104 lines
2 KiB
C
104 lines
2 KiB
C
#include "bootloader.h"
|
|
#include "dev/watchdog.h"
|
|
#include <util/delay.h>
|
|
#include <avr/wdt.h>
|
|
#include <avr/interrupt.h>
|
|
#include <avr/pgmspace.h>
|
|
#include "dev/usb/usb_drv.h"
|
|
#include <avr/pgmspace.h>
|
|
#include <avr/interrupt.h>
|
|
#include <avr/eeprom.h>
|
|
|
|
/* MCUSR is a deprecated name but older avr-libc versions may define it */
|
|
#if !defined (MCUCSR)
|
|
# if defined (MCUSR)
|
|
# define MCUCSR MCUSR
|
|
# endif
|
|
#endif
|
|
|
|
#ifndef EEPROM_MAGIC_BYTE_ADDR
|
|
#define EEPROM_MAGIC_BYTE_ADDR (uint8_t*)(E2END-3)
|
|
#endif
|
|
|
|
volatile uint32_t Boot_Key ATTR_NO_INIT;
|
|
|
|
extern void Bootloader_Jump_Check(void) ATTR_INIT_SECTION(3);
|
|
|
|
bool
|
|
bootloader_is_present(void)
|
|
{
|
|
#if defined(BOOTLOADER_START_ADDRESS)
|
|
return pgm_read_word_far(BOOTLOADER_START_ADDRESS) != 0xFFFF;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
void
|
|
Jump_To_Bootloader(void)
|
|
{
|
|
/* Disable all interrupts */
|
|
cli();
|
|
|
|
#ifdef UDCON
|
|
/* If USB is used, detach from the bus */
|
|
Usb_detach();
|
|
|
|
uint8_t i;
|
|
|
|
/* Wait two seconds for the USB detachment to register on the host */
|
|
for(i = 0; i < 200; i++) {
|
|
_delay_ms(10);
|
|
watchdog_periodic();
|
|
}
|
|
#endif
|
|
|
|
/* Set the bootloader key to the magic value and force a reset */
|
|
Boot_Key = MAGIC_BOOT_KEY;
|
|
|
|
eeprom_write_byte(EEPROM_MAGIC_BYTE_ADDR, 0xFF);
|
|
|
|
/* Enable interrupts */
|
|
sei();
|
|
|
|
watchdog_reboot();
|
|
}
|
|
|
|
void
|
|
Bootloader_Jump_Check(void)
|
|
{
|
|
/* If the reset source was the bootloader and the key is correct,
|
|
* clear it and jump to the bootloader
|
|
*/
|
|
if(MCUCSR & (1 << WDRF)) {
|
|
MCUCSR = 0;
|
|
if(Boot_Key == MAGIC_BOOT_KEY) {
|
|
Boot_Key = 0;
|
|
wdt_disable();
|
|
|
|
/* Disable all interrupts */
|
|
cli();
|
|
|
|
eeprom_write_byte(EEPROM_MAGIC_BYTE_ADDR, 0xFF);
|
|
|
|
/* Enable interrupts */
|
|
sei();
|
|
|
|
((void (*)(void))(BOOTLOADER_START_ADDRESS)) ();
|
|
} else {
|
|
/* The watchdog fired. Probably means we
|
|
* crashed. Wait two seconds before continuing.
|
|
*/
|
|
|
|
Boot_Key++;
|
|
uint8_t i;
|
|
|
|
for(i = 0; i < 200; i++) {
|
|
_delay_ms(10);
|
|
watchdog_periodic();
|
|
}
|
|
}
|
|
} else {
|
|
Boot_Key = MAGIC_BOOT_KEY - 4;
|
|
}
|
|
}
|