// File: lm75a_cgi.cpp // Read and report the temperature from an lm75a in a web page. // Author: Jim Merkle, 11/03/2017 #include #include #include #include #include #include #include #include #include #include #include #include #include // nanosleep #define PROG_NAME "lm75a_cgi" // I2C Address #define LM75A 0x48 // LM75A Registers #define LM75A_REG_TEMP 0 #define LM75A_REG_CONF 1 #define LM75A_REG_THYST 2 #define LM75A_REG_TOS 3 #define LM75A_REG_PROD_ID 7 /* only applicable for TI manufactured part */ // Power On Defaults: // 0: Read only temperature - 2 bytes // 1: 0x00 Configuration // 2: 0x4B THYST - 75 degrees C - 2 bytes // 3: 0x50 TOS - 80 degrees C - 2 bytes // 7: 0xA1 (TI only) void msleep(int milliseconds) { //usleep(milliseconds * 1000); // limited method struct timespec ts; ts.tv_sec = milliseconds / 1000; ts.tv_nsec = (milliseconds % 1000) * 1000000; nanosleep(&ts, NULL); } void printbinary16(uint16_t number) { uint16_t mask = 0x8000; // most significant bit set while (mask) { printf("%c",mask & number?'1':'0'); mask >>= 1; } } // Return number of i2c_msg's transferred, else negative value for error int i2cWriteRead(int fd, int i2cAddress, uint8_t * pWriteData, int writeBytes, uint8_t * pReadData, int readBytes) { struct i2c_msg msg[2]; // declare a two i2c_msg array struct i2c_rdwr_ioctl_data i2c_data; // declare our i2c_rdwr_ioctl_data structure int msgsUsed=0; // how many i2c_msg structures are used int result; // return value from ioctl() call // Assume the first i2c_msg is used for writing if(pWriteData && writeBytes) { msg[0].addr = i2cAddress; msg[0].flags = 0; msg[0].buf = pWriteData; msg[0].len = writeBytes; msgsUsed++; // Load the second i2c_msg for receiving if the first one is used if(pReadData && readBytes) { // Load up receive msg msg[1].addr = i2cAddress; msg[1].flags = I2C_M_RD; msg[1].buf = pReadData; msg[1].len = readBytes; msgsUsed++; } } else if(pReadData && readBytes) { // Load the first i2c_msg for receiving (no transmit data) msg[0].addr = i2cAddress; msg[0].flags = I2C_M_RD; msg[0].buf = pReadData; msg[0].len = readBytes; msgsUsed++; } // Continue if we have data to transfer if(msgsUsed) { // i2c_msg array is loaded up. Now load i2c_rdwr_ioctl_data structure. i2c_data.msgs = msg; i2c_data.nmsgs = msgsUsed; // With our open file descriptor, perform I2C message transfers // This function call returns the number of i2c_msg structures processed, // or a negative error value result = ioctl(fd,I2C_RDWR,&i2c_data); if(result < 0) { printf("ioctl error: %s\n", strerror(errno)); return -2; } } else { printf("i2cWriteRead: No i2c_msg's processed\n"); } return result; // return number of i2c_msg structures processed } int main(int argc, char * argv[]) { // The following printf() gives some CGI interface some data to begin constructing a web page // The CGI interface will now wait for more data printf("Content-type: text/html\n\n"); // Open the i2c bus device where the chip resides int fd = open("/dev/i2c-1",O_RDWR); if(fd<0) { printf("Failed to open the bus."); /* ERROR HANDLING; you can check errno to see what went wrong */ exit(1); } // By default, the LM75A configuration register is 0x00 // Make sure the part isn't in "Shutdown" mode (bit 0 set) uint8_t u8WriteConfig[2] = {LM75A_REG_CONF,0}; i2cWriteRead(fd, LM75A, u8WriteConfig, sizeof(u8WriteConfig), NULL, 0); // Read two temperature bytes after setting temperature index uint8_t index = LM75A_REG_TEMP; // Read temperature register uint8_t temperatureRegs[2]; // The MSB (first byte) is degree C units, the LSB is fractional component i2cWriteRead(fd, LM75A, &index, 1, temperatureRegs, 2); // Combine the two bytes into a 16-bit value (High byte is temperature - whole numbers, low byte is binary fraction (1/256 units) uint16_t uTemp = (uint16_t)temperatureRegs[0] << 8 | (uint16_t)temperatureRegs[1]; // high byte followed by low byte float fTemp = (short)uTemp; fTemp /= 256; // The float will keep the remainder // Convert into Fahrenheit float fTempf = fTemp*9/5 + 32; //printf("\033cTemp %0.1fF\n",fTempf); //printbinary16(uTemp); //printf(" 0x%04X %0.3fC ",uTemp,fTemp); printf("The Raspberry Pi with LM75A says it's %0.1fF\n",fTempf); // clean up close(fd); return 0; }