2021-03-19 09:16:27 +01:00
|
|
|
#pragma once
|
|
|
|
|
2022-10-12 04:26:35 +02:00
|
|
|
#include "esphome/components/sensor/sensor.h"
|
2021-03-19 09:16:27 +01:00
|
|
|
#include "esphome/core/component.h"
|
2021-09-20 11:47:51 +02:00
|
|
|
#include "esphome/core/hal.h"
|
2021-03-19 09:16:27 +01:00
|
|
|
#include "esphome/core/helpers.h"
|
|
|
|
|
2023-10-18 08:33:36 +02:00
|
|
|
#include <cinttypes>
|
|
|
|
|
2021-03-19 09:16:27 +01:00
|
|
|
namespace esphome {
|
|
|
|
namespace pulse_meter {
|
|
|
|
|
|
|
|
class PulseMeterSensor : public sensor::Sensor, public Component {
|
|
|
|
public:
|
2022-02-20 21:32:35 +01:00
|
|
|
enum InternalFilterMode {
|
|
|
|
FILTER_EDGE = 0,
|
|
|
|
FILTER_PULSE,
|
|
|
|
};
|
|
|
|
|
2021-09-20 11:47:51 +02:00
|
|
|
void set_pin(InternalGPIOPin *pin) { this->pin_ = pin; }
|
2021-03-19 09:16:27 +01:00
|
|
|
void set_filter_us(uint32_t filter) { this->filter_us_ = filter; }
|
|
|
|
void set_timeout_us(uint32_t timeout) { this->timeout_us_ = timeout; }
|
|
|
|
void set_total_sensor(sensor::Sensor *sensor) { this->total_sensor_ = sensor; }
|
2023-08-07 22:14:20 +02:00
|
|
|
void set_filter_mode(InternalFilterMode mode) { this->filter_mode_ = mode; }
|
2023-10-19 23:28:05 +02:00
|
|
|
|
|
|
|
void set_total_pulses(uint32_t pulses);
|
2021-05-14 10:06:31 +02:00
|
|
|
|
2021-03-19 09:16:27 +01:00
|
|
|
void setup() override;
|
|
|
|
void loop() override;
|
2023-08-07 22:14:20 +02:00
|
|
|
float get_setup_priority() const override;
|
2021-03-19 09:16:27 +01:00
|
|
|
void dump_config() override;
|
|
|
|
|
|
|
|
protected:
|
2023-08-07 22:14:20 +02:00
|
|
|
static void edge_intr(PulseMeterSensor *sensor);
|
|
|
|
static void pulse_intr(PulseMeterSensor *sensor);
|
2021-03-19 09:16:27 +01:00
|
|
|
|
2022-09-15 01:53:02 +02:00
|
|
|
InternalGPIOPin *pin_{nullptr};
|
2021-03-19 09:16:27 +01:00
|
|
|
uint32_t filter_us_ = 0;
|
|
|
|
uint32_t timeout_us_ = 1000000UL * 60UL * 5UL;
|
2022-09-15 01:53:02 +02:00
|
|
|
sensor::Sensor *total_sensor_{nullptr};
|
2022-02-20 21:32:35 +01:00
|
|
|
InternalFilterMode filter_mode_{FILTER_EDGE};
|
2021-03-19 09:16:27 +01:00
|
|
|
|
2023-08-07 22:14:20 +02:00
|
|
|
// Variables used in the loop
|
2023-09-21 00:04:03 +02:00
|
|
|
enum class MeterState { INITIAL, RUNNING, TIMED_OUT };
|
|
|
|
MeterState meter_state_ = MeterState::INITIAL;
|
2024-05-08 00:13:15 +02:00
|
|
|
bool peeked_edge_ = false;
|
2023-08-07 22:14:20 +02:00
|
|
|
uint32_t total_pulses_ = 0;
|
|
|
|
uint32_t last_processed_edge_us_ = 0;
|
|
|
|
|
|
|
|
// This struct (and the two pointers) are used to pass data between the ISR and loop.
|
|
|
|
// These two pointers are exchanged each loop.
|
|
|
|
// Therefore you can't use data in the pointer to loop receives to set values in the pointer to loop sends.
|
|
|
|
// As a result it's easiest if you only use these pointers to send data from the ISR to the loop.
|
|
|
|
// (except for resetting the values)
|
|
|
|
struct State {
|
|
|
|
uint32_t last_detected_edge_us_ = 0;
|
2024-05-08 00:13:15 +02:00
|
|
|
uint32_t last_rising_edge_us_ = 0;
|
2023-08-07 22:14:20 +02:00
|
|
|
uint32_t count_ = 0;
|
|
|
|
};
|
|
|
|
State state_[2];
|
|
|
|
volatile State *set_ = state_;
|
|
|
|
volatile State *get_ = state_ + 1;
|
2021-03-19 09:16:27 +01:00
|
|
|
|
2023-08-07 22:14:20 +02:00
|
|
|
// Only use these variables in the ISR
|
|
|
|
ISRInternalGPIOPin isr_pin_;
|
2024-05-08 00:13:15 +02:00
|
|
|
|
|
|
|
/// Filter state for edge mode
|
|
|
|
struct EdgeState {
|
|
|
|
uint32_t last_sent_edge_us_ = 0;
|
|
|
|
};
|
|
|
|
EdgeState edge_state_{};
|
|
|
|
|
|
|
|
/// Filter state for pulse mode
|
|
|
|
struct PulseState {
|
|
|
|
uint32_t last_intr_ = 0;
|
|
|
|
bool latched_ = false;
|
|
|
|
bool last_pin_val_ = false;
|
|
|
|
};
|
|
|
|
PulseState pulse_state_{};
|
2021-03-19 09:16:27 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace pulse_meter
|
|
|
|
} // namespace esphome
|