Tasmota/sonoff/xsns_htu21.ino

303 lines
7.6 KiB
Arduino
Raw Normal View History

2017-01-28 13:41:01 +00:00
/*
xsns_htu21.ino - HTU21 temperature and humidity sensor support for Sonoff-Tasmota
Copyright (C) 2017 Heiko Krupp and Theo Arends
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/>.
2017-01-28 13:41:01 +00:00
*/
#ifdef USE_I2C
#ifdef USE_HTU
/*********************************************************************************************\
* HTU21 - Temperature and Humidy
*
* Source: Heiko Krupp
\*********************************************************************************************/
#define HTU21_ADDR 0x40
#define SI7013_CHIPID 0x0D
#define SI7020_CHIPID 0x14
#define SI7021_CHIPID 0x15
2017-01-28 13:41:01 +00:00
#define HTU21_CHIPID 0x32
#define HTU21_READTEMP 0xE3
#define HTU21_READHUM 0xE5
#define HTU21_WRITEREG 0xE6
#define HTU21_READREG 0xE7
#define HTU21_RESET 0xFE
#define HTU21_HEATER_WRITE 0x51
#define HTU21_HEATER_READ 0x11
#define HTU21_SERIAL2_READ1 0xFC /* Read 3rd two Serial bytes */
#define HTU21_SERIAL2_READ2 0xC9 /* Read 4th two Serial bytes */
#define HTU21_HEATER_ON 0x04
#define HTU21_HEATER_OFF 0xFB
#define HTU21_RES_RH12_T14 0x00 // Default
2017-01-28 13:41:01 +00:00
#define HTU21_RES_RH8_T12 0x01
#define HTU21_RES_RH10_T13 0x80
#define HTU21_RES_RH11_T11 0x81
#define HTU21_CRC8_POLYNOM 0x13100
uint8_t htu_address;
uint8_t htu_type = 0;
uint8_t delay_temp;
uint8_t delay_humidity = 50;
char htu_types[7];
2017-01-28 13:41:01 +00:00
uint8_t HtuCheckCrc8(uint16_t data)
2017-01-28 13:41:01 +00:00
{
for (uint8_t bit = 0; bit < 16; bit++) {
if (data & 0x8000) {
2017-01-28 13:41:01 +00:00
data = (data << 1) ^ HTU21_CRC8_POLYNOM;
} else {
2017-01-28 13:41:01 +00:00
data <<= 1;
}
2017-01-28 13:41:01 +00:00
}
return data >>= 8;
}
uint8_t HtuReadDeviceId(void)
2017-01-28 13:41:01 +00:00
{
uint16_t deviceID = 0;
uint8_t checksum = 0;
Wire.beginTransmission(HTU21_ADDR);
Wire.write(HTU21_SERIAL2_READ1);
Wire.write(HTU21_SERIAL2_READ2);
Wire.endTransmission();
Wire.requestFrom(HTU21_ADDR, 3);
deviceID = Wire.read() << 8;
deviceID |= Wire.read();
checksum = Wire.read();
if (HtuCheckCrc8(deviceID) == checksum) {
2017-01-28 13:41:01 +00:00
deviceID = deviceID >> 8;
} else {
deviceID = 0;
}
return (uint8_t)deviceID;
}
void HtuSetResolution(uint8_t resolution)
2017-01-28 13:41:01 +00:00
{
uint8_t current = I2cRead8(HTU21_ADDR, HTU21_READREG);
2017-01-28 13:41:01 +00:00
current &= 0x7E; // Replace current resolution bits with 0
current |= resolution; // Add new resolution bits to register
I2cWrite8(HTU21_ADDR, HTU21_WRITEREG, current);
2017-01-28 13:41:01 +00:00
}
void HtuReset(void)
2017-01-28 13:41:01 +00:00
{
Wire.beginTransmission(HTU21_ADDR);
Wire.write(HTU21_RESET);
Wire.endTransmission();
delay(15); // Reset takes 15ms
}
void HtuHeater(uint8_t heater)
2017-01-28 13:41:01 +00:00
{
uint8_t current = I2cRead8(HTU21_ADDR, HTU21_READREG);
2017-01-28 13:41:01 +00:00
switch(heater)
{
case HTU21_HEATER_ON : current |= heater;
break;
case HTU21_HEATER_OFF : current &= heater;
break;
default : current &= heater;
break;
}
I2cWrite8(HTU21_ADDR, HTU21_WRITEREG, current);
2017-01-28 13:41:01 +00:00
}
boolean HtuInit()
2017-01-28 13:41:01 +00:00
{
HtuReset();
HtuHeater(HTU21_HEATER_OFF);
HtuSetResolution(HTU21_RES_RH12_T14);
2017-01-28 13:41:01 +00:00
return true;
}
float HtuReadHumidity(void)
2017-01-28 13:41:01 +00:00
{
uint8_t checksum = 0;
uint16_t sensorval = 0;
float humidity = 0.0;
2017-01-28 13:41:01 +00:00
Wire.beginTransmission(HTU21_ADDR);
Wire.write(HTU21_READHUM);
if (Wire.endTransmission() != 0) {
return 0.0; // In case of error
}
delay(delay_humidity); // Sensor time at max resolution
2017-01-28 13:41:01 +00:00
Wire.requestFrom(HTU21_ADDR, 3);
if (3 <= Wire.available()) {
2017-01-28 13:41:01 +00:00
sensorval = Wire.read() << 8; // MSB
sensorval |= Wire.read(); // LSB
checksum = Wire.read();
}
if (HtuCheckCrc8(sensorval) != checksum) {
return 0.0; // Checksum mismatch
}
2017-01-28 13:41:01 +00:00
sensorval ^= 0x02; // clear status bits
humidity = 0.001907 * (float)sensorval - 6;
if (humidity > 100) {
return 100.0;
}
if (humidity < 0) {
return 0.01;
}
2017-01-28 13:41:01 +00:00
return humidity;
}
float HtuReadTemperature()
2017-01-28 13:41:01 +00:00
{
uint8_t checksum=0;
uint16_t sensorval=0;
float t;
Wire.beginTransmission(HTU21_ADDR);
Wire.write(HTU21_READTEMP);
if (Wire.endTransmission() != 0) {
return 0.0; // In case of error
}
delay(delay_temp); // Sensor time at max resolution
2017-01-28 13:41:01 +00:00
Wire.requestFrom(HTU21_ADDR, 3);
if (3 == Wire.available()) {
2017-01-28 13:41:01 +00:00
sensorval = Wire.read() << 8; // MSB
sensorval |= Wire.read(); // LSB
checksum = Wire.read();
}
if (HtuCheckCrc8(sensorval) != checksum) {
return 0.0; // Checksum mismatch
}
2017-01-28 13:41:01 +00:00
t = ConvertTemp(0.002681 * (float)sensorval - 46.85);
2017-01-28 13:41:01 +00:00
return t;
}
float HtuCompensatedHumidity(float humidity, float temperature)
2017-01-28 13:41:01 +00:00
{
if(humidity == 0.00 && temperature == 0.00) {
return 0.0;
}
if(temperature > 0.00 && temperature < 80.00) {
2017-01-28 13:41:01 +00:00
return (-0.15)*(25-temperature)+humidity;
}
2017-01-28 13:41:01 +00:00
}
uint8_t HtuDetect()
2017-01-28 13:41:01 +00:00
{
if (htu_type) {
return true;
}
2017-01-28 13:41:01 +00:00
boolean success = false;
htu_address = HTU21_ADDR;
htu_type = HtuReadDeviceId();
success = HtuInit();
switch (htu_type) {
2017-01-28 13:41:01 +00:00
case HTU21_CHIPID:
strcpy_P(htu_types, PSTR("HTU21"));
delay_temp=50;
delay_humidity=16;
break;
case SI7013_CHIPID:
strcpy_P(htu_types, PSTR("SI7013"));
delay_temp=12;
delay_humidity=23;
break;
case SI7020_CHIPID:
strcpy_P(htu_types, PSTR("SI7020"));
delay_temp=12;
delay_humidity=23;
break;
case SI7021_CHIPID:
strcpy_P(htu_types, PSTR("SI7021"));
delay_temp=12;
delay_humidity=23;
break;
default:
strcpy_P(htu_types, PSTR("T/RH?"));
delay_temp=50;
delay_humidity=23;
2017-01-28 13:41:01 +00:00
}
if (success) {
snprintf_P(log_data, sizeof(log_data), PSTR(D_LOG_I2C "%s " D_FOUND_AT " 0x%x"), htu_types, htu_address);
AddLog(LOG_LEVEL_DEBUG);
2017-01-28 13:41:01 +00:00
} else {
htu_type = 0;
2017-01-28 13:41:01 +00:00
}
return success;
}
/*********************************************************************************************\
* Presentation
\*********************************************************************************************/
void MqttShowHtu(uint8_t* djson)
2017-01-28 13:41:01 +00:00
{
if (!htu_type) {
return;
}
2017-01-28 13:41:01 +00:00
char stemp1[10];
char stemp2[10];
2017-01-28 13:41:01 +00:00
float t = HtuReadTemperature();
float h = HtuReadHumidity();
h = HtuCompensatedHumidity(h, t);
dtostrfd(t, Settings.flag.temperature_resolution, stemp1);
dtostrfd(h, Settings.flag.humidity_resolution, stemp2);
snprintf_P(mqtt_data, sizeof(mqtt_data), JSON_SNS_TEMPHUM, mqtt_data, htu_types, stemp1, stemp2);
2017-01-28 13:41:01 +00:00
*djson = 1;
#ifdef USE_DOMOTICZ
DomoticzTempHumSensor(stemp1, stemp2);
2017-01-28 13:41:01 +00:00
#endif // USE_DOMOTICZ
}
#ifdef USE_WEBSERVER
String WebShowHtu()
2017-01-28 13:41:01 +00:00
{
String page = "";
if (htu_type) {
char stemp[10];
char sensor[80];
2017-01-28 13:41:01 +00:00
float t_htu21 = HtuReadTemperature();
float h_htu21 = HtuReadHumidity();
h_htu21 = HtuCompensatedHumidity(h_htu21, t_htu21);
dtostrfi(t_htu21, Settings.flag.temperature_resolution, stemp);
snprintf_P(sensor, sizeof(sensor), HTTP_SNS_TEMP, htu_types, stemp, TempUnit());
page += sensor;
dtostrfi(h_htu21, Settings.flag.humidity_resolution, stemp);
snprintf_P(sensor, sizeof(sensor), HTTP_SNS_HUM, htu_types, stemp);
page += sensor;
2017-01-28 13:41:01 +00:00
}
return page;
}
#endif // USE_WEBSERVER
#endif // USE_HTU
#endif // USE_I2C