mirror of
https://github.com/travisgoodspeed/goodwatch
synced 2024-11-22 08:09:13 +00:00
182 lines
4.7 KiB
C
182 lines
4.7 KiB
C
/*! \file main.c
|
|
|
|
\brief Main module. This version just initializes the LCD and then
|
|
drops to a low power mode, letting the WDT do the work on a slow
|
|
interval.
|
|
*/
|
|
|
|
#include <msp430.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
|
|
#include "lcd.h"
|
|
#include "lcdtext.h"
|
|
#include "rtc.h"
|
|
#include "keypad.h"
|
|
#include "apps.h"
|
|
#include "sidebutton.h"
|
|
#include "radio.h"
|
|
#include "dmesg.h"
|
|
|
|
//! Initialize the XT1 crystal, and stabilize it.
|
|
void xtal_init(){
|
|
P5SEL |= BIT0 + BIT1; // Select XT1
|
|
UCSCTL6 |= XCAP_3; // Internal load cap
|
|
|
|
// Loop until XT1 & DCO stabilizes
|
|
do{
|
|
UCSCTL7 &= ~(XT1LFOFFG + DCOFFG);
|
|
// Clear LFXT1,DCO fault flags
|
|
SFRIFG1 &= ~OFIFG; // Clear fault flags
|
|
}while (SFRIFG1 & OFIFG); // Test oscillator fault flag
|
|
|
|
UCSCTL6 &= ~(XT1DRIVE_3); // Xtal is now stable, reduce drive
|
|
// strength
|
|
//See page 125 of the family guide.
|
|
//UCSCTL4 = SELM_3 + SELS_0 + SELA_0; //XT1 for ACLK and SMCLK, MCLK from DCO.
|
|
UCSCTL4 = SELM_0 + SELS_0 + SELA_0; //XT1 for everything; very slow CPU.
|
|
}
|
|
|
|
//! Power On Self Test
|
|
int post(){
|
|
if(LCDBIV || (LCDBCTL1&LCDNOCAPIFG)){
|
|
/* When the LCD cap is missing, of the wrong value, or
|
|
mis-soldered, the charge pump will be disabled out of
|
|
self-protection. This looks like a normally dark screen
|
|
abruptly fading to nothing, but you might still be able to
|
|
see the response at the right angle.
|
|
*/
|
|
lcd_string("lcd lcd");
|
|
}else if(UCSCTL7&2){
|
|
/* This flag is triggered when the 32kHz crystal has a fault, such
|
|
as if it is not populated or if a little drag of solder reaches
|
|
ground. Watches with this problem will lose minutes a day, but
|
|
the '430 will fall back to an internal oscillator so that
|
|
non-watch functions still work.
|
|
*/
|
|
lcd_string(" crystal");
|
|
|
|
/*Can't run this test because of an unfixed errata.
|
|
}else if(has_radio && RF1AIFERR & 1){
|
|
lcd_string("RF LOWV");
|
|
*/
|
|
}else if(has_radio && RF1AIFERR & 2){
|
|
lcd_string("RF OPERR");
|
|
}else if(has_radio && RF1AIFERR & 4){
|
|
lcd_string("RFOUTERR");
|
|
}else if(has_radio && RF1AIFERR & 8){
|
|
lcd_string("RF OVERW");
|
|
}else{
|
|
/* Return zero if everything is hunky dory.
|
|
*/
|
|
lcd_string("all good");
|
|
return 0;
|
|
}
|
|
|
|
//We had a failure, indicated above.
|
|
return 1;
|
|
}
|
|
|
|
//! Main method.
|
|
int main(void) {
|
|
WDTCTL = WDTPW + WDTHOLD; // Stop WDT
|
|
|
|
//Initialize the various modules.
|
|
dmesg_init();
|
|
lcd_init();
|
|
|
|
lcd_zero();
|
|
lcd_string("RTC INIT");
|
|
rtc_init();
|
|
|
|
lcd_zero();
|
|
lcd_string("KEY INIT");
|
|
key_init();
|
|
|
|
lcd_zero();
|
|
lcd_string("BUT INIT");
|
|
sidebutton_init();
|
|
|
|
lcd_zero();
|
|
lcd_string("APP INIT");
|
|
app_init();
|
|
|
|
lcd_zero();
|
|
lcd_string("OSC INIT");
|
|
xtal_init();
|
|
|
|
lcd_zero();
|
|
lcd_string("RAD INIT");
|
|
radio_init();
|
|
|
|
// Setup and enable WDT 250ms, ACLK, interval timer
|
|
WDTCTL = WDT_ADLY_250;
|
|
SFRIE1 |= WDTIE;
|
|
|
|
// Turn off SVSH, SVSM
|
|
PMMCTL0_H = 0xA5;
|
|
SVSMHCTL = 0;
|
|
SVSMLCTL = 0;
|
|
PMMCTL0_H = 0x00;
|
|
|
|
|
|
|
|
|
|
lcd_string("POSTPOST");
|
|
// Run the POST until it passes.
|
|
while(post());
|
|
|
|
lcd_string("LPM3LPM3");
|
|
__bis_SR_register(LPM3_bits + GIE); // Enter LPM3
|
|
//__bis_SR_register(LPM0_bits + GIE); // Enter LPM0 w/interrupt
|
|
while(1);
|
|
}
|
|
|
|
//! Watchdog Timer interrupt service routine, calls back to handler functions.
|
|
void __attribute__ ((interrupt(WDT_VECTOR))) watchdog_timer (void) {
|
|
static int latch=0;
|
|
|
|
|
|
if(sidebutton_mode()){
|
|
/* So if the side button is being pressed, we increment the latch
|
|
and move to the next application. Some applications, such as the
|
|
calculator, might hijack the call, so if we are latched for too
|
|
many poling cycles, we forcibly revert to the clock applicaiton.
|
|
*/
|
|
|
|
lcd_zero();
|
|
|
|
//Politely move to the next app if requested.
|
|
if(!(latch++))
|
|
app_next();
|
|
|
|
//Force a shift to the home if held for 4 seconds (16 polls)
|
|
if(latch>16)
|
|
app_forcehome();
|
|
|
|
|
|
}else if(sidebutton_set()){
|
|
/* Similarly, we'll reboot if the SET/PRGM button has been held for 10
|
|
seconds (40 polls). We'll draw a countdown if getting close, so
|
|
there's no ambiguity as to whether the chip reset.
|
|
|
|
The other features of this button are handled within each application's
|
|
draw function.
|
|
*/
|
|
if(latch++>40)
|
|
PMMCTL0 = PMMPW | PMMSWPOR;
|
|
|
|
}else{
|
|
latch=0;
|
|
}
|
|
|
|
|
|
|
|
/* The applet is drawn four times per second. We handle
|
|
double-buffering, so that incomplete drawings won't be shown to
|
|
the user, but everything else is the app's responsibility. */
|
|
lcd_predraw();
|
|
app_draw();
|
|
lcd_postdraw();
|
|
}
|