ESPHome 2025.5.0
Loading...
Searching...
No Matches
as3935.cpp
Go to the documentation of this file.
1#include "as3935.h"
2#include "esphome/core/log.h"
3
4namespace esphome {
5namespace as3935 {
6
7static const char *const TAG = "as3935";
8
10 ESP_LOGCONFIG(TAG, "Setting up AS3935...");
11
12 this->irq_pin_->setup();
13 LOG_PIN(" IRQ Pin: ", this->irq_pin_);
14
15 // Write properties to sensor
16 this->write_indoor(this->indoor_);
17 this->write_noise_level(this->noise_level_);
18 this->write_watchdog_threshold(this->watchdog_threshold_);
19 this->write_spike_rejection(this->spike_rejection_);
20 this->write_lightning_threshold(this->lightning_threshold_);
21 this->write_mask_disturber(this->mask_disturber_);
22 this->write_div_ratio(this->div_ratio_);
23 this->write_capacitance(this->capacitance_);
24
25 // Handle setting up tuning or auto-calibration
26 if (this->tune_antenna_) {
27 ESP_LOGCONFIG(TAG, " Antenna tuning: ENABLED - lightning detection will not function in this mode");
28 this->tune_antenna();
29 } else if (this->calibration_) {
30 this->calibrate_oscillator();
31 }
32}
33
34void AS3935Component::dump_config() {
35 ESP_LOGCONFIG(TAG, "AS3935:");
36 LOG_PIN(" Interrupt Pin: ", this->irq_pin_);
37#ifdef USE_BINARY_SENSOR
38 LOG_BINARY_SENSOR(" ", "Thunder alert", this->thunder_alert_binary_sensor_);
39#endif
40#ifdef USE_SENSOR
41 LOG_SENSOR(" ", "Distance", this->distance_sensor_);
42 LOG_SENSOR(" ", "Lightning energy", this->energy_sensor_);
43#endif
44}
45
46float AS3935Component::get_setup_priority() const { return setup_priority::DATA; }
47
48void AS3935Component::loop() {
49 if (!this->irq_pin_->digital_read())
50 return;
51
52 uint8_t int_value = this->read_interrupt_register_();
53 if (int_value == NOISE_INT) {
54 ESP_LOGI(TAG, "Noise was detected - try increasing the noise level value!");
55 } else if (int_value == DISTURBER_INT) {
56 ESP_LOGI(TAG, "Disturber was detected - try increasing the spike rejection value!");
57 } else if (int_value == LIGHTNING_INT) {
58 ESP_LOGI(TAG, "Lightning has been detected!");
59#ifdef USE_BINARY_SENSOR
60 if (this->thunder_alert_binary_sensor_ != nullptr) {
61 this->thunder_alert_binary_sensor_->publish_state(true);
62 this->set_timeout(10, [this]() { this->thunder_alert_binary_sensor_->publish_state(false); });
63 }
64#endif
65#ifdef USE_SENSOR
66 uint8_t distance = this->get_distance_to_storm_();
67 if (this->distance_sensor_ != nullptr)
68 this->distance_sensor_->publish_state(distance);
69
70 uint32_t energy = this->get_lightning_energy_();
71 if (this->energy_sensor_ != nullptr)
72 this->energy_sensor_->publish_state(energy);
73#endif
74 }
75}
76
77void AS3935Component::write_indoor(bool indoor) {
78 ESP_LOGV(TAG, "Setting indoor to %d", indoor);
79 if (indoor) {
81 } else {
83 }
84}
85// REG0x01, bits[3:0], manufacturer default: 0010 (2).
86// This setting determines the threshold for events that trigger the
87// IRQ Pin.
88void AS3935Component::write_watchdog_threshold(uint8_t watchdog_threshold) {
89 ESP_LOGV(TAG, "Setting watchdog sensitivity to %d", watchdog_threshold);
90 if ((watchdog_threshold < 1) || (watchdog_threshold > 10)) // 10 is the max sensitivity setting
91 return;
92 this->write_register(THRESHOLD, THRESH_MASK, watchdog_threshold, 0);
93}
94
95// REG0x01, bits [6:4], manufacturer default: 010 (2).
96// The noise floor level is compared to a known reference voltage. If this
97// level is exceeded the chip will issue an interrupt to the IRQ pin,
98// broadcasting that it can not operate properly due to noise (INT_NH).
99// Check datasheet for specific noise level tolerances when setting this register.
100void AS3935Component::write_noise_level(uint8_t noise_level) {
101 ESP_LOGV(TAG, "Setting noise level to %d", noise_level);
102 if ((noise_level < 1) || (noise_level > 7))
103 return;
104
105 this->write_register(THRESHOLD, NOISE_FLOOR_MASK, noise_level, 4);
106}
107// REG0x02, bits [3:0], manufacturer default: 0010 (2).
108// This setting, like the watchdog threshold, can help determine between false
109// events and actual lightning. The shape of the spike is analyzed during the
110// chip's signal validation routine. Increasing this value increases robustness
111// at the cost of sensitivity to distant events.
112void AS3935Component::write_spike_rejection(uint8_t spike_rejection) {
113 ESP_LOGV(TAG, "Setting spike rejection to %d", spike_rejection);
114 if ((spike_rejection < 1) || (spike_rejection > 11))
115 return;
116
117 this->write_register(LIGHTNING_REG, SPIKE_MASK, spike_rejection, 0);
118}
119// REG0x02, bits [5:4], manufacturer default: 0 (single lightning strike).
120// The number of lightning events before IRQ is set high. 15 minutes is The
121// window of time before the number of detected lightning events is reset.
122// The number of lightning strikes can be set to 1,5,9, or 16.
123void AS3935Component::write_lightning_threshold(uint8_t lightning_threshold) {
124 ESP_LOGV(TAG, "Setting lightning threshold to %d", lightning_threshold);
125 switch (lightning_threshold) {
126 case 1:
127 this->write_register(LIGHTNING_REG, ((1 << 5) | (1 << 4)), 0, 4); // Demonstrative
128 break;
129 case 5:
130 this->write_register(LIGHTNING_REG, ((1 << 5) | (1 << 4)), 1, 4);
131 break;
132 case 9:
133 this->write_register(LIGHTNING_REG, ((1 << 5) | (1 << 4)), 1, 5);
134 break;
135 case 16:
136 this->write_register(LIGHTNING_REG, ((1 << 5) | (1 << 4)), 3, 4);
137 break;
138 default:
139 return;
140 }
141}
142// REG0x03, bit [5], manufacturer default: 0.
143// This setting will return whether or not disturbers trigger the IRQ Pin.
144void AS3935Component::write_mask_disturber(bool enabled) {
145 ESP_LOGV(TAG, "Setting mask disturber to %d", enabled);
146 if (enabled) {
147 this->write_register(INT_MASK_ANT, (1 << 5), 1, 5);
148 } else {
149 this->write_register(INT_MASK_ANT, (1 << 5), 0, 5);
150 }
151}
152// REG0x03, bit [7:6], manufacturer default: 0 (16 division ratio).
153// The antenna is designed to resonate at 500kHz and so can be tuned with the
154// following setting. The accuracy of the antenna must be within 3.5 percent of
155// that value for proper signal validation and distance estimation.
156void AS3935Component::write_div_ratio(uint8_t div_ratio) {
157 ESP_LOGV(TAG, "Setting div ratio to %d", div_ratio);
158 switch (div_ratio) {
159 case 16:
160 this->write_register(INT_MASK_ANT, ((1 << 7) | (1 << 6)), 0, 6);
161 break;
162 case 22:
163 this->write_register(INT_MASK_ANT, ((1 << 7) | (1 << 6)), 1, 6);
164 break;
165 case 64:
166 this->write_register(INT_MASK_ANT, ((1 << 7) | (1 << 6)), 1, 7);
167 break;
168 case 128:
169 this->write_register(INT_MASK_ANT, ((1 << 7) | (1 << 6)), 3, 6);
170 break;
171 default:
172 return;
173 }
174}
175// REG0x08, bits [3:0], manufacturer default: 0.
176// This setting will add capacitance to the series RLC antenna on the product
177// to help tune its resonance. The datasheet specifies being within 3.5 percent
178// of 500kHz to get optimal lightning detection and distance sensing.
179// It's possible to add up to 120pF in steps of 8pF to the antenna.
180void AS3935Component::write_capacitance(uint8_t capacitance) {
181 ESP_LOGV(TAG, "Setting tune cap to %d pF", capacitance * 8);
182 this->write_register(FREQ_DISP_IRQ, CAP_MASK, capacitance, 0);
183}
184
185// REG0x03, bits [3:0], manufacturer default: 0.
186// When there is an event that exceeds the watchdog threshold, the register is written
187// with the type of event. This consists of two messages: INT_D (disturber detected) and
188// INT_L (Lightning detected). A third interrupt INT_NH (noise level too HIGH)
189// indicates that the noise level has been exceeded and will persist until the
190// noise has ended. Events are active HIGH. There is a one second window of time to
191// read the interrupt register after lightning is detected, and 1.5 after
192// disturber.
194 // A 2ms delay is added to allow for the memory register to be populated
195 // after the interrupt pin goes HIGH. See "Interrupt Management" in
196 // datasheet.
197 ESP_LOGV(TAG, "Calling read_interrupt_register_");
198 delay(2);
199 return this->read_register_(INT_MASK_ANT, INT_MASK);
200}
201
202// REG0x02, bit [6], manufacturer default: 1.
203// This register clears the number of lightning strikes that has been read in
204// the last 15 minute block.
206 // Write high, then low, then high to clear.
207 ESP_LOGV(TAG, "Calling clear_statistics_");
208 this->write_register(LIGHTNING_REG, (1 << 6), 1, 6);
209 this->write_register(LIGHTNING_REG, (1 << 6), 0, 6);
210 this->write_register(LIGHTNING_REG, (1 << 6), 1, 6);
211}
212
213// REG0x07, bit [5:0], manufacturer default: 0.
214// This register holds the distance to the front of the storm and not the
215// distance to a lightning strike.
217 ESP_LOGV(TAG, "Calling get_distance_to_storm_");
219}
220
222 ESP_LOGV(TAG, "Calling get_lightning_energy_");
223 uint32_t pure_light = 0; // Variable for lightning energy which is just a pure number.
224 uint32_t temp = 0;
225 // Temp variable for lightning energy.
227 // Temporary Value is large enough to handle a shift of 16 bits.
228 pure_light = temp << 16;
229 temp = this->read_register(ENERGY_LIGHT_MSB);
230 // Temporary value is large enough to handle a shift of 8 bits.
231 pure_light |= temp << 8;
232 // No shift here, directly OR'ed into pure_light variable.
233 temp = this->read_register(ENERGY_LIGHT_LSB);
234 pure_light |= temp;
235 return pure_light;
236}
237
238// REG0x03, bit [7:6], manufacturer default: 0 (16 division ratio).
239// This function returns the current division ratio of the resonance frequency.
240// The antenna resonance frequency should be within 3.5 percent of 500kHz, and
241// so when modifying the resonance frequency with the internal capacitors
242// (tuneCap()) it's important to keep in mind that the displayed frequency on
243// the IRQ pin is divided by this number.
244uint8_t AS3935Component::read_div_ratio() {
245 ESP_LOGV(TAG, "Calling read_div_ratio");
246 uint8_t reg_val = this->read_register_(INT_MASK_ANT, DIV_MASK);
247 reg_val >>= 6; // Front of the line.
248
249 if (reg_val == 0) {
250 return 16;
251 } else if (reg_val == 1) {
252 return 32;
253 } else if (reg_val == 2) {
254 return 64;
255 } else if (reg_val == 3) {
256 return 128;
257 }
258 ESP_LOGW(TAG, "Unknown response received for div_ratio");
259 return 0;
260}
261
262uint8_t AS3935Component::read_capacitance() {
263 ESP_LOGV(TAG, "Calling read_capacitance");
264 uint8_t reg_val = this->read_register_(FREQ_DISP_IRQ, CAP_MASK) * 8;
265 return (reg_val);
266}
267
268// REG0x08, bits [5,6,7], manufacturer default: 0.
269// This will send the frequency of the oscillators to the IRQ pin.
270// _osc 1, bit[5] = TRCO - System RCO at 32.768kHz
271// _osc 2, bit[6] = SRCO - Timer RCO Oscillators 1.1MHz
272// _osc 3, bit[7] = LCO - Frequency of the Antenna
273void AS3935Component::display_oscillator(bool state, uint8_t osc) {
274 if ((osc < 1) || (osc > 3))
275 return;
276
277 this->write_register(FREQ_DISP_IRQ, OSC_MASK, state, 4 + osc);
278}
279
280// REG0x3D, bits[7:0]
281// This function calibrates both internal oscillators The oscillators are tuned
282// based on the resonance frequency of the antenna and so it should be trimmed
283// before the calibration is done.
284bool AS3935Component::calibrate_oscillator() {
285 ESP_LOGI(TAG, "Starting oscillators calibration...");
286 this->write_register(CALIB_RCO, WIPE_ALL, DIRECT_COMMAND, 0); // Send command to calibrate the oscillators
287
288 this->display_oscillator(true, 2);
289 delay(2); // Give time for the internal oscillators to start up.
290 this->display_oscillator(false, 2);
291
292 // Check it they were calibrated successfully.
293 uint8_t reg_val_srco = this->read_register_(CALIB_SRCO, CALIB_MASK_NOK);
294 uint8_t reg_val_trco = this->read_register_(CALIB_TRCO, CALIB_MASK_NOK);
295
296 // reg_val_srco &= CALIB_MASK;
297 // reg_val_srco >>= 6;
298 // reg_val_trco &= CALIB_MASK;
299 // reg_val_trco >>= 6;
300 if (!reg_val_srco && !reg_val_trco) { // Zero upon success
301 ESP_LOGI(TAG, "Calibration was succesful");
302 return true;
303 } else {
304 ESP_LOGW(TAG, "Calibration was NOT succesful");
305 return false;
306 }
307}
308
309void AS3935Component::tune_antenna() {
310 ESP_LOGI(TAG, "Starting antenna tuning...");
311 uint8_t div_ratio = this->read_div_ratio();
312 uint8_t tune_val = this->read_capacitance();
313 ESP_LOGI(TAG, "Division Ratio is set to: %d", div_ratio);
314 ESP_LOGI(TAG, "Internal Capacitor is set to: %d", tune_val);
315 ESP_LOGI(TAG, "Displaying oscillator on INT pin. Measure its frequency - multiply value by Division Ratio");
316 this->display_oscillator(true, ANTFREQ);
317}
318
319uint8_t AS3935Component::read_register_(uint8_t reg, uint8_t mask) {
320 uint8_t value = this->read_register(reg);
321 value &= (~mask);
322 return value;
323}
324
325} // namespace as3935
326} // namespace esphome
virtual void setup()
Where the component's initialization should happen.
Definition component.cpp:51
void set_timeout(const std::string &name, uint32_t timeout, std::function< void()> &&f)
Set a timeout function with a unique name.
Definition component.cpp:72
virtual void setup()=0
virtual bool digital_read()=0
virtual uint8_t read_register(uint8_t reg)=0
virtual void write_register(uint8_t reg, uint8_t mask, uint8_t bits, uint8_t start_position)=0
uint8_t read_register_(uint8_t reg, uint8_t mask)
Definition as3935.cpp:319
bool state
Definition fan.h:0
@ ENERGY_LIGHT_MMSB
Definition as3935.h:26
const float DATA
For components that import data from directly connected sensors like DHT.
Definition component.cpp:19
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
void IRAM_ATTR HOT delay(uint32_t ms)
Definition core.cpp:28