Tasmota/tasmota/xsns_92_hdc1080.ino

289 lines
7.7 KiB
Arduino
Raw Normal View History

2020-03-08 20:54:28 +00:00
/*
xsns_92_hdc1080.ino - Texas Instruments HDC1080 temperature and humidity sensor support for Tasmota
Copyright (C) 2020 Luis Teixeira
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#ifdef USE_I2C
#ifdef USE_HDC1080
/*********************************************************************************************\
* HDC1080 - Temperature and Humidy sensor
*
* Source: Luis Teixeira
*
* I2C Address: 0x40
\*********************************************************************************************/
#define XSNS_92 92
#define XI2C_92 92 // See I2CDEVICES.md
#define HDC1080_ADDR 0x40
// Registers:
#define HDC_REG_TEMP 0x00 // Temperature register
#define HDC_REG_RH 0x01 // Humidity register
#define HDC_REG_CONFIG 0x02 // Configuration register
#define HDC_REG_SERIAL1 0xFB // First 2 bytes of the serial ID
#define HDC_REG_SERIAL2 0xFC // Mid 2 bytes of the serial ID
#define HDC_REG_SERIAL3 0xFD // Last bits of the serial ID
#define HDC_REG_MAN_ID 0xFE // Manufacturer ID
#define HDC_REG_DEV_ID 0xFF // Device ID
// Expected constant values of some of the registers:
#define HDC1080_MAN_ID 0x5449 // Manufacturer ID (Texas Instruments)
#define HDC1080_DEV_ID 0x1050 // Device ID (valid for the HDC1080)
// Possible values for the configuration register fields:
#define HDC1080_RST_ON 0x8000
#define HDC1080_HEAT_ON 0x2000
#define HDC1080_MODE_ON 0x1000 // acquision mode (temperature + humidity)
#define HDC1080_TRES_11 0x400
#define HDC1080_HRES_11 0x100
#define HDC1080_HRES_8 0x80
// Constants:
#define HDC1080_CONV_TIME 50 // Assume 6.50 + 6.35 ms + x of conversion delay for this device
2020-03-08 20:54:28 +00:00
#define HDC1080_TEMP_MULT 0.0025177
#define HDC1080_RH_MULT 0.0025177
#define HDC1080_TEMP_OFFSET 40.0
2020-03-08 20:54:28 +00:00
const char* hdc_type_name = "HDC1080";
uint16_t hdc_manufacturer_id = 0;
uint16_t hdc_device_id = 0;
float hdc_temperature = 0.0;
float hdc_humidity = 0.0;
2020-03-08 20:54:28 +00:00
uint8_t hdc_valid = 0;
/**
* Reads the device ID register.
*
*/
uint16_t HdcReadDeviceId(void) {
return I2cRead16(HDC1080_ADDR, HDC_REG_DEV_ID);
}
2020-03-08 20:54:28 +00:00
/**
* Reads the manufacturer ID register.
*
*/
uint16_t HdcReadManufacturerId(void) {
return I2cRead16(HDC1080_ADDR, HDC_REG_MAN_ID);
2020-03-08 20:54:28 +00:00
}
/**
* Overwrites the configuration register with the provided config
*/
void HdcConfig(uint16_t config) {
I2cWrite16(HDC1080_ADDR, HDC_REG_CONFIG, config);
2020-03-08 20:54:28 +00:00
}
/**
* Performs a soft reset on the device.
2020-03-08 20:54:28 +00:00
*
* RST = 1 -> software reset
2020-03-08 20:54:28 +00:00
*
*/
void HdcReset(void) {
2020-03-08 20:54:28 +00:00
uint16_t current = I2cRead16(HDC1080_ADDR, HDC_REG_CONFIG);
// bit 15 of the configuration register contains the RST flag
// so we set it to 1:
2020-03-08 20:54:28 +00:00
current |= 0x8000;
2020-03-08 20:54:28 +00:00
I2cWrite16(HDC1080_ADDR, HDC_REG_CONFIG, current);
delay(HDC1080_CONV_TIME); // Not sure how long it takes to reset. Assuming this value.
2020-03-08 20:54:28 +00:00
}
/**
* Performs the single transaction read of the HDC1080, providing the
* adequate delay for the acquisition.
*
*/
int8_t HdcReadBuffer(uint8_t addr, uint8_t reg, uint8_t *reg_data, uint16_t len) {
Wire.beginTransmission((uint8_t)addr);
Wire.write((uint8_t)reg);
Wire.endTransmission();
delay(HDC1080_CONV_TIME);
if (len != Wire.requestFrom((uint8_t)addr, (uint8_t)len)) {
return 1;
}
while (len--) {
*reg_data = (uint8_t)Wire.read();
reg_data++;
}
return 0;
}
/**
* The various initialization steps for this sensor.
*
*/
void HdcInit(void) {
2020-03-08 20:54:28 +00:00
HdcReset();
HdcConfig(HDC1080_MODE_ON);
2020-03-08 20:54:28 +00:00
}
/**
* Performs a temperature and humidity measurement, and calls
* the conversion function providing the results in the correct
* unit according to the device settings.
*
*/
bool HdcRead(void) {
int8_t status = 0;
uint8_t sensor_data[4];
uint16_t temp_data = 0;
uint16_t rh_data = 0;
2020-03-08 20:54:28 +00:00
status = HdcReadBuffer(HDC1080_ADDR, HDC_REG_TEMP, sensor_data, 4);
2020-03-08 20:54:28 +00:00
temp_data = (uint16_t) ((sensor_data[0] << 8) | sensor_data[1]);
rh_data = (uint16_t) ((sensor_data[2] << 8) | sensor_data[3]);
2020-03-08 20:54:28 +00:00
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("HdcRead: temperature raw data: 0x%04x; humidity raw data: 0x%04x"), temp_data, rh_data);
2020-03-08 20:54:28 +00:00
if(status != 0) {
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("HdcRead: failed to read HDC_REG_TEMP. Status = %d"), status);
return false;
}
// read the temperature from the first 16 bits of the result
2020-03-08 20:54:28 +00:00
hdc_temperature = ConvertTemp(HDC1080_TEMP_MULT * (float) (temp_data) - HDC1080_TEMP_OFFSET);
hdc_humidity = HDC1080_RH_MULT * (float) (rh_data);
if (hdc_humidity > 100) { hdc_humidity = 100.0; }
2020-03-08 20:54:28 +00:00
if (hdc_humidity < 0) { hdc_humidity = 0.01; }
ConvertHumidity(hdc_humidity); // Set global humidity
hdc_valid = SENSOR_MAX_MISS;
return true;
}
/********************************************************************************************/
void HdcDetect(void) {
if (I2cActive(HDC1080_ADDR)) {
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("HdcDetect: Address = 0x%02X already in use."), HDC1080_ADDR);
2020-03-08 20:54:28 +00:00
return;
}
hdc_manufacturer_id = HdcReadManufacturerId();
hdc_device_id = HdcReadDeviceId();
2020-03-08 20:54:28 +00:00
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("HdcDetect: detected device with manufacturerId = 0x%04X and deviceId = 0x%04X"), hdc_manufacturer_id, hdc_device_id);
2020-03-08 20:54:28 +00:00
if (hdc_device_id == HDC1080_DEV_ID) {
2020-03-08 20:54:28 +00:00
HdcInit();
I2cSetActiveFound(HDC1080_ADDR, hdc_type_name);
2020-03-08 20:54:28 +00:00
}
}
void HdcEverySecond(void) {
if (uptime &1) { // Every 2 seconds
if (!HdcRead()) {
AddLogMissed((char*) hdc_type_name, hdc_valid);
2020-03-08 20:54:28 +00:00
}
}
}
void HdcShow(bool json) {
if (hdc_valid) {
char temperature[33];
dtostrfd(hdc_temperature, Settings.flag2.temperature_resolution, temperature);
char humidity[33];
dtostrfd(hdc_humidity, Settings.flag2.humidity_resolution, humidity);
if (json) {
ResponseAppend_P(JSON_SNS_TEMPHUM, hdc_type_name, temperature, humidity);
2020-03-08 20:54:28 +00:00
#ifdef USE_DOMOTICZ
if (0 == tele_period) {
DomoticzTempHumSensor(temperature, humidity);
}
#endif // USE_DOMOTICZ
#ifdef USE_KNX
if (0 == tele_period) {
KnxSensor(KNX_TEMPERATURE, hdc_temperature);
KnxSensor(KNX_HUMIDITY, hdc_humidity);
}
#endif // USE_KNX
#ifdef USE_WEBSERVER
} else {
WSContentSend_PD(HTTP_SNS_TEMP, hdc_type_name, temperature, TempUnit());
WSContentSend_PD(HTTP_SNS_HUM, hdc_type_name, humidity);
2020-03-08 20:54:28 +00:00
#endif // USE_WEBSERVER
}
}
}
/*********************************************************************************************\
* Interface
\*********************************************************************************************/
bool Xsns92(uint8_t function)
{
if (!I2cEnabled(XI2C_92)) {
AddLog_P(LOG_LEVEL_DEBUG, PSTR("Xsns92: I2C driver not enabled for this device."));
return false;
}
2020-03-08 20:54:28 +00:00
bool result = false;
if (FUNC_INIT == function) {
HdcDetect();
}
else if (hdc_device_id) {
2020-03-08 20:54:28 +00:00
switch (function) {
case FUNC_EVERY_SECOND:
HdcEverySecond();
break;
case FUNC_JSON_APPEND:
HdcShow(1);
break;
#ifdef USE_WEBSERVER
case FUNC_WEB_SENSOR:
HdcShow(0);
break;
#endif // USE_WEBSERVER
}
}
return result;
}
#endif // USE_HDC1080
#endif // USE_I2C