3      Copyright (C) Dean Camera, 2009. 
   5   dean [at] fourwalledcubicle [dot] com 
   6       www.fourwalledcubicle.com 
  10   Copyright 2009  Dean Camera (dean [at] fourwalledcubicle [dot] com) 
  12   Permission to use, copy, modify, and distribute this software 
  13   and its documentation for any purpose and without fee is hereby 
  14   granted, provided that the above copyright notice appear in all 
  15   copies and that both that the copyright notice and this 
  16   permission notice and warranty disclaimer appear in supporting 
  17   documentation, and that the name of the author not be used in 
  18   advertising or publicity pertaining to distribution of the 
  19   software without specific, written prior permission. 
  21   The author disclaim all warranties with regard to this 
  22   software, including all implied warranties of merchantability 
  23   and fitness.  In no event shall the author be liable for any 
  24   special, indirect or consequential damages or any damages 
  25   whatsoever resulting from loss of use, data or profits, whether 
  26   in an action of contract, negligence or other tortious action, 
  27   arising out of or in connection with the use or performance of 
  33  *  Main source file for the CDC class bootloader. This file contains the complete bootloader logic. 
  36 #define  INCLUDE_FROM_BOOTLOADERCDC_C 
  37 #include "BootloaderCDC.h" 
  40 /** Line coding options for the virtual serial port. Although the virtual serial port data is never 
  41  *  sent through a physical serial port, the line encoding data must still be read and preserved from 
  42  *  the host, or the host will detect a problem and fail to open the port. This structure contains the 
  43  *  current encoding options, including baud rate, character format, parity mode and total number of  
  44  *  bits in each data chunk. 
  46 CDC_Line_Coding_t LineCoding 
= { BaudRateBPS
: 9600, 
  47                                  CharFormat
:  OneStopBit
, 
  48                                  ParityType
:  Parity_None
, 
  51 /** Current address counter. This stores the current address of the FLASH or EEPROM as set by the host, 
  52  *  and is used when reading or writing to the AVRs memory (either FLASH or EEPROM depending on the issued 
  57 /** Flag to indicate if the bootloader should be running, or should exit and allow the application code to run 
  58  *  via a soft reset. When cleared, the bootloader will abort, the USB interface will shut down and the application 
  59  *  jumped to via an indirect jump to location 0x0000. 
  61 bool RunBootloader 
= true; 
  64 /** Main program entry point. This routine configures the hardware required by the bootloader, then continuously  
  65  *  runs the bootloader processing routine until instructed to soft-exit, or hard-reset via the watchdog to start 
  66  *  the loaded application code. 
  70         /* Disable watchdog if enabled by bootloader/fuses */ 
  71         MCUSR 
&= ~(1 << WDRF
); 
  74         /* Disable clock division */ 
  75         clock_prescale_set(clock_div_1
); 
  77         /* Relocate the interrupt vector table to the bootloader section */ 
  81         /* Initialize USB Subsystem */ 
  90         Endpoint_SelectEndpoint(CDC_TX_EPNUM
); 
  92         /* Wait until any pending transmissions have completed before shutting down */ 
  93         while (!(Endpoint_IsINReady())); 
  95         /* Shut down the USB subsystem */ 
  98         /* Relocate the interrupt vector table back to the application section */ 
 102         /* Reset any used hardware ports back to their defaults */ 
 111         /* Re-enable RWW section */ 
 114         /* Start the user application */ 
 115         AppPtr_t AppStartPtr 
= (AppPtr_t
)0x0000; 
 119 /** Event handler for the USB_Disconnect event. This indicates that the bootloader should exit and the user 
 120  *  application started. 
 122 EVENT_HANDLER(USB_Disconnect
) 
 124         /* Upon disconnection, run user application */ 
 125         RunBootloader 
