/********************************************************************************* * Copyright: (C) 2023 LingYun IoT System Studio * All rights reserved. * * Filename: tsl2561.c * Description: This file is the Lux sensor TSL2561 code * * Version: 1.0.0(10/08/23) * Author: Guo Wenxue * ChangeLog: 1, Release initial version on "10/08/23 17:52:00" * * Pin connection: * STH20 Module Raspberry Pi Board * VCC <-----> #Pin1(3.3V) * SDA0 <-----> #Pin27(SDA, BCM GPIO0) * SCL0 <-----> #Pin28(SCL, BCM GPIO1) * GND <-----> GND * * /boot/config.txt: * dtoverlay=i2c0,pins_0_1 * ********************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "util_proc.h" #include "logger.h" #include "tsl2561.h" #define CONTROL_REG 0x80 #define REG_COUNT 4 #define POWER_UP 0x03 #define POWER_DOWN 0x00 #define OFF 0 #define ON 1 /* Register Address */ enum { /* Channel_0 = DATA0HIGH<<8 + DATA0LOW */ DATA0LOW = 0x8C, DATA0HIGH, /* Channel_1 = DATA1HIGH<<8 + DATA1LOW */ DATA1LOW, DATA1HIGH, }; static const int regs_addr[REG_COUNT]={DATA0LOW, DATA0HIGH, DATA1LOW, DATA1HIGH}; void tsl2561_power(int fd, int cmd) { struct i2c_msg msg; struct i2c_rdwr_ioctl_data data; unsigned char buf[2]; msg.addr= TSL2561_I2CADDR; msg.flags=0; /* write */ msg.len= 1; msg.buf= buf; data.nmsgs= 1; data.msgs= &msg; msg.buf[0]=CONTROL_REG; if( ioctl(fd, I2C_RDWR, &data) < 0 ) { log_error("%s() ioctl failure: %s\n", __func__, strerror(errno)); return ; } if( cmd ) msg.buf[0]=POWER_UP; else msg.buf[0]=POWER_DOWN; if( ioctl(fd, I2C_RDWR, &data) < 0 ) { log_error("%s() ioctl failure: %s\n", __func__, strerror(errno)); return ; } return ; } int tsl2561_readreg(int fd, unsigned char regaddr, unsigned char *regval) { struct i2c_msg msg; struct i2c_rdwr_ioctl_data data; unsigned char buf[2]; msg.addr= TSL2561_I2CADDR; msg.flags=0; /* write */ msg.len= 1; msg.buf= buf; msg.buf[0] = regaddr; data.nmsgs= 1; data.msgs= &msg; if( ioctl(fd, I2C_RDWR, &data) < 0 ) { log_error("%s() ioctl failure: %s\n", __func__, strerror(errno)); return -1; } memset(buf, 0, sizeof(buf)); msg.addr= TSL2561_I2CADDR; msg.flags=I2C_M_RD; /* read */ msg.len= 1; msg.buf= buf; data.nmsgs= 1; data.msgs= &msg; if( ioctl(fd, I2C_RDWR, &data) < 0 ) { log_error("%s() ioctl failure: %s\n", __func__, strerror(errno)); return -1; } *regval = msg.buf[0]; return 0; } int tsl2561_get_lux(float *lux) { int i, fd; int rv = 0; char *dev = TSL2561_I2CDEV; float div = 0.0; unsigned char reg_data[REG_COUNT]; unsigned char buf; int chn0_data = 0; int chn1_data = 0; if( !lux ) { log_error("Invalid input arguments\n"); return -1; } if( (fd=open(dev, O_RDWR)) < 0) { log_error("i2c device '%s' open failed: %s\n", dev, strerror(errno)); return -2; } tsl2561_power(fd, ON); msleep(410); /* t(CONV) MAX 400ms */ /* Read register Channel0 and channel1 data from register */ for(i=0; i0 && div<=0.5 ) *lux = 0.304*chn0_data-0.062*chn0_data*pow(div,1.4); else if( div>0.5 && div<=0.61 ) *lux = 0.0224*chn0_data-0.031*chn1_data; else if( div>0.61 && div<=0.8 ) *lux = 0.0128*chn0_data-0.0153*chn1_data; else if( div>0.8 && div<=1.3 ) *lux = 0.00146*chn0_data-0.00112*chn1_data; else if( div>1.3 ) *lux = 0.0; OUT: tsl2561_power(fd, OFF); return rv; }