sketchy slow test with LEDs
[cortex-from-scratch] / rtc.c
1 /* (CC-BY-NC-SA) ROBIN KRENS - ROBIN @ ROBINKRENS.NL
2  * 
3  * $LOG$
4  * 2019/8/14 - ROBIN KRENS      
5  * Initial version 
6  * 
7  * $DESCRIPTION$
8  * Real Time Clock configuration. A Real Time Clock is configured to use
9  * the more accurate external chrystal (LSE) and ticks at exactly
10  * one second. (Can be further calibrated using a tamper pin)
11  * Each clock tick triggers an interrupt and invokes the real time clock
12  * service routine.
13  *
14  * Note: this is not the Cortex M3/M4 SysTick!
15  * 
16  * */
17
18 #include <stdbool.h>
19 #include <stddef.h>
20 #include <stdint.h>
21
22 #include <sys/robsys.h>
23 #include <sys/mmap.h>
24
25 #include <lib/regfunc.h>
26 #include <lib/tinyprintf.h>
27
28 #include <drivers/led.h>
29 #include <drivers/tm1637.h>
30
31
32 /* This handler is invoked each clock tick. I've included two examples 
33  * one is writing the clock count to a 4 grid led display. Another is a simple LED blink */
34 void * rtc_handler() {
35
36         // Real time clock output on a LED display 
37         /* int cntvalue = *RTC_CNTL;
38         if (cntvalue < 9999) {
39
40         unsigned char dn[10] = {0xFC, 0x60, 0xDA, 0xF2, 0x66, 0xB6, 0xBE, 0xE0, 0xFE, 0xF6};
41                 
42         int grid0, grid1, grid2, grid3;
43
44         grid0 = cntvalue % 10;
45         grid1 = ((cntvalue % 100) - grid0) / 10;
46         grid2 = ((cntvalue % 1000) - grid0 - grid1) / 100;
47         grid3 = ((cntvalue % 10000) - grid0 - grid1 - grid2) / 1000;
48
49         //printf("%d, %d, %d, %d\n", grid0, grid1, grid2, grid3);
50         
51         char current[4] = { dn[grid3], dn[grid2], dn[grid1], dn[grid0] }; 
52
53         for (int i = 0; i < 4; i++) {
54                 set_grid(i, current[i], false);
55         }
56
57         set_display(true, 0); 
58         } */
59
60         // Simple LED blink 
61         uint32_t curr = *RTC_CNTL;
62         int even = *RTC_CNTL % 2;
63         (!even) ? led_off() : led_on();
64
65         rclrbit(RTC_CRL, 0); /* clear interrupt flag */
66 }
67
68 /* Enable a periodic interrupt. At each clock count a interrupt is triggered
69  * and consequently the handler is invoked. */
70 static void periodic_intr() {
71
72         while(!rchkbit(RTC_CRL, 5)); // Check last write is terminated
73         rsetbit(RTC_CRL, 4); // start configure
74         rsetbit(RTC_CRH, 0); // enable periodic (second) interrupt
75         
76         while(!rchkbit(RTC_CRL, 5)); 
77         rwrite(RTC_PRLL, 0x7FFF); // 0x7FFF equals 1 second clock configuration 
78         rclrbit(RTC_CRL, 4); // stop configure
79         while(!rchkbit(RTC_CRL, 5)); // Check last write is terminated
80         rsetbit(NVIC_ISER0, 3); // enable in register vector
81 }
82
83 /* Setup the Real time clock to work with the Low speed external 
84  * chrystal (32.768 kHz) */
85 static void setup_rtc() {
86
87         /* Enable PWREN and BKPEN */
88         rsetbit(RCC_APB1ENR, 28);
89         rsetbit(RCC_APB1ENR, 27);
90
91         /* Enable access to backup registers and RTC */
92         rsetbit(PWR_CR, 8);
93
94         rsetbit(RCC_BDCR, 0); /* LSE enable */
95         while(!rchkbit(RCC_BDCR, 1)); /* wait for LSE to come up */
96         
97         rsetbitsfrom(RCC_BDCR, 8, 0x1); /* use LSE as RTC source */
98         rsetbit(RCC_BDCR, 15); /* enable RTC */
99         
100         ivt_set_gate(19, rtc_handler, 0); /* setup interrupt handler */
101
102         //calibrate_rtc(); // uncomment if need, needs TAMPER PIN
103         
104         periodic_intr();// setup periodic interrupt
105
106 }
107
108 /* Function to calibrate the RTC even more accurate. 
109  * Output to a so-called tamper pin. Uncomment if needed */
110 static void calibrate_rtc() {
111                 /* rsetbit(BKP_RTCCR, 7); // enable CC0, 
112         while(!rchkbit(RTC_CRL, 5)); // Check last write is terminated
113         rsetbit(RTC_CRL, 4);
114         // Set up and check tamper pin
115         rclrbit(RTC_CRL, 4);
116         while(!rchkbit(RTC_CRL, 5)); // Check last write is terminated */
117 }
118
119 void rtc_init() {
120
121 #ifdef ENABLE_RTC
122         setup_rtc();
123 #endif
124
125 }