= false; 
 128 /** Event handler for the USB_ConfigurationChanged event. This configures the device's endpoints ready 
 129  *  to relay data to and from the attached USB host. 
 131 EVENT_HANDLER(USB_ConfigurationChanged
) 
 133         /* Setup CDC Notification, Rx and Tx Endpoints */ 
 134         Endpoint_ConfigureEndpoint(CDC_NOTIFICATION_EPNUM
, EP_TYPE_INTERRUPT
, 
 135                                        ENDPOINT_DIR_IN
, CDC_NOTIFICATION_EPSIZE
, 
 136                                    ENDPOINT_BANK_SINGLE
); 
 138         Endpoint_ConfigureEndpoint(CDC_TX_EPNUM
, EP_TYPE_BULK
, 
 139                                        ENDPOINT_DIR_IN
, CDC_TXRX_EPSIZE
, 
 140                                    ENDPOINT_BANK_SINGLE
); 
 142         Endpoint_ConfigureEndpoint(CDC_RX_EPNUM
, EP_TYPE_BULK
, 
 143                                        ENDPOINT_DIR_OUT
, CDC_TXRX_EPSIZE
, 
 144                                    ENDPOINT_BANK_SINGLE
); 
 147 /** Event handler for the USB_UnhandledControlPacket event. This is used to catch standard and class specific 
 148  *  control requests that are not handled internally by the USB library, so that they can be handled appropriately 
 149  *  for the application. 
 151 EVENT_HANDLER(USB_UnhandledControlPacket
) 
 153         uint8_t* LineCodingData 
= (uint8_t*)&LineCoding
; 
 155         Endpoint_Discard_Word(); 
 157         /* Process CDC specific control requests */ 
 160                 case REQ_GetLineEncoding
: 
 161                         if (bmRequestType 
== (REQDIR_DEVICETOHOST 
| REQTYPE_CLASS 
| REQREC_INTERFACE
)) 
 163                                 Endpoint_ClearControlSETUP(); 
 165                                 for (uint8_t i 
= 0; i 
< sizeof(LineCoding
); i
++) 
 166                                   Endpoint_Write_Byte(*(LineCodingData
++));      
 168                                 Endpoint_ClearControlIN(); 
 170                                 /* Acknowledge status stage */ 
 171                                 while (!(Endpoint_IsOUTReceived())); 
 172                                 Endpoint_ClearControlOUT(); 
 176                 case REQ_SetLineEncoding
: 
 177                         if (bmRequestType 
== (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
)) 
 179                                 Endpoint_ClearControlSETUP(); 
 181                                 while (!(Endpoint_IsOUTReceived())); 
 183                                 for (uint8_t i 
= 0; i 
< sizeof(LineCoding
); i
++) 
 184                                   *(LineCodingData
++) = Endpoint_Read_Byte(); 
 186                                 Endpoint_ClearControlOUT(); 
 188                                 /* Acknowledge status stage */ 
 189                                 while (!(Endpoint_IsINReady())); 
 190                                 Endpoint_ClearControlIN(); 
 194                 case REQ_SetControlLineState
: 
 195                         if (bmRequestType 
== (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
)) 
 197                                 Endpoint_ClearControlSETUP(); 
 199                                 /* Acknowledge status stage */ 
 200                                 while (!(Endpoint_IsINReady())); 
 201                                 Endpoint_ClearControlIN(); 
 208 /** Reads or writes a block of EEPROM or FLASH memory to or from the appropriate CDC data endpoint, depending 
 209  *  on the AVR910 protocol command issued. 
 211  *  \param Command  Single character AVR910 protocol command indicating what memory operation to perform 
 213 static void ReadWriteMemoryBlock(const uint8_t Command
) 
 218         bool     HighByte 
= false; 
 221         BlockSize  
= (FetchNextCommandByte() << 8); 
 222         BlockSize 
|=  FetchNextCommandByte(); 
 224         MemoryType 
=  FetchNextCommandByte(); 
 226         if ((MemoryType 
!= 'E') && (MemoryType 
!= 'F')) 
 228                 /* Send error byte back to the host */ 
 229                 WriteNextResponseByte('?'); 
 234         /* Check if command is to read memory */ 
 237                 /* Re-enable RWW section */ 
 242                         if (MemoryType 
== 'E') 
 244                                 /* Read the next EEPROM byte into the endpoint */ 
 245                                 WriteNextResponseByte(eeprom_read_byte((uint8_t*)(uint16_t)(CurrAddress 
>> 1))); 
 247                                 /* Increment the address counter after use */ 
 252                                 /* Read the next FLASH byte from the current FLASH page */ 
 254                                 WriteNextResponseByte(pgm_read_byte_far(CurrAddress 
| HighByte
)); 
 256                                 WriteNextResponseByte(pgm_read_byte(CurrAddress 
| HighByte
));                                    
 259                                 /* If both bytes in current word have been read, increment the address counter */ 
 263                                 HighByte 
