|
|
- // -----------------------------------------------------------------------------
- // Abstract Energy Monitor Sensor (other EMON sensors extend this class)
- // Copyright (C) 2017 by Xose Pérez <xose dot perez at gmail dot com>
- // -----------------------------------------------------------------------------
-
- #pragma once
-
- #include "Arduino.h"
- #include "BaseSensor.h"
-
- class EmonSensor : public BaseSensor {
-
- public:
-
- // ---------------------------------------------------------------------
- // Public
- // ---------------------------------------------------------------------
-
- EmonSensor(): BaseSensor() {
- #if EMON_REPORT_CURRENT
- ++_magnitudes;
- #endif
- #if EMON_REPORT_POWER
- ++_magnitudes;
- #endif
- #if EMON_REPORT_ENERGY
- ++_magnitudes;
- #endif
- }
-
- void setVoltage(double voltage) {
- if (_voltage != voltage) _dirty = true;
- _voltage = voltage;
- }
-
- void setReference(double reference) {
- if (_reference != reference) _dirty = true;
- _reference = reference;
- }
-
- void setCurrentRatio(double current_ratio) {
- if (_current_ratio != current_ratio) _dirty = true;
- _current_ratio = current_ratio;
- }
-
- // ---------------------------------------------------------------------
- // Sensor API
- // ---------------------------------------------------------------------
-
- void begin() {
-
- // Resolution
- _adc_counts = 1 << _resolution;
-
- // Calculate factor
- _current_factor = _current_ratio * _reference / _adc_counts;
-
- // Calculate multiplier
- unsigned int s = 1;
- unsigned int i = 1;
- unsigned int m = s * i;
- while (m * _current_factor < 1) {
- _multiplier = m;
- i = (i == 1) ? 2 : (i == 2) ? 5 : 1;
- if (i == 1) s *= 10;
- m = s * i;
- }
-
- #if SENSOR_DEBUG
- Serial.print("[EMON] Current ratio: "); Serial.println(ratio);
- Serial.print("[EMON] Ref. Voltage: "); Serial.println(ref);
- Serial.print("[EMON] ADC Counts: "); Serial.println(_adc_counts);
- Serial.print("[EMON] Current factor: "); Serial.println(_current_factor);
- Serial.print("[EMON] Multiplier: "); Serial.println(_multiplier);
- #endif
-
- }
-
- protected:
-
- // ---------------------------------------------------------------------
- // Protected
- // ---------------------------------------------------------------------
-
- virtual unsigned int readADC(unsigned char channel) {}
-
- double read(unsigned char channel, double &pivot) {
-
- int sample;
- int max = 0;
- int min = _adc_counts;
- double filtered;
- double sum = 0;
-
- unsigned long time_span = millis();
- for (unsigned long i=0; i<_samples; i++) {
-
- // Read analog value
- sample = readADC(channel);
- if (sample > max) max = sample;
- if (sample < min) min = sample;
-
- // Digital low pass filter extracts the VDC offset
- pivot = (pivot + (sample - pivot) / EMON_FILTER_SPEED);
- filtered = sample - pivot;
-
- // Root-mean-square method
- sum += (filtered * filtered);
-
- }
- time_span = millis() - time_span;
-
- // Quick fix
- if (pivot < min || max < pivot) {
- pivot = (max + min) / 2.0;
- }
-
- // Calculate current
- double rms = _samples > 0 ? sqrt(sum / _samples) : 0;
- double current = _current_factor * rms;
- current = (double) (int(current * _multiplier) - 1) / _multiplier;
- if (current < 0) current = 0;
-
- #if SENSOR_DEBUG
- Serial.print("[EMON] Total samples: "); Serial.println(_samples);
- Serial.print("[EMON] Total time (ms): "); Serial.println(time_span);
- Serial.print("[EMON] Sample frequency (Hz): "); Serial.println(1000 * _samples / time_span);
- Serial.print("[EMON] Max value: "); Serial.println(max);
- Serial.print("[EMON] Min value: "); Serial.println(min);
- Serial.print("[EMON] Midpoint value: "); Serial.println(pivot);
- Serial.print("[EMON] RMS value: "); Serial.println(rms);
- Serial.print("[EMON] Current: "); Serial.println(current);
- #endif
-
- // Check timing
- if ((time_span > EMON_MAX_TIME)
- || ((time_span < EMON_MAX_TIME) && (_samples < EMON_MAX_SAMPLES))) {
- _samples = (_samples * EMON_MAX_TIME) / time_span;
- }
-
- return current;
-
- }
-
- unsigned char _magnitudes = 0;
- unsigned long _samples = EMON_MAX_SAMPLES;
-
- unsigned int _multiplier = 1;
- unsigned char _resolution = 10;
-
- double _voltage = EMON_MAINS_VOLTAGE;
- double _reference = EMON_REFERENCE_VOLTAGE;
- double _current_ratio = EMON_CURRENT_RATIO;
-
- unsigned long _adc_counts;
- double _current_factor;
-
-
-
- };
|