|
|
@ -52,12 +52,12 @@ int main(void)
|
|
|
|
while (RunBootloader)
|
|
|
|
while (RunBootloader)
|
|
|
|
USB_USBTask();
|
|
|
|
USB_USBTask();
|
|
|
|
|
|
|
|
|
|
|
|
/* Turn off the USB interface, disconnect from the host */
|
|
|
|
/* Disconnect from the host - USB interface will be reset later along with the AVR */
|
|
|
|
USB_ShutDown();
|
|
|
|
USB_Detach();
|
|
|
|
|
|
|
|
|
|
|
|
/* Enable the watchdog and force a timeout to reset the AVR */
|
|
|
|
/* Enable the watchdog and force a timeout to reset the AVR */
|
|
|
|
wdt_enable(WDTO_250MS);
|
|
|
|
wdt_enable(WDTO_250MS);
|
|
|
|
|
|
|
|
|
|
|
|
for (;;);
|
|
|
|
for (;;);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -100,53 +100,50 @@ void EVENT_USB_Device_UnhandledControlRequest(void)
|
|
|
|
switch (USB_ControlRequest.bRequest)
|
|
|
|
switch (USB_ControlRequest.bRequest)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
case REQ_SetReport:
|
|
|
|
case REQ_SetReport:
|
|
|
|
if (USB_ControlRequest.bmRequestType == (REQDIR_HOSTTODEVICE | REQTYPE_CLASS | REQREC_INTERFACE))
|
|
|
|
Endpoint_ClearSETUP();
|
|
|
|
{
|
|
|
|
|
|
|
|
Endpoint_ClearSETUP();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Wait until the command has been sent by the host */
|
|
|
|
|
|
|
|
while (!(Endpoint_IsOUTReceived()));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Read in the write destination address */
|
|
|
|
/* Wait until the command has been sent by the host */
|
|
|
|
uint16_t PageAddress = Endpoint_Read_Word_LE();
|
|
|
|
while (!(Endpoint_IsOUTReceived()));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Read in the write destination address */
|
|
|
|
|
|
|
|
uint16_t PageAddress = Endpoint_Read_Word_LE();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Check if the command is a program page command, or a start application command */
|
|
|
|
|
|
|
|
if (PageAddress == TEENSY_STARTAPPLICATION)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
RunBootloader = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
/* Erase the given FLASH page, ready to be programmed */
|
|
|
|
|
|
|
|
boot_page_erase(PageAddress);
|
|
|
|
|
|
|
|
boot_spm_busy_wait();
|
|
|
|
|
|
|
|
|
|
|
|
/* Check if the command is a program page command, or a start application command */
|
|
|
|
/* Write each of the FLASH page's bytes in sequence */
|
|
|
|
if (PageAddress == TEENSY_STARTAPPLICATION)
|
|
|
|
for (uint8_t PageByte = 0; PageByte < SPM_PAGESIZE; PageByte += 2)
|
|
|
|
{
|
|
|
|
|
|
|
|
RunBootloader = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
/* Erase the given FLASH page, ready to be programmed */
|
|
|
|
/* Check if endpoint is empty - if so clear it and wait until ready for next packet */
|
|
|
|
boot_page_erase(PageAddress);
|
|
|
|
if (!(Endpoint_BytesInEndpoint()))
|
|
|
|
boot_spm_busy_wait();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Write each of the FLASH page's bytes in sequence */
|
|
|
|
|
|
|
|
for (uint8_t PageByte = 0; PageByte < SPM_PAGESIZE; PageByte += 2)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
/* Check if endpoint is empty - if so clear it and wait until ready for next packet */
|
|
|
|
Endpoint_ClearOUT();
|
|
|
|
if (!(Endpoint_BytesInEndpoint()))
|
|
|
|
while (!(Endpoint_IsOUTReceived()));
|
|
|
|
{
|
|
|
|
|
|
|
|
Endpoint_ClearOUT();
|
|
|
|
|
|
|
|
while (!(Endpoint_IsOUTReceived()));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Write the next data word to the FLASH page */
|
|
|
|
|
|
|
|
boot_page_fill(PageAddress + PageByte, Endpoint_Read_Word_LE());
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Write the filled FLASH page to memory */
|
|
|
|
/* Write the next data word to the FLASH page */
|
|
|
|
boot_page_write(PageAddress);
|
|
|
|
boot_page_fill(PageAddress + PageByte, Endpoint_Read_Word_LE());
|
|
|
|
boot_spm_busy_wait();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Re-enable RWW section */
|
|
|
|
|
|
|
|
boot_rww_enable();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Endpoint_ClearOUT();
|
|
|
|
/* Write the filled FLASH page to memory */
|
|
|
|
|
|
|
|
boot_page_write(PageAddress);
|
|
|
|
|
|
|
|
boot_spm_busy_wait();
|
|
|
|
|
|
|
|
|
|
|
|
Endpoint_ClearStatusStage();
|
|
|
|
/* Re-enable RWW section */
|
|
|
|
|
|
|
|
boot_rww_enable();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Endpoint_ClearOUT();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Endpoint_ClearStatusStage();
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|