ESP32-PaxCounter/src/if482.cpp

199 lines
6.1 KiB
C++
Raw Normal View History

2019-01-29 09:04:31 +01:00
#if defined HAS_IF482 && defined RTC_INT
2019-01-28 21:47:44 +01:00
/* NOTE:
The IF482 Generator needs an high precise 1 Hz clock signal which cannot be
acquired in suitable precision on the ESP32 SoC itself. Additional clocking
hardware is required, ususally the clock signal is generated by external RTC or
2019-01-29 09:04:31 +01:00
GPS which can generate a precise time pulse signal (+/- 2ppm).
In this example code we use a Maxim DS3231 RTC chip, and configure the chips's
interrupt output pin as clock. The clock signal triggers an interrupt on the
ESP32, which controls the realtime output of IF482 telegram. This is why code in
2019-01-28 21:47:44 +01:00
IF482.cpp depends on code in RTCTIME.cpp.
*/
2019-01-28 21:47:44 +01:00
///////////////////////////////////////////////////////////////////////////////
/*
2019-01-26 12:32:58 +01:00
IF482 Generator to control clocks with IF482 telegram input (e.g. BÜRK BU190)
2019-01-29 09:04:31 +01:00
2019-01-26 12:32:58 +01:00
Example IF482 telegram: "OAL160806F170400"
IF482 Specification:
2019-01-26 12:32:58 +01:00
http://www.mobatime.com/fileadmin/user_upload/downloads/TE-112023.pdf
The IF 482 telegram is a time telegram, which sends the time and date
information as ASCII characters through the serial interface RS 232 or RS 422.
Communication parameters:
Baud rate: 9600 Bit/s
Data bits 7
Parity: even
Stop bit: 1
Jitter: < 50ms
Interface : RS232 or RS422
Synchronization: Telegram ends at the beginning of the second
specified in the telegram
Cycle: 1 second
Format of ASCII telegram string:
Byte Meaning ASCII Hex
1 Start of telegram O 4F
2 Monitoring* A 41
3 Time-Season** W/S/U/L 57 or 53
4 Year tens 0 .. 9 30 .. 39
5 Year unit 0 .. 9 30 .. 39
6 Month tens 0 or 1 30 or 31
7 Month unit 0 .. 9 30 .. 39
8 Day tens 0 .. 3 30 .. 33
9 Day unit 0 .. 9 30 .. 39
10 Day of week*** 1 .. 7 31 .. 37
11 Hours tens 0 .. 2 30 .. 32
12 Hours unit 0 .. 9 30 .. 39
13 Minutes tens 0 .. 5 30 .. 35
14 Minutes unit 0 .. 9 30 .. 39
15 Seconds tens 0 .. 5 30 .. 35
16 Seconds unit 0 .. 9 30 .. 39
17 End of telegram CR 0D
*) Monitoring:
With a correctly received time in the sender unit, the ASCII character 'A' is
issued. If 'M' is issued, this indicates that the sender was unable to receive
any time signal for over 12 hours (time is accepted with A and M).
**) Season:
W: Standard time,
S: Season time,
U: UTC time (not supported by all systems),
L: Local Time
***) Day of week:
not evaluated by model BU-190
*/
2019-01-28 21:47:44 +01:00
///////////////////////////////////////////////////////////////////////////////
#include "if482.h"
// Local logging tag
static const char TAG[] = "main";
TaskHandle_t IF482Task;
HardwareSerial IF482(2); // use UART #2 (note: #1 may be in use for serial GPS)
// initialize and configure GPS
int if482_init(void) {
2019-01-26 12:32:58 +01:00
// open serial interface
IF482.begin(HAS_IF482);
2019-01-28 21:47:44 +01:00
// use external rtc 1Hz clock for triggering IF482 telegram
2019-02-02 10:35:20 +01:00
if (I2C_MUTEX_LOCK()) {
Rtc.SetSquareWavePinClockFrequency(DS3231SquareWaveClock_1Hz);
Rtc.SetSquareWavePin(DS3231SquareWavePin_ModeClock);
I2C_MUTEX_UNLOCK();
} else {
ESP_LOGE(TAG, "I2c bus busy - IF482 initialization error");
return 0;
}
2019-02-04 20:02:30 +01:00
xTaskCreatePinnedToCore(if482_loop, // task function
"if482loop", // name of task
2048, // stack size of task
(void *)1, // parameter of the task
3, // priority of the task
&IF482Task, // task handle
0); // CPU core
assert(IF482Task); // has if482loop task started?
// setup external interupt for active low RTC INT pin
2019-01-26 18:49:53 +01:00
pinMode(RTC_INT, INPUT_PULLUP);
2019-02-04 20:02:30 +01:00
attachInterrupt(digitalPinToInterrupt(RTC_INT), IF482IRQ, FALLING);
return 1;
} // if482_init
2019-02-02 09:15:31 +01:00
String if482Telegram(time_t tt) {
time_t t = myTZ.toLocal(tt);
2019-01-26 18:49:53 +01:00
char mon;
char buf[14] = "000000F000000";
char out[17];
switch (timeStatus()) { // indicates if time has been set and recently synced
case timeSet: // time is set and is synced
2019-01-26 18:49:53 +01:00
mon = 'A';
break;
case timeNeedsSync: // time had been set but sync attempt did not succeed
2019-01-26 18:49:53 +01:00
mon = 'M';
break;
default: // time not set, no valid time
2019-01-26 18:49:53 +01:00
mon = '?';
break;
} // switch
2019-02-02 09:15:31 +01:00
if ((timeStatus() == timeSet) ||
(timeStatus() == timeNeedsSync)) // do we have valid time?
snprintf(buf, sizeof buf, "%02u%02u%02u%1u%02u%02u%02u", year(t) - 2000,
month(t), day(t), weekday(t), hour(t), minute(t), second(t));
2019-01-26 18:49:53 +01:00
snprintf(out, sizeof out, "O%cL%s\r", mon, buf);
2019-02-04 20:02:30 +01:00
ESP_LOGD(TAG, "IF482 = %s", out);
return out;
}
void if482_loop(void *pvParameters) {
configASSERT(((uint32_t)pvParameters) == 1); // FreeRTOS check
TickType_t wakeTime;
time_t t, tt;
2019-02-02 09:15:31 +01:00
const TickType_t timeOffset =
pdMS_TO_TICKS(IF482_OFFSET); // duration of telegram transmit
const TickType_t startTime = xTaskGetTickCount(); // now
2019-02-02 09:15:31 +01:00
// wait until begin of a new second
t = tt = now();
do {
tt = now();
} while (t == tt);
2019-02-04 20:02:30 +01:00
BitsPending = true; // start blink in display
2019-02-02 09:15:31 +01:00
// take timestamp at moment of start of new second
const TickType_t shotTime = xTaskGetTickCount() - startTime - timeOffset;
// task remains in blocked state until it is notified by isr
for (;;) {
xTaskNotifyWait(
0x00, // don't clear any bits on entry
ULONG_MAX, // clear all bits on exit
&wakeTime, // receives moment of call from isr
portMAX_DELAY); // wait forever (missing error handling here...)
2019-02-02 09:15:31 +01:00
// now we're synced to start of second tt and wait
// until it's time to start transmit telegram for tt+1
vTaskDelayUntil(&wakeTime, shotTime); // sets waketime to moment of shot
IF482.print(if482Telegram(now() + 1));
}
2019-02-04 20:02:30 +01:00
BitsPending = false; // stop blink in display
vTaskDelete(IF482Task); // shoud never be reached
} // if482_loop()
// interrupt service routine triggered by RTC 1Hz precise clock
void IRAM_ATTR IF482IRQ() {
xTaskNotifyFromISR(IF482Task, xTaskGetTickCountFromISR(), eSetBits, NULL);
portYIELD_FROM_ISR();
}
#endif // HAS_IF482