-/*! \file goodfet.c\r
- \author Travis Goodspeed\r
- \brief Main module.\r
- \r
- This is the main module of the GoodFET, which calls the initialization\r
- routines and delegates commands to the various applications.\r
-*/\r
-\r
-\r
-#include "platform.h"\r
-#include "command.h"\r
-#include "apps.h"\r
-\r
-\r
-\r
-//LED on P1.0\r
-//IO on P5\r
-\r
-//! Initialize registers and all that jazz.\r
-void init(){\r
- WDTCTL = WDTPW + WDTHOLD; // Stop watchdog timer\r
- \r
- //LED out and on.\r
- PLEDDIR |= PLEDPIN;\r
- PLEDOUT |= PLEDPIN;\r
- \r
- //Setup clocks, unique to each '430.\r
- msp430_init_dco();\r
- msp430_init_uart();\r
- \r
- //Enable Interrupts.\r
- //eint();\r
-}\r
-\r
-//! Handle a command.\r
-void handle(unsigned char app,\r
- unsigned char verb,\r
- unsigned long len){\r
- //debugstr("GoodFET");\r
- switch(app){\r
- case MONITOR:\r
- monitorhandle(app,verb,len);\r
- break;\r
- case SPI:\r
- spihandle(app,verb,len);\r
- break;\r
- case I2CAPP:\r
- i2chandle(app,verb,len);\r
- break;\r
- case CHIPCON:\r
- cchandle(app,verb,len);\r
- break;\r
- case JTAG:\r
- jtaghandle(app,verb,len);\r
- break;\r
- case JTAG430: //Also JTAG430X, JTAG430X2\r
- jtag430x2handle(app,verb,len);\r
- break;\r
- default:\r
- #ifdef HANDLEOTHER\r
- HANDLEOTHER(app,verb,len);\r
- #else\r
- txdata(app,NOK,0);\r
- #endif\r
- break;\r
- }\r
-}\r
-\r
-//! Main loop.\r
-int main(void)\r
-{\r
- volatile unsigned int i;\r
- unsigned char app, verb;\r
- unsigned long len;\r
- \r
- init();\r
- \r
- txstring(MONITOR,OK,"http://goodfet.sf.net/");\r
- \r
- //Command loop. There's no end!\r
- while(1){\r
- //Magic 3\r
- app=serial_rx();\r
- verb=serial_rx();\r
- //len=serial_rx();\r
- len=rxword();\r
- \r
- //Read data, looking for buffer overflow.y\r
- if(len<=CMDDATALEN){\r
- for(i=0;i<len;i++){\r
- cmddata[i]=serial_rx();\r
- }\r
- handle(app,verb,len);\r
- }else{\r
- //Listen to the blaberring.\r
- for(i-0;i<len;i++)\r
- serial_rx();\r
- //Reply with an error.\r
- debugstr("Buffer length exceeded.");\r
- txdata(MONITOR,NOK,0);\r
- }\r
- }\r
-}\r
-\r
+/*! \file goodfet.c
+ \author Travis Goodspeed
+ \brief Main module.
+
+ This is the main module of the GoodFET, which calls the initialization
+ routines and delegates commands to the various applications.
+*/
+
+
+#include "platform.h"
+#include "command.h"
+#include "apps.h"
+#include "glitch.h"
+
+#if (platform == tilaunchpad)
+#include <setjmp.h>
+jmp_buf warmstart;
+void coldstart();
+#include "msp430_serial.h"
+#endif
+
+#define RESET 0x80 // not a real app -- causes firmware to reset
+#define DEBUGAPP 0xFF
+
+//! General init function, calls platform-specific one.
+void init(){
+#ifdef MSP430
+ #define INITCHIP msp430_init();
+#endif
+
+#ifdef ARDUINO
+ #define INITCHIP arduino_init();
+#endif
+
+#if (platform == donbfet)
+# define INITCHIP donbfet_init();
+#endif
+
+#ifdef INITCHIP
+INITCHIP
+#else
+#warning "No init() routine for this platform!"
+#endif
+
+#ifdef INITPLATFORM
+ INITPLATFORM
+#endif
+}
+
+
+
+//! Handle a command.
+void handle(uint8_t const app,
+ uint8_t const verb,
+ uint32_t const len){
+ int i;
+
+ //debugstr("GoodFET");
+ //led_off();
+
+ // find the app and call the handle fn
+ for(i = 0; i < num_apps; i++){
+ if(apps[i]->app == app){
+ // call the app's handle fn
+ (*(apps[i]->handle))(app, verb, len);
+
+ // exit early
+ return;
+ }
+ }
+
+ // if we get here, then the desired app is not compiled into
+ // this firmware
+ debugstr("App missing.");
+ debughex(app);
+ txdata(app, NOK, 0);
+}
+
+
+//! Main loop.
+int main(void){
+ volatile unsigned int i;
+ unsigned char app, verb;
+ unsigned long len;
+ // MSP reboot count for reset input & reboot function located at 0xFFFE
+ volatile unsigned int reset_count = 0;
+
+ silent=0; //Don't trust globals.
+
+#if (platform == tilaunchpad)
+ int ret=0;
+
+ //ret = setjmp(warmstart);// needs to be here since context from init() would be gone
+ warmstart:
+ if (ret == 0) {
+ coldstart(); // basic hardware setup, clock to TUSB3410, and enable
+ } else if (ret == 2) {
+ dputs("\nalmost BSL only one RTS change\n");
+ } else if (ret > 2) { // reset released after more than two tst transisitions
+ // We could write a BSL, a nice exercise for a Sunday afternoon.
+ dputs("\nBSL\n");
+ //call_BSL(); // once you are done uncomment ;-)
+ } else { // we come here after DTR high (release reset)
+ dputs("\nWarmstart\n");
+ }
+#endif
+
+#if (platform == donbfet)
+ extern void donbfet_reboot(void);
+ void (*reboot_function)(void) = donbfet_reboot;
+#elif (platform == zigduino)
+ extern void zigduino_reboot(void);
+ void (*reboot_function)(void) = zigduino_reboot;
+#else
+ void (*reboot_function)(void) = (void *) 0xFFFE;
+#endif
+
+ init();
+
+ txstring(MONITOR,OK,"http://goodfet.sf.net/");
+ //txstring(0xab,0xcd,"http://goodfet.sf.net/");
+
+
+ //Command loop. There's no end!
+ while(1){
+ //Magic 3
+ app = serial_rx();
+
+ // If the app is the reset byte (0x80) increment and loop
+ if (app == RESET){
+ reset_count++;
+
+ if (reset_count > 4){
+ // We could trigger the WDT with either:
+ // WDTCTL = 0;
+ // or
+ // WDTCTL = WDTPW + WDTCNTCL + WDTSSEL + 0x00;
+ // but instead we'll jump to our reboot function pointer
+ (*reboot_function)();
+ debugstr("Rebooting not supported on this platform.");
+ }
+
+ continue;
+ }else {
+ reset_count = 0;
+ }
+
+ verb = serial_rx();
+ len = rxword();
+
+ //Read data, looking for buffer overflow.
+ if(len <= CMDDATALEN){
+ for(i = 0; i < len; i++)
+ cmddata[i] = serial_rx();
+
+ handle(app,verb,len);
+ }else {
+ //Listen to the blaberring.
+ for(i = 0; i < len; i++)
+ serial_rx();
+
+ //Reply with an error.
+ debugstr("Buffer length exceeded.");
+ txdata(MONITOR,NOK,0);
+ }
+ }
+}
+