@ -66,6 +66,8 @@ void ISPProtocol_EnterISPMode(void)
ISPProtocol_DelayMS ( Enter_ISP_Params . ExecutionDelayMS ) ;
SPI_Init ( ISPTarget_GetSPIPrescalerMask ( ) | SPI_SCK_LEAD_RISING | SPI_SAMPLE_LEADING | SPI_MODE_MASTER ) ;
/* Continuously attempt to synchronize with the target until either the number of attempts specified
* by the host has exceeded , or the the device sends back the expected response values */
while ( Enter_ISP_Params . SynchLoops - - & & ( ResponseStatus = = STATUS_CMD_FAILED ) )
{
uint8_t ResponseBytes [ 4 ] ;
@ -110,6 +112,7 @@ void ISPProtocol_LeaveISPMode(void)
Endpoint_ClearOUT ( ) ;
Endpoint_SetEndpointDirection ( ENDPOINT_DIR_IN ) ;
/* Perform pre-exit delay, release the target /RESET, disable the SPI bus and perform the post-exit delay */
ISPProtocol_DelayMS ( Leave_ISP_Params . PreDelayMS ) ;
ISPTarget_ChangeTargetResetLine ( false ) ;
SPI_ShutDown ( ) ;
@ -166,6 +169,9 @@ void ISPProtocol_ProgramMemory(uint8_t V2Command)
Write_Memory_Params . PollValue2 ;
uint8_t * NextWriteByte = Write_Memory_Params . ProgData ;
/* Check to see if the host has issued a SET ADDRESS command and we haven't sent a
* LOAD EXTENDED ADDRESS command ( if needed , used when operating beyond the 128 KB
* FLASH barrier ) */
if ( MustSetAddress )
{
if ( CurrentAddress & ( 1UL < < 31 ) )
@ -174,6 +180,7 @@ void ISPProtocol_ProgramMemory(uint8_t V2Command)
MustSetAddress = false ;
}
/* Check the programming mode desired by the host, either Paged or Word memory writes */
if ( Write_Memory_Params . ProgrammingMode & PROG_MODE_PAGED_WRITES_MASK )
{
uint16_t StartAddress = ( CurrentAddress & 0xFFFF ) ;
@ -184,16 +191,16 @@ void ISPProtocol_ProgramMemory(uint8_t V2Command)
bool IsOddByte = ( CurrentByte & 0x01 ) ;
uint8_t ByteToWrite = * ( NextWriteByte + + ) ;
if ( IsOddByte & & ( V2Command = = CMD_PROGRAM_FLASH_ISP ) )
Write_Memory_Params . ProgrammingCommands [ 0 ] | = READ_WRITE_HIGH_BYTE_MASK ;
else
Write_Memory_Params . ProgrammingCommands [ 0 ] & = ~ READ_WRITE_HIGH_BYTE_MASK ;
SPI_SendByte ( Write_Memory_Params . ProgrammingCommands [ 0 ] ) ;
SPI_SendByte ( CurrentAddress > > 8 ) ;
SPI_SendByte ( CurrentAddress & 0xFF ) ;
SPI_SendByte ( ByteToWrite ) ;
/* AVR FLASH addressing requires us to modify the write command based on if we are writing a high
* or low byte at the current word address */
Write_Memory_Params . ProgrammingCommands [ 0 ] ^ = READ_WRITE_HIGH_BYTE_MASK ;
/* Check to see the write completion method, to see if we have a valid polling address */
if ( ! ( PollAddress ) & & ( ByteToWrite ! = PollValue ) )
{
if ( IsOddByte & & ( V2Command = = CMD_PROGRAM_FLASH_ISP ) )
@ -289,6 +296,9 @@ void ISPProtocol_ReadMemory(uint8_t V2Command)
Endpoint_Write_Byte ( V2Command ) ;
Endpoint_Write_Byte ( STATUS_CMD_OK ) ;
/* Check to see if the host has issued a SET ADDRESS command and we haven't sent a
* LOAD EXTENDED ADDRESS command ( if needed , used when operating beyond the 128 KB
* FLASH barrier ) */
if ( MustSetAddress )
{
if ( CurrentAddress & ( 1UL < < 31 ) )
@ -297,28 +307,30 @@ void ISPProtocol_ReadMemory(uint8_t V2Command)
MustSetAddress = false ;
}
/* Read each byte from the device and write them to the packet for the host */
for ( uint16_t CurrentByte = 0 ; CurrentByte < Read_Memory_Params . BytesToRead ; CurrentByte + + )
{
bool IsOddByte = ( CurrentByte & 0x01 ) ;
if ( IsOddByte & & ( V2Command = = CMD_READ_FLASH_ISP ) )
Read_Memory_Params . ReadMemoryCommand | = READ_WRITE_HIGH_BYTE_MASK ;
else
Read_Memory_Params . ReadMemoryCommand & = ~ READ_WRITE_HIGH_BYTE_MASK ;
/* Read the next byte from the desired memory space in the device */
SPI_SendByte ( Read_Memory_Params . ReadMemoryCommand ) ;
SPI_SendByte ( CurrentAddress > > 8 ) ;
SPI_SendByte ( CurrentAddress & 0xFF ) ;
Endpoint_Write_Byte ( SPI_ReceiveByte ( ) ) ;
/* Check if the endpoint bank is currently full */
/* Check if the endpoint bank is currently full, if so send the packet */
if ( ! ( Endpoint_IsReadWriteAllowed ( ) ) )
{
Endpoint_ClearIN ( ) ;
Endpoint_WaitUntilReady ( ) ;
}
if ( ( IsOddByte & & ( V2Command = = CMD_READ_FLASH_ISP ) ) | | ( V2Command = = CMD_READ_EEPROM_ISP ) )
/* AVR FLASH addressing requires us to modify the read command based on if we are reading a high
* or low byte at the current word address */
if ( V2Command = = CMD_READ_FLASH_ISP )
Read_Memory_Params . ReadMemoryCommand ^ = READ_WRITE_HIGH_BYTE_MASK ;
/* Only increment the current address if we have read both bytes in the current word when in FLASH
* read mode , or for each byte when in EEPROM read mode */
if ( ( ( CurrentByte & 0x01 ) & & ( V2Command = = CMD_READ_FLASH_ISP ) ) | | ( V2Command = = CMD_READ_EEPROM_ISP ) )
CurrentAddress + + ;
}
@ -353,9 +365,11 @@ void ISPProtocol_ChipErase(void)
uint8_t ResponseStatus = STATUS_CMD_OK ;
/* Send the chip erase commands as given by the host to the device */
for ( uint8_t SByte = 0 ; SByte < sizeof ( Erase_Chip_Params . EraseCommandBytes ) ; SByte + + )
SPI_SendByte ( Erase_Chip_Params . EraseCommandBytes [ SByte ] ) ;
/* Use appropriate command completion check as given by the host (delay or busy polling) */
if ( ! ( Erase_Chip_Params . PollMethod ) )
ISPProtocol_DelayMS ( Erase_Chip_Params . EraseDelayMS ) ;
else
@ -386,6 +400,7 @@ void ISPProtocol_ReadFuseLockSigOSCCAL(uint8_t V2Command)
uint8_t ResponseBytes [ 4 ] ;
/* Send the Fuse or Lock byte read commands as given by the host to the device, store response */
for ( uint8_t RByte = 0 ; RByte < sizeof ( ResponseBytes ) ; RByte + + )
ResponseBytes [ RByte ] = SPI_TransferByte ( Read_FuseLockSigOSCCAL_Params . ReadCommandBytes [ RByte ] ) ;
@ -413,6 +428,7 @@ void ISPProtocol_WriteFuseLock(uint8_t V2Command)
Endpoint_ClearOUT ( ) ;
Endpoint_SetEndpointDirection ( ENDPOINT_DIR_IN ) ;
/* Send the Fuse or Lock byte program commands as given by the host to the device */
for ( uint8_t SByte = 0 ; SByte < sizeof ( Write_FuseLockSig_Params . WriteCommandBytes ) ; SByte + + )
SPI_SendByte ( Write_FuseLockSig_Params . WriteCommandBytes [ SByte ] ) ;
@ -464,11 +480,28 @@ void ISPProtocol_SPIMulti(void)
else
Endpoint_Write_Byte ( SPI_ReceiveByte ( ) ) ;
/* Check to see if we have filled the endpoint bank and need to send the packet */
if ( ! ( Endpoint_IsReadWriteAllowed ( ) ) )
{
Endpoint_ClearIN ( ) ;
Endpoint_WaitUntilReady ( ) ;
}
CurrRxPos + + ;
}
Endpoint_Write_Byte ( STATUS_CMD_OK ) ;
bool IsEndpointFull = ! ( Endpoint_IsReadWriteAllowed ( ) ) ;
Endpoint_ClearIN ( ) ;
/* Ensure last packet is a short packet to terminate the transfer */
if ( IsEndpointFull )
{
Endpoint_WaitUntilReady ( ) ;
Endpoint_ClearIN ( ) ;
Endpoint_WaitUntilReady ( ) ;
}
}
# endif