= !HighByte
; 
 269                 uint32_t PageStartAddress 
= CurrAddress
; 
 271                 if (MemoryType 
== 'F') 
 273                         boot_page_erase(PageStartAddress
); 
 274                         boot_spm_busy_wait(); 
 279                         if (MemoryType 
== 'F') 
 281                                 /* If both bytes in current word have been written, increment the address counter */ 
 284                                         /* Write the next FLASH word to the current FLASH page */ 
 285                                         boot_page_fill(CurrAddress
, ((FetchNextCommandByte() << 8) | LowByte
)); 
 287                                         /* Increment the address counter after use */ 
 294                                         LowByte 
= FetchNextCommandByte(); 
 301                                 /* Write the next EEPROM byte from the endpoint */ 
 302                                 eeprom_write_byte((uint8_t*)(uint16_t)(CurrAddress 
>> 1), FetchNextCommandByte());                                       
 304                                 /* Increment the address counter after use */ 
 309                 /* If in FLASH programming mode, commit the page after writing */ 
 310                 if (MemoryType 
== 'F') 
 312                         /* Commit the flash page to memory */ 
 313                         boot_page_write(PageStartAddress
); 
 315                         /* Wait until write operation has completed */ 
 316                         boot_spm_busy_wait(); 
 319                 /* Send response byte back to the host */ 
 320                 WriteNextResponseByte('\r');             
 324 /** Retrieves the next byte from the host in the CDC data OUT endpoint, and clears the endpoint bank if needed 
 325  *  to allow reception of the next data packet from the host. 
 327  *  \return Next received byte from the host in the CDC data OUT endpoint 
 329 static uint8_t FetchNextCommandByte(void) 
 331         /* Select the OUT endpoint so that the next data byte can be read */ 
 332         Endpoint_SelectEndpoint(CDC_RX_EPNUM
); 
 334         /* If OUT endpoint empty, clear it and wait for the next packet from the host */ 
 335         while (!(Endpoint_IsReadWriteAllowed())) 
 338                 while (!(Endpoint_IsOUTReceived())); 
 341         /* Fetch the next byte from the OUT endpoint */ 
 342         return Endpoint_Read_Byte(); 
 345 /** Writes the next response byte to the CDC data IN endpoint, and sends the endpoint back if needed to free up the 
 346  *  bank when full ready for the next byte in the packet to the host. 
 348  *  \param Response  Next response byte to send to the host 
 350 static void WriteNextResponseByte(const uint8_t Response
) 
 352         /* Select the IN endpoint so that the next data byte can be written */ 
 353         Endpoint_SelectEndpoint(CDC_TX_EPNUM
); 
 355         /* If OUT endpoint empty, clear it and wait for the next packet from the host */ 
 356         if (!(Endpoint_IsReadWriteAllowed())) 
 359                 while (!(Endpoint_IsINReady())); 
 362         /* Write the next byte to the OUT endpoint */ 
 363         Endpoint_Write_Byte(Response
); 
 366 /** Task to read in AVR910 commands from the CDC data OUT endpoint, process them, perform the required actions 
 367  *  and send the appropriate response back to the host. 
 371         /* Select the OUT endpoint */ 
 372         Endpoint_SelectEndpoint(CDC_RX_EPNUM
); 
 374         /* Check if endpoint has a command in it sent from the host */ 
 375         if (Endpoint_IsOUTReceived()) 
 377                 /* Read in the bootloader command (first byte sent from host) */ 
 378                 uint8_t Command 
