HotWaterManager/HotWaterManager.ino
2025-04-02 09:05:47 +00:00

1544 lines
65 KiB
C++

/* Hot water manager
* -----------------
* This is a feature rich smart hot water management system
*
* More details and explanation of this project can be found at https://www.thefloatinglab.world/en/boilers.html
* This project is developed and maintained at https://git.thefloatinglab.world/thefloatinglab/hotwatermanager
*
* License
* -------
* HotWaterManager, a feature rich smart hot water management system.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*
* Objective
* ---------
* Objective was to create a hot water management system based on an Arduino architecture.
*
* More details and a list of features can be found at https://www.thefloatinglab.world/en/boilers.html
*
* Requirements
* ------------
* The microcontroller should be a ATMEGA328.
* The microcontroller clock speed needs to be at least 4 MHz
* The power supply should be 5V, with lower voltages the voltage measurement and range degrades and a standard 5V LCD will not work.
*/
// Libraries
#include "HotWaterManager.h"
#include <avr/eeprom.h>
#include <avr/wdt.h>
#include <avr/sleep.h>
#include <avr/boot.h>
#ifdef LCDADDR
#include <LiquidCrystal_I2C.h>
LiquidCrystal_I2C lcd(LCDADDR,16,2); // set the LCD address to LCDADDR for a 16 chars and 2 line display
#else
#include <LiquidCrystal.h>
// initialize the library by associating any needed LCD interface pin
// with the arduino pin number it is connected to
LiquidCrystal lcd(LCD_RS, LCD_EN, LCD_D4, LCD_D5, LCD_D6, LCD_D7);
#endif
#include <OneWire.h>
OneWire ds(TEMP); // Connect 1-wire devices
rom_t rom = {EPROMCHK,BRIGHTNESS,DISPTIMEOUT,VREQ,VFLOAT,VOFF,PAUSE,false,TMIN,TTARGET,TSANITY,TMAX,TSANDAYS,TNOSAN, 1000, PUMPDELTA, PUMPAFTERRUN, HWVTIMER, false, HWVPWM, 0, LEDM_TOUCH, false, HOLDON, 0, NIGHTSENSOR, RELAYPWM, false};
sensors_t sensors = {EPROMCHK,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, THERMISTOR, THERMBETA, true};
// Reserve space for two DS18B20 sensors
tempdata_t tempdata[] = {{0,0,1000},{0,0,-1000}};
button_t button = {BUTTON,0,false};
// Special characters for LCD
uint8_t hwvchar[8] = {0b10001,0b10001,0b10001,0b10001,0b10001,0b10001,0b10001,0b00000};
uint8_t Anchor[8] = {0b00100,0b01010,0b00100,0b00100,0b00100,0b10101,0b01110,0b00000};
uint8_t Sailing[8] = {0b00010,0b00110,0b01111,0b01111,0b11111,0b00010,0b11111,0b01110};
uint8_t engine[8] = {0b00000,0b01111,0b00100,0b01111,0b11111,0b01111,0b00000,0b00000};
uint8_t solar[8] = {0b11111,0b11111,0b01110,0b00000,0b10101,0b10101,0b10101,0b00000};
uint8_t legion[8] = {0b01110,0b11111,0b10101,0b11111,0b11011,0b11011,0b01110,0b00000};
uint8_t down[8] = {0b00100,0b00100,0b00100,0b00100,0b10101,0b01110,0b00100,0b00000};
uint8_t up[8] = {0b00100,0b01110,0b10101,0b00100,0b00100,0b00100,0b00100,0b00000};
uint8_t lightning[8]={0b00011,0b00110,0b01100,0b11110,0b00110,0b01100,0b01000,0b10000};
uint8_t Tmr[8] = {0b00000,0b01110,0b10101,0b10111,0b10001,0b01110,0b00000,0b00000};
// Variables
uint8_t mode=0;
uint8_t menusel=0;
boolean menuedit=false;
boolean swapTsens=false;
int16_t lastTemperature=0;
boolean lcdon=true;
boolean romdirty=false;
boolean freset=false;
boolean sreset=false;
boolean reboot=false;
boolean paused=true;
boolean restartTmrReq=true;
boolean hwv=false;
uint32_t secs=0;
uint32_t disptimer=0;
uint32_t menuseltimer=0;
uint32_t pauseTmr=0;
uint32_t restartTmr=0;
uint32_t pumpTimer=0;
uint32_t HWVtmr=0;
uint32_t lastlum=0;
uint16_t hwv_timer=0;
uint16_t rawVoltages[5];
uint16_t thermists[5];
uint8_t dummy;
uint8_t bright;
mma_t lum;
// menu code, 8 bits
#define MENU_8 B00001000
#define MENU_16 B00010000
#define MENU_SIGNED B10000000
#define MENU_FLOAT B00000011
#define MENU_BOOLEAN B01000000
#define MENU_TEMP B00000100
#define MENU_VOLTS B00100000
#define MENU_SHOW B11111111
// SFX extension code, 4 bits
#define NOALT B01000000
#define FNALT B00100000
// Suffixes for some menu options
const char sfx_seconds[] PROGMEM = " seconds";
const char sfx_minutes[] PROGMEM = " minutes";
const char sfx_days[] PROGMEM = " days";
const char sfx_percent[] PROGMEM = " percent";
const char sfx_volts[] PROGMEM = "V";
const char sfx_deg[] PROGMEM = "\xDF""K";
const char sfx_ohms[] PROGMEM = "\xF4";
const char* const suffixes[] = {sfx_seconds,sfx_minutes,sfx_days,sfx_percent,sfx_volts,sfx_deg,sfx_ohms};
// Build the menu table structure
typedef void (*funcptr_t)();
struct menu_t {
const char txt[17];
const uint8_t sfx;
const uint8_t code;
const int16_t min;
const int16_t max;
const int16_t step;
funcptr_t funcptr;
const void *varptr;
};
const menu_t PROGMEM menu[]= {
// Display, suffix, Menu flags min, max, step, function call, address of variable
{"System Volts", 0, MENU_8, 0, 1, 1, &fptr_vmul, &rom.vmul},
{"Temp Units", 0, MENU_8, 0, 1, 1, &fptr_units, &rom.fahrenheit},
{"Min Temperature", 0, MENU_16+MENU_SIGNED+MENU_TEMP+1, 0, 100, 10, &fptr_dummy, &rom.tmin},
{"Target Temp", 0, MENU_16+MENU_SIGNED+MENU_TEMP+1, 340, 520, 20, &fptr_dummy, &rom.ttarget},
{"Max Temperature", 0, MENU_16+MENU_SIGNED+MENU_TEMP+1, 600, 900, 50, &fptr_dummy, &rom.tmax},
{"Sanity Temp", 0, MENU_16+MENU_SIGNED+MENU_TEMP+1, 560, 680, 20, &fptr_dummy, &rom.tsanity},
{"Sanity interval", 3, MENU_16+MENU_SIGNED, 2, 14, 1, &fptr_dummy, &rom.tsandays},
{"No Sanity below", 0, MENU_16+MENU_SIGNED+MENU_TEMP+1, 200, 300, 10, &fptr_dummy, &rom.tnosan},
{"Start Voltage", 5, MENU_16+MENU_VOLTS+1, 138, 148, 1, &fptr_dummy, &rom.vreq},
{"Float Voltage", 5, MENU_16+MENU_VOLTS+1, 125, 140, 1, &fptr_dummy, &rom.vfloat},
{"Abort Voltage", 5, MENU_16+MENU_VOLTS+1, 120, 135, 1, &fptr_dummy, &rom.voff},
{"Pause duration", 2, MENU_16+MENU_SIGNED, 5, 45, 5, &fptr_dummy, &rom.pause},
{"Hold On time", 1, MENU_8, 0, 10, 1, &fptr_dummy, &rom.holdon},
#if HWMODEL != HWV_1_0
{"LCD Brightness", 0, MENU_8, 1, 7, 1, &fptr_brightness, &rom.brightness},
#if HWMODEL > HWV_1_1
{"LCD NightBright", 0, MENU_8, 0, 7, 1, &fptr_dummy, &rom.nightbright},
{"LCD NightSensor", 0, MENU_8, 6, 16, 1, &fptr_lumsens, &rom.nightsensor},
#endif
{"LCD Glow", 0, MENU_BOOLEAN, 0, 1, 1, &fptr_dummy, &rom.lcdglow},
#endif
{"LCD TimeOut", 1, MENU_8, 0, 10, 1, &fptr_dummy, &rom.disptimeout},
{"LED Mode", 0, MENU_8, 0, 4, 1, &fptr_ledmode, &rom.ledmode},
{"Vlt Calibration", 0, MENU_16, 920, 1080, 1, &fptr_showvoltage, &rom.calibration},
{"Pump Source", 0, MENU_8, 0, 2, 1, &fptr_pumpsource, &rom.pumpMode},
{"Pump delta", 6, MENU_16+MENU_SIGNED+1, 5, 100, 5, &fptr_dummy, &rom.pumpdelta},
{"Pump afterrun", 1, MENU_16, 0, 300, 30, &fptr_dummy, &rom.pumpafterrun},
{"Hot Water timer", 2, MENU_16, 0, 180, 15, &fptr_dummy, &rom.hwv_timer},
{"Hot Water pwr", 4, MENU_8, 50, 100, 10, &fptr_dummy, &rom.hwv_pwm},
#if HWMODEL != HWV_1_0
{"Relay hold pwr", 4, MENU_8, 50, 100, 10, &fptr_dummy, &rom.relaypwm},
#endif
{"Thermistor R", 7, MENU_16, 5000, 10000,5000, &fptr_showTemp, &sensors.thermistor},
{"Thermistor Beta", 0, MENU_16, 3000, 4200, 50, &fptr_showTemp, &sensors.thermbeta},
{"Swap temp sens", 0, MENU_BOOLEAN, 0, 1, 1, &fptr_showtemps, &swapTsens},
{"Reboot", 0, MENU_BOOLEAN, 0, 1, 1, &fptr_dummy, &reboot},
{"Factory reset", 0, MENU_BOOLEAN, 0, 1, 1, &fptr_dummy, &freset},
{"Sensors reset", 0, MENU_BOOLEAN, 0, 1, 1, &fptr_dummy, &sreset},
{"Versions", 0, MENU_SHOW, 0, 1, 1, &fptr_versions, &dummy},
#if HWMODEL != ARDUINO_
{"Serial number", 0, MENU_8, 0, 1, 1, &fptr_serialnr, &dummy},
#endif
{"",0,0,0,0,0,NULL,NULL}
};
// Interrupt routines
EMPTY_INTERRUPT(INT0_vect); // external interrupt 0 (button press) wakes the MCU
ISR(WDT_vect) { // Watchdog interrupt for the sleep timer
wdt_disable(); // disable watchdog
}
void(* resetFunc) (void) = 0; // declare reset function at address 0
// If no bootloader is used, this code is necessary to prevent a watchdog reset repeating itself after the reboot
void wdt_init(void) __attribute__((naked)) __attribute__((section(".init3")));
void wdt_init(void) {
MCUSR = 0;
wdt_disable();
return;
}
void setup() {
MCUSR = 0; // Just to make sure the watchdog from the previous run...
wdt_disable(); // ... does not continue to run
pinMode(BUTTON,INPUT_PULLUP);
pinMode(ENGINE,INPUT);
pinMode(LED,OUTPUT);
pinMode(HEATER,OUTPUT);
pinMode(AUX,OUTPUT);
pinMode(HWV,OUTPUT);
#if HWMODEL != ARDUINO_
pinMode(PIN_PB6,INPUT_PULLUP); // We use the internal oscillator, prevent this pin from floating
pinMode(PIN_PB7,INPUT_PULLUP); // We use the internal oscillator, prevent this pin from floating
#endif
#if HWMODEL > HWV_1_1
pinMode(LUMSENS,OUTPUT); // We use this ADC input to use the LED as a light sensor
#endif
brightness(BRIGHTNESS); // The brightness routine will define output ports as required
initLCD(); // Now the brightness has been set, initialize the LCD
boolean resetpressed=false;
uint32_t tmr=millis();
while(millis()-tmr<3000) { // If in these 3 seconds...
if(!digitalRead(BUTTON)) // the button is pressed...
resetpressed=true; // record the reset request
}
lcd.clear();
// Read the sensor addresses
eeprom_read_block((void*)&sensors, (void*)128, sizeof(sensors));
// If the reset button is pressed and we already have sensor addresses, offer the user to reset the sensor addresses as well
if(resetpressed && sensors.epromchk==EPROMCHK) {
lcd.print(F("Hold button to"));
lcd.setCursor(0,1);
lcd.print(F("reset sensors"));
delay(5000);
}
lcd.clear();
// Button still pressed or sensor addresses never initialized?
if(!digitalRead(BUTTON) || sensors.epromchk!=EPROMCHK || (!sensors.temphwa[0][0] && sensors.useDS)) {
lcd.print(F("Sensors reset"));
delay(2000);
lcd.clear();
sensors.epromchk=EPROMCHK;
sensors.useDS=true;
sensors.thermistor=THERMISTOR;
sensors.thermbeta=THERMBETA;
uint8_t x=1;
if(analogRead(TEMP)<980 && analogRead(TEMP)>200) {
sensors.useDS=false;
lcd.print(F("1 analog"));
} else {
#if HWMODEL > HWV_1_0
pinMode(OWPULLUP,OUTPUT);
digitalWrite(OWPULLUP,HIGH); // Switch on the extra pullup resistor for the digital sensors
#endif
x=searchSensors();
lcd.print(x);
lcd.print(F(" digital"));
}
lcd.setCursor(0,1);
lcd.print(F("sensors found"));
#ifndef SIMTEMP
if(!x) // If no temperature sensors have been found...
while(true); // continuation is pointless
#endif
eeprom_write_block((void*)&sensors, (void*)128, sizeof(sensors));
}
if(sensors.useDS) { // Do we use Digital Sensors?
#if HWMODEL > HWV_1_0
pinMode(OWPULLUP,OUTPUT);
digitalWrite(OWPULLUP,HIGH); // Switch on the extra pullup resistor for the digital sensors
#endif
ds.reset(); // Reset 'em
ds.select(sensors.temphwa[0]); // Request first temperature conversion
ds.write(0x44,1); // Temperature conversion
}
// Prime the Voltage and thermistor median arrays
for (int i = 0; i < 5; i++) {
rawVoltages[i] = readvoltage(10);
thermists[i] = analogRead(TEMP);
delay(15 + i * 10); // Use a non constant delay to avoid any pulsing loads to sync with the sampling frequency
}
eeprom_read_block((void*)&rom, (void*)0, 2); // Read the ROM header
if(resetpressed && rom.epromchk==EPROMCHK) { // Reset pressed, and we have an already configured EPROM?
lcd.print(F("Hold button to")); // Offer user to reset the ROM
lcd.setCursor(0,1);
lcd.print(F("Factory Reset"));
delay(5000);
lcd.clear();
}
if(resetpressed && !digitalRead(BUTTON) || rom.epromchk!=EPROMCHK) {// button is pressed or ROM not yet initialized
rom.epromchk=EPROMCHK;
if(getvoltage()>180) { // If the voltage is above 18 volts
rom.vreq*=2; // Multiply all default voltages by 2
rom.vfloat*=2;
rom.voff*=2;
rom.vmul=true; // Set the system up for 24 Volts
}
eeprom_write_block((void*)&rom, (void*)0, sizeof(rom));
lcd.clear();
lcd.print(F("Reset OK"));
}
#ifndef USEDEFAULT
eeprom_read_block((void*)&rom, (void*)0, sizeof(rom)); // These are the configuration settings.
#endif
brightness(rom.brightness);
// Prime the "history" of the temperature. Make sure we start with a downward trend
delay(1500);
lastTemperature=readtemp(0)+TRENDDELTA;
lcd.clear();
wdt_enable(WDTO_4S); // Enable watchdog, Use a 4 second interval
}
void initLCD() {
digitalWrite(LED,HIGH);
#ifdef LCDADDR
lcd.begin();
#else
lcd.begin(16, 2);
#endif
lcd.createChar(1, Anchor);
lcd.createChar(2, Sailing);
delay(100);
for(int offset=0;offset<6;offset++) {
lcd.setCursor(offset,0);
lcd.print(" ");
lcd.write((uint8_t)2);
delay(400);
}
delay(800);
lcd.setCursor(6,1);
lcd.write((uint8_t)1);
delay(1000);
lcd.clear();
lcd.print(F(" Hot Water"));
lcd.setCursor(2,1);
lcd.print(F("Manager V" VERSION));
lcd.createChar(1, engine);
lcd.createChar(2, solar);
lcd.createChar(3, lightning);
lcd.createChar(4, legion);
lcd.createChar(5, down);
lcd.createChar(6, up);
lcd.createChar(7, Tmr);
digitalWrite(LED,LOW);
}
// Main menu string constants
const char mode0[] PROGMEM = "Automatic";
const char mode1[] PROGMEM = "Pause";
const char mode2[] PROGMEM = "WarmUp";
const char mode3[] PROGMEM = "Sanitize";
const char mode4[] PROGMEM = "Thermostat";
const char mode5[] PROGMEM = "AntiFreeze";
const char mode6[] PROGMEM = "Heater On";
const char mode7[] PROGMEM = "Pump on";
const char mode8[] PROGMEM = "Off";
const char mode9[] PROGMEM = "Setup";
const char* const modes[] = {mode0,mode1,mode2,mode3,mode4,mode5,mode6,mode7,mode8,mode9,NULL};
void modemenu() {
// This function is used to traverse through the main menu
if(modes[menusel]==NULL) // Reached the end?
menusel=0; // Go to the first one
lcd.print("> ");
pgmLcdPrint(modes[menusel]); // Print the menu option from PROGMEM
lcd.setCursor(0,1); // On the next line...
lcd.print(F("Hold to select")); // Print this
menusel++; // Advance to the next menu option
}
void loop() {
wdt_reset(); // pat the watchdog, at least every 4 seconds, otherwise the microcontroller will reset
#if HWMODEL > HWV_1_1
// Use the LED to sense ambient light, so we can dim the LCD in the dark
if(rom.nightbright && rom.nightbright<rom.brightness) { // Valid configuration of rom.nightbright?
if(seconds()-lastlum>1) { // Only once per 2 seconds
lastlum=seconds();
boolean ledstate=digitalRead(LED); // Remember the current state of the LED
analogRead(LUMSENS); // Prime the ADC (we arrive here after sleep mode, so extra long read)
analogRead(LUMSENS); // Throw away the next read as well (just to be sure)
pinMode(LUMSENS,OUTPUT); // Set LUMSENS to an output again
digitalWrite(LUMSENS,HIGH); // Switch everything to high
digitalWrite(LED,HIGH); // Switch the LED to high
// See comments below why we manipulate the register manually
DDRE &= ~4; // Switch the LUMSENS pin to an INPUT, and since the output was HIGH, it is now an INPUT_PULLUP
uint16_t sens=analogRead(LUMSENS); // Read full voltage (should be close to 1023)
pinMode(LUMSENS,OUTPUT); // Switch back to output again
digitalWrite(LUMSENS,HIGH); // Reverse bias the LED
digitalWrite(LED,LOW); // Reverse bias the LED
// What we want to do:
// pinMode(LUMSENS,INPUT_PULLUP); // Switch it to input pullup
// But if pinMode first switches the output to LOW, and then switches to an input, we lost some charge from the LED.
// So we do it manually, FIRST swich to an INPUT, THEN get rid of the pullup resistor.
DDRE &= ~4; // Switch the LUMSENS pin to an INPUT, and since the output was HIGH, it is now an INPUT_PULLUP
PORTE &= ~4; // Switch off the PULLUP resistor.
delay(5); // Wait 5 milliseconds
sens-=analogRead(LUMSENS); // Read the difference
pinMode(LUMSENS,OUTPUT); // Switch LUMSENS back to an output again
digitalWrite(LUMSENS,LOW); // Make sure it is low again
digitalWrite(LED,ledstate); // Reset the led to whatever state it had
if(sens>1024) // Full voltage lower than what we measured on the LED?
sens=0; // Makes no sens...
int16_t br=rom.nightbright+(modMovAvg(&lum,sens,LUMSAMPLES)-rom.nightsensor);
if(br>rom.brightness) // If brightness is higher than what we allow
br=rom.brightness; // Limit it
if(br<rom.nightbright) // If brightness is lower than what we allow
br=rom.nightbright; // Limit it
bright=br; // convert int16 to uint8, it now falls in range
if(lcdon) // Only if the LCD is on:
brightness(bright); // Set the actual new brightness
} else {
bright=rom.brightness; // If nightbright is switched off, fall back to normal brightness
}
}
#endif
int16_t butt=chkbutton(&button); // Capture button presses
switch(rom.ledmode) { // Process the LED modes
case LEDM_WARMWATER: { // Do we have hot water?
digitalWrite(LED,(readtemp(0) >= rom.ttarget));
}
break;
case LEDM_HEATING: { // Are we currently heating?
digitalWrite(LED,(digitalRead(HEATER) || digitalRead(AUX)));
}
break;
case LEDM_HWVALVE: { // Is the hot water timer running?
digitalWrite(LED,hwv && hwv_timer);
}
break;
case LEDM_LEGIONELLA: { // Is the water unsafe?
digitalWrite(LED,sanitycheck());
}
break;
default: { // Is the button pressed?
digitalWrite(LED,!digitalRead(button.button));
}
}
// Deal with the LCD backlight
if(butt) { // Did we have a button press?
disptimer=seconds(); // Restart the display timeout timer
if(!lcdon) { // If the backlight was not already on, turn it on now
brightness(bright); // Set the configured brightness
lcdon=true; // Remember that the LCD backlight is on
butt=0; // This keypress is "used up" to switch the LCD backlight on.
}
}
if(butt) { // We still have the button press? (Could be "eaten up" by the LCD backlight timeout)
menuseltimer=seconds(); // Reset the menu timeout timer
if(butt==2) { // Long button press
if(menusel) { // Are we already in the menu?
if(menusel>=100) { // Are we in the setup menu?
menuedit=!menuedit; // Long press is either exit or entry of a setup function
lcd.clear();
// Show the associated menu option
printMenu(menusel-100,menuedit);
if(!menuedit && romdirty) { // We are not in edit mode but made the rom "dirty"?
if(freset)
rom.epromchk=0;
if(sreset)
sensors.epromchk=0;
#ifndef SIMTEMP
if(swapTsens) { // If we need to swap the sensors, do it now...
for(uint8_t i=0;i<8;i++) {
uint8_t x=sensors.temphwa[0][i];
sensors.temphwa[0][i]=sensors.temphwa[1][i];
sensors.temphwa[1][i]=x;
}
swapTsens=false;
}
#endif
if(reboot) // Only a reboot?
resetFunc(); // Don't write EEPROM, just reboot.
// Save the changed entries in the ROM
eeprom_write_block((void*)&rom, (void*)0, sizeof(rom));
eeprom_write_block((void*)&sensors, (void*)128, sizeof(sensors));
if(freset || sreset) // Reset rom or sensors?
resetFunc(); // Follow it with a reset
}
} else { // We are not in the setup menu
if(modes[menusel]==NULL) { // We reached the setup menu
menusel=99; // Select it
butt=1; // Simulate short press to enter it
} else { // It is a long press on a menu option
paused=false;
mode=menusel-1; // so we selected this one
menusel=0; // Exit the menu
lcd.clear();
}
}
} else { // Not yet in the menu
lcd.clear();
modemenu(); // So start it now
}
}
if(butt==1) { // Short press
lcd.clear();
// Are we editing a setup menu item?
if(menuedit) { // Are we in edit mode?
editItem(menusel-100); // Edit the item
printMenu(menusel-100,menuedit);
} else { // Not in edit mode
if(menusel>=99) { // Are we in the setup menu?
nextMenuItem(); // Advance to the next menu item
} else { // Not in the setup menu
if(!menusel && rom.hwv_timer) { // Have we configured a hot water valve timer?
// Toggle the hot water valve timer
setHWV(!(hwv && hwv_timer>HWVBLEED),rom.hwv_timer*60);
} else { // No HWV...
modemenu(); // ...just go into the main menu
}
}
}
}
} else { // No button was pressed
if(menusel>=100 && menuedit && getPgmVal(&menu[menusel-100].code)==MENU_SHOW) {
lcd.setCursor(0,0);
printMenu(menusel-100,menuedit); // This menu option has no timeout
} else {
if(menusel && seconds()-menuseltimer>(MENUTIMEOUT)) {
menusel=0; // Menu timed out, get out of it
menuedit=false; // Exit edit mode
romdirty=false; // If we were in edit mode, discard the change
lcd.clear();
}
}
if(!menusel) { // If we are not in a menu...
processmodes(); // ... Process the mode
}
}
if(!menusel && !butt && !button.pressed && !hwv && !digitalRead(HEATER)) {
// We are not in a menu, no button is pressed, and we have no hot water valve timer going...
// Let's get some sleep!
// We are going to use the Watchdog timer for this.
// We will shut down everything else.
// The sleep will end when either the watchdog timer or an interrupt (button press) fires.
uint8_t wdt_prescale = 6; // Sleep for 1 second
uint8_t wdtcsr = bit(WDIE) | (wdt_prescale&7) | ((wdt_prescale&8)<<2);
wdt_disable(); // Disable the watchdog timer for now.
uint8_t adcsra=ADCSRA; // Store current state of ADCSRA
ADCSRA = 0; // disable the ADC
MCUSR = 0; // clear various "reset" flags
WDTCSR = bit (WDCE) | bit (WDE); // allow changes, disable reset
WDTCSR = wdtcsr; // set WDIE, and prescaler
wdt_reset(); // pat the dog
EICRA = _BV(ISC01); // configure INT0 to trigger on falling edge
EIMSK = _BV(INT0); // enable INT0, so we wake up immediately if the button is pressed
set_sleep_mode(SLEEP_MODE_PWR_DOWN); // Power down everything
sleep_enable();
sei(); // ensure interrupts enabled so we can wake up again
sleep_cpu(); // go to sleep. Here we sleep
// Zzzzzzz
sleep_disable(); // We woke up, don't fall asleep again, there is work to do
ADCSRA=adcsra; // Restore ADCSRA
wdt_enable(WDTO_4S); // Re-enable the watch dog
secs++; // The timer stopped during sleep, make good for it with the one second we slept
// _millis+=(16<<wdt_prescale); // Add the time spent sleeping to our timer (use this for sleep periods other than 1 second)
}
}
void processmodes() {
// This is the function that controls the hot water management
uint16_t voltage=getvoltage();
static uint16_t LCDvoltage;
static uint32_t vfreq;
// Did the display backlight timeout?
if(lcdon && rom.disptimeout && seconds()-disptimer>rom.disptimeout) {
lcdon=false;
brightness(0); // Switch off the back light
}
// Change the value displayed on the LCD only once per second
if(!LCDvoltage || seconds()-vfreq>1) {
LCDvoltage=voltage;
vfreq=seconds();
}
lcd.setCursor(0,0);
// If the alternate temp sensor has a higher value than the tank sensor, show both of them alternately.
uint8_t src=0;
if(rom.pumpMode && readtemp(1)-readtemp(0)>rom.pumpdelta && (seconds() & 2)) {
lcd.print(F("Source"));
src=1;
} else {
lcd.print(F("Tank "));
}
printTemperature(readtemp(src),1); // Display the temperature
lcd.setCursor(0,1);
pgmLcdPrint(modes[mode]); // Print the current operation mode
lcd.setCursor(11,1);
lcd.print(((float)LCDvoltage)/10.0,1); // Display the voltage
lcd.print("V ");
// Update the temperature trend arrow
lcd.setCursor(13,0);
if(tempdata[src].failures) // If we failed to receive valid data from the sensor...
lcd.write('?'); // ... indicate this with a question mark...
else {
#ifdef SIMTEMP
lcd.write(6); // Simulate upward trend
#else
if(readtemp(0)-lastTemperature>=TRENDDELTA) {
lcd.write(6); // Trend is going up
lastTemperature=readtemp(0);
} else
if(lastTemperature-readtemp(0)>=TRENDDELTA) {
lcd.write(5); // Trend is going down
lastTemperature=readtemp(0);
}
}
#endif
lcd.setCursor(14,0);
if(hwv && hwv_timer) { // If the hot water valve is selected...
for(uint8_t i=0; i < 7 ;i++) { // build the character
if(i<6 && ((seconds()-HWVtmr)*7)/hwv_timer > i)
hwvchar[i] = 0b10001;
else
hwvchar[i] = 0b11111;
}
lcd.createChar(0, hwvchar); // create the character
lcd.setCursor(14,0); // needed after createChar
lcd.write((uint8_t)0); // Display the character
} else
if(sanitycheck())
lcd.write(4); // Display the legionella warning
else // Or...
lcd.write(' '); // ... clear it
lcd.setCursor(15,0);
if(!mode && digitalRead(ENGINE)) // AUtomatic mode and engine contact is on?
lcd.write(1); // Show the engine symbol
else
if(digitalRead(AUX)) // Is the auxilliary pump running?
lcd.write(3 - (int8_t) rom.pumpMode); // Show the respective symbol
else
if(paused) // If we are pausing...
lcd.write(7); // ... show it on the screen
else
if(digitalRead(HEATER)) // If the heating element is energized
lcd.write(3); // Show the symbol on the display
else // Or...
lcd.write(' '); // ... clear it when the heating element is de-energized.
if(hwv && seconds()-HWVtmr > hwv_timer) // If the HWV is enabled and expired...
setHWV(false,0); // ... switch it off
if(voltage<rom.vfloat) // If the voltage has become below the float voltage...
restartTmrReq=true; // ... we need to restart the timer
if(readtemp(0)>rom.tmax) { // If Temp is higher than max...
heaterState(false); // Switch off the heating element
pumpState(LOW); // Switch off the auxilliary pump
return; // Nothing else needs to be done, we're finished
}
if(voltage<rom.voff) { // If voltage below the lower threshold, pause whatever program is running
heaterState(false);
pumpState(LOW); // The pump also consumes power, so let's keep it off until the voltage goes up.
paused=true;
pauseTmr=0;
}
if(paused || mode==1) { // If Paused, keep boiler off until the timer expires
if(!paused) { // Was the mode set manually (mode==1) to pause?
pauseTmr=0; // Then we actually need to start the pause timer.
paused=true;
}
if(voltage>=rom.vreq) { // Voltage reached the required value
if(!pauseTmr) { // Did we already start the countdown?
pauseTmr=seconds(); // If not, start it now
} else {
if(seconds()-pauseTmr>rom.pause*60L) { // Has the timer already expired?
paused=false; // Cancel pause mode
if(mode==1) // If the pause mode was manually selected...
mode=0; // ... reset it to automatic mode
} else { // Nope, we still need to pause
heaterState(false); // Make sure the water heater is off
// Do not return, maybe solar can still input some heat
}
}
}
}
// We've handled all hard limits, now look at the modes
if(mode==8) { // If mode is OFF...
heaterState(false); // - Shut down the water heater
pumpState(LOW); // - Shut down also the auxilliary pump
return; // - No further actions required.
}
if(mode==7) { // Is mode is PUMP
pumpState(HIGH); // Run the pump
return; // Do nothing else
}
// If we have an auxilliary pump,
if(rom.pumpMode) { // Do we have an auxilliary pump?
if(readtemp(1)-readtemp(0)>rom.pumpdelta) { // Is there a worthwile temperature difference between the source and hot water tank?
pumpState(HIGH); // Then start the pump
heaterState(false); // And disable the heating element
pumpTimer=seconds(); // Record the time
if(mode!=6) // Unless the mode is "heater on"...
return; // We are warming the boiler, as long as we can do that no electricity is needed, skip the rest.
} else { // Not enough temperature difference
if(seconds()-pumpTimer>rom.pumpafterrun) // Unless the "aterrun" timer is still running
pumpState(LOW); // Switch off the pump
}
}
if(mode==6) { // If program=ON...
heaterState(true); // ... just keep the heater on. Overtemp and pause is already dealt with above.
return; // Nothing else needs to be done
}
if(mode==2) { // If the program is in WarmUp mode...
if(readtemp(0)<rom.ttarget) { // and temperature below the target
heaterState(true); // Keep the heater on
return; // Finished here
} else { // Temperature reached the target?
mode=0; // Then revert to automatic mode
}
}
if(mode==3) { // If the program is in Sanitize mode...
if(readtemp(0)<rom.tsanity) { // and temperature below the target
heaterState(true); // Keep the heater on
return; // Finished here
} else { // Temperature reached the target?
mode=0; // Then revert to automatic mode
}
}
if(mode==4) { // If the mode is "thermostat"
if(readtemp(0)<rom.ttarget) { // If the temperature is below the target
heaterState(true); // Keep the heater on
} else { // Otherwise
heaterState(sanitycheck()); // Use the heater when sanitize is required, otherwise switch it off
}
return; // We're finished
}
// If we arrive here, we are either running in auto or antifreeze mode.
if(paused) { // In pause mode
heaterState(false); // Keep the heater off
return;
}
if(readtemp(0)<rom.tmin) { // Hot water tank about to freeze?
heaterState(true); // Turn the heating element on
return; // Finished
}
if(mode==5) { // In antifreeze mode, if we made it until here, the water is warm enough
heaterState(false); // So switch off the heater
return; // We're finished
}
// If we make it until here, we are in automatic mode
if(digitalRead(ENGINE)) { // Engine contact is on?
heaterState(false); // Don't use the heater
return;
}
// Is the boiler off? Maybe we can switch it on?
if(!digitalRead(HEATER)) {
if(voltage<rom.vreq) // If voltage too low...
return; // ... Forget about it
// The voltage is high enough, so check to see if we need to reset the timer
if(restartTmrReq) {
restartTmrReq=false;
restartTmr=seconds();
}
uint32_t holdoff=1;
// Did we already reach the target temperature, then we can be a bit more conservative with energy consumption
if(readtemp(0)>rom.ttarget) {
if(sanitycheck()) {
holdoff=2; // We are due to run a sanity cycle, so let's try to reach it with moderate conservatism
} else {
holdoff=5; // Sanity cycle is not due, so let's not try too hard to get the temperature up to the sanity level
}
}
if(readtemp(0)>rom.tsanity) // Did we reach the sanity temperature?
holdoff=10; // We just heat the water further but only if we are otherwise waisting available energy.
if(seconds()-restartTmr>holdoff*60L) // Now we need to see how long ago we reached (again) VREQ. Can we already start?
heaterState(true);
return;
}
// If we arrive here, the boiler is on. Do we want to keep it that way?
if(voltage<rom.vfloat) { // If boiler is on but we get under the float voltage, switch it off
heaterState(false);
return;
}
}
void heaterState(boolean state) {
static uint32_t lastChange=0;
// Take care of the relay PWM if it has been configured
if(state && rom.relaypwm<100 && digitalRead(HEATER) && seconds()-lastChange>2)
analogWrite12V(HEATER,rom.relaypwm);
// If we want to change state, do so only after at least some time has passed since the last change
if((!state && seconds()-lastChange>rom.holdon) || (state && seconds()-lastChange>HOLD_OFF)) {
if(state) // If we switch the heater on...
pumpState(LOW); // Switch off the auxilliary pump
if(digitalRead(HEATER)!=state) { // Need to change the heater state?
lastChange=seconds();
delay(20);
if(state)
analogWrite12V(HEATER,100); // Set heater output to 100%
else
analogWrite12V(HEATER,0); // Set heater output to 0%
// digitalWrite(HEATER,state); // Switch the heater
if(rom.hwv_timer && !hwv && state) // If we have a HWV and it is not on, and we switch on the heater...
setHWV(true,HWVBLEED); // ... switch on the HWV to equalize the pressure
}
}
}
void pumpState(boolean state) {
digitalWrite(AUX,state);
}
void setHWV(bool state, uint16_t hwvtmr) {
if(state) { // We want to switch on the Hot Water Valve?
HWVtmr=seconds(); // Record the time
hwv_timer=hwvtmr; // Set the duration
}
hwv=state; // Register the new state
if(state) // We want to switch it on
analogWrite12V(HWV,rom.hwv_pwm); // Use PWM for this
else // Otherwise
digitalWrite(HWV,LOW); // Switch it off
}
void analogWrite12V(uint8_t port,uint8_t percent) {
if(percent>=100) {
digitalWrite(port,HIGH); // Set port to high so we can read back port state
if(rom.vmul) // If 24V
analogWrite(port,127); // Set an initial value of 50%.
} else {
uint16_t val = ((uint16_t)percent*256)/100;
if(rom.vmul) // In case of 24V
val/=2; // Divide PWM by 2
analogWrite(port,val);
}
if(!percent) // If we want to switch it completely off
digitalWrite(port,LOW); // Do a digitalWrite
}
// This routine will keep track whether there is a legionella chance
boolean sanitycheck() {
static boolean sancounting=false;
static boolean keepalarm=false; // Prevent the alarm switching off after timer rollover
static uint32_t santotal=0;
static uint32_t sancount=0x80000000;
if(readtemp(0)>=rom.tsanity) { // The temperature in boiler is high enough to kill legionella, so clear all alarms.
santotal=0;
sancounting=false;
keepalarm=false;
} else {
if(readtemp(0)>rom.tnosan) {
if(!sancounting) {
sancounting=true;
sancount=seconds();
}
if(santotal+(seconds()-sancount)>((uint32_t)rom.tsandays)*60L*60L*24L) {
keepalarm=true;
}
} else {
if(sancounting) {
sancounting=false;
santotal+=(seconds()-sancount);
}
}
}
return keepalarm;
}
void brightness(uint8_t brightness) {
// Controlling the brightness of the LCD backlight could be done with a single PWM output, but then we couldn't use the sleep function of the processor anymore.
// We have enough IO ports anyway, so we use three outputs with different resistors with which we can make 8 different brightness values.
// Setup an array with the output ports in binary order
const uint8_t blports[]={BACKLIGHT_B0, BACKLIGHT_B1, BACKLIGHT_B2};
if(brightness)
bright=brightness; // Remember the new brightness
uint8_t inputmode=INPUT; // If a port is not used, set it as input
if(!brightness && rom.lcdglow) // but if the brightness is 0 and we selected "glow"....
inputmode=INPUT_PULLUP; // ...make it input_pullup to allow some leak current through the LCD backlight
for(uint8_t i=0; i<3; i++) { // Process all three bits
if(brightness & (1<<i)) { // Isolate the bit we're looking for. If this bit is on...
pinMode(blports[i],OUTPUT); // ... select it as an output
digitalWrite(blports[i],HIGH); // ... and pull it high
} else // otherwise
pinMode(blports[i],inputmode); // select it as an input to neutralize it (or put it into glow mode)
}
}
// Function Pointer routines for some special setup menu processing. The addresses of these functions are referred to in the menu table
void fptr_dummy(void) {
}
void fptr_brightness() {
brightness(rom.brightness);
}
void fptr_showvoltage() {
lcd.setCursor(11,1);
lcd.print((float)readvoltage(10)/10.0,1); // We use readvoltage here because getvoltage only reads once per second and is too sluggish
lcd.write('V');
getvoltage(); // Update the voltage in the median array, so when we exit the menu a matching voltage is shown right away instead of the old value
}
void fptr_showTemp() {
lcd.setCursor(9,1);
tempdata[0].mmaTemp.samples=0;
tempdata[0].mmaTemp.sum=0;
printTemperature(readtemp(0),1);
}
void fptr_pumpsource() {
switch(rom.pumpMode) {
case 2: {
lcd.print(F("Solar"));
}
break;
case 1: {
lcd.print(F("Engine"));
}
break;
default: {
lcd.print(F("None"));
}
}
}
void fptr_showtemps() {
lcd.setCursor(5,1);
if(sensors.useDS && sensors.temphwa[1][0]) {
printTemperature(readtemp(0),0);
lcd.print(" ");
printTemperature(readtemp(1),0);
} else {
lcd.print("N/A");
swapTsens=false;
}
}
void fptr_vmul() {
if(rom.vmul) { // If 24V range is selected
lcd.print(24);
if(rom.vreq<180) { // Multiply all configurable voltages by 2 if the configured voltage is below 18 volts
rom.vreq*=2;
rom.vfloat*=2;
rom.voff*=2;
}
} else { // If 12V range is selected
lcd.print(12);
if(rom.vreq>180) { // Divide all configurable voltages by 2 if the configured voltage is above 18 volts
rom.vreq/=2;
rom.vfloat/=2;
rom.voff/=2;
}
}
lcd.print(F("Volt"));
}
void fptr_units() {
if(rom.fahrenheit)
lcd.print(F("Fahrenheit"));
else
lcd.print(F("Celsius"));
}
void fptr_ledmode() {
static const char lmod0[] PROGMEM = "Touch";
static const char lmod1[] PROGMEM = "Warm water";
static const char lmod2[] PROGMEM = "Heating up";
static const char lmod3[] PROGMEM = "Solenoid";
static const char lmod4[] PROGMEM = "Legionella";
static const char* const ledmods[] = {lmod0,lmod1,lmod2,lmod3,lmod4};
lcd.setCursor(4,1);
pgmLcdPrint(ledmods[rom.ledmode]);
}
void fptr_versions() {
lcd.print(F("SW " VERSION " HW " HWVER ));
}
#if HWMODEL != ARDUINO_
void fptr_lumsens() {
lcd.setCursor(8,1);
lcd.print(lum.avg); // Show the current sensor output
lcd.print(" ");
}
void fptr_serialnr() {
char signature[10];
char base64str[17];
lcd.setCursor(0,1);
for(uint8_t i=0;i<10;i++)
signature[i]=boot_signature_byte_get(i+14);
base64_encode(base64str, signature, 10);
lcd.print(base64str);
}
#endif
void printMenu(uint8_t item, boolean edit) {
uint8_t code = getPgmVal(&menu[item].code);
uint16_t value = getRomValue(code,item);
if(!edit)
lcd.write('>');
if(code==MENU_SHOW) {
if(edit)
lcd.print(F("Hold to exit"));
else
pgmLcdPrint(menu[item].txt);
lcd.setCursor(2,1);
} else {
pgmLcdPrint(menu[item].txt);
lcd.setCursor(0,1);
if(edit)
lcd.print("> ");
else
lcd.print(" ");
if(code & (MENU_8 | MENU_16)) {
if(code & MENU_FLOAT)
if(code & MENU_TEMP)
printTemperature(value,0);
else
lcd.print((float) ((float)value/pow(10.0,(float)(code & MENU_FLOAT))), code & MENU_FLOAT);
else
if(code & MENU_SIGNED)
lcd.print((int)value);
else
lcd.print(value);
}
uint8_t i=pgm_read_byte(&menu[item].sfx) & 0x0F;
if(i)
pgmLcdPrint(suffixes[i-1]);
if(code & MENU_BOOLEAN)
if(value)
lcd.print("Yes");
else
lcd.print("No ");
}
// If the menu has a function pointer, process it now
lcd.setCursor(2,1);
void (*function)(void); // function buffer
function = (funcptr_t)pgm_read_ptr(&menu[item].funcptr);// get it from PROGMEM
function(); // Call the function
}
uint16_t getRomValue(uint8_t code, uint8_t item) {
if(code & (MENU_8 | MENU_BOOLEAN)) {
if(code & MENU_SIGNED)
return (uint16_t)((int16_t)*((int8_t *)pgm_read_word(&menu[item].varptr)));
else
return (uint16_t)*((uint8_t *)pgm_read_word(&menu[item].varptr));
}
if(code & MENU_16)
return (uint16_t)*((uint16_t *)pgm_read_word(&menu[item].varptr));
}
void editItem(uint8_t item) {
uint8_t code = getPgmVal(&menu[item].code);
if(code==MENU_SHOW)
return;
int16_t value = getRomValue(code,item);
int16_t step=getPgmVal(&menu[item].step);
int16_t max=getPgmVal(&menu[item].max);
int16_t min=getPgmVal(&menu[item].min);
romdirty=true; // Remember to save changes in EEPROM
if(code & MENU_VOLTS && rom.vmul) { // If some voltage menu option is selected and we operate on a 24V system...
step*=2; max*=2; min*=2; // Multiply all voltage steps, minimums and maximums by two.
}
value+=step;
value-=value%step;
if(value>max || value<min)
value=min;
if(code & (MENU_8 | MENU_BOOLEAN)) {
if(code & MENU_SIGNED)
*((int8_t *)pgm_read_word(&menu[item].varptr)) = (int8_t)value;
else
*((uint8_t *)pgm_read_word(&menu[item].varptr)) = (uint8_t)value;
}
if(code & MENU_16)
if(code & MENU_SIGNED)
*((int16_t *)pgm_read_word(&menu[item].varptr)) = value;
else
*((uint16_t *)pgm_read_word(&menu[item].varptr)) = value;
}
void nextMenuItem() {
uint8_t fn;
do {
menusel++; // Advance to the next item
if(getPgmVal(menu[menusel-100].txt)=='\0')
menusel=100; // Reached the last one, rewind to the first one
fn=getPgmVal(&menu[menusel-100].sfx);
} while((rom.altfunc && ( fn & NOALT)) || (!rom.altfunc && (fn & FNALT)) );
printMenu(menusel-100,menuedit);
}
uint16_t readvoltage(uint32_t mag) {
#ifdef SIMVOLT
return SIMVOLT;
#endif
analogRead(VCCREF);
return (uint16_t)(((((((((uint32_t)analogRead(VCCREF))*(R1+R2))/R2)*ADCREF)+(rom.calibration/2))/(uint32_t)rom.calibration)+(mag*5))/(mag*10));
}
uint16_t getvoltage() {
static uint8_t index=0;
rawVoltages[index++]=readvoltage(10);
index%=5;
return medianp(rawVoltages);
}
int16_t readtemp(uint8_t sensor) {
uint8_t data[9];
static uint8_t index=0;
static uint32_t temptimer=0;
static uint8_t currsensor=0;
#ifdef SIMTEMP
return SIMTEMP;
#endif
if(sensors.useDS) {
if(!sensors.temphwa[sensor][0]) // No hardware address for this sensor slot?
return tempdata[sensor].fallback; // Return a "safe" but indicative value.
} else {
analogRead(TEMP);
thermists[index++]=analogRead(TEMP); // Always read the thermistor value and put it in the median array.
index%=5; // The array can only hold 5 values
}
if(seconds()-temptimer<1) // Do we need a new temperature now?
return tempdata[sensor].temperature; // No, use the old one
temptimer=seconds();
if(sensors.useDS) {
if(sensors.temphwa[currsensor][0]) {
ds.reset();
ds.select(sensors.temphwa[currsensor]); // Select the current sensor
ds.write(0xBE); // Issue Read scratchpad command
for(byte i = 0; i < 9; i++) // Receive 9 bytes
data[i] = ds.read();
if(OneWire::crc8(data,8)==data[8]) { // Check what we've got. Is the data valid?
int16_t t = (int16_t)((data[1] << 8) + data[0] ) * 10 / 16;
if(t>1100 || t<-350)
tempdata[currsensor].failures++;
else {
tempdata[currsensor].temperature = t;
tempdata[currsensor].failures=0;
}
} else {
tempdata[currsensor].failures++;
}
}
// Setup the sensors for measuring the temperature, so we have new data ready at the next reading attempt
ds.reset();
currsensor=1-currsensor;
ds.select(sensors.temphwa[currsensor]);
ds.write(0x44,1); // Temperature conversion
} else {
float ratio = ((float)medianp(thermists))/1024.0;
float R = TEMPR * ratio / (1.0-ratio); // Calculate thermistor resistor value
tempdata[0].temperature = modMovAvg(&tempdata[0].mmaTemp,(int16_t)round(10.0* ((1.0 / ((log(R/sensors.thermistor) / sensors.thermbeta) + (1.0 / 298.15)))-273.15)),TEMPSAMPLES);
}
if(tempdata[sensor].failures>10)
// Too many failures, start returning a "safe" but unusual value.
tempdata[sensor].temperature = tempdata[sensor].fallback;
return tempdata[sensor].temperature;
}
uint8_t searchSensors() {
uint8_t owaddr[8];
uint8_t sensor=0;
sensors.temphwa[0][0]=0;
sensors.temphwa[1][0]=0;
ds.reset();
while(ds.search(owaddr) && sensor<2) { // We have found a one-wire device. Is it a temperature sensor and is the data valid?
if(owaddr[0]==0x28 && OneWire::crc8( owaddr, 7) == owaddr[7]) {
// Seems to be ok. So copy this hardware address
for(uint8_t i=0;i<8;i++)
sensors.temphwa[sensor][i]=owaddr[i];
// Now we have it, configure the sensor
ds.reset();
ds.select(sensors.temphwa[sensor]);
ds.write(0x4E); // Write scratchpad command
ds.write(0); // TL data
ds.write(0); // TH data
ds.write(0x7F); // Configuration Register (resolution) 7F=12bits 5F=11bits 3F=10bits 1F=9bits
ds.reset(); // This "reset" sequence is mandatory, it allows the DS18B20 to understand the copy scratchpad to EEPROM command
ds.select(sensors.temphwa[sensor]);
ds.write(0x48); // Copy Scratchpad command
sensor++;
ds.reset();
}
}
return sensor;
}
int16_t getPgmVal(const int16_t *ptr) {
return pgm_read_word(ptr);
}
uint16_t getPgmVal(const uint16_t *ptr) {
return pgm_read_word(ptr);
}
char getPgmVal(const char *ptr) {
return pgm_read_byte(ptr);
}
byte getPgmVal(const byte *ptr) {
return pgm_read_byte(ptr);
}
// Prints a string from flash memory
const char* pgmLcdPrint(const char *txt) {
char c=pgm_read_byte_near(txt);
while(c) {
lcd.write(c);
txt++;
c=pgm_read_byte_near(txt);
}
txt++;
return txt;
}
void printTemperature(int16_t value,uint8_t precision) {
if(rom.fahrenheit) {
lcd.print(((float)value*0.18)+32.0,precision);
lcd.print("\xDF""F");
if(precision && (float)value*0.18+32.0<100)
lcd.write(' ');
} else {
lcd.print((float)value/10.0,precision);
lcd.print("\xDF""C");
if(precision) {
if(value<100 && value>0)
lcd.write(' ');
if(value>-100)
lcd.write(' ');
}
}
}
uint32_t seconds() {
static uint32_t prevmillis=0;
uint32_t dsecs=(millis()-prevmillis)/1000L;
if(dsecs) {
secs+=dsecs;
prevmillis+=dsecs*1000L;
}
return secs;
}
// This function handles the button functionality. Button presses are separated into long, short and double clicks presses.
int chkbutton(struct button_t* button) {
if(digitalRead(button->button)==LOW) { // button pressed
if(button->pressed==0)
button->pressed=millis();
if(millis()-button->pressed>=LONGPRESS) {
if(!button->longpressed) {
button->longpressed=true;
return 2;
}
}
return 0;
} else { // button released
if(button->longpressed) {
button->longpressed=false;
button->pressed=0;
}
if(button->pressed==0)
return 0;
unsigned long pressed=millis()-button->pressed;
button->pressed=0;
#ifdef SHORTPRESS
if(pressed<SHORTPRESS)
return 0;
#endif
if(pressed<LONGPRESS) {
return 1;
}
return 0;
}
}
// ******** Modified Moving Average filter *********
int16_t modMovAvg(mma_t* mma, const int16_t val, const uint16_t maxSamples) {
if(mma->samples<maxSamples)
mma->samples++; // This is going to be an additional sample
else
mma->sum-=mma->avg; // Max amount of samples reached, substract average to make room for new value
mma->sum+=val;
mma->avg=(int16_t)((mma->sum+(mma->samples>>1))/mma->samples);
return mma->avg;
}
// ******** Routine to find the median in an array of 5 values *********
// We use this to reject voltage spikes and erroneous temperature readings, rather than averaging them.
// Trick using XOR to swap two variables
#define swap(a,b) a ^= b; b ^= a; a ^= b;
#define sort(a,b) if(a>b){ swap(a,b); }
uint16_t median(uint16_t a, uint16_t b, uint16_t c, uint16_t d, uint16_t e) {
sort(a, b);
sort(d, e);
sort(a, c);
sort(b, c);
sort(a, d);
sort(c, d);
sort(b, e);
sort(b, c);
return c;
}
uint16_t medianp(uint16_t values[5]) {
return median(values[0], values[1], values[2], values[3], values[4]);
}
#if HWMODEL != ARDUINO_
// ********* These routine are necessary to represent the serial number in Base64 ************
// In the commercial version we like to keep track of serial numbers.
// In the Arduino versions we don't need that.
// Routines to encode to base64
unsigned char binary_to_base64(unsigned char v) {
if(v < 26) return v + 'A'; // Capital letters - 'A' is ascii 65 and base64 0
if(v < 52) return v + 71; // Lowercase letters - 'a' is ascii 97 and base64 26
if(v < 62) return v - 4; // Digits - '0' is ascii 48 and base64 52
if(v == 62) return '+'; // Special case for value 62
if(v == 63) return '/'; // Special case for value 63
return 64;
}
void a3_to_a4(uint8_t *a4, uint8_t *a3) {
a4[0] = (a3[0] & 0xfc) >> 2;
a4[1] = ((a3[0] & 0x03) << 4) + ((a3[1] & 0xf0) >> 4);
a4[2] = ((a3[1] & 0x0f) << 2) + ((a3[2] & 0xc0) >> 6);
a4[3] = (a3[2] & 0x3f);
}
int8_t base64_encode(char *output, char *input, int8_t inputLen) {
int8_t i = 0, j = 0;
int8_t encLen = 0;
uint8_t a3[3];
uint8_t a4[4];
while(inputLen--) {
a3[i++] = *(input++);
if(i == 3) {
a3_to_a4(a4, a3);
for(i = 0; i < 4; i++)
output[encLen++] = binary_to_base64(a4[i]);
i = 0;
}
}
if(i) {
for(j = i; j < 3; j++)
a3[j] = '\0';
a3_to_a4(a4, a3);
for(j = 0; j < i + 1; j++)
output[encLen++] = binary_to_base64(a4[j]);
while((i++ < 3))
output[encLen++] = '=';
}
output[encLen] = '\0';
return encLen;
}
#endif