mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-31 15:12:06 +00:00 
			
		
		
		
	Add AHT10 sensor (#949)
This commit is contained in:
		
							
								
								
									
										0
									
								
								esphome/components/aht10/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								esphome/components/aht10/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										127
									
								
								esphome/components/aht10/aht10.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								esphome/components/aht10/aht10.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | ||||
| // Implementation based on: | ||||
| //  - AHT10: https://github.com/Thinary/AHT10 | ||||
| //  - Official Datasheet (cn): | ||||
| //  http://www.aosong.com/userfiles/files/media/aht10%E8%A7%84%E6%A0%BC%E4%B9%A6v1_1%EF%BC%8820191015%EF%BC%89.pdf | ||||
| //  - Unofficial Translated Datasheet (en): | ||||
| //  https://wiki.liutyi.info/download/attachments/30507639/Aosong_AHT10_en_draft_0c.pdf | ||||
| // | ||||
| // When configured for humidity, the log 'Components should block for at most 20-30ms in loop().' will be generated in | ||||
| // verbose mode. This is due to technical specs of the sensor and can not be avoided. | ||||
| // | ||||
| // According to the datasheet, the component is supposed to respond in more than 75ms. In fact, it can answer almost | ||||
| // immediately for temperature. But for humidity, it takes >90ms to get a valid data. From experience, we have best | ||||
| // results making successive requests; the current implementation make 3 attemps with a delay of 30ms each time. | ||||
|  | ||||
| #include "aht10.h" | ||||
| #include "esphome/core/log.h" | ||||
|  | ||||
| namespace esphome { | ||||
| namespace aht10 { | ||||
|  | ||||
| static const char *TAG = "aht10"; | ||||
| static const uint8_t AHT10_CALIBRATE_CMD[] = {0xE1}; | ||||
| static const uint8_t AHT10_MEASURE_CMD[] = {0xAC, 0x33, 0x00}; | ||||
| static const uint8_t AHT10_DEFAULT_DELAY = 5;    // ms, for calibration and temperature measurement | ||||
| static const uint8_t AHT10_HUMIDITY_DELAY = 30;  // ms | ||||
| static const uint8_t AHT10_ATTEMPS = 3;          // safety margin, normally 3 attemps are enough: 3*30=90ms | ||||
|  | ||||
| void AHT10Component::setup() { | ||||
|   ESP_LOGCONFIG(TAG, "Setting up AHT10..."); | ||||
|  | ||||
|   if (!this->write_bytes(0, AHT10_CALIBRATE_CMD, sizeof(AHT10_CALIBRATE_CMD))) { | ||||
|     ESP_LOGE(TAG, "Communication with AHT10 failed!"); | ||||
|     this->mark_failed(); | ||||
|     return; | ||||
|   } | ||||
|   uint8_t data; | ||||
|   if (!this->read_byte(0, &data, AHT10_DEFAULT_DELAY)) { | ||||
|     ESP_LOGD(TAG, "Communication with AHT10 failed!"); | ||||
|     this->mark_failed(); | ||||
|     return; | ||||
|   } | ||||
|   if ((data & 0x68) != 0x08) {  // Bit[6:5] = 0b00, NORMAL mode and Bit[3] = 0b1, CALIBRATED | ||||
|     ESP_LOGE(TAG, "AHT10 calibration failed!"); | ||||
|     this->mark_failed(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   ESP_LOGV(TAG, "AHT10 calibrated"); | ||||
| } | ||||
|  | ||||
| void AHT10Component::update() { | ||||
|   if (!this->write_bytes(0, AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD))) { | ||||
|     ESP_LOGE(TAG, "Communication with AHT10 failed!"); | ||||
|     this->status_set_warning(); | ||||
|     return; | ||||
|   } | ||||
|   uint8_t data[6]; | ||||
|   uint8_t delay = AHT10_DEFAULT_DELAY; | ||||
|   if (this->humidity_sensor_ != nullptr) | ||||
|     delay = AHT10_HUMIDITY_DELAY; | ||||
|   for (int i = 0; i < AHT10_ATTEMPS; ++i) { | ||||
|     ESP_LOGVV(TAG, "Attemps %u at %6ld", i, millis()); | ||||
|     if (!this->read_bytes(0, data, 6, delay)) { | ||||
|       ESP_LOGD(TAG, "Communication with AHT10 failed, waiting..."); | ||||
|     } else if ((data[0] & 0x80) == 0x80) {  // Bit[7] = 0b1, device is busy | ||||
|       ESP_LOGD(TAG, "AHT10 is busy, waiting..."); | ||||
|     } else if (data[1] == 0x0 && data[2] == 0x0 && (data[3] >> 4) == 0x0) { | ||||
|       // Unrealistic humidity (0x0) | ||||
|       if (this->humidity_sensor_ == nullptr) { | ||||
|         ESP_LOGVV(TAG, "ATH10 Unrealistic humidity (0x0), but humidity is not required"); | ||||
|         break; | ||||
|       } else { | ||||
|         ESP_LOGD(TAG, "ATH10 Unrealistic humidity (0x0), retrying..."); | ||||
|         if (!this->write_bytes(0, AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD))) { | ||||
|           ESP_LOGE(TAG, "Communication with AHT10 failed!"); | ||||
|           this->status_set_warning(); | ||||
|           return; | ||||
|         } | ||||
|       } | ||||
|     } else { | ||||
|       // data is valid, we can break the loop | ||||
|       ESP_LOGVV(TAG, "Answer at %6ld", millis()); | ||||
|       break; | ||||
|     } | ||||
|   } | ||||
|   if ((data[0] & 0x80) == 0x80) { | ||||
|     ESP_LOGE(TAG, "Measurements reading timed-out!"); | ||||
|     this->status_set_warning(); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   uint32_t raw_temperature = ((data[3] & 0x0F) << 16) | (data[4] << 8) | data[5]; | ||||
|   uint32_t raw_humidity = ((data[1] << 16) | (data[2] << 8) | data[3]) >> 4; | ||||
|  | ||||
|   float temperature = ((200.0 * (float) raw_temperature) / 1048576.0) - 50.0; | ||||
|   float humidity; | ||||
|   if (raw_humidity == 0) {  // unrealistic value | ||||
|     humidity = NAN; | ||||
|   } else { | ||||
|     humidity = (float) raw_humidity * 100.0 / 1048576.0; | ||||
|   } | ||||
|  | ||||
|   if (this->temperature_sensor_ != nullptr) { | ||||
|     this->temperature_sensor_->publish_state(temperature); | ||||
|   } | ||||
|   if (this->humidity_sensor_ != nullptr) { | ||||
|     if (isnan(humidity)) | ||||
|       ESP_LOGW(TAG, "Invalid humidity! Sensor reported 0%% Hum"); | ||||
|     this->humidity_sensor_->publish_state(humidity); | ||||
|   } | ||||
|   this->status_clear_warning(); | ||||
| } | ||||
|  | ||||
| float AHT10Component::get_setup_priority() const { return setup_priority::DATA; } | ||||
|  | ||||
| void AHT10Component::dump_config() { | ||||
|   ESP_LOGCONFIG(TAG, "AHT10:"); | ||||
|   LOG_I2C_DEVICE(this); | ||||
|   if (this->is_failed()) { | ||||
|     ESP_LOGE(TAG, "Communication with AHT10 failed!"); | ||||
|   } | ||||
|   LOG_SENSOR("  ", "Temperature", this->temperature_sensor_); | ||||
|   LOG_SENSOR("  ", "Humidity", this->humidity_sensor_); | ||||
| } | ||||
|  | ||||
| }  // namespace aht10 | ||||
| }  // namespace esphome | ||||
							
								
								
									
										26
									
								
								esphome/components/aht10/aht10.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								esphome/components/aht10/aht10.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include "esphome/core/component.h" | ||||
| #include "esphome/components/sensor/sensor.h" | ||||
| #include "esphome/components/i2c/i2c.h" | ||||
|  | ||||
| namespace esphome { | ||||
| namespace aht10 { | ||||
|  | ||||
| class AHT10Component : public PollingComponent, public i2c::I2CDevice { | ||||
|  public: | ||||
|   void setup() override; | ||||
|   void update() override; | ||||
|   void dump_config() override; | ||||
|   float get_setup_priority() const override; | ||||
|  | ||||
|   void set_temperature_sensor(sensor::Sensor *temperature_sensor) { temperature_sensor_ = temperature_sensor; } | ||||
|   void set_humidity_sensor(sensor::Sensor *humidity_sensor) { humidity_sensor_ = humidity_sensor; } | ||||
|  | ||||
|  protected: | ||||
|   sensor::Sensor *temperature_sensor_; | ||||
|   sensor::Sensor *humidity_sensor_; | ||||
| }; | ||||
|  | ||||
| }  // namespace aht10 | ||||
| }  // namespace esphome | ||||
							
								
								
									
										30
									
								
								esphome/components/aht10/sensor.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								esphome/components/aht10/sensor.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| import esphome.codegen as cg | ||||
| import esphome.config_validation as cv | ||||
| from esphome.components import i2c, sensor | ||||
| from esphome.const import CONF_HUMIDITY, CONF_ID, CONF_TEMPERATURE, \ | ||||
|     UNIT_CELSIUS, ICON_THERMOMETER, ICON_WATER_PERCENT, UNIT_PERCENT | ||||
|  | ||||
| DEPENDENCIES = ['i2c'] | ||||
|  | ||||
| aht10_ns = cg.esphome_ns.namespace('aht10') | ||||
| AHT10Component = aht10_ns.class_('AHT10Component', cg.PollingComponent, i2c.I2CDevice) | ||||
|  | ||||
| CONFIG_SCHEMA = cv.Schema({ | ||||
|     cv.GenerateID(): cv.declare_id(AHT10Component), | ||||
|     cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(UNIT_CELSIUS, ICON_THERMOMETER, 2), | ||||
|     cv.Optional(CONF_HUMIDITY): sensor.sensor_schema(UNIT_PERCENT, ICON_WATER_PERCENT, 2), | ||||
| }).extend(cv.polling_component_schema('60s')).extend(i2c.i2c_device_schema(0x38)) | ||||
|  | ||||
|  | ||||
| def to_code(config): | ||||
|     var = cg.new_Pvariable(config[CONF_ID]) | ||||
|     yield cg.register_component(var, config) | ||||
|     yield i2c.register_i2c_device(var, config) | ||||
|  | ||||
|     if CONF_TEMPERATURE in config: | ||||
|         sens = yield sensor.new_sensor(config[CONF_TEMPERATURE]) | ||||
|         cg.add(var.set_temperature_sensor(sens)) | ||||
|  | ||||
|     if CONF_HUMIDITY in config: | ||||
|         sens = yield sensor.new_sensor(config[CONF_HUMIDITY]) | ||||
|         cg.add(var.set_humidity_sensor(sens)) | ||||
| @@ -221,6 +221,11 @@ sensor: | ||||
|   - platform: homeassistant | ||||
|     entity_id: sensor.hello_world | ||||
|     id: ha_hello_world | ||||
|   - platform: aht10   | ||||
|     temperature: | ||||
|       name: "Temperature" | ||||
|     humidity: | ||||
|       name: "Humidity" | ||||
|   - platform: am2320 | ||||
|     temperature: | ||||
|       name: "Temperature" | ||||
|   | ||||
		Reference in New Issue
	
	Block a user