-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.cpp
383 lines (343 loc) · 13.3 KB
/
main.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
//////////////////////////////////////////////////////////////////////////////
/// @file main.cpp
/// @author Kai R.
/// @brief Clock with DCF77 receiver and DS3231 RTC module.
/// The RTC module is synchronized using the DCF77 receiver.
///
/// Used PINS:
/// Display
/// Pin D13: SCK/SCLK = CLK (Pin 29)
/// Pin D10: ChipSelect (CS/SS) = CSB (Pin 38)
/// Pin D11: MOSI = SI (Pin 28)
/// Pin D12: MISO - not used
///
/// Pin 07: RS = Read/Write Data from/to RAM (Pin 39)
/// Pin 08: RST = Reset des Displays (Pin 40)
/// Pin 09: Brightness
///
/// I2C - RTC
/// Pin 21: (A4) I2C SDA
/// Pin 22: (A5) I2C SCL
///
/// Other Control Pins
/// Pin 02: Interrupt Pin 0 = Processing of dcf77 signal.
/// Pin 03: Interrupt Pin 1 = evaluate the 1Hz signal of the RTC.
/// Pin 04: Button for switching the backlight
/// Pin 05: Button to switch on the date
/// PIN 06 if not ATtiny88
/// else PIN 14: Switch DCF77 Receiver on or off
/// Pin 06: Reserved for debug output (Serial.print - Only Attiny88)
///
///
/// @date 2022-05-20
/// @version 1.1
///
/// @date 2022-06-03
/// Refactornig, added animated dots to the time display.
///
/// @date 2022-06-04
/// When debugging is turned on on the serial console, certain code parts are now turned off
/// during compilation so that enough memory is available for the debugging output.
/// (define DEBUG_ENABLED)
///
/// @date 2022-10-08
/// Button status designations changed
///
/// @copyright Copyright (c) 2022
///
//////////////////////////////////////////////////////////////////////////////
#include <Arduino.h>
#include <digitalWriteFast.h>
#include <avr/wdt.h>
#include <avr/power.h>
#include <Button_SL.hpp>
#include "bcdconv.hpp"
#include "dcf77.hpp"
#include "display.hpp"
#include "DS3231Wire.h"
//////////////////////////////////////////////////
// Definitions
//////////////////////////////////////////////////
#if !(defined(__AVR_ATtinyX8__) || defined(__AVR_ATmega328P__) || defined(__AVR_ATmega328PB__) || \
defined(__AVR_ATmega88P__) || defined(__AVR_ATmega8__))
#error Software not suitable for the microcontroller
#endif
// PlatformIO: Set it in platform.ini (mybuild_flags)
// If the code is to be used for a developer board (Uno/Nano/Micro) then uncomment DEV_BOARD.
// On the development boards with USB connection, pin6 is used instead of pin14 for switching the DCF77 module on and
// off.
// #define DEV_BOARD
// Uncomment for debug output on the serial console or to switch on I2C/Wire Fast Mode
// #define WIRE_FAST_MODE
// #define PRINT_TIME_SERIAL
// #define DEBUG_DCF77CONTROL
// #define DEBUG_ISR
// #define DEBUG_INT1
// #define DEBUG_DCF77_SEQ_ADD_CHECK
// #define DEBUG_DCF77_SEQ
// #define SET_TEST_TIME
// If DEBUG_ENABLED is defined, then certain parts of the program are disabled ( in loop() ) to
// free up space for the debug output on the serial console.
#if defined(PRINT_TIME_SERIAL) || defined(DEBUG_DCF77CONTROL) || defined(DEBUG_ISR) || defined(DEBUG_INT1) || \
defined(DEBUG_DCF77_SEQ_ADD_CHECK) || defined(DEBUG_DCF77_SEQ)
#define DEBUG_ENABLED
#endif
#ifdef WIRE_FAST_MODE
constexpr uint32_t WIRE_SPEED{400000}; // I2C Fast Mode
#else
constexpr uint32_t WIRE_SPEED{100000}; // I2C Normal Mode
#endif
//////////////////////////////////////////////////
// Global constants and variables / objects
//////////////////////////////////////////////////
#if defined(DEV_BOARD)
constexpr uint8_t DCF77_ON_OFF_PIN{6}; // Switch DCF77 Receiver on or off
#else
constexpr uint8_t DCF77_ON_OFF_PIN{14}; // Switch DCF77 Receiver on or off
#endif
constexpr uint32_t DCF77_SLEEP{28790}; // Period (in seconds) for which the radio clock is switched off.
// Here 28790 Seconds.
// int1_second is just a counter that increases every second.
// It is not necessarily in sync with the RTC seconds
volatile uint8_t int1_second{0}; // Second Tick in loop(), set in INT1
DCF77Clock dcf77;
ClockData clockData;
dogm_7036 lcd;
#ifndef DEBUG_ENABLED
Btn::ButtonSL dtButton(BUTTON_DT_PIN);
Btn::ButtonSL blButton(BUTTON_BL_PIN);
#endif
//////////////////////////////////////////////////
// Function forward declaration
//////////////////////////////////////////////////
void optimizePowerConsumption(void);
bool rtcNeedsSync(void);
void check1HzSig(void);
//////////////////////////////////////////////////////////////////////////////
/// @brief Initialize the program.
///
//////////////////////////////////////////////////////////////////////////////
void setup() {
optimizePowerConsumption();
// With ATtiny controllers, the serial output is via the Serial -TX pin. An FTD232 adapter is required.
#ifdef DEBUG_ENABLED
Serial.begin(9600);
#ifndef ESP8266
while (!Serial)
; // wait for serial port to connect. Needed for native USB
#endif
#endif
pinModeFast(DCF77_ON_OFF_PIN, OUTPUT);
digitalWriteFast(DCF77_ON_OFF_PIN, LOW); // Switch DCF77 receiver on (P-Channel MOSFet as switch)
#ifndef DEBUG_ENABLED
dtButton.begin();
blButton.begin();
#endif
// init DOGM-LCD
initDisplay(lcd);
// init DCF77
dcf77.begin();
dcf77.setActiveLow(true); // ELV DCF77 Modul works with active low signals
// Init RTC
Wire.begin();
Wire.setClock(WIRE_SPEED);
DS3231::disable32kHz();
DS3231::enableSw1Hz();
attachInterrupt(digitalPinToInterrupt(PIND3), check1HzSig, RISING);
#ifdef SET_TEST_TIME
DS3231::setDateTime(BCDConv::decToBcd(0), BCDConv::decToBcd(1), BCDConv::decToBcd(1), BCDConv::decToBcd(17),
BCDConv::decToBcd(1),
BCDConv::decToBcd(15)); // Reset RTC for testing purposes
#endif
}
//////////////////////////////////////////////////////////////////////////////
/// @brief Time display and time synchronization.
///
//////////////////////////////////////////////////////////////////////////////
void loop() {
static uint8_t tickSecond{LEAP_SECOND + 1}; // 61 to prevent initial tickSecond = int1_second.
static bool showDate{false};
static bool dcf77PoweredOn{true};
#ifndef DEBUG_ENABLED
static uint8_t dateVisibleOffTime{0};
static uint32_t dcf77SleepCounter{0};
#endif
if (dcf77PoweredOn) {
if (!rtcNeedsSync()) { // If returns 0 (false) both clocks are synchronous.
#ifndef DEBUG_ENABLED
// If both clocks synchronous switch dcf77 clock off for the DCF77_SLEEP time.
digitalWriteFast(DCF77_ON_OFF_PIN, HIGH);
dcf77PoweredOn = false;
#endif
clockData.clockSeparator().setTimeSeparator(Separators::COLUP, 0);
}
}
#ifndef DEBUG_ENABLED
if (dtButton.tick() != Btn::ButtonState::notPressed) {
showDate = true;
printRtcTime(lcd, clockData,
showDate); // Don't wait until the next second after the button is pressed to show the date.
}
switchBacklight(int1_second, blButton.tick()); // Switch backlight on if button has been pressed.
#endif
// Do the following every second.
// To limit the read operations to the RTC (read the second), use the count variable of INT1.
// Using millis() is too imprecise.
// The clock comes from the 1Hz signal of the RTC which is present at the INT1 pin.
if (int1_second != tickSecond) {
tickSecond = int1_second;
#ifndef DEBUG_ENABLED
if (!dcf77PoweredOn) {
++dcf77SleepCounter;
if (dcf77SleepCounter == DCF77_SLEEP) {
digitalWriteFast(DCF77_ON_OFF_PIN, LOW); // Switch DCFAvtive-Pin - Clock ON
dcf77PoweredOn = true;
dcf77SleepCounter = 0;
clockData.clockSeparator().setTimeSeparator(Separators::SPACE, 0);
}
}
if (showDate) {
switch (dateVisibleOffTime < SHOW_DATE_DURATION) {
case true: ++dateVisibleOffTime; break;
case false:
showDate = false; // showDate becomes false when the display time for the date has passed.
dateVisibleOffTime = 0;
}
}
#endif
printRtcTime(lcd, clockData, showDate);
}
}
//////////////////////////////////////////////////////////////////////////////
/// @brief Control the synchronization between the two clocks
///
/// @return true There is a time difference between the dcf77 and the RTC time.
/// @return false There is no time difference. Both clocks are synchronous.
//////////////////////////////////////////////////////////////////////////////
bool rtcNeedsSync() {
decltype(rtcNeedsSync()) rtcSetTime{true};
//
// If the sequenceflag != MAX_SECOND then the sequence was not received correctly,
// unless it is a leap second sequence.
// In this case, the second counter must not be unequal to MAX_SECONDS + 1.
//
DCF77Sequence state = dcf77.getSequenceFlag();
if (state == MAX_SECONDS || (state == LEAP_SECOND && dcf77.getLeapSecond())) {
switch (dcf77.decodeSequence()) {
case false:
break; // if the parity of the data sequence is not OK do nothing here.
// Compare the DCF77 time with the RTC time. The RTC will only be set if there is a time difference
// Because only every full minute is checked, the dcf77 seconds are always 0.
case true:
uint8_t dcf77Compare = dcf77.getMinutes() + dcf77.getHours();
uint8_t rtcCompare = BCDConv::bcdToDec(DS3231::readRegister(DS3231::SECONDS)) +
BCDConv::bcdToDec(DS3231::readRegister(DS3231::MINUTES)) +
BCDConv::bcdToDec(DS3231::readRegister(DS3231::HOURS));
uint8_t timeCompareDiff = rtcCompare - dcf77Compare;
#ifdef DEBUG_DCF77CONTROL
Serial.println("");
Serial.print(F("int1_second : "));
Serial.println(int1_second);
Serial.print(F("RTC Compare Value: "));
Serial.println(rtcCompare);
Serial.print(F("DCF77 Compare Value: "));
Serial.println(dcf77Compare);
Serial.print(F("Diff. RTC to DCF77 : "));
Serial.println(timeCompareDiff);
Serial.print(F("Parity : "));
Serial.println(parityOK);
#endif
// When timeCompareDiff is non-zero and not 59 at an hour change
// there is a time difference -> set RTC Clock.
switch (timeCompareDiff && timeCompareDiff != HOUR_CHANGE) {
case true:
DS3231::setDateTime(dcf77.getBcdYear(), dcf77.getBcdMonth(), dcf77.getBcdDay(), dcf77.getBcdHours(),
dcf77.getBcdMinutes(),
1 // Second.
);
// no break!
case false: rtcSetTime = false; break;
} // Switch2
break;
} // Switch1
}
return rtcSetTime;
}
//////////////////////////////////////////////////////////////////////////////
/// @brief Count the seconds using the 1Hz signal from the RTC.
///
//////////////////////////////////////////////////////////////////////////////
void check1HzSig() {
int1_second = (int1_second + 1) % 60;
#ifdef DEBUG_INT1
Serial.println(int1_second);
#endif
}
//////////////////////////////////////////////////////////////////////////////
/// @brief Disable unused peripherals and set unused Pins to input with internal
/// pullups
///
//////////////////////////////////////////////////////////////////////////////
void optimizePowerConsumption(void) {
// Disable digital input buffer on ADC pins
DIDR0 = bit(ADC5D) | bit(ADC4D) | bit(ADC3D) | bit(ADC2D) | bit(ADC1D) | bit(ADC0D);
DIDR0 |= 0xC0; // ADC7D and ADC6D are undefined in header file so set bits this way
// Disable digital input buffer on Analog comparator pins
DIDR1 |= bit(AIN1D) | bit(AIN0D);
// Disable Analog Comparator
ACSR |= bit(ACD);
// Power shutdown to unused peripherals (Only Atmega328PB / ATTINY88)
#if defined(__AVR_ATtinyX8__) || defined(__AVR_ATmega328P__)
power_adc_disable();
#endif
// Unused and unconnected pins consume power. The un-needed power load of floating pins
// can be avoided by using the AVR's internal pull-up resistors on the unused pins.
// Port pins can be set to input pull-ups by setting the DDxn bit to 0 and PORTxn bit
// to 1. (where x is PORT B,C,D,E and n is 0 to 7)
// Unused pins set as input pull up
// Pins Used
// Pin 02 PD2 Used PD -> 1011 1100
// Pin 03 PD3 -> 1111 1100 If NOT ATtiny88
// Pin 04 PD4
// Pin 05 PD5
// Pin 06 PD6 only if NOT ATtiny88
// Pin 07 PD7
//
// Pin 08 PB0 Used PB -> 0010 1111
// Pin 09 PB1 0110 1111 If ATtiny88
// Pin 10 PB2
// Pin 11 PB3
// Pin 13 PB5
// PIN 14 PB6 only if ATtiny88
//
// Pin A4 PC4 Used PC -> 0011 0000
// Pin A5 PC5
#if defined(DEV_BOARD)
#define PORTSB 0x2F
#else
#define PORTSB 0x6F
#endif
#define PORTSC 0x30
#if defined(DEV_BOARD)
#define PORTSD 0xFC
#else
#define PORTSD 0xBC
#endif
DDRB &= (PORTSB);
DDRC &= (PORTSC);
DDRD &= (PORTSD);
PORTB |= ~(PORTSB);
PORTC |= ~(PORTSC);
PORTD |= ~(PORTSD);
cli(); // Disable interrupts
// turn off brown-out enable in software
MCUCR = bit(BODS) | bit(BODSE); // turn on brown-out enable select
MCUCR = bit(BODS); // this must be done within 4 clock cycles of above
// Watchdog Timer OFF
wdt_reset(); // Reset watchdog timer
MCUSR &= ~_BV(WDRF); // Clear WDRF in MCUSR
WDTCSR = 0x00; // Turn off WDT
CLKPR |= 0x80; // Prescaler change enable to 1Mhz
CLKPR = bit(CLKPS1) | bit(CLKPS0); // set prescaler to 8
sei();
}