= FetchNextCommandByte(); 
 380                 if ((Command 
== 'L') || (Command 
== 'P') || (Command 
== 'T') || (Command 
== 'E')) 
 383                           RunBootloader 
= false; 
 385                           FetchNextCommandByte(); 
 387                         /* Send confirmation byte back to the host */ 
 388                         WriteNextResponseByte('\r');                     
 390                 else if (Command 
== 't') 
 392                         /* Return ATMEGA128 part code - this is only to allow AVRProg to use the bootloader */ 
 393                         WriteNextResponseByte(0x44); 
 395                         WriteNextResponseByte(0x00); 
 397                 else if (Command 
== 'a') 
 399                         /* Indicate auto-address increment is supported */ 
 400                         WriteNextResponseByte('Y'); 
 402                 else if (Command 
== 'A') 
 404                         /* Set the current address to that given by the host */ 
 405                         CurrAddress   
= (FetchNextCommandByte() << 9); 
 406                         CurrAddress  
|= (FetchNextCommandByte() << 1); 
 408                         /* Send confirmation byte back to the host */ 
 409                         WriteNextResponseByte('\r'); 
 411                 else if (Command 
== 'p') 
 413                         /* Indicate serial programmer back to the host */ 
 414                         WriteNextResponseByte('S');               
 416                 else if (Command 
== 'S') 
 418                         /* Write the 7-byte software identifier to the endpoint */ 
 419                         for (uint8_t CurrByte 
= 0; CurrByte 
< 7; CurrByte
++) 
 420                           WriteNextResponseByte(SOFTWARE_IDENTIFIER
[CurrByte
]);          
 422                 else if (Command 
== 'V') 
 424                         WriteNextResponseByte('0' + BOOTLOADER_VERSION_MAJOR
); 
 425                         WriteNextResponseByte('0' + BOOTLOADER_VERSION_MINOR
); 
 427                 else if (Command 
== 's') 
 429                         WriteNextResponseByte(SIGNATURE_0
); 
 430                         WriteNextResponseByte(SIGNATURE_1
); 
 431                         WriteNextResponseByte(SIGNATURE_2
);              
 433                 else if (Command 
== 'b') 
 435                         WriteNextResponseByte('Y'); 
 437                         /* Send block size to the host */ 
 438                         WriteNextResponseByte(SPM_PAGESIZE 
>> 8); 
 439                         WriteNextResponseByte(SPM_PAGESIZE 
& 0xFF);              
 441                 else if (Command 
== 'e') 
 443                         /* Clear the application section of flash */ 
 444                         for (uint32_t CurrFlashAddress 
= 0; CurrFlashAddress 
< BOOT_START_ADDR
; CurrFlashAddress
++) 
 446                                 boot_page_erase(CurrFlashAddress
); 
 447                                 boot_spm_busy_wait(); 
 448                                 boot_page_write(CurrFlashAddress
); 
 449                                 boot_spm_busy_wait(); 
 451                                 CurrFlashAddress 
