main.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411
  1. /*! \mainpage SIR firmware documentation
  2. *
  3. * \section intro Introduction
  4. * A collection of HTML-files has been generated using the documentation in the sourcefiles to
  5. * allow the developer to browse through the technical documentation of this project.
  6. * \par
  7. * \note these HTML files are automatically generated (using DoxyGen) and all modifications in the
  8. * documentation should be done via the sourcefiles.
  9. */
  10. /*! \file
  11. * COPYRIGHT (C) SaltyRadio 2016
  12. * \date 20-02-2016
  13. */
  14. #define LOG_MODULE LOG_MAIN_MODULE
  15. /*--------------------------------------------------------------------------*/
  16. /* Include files */
  17. /*--------------------------------------------------------------------------*/
  18. #include <stdio.h>
  19. #include <string.h>
  20. #include <sys/thread.h>
  21. #include <sys/timer.h>
  22. #include <sys/version.h>
  23. #include <dev/irqreg.h>
  24. #include "displayHandler.h"
  25. #include "system.h"
  26. #include "portio.h"
  27. #include "display.h"
  28. #include "remcon.h"
  29. #include "keyboard.h"
  30. #include "led.h"
  31. #include "log.h"
  32. #include "uart0driver.h"
  33. #include "mmc.h"
  34. #include "watchdog.h"
  35. #include "flash.h"
  36. #include "spidrv.h"
  37. #include "network.h"
  38. #include "typedefs.h"
  39. #include <time.h>
  40. #include "rtc.h"
  41. #include "alarm.h"
  42. #include "ntp.h"
  43. #include "httpstream.h"
  44. #include "contentparser.h"
  45. /*-------------------------------------------------------------------------*/
  46. /* local routines (prototyping) */
  47. /*-------------------------------------------------------------------------*/
  48. static void SysMainBeatInterrupt(void*);
  49. static void SysControlMainBeat(u_char);
  50. /*-------------------------------------------------------------------------*/
  51. /* Stack check variables placed in .noinit section */
  52. /*-------------------------------------------------------------------------*/
  53. /*!
  54. * \addtogroup System
  55. */
  56. /*@{*/
  57. /*-------------------------------------------------------------------------*/
  58. /* start of code */
  59. /*-------------------------------------------------------------------------*/
  60. /*!
  61. * \brief ISR MainBeat Timer Interrupt (Timer 2 for Mega128, Timer 0 for Mega256).
  62. *
  63. * This routine is automatically called during system
  64. * initialization.
  65. *
  66. * resolution of this Timer ISR is 4,448 msecs
  67. *
  68. * \param *p not used (might be used to pass parms from the ISR)
  69. */
  70. static void SysMainBeatInterrupt(void *p)
  71. {
  72. /*
  73. * scan for valid keys AND check if a MMCard is inserted or removed
  74. */
  75. KbScan();
  76. CardCheckCard();
  77. }
  78. /*!
  79. * \brief Initialise Digital IO
  80. * init inputs to '0', outputs to '1' (DDRxn='0' or '1')
  81. *
  82. * Pull-ups are enabled when the pin is set to input (DDRxn='0') and then a '1'
  83. * is written to the pin (PORTxn='1')
  84. */
  85. void SysInitIO(void)
  86. {
  87. /*
  88. * Port B: VS1011, MMC CS/WP, SPI
  89. * output: all, except b3 (SPI Master In)
  90. * input: SPI Master In
  91. * pull-up: none
  92. */
  93. outp(0xF7, DDRB);
  94. /*
  95. * Port C: Address bus
  96. */
  97. /*
  98. * Port D: LCD_data, Keypad Col 2 & Col 3, SDA & SCL (TWI)
  99. * output: Keyboard colums 2 & 3
  100. * input: LCD_data, SDA, SCL (TWI)
  101. * pull-up: LCD_data, SDA & SCL
  102. */
  103. outp(0x0C, DDRD);
  104. outp((inp(PORTD) & 0x0C) | 0xF3, PORTD);
  105. /*
  106. * Port E: CS Flash, VS1011 (DREQ), RTL8019, LCD BL/Enable, IR, USB Rx/Tx
  107. * output: CS Flash, LCD BL/Enable, USB Tx
  108. * input: VS1011 (DREQ), RTL8019, IR
  109. * pull-up: USB Rx
  110. */
  111. outp(0x8E, DDRE);
  112. outp((inp(PORTE) & 0x8E) | 0x01, PORTE);
  113. /*
  114. * Port F: Keyboard_Rows, JTAG-connector, LED, LCD RS/RW, MCC-detect
  115. * output: LCD RS/RW, LED
  116. * input: Keyboard_Rows, MCC-detect
  117. * pull-up: Keyboard_Rows, MCC-detect
  118. * note: Key row 0 & 1 are shared with JTAG TCK/TMS. Cannot be used concurrent
  119. */
  120. #ifndef USE_JTAG
  121. sbi(JTAG_REG, JTD); // disable JTAG interface to be able to use all key-rows
  122. sbi(JTAG_REG, JTD); // do it 2 times - according to requirements ATMEGA128 datasheet: see page 256
  123. #endif //USE_JTAG
  124. outp(0x0E, DDRF);
  125. outp((inp(PORTF) & 0x0E) | 0xF1, PORTF);
  126. /*
  127. * Port G: Keyboard_cols, Bus_control
  128. * output: Keyboard_cols
  129. * input: Bus Control (internal control)
  130. * pull-up: none
  131. */
  132. outp(0x18, DDRG);
  133. }
  134. /*!
  135. * \brief Starts or stops the 4.44 msec mainbeat of the system
  136. * \param OnOff indicates if the mainbeat needs to start or to stop
  137. */
  138. static void SysControlMainBeat(u_char OnOff)
  139. {
  140. int nError = 0;
  141. if (OnOff==ON)
  142. {
  143. nError = NutRegisterIrqHandler(&OVERFLOW_SIGNAL, SysMainBeatInterrupt, NULL);
  144. if (nError == 0)
  145. {
  146. init_8_bit_timer();
  147. }
  148. }
  149. else
  150. {
  151. // disable overflow interrupt
  152. disable_8_bit_timer_ovfl_int();
  153. }
  154. }
  155. /*-------------------------------------------------------------------------*/
  156. /* global variable definitions */
  157. /*-------------------------------------------------------------------------*/
  158. bool isAlarmSyncing = false;
  159. bool initialized = false;
  160. bool running = false;
  161. u_char VS_volume = 7; //[0-15];
  162. /*-------------------------------------------------------------------------*/
  163. /* local variable definitions */
  164. /*-------------------------------------------------------------------------*/
  165. /*-------------------------------------------------------------------------*/
  166. /* Thread init */
  167. /*-------------------------------------------------------------------------*/
  168. THREAD(StartupInit, arg)
  169. {
  170. NutThreadSetPriority(5);
  171. NetworkInit();
  172. initialized = true;
  173. NutThreadExit();
  174. }
  175. THREAD(AlarmSync, arg)
  176. {
  177. NutThreadSetPriority(200);
  178. while(initialized == false){
  179. NutSleep(1000);
  180. }
  181. NtpSync();
  182. for(;;)
  183. {
  184. if((initialized == true) && (hasNetworkConnection() == true))
  185. {
  186. isAlarmSyncing = true;
  187. char url[49];
  188. sprintf(url, "/getAlarmen.php?radiomac=%s&tz=%d", getMacAdress(), getTimeZone());
  189. httpGet(url, parseAlarmJson);
  190. char url2[43];
  191. sprintf(url2, "/getTwitch.php?radiomac=%s", getMacAdress());
  192. httpGet(url2, parseTwitch);
  193. isAlarmSyncing = false;
  194. }
  195. NutSleep(3000);
  196. }
  197. NutThreadExit();
  198. }
  199. /*-------------------------------------------------------------------------*/
  200. /* Global functions */
  201. /*-------------------------------------------------------------------------*/
  202. int timer(time_t start){
  203. time_t diff = time(0) - start;
  204. return diff;
  205. }
  206. long timerStruct(struct _tm s){
  207. struct _tm ct;
  208. X12RtcGetClock(&ct);
  209. long stime = (s.tm_hour * 3600) + (s.tm_min * 60) + s.tm_sec;
  210. long ctime = (ct.tm_hour * 3600) + (ct.tm_min * 60) + ct.tm_sec;
  211. return ctime - stime;
  212. }
  213. int checkOffPressed(){
  214. if (KbGetKey() != KEY_UNDEFINED){
  215. LcdBackLight(LCD_BACKLIGHT_ON);
  216. return 1;
  217. } else {
  218. return 0;
  219. }
  220. }
  221. int main(void)
  222. {
  223. initialized = 0;
  224. int VOL2 = 127;
  225. struct _tm timeCheck;
  226. struct _tm start;
  227. int idx = 0;
  228. WatchDogDisable();
  229. NutDelay(100);
  230. SysInitIO();
  231. SPIinit();
  232. LedInit();
  233. LcdLowLevelInit();
  234. Uart0DriverInit();
  235. Uart0DriverStart();
  236. LogInit();
  237. CardInit();
  238. X12Init();
  239. VsPlayerInit();
  240. NtpInit();
  241. NutThreadCreate("BackgroundThread", StartupInit, NULL, 1024);
  242. NutThreadCreate("BackgroundThread", AlarmSync, NULL, 2500);
  243. //NutThreadCreate("BackgroundThread", NTPSync, NULL, 700);
  244. /** Quick fix for turning off the display after 10 seconds boot */
  245. RcInit();
  246. KbInit();
  247. SysControlMainBeat(ON); // enable 4.4 msecs heartbeat interrupt
  248. /*
  249. * Increase our priority so we can feed the watchdog.
  250. */
  251. NutThreadSetPriority(1);
  252. /* Enable global interrupts */
  253. sei();
  254. LcdBackLight(LCD_BACKLIGHT_OFF);
  255. X12RtcGetClock(&timeCheck);
  256. X12RtcGetClock(&start);
  257. for (;;)
  258. {
  259. //printf("running = %d, time = %d\n", running, timerStruct(start));
  260. if (timerStruct(start) < 0){
  261. X12RtcGetClock(&start);
  262. }
  263. if (timerStruct(timeCheck) < 0){
  264. X12RtcGetClock(&timeCheck);
  265. }
  266. //Check if a button is pressed
  267. if (checkOffPressed() == 1){
  268. X12RtcGetClock(&start);
  269. running = true;
  270. LcdBackLight(LCD_BACKLIGHT_ON);
  271. }
  272. //If escape is pressed, stop displaying custom message
  273. if(KbGetKey() == KEY_ESC){
  274. setDisplayingCustomMessage(false);
  275. }
  276. //Check if background LED is on, and compare to timer
  277. if (running == true){
  278. if (timerStruct(start) >= 10 || running > 1){
  279. running = false;
  280. LcdBackLight(LCD_BACKLIGHT_OFF);
  281. }
  282. }
  283. if(KbGetKey() == KEY_DOWN)
  284. {
  285. NutSleep(150);
  286. X12RtcGetClock(&timeCheck);
  287. if (VS_volume > 0){
  288. --VS_volume;
  289. VS_volume = VS_volume % 17;
  290. VsSetVolume((127 - (VS_volume * 8)) % 128);
  291. }
  292. displayVolume(VS_volume);
  293. }
  294. else if(KbGetKey() == KEY_UP)
  295. {
  296. NutSleep(150);
  297. X12RtcGetClock(&timeCheck);
  298. if (VS_volume < 16){
  299. ++VS_volume;
  300. VS_volume = VS_volume % 17;
  301. VsSetVolume((127 - (VS_volume * 8)) % 128);
  302. }
  303. displayVolume(VS_volume);
  304. }
  305. else if(timerStruct(timeCheck) >= 5 && checkAlarms() == 1)
  306. {
  307. for (idx = 0; idx < 5; idx++){
  308. if (getState(idx) == 1){
  309. displayAlarm(0,1,idx);
  310. if (KbGetKey() == KEY_ESC){
  311. //NutDelay(50);
  312. handleAlarm(idx);
  313. //NutDelay(50);
  314. LcdBackLight(LCD_BACKLIGHT_OFF);
  315. stopStream();
  316. } else if (KbGetKey() == KEY_01 || KbGetKey() == KEY_02 || KbGetKey() == KEY_03 || KbGetKey() == KEY_04 || KbGetKey() == KEY_05 || KbGetKey() == KEY_ALT){
  317. setSnooze(idx);
  318. LcdBackLight(LCD_BACKLIGHT_OFF);
  319. stopStream();
  320. }
  321. }
  322. }
  323. }else if(isDisplayingCustomMessage() == true){
  324. }
  325. else if (timerStruct(timeCheck) >= 5){
  326. displayTime(0);
  327. displayDate(1);
  328. }
  329. WatchDogRestart();
  330. }
  331. return(0);
  332. }