mirror of
https://github.com/esphome/esphome.git
synced 2025-01-23 21:04:29 +01:00
some cleanup
This commit is contained in:
parent
66bb6bdfda
commit
953d4e5f50
4 changed files with 236 additions and 209 deletions
|
@ -9,34 +9,67 @@ namespace as7343 {
|
||||||
|
|
||||||
static const char *const TAG = "as7343";
|
static const char *const TAG = "as7343";
|
||||||
|
|
||||||
static constexpr float CONST_H = 6.6260695e-34f;
|
// pimoroni?
|
||||||
static constexpr float CONST_C = 299792458;
|
static constexpr float CHANNEL_COMPENSATION_GAIN[AS7343_NUM_CHANNELS] = {1.84, 6.03, 4.88, 13.74, 3.37, 2.82, 6.72,
|
||||||
|
|
||||||
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,
|
|
||||||
2.22, 3.17, 1.95, 12.25, 1.00, 1};
|
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.59633, 0.56242, 0.65645, 0.68882, 0.79980,
|
||||||
0.70423, 0.40366, 0.38516};
|
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<const char *, AS7343_NUM_CHANNELS> 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<float, AS7343_NUM_CHANNELS> 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<float, AS7343_NUM_CHANNELS> 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,
|
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};
|
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,
|
// 1.5 Irradiation in mW/m² per basic count ?
|
||||||
0.000281, 0.000281, 0.000422, 0.000281, 0.000422, 0.000281};
|
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,
|
// 1.6 band energy contribution precalculated based on band widths and central wavelengths (photon energy)
|
||||||
600, 640, 690, 745, 855, 718};
|
// value = band energy / sum of all bands energies, except for VIS
|
||||||
static constexpr float CHANNEL_NM_WIDTH[NUM_USEFUL_CHANNELS] = {30, 22, 55, 30, 40, 100, 35, 80, 50, 55, 60, 54, 0};
|
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.0073000000000,
|
||||||
0.0380000000000,
|
0.0380000000000,
|
||||||
0.1126000000000,
|
0.1126000000000,
|
||||||
|
@ -50,31 +83,8 @@ static constexpr float CHANNEL_PHOTOPIC_LUMINOSITY[NUM_USEFUL_CHANNELS] = {0.000
|
||||||
0,
|
0,
|
||||||
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] = {
|
// 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};
|
// 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() {
|
void AS7343Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AS7343...");
|
ESP_LOGCONFIG(TAG, "Setting up AS7343...");
|
||||||
|
@ -143,58 +153,34 @@ void AS7343Component::dump_config() {
|
||||||
|
|
||||||
float AS7343Component::get_setup_priority() const { return setup_priority::DATA; }
|
float AS7343Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
|
void log13_f(const char *TAG, const char *str, const std::array<float, 13> &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<uint16_t, 13> &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<const char *, 13> &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() {
|
void AS7343Component::update() {
|
||||||
// this->optimizer_(1000);
|
|
||||||
// delay(20);
|
|
||||||
|
|
||||||
this->enable_spectral_measurement(true);
|
this->enable_spectral_measurement(true);
|
||||||
this->read_18_channels(this->channel_readings_);
|
this->read_all_channels();
|
||||||
this->enable_spectral_measurement(false);
|
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;
|
log13_s(TAG, "Channel", CHANNEL_NAMES);
|
||||||
this->channel_readings_[AS7343_CHANNEL_FD_0] = 0;
|
log13_f(TAG, "Nm", CHANNEL_NM);
|
||||||
this->channel_readings_[AS7343_CHANNEL_FD_1] = 0;
|
log13_d(TAG, "Counts", this->readings_.raw_counts);
|
||||||
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_();
|
|
||||||
|
|
||||||
uint16_t max_adc = this->get_maximum_spectral_adc_();
|
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) {
|
if (highest_adc >= max_adc) {
|
||||||
ESP_LOGW(TAG, "Max ADC: %u, Highest ADC: %u", max_adc, highest_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, "Max ADC: %u, Highest ADC: %u", max_adc, highest_adc);
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGD(TAG, " ,Gain , %.1f,X", gain_x);
|
ESP_LOGD(TAG, " ,Gain , %.1f,X", this->readings_.gain_x);
|
||||||
ESP_LOGD(TAG, " ,ATIME, %u,", atime);
|
ESP_LOGD(TAG, " ,ATIME, %u,", this->readings_.atime);
|
||||||
ESP_LOGD(TAG, " ,ASTEP, %u,", astep);
|
ESP_LOGD(TAG, " ,ASTEP, %u,", this->readings_.astep);
|
||||||
ESP_LOGD(TAG, " ,TINT , %.2f,", tint_ms);
|
ESP_LOGD(TAG, " ,TINT , %.2f,", this->readings_.t_int);
|
||||||
ESP_LOGD(TAG, " ,TINT2 , %.2f,", tint2_ms);
|
|
||||||
|
|
||||||
ESP_LOGD(TAG, ",nm, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, %.0f, ", CHANNEL_NM[0],
|
// 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[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]);
|
// 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, ",
|
// 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[0]], this->channel_readings_[CHANNEL_IDX[1]],
|
||||||
this->channel_readings_[CHANNEL_IDX[2]], this->channel_readings_[CHANNEL_IDX[3]],
|
// 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[4]], this->channel_readings_[CHANNEL_IDX[5]],
|
||||||
this->channel_readings_[CHANNEL_IDX[6]], this->channel_readings_[CHANNEL_IDX[7]],
|
// 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[8]], this->channel_readings_[CHANNEL_IDX[9]],
|
||||||
this->channel_readings_[CHANNEL_IDX[10]], this->channel_readings_[CHANNEL_IDX[11]],
|
// this->channel_readings_[CHANNEL_IDX[10]], this->channel_readings_[CHANNEL_IDX[11]],
|
||||||
this->channel_readings_[CHANNEL_IDX[12]]);
|
// this->channel_readings_[CHANNEL_IDX[12]]);
|
||||||
|
|
||||||
float irradiance;
|
float irradiance;
|
||||||
float lux;
|
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, " ,Irradiance, %f, W/m²", irradiance);
|
||||||
ESP_LOGD(TAG, " ,Lux solar , %f, lx", lux);
|
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);
|
ESP_LOGD(TAG, " ,PPFD , %.2f, µmol/s⋅m²", ppfd);
|
||||||
|
|
||||||
if (this->illuminance_ != nullptr) {
|
if (this->illuminance_ != nullptr) {
|
||||||
|
@ -239,17 +227,18 @@ void AS7343Component::update() {
|
||||||
if (this->ppfd_ != nullptr) {
|
if (this->ppfd_ != nullptr) {
|
||||||
this->ppfd_->publish_state(ppfd);
|
this->ppfd_->publish_state(ppfd);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t i = 0;
|
uint8_t i = 0;
|
||||||
float max_val = 0;
|
float max_val = 0;
|
||||||
float normalized_readings[NUM_USEFUL_CHANNELS];
|
float normalized_readings[AS7343_NUM_CHANNELS];
|
||||||
for (i = 0; i < NUM_USEFUL_CHANNELS; i++) {
|
for (i = 0; i < AS7343_NUM_CHANNELS; i++) {
|
||||||
normalized_readings[i] = (float) this->channel_readings_[CHANNEL_IDX[i]] / CHANNEL_SENS[i];
|
normalized_readings[i] = (float) this->readings_.basic_counts[i] / CHANNEL_SENS[i];
|
||||||
if (max_val < normalized_readings[i]) {
|
if (max_val < normalized_readings[i]) {
|
||||||
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;
|
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);
|
return this->write_byte((uint8_t) AS7343Registers::CFG1, gain);
|
||||||
}
|
}
|
||||||
|
|
||||||
float AS7343Component::calculate_ppfd(float tint_ms, float gain_x, AS7343Gain gain) {
|
void AS7343Component::calculate_basic_counts() {
|
||||||
float pfd = 0;
|
float divisor = this->readings_.gain_x * this->readings_.t_int;
|
||||||
float bc[NUM_USEFUL_CHANNELS] = {0};
|
for (size_t i = 0; i < AS7343_NUM_CHANNELS; i++) {
|
||||||
float bcc[NUM_USEFUL_CHANNELS] = {0};
|
// 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++) {
|
this->readings_.basic_counts[i] = basic_count;
|
||||||
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];
|
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) {
|
if (CHANNEL_NM[i] < 400 || CHANNEL_NM[i] > 700) {
|
||||||
continue;
|
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;
|
// assume channels cover whole range
|
||||||
// probably is it not mW/m2 but uW/cm2!!!! so try divide by 10k not 1000
|
ppfd += photon_flux * CHANNEL_NM_WIDTH[i] / 1e9f; // nm to meters
|
||||||
// 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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
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,
|
void AS7343Component::calculate_irradiance(float &irradiance_in_w_per_m2, float &lux) {
|
||||||
AS7343Gain gain) {
|
// Total irradiance in a whole wavelenght interval
|
||||||
float irr_band;
|
float irr_band;
|
||||||
|
float lux_band;
|
||||||
irradiance_in_w_per_m2 = 0;
|
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]];
|
for (uint8_t i = 0; i < AS7343_NUM_CHANNELS - 1; i++) {
|
||||||
if (reading == 0) {
|
irr_band = this->readings_.basic_counts[i] * CHANNEL_IRRAD_MW_PER_BASIC_COUNT[i] / 1000;
|
||||||
continue;
|
|
||||||
}
|
// irr_band *= CHANNEL_ENERGY_CONTRIBUTION[i]; ?
|
||||||
float basic_count = reading / (gain_x * tint_ms);
|
lux_band = irr_band * CHANNEL_PHOTOPIC_LUMINOSITY[i];
|
||||||
// ESP_LOGD(TAG, "[%2d] Basic count %f", i, basic_count);
|
|
||||||
basic_count *= AS7343_GAIN_CORRECTION[(uint8_t) gain][i];
|
lux += lux_band;
|
||||||
// ESP_LOGD(TAG, "[%2d] gain corrected %f", i, basic_count);
|
irradiance_in_w_per_m2 += irr_band;
|
||||||
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]);
|
|
||||||
}
|
}
|
||||||
// sunlight equivalent
|
// sunlight equivalent
|
||||||
// 1 W/m2 = 116 ± 3 lx solar
|
// 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;
|
lux *= 116;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AS7343Component::read_18_channels(std::array<uint16_t, AS7343_NUM_CHANNELS> &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<uint16_t, AS7343_NUM_CHANNELS_MAX> data;
|
||||||
|
|
||||||
this->wait_for_data();
|
this->wait_for_data();
|
||||||
|
|
||||||
AS7343RegStatus status{0};
|
AS7343RegStatus status{0};
|
||||||
|
@ -426,10 +433,22 @@ bool AS7343Component::read_18_channels(std::array<uint16_t, AS7343_NUM_CHANNELS>
|
||||||
if (astatus.asat_status) {
|
if (astatus.asat_status) {
|
||||||
ESP_LOGW(TAG, "AS7343 affected by analog or digital saturation. Readings are not reliable.");
|
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) {
|
bool AS7343Component::wait_for_data(uint16_t timeout) {
|
||||||
|
@ -836,45 +855,46 @@ template<typename T, size_t N> T AS7343Component::get_highest_value(std::array<T
|
||||||
|
|
||||||
bool AS7343Component::spectral_post_process_(bool fire_at_will) {
|
bool AS7343Component::spectral_post_process_(bool fire_at_will) {
|
||||||
bool need_to_repeat = false;
|
bool need_to_repeat = false;
|
||||||
uint16_t highest_value, maximum_adc;
|
// uint16_t highest_value, maximum_adc;
|
||||||
bool is_saturation{false};
|
// bool is_saturation{false};
|
||||||
uint8_t current_gain, new_gain;
|
// uint8_t current_gain, new_gain;
|
||||||
|
|
||||||
uint16_t max_adc = this->get_maximum_spectral_adc_();
|
// 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->channel_readings_);
|
||||||
|
|
||||||
current_gain = this->readings_gain_;
|
// current_gain = this->readings_gain_;
|
||||||
new_gain = current_gain;
|
// new_gain = current_gain;
|
||||||
this->get_optimized_gain_(max_adc, highest_adc, AS7343Gain::AS7343_GAIN_0_5X, AS7343Gain::AS7343_GAIN_128X, new_gain,
|
// this->get_optimized_gain_(max_adc, highest_adc, AS7343Gain::AS7343_GAIN_0_5X, AS7343Gain::AS7343_GAIN_128X,
|
||||||
is_saturation);
|
// new_gain,
|
||||||
if (new_gain != current_gain) {
|
// is_saturation);
|
||||||
if (fire_at_will) {
|
// if (new_gain != current_gain) {
|
||||||
// need to repeat the measurement
|
// if (fire_at_will) {
|
||||||
this->set_gain((AS7343Gain) new_gain);
|
// // need to repeat the measurement
|
||||||
this->setup_gain((AS7343Gain) new_gain);
|
// this->set_gain((AS7343Gain) new_gain);
|
||||||
}
|
// this->setup_gain((AS7343Gain) new_gain);
|
||||||
need_to_repeat = true;
|
// }
|
||||||
} else if (is_saturation) {
|
// need_to_repeat = true;
|
||||||
// digital saturation
|
// } else if (is_saturation) {
|
||||||
// but can't change gain? try change time ?
|
// // digital saturation
|
||||||
ESP_LOGW(TAG, "Spectral post process: OPTIMIZE saturation detected");
|
// // but can't change gain? try change time ?
|
||||||
}
|
// ESP_LOGW(TAG, "Spectral post process: OPTIMIZE saturation detected");
|
||||||
if (!is_saturation) {
|
// }
|
||||||
// no saturation
|
// if (!is_saturation) {
|
||||||
for (uint8_t i = 0; i < AS7343_NUM_CHANNELS; i++) {
|
// // no saturation
|
||||||
// todo - update reading with gain factor first, then compare
|
// for (uint8_t i = 0; i < AS7343_NUM_CHANNELS; i++) {
|
||||||
if (this->channel_readings_[i] >= max_adc) { // check both values - before and after gain factor application
|
// // todo - update reading with gain factor first, then compare
|
||||||
this->channel_readings_[i] = ADC_SATURATED_VALUE;
|
// if (this->channel_readings_[i] >= max_adc) { // check both values - before and after gain factor application
|
||||||
is_saturation = true;
|
// this->channel_readings_[i] = ADC_SATURATED_VALUE;
|
||||||
}
|
// is_saturation = true;
|
||||||
}
|
// }
|
||||||
if (is_saturation) {
|
// }
|
||||||
ESP_LOGW(TAG, "Spectral post process: CHANNEL saturation detected");
|
// if (is_saturation) {
|
||||||
}
|
// ESP_LOGW(TAG, "Spectral post process: CHANNEL saturation detected");
|
||||||
}
|
// }
|
||||||
|
// }
|
||||||
/// what to do with saturation and !need_to_repeat ?
|
/// 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,
|
// ESP_LOGW(TAG, "Spectral post process: gain %u, saturation %u, need to repeat %u", new_gain, is_saturation,
|
||||||
need_to_repeat);
|
// need_to_repeat);
|
||||||
return need_to_repeat;
|
return need_to_repeat;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -50,10 +50,11 @@ class AS7343Component : public PollingComponent, public i2c::I2CDevice {
|
||||||
|
|
||||||
float get_gain_multiplier(AS7343Gain gain);
|
float get_gain_multiplier(AS7343Gain gain);
|
||||||
|
|
||||||
bool read_18_channels(std::array<uint16_t, AS7343_NUM_CHANNELS> &data);
|
bool read_all_channels();
|
||||||
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);
|
void calculate_basic_counts();
|
||||||
float calculate_spectre_();
|
void calculate_ppfd(float &ppfd);
|
||||||
|
void calculate_irradiance(float &irradiance, float &lux);
|
||||||
|
|
||||||
bool wait_for_data(uint16_t timeout = 1000);
|
bool wait_for_data(uint16_t timeout = 1000);
|
||||||
bool is_data_ready();
|
bool is_data_ready();
|
||||||
|
@ -94,11 +95,16 @@ class AS7343Component : public PollingComponent, public i2c::I2CDevice {
|
||||||
AS7343Gain gain_;
|
AS7343Gain gain_;
|
||||||
uint8_t atime_;
|
uint8_t atime_;
|
||||||
|
|
||||||
std::array<uint16_t, AS7343_NUM_CHANNELS> channel_readings_;
|
struct {
|
||||||
AS7343Gain readings_gain_;
|
std::array<uint16_t, AS7343_NUM_CHANNELS> raw_counts;
|
||||||
uint8_t readings_done_;
|
std::array<float, AS7343_NUM_CHANNELS> basic_counts;
|
||||||
|
AS7343Gain gain;
|
||||||
|
uint8_t atime;
|
||||||
|
uint16_t astep;
|
||||||
|
|
||||||
std::array<float, AS7343_NUM_CHANNELS> channel_basic_readings_;
|
float gain_x;
|
||||||
|
float t_int;
|
||||||
|
} readings_;
|
||||||
|
|
||||||
float get_tint_();
|
float get_tint_();
|
||||||
void optimizer_(float max_TINT);
|
void optimizer_(float max_TINT);
|
||||||
|
|
|
@ -27,10 +27,10 @@ const float AS7343_GAIN_CORRECTION[13][12 + 1] PROGMEM = {
|
||||||
1.044000, 1.053000}, // 256x
|
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, 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}, // 512x
|
||||||
{1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000,
|
{1.033303857, 1.034763813, 1.029855013, 1.036911249, 0.988150537, 1.029003501, 1.021363258, 0.98762089, 1.021713376,
|
||||||
1.000000, 1.000000}, // 1024x
|
1.040108204, 0.987417698, 1.019481421, 0.987270057}, // 1024x
|
||||||
{1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000,
|
{0.986859143, 0.980877221, 0.9469769, 0.985535324, 0.926243961, 0.959814787, 1.007963896, 0.942685187, 1.007630587,
|
||||||
1.000000, 1.000000}, // 2048x
|
1.025460005, 0.928696275, 1.001296639, 1.041275263}, // 2048x
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -92,7 +92,8 @@ enum AS7343Channel : uint8_t {
|
||||||
AS7343_CHANNEL_FD,
|
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 {
|
union AS7343RegCfg20 {
|
||||||
uint8_t raw;
|
uint8_t raw;
|
||||||
|
|
Loading…
Add table
Reference in a new issue