Update according to comments

This commit is contained in:
Otto winter 2021-08-03 19:52:37 +02:00
parent 157e2731cc
commit a8a14c761b
No known key found for this signature in database
GPG key ID: 48ED2DDB96D7682C
5 changed files with 89 additions and 145 deletions

View file

@ -139,8 +139,6 @@ class I2CDevice;
class I2CMultiplexer; class I2CMultiplexer;
class I2CRegister { class I2CRegister {
public: public:
I2CRegister(I2CDevice *parent, uint8_t a_register) : parent_(parent), register_(a_register) {}
I2CRegister &operator=(uint8_t value); I2CRegister &operator=(uint8_t value);
I2CRegister &operator=(const std::vector<uint8_t> &value); I2CRegister &operator=(const std::vector<uint8_t> &value);
I2CRegister &operator&=(uint8_t value); I2CRegister &operator&=(uint8_t value);
@ -149,6 +147,10 @@ class I2CRegister {
uint8_t get(); uint8_t get();
protected: protected:
friend class I2CDevice;
I2CRegister(I2CDevice *parent, uint8_t a_register) : parent_(parent), register_(a_register) {}
I2CDevice *parent_; I2CDevice *parent_;
uint8_t register_; uint8_t register_;
}; };

View file

@ -5,83 +5,12 @@
namespace esphome { namespace esphome {
namespace ltr390 { namespace ltr390 {
static const char *TAG = "ltr390"; static const char *const TAG = "ltr390";
static const float GAINVALUES[5] = {1.0, 3.0, 6.0, 9.0, 18.0}; static const float GAINVALUES[5] = {1.0, 3.0, 6.0, 9.0, 18.0};
static const float RESOLUTIONVALUE[6] = {4.0, 2.0, 1.0, 0.5, 0.25, 0.125}; static const float RESOLUTIONVALUE[6] = {4.0, 2.0, 1.0, 0.5, 0.25, 0.125};
static const uint32_t MODEADDRESSES[2] = {0x0D, 0x10}; static const uint32_t MODEADDRESSES[2] = {0x0D, 0x10};
bool LTR390Component::enabled_() {
std::bitset<8> crtl_value(this->ctrl_reg_->get());
return (bool) crtl_value[LTR390_CTRL_EN];
}
void LTR390Component::enable_(bool en) {
std::bitset<8> crtl_value(this->ctrl_reg_->get());
crtl_value[LTR390_CTRL_EN] = en;
*this->ctrl_reg_ = crtl_value.to_ulong();
}
bool LTR390Component::reset_() {
std::bitset<8> crtl_value(this->ctrl_reg_->get());
crtl_value[LTR390_CTRL_RST] = true;
*this->ctrl_reg_ = crtl_value.to_ulong();
delay(10);
// Read after reset
crtl_value = std::bitset<8>(this->ctrl_reg_->get());
return !(bool) crtl_value.to_ulong();
}
void LTR390Component::set_mode_(LTR390MODE mode) {
std::bitset<8> crtl_value(this->ctrl_reg_->get());
crtl_value[LTR390_CTRL_MODE] = mode;
*this->ctrl_reg_ = crtl_value.to_ulong();
}
LTR390MODE LTR390Component::get_mode_() {
std::bitset<8> crtl_value(this->ctrl_reg_->get());
return (LTR390MODE)(int) crtl_value[LTR390_CTRL_MODE];
}
void LTR390Component::set_gain_(LTR390GAIN gain) { *this->gain_reg_ = gain; }
LTR390GAIN LTR390Component::get_gain_() {
std::bitset<8> gain_value(this->gain_reg_->get());
return (LTR390GAIN) gain_value.to_ulong();
}
void LTR390Component::set_resolution_(LTR390RESOLUTION res) {
std::bitset<8> res_value(this->res_reg_->get());
std::bitset<3> new_res_value(res);
for (int i = 0; i < 3; i++) {
res_value[4 + i] = new_res_value[i];
}
*this->res_reg_ = res_value.to_ulong();
}
LTR390RESOLUTION LTR390Component::get_resolution_() {
std::bitset<8> res_value(this->res_reg_->get());
std::bitset<3> output_value(0);
for (int i = 0; i < 3; i++) {
output_value[i] = res_value[4 + i];
}
return (LTR390RESOLUTION) output_value.to_ulong();
}
bool LTR390Component::new_data_available_() {
std::bitset<8> status_value(this->status_reg_->get());
return (bool) status_value[3];
}
uint32_t little_endian_bytes_to_int(const uint8_t *buffer, uint8_t num_bytes) { uint32_t little_endian_bytes_to_int(const uint8_t *buffer, uint8_t num_bytes) {
uint32_t value = 0; uint32_t value = 0;
@ -93,22 +22,39 @@ uint32_t little_endian_bytes_to_int(const uint8_t *buffer, uint8_t num_bytes) {
return value; return value;
} }
uint32_t LTR390Component::read_sensor_data_(LTR390MODE mode) { optional<uint32_t> LTR390Component::read_sensor_data_(LTR390MODE mode) {
const uint8_t num_bytes = 3; const uint8_t num_bytes = 3;
uint8_t buffer[num_bytes]; uint8_t buffer[num_bytes];
while (!this->new_data_available_()) { // Wait until data available
ESP_LOGD(TAG, "WAITING FOR DATA"); const uint32_t now = millis();
while (true) {
std::bitset<8> status = this->reg(LTR390_MAIN_STATUS).get();
bool available = status[3];
if (available)
break;
if (millis() - now > 100) {
ESP_LOGW(TAG, "Sensor didn't return any data, aborting");
return {};
}
ESP_LOGD(TAG, "Waiting for data");
delay(2); delay(2);
} }
this->read_bytes(MODEADDRESSES[mode], buffer, num_bytes); if (!this->read_bytes(MODEADDRESSES[mode], buffer, num_bytes)) {
ESP_LOGW(TAG, "Reading data from sensor failed!");
return {};
}
return little_endian_bytes_to_int(buffer, num_bytes); return little_endian_bytes_to_int(buffer, num_bytes);
} }
void LTR390Component::read_als_() { void LTR390Component::read_als_() {
uint32_t als = this->read_sensor_data_(LTR390_MODE_ALS); auto val = this->read_sensor_data_(LTR390_MODE_ALS);
if (!val.has_value())
return;
uint32_t als = *val;
if (this->light_sensor_ != nullptr) { if (this->light_sensor_ != nullptr) {
float lux = (0.6 * als) / (GAINVALUES[this->gain_] * RESOLUTIONVALUE[this->res_]) * this->wfac_; float lux = (0.6 * als) / (GAINVALUES[this->gain_] * RESOLUTIONVALUE[this->res_]) * this->wfac_;
@ -121,7 +67,10 @@ void LTR390Component::read_als_() {
} }
void LTR390Component::read_uvs_() { void LTR390Component::read_uvs_() {
uint32_t uv = this->read_sensor_data_(LTR390_MODE_UVS); auto val = this->read_sensor_data_(LTR390_MODE_UVS);
if (!val.has_value())
return;
uint32_t uv = *val;
if (this->uvi_sensor_ != nullptr) { if (this->uvi_sensor_ != nullptr) {
this->uvi_sensor_->publish_state(uv / LTR390_SENSITIVITY * this->wfac_); this->uvi_sensor_->publish_state(uv / LTR390_SENSITIVITY * this->wfac_);
@ -134,16 +83,20 @@ void LTR390Component::read_uvs_() {
void LTR390Component::read_mode_(int mode_index) { void LTR390Component::read_mode_(int mode_index) {
// Set mode // Set mode
this->set_mode_(std::get<0>(this->mode_funcs_->at(mode_index))); LTR390MODE mode = std::get<0>(this->mode_funcs_[mode_index]);
std::bitset<8> ctrl = this->reg(LTR390_MAIN_CTRL).get();
ctrl[LTR390_CTRL_MODE] = mode;
this->reg(LTR390_MAIN_CTRL) = ctrl.to_ulong();
// After the sensor integration time do the following // After the sensor integration time do the following
this->set_timeout(((unsigned int) RESOLUTIONVALUE[this->res_]) * 100, [this, mode_index]() { this->set_timeout(((uint32_t) RESOLUTIONVALUE[this->res_]) * 100, [this, mode_index]() {
// Read from the sensor // Read from the sensor
std::get<1>(this->mode_funcs_->at(mode_index))(); std::get<1>(this->mode_funcs_[mode_index])();
// If there are more modes to read then begin the next // If there are more modes to read then begin the next
// otherwise stop // otherwise stop
if (mode_index + 1 < this->mode_funcs_->size()) { if (mode_index + 1 < this->mode_funcs_.size()) {
this->read_mode_(mode_index + 1); this->read_mode_(mode_index + 1);
} else { } else {
this->reading_ = false; this->reading_ = false;
@ -154,40 +107,48 @@ void LTR390Component::read_mode_(int mode_index) {
void LTR390Component::setup() { void LTR390Component::setup() {
ESP_LOGCONFIG(TAG, "Setting up ltr390..."); ESP_LOGCONFIG(TAG, "Setting up ltr390...");
this->ctrl_reg_ = new i2c::I2CRegister(this, LTR390_MAIN_CTRL); // reset
this->status_reg_ = new i2c::I2CRegister(this, LTR390_MAIN_STATUS); std::bitset<8> ctrl = this->reg(LTR390_MAIN_CTRL).get();
this->gain_reg_ = new i2c::I2CRegister(this, LTR390_GAIN); ctrl[LTR390_CTRL_RST] = true;
this->res_reg_ = new i2c::I2CRegister(this, LTR390_MEAS_RATE); this->reg(LTR390_MAIN_CTRL) = ctrl.to_ulong();
delay(10);
this->reset_(); // Enable
ctrl = this->reg(LTR390_MAIN_CTRL).get();
ctrl[LTR390_CTRL_EN] = true;
this->reg(LTR390_MAIN_CTRL) = ctrl.to_ulong();
this->enable_(true); // check enabled
ESP_LOGD(TAG, "%s", this->enabled_() ? "ENABLED" : "DISABLED"); ctrl = this->reg(LTR390_MAIN_CTRL).get();
if (!this->enabled_()) { bool enabled = ctrl[LTR390_CTRL_EN];
if (!enabled) {
ESP_LOGW(TAG, "Sensor didn't respond with enabled state");
this->mark_failed(); this->mark_failed();
return; return;
} }
// Set gain // Set gain
this->set_gain_(this->gain_); this->reg(LTR390_GAIN) = gain_;
// Set resolution // Set resolution
this->set_resolution_(this->res_); uint8_t res = this->reg(LTR390_MEAS_RATE).get();
// resolution is in bits 5-7
res &= ~0b01110000;
res |= res << 4;
this->reg(LTR390_MEAS_RATE) = res;
// Set sensor read state // Set sensor read state
this->reading_ = false; this->reading_ = false;
// Create a list of modes and corresponding read functions
this->mode_funcs_ = new std::vector<std::tuple<LTR390MODE, std::function<void(void)> > >();
// If we need the light sensor then add to the list // If we need the light sensor then add to the list
if (this->light_sensor_ != nullptr || this->als_sensor_ != nullptr) { if (this->light_sensor_ != nullptr || this->als_sensor_ != nullptr) {
this->mode_funcs_->push_back(std::make_tuple(LTR390_MODE_ALS, std::bind(&LTR390Component::read_als_, this))); this->mode_funcs_.emplace_back(LTR390_MODE_ALS, std::bind(&LTR390Component::read_als_, this));
} }
// If we need the UV sensor then add to the list // If we need the UV sensor then add to the list
if (this->uvi_sensor_ != nullptr || this->uv_sensor_ != nullptr) { if (this->uvi_sensor_ != nullptr || this->uv_sensor_ != nullptr) {
this->mode_funcs_->push_back(std::make_tuple(LTR390_MODE_UVS, std::bind(&LTR390Component::read_uvs_, this))); this->mode_funcs_.emplace_back(LTR390_MODE_UVS, std::bind(&LTR390Component::read_uvs_, this));
} }
} }

View file

@ -1,10 +1,9 @@
#pragma once #pragma once
#include "esphome/core/component.h" #include "esphome/core/component.h"
#include "esphome/core/optional.h"
#include "esphome/components/sensor/sensor.h" #include "esphome/components/sensor/sensor.h"
#include "esphome/components/i2c/i2c.h" #include "esphome/components/i2c/i2c.h"
#include <map>
#include <atomic>
#include <tuple> #include <tuple>
namespace esphome { namespace esphome {
@ -23,8 +22,7 @@ static const uint8_t LTR390_MEAS_RATE = 0x04;
static const uint8_t LTR390_GAIN = 0x05; static const uint8_t LTR390_GAIN = 0x05;
static const uint8_t LTR390_PART_ID = 0x06; static const uint8_t LTR390_PART_ID = 0x06;
static const uint8_t LTR390_MAIN_STATUS = 0x07; static const uint8_t LTR390_MAIN_STATUS = 0x07;
static const float LTR390_SENSITIVITY = 2300.0;
#define LTR390_SENSITIVITY 2300.0
// Sensing modes // Sensing modes
enum LTR390MODE { enum LTR390MODE {
@ -68,36 +66,17 @@ class LTR390Component : public PollingComponent, public i2c::I2CDevice {
void set_uv_sensor(sensor::Sensor *uv_sensor) { this->uv_sensor_ = uv_sensor; } void set_uv_sensor(sensor::Sensor *uv_sensor) { this->uv_sensor_ = uv_sensor; }
protected: protected:
bool enabled_(); optional<uint32_t> read_sensor_data_(LTR390MODE mode);
void enable_(bool en);
bool reset_();
void set_mode_(LTR390MODE mode);
LTR390MODE get_mode_();
void set_gain_(LTR390GAIN gain);
LTR390GAIN get_gain_();
void set_resolution_(LTR390RESOLUTION res);
LTR390RESOLUTION get_resolution_();
bool new_data_available_();
uint32_t read_sensor_data_(LTR390MODE mode);
void read_als_(); void read_als_();
void read_uvs_(); void read_uvs_();
void read_mode_(int mode_index); void read_mode_(int mode_index);
std::atomic<bool> reading_; bool reading_;
std::vector<std::tuple<LTR390MODE, std::function<void(void)> > > *mode_funcs_; // a list of modes and corresponding read functions
std::vector<std::tuple<LTR390MODE, std::function<void()>>> mode_funcs_;
i2c::I2CRegister *ctrl_reg_;
i2c::I2CRegister *status_reg_;
i2c::I2CRegister *gain_reg_;
i2c::I2CRegister *res_reg_;
LTR390GAIN gain_; LTR390GAIN gain_;
LTR390RESOLUTION res_; LTR390RESOLUTION res_;

View file

@ -20,10 +20,10 @@ LTR390Component = ltr390_ns.class_(
"LTR390Component", cg.PollingComponent, i2c.I2CDevice "LTR390Component", cg.PollingComponent, i2c.I2CDevice
) )
CONF_ALS = "als" CONF_AMBIENT_LIGHT = "ambient_light"
CONF_UVI = "uvi" CONF_UV_INDEX = "uv_index"
CONF_UV = "uv" CONF_UV = "uv"
CONF_WFAC = "wfac" CONF_WINDOW_CORRECTION_FACTOR = "window_correction_factor"
UNIT_COUNTS = "#" UNIT_COUNTS = "#"
UNIT_UVI = "UVI" UNIT_UVI = "UVI"
@ -47,17 +47,17 @@ RES_OPTIONS = {
13: LTR390RESOLUTION.LTR390_RESOLUTION_13BIT, 13: LTR390RESOLUTION.LTR390_RESOLUTION_13BIT,
} }
CONFIG_SCHEMA = ( CONFIG_SCHEMA = cv.All(
cv.Schema( cv.Schema(
{ {
cv.GenerateID(): cv.declare_id(LTR390Component), cv.GenerateID(): cv.declare_id(LTR390Component),
cv.Optional(CONF_LIGHT): sensor.sensor_schema( cv.Optional(CONF_LIGHT): sensor.sensor_schema(
UNIT_LUX, ICON_BRIGHTNESS_5, 1, DEVICE_CLASS_ILLUMINANCE UNIT_LUX, ICON_BRIGHTNESS_5, 1, DEVICE_CLASS_ILLUMINANCE
), ),
cv.Optional(CONF_ALS): sensor.sensor_schema( cv.Optional(CONF_AMBIENT_LIGHT): sensor.sensor_schema(
UNIT_COUNTS, ICON_BRIGHTNESS_5, 1, DEVICE_CLASS_ILLUMINANCE UNIT_COUNTS, ICON_BRIGHTNESS_5, 1, DEVICE_CLASS_ILLUMINANCE
), ),
cv.Optional(CONF_UVI): sensor.sensor_schema( cv.Optional(CONF_UV_INDEX): sensor.sensor_schema(
UNIT_UVI, ICON_BRIGHTNESS_5, 5, DEVICE_CLASS_ILLUMINANCE UNIT_UVI, ICON_BRIGHTNESS_5, 5, DEVICE_CLASS_ILLUMINANCE
), ),
cv.Optional(CONF_UV): sensor.sensor_schema( cv.Optional(CONF_UV): sensor.sensor_schema(
@ -65,32 +65,34 @@ CONFIG_SCHEMA = (
), ),
cv.Optional(CONF_GAIN, default="X3"): cv.enum(GAIN_OPTIONS), cv.Optional(CONF_GAIN, default="X3"): cv.enum(GAIN_OPTIONS),
cv.Optional(CONF_RESOLUTION, default=18): cv.enum(RES_OPTIONS), cv.Optional(CONF_RESOLUTION, default=18): cv.enum(RES_OPTIONS),
cv.Optional(CONF_WFAC, default=1.0): cv.float_range(min=1.0), cv.Optional(CONF_WINDOW_CORRECTION_FACTOR, default=1.0): cv.float_range(
min=1.0
),
} }
) )
.extend(cv.polling_component_schema("60s")) .extend(cv.polling_component_schema("60s"))
.extend(i2c.i2c_device_schema(0x53)) .extend(i2c.i2c_device_schema(0x53)),
cv.has_at_least_one_key(CONF_LIGHT, CONF_AMBIENT_LIGHT, CONF_UV_INDEX, CONF_UV),
) )
TYPES = { TYPES = {
CONF_LIGHT: "set_light_sensor", CONF_LIGHT: "set_light_sensor",
CONF_ALS: "set_als_sensor", CONF_AMBIENT_LIGHT: "set_als_sensor",
CONF_UVI: "set_uvi_sensor", CONF_UV_INDEX: "set_uvi_sensor",
CONF_UV: "set_uv_sensor", CONF_UV: "set_uv_sensor",
} }
def to_code(config): async def to_code(config):
var = cg.new_Pvariable(config[CONF_ID]) var = cg.new_Pvariable(config[CONF_ID])
yield cg.register_component(var, config) await cg.register_component(var, config)
yield i2c.register_i2c_device(var, config) await i2c.register_i2c_device(var, config)
cg.add(var.set_gain_value(config[CONF_GAIN])) cg.add(var.set_gain_value(config[CONF_GAIN]))
cg.add(var.set_res_value(config[CONF_RESOLUTION])) cg.add(var.set_res_value(config[CONF_RESOLUTION]))
cg.add(var.set_wfac_value(config[CONF_WFAC])) cg.add(var.set_wfac_value(config[CONF_WINDOW_CORRECTION_FACTOR]))
for key, funcName in TYPES.items(): for key, funcName in TYPES.items():
if key in config: if key in config:
sens = yield sensor.new_sensor(config[key]) sens = await sensor.new_sensor(config[key])
cg.add(getattr(var, funcName)(sens)) cg.add(getattr(var, funcName)(sens))

View file

@ -239,15 +239,15 @@ sensor:
- platform: ltr390 - platform: ltr390
uv: uv:
name: "LTR390 UV" name: "LTR390 UV"
uvi: uv_index:
name: "LTR390 UVI" name: "LTR390 UVI"
light: light:
name: "LTR390 Light" name: "LTR390 Light"
als: ambient_light:
name: "LTR390 ALS" name: "LTR390 ALS"
gain: "X3" gain: "X3"
resolution: 18 resolution: 18
wfac: 1.0 window_correction_factor: 1.0
address: 0x53 address: 0x53
update_interval: 60s update_interval: 60s
- platform: sgp40 - platform: sgp40