+= SPM_PAGESIZE
; 
 454                         /* Send confirmation byte back to the host */ 
 455                         WriteNextResponseByte('\r');             
 457                 else if (Command 
== 'l') 
 459                         /* Set the lock bits to those given by the host */ 
 460                         boot_lock_bits_set(FetchNextCommandByte()); 
 462                         /* Send confirmation byte back to the host */ 
 463                         WriteNextResponseByte('\r'); 
 465                 else if (Command 
== 'r') 
 467                         WriteNextResponseByte(boot_lock_fuse_bits_get(GET_LOCK_BITS
));           
 469                 else if (Command 
== 'F') 
 471                         WriteNextResponseByte(boot_lock_fuse_bits_get(GET_LOW_FUSE_BITS
)); 
 473                 else if (Command 
== 'N') 
 475                         WriteNextResponseByte(boot_lock_fuse_bits_get(GET_HIGH_FUSE_BITS
));              
 477                 else if (Command 
== 'Q') 
 479                         WriteNextResponseByte(boot_lock_fuse_bits_get(GET_EXTENDED_FUSE_BITS
));          
 481                 else if (Command 
== 'C') 
 483                         /* Write the high byte to the current flash page */ 
 484                         boot_page_fill(CurrAddress
, FetchNextCommandByte()); 
 486                         /* Send confirmation byte back to the host */ 
 487                         WriteNextResponseByte('\r');             
 489                 else if (Command 
== 'c') 
 491                         /* Write the low byte to the current flash page */ 
 492                         boot_page_fill(CurrAddress 
| 1, FetchNextCommandByte()); 
 494                         /* Increment the address */ 
 497                         /* Send confirmation byte back to the host */ 
 498                         WriteNextResponseByte('\r');             
 500                 else if (Command 
== 'm') 
 502                         /* Commit the flash page to memory */ 
 503                         boot_page_write(CurrAddress
); 
 505                         /* Wait until write operation has completed */ 
 506                         boot_spm_busy_wait(); 
 508                         /* Send confirmation byte back to the host */ 
 509                         WriteNextResponseByte('\r');             
 511                 else if ((Command 
== 'B') || (Command 
== 'g')) 
 513                         /* Delegate the block write/read to a separate function for clarity */ 
 514                         ReadWriteMemoryBlock(Command
); 
 516                 else if (Command 
== 'R') 
 519                         uint16_t ProgramWord 
= pgm_read_word_far(CurrAddress
); 
 521                         uint16_t ProgramWord 
= pgm_read_word(CurrAddress
);                       
 524                         WriteNextResponseByte(ProgramWord 
>> 8); 
 525                         WriteNextResponseByte(ProgramWord 
& 0xFF); 
 527                 else if (Command 
== 'D') 
 529                         /* Read the byte from the endpoint and write it to the EEPROM */ 
 530                         eeprom_write_byte((uint8_t*)(uint16_t)(CurrAddress 
>> 1), FetchNextCommandByte()); 
 532                         /* Increment the address after use */                    
 535                         /* Send confirmation byte back to the host */ 
 536                         WriteNextResponseByte('\r');             
 538                 else if (Command 
== 'd') 
 540                         /* Read the EEPROM byte and write it to the endpoint */ 
 541                         WriteNextResponseByte(eeprom_read_byte((uint8_t*)(uint16_t)(CurrAddress 
>> 1))); 
 543                         /* Increment the address after use */ 
 546                 else if (Command 
== 27) 
 548                         /* Escape is sync, ignore */ 
 552                         /* Unknown command, return fail code */ 
 553                         WriteNextResponseByte('?'); 
 556                 /* Select the IN endpoint */ 
 557                 Endpoint_SelectEndpoint(CDC_TX_EPNUM
); 
 559                 /* Remember if the endpoint is completely full before clearing it */ 
 560                 bool IsEndpointFull 
= !(Endpoint_IsReadWriteAllowed()); 
 562                 /* Send the endpoint data to the host */ 
 565                 /* If a full endpoint's worth of data was sent, we need to send an empty packet afterwards to signal end of transfer */ 
 568                         while (!(Endpoint_IsINReady())); 
 572                 /* Select the OUT endpoint */ 
 573                 Endpoint_SelectEndpoint(CDC_RX_EPNUM
); 
 575                 /* Acknowledge the command from the host */