MySensors Library & Examples  2.3.2-62-ge298769
EnergyMeterPulseSensor.ino
1 /*
2  The MySensors Arduino library handles the wireless radio link and protocol
3  between your home built sensors/actuators and HA controller of choice.
4  The sensors forms a self healing radio network with optional repeaters. Each
5  repeater and gateway builds a routing tables in EEPROM which keeps track of the
6  network topology allowing messages to be routed to nodes.
7 
8  Created by Henrik Ekblad <[email protected]>
9  Copyright (C) 2013-2022 Sensnology AB
10  Full contributor list: https://github.com/mysensors/MySensors/graphs/contributors
11 
12  Documentation: http://www.mysensors.org
13  Support Forum: http://forum.mysensors.org
14 
15  This program is free software; you can redistribute it and/or
16  modify it under the terms of the GNU General Public License
17  version 2 as published by the Free Software Foundation.
18 
19  *******************************
20 
21  REVISION HISTORY
22  Version 1.0 - Henrik Ekblad
23  Version 1.1 - Peter Andersson added millis watt calculation if time between pulses > 1h
24 
25  DESCRIPTION
26  This sketch provides an example how to implement a LM393 PCB
27  Use this sensor to measure kWh and Watt of your house meter
28  You need to set the correct pulsefactor of your meter (blinks per kWh).
29  The sensor starts by fetching current kWh value from gateway.
30  Reports both kWh and Watt back to gateway.
31 
32  Unfortunately millis() won't increment when the Arduino is in
33  sleepmode. So we cannot make this sensor sleep if we also want
34  to calculate/report watt value.
35  http://www.mysensors.org/build/pulse_power
36 */
37 
38 // Enable debug prints
39 #define MY_DEBUG
40 
41 // Enable and select radio type attached
42 #define MY_RADIO_RF24
43 //#define MY_RADIO_NRF5_ESB
44 //#define MY_RADIO_RFM69
45 //#define MY_RADIO_RFM95
46 
47 #include <MySensors.h>
48 
49 #define DIGITAL_INPUT_SENSOR 3 // The digital input you attached your light sensor. (Only 2 and 3 generates interrupt!)
50 #define PULSE_FACTOR 1000 // Number of blinks per kWh of your meter. Normally 1000.
51 #define SLEEP_MODE false // Watt value can only be reported when sleep mode is false.
52 #define MAX_WATT 10000 // Max watt value to report. This filters outliers.
53 #define CHILD_ID 1 // Id of the sensor child
54 
55 uint32_t SEND_FREQUENCY =
56  20000; // Minimum time between send (in milliseconds). We don't want to spam the gateway.
57 double ppwh = ((double)PULSE_FACTOR) / 1000; // Pulses per watt hour
58 bool pcReceived = false;
59 volatile uint32_t pulseCount = 0;
60 volatile uint32_t lastBlinkmicros = 0;
61 volatile uint32_t lastBlinkmillis = 0;
62 volatile uint32_t watt = 0;
63 uint32_t oldPulseCount = 0;
64 uint32_t oldWatt = 0;
65 double oldkWh;
66 uint32_t lastSend;
67 MyMessage wattMsg(CHILD_ID, V_WATT);
68 MyMessage kWhMsg(CHILD_ID, V_KWH);
69 MyMessage pcMsg(CHILD_ID, V_VAR1);
70 
71 void IRQ_HANDLER_ATTR onPulse()
72 {
73  if (!SLEEP_MODE) {
74  uint32_t newBlinkmicros = micros();
75  uint32_t newBlinkmillis = millis();
76  uint32_t intervalmicros = newBlinkmicros - lastBlinkmicros;
77  uint32_t intervalmillis = newBlinkmillis - lastBlinkmillis;
78  if (intervalmicros < 10000L && intervalmillis < 10L) { // Sometimes we get interrupt on RISING
79  return;
80  }
81  if (intervalmillis < 360000) { // Less than an hour since last pulse, use microseconds
82  watt = (3600000000.0 / intervalmicros) / ppwh;
83  } else {
84  watt = (3600000.0 / intervalmillis) /
85  ppwh; // more thAn an hour since last pulse, use milliseconds as micros will overflow after 70min
86  }
87  lastBlinkmicros = newBlinkmicros;
88  lastBlinkmillis = newBlinkmillis;
89  }
90  pulseCount++;
91 }
92 
93 void setup()
94 {
95  // Fetch last known pulse count value from gw
96  request(CHILD_ID, V_VAR1);
97 
98  // Use the internal pullup to be able to hook up this sketch directly to an energy meter with S0 output
99  // If no pullup is used, the reported usage will be too high because of the floating pin
100  pinMode(DIGITAL_INPUT_SENSOR, INPUT_PULLUP);
101 
102  attachInterrupt(digitalPinToInterrupt(DIGITAL_INPUT_SENSOR), onPulse, RISING);
103  lastSend = millis();
104 }
105 
107 {
108  // Send the sketch version information to the gateway and Controller
109  sendSketchInfo(F("Energy Meter"), F("1.1"));
110 
111  // Register this device as power sensor
112  present(CHILD_ID, S_POWER);
113 }
114 
115 void loop()
116 {
117  uint32_t now = millis();
118  // Only send values at a maximum frequency or woken up from sleep
119  bool sendTime = now - lastSend > SEND_FREQUENCY;
120  if (pcReceived && (SLEEP_MODE || sendTime)) {
121  // New watt value has been calculated
122  if (!SLEEP_MODE && watt != oldWatt) {
123  // Check that we don't get unreasonable large watt value, which
124  // could happen when long wraps or false interrupt triggered
125  if (watt < ((uint32_t)MAX_WATT)) {
126  send(wattMsg.set(watt)); // Send watt value to gw
127  }
128  Serial.print("Watt:");
129  Serial.println(watt);
130  oldWatt = watt;
131  }
132 
133  // Pulse count value has changed
134  if (pulseCount != oldPulseCount) {
135  send(pcMsg.set(pulseCount)); // Send pulse count value to gw
136  double kWh = ((double)pulseCount / ((double)PULSE_FACTOR));
137  oldPulseCount = pulseCount;
138  if (kWh != oldkWh) {
139  send(kWhMsg.set(kWh, 4)); // Send kWh value to gw
140  oldkWh = kWh;
141  }
142  }
143  lastSend = now;
144  } else if (sendTime && !pcReceived) {
145  // No pulse count value received from controller. Try requesting it again.
146  request(CHILD_ID, V_VAR1);
147  lastSend = now;
148  }
149 
150  if (SLEEP_MODE) {
151  sleep(SEND_FREQUENCY, false);
152  }
153 }
154 
155 void receive(const MyMessage &message)
156 {
157  if (message.getType()==V_VAR1) {
158  pulseCount = oldPulseCount = message.getLong();
159  Serial.print("Received last pulse count value from gw:");
160  Serial.println(pulseCount);
161  pcReceived = true;
162  }
163 }
sendSketchInfo
bool sendSketchInfo(const char *name, const char *version, const bool requestEcho=false)
receive
void receive(const MyMessage &message)
Callback for incoming messages.
Definition: EnergyMeterPulseSensor.ino:155
loop
void loop()
Main loop.
Definition: EnergyMeterPulseSensor.ino:115
presentation
void presentation()
Node presentation.
Definition: EnergyMeterPulseSensor.ino:106
MyMessage::getLong
int32_t getLong(void) const
Get signed 32-bit integer payload.
MyMessage::set
MyMessage & set(const void *payload, const size_t length)
Set entire payload.
send
bool send(MyMessage &msg, const bool requestEcho=false)
MyMessage::getType
uint8_t getType(void) const
Get message type.
request
bool request(const uint8_t childSensorId, const uint8_t variableType, const uint8_t destination=GATEWAY_ADDRESS)
present
bool present(const uint8_t sensorId, const mysensors_sensor_t sensorType, const char *description="", const bool requestEcho=false)
setup
void setup()
Called after node initialises but before main loop.
Definition: EnergyMeterPulseSensor.ino:93
sleep
int8_t sleep(const uint32_t sleepingMS, const bool smartSleep=false)
IRQ_HANDLER_ATTR
#define IRQ_HANDLER_ATTR
ESP8266/ESP32 IRQ handlers need to be stored in IRAM.
Definition: MyHwHAL.h:52
MySensors.h
API declaration for MySensors.
MyMessage
MyMessage is used to create, manipulate, send and read MySensors messages.
Definition: MyMessage.h:290