mirror of
https://github.com/esphome/esphome.git
synced 2024-11-29 10:14:13 +01:00
Support for AM2320 temperature and humidity sensor (#554)
Support for AM2320 temperature and humidity sensor Co-authored-by: Mikkonen Teemu <teemu.mikkonen@iki.fi>
This commit is contained in:
parent
b7daee533a
commit
3b48aa5911
5 changed files with 171 additions and 0 deletions
0
esphome/components/am2320/__init__.py
Normal file
0
esphome/components/am2320/__init__.py
Normal file
107
esphome/components/am2320/am2320.cpp
Normal file
107
esphome/components/am2320/am2320.cpp
Normal file
|
@ -0,0 +1,107 @@
|
||||||
|
// Implementation based on:
|
||||||
|
// - ESPEasy: https://github.com/letscontrolit/ESPEasy/blob/mega/src/_P034_DHT12.ino
|
||||||
|
// - DHT12_sensor_library: https://github.com/xreef/DHT12_sensor_library/blob/master/DHT12.cpp
|
||||||
|
// - Arduino - AM2320: https://github.com/EngDial/AM2320/blob/master/src/AM2320.cpp
|
||||||
|
|
||||||
|
#include "am2320.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace am2320 {
|
||||||
|
|
||||||
|
static const char *TAG = "am2320";
|
||||||
|
|
||||||
|
// ---=== Calc CRC16 ===---
|
||||||
|
uint16_t crc_16(uint8_t *ptr, uint8_t length) {
|
||||||
|
uint16_t crc = 0xFFFF;
|
||||||
|
uint8_t i;
|
||||||
|
//------------------------------
|
||||||
|
while (length--) {
|
||||||
|
crc ^= *ptr++;
|
||||||
|
for (i = 0; i < 8; i++)
|
||||||
|
if ((crc & 0x01) != 0) {
|
||||||
|
crc >>= 1;
|
||||||
|
crc ^= 0xA001;
|
||||||
|
} else
|
||||||
|
crc >>= 1;
|
||||||
|
}
|
||||||
|
return crc;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AM2320Component::update() {
|
||||||
|
uint8_t data[8];
|
||||||
|
data[0] = 0;
|
||||||
|
data[1] = 4;
|
||||||
|
if (!this->read_data_(data)) {
|
||||||
|
this->status_set_warning();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
float temperature = (((data[4] & 0x7F) << 8) + data[5]) / 10.0;
|
||||||
|
temperature = (data[4] & 0x80) ? -temperature : temperature;
|
||||||
|
float humidity = ((data[2] << 8) + data[3]) / 10.0;
|
||||||
|
|
||||||
|
ESP_LOGD(TAG, "Got temperature=%.1f°C humidity=%.1f%%", temperature, humidity);
|
||||||
|
if (this->temperature_sensor_ != nullptr)
|
||||||
|
this->temperature_sensor_->publish_state(temperature);
|
||||||
|
if (this->humidity_sensor_ != nullptr)
|
||||||
|
this->humidity_sensor_->publish_state(humidity);
|
||||||
|
this->status_clear_warning();
|
||||||
|
}
|
||||||
|
void AM2320Component::setup() {
|
||||||
|
ESP_LOGCONFIG(TAG, "Setting up AM2320...");
|
||||||
|
uint8_t data[8];
|
||||||
|
data[0] = 0;
|
||||||
|
data[1] = 4;
|
||||||
|
if (!this->read_data_(data)) {
|
||||||
|
this->mark_failed();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void AM2320Component::dump_config() {
|
||||||
|
ESP_LOGD(TAG, "AM2320:");
|
||||||
|
LOG_I2C_DEVICE(this);
|
||||||
|
if (this->is_failed()) {
|
||||||
|
ESP_LOGE(TAG, "Communication with AM2320 failed!");
|
||||||
|
}
|
||||||
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
}
|
||||||
|
float AM2320Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
|
bool AM2320Component::read_bytes_(uint8_t a_register, uint8_t *data, uint8_t len, uint32_t conversion) {
|
||||||
|
if (!this->write_bytes(a_register, data, 2)) {
|
||||||
|
ESP_LOGW(TAG, "Writing bytes for AM2320 failed!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (conversion > 0)
|
||||||
|
delay(conversion);
|
||||||
|
return this->parent_->raw_receive(this->address_, data, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AM2320Component::read_data_(uint8_t *data) {
|
||||||
|
// Wake up
|
||||||
|
this->write_bytes(0, data, 0);
|
||||||
|
|
||||||
|
// Write instruction 3, 2 bytes, get 8 bytes back (2 preamble, 2 bytes temperature, 2 bytes humidity, 2 bytes CRC)
|
||||||
|
if (!this->read_bytes_(3, data, 8, 2)) {
|
||||||
|
ESP_LOGW(TAG, "Updating AM2320 failed!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t checksum;
|
||||||
|
|
||||||
|
checksum = data[7] << 8;
|
||||||
|
checksum += data[6];
|
||||||
|
|
||||||
|
if (crc_16(data, 6) != checksum) {
|
||||||
|
ESP_LOGW(TAG, "AM2320 Checksum invalid!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace am2320
|
||||||
|
} // namespace esphome
|
29
esphome/components/am2320/am2320.h
Normal file
29
esphome/components/am2320/am2320.h
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
#include "esphome/components/sensor/sensor.h"
|
||||||
|
#include "esphome/components/i2c/i2c.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace am2320 {
|
||||||
|
|
||||||
|
class AM2320Component : public PollingComponent, public i2c::I2CDevice {
|
||||||
|
public:
|
||||||
|
void setup() override;
|
||||||
|
void dump_config() override;
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
void update() 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:
|
||||||
|
bool read_data_(uint8_t *data);
|
||||||
|
bool read_bytes_(uint8_t a_register, uint8_t *data, uint8_t len, uint32_t conversion = 0);
|
||||||
|
|
||||||
|
sensor::Sensor *temperature_sensor_;
|
||||||
|
sensor::Sensor *humidity_sensor_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace am2320
|
||||||
|
} // namespace esphome
|
30
esphome/components/am2320/sensor.py
Normal file
30
esphome/components/am2320/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']
|
||||||
|
|
||||||
|
am2320_ns = cg.esphome_ns.namespace('am2320')
|
||||||
|
AM2320Component = am2320_ns.class_('AM2320Component', cg.PollingComponent, i2c.I2CDevice)
|
||||||
|
|
||||||
|
CONFIG_SCHEMA = cv.Schema({
|
||||||
|
cv.GenerateID(): cv.declare_id(AM2320Component),
|
||||||
|
cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(UNIT_CELSIUS, ICON_THERMOMETER, 1),
|
||||||
|
cv.Optional(CONF_HUMIDITY): sensor.sensor_schema(UNIT_PERCENT, ICON_WATER_PERCENT, 1),
|
||||||
|
}).extend(cv.polling_component_schema('60s')).extend(i2c.i2c_device_schema(0x5C))
|
||||||
|
|
||||||
|
|
||||||
|
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))
|
|
@ -94,6 +94,11 @@ sensor:
|
||||||
- platform: homeassistant
|
- platform: homeassistant
|
||||||
entity_id: sensor.hello_world
|
entity_id: sensor.hello_world
|
||||||
id: ha_hello_world
|
id: ha_hello_world
|
||||||
|
- platform: am2320
|
||||||
|
temperature:
|
||||||
|
name: "Temperature"
|
||||||
|
humidity:
|
||||||
|
name: "Humidity"
|
||||||
- platform: adc
|
- platform: adc
|
||||||
pin: VCC
|
pin: VCC
|
||||||
id: my_sensor
|
id: my_sensor
|
||||||
|
|
Loading…
Reference in a new issue