diff --git a/esphome/components/as7343/as7343.cpp b/esphome/components/as7343/as7343.cpp index 5bd0b27f5d..4fa9388de3 100644 --- a/esphome/components/as7343/as7343.cpp +++ b/esphome/components/as7343/as7343.cpp @@ -9,34 +9,67 @@ namespace as7343 { static const char *const TAG = "as7343"; -static constexpr float CONST_H = 6.6260695e-34f; -static constexpr float CONST_C = 299792458; - -static constexpr uint8_t NUM_USEFUL_CHANNELS = 13; -static constexpr uint8_t CHANNEL_IDX[NUM_USEFUL_CHANNELS] = { - AS7343_CHANNEL_405_F1, AS7343_CHANNEL_425_F2, AS7343_CHANNEL_450_FZ, AS7343_CHANNEL_475_F3, AS7343_CHANNEL_515_F4, - AS7343_CHANNEL_555_FY, AS7343_CHANNEL_550_F5, AS7343_CHANNEL_600_FXL, AS7343_CHANNEL_640_F6, AS7343_CHANNEL_690_F7, - AS7343_CHANNEL_745_F8, AS7343_CHANNEL_855_NIR, AS7343_CHANNEL_CLEAR}; - -// pimoroni -static constexpr float CHANNEL_COMPENSATION_GAIN[NUM_USEFUL_CHANNELS] = {1.84, 6.03, 4.88, 13.74, 3.37, 2.82, 6.72, +// pimoroni? +static constexpr float CHANNEL_COMPENSATION_GAIN[AS7343_NUM_CHANNELS] = {1.84, 6.03, 4.88, 13.74, 3.37, 2.82, 6.72, 2.22, 3.17, 1.95, 12.25, 1.00, 1}; -static constexpr float CHANNEL_SENS[NUM_USEFUL_CHANNELS] = {0.19402, 0.26647, 0.35741, 0.41753, 0.52235, +static constexpr float CHANNEL_SENS[AS7343_NUM_CHANNELS] = {0.19402, 0.26647, 0.35741, 0.41753, 0.52235, 0.59633, 0.56242, 0.65645, 0.68882, 0.79980, 0.70423, 0.40366, 0.38516}; -static constexpr float CHANNEL_BASIC_CORRECTIONS[NUM_USEFUL_CHANNELS] = { + +// +// 1. AS7343 Datasheet-based +// +// 1.1 Names +// static constexpr char *CHANNEL_NAMES[AS7343_NUM_CHANNELS] = {"F1", "F2", "FZ", "F3", "F4", "FY", "F5", +// "FXL", "F6", "F7", "F8", "NIR", "Clear"}; + +static const std::array CHANNEL_NAMES = { + "F1", "F2", "FZ", "F3", "F4", "FY", "F5", "FXL", "F6", "F7", "F8", "NIR", "Clear"}; +// 1.2 Each channel central wavelength in nm +static const std::array CHANNEL_NM = {405, 425, 450, 475, 515, 555, 550, + 600, 640, 690, 745, 855, 718}; +// 1.3 Each channel width in nm +static const std::array CHANNEL_NM_WIDTH = {30, 22, 55, 30, 40, 100, 35, + 80, 50, 55, 60, 54, 0}; + +// 1.4 Gain correctoin for each channel +static constexpr float CHANNEL_BASIC_CORRECTIONS[AS7343_NUM_CHANNELS] = { 1.055464349, 1.043509797, 1.029576268, 1.0175052, 1.00441899, 0.987356499, 0.957597044, 0.995863485, 1.014628964, 0.996500814, 0.933072749, 1.052236338, 0.999570232}; -static const float OFFSETS[NUM_USEFUL_CHANNELS] = {0.000281, 0.000281, 0.000281, 0.000281, 0.000281, 0.000281, - 0.000281, 0.000281, 0.000422, 0.000281, 0.000422, 0.000281}; +// 1.5 Irradiation in mW/m² per basic count ? +static constexpr float CHANNEL_IRRAD_MW_PER_BASIC_COUNT[AS7343_NUM_CHANNELS] = { + 767.5101757, 2512.765376, 2034.308898, 5730.41039, 1404.780643, 1177.586336, 2803.31385, + 923.8726968, 1322.666667, 811.8520699, 5106.962963, 417.0131368, 78.70319635}; -static constexpr float CHANNEL_NM[NUM_USEFUL_CHANNELS] = {405, 425, 450, 475, 515, 555, 550, - 600, 640, 690, 745, 855, 718}; -static constexpr float CHANNEL_NM_WIDTH[NUM_USEFUL_CHANNELS] = {30, 22, 55, 30, 40, 100, 35, 80, 50, 55, 60, 54, 0}; +// 1.6 band energy contribution precalculated based on band widths and central wavelengths (photon energy) +// value = band energy / sum of all bands energies, except for VIS +static constexpr float CHANNEL_ENERGY_CONTRIBUTION[AS7343_NUM_CHANNELS] = {0.069385773, + 0.04848841, + 0.114486525, + 0.059160501, + 0.072754014, + 0.168776203, + 0.059608686, + 0.124894391, + 0.073180307, + 0.074665125, + 0.075439565, + 0.059160501, + 0}; -static constexpr float CHANNEL_PHOTOPIC_LUMINOSITY[NUM_USEFUL_CHANNELS] = {0.0006400000000, +// +// 2. Scientific data +// +// 2.1 Photon energy for selected bands +// E = h*c/lambda +static constexpr float PHOTON_ENERGIES[AS7343_NUM_CHANNELS] = { + 4.9048E-19f, 4.67399E-19f, 4.41432E-19f, 4.18199E-19f, 3.85718E-19f, 3.57918E-19f, 3.61172E-19f, + 3.31074E-19f, 3.10382E-19f, 2.87891E-19f, 2.66637E-19f, 2.32333E-19f, 2.76664E-19f}; + +// 2.2 CIE 1923 Photopic Luminosity Function for selected bands +static constexpr float CHANNEL_PHOTOPIC_LUMINOSITY[AS7343_NUM_CHANNELS] = {0.0006400000000, 0.0073000000000, 0.0380000000000, 0.1126000000000, @@ -50,31 +83,8 @@ static constexpr float CHANNEL_PHOTOPIC_LUMINOSITY[NUM_USEFUL_CHANNELS] = {0.000 0, 0}; -// Irradiation in mW/m² per basic count -static constexpr float CHANNEL_IRRAD_MW_PER_BASIC_COUNT[NUM_USEFUL_CHANNELS] = { - 767.5101757, 2512.765376, 2034.308898, 5730.41039, 1404.780643, 1177.586336, 2803.31385, - 923.8726968, 1322.666667, 811.8520699, 5106.962963, 417.0131368, 78.70319635}; - -// E = h*c/lambda -static constexpr float PHOTON_ENERGIES[NUM_USEFUL_CHANNELS] = { - 4.9048E-19f, 4.67399E-19f, 4.41432E-19f, 4.18199E-19f, 3.85718E-19f, 3.57918E-19f, 3.61172E-19f, - 3.31074E-19f, 3.10382E-19f, 2.87891E-19f, 2.66637E-19f, 2.32333E-19f, 2.76664E-19f}; - // static constexpr float CHANNEL_CONTRIB[NUM_USEFUL_CHANNELS] = { // 0.0603622, 0.0442656, 0.110664, 0.0603622, 0.0804829, 0.201207, 0.0704225, 0.160966, 0.100604, 0.110664, 0}; -static constexpr float CHANNEL_ENERGY_CONTRIBUTION[NUM_USEFUL_CHANNELS] = {0.069385773, - 0.04848841, - 0.114486525, - 0.059160501, - 0.072754014, - 0.168776203, - 0.059608686, - 0.124894391, - 0.073180307, - 0.074665125, - 0.075439565, - 0.059160501, - 0}; void AS7343Component::setup() { ESP_LOGCONFIG(TAG, "Setting up AS7343..."); @@ -143,58 +153,34 @@ void AS7343Component::dump_config() { float AS7343Component::get_setup_priority() const { return setup_priority::DATA; } +void log13_f(const char *TAG, const char *str, const std::array &arr) { + ESP_LOGD(TAG, "%s, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, ", str, arr[0], + arr[1], arr[2], arr[3], arr[4], arr[5], arr[6], arr[7], arr[8], arr[9], arr[10], arr[11], arr[12]); +} + +void log13_d(const char *TAG, const char *str, const std::array &arr) { + ESP_LOGD(TAG, "%s, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, ", str, arr[0], arr[1], arr[2], arr[3], arr[4], + arr[5], arr[6], arr[7], arr[8], arr[9], arr[10], arr[11], arr[12]); +} + +void log13_s(const char *TAG, const char *str, const std::array &arr) { + ESP_LOGD(TAG, "%s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s, ", str, arr[0], arr[1], arr[2], arr[3], arr[4], + arr[5], arr[6], arr[7], arr[8], arr[9], arr[10], arr[11], arr[12]); +} + void AS7343Component::update() { - // this->optimizer_(1000); - // delay(20); - this->enable_spectral_measurement(true); - this->read_18_channels(this->channel_readings_); + this->read_all_channels(); this->enable_spectral_measurement(false); + this->calculate_basic_counts(); - for (uint8_t i = 0; i < NUM_USEFUL_CHANNELS; i++) { - ESP_LOGD(TAG, "Channel %d: %d", i, this->channel_readings_[CHANNEL_IDX[i]]); - } - this->channel_readings_[AS7343_CHANNEL_FD] = 0; - this->channel_readings_[AS7343_CHANNEL_FD_0] = 0; - this->channel_readings_[AS7343_CHANNEL_FD_1] = 0; - if (0) { - this->channel_readings_[AS7343_CHANNEL_CLEAR] = 0; - this->channel_readings_[AS7343_CHANNEL_CLEAR_0] = 0; - this->channel_readings_[AS7343_CHANNEL_CLEAR_1] = 0; - this->channel_readings_[AS7343_CHANNEL_FD] = 0; - this->channel_readings_[AS7343_CHANNEL_FD_0] = 0; - this->channel_readings_[AS7343_CHANNEL_FD_1] = 0; - if (this->spectral_post_process_()) { - ESP_LOGW(TAG, "Spectral post process - need to repeat 1"); - delay(20); - this->enable_spectral_measurement(true); - this->read_18_channels(this->channel_readings_); - this->enable_spectral_measurement(false); - this->channel_readings_[AS7343_CHANNEL_CLEAR] = 0; - this->channel_readings_[AS7343_CHANNEL_CLEAR_0] = 0; - this->channel_readings_[AS7343_CHANNEL_CLEAR_1] = 0; - this->channel_readings_[AS7343_CHANNEL_FD] = 0; - this->channel_readings_[AS7343_CHANNEL_FD_0] = 0; - this->channel_readings_[AS7343_CHANNEL_FD_1] = 0; - } - } - - if (this->spectral_post_process_(false)) { - ESP_LOGW(TAG, "Spectral post process - need to repeat 2"); - } - - AS7343Gain gain = this->readings_gain_; - uint8_t atime = get_atime(); - uint16_t astep = get_astep(); - - float tint_ms = (1 + atime) * (1 + astep) * 2.78 / 1000; // us to ms - float gain_x = get_gain_multiplier(gain); - - float tint2_ms = this->get_tint_(); + log13_s(TAG, "Channel", CHANNEL_NAMES); + log13_f(TAG, "Nm", CHANNEL_NM); + log13_d(TAG, "Counts", this->readings_.raw_counts); uint16_t max_adc = this->get_maximum_spectral_adc_(); - uint16_t highest_adc = this->get_highest_value(this->channel_readings_); + uint16_t highest_adc = this->get_highest_value(this->readings_.raw_counts); if (highest_adc >= max_adc) { ESP_LOGW(TAG, "Max ADC: %u, Highest ADC: %u", max_adc, highest_adc); @@ -202,30 +188,32 @@ void AS7343Component::update() { ESP_LOGD(TAG, "Max ADC: %u, Highest ADC: %u", max_adc, highest_adc); } - ESP_LOGD(TAG, " ,Gain , %.1f,X", gain_x); - ESP_LOGD(TAG, " ,ATIME, %u,", atime); - ESP_LOGD(TAG, " ,ASTEP, %u,", astep); - ESP_LOGD(TAG, " ,TINT , %.2f,", tint_ms); - ESP_LOGD(TAG, " ,TINT2 , %.2f,", tint2_ms); + ESP_LOGD(TAG, " ,Gain , %.1f,X", this->readings_.gain_x); + ESP_LOGD(TAG, " ,ATIME, %u,", this->readings_.atime); + ESP_LOGD(TAG, " ,ASTEP, %u,", this->readings_.astep); + ESP_LOGD(TAG, " ,TINT , %.2f,", this->readings_.t_int); - ESP_LOGD(TAG, ",nm, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, ", CHANNEL_NM[0], - CHANNEL_NM[1], CHANNEL_NM[2], CHANNEL_NM[3], CHANNEL_NM[4], CHANNEL_NM[5], CHANNEL_NM[6], CHANNEL_NM[7], - CHANNEL_NM[8], CHANNEL_NM[9], CHANNEL_NM[10], CHANNEL_NM[11], CHANNEL_NM[12]); - ESP_LOGD(TAG, ",counts, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, ", - this->channel_readings_[CHANNEL_IDX[0]], this->channel_readings_[CHANNEL_IDX[1]], - this->channel_readings_[CHANNEL_IDX[2]], this->channel_readings_[CHANNEL_IDX[3]], - this->channel_readings_[CHANNEL_IDX[4]], this->channel_readings_[CHANNEL_IDX[5]], - this->channel_readings_[CHANNEL_IDX[6]], this->channel_readings_[CHANNEL_IDX[7]], - this->channel_readings_[CHANNEL_IDX[8]], this->channel_readings_[CHANNEL_IDX[9]], - this->channel_readings_[CHANNEL_IDX[10]], this->channel_readings_[CHANNEL_IDX[11]], - this->channel_readings_[CHANNEL_IDX[12]]); + // ESP_LOGD(TAG, ",nm, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, ", CHANNEL_NM[0], + // CHANNEL_NM[1], CHANNEL_NM[2], CHANNEL_NM[3], CHANNEL_NM[4], CHANNEL_NM[5], CHANNEL_NM[6], CHANNEL_NM[7], + // CHANNEL_NM[8], CHANNEL_NM[9], CHANNEL_NM[10], CHANNEL_NM[11], CHANNEL_NM[12]); + // ESP_LOGD(TAG, ",counts, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, ", + // this->channel_readings_[CHANNEL_IDX[0]], this->channel_readings_[CHANNEL_IDX[1]], + // this->channel_readings_[CHANNEL_IDX[2]], this->channel_readings_[CHANNEL_IDX[3]], + // this->channel_readings_[CHANNEL_IDX[4]], this->channel_readings_[CHANNEL_IDX[5]], + // this->channel_readings_[CHANNEL_IDX[6]], this->channel_readings_[CHANNEL_IDX[7]], + // this->channel_readings_[CHANNEL_IDX[8]], this->channel_readings_[CHANNEL_IDX[9]], + // this->channel_readings_[CHANNEL_IDX[10]], this->channel_readings_[CHANNEL_IDX[11]], + // this->channel_readings_[CHANNEL_IDX[12]]); float irradiance; float lux; - this->calculate_irradiance(tint_ms, gain_x, irradiance, lux, gain); + float ppfd; + + this->calculate_irradiance(irradiance, lux); + this->calculate_ppfd(ppfd); + ESP_LOGD(TAG, " ,Irradiance, %f, W/m²", irradiance); ESP_LOGD(TAG, " ,Lux solar , %f, lx", lux); - float ppfd = this->calculate_ppfd(tint_ms, gain_x, gain); ESP_LOGD(TAG, " ,PPFD , %.2f, µmol/s⋅m²", ppfd); if (this->illuminance_ != nullptr) { @@ -239,17 +227,18 @@ void AS7343Component::update() { if (this->ppfd_ != nullptr) { this->ppfd_->publish_state(ppfd); } + uint8_t i = 0; float max_val = 0; - float normalized_readings[NUM_USEFUL_CHANNELS]; - for (i = 0; i < NUM_USEFUL_CHANNELS; i++) { - normalized_readings[i] = (float) this->channel_readings_[CHANNEL_IDX[i]] / CHANNEL_SENS[i]; + float normalized_readings[AS7343_NUM_CHANNELS]; + for (i = 0; i < AS7343_NUM_CHANNELS; i++) { + normalized_readings[i] = (float) this->readings_.basic_counts[i] / CHANNEL_SENS[i]; if (max_val < normalized_readings[i]) { max_val = normalized_readings[i]; } } - for (i = 0; i < NUM_USEFUL_CHANNELS; i++) { + for (i = 0; i < AS7343_NUM_CHANNELS; i++) { normalized_readings[i] = normalized_readings[i] * 100 / max_val; } @@ -344,64 +333,74 @@ bool AS7343Component::change_gain(AS7343Gain gain) { return this->write_byte((uint8_t) AS7343Registers::CFG1, gain); } -float AS7343Component::calculate_ppfd(float tint_ms, float gain_x, AS7343Gain gain) { - float pfd = 0; - float bc[NUM_USEFUL_CHANNELS] = {0}; - float bcc[NUM_USEFUL_CHANNELS] = {0}; +void AS7343Component::calculate_basic_counts() { + float divisor = this->readings_.gain_x * this->readings_.t_int; + for (size_t i = 0; i < AS7343_NUM_CHANNELS; i++) { + // 1. raw -> basic + float basic_count = this->readings_.raw_counts[i] / divisor; + // 2. gain correction + basic_count *= AS7343_GAIN_CORRECTION[(uint8_t) this->readings_.gain][i]; - for (uint8_t i = 0; i < NUM_USEFUL_CHANNELS - 1; i++) { - float basic_count = this->channel_readings_[CHANNEL_IDX[i]] / (gain_x * tint_ms); - bc[i] = basic_count * AS7343_GAIN_CORRECTION[(uint8_t) gain][i]; - bcc[i] = basic_count / CHANNEL_SENS[i]; - this->channel_basic_readings_[i] = bcc[i]; + this->readings_.basic_counts[i] = basic_count; + } +} + +void AS7343Component::calculate_ppfd(float &ppfd) { + /* + Given the spectral irradiance Iλ, defined as the radiant flux + per unit wavelength and unit area, the photon flux density + in the PAR wavelength range can be calculated using the + following expression: + + φ = ∫ Iλ(λ) * dλ / (hc/λ) (1) + + where h is Planck’s constant, c is the speed of light, and λ is the wavelength. + + The PPFD is simply the photon density flux measured in units of + micro-moles of photons per square meter per second (µmol m−2 s−1), + where a mole of photons is defined as NA = 6.022×10^23 (Avogadro’s number) photons. + + PPFD (or Qpar) can be calculated as + Qpar = φ / (NA * 10^6) (2) + + Combining (1) and (2) gives the following expression for PPFD: + + Qpar = ∫ Iλ(λ) * dλ / (hc/λ * NA * 10^6) (3) + + where 1/hc/NA*10^6 is a constant equal to 0.836e-2. + + */ + ppfd = 0; + + // assume we integrate using rectangles - mid point is channel wavelength, width is channel width in nm + for (uint8_t i = 0; i < AS7343_NUM_CHANNELS; i++) { if (CHANNEL_NM[i] < 400 || CHANNEL_NM[i] > 700) { continue; } - // float irradiance_in_w_per_m2 = basic_count * CHANNEL_IRRAD_MW_PER_BASIC_COUNT[i] /1000 ; + // Iλ(λ) + float irradiance_in_w_per_m2 = this->readings_.basic_counts[i] * CHANNEL_IRRAD_MW_PER_BASIC_COUNT[i] / 1000; + float photon_flux = irradiance_in_w_per_m2 * CHANNEL_NM[i] * 0.836e-2; - float irradiance_in_w_per_m2 = basic_count * CHANNEL_IRRAD_MW_PER_BASIC_COUNT[i] / 10000; - // probably is it not mW/m2 but uW/cm2!!!! so try divide by 10k not 1000 - // 1 W/m2 = 100 uW/cm2 - - // // https://www.berthold.com/en/bioanalytic/knowledge/faq/irradiance-to-photon-flux/ - // float photon_flux = watts * CHANNEL_NM[i] * 0.836e-2; - // photon_flux *= CHANNEL_ENERGY_CONTRIBUTION[i]; - - irradiance_in_w_per_m2 *= CHANNEL_ENERGY_CONTRIBUTION[i]; - - float photon_count = irradiance_in_w_per_m2 / PHOTON_ENERGIES[i]; - float pfd_in_micromols = 1e6f * photon_count / 6.02214179e23f; - - pfd += pfd_in_micromols; + // assume channels cover whole range + ppfd += photon_flux * CHANNEL_NM_WIDTH[i] / 1e9f; // nm to meters } - - ESP_LOGD(TAG, ",basic_counts, %f, %f, %f, %f, %f, %f, %f, %f, %f, %f, %f, %f", bc[0], bc[1], bc[2], bc[3], bc[4], - bc[5], bc[6], bc[7], bc[8], bc[9], bc[10], bc[11]); - ESP_LOGD(TAG, ",basic_counts_sens_corrected, %f, %f, %f, %f, %f, %f, %f, %f, %f, %f, %f, %f", bcc[0], bcc[1], bcc[2], - bcc[3], bcc[4], bcc[5], bcc[6], bcc[7], bcc[8], bcc[9], bcc[10], bcc[11]); - return pfd; } -void AS7343Component::calculate_irradiance(float tint_ms, float gain_x, float &irradiance_in_w_per_m2, float &lux, - AS7343Gain gain) { +void AS7343Component::calculate_irradiance(float &irradiance_in_w_per_m2, float &lux) { + // Total irradiance in a whole wavelenght interval float irr_band; + float lux_band; irradiance_in_w_per_m2 = 0; - for (uint8_t i = 0; i < NUM_USEFUL_CHANNELS - 1; i++) { - uint16_t reading = this->channel_readings_[CHANNEL_IDX[i]]; - if (reading == 0) { - continue; - } - float basic_count = reading / (gain_x * tint_ms); - // ESP_LOGD(TAG, "[%2d] Basic count %f", i, basic_count); - basic_count *= AS7343_GAIN_CORRECTION[(uint8_t) gain][i]; - // ESP_LOGD(TAG, "[%2d] gain corrected %f", i, basic_count); - irr_band = basic_count * CHANNEL_IRRAD_MW_PER_BASIC_COUNT[i] / 1000; // 1000 - if mW/m2, 100 if its uW/cm2 - // ESP_LOGD(TAG, "[%2d] irradiance %f", i, irr_band); - irr_band *= CHANNEL_ENERGY_CONTRIBUTION[i]; - lux += irr_band * CHANNEL_PHOTOPIC_LUMINOSITY[i]; - irradiance_in_w_per_m2 += irr_band ; - // ESP_LOGD(TAG, "[%2d] band irradiance %f", i, irr_band * CHANNEL_ENERGY_CONTRIBUTION[i]); + + for (uint8_t i = 0; i < AS7343_NUM_CHANNELS - 1; i++) { + irr_band = this->readings_.basic_counts[i] * CHANNEL_IRRAD_MW_PER_BASIC_COUNT[i] / 1000; + + // irr_band *= CHANNEL_ENERGY_CONTRIBUTION[i]; ? + lux_band = irr_band * CHANNEL_PHOTOPIC_LUMINOSITY[i]; + + lux += lux_band; + irradiance_in_w_per_m2 += irr_band; } // sunlight equivalent // 1 W/m2 = 116 ± 3 lx solar @@ -409,7 +408,15 @@ void AS7343Component::calculate_irradiance(float tint_ms, float gain_x, float &i lux *= 116; } -bool AS7343Component::read_18_channels(std::array &data) { +bool AS7343Component::read_all_channels() { + static constexpr uint8_t CHANNEL_MAP[AS7343_NUM_CHANNELS] = { + AS7343_CHANNEL_405_F1, AS7343_CHANNEL_425_F2, AS7343_CHANNEL_450_FZ, AS7343_CHANNEL_475_F3, + AS7343_CHANNEL_515_F4, AS7343_CHANNEL_555_FY, AS7343_CHANNEL_550_F5, AS7343_CHANNEL_600_FXL, + AS7343_CHANNEL_640_F6, AS7343_CHANNEL_690_F7, AS7343_CHANNEL_745_F8, AS7343_CHANNEL_855_NIR, + AS7343_CHANNEL_CLEAR}; + + std::array data; + this->wait_for_data(); AS7343RegStatus status{0}; @@ -426,10 +433,22 @@ bool AS7343Component::read_18_channels(std::array if (astatus.asat_status) { ESP_LOGW(TAG, "AS7343 affected by analog or digital saturation. Readings are not reliable."); } - this->readings_saturated_ = astatus.asat_status; - this->readings_gain_ = astatus.again_status; - return this->read_bytes_16((uint8_t) AS7343Registers::DATA_O, data.data(), AS7343_NUM_CHANNELS); + auto ret = this->read_bytes_16((uint8_t) AS7343Registers::DATA_O, data.data(), AS7343_NUM_CHANNELS_MAX); + + for (uint8_t i = 0; i < AS7343_NUM_CHANNELS; i++) { + this->readings_.raw_counts[i] = data[CHANNEL_MAP[i]]; + } + + this->readings_.gain = astatus.again_status; + this->readings_.gain_x = get_gain_multiplier(this->readings_.gain); + this->readings_.atime = get_atime(); + this->readings_.astep = get_astep(); + this->readings_.t_int = (1 + this->readings_.atime) * (1 + this->readings_.astep) * 2.78 / 1000; // us to ms + + this->readings_saturated_ = astatus.asat_status; + + return ret; } bool AS7343Component::wait_for_data(uint16_t timeout) { @@ -836,45 +855,46 @@ template T AS7343Component::get_highest_value(std::arrayget_maximum_spectral_adc_(); - uint16_t highest_adc = this->get_highest_value(this->channel_readings_); + // uint16_t max_adc = this->get_maximum_spectral_adc_(); + // uint16_t highest_adc = this->get_highest_value(this->channel_readings_); - current_gain = this->readings_gain_; - new_gain = current_gain; - this->get_optimized_gain_(max_adc, highest_adc, AS7343Gain::AS7343_GAIN_0_5X, AS7343Gain::AS7343_GAIN_128X, new_gain, - is_saturation); - if (new_gain != current_gain) { - if (fire_at_will) { - // need to repeat the measurement - this->set_gain((AS7343Gain) new_gain); - this->setup_gain((AS7343Gain) new_gain); - } - need_to_repeat = true; - } else if (is_saturation) { - // digital saturation - // but can't change gain? try change time ? - ESP_LOGW(TAG, "Spectral post process: OPTIMIZE saturation detected"); - } - if (!is_saturation) { - // no saturation - for (uint8_t i = 0; i < AS7343_NUM_CHANNELS; i++) { - // todo - update reading with gain factor first, then compare - if (this->channel_readings_[i] >= max_adc) { // check both values - before and after gain factor application - this->channel_readings_[i] = ADC_SATURATED_VALUE; - is_saturation = true; - } - } - if (is_saturation) { - ESP_LOGW(TAG, "Spectral post process: CHANNEL saturation detected"); - } - } + // current_gain = this->readings_gain_; + // new_gain = current_gain; + // this->get_optimized_gain_(max_adc, highest_adc, AS7343Gain::AS7343_GAIN_0_5X, AS7343Gain::AS7343_GAIN_128X, + // new_gain, + // is_saturation); + // if (new_gain != current_gain) { + // if (fire_at_will) { + // // need to repeat the measurement + // this->set_gain((AS7343Gain) new_gain); + // this->setup_gain((AS7343Gain) new_gain); + // } + // need_to_repeat = true; + // } else if (is_saturation) { + // // digital saturation + // // but can't change gain? try change time ? + // ESP_LOGW(TAG, "Spectral post process: OPTIMIZE saturation detected"); + // } + // if (!is_saturation) { + // // no saturation + // for (uint8_t i = 0; i < AS7343_NUM_CHANNELS; i++) { + // // todo - update reading with gain factor first, then compare + // if (this->channel_readings_[i] >= max_adc) { // check both values - before and after gain factor application + // this->channel_readings_[i] = ADC_SATURATED_VALUE; + // is_saturation = true; + // } + // } + // if (is_saturation) { + // ESP_LOGW(TAG, "Spectral post process: CHANNEL saturation detected"); + // } + // } /// what to do with saturation and !need_to_repeat ? - ESP_LOGW(TAG, "Spectral post process: gain %u, saturation %u, need to repeat %u", new_gain, is_saturation, - need_to_repeat); + // ESP_LOGW(TAG, "Spectral post process: gain %u, saturation %u, need to repeat %u", new_gain, is_saturation, + // need_to_repeat); return need_to_repeat; } diff --git a/esphome/components/as7343/as7343.h b/esphome/components/as7343/as7343.h index bb4a3de965..79c4967c4e 100644 --- a/esphome/components/as7343/as7343.h +++ b/esphome/components/as7343/as7343.h @@ -50,10 +50,11 @@ class AS7343Component : public PollingComponent, public i2c::I2CDevice { float get_gain_multiplier(AS7343Gain gain); - bool read_18_channels(std::array &data); - float calculate_ppfd(float tint_ms, float gain_x, AS7343Gain gain); - void calculate_irradiance(float tint_ms, float gain_x, float &irradiance, float &lux, AS7343Gain gain); - float calculate_spectre_(); + bool read_all_channels(); + + void calculate_basic_counts(); + void calculate_ppfd(float &ppfd); + void calculate_irradiance(float &irradiance, float &lux); bool wait_for_data(uint16_t timeout = 1000); bool is_data_ready(); @@ -94,11 +95,16 @@ class AS7343Component : public PollingComponent, public i2c::I2CDevice { AS7343Gain gain_; uint8_t atime_; - std::array channel_readings_; - AS7343Gain readings_gain_; - uint8_t readings_done_; + struct { + std::array raw_counts; + std::array basic_counts; + AS7343Gain gain; + uint8_t atime; + uint16_t astep; - std::array channel_basic_readings_; + float gain_x; + float t_int; + } readings_; float get_tint_(); void optimizer_(float max_TINT); diff --git a/esphome/components/as7343/as7343_calibration.cpp b/esphome/components/as7343/as7343_calibration.cpp index 55dfaa0064..d82e0c4c02 100644 --- a/esphome/components/as7343/as7343_calibration.cpp +++ b/esphome/components/as7343/as7343_calibration.cpp @@ -27,10 +27,10 @@ const float AS7343_GAIN_CORRECTION[13][12 + 1] PROGMEM = { 1.044000, 1.053000}, // 256x {1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000}, // 512x - {1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, - 1.000000, 1.000000}, // 1024x - {1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, - 1.000000, 1.000000}, // 2048x + {1.033303857, 1.034763813, 1.029855013, 1.036911249, 0.988150537, 1.029003501, 1.021363258, 0.98762089, 1.021713376, + 1.040108204, 0.987417698, 1.019481421, 0.987270057}, // 1024x + {0.986859143, 0.980877221, 0.9469769, 0.985535324, 0.926243961, 0.959814787, 1.007963896, 0.942685187, 1.007630587, + 1.025460005, 0.928696275, 1.001296639, 1.041275263}, // 2048x }; diff --git a/esphome/components/as7343/as7343_registers.h b/esphome/components/as7343/as7343_registers.h index 620465d6a0..17d02aaa85 100644 --- a/esphome/components/as7343/as7343_registers.h +++ b/esphome/components/as7343/as7343_registers.h @@ -92,7 +92,8 @@ enum AS7343Channel : uint8_t { AS7343_CHANNEL_FD, }; -static constexpr uint8_t AS7343_NUM_CHANNELS = 18; +static constexpr uint8_t AS7343_NUM_CHANNELS_MAX = 18; +static constexpr uint8_t AS7343_NUM_CHANNELS = 13; union AS7343RegCfg20 { uint8_t raw;