ESPHome 2026.2.4
Loading...
Searching...
No Matches
atm90e32.h
Go to the documentation of this file.
1#pragma once
2
3#include <span>
4#include <unordered_map>
5#include "atm90e32_reg.h"
10#include "esphome/core/gpio.h"
13
14namespace esphome {
15namespace atm90e32 {
16
18 public spi::SPIDevice<spi::BIT_ORDER_MSB_FIRST, spi::CLOCK_POLARITY_HIGH,
19 spi::CLOCK_PHASE_TRAILING, spi::DATA_RATE_1MHZ> {
20 public:
21 static const uint8_t PHASEA = 0;
22 static const uint8_t PHASEB = 1;
23 static const uint8_t PHASEC = 2;
24 const char *phase_labels[3] = {"A", "B", "C"};
25 // these registers are not sucessive, so we can't just do 'base + phase'
26 const uint16_t voltage_gain_registers[3] = {ATM90E32_REGISTER_UGAINA, ATM90E32_REGISTER_UGAINB,
27 ATM90E32_REGISTER_UGAINC};
28 const uint16_t current_gain_registers[3] = {ATM90E32_REGISTER_IGAINA, ATM90E32_REGISTER_IGAINB,
29 ATM90E32_REGISTER_IGAINC};
30 const uint16_t voltage_offset_registers[3] = {ATM90E32_REGISTER_UOFFSETA, ATM90E32_REGISTER_UOFFSETB,
31 ATM90E32_REGISTER_UOFFSETC};
32 const uint16_t current_offset_registers[3] = {ATM90E32_REGISTER_IOFFSETA, ATM90E32_REGISTER_IOFFSETB,
33 ATM90E32_REGISTER_IOFFSETC};
34 const uint16_t power_offset_registers[3] = {ATM90E32_REGISTER_POFFSETA, ATM90E32_REGISTER_POFFSETB,
35 ATM90E32_REGISTER_POFFSETC};
36 const uint16_t reactive_power_offset_registers[3] = {ATM90E32_REGISTER_QOFFSETA, ATM90E32_REGISTER_QOFFSETB,
37 ATM90E32_REGISTER_QOFFSETC};
38 const uint16_t over_voltage_flags[3] = {ATM90E32_STATUS_S0_OVPHASEAST, ATM90E32_STATUS_S0_OVPHASEBST,
39 ATM90E32_STATUS_S0_OVPHASECST};
40 const uint16_t voltage_sag_flags[3] = {ATM90E32_STATUS_S1_SAGPHASEAST, ATM90E32_STATUS_S1_SAGPHASEBST,
41 ATM90E32_STATUS_S1_SAGPHASECST};
42 const uint16_t phase_loss_flags[3] = {ATM90E32_STATUS_S1_PHASELOSSAST, ATM90E32_STATUS_S1_PHASELOSSBST,
43 ATM90E32_STATUS_S1_PHASELOSSCST};
44 void loop() override;
45 void setup() override;
46 void dump_config() override;
47 float get_setup_priority() const override;
48 void update() override;
49 void set_voltage_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].voltage_sensor_ = obj; }
50 void set_current_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].current_sensor_ = obj; }
51 void set_power_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].power_sensor_ = obj; }
52 void set_reactive_power_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].reactive_power_sensor_ = obj; }
53 void set_apparent_power_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].apparent_power_sensor_ = obj; }
55 this->phase_[phase].forward_active_energy_sensor_ = obj;
56 }
58 this->phase_[phase].reverse_active_energy_sensor_ = obj;
59 }
60 void set_power_factor_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].power_factor_sensor_ = obj; }
61 void set_phase_angle_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].phase_angle_sensor_ = obj; }
63 this->phase_[phase].harmonic_active_power_sensor_ = obj;
64 }
65 void set_peak_current_sensor(int phase, sensor::Sensor *obj) { this->phase_[phase].peak_current_sensor_ = obj; }
66 void set_volt_gain(int phase, uint16_t gain) {
67 this->phase_[phase].voltage_gain_ = gain;
68 this->has_config_voltage_gain_[phase] = true;
69 }
70 void set_ct_gain(int phase, uint16_t gain) {
71 this->phase_[phase].ct_gain_ = gain;
72 this->has_config_current_gain_[phase] = true;
73 }
74 void set_voltage_offset(uint8_t phase, int16_t offset) {
75 this->offset_phase_[phase].voltage_offset_ = offset;
76 this->has_config_voltage_offset_[phase] = true;
77 }
78 void set_current_offset(uint8_t phase, int16_t offset) {
79 this->offset_phase_[phase].current_offset_ = offset;
80 this->has_config_current_offset_[phase] = true;
81 }
82 void set_active_power_offset(uint8_t phase, int16_t offset) {
83 this->power_offset_phase_[phase].active_power_offset = offset;
84 this->has_config_active_power_offset_[phase] = true;
85 }
86 void set_reactive_power_offset(uint8_t phase, int16_t offset) {
87 this->power_offset_phase_[phase].reactive_power_offset = offset;
88 this->has_config_reactive_power_offset_[phase] = true;
89 }
90 void set_freq_sensor(sensor::Sensor *freq_sensor) { freq_sensor_ = freq_sensor; }
92 void set_chip_temperature_sensor(sensor::Sensor *chip_temperature_sensor) {
93 chip_temperature_sensor_ = chip_temperature_sensor;
94 }
95 void set_line_freq(int freq) { line_freq_ = freq; }
96 void set_current_phases(int phases) { current_phases_ = phases; }
97 void set_pga_gain(uint16_t gain) { pga_gain_ = gain; }
105 int16_t calibrate_offset(uint8_t phase, bool voltage);
106 int16_t calibrate_power_offset(uint8_t phase, bool reactive);
108#ifdef USE_NUMBER
109 void set_reference_voltage(uint8_t phase, number::Number *ref_voltage) { ref_voltages_[phase] = ref_voltage; }
110 void set_reference_current(uint8_t phase, number::Number *ref_current) { ref_currents_[phase] = ref_current; }
111#endif
112 float get_reference_voltage(uint8_t phase) {
113#ifdef USE_NUMBER
114 return (phase >= 0 && phase < 3 && ref_voltages_[phase]) ? ref_voltages_[phase]->state : 120.0; // Default voltage
115#else
116 return 120.0; // Default voltage
117#endif
118 }
119 float get_reference_current(uint8_t phase) {
120#ifdef USE_NUMBER
121 return (phase >= 0 && phase < 3 && ref_currents_[phase]) ? ref_currents_[phase]->state : 5.0f; // Default current
122#else
123 return 5.0f; // Default current
124#endif
125 }
126 bool using_saved_calibrations_ = false; // Track if stored calibrations are being used
127#ifdef USE_TEXT_SENSOR
128 void check_phase_status();
129 void check_freq_status();
130 void check_over_current();
132 this->phase_status_text_sensor_[phase] = sensor;
133 }
135#endif
136 uint16_t calculate_voltage_threshold(int line_freq, uint16_t ugain, float multiplier);
138
139 protected:
140#ifdef USE_NUMBER
141 number::Number *ref_voltages_[3]{nullptr, nullptr, nullptr};
142 number::Number *ref_currents_[3]{nullptr, nullptr, nullptr};
143#endif
144 uint16_t read16_(uint16_t a_register);
145 int read32_(uint16_t addr_h, uint16_t addr_l);
146 void write16_(uint16_t a_register, uint16_t val, bool validate = true);
147 float get_local_phase_voltage_(uint8_t phase);
148 float get_local_phase_current_(uint8_t phase);
149 float get_local_phase_active_power_(uint8_t phase);
150 float get_local_phase_reactive_power_(uint8_t phase);
151 float get_local_phase_apparent_power_(uint8_t phase);
152 float get_local_phase_power_factor_(uint8_t phase);
153 float get_local_phase_forward_active_energy_(uint8_t phase);
154 float get_local_phase_reverse_active_energy_(uint8_t phase);
155 float get_local_phase_angle_(uint8_t phase);
156 float get_local_phase_harmonic_active_power_(uint8_t phase);
157 float get_local_phase_peak_current_(uint8_t phase);
158 float get_phase_voltage_(uint8_t phase);
159 float get_phase_voltage_avg_(uint8_t phase);
160 float get_phase_current_(uint8_t phase);
161 float get_phase_current_avg_(uint8_t phase);
162 float get_phase_active_power_(uint8_t phase);
163 float get_phase_reactive_power_(uint8_t phase);
164 float get_phase_apparent_power_(uint8_t phase);
165 float get_phase_power_factor_(uint8_t phase);
166 float get_phase_forward_active_energy_(uint8_t phase);
167 float get_phase_reverse_active_energy_(uint8_t phase);
168 float get_phase_angle_(uint8_t phase);
169 float get_phase_harmonic_active_power_(uint8_t phase);
170 float get_phase_peak_current_(uint8_t phase);
171 float get_frequency_();
172 float get_chip_temperature_();
181 void write_offsets_to_registers_(uint8_t phase, int16_t voltage_offset, int16_t current_offset);
182 void write_power_offsets_to_registers_(uint8_t phase, int16_t p_offset, int16_t q_offset);
184 bool verify_gain_writes_();
185 bool validate_spi_read_(uint16_t expected, const char *context = nullptr);
187 void get_cs_summary_(std::span<char, GPIO_SUMMARY_MAX_LEN> buffer);
188
221
226
228
233
235
237 uint16_t voltage_gain{1};
238 uint16_t current_gain{1};
240
242
243 bool has_config_voltage_offset_[3]{false, false, false};
244 bool has_config_current_offset_[3]{false, false, false};
245 bool has_config_active_power_offset_[3]{false, false, false};
246 bool has_config_reactive_power_offset_[3]{false, false, false};
247 bool has_config_voltage_gain_[3]{false, false, false};
248 bool has_config_current_gain_[3]{false, false, false};
249
253
255#ifdef USE_TEXT_SENSOR
258#endif
260 uint16_t pga_gain_{0x15};
261 int line_freq_{60};
271 bool offset_calibration_mismatch_[3]{false, false, false};
272 bool power_offset_calibration_mismatch_[3]{false, false, false};
273 bool gain_calibration_mismatch_[3]{false, false, false};
274};
275
276} // namespace atm90e32
277} // namespace esphome
This class simplifies creating components that periodically check a state.
Definition component.h:512
void set_chip_temperature_sensor(sensor::Sensor *chip_temperature_sensor)
Definition atm90e32.h:92
void set_freq_sensor(sensor::Sensor *freq_sensor)
Definition atm90e32.h:90
float get_local_phase_reactive_power_(uint8_t phase)
Definition atm90e32.cpp:442
float get_phase_forward_active_energy_(uint8_t phase)
Definition atm90e32.cpp:525
void set_freq_status_text_sensor(text_sensor::TextSensor *sensor)
Definition atm90e32.h:134
float get_phase_current_avg_(uint8_t phase)
Definition atm90e32.cpp:484
float get_local_phase_apparent_power_(uint8_t phase)
Definition atm90e32.cpp:444
void write16_(uint16_t a_register, uint16_t val, bool validate=true)
Definition atm90e32.cpp:421
void set_apparent_power_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:53
text_sensor::TextSensor * freq_status_text_sensor_
Definition atm90e32.h:257
ESPPreferenceObject power_offset_pref_
Definition atm90e32.h:251
const uint16_t voltage_gain_registers[3]
Definition atm90e32.h:26
float get_phase_voltage_avg_(uint8_t phase)
Definition atm90e32.cpp:470
void write_power_offsets_to_registers_(uint8_t phase, int16_t p_offset, int16_t q_offset)
Definition atm90e32.cpp:813
const uint16_t current_gain_registers[3]
Definition atm90e32.h:28
float get_reference_voltage(uint8_t phase)
Definition atm90e32.h:112
number::Number * ref_voltages_[3]
Definition atm90e32.h:141
void set_power_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:51
struct esphome::atm90e32::ATM90E32Component::GainCalibration gain_phase_[3]
void set_current_offset(uint8_t phase, int16_t offset)
Definition atm90e32.h:78
const uint16_t current_offset_registers[3]
Definition atm90e32.h:32
static const uint8_t PHASEB
Definition atm90e32.h:22
float get_phase_reverse_active_energy_(uint8_t phase)
Definition atm90e32.cpp:536
void set_reactive_power_offset(uint8_t phase, int16_t offset)
Definition atm90e32.h:86
float get_local_phase_harmonic_active_power_(uint8_t phase)
Definition atm90e32.cpp:458
float get_phase_angle_(uint8_t phase)
Definition atm90e32.cpp:552
float get_local_phase_current_(uint8_t phase)
Definition atm90e32.cpp:438
bool validate_spi_read_(uint16_t expected, const char *context=nullptr)
void set_peak_current_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:65
const uint16_t reactive_power_offset_registers[3]
Definition atm90e32.h:36
const uint16_t over_voltage_flags[3]
Definition atm90e32.h:38
void set_reverse_active_energy_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:57
float get_phase_voltage_(uint8_t phase)
Definition atm90e32.cpp:464
void set_current_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:50
GainCalibration config_gain_phase_[3]
Definition atm90e32.h:241
int16_t calibrate_offset(uint8_t phase, bool voltage)
float get_local_phase_reverse_active_energy_(uint8_t phase)
Definition atm90e32.cpp:452
void set_harmonic_active_power_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:62
void set_current_phases(int phases)
Definition atm90e32.h:96
float get_local_phase_forward_active_energy_(uint8_t phase)
Definition atm90e32.cpp:448
OffsetCalibration config_offset_phase_[3]
Definition atm90e32.h:227
struct esphome::atm90e32::ATM90E32Component::OffsetCalibration offset_phase_[3]
void set_active_power_offset(uint8_t phase, int16_t offset)
Definition atm90e32.h:82
uint16_t calculate_voltage_threshold(int line_freq, uint16_t ugain, float multiplier)
float get_local_phase_power_factor_(uint8_t phase)
Definition atm90e32.cpp:446
float get_phase_reactive_power_(uint8_t phase)
Definition atm90e32.cpp:509
void set_enable_gain_calibration(bool flag)
Definition atm90e32.h:104
const uint16_t phase_loss_flags[3]
Definition atm90e32.h:42
float get_phase_apparent_power_(uint8_t phase)
Definition atm90e32.cpp:514
float get_local_phase_voltage_(uint8_t phase)
Definition atm90e32.cpp:436
void set_phase_status_text_sensor(uint8_t phase, text_sensor::TextSensor *sensor)
Definition atm90e32.h:131
void set_pga_gain(uint16_t gain)
Definition atm90e32.h:97
void set_reference_current(uint8_t phase, number::Number *ref_current)
Definition atm90e32.h:110
ESPPreferenceObject gain_calibration_pref_
Definition atm90e32.h:252
void write_offsets_to_registers_(uint8_t phase, int16_t voltage_offset, int16_t current_offset)
Definition atm90e32.cpp:797
static const uint8_t PHASEA
Definition atm90e32.h:21
struct esphome::atm90e32::ATM90E32Component::PowerOffsetCalibration power_offset_phase_[3]
float get_reference_current(uint8_t phase)
Definition atm90e32.h:119
number::Number * ref_currents_[3]
Definition atm90e32.h:142
float get_phase_peak_current_(uint8_t phase)
Definition atm90e32.cpp:557
void set_voltage_offset(uint8_t phase, int16_t offset)
Definition atm90e32.h:74
float get_phase_harmonic_active_power_(uint8_t phase)
Definition atm90e32.cpp:547
const uint16_t voltage_sag_flags[3]
Definition atm90e32.h:40
float get_phase_active_power_(uint8_t phase)
Definition atm90e32.cpp:504
PowerOffsetCalibration config_power_offset_phase_[3]
Definition atm90e32.h:234
void get_cs_summary_(std::span< char, GPIO_SUMMARY_MAX_LEN > buffer)
Definition atm90e32.cpp:111
static const uint8_t PHASEC
Definition atm90e32.h:23
const uint16_t power_offset_registers[3]
Definition atm90e32.h:34
float get_setup_priority() const override
Definition atm90e32.cpp:388
uint16_t read16_(uint16_t a_register)
Definition atm90e32.cpp:393
int read32_(uint16_t addr_h, uint16_t addr_l)
Definition atm90e32.cpp:408
sensor::Sensor * chip_temperature_sensor_
Definition atm90e32.h:259
float get_phase_power_factor_(uint8_t phase)
Definition atm90e32.cpp:519
void set_voltage_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:49
float get_local_phase_angle_(uint8_t phase)
Definition atm90e32.cpp:456
const uint16_t voltage_offset_registers[3]
Definition atm90e32.h:30
void set_ct_gain(int phase, uint16_t gain)
Definition atm90e32.h:70
void set_phase_angle_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:61
void set_reference_voltage(uint8_t phase, number::Number *ref_voltage)
Definition atm90e32.h:109
void set_power_factor_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:60
void set_forward_active_energy_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:54
void set_volt_gain(int phase, uint16_t gain)
Definition atm90e32.h:66
text_sensor::TextSensor * phase_status_text_sensor_[3]
Definition atm90e32.h:256
float get_phase_current_(uint8_t phase)
Definition atm90e32.cpp:498
void set_peak_current_signed(bool flag)
Definition atm90e32.h:91
float get_local_phase_peak_current_(uint8_t phase)
Definition atm90e32.cpp:462
void set_publish_interval_flag_(bool flag)
Definition atm90e32.h:174
int16_t calibrate_power_offset(uint8_t phase, bool reactive)
struct esphome::atm90e32::ATM90E32Component::ATM90E32Phase phase_[3]
void set_enable_offset_calibration(bool flag)
Definition atm90e32.h:103
ESPPreferenceObject offset_pref_
Definition atm90e32.h:250
void set_reactive_power_sensor(int phase, sensor::Sensor *obj)
Definition atm90e32.h:52
float get_local_phase_active_power_(uint8_t phase)
Definition atm90e32.cpp:440
Base-class for all numbers.
Definition number.h:29
Base-class for all sensors.
Definition sensor.h:43
The SPIDevice is what components using the SPI will create.
Definition spi.h:427
AlsGain501 gain
mopeka_std_values val[4]
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
uint32_t IRAM_ATTR HOT millis()
Definition core.cpp:25