mirror of
https://github.com/opentx/opentx.git
synced 2025-07-23 16:25:16 +03:00
971 lines
28 KiB
C
971 lines
28 KiB
C
/* ----------------------------------------------------------------------------
|
|
* ATMEL Microcontroller Software Support
|
|
* ----------------------------------------------------------------------------
|
|
* Copyright (c) 2011, Atmel Corporation
|
|
*
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* - Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the disclaimer below.
|
|
*
|
|
* Atmel's name may not be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
* ----------------------------------------------------------------------------
|
|
*/
|
|
|
|
|
|
/*----------------------------------------------------------------------------
|
|
* Headers
|
|
*----------------------------------------------------------------------------*/
|
|
|
|
#include "../common/applet.h"
|
|
#include <board.h>
|
|
//#include "libspiflash.h"
|
|
#include <string.h>
|
|
|
|
#undef DYN_TRACES
|
|
#define DYN_TRACES 0
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------
|
|
* Internal definitions
|
|
*----------------------------------------------------------------------------*/
|
|
|
|
/** SPI clock frequency in Hz. */
|
|
#define SPCK 8000000
|
|
|
|
|
|
/** Stack size in SRAM */
|
|
#define STACK_SIZE 0x100
|
|
|
|
/** Last erased sector id, this will avoid to erase if the block is already erased. */
|
|
//static unsigned short lastErasedBlock = 0xFFFF;
|
|
|
|
/** Indicate the farthest memory offset ever erase, if current write address is less
|
|
than the address, the related block should be erase again before write. */
|
|
//static uint32_t writtenAddress = 0;
|
|
|
|
|
|
/** Max size of data we can tranfsert in one shot */
|
|
#define MAX_COUNT 0x6000
|
|
|
|
/** Chip select value used to select the AT25 chip. */
|
|
#define SPI_CS 0
|
|
/** SPI peripheral pins to configure to access the serial flash. */
|
|
#define SPI_PINS PINS_SPI0, PIN_SPI0_NPCS0
|
|
|
|
/*----------------------------------------------------------------------------
|
|
* Types
|
|
*----------------------------------------------------------------------------*/
|
|
/** Structure for storing parameters for each command that can be performed by
|
|
the applet. */
|
|
struct _Mailbox {
|
|
|
|
/** Command send to the monitor to be executed. */
|
|
uint32_t command;
|
|
/** Returned status, updated at the end of the monitor execution.*/
|
|
uint32_t status;
|
|
|
|
/** Input Arguments in the argument area*/
|
|
union {
|
|
|
|
/** Input arguments for the Init command.*/
|
|
struct {
|
|
|
|
/** Communication link used.*/
|
|
uint32_t comType;
|
|
/**Trace level.*/
|
|
uint32_t traceLevel;
|
|
/** Serial flash index.*/
|
|
uint32_t at25Idx;
|
|
|
|
} inputInit;
|
|
|
|
/** Output arguments for the Init command.*/
|
|
struct {
|
|
|
|
/** Memory size.*/
|
|
uint32_t memorySize;
|
|
/** Buffer address.*/
|
|
uint32_t bufferAddress;
|
|
/** Buffer size.*/
|
|
uint32_t bufferSize;
|
|
|
|
} outputInit;
|
|
|
|
/** Input arguments for the Write command.*/
|
|
struct {
|
|
|
|
/** Buffer address.*/
|
|
uint32_t bufferAddr;
|
|
/** Buffer size.*/
|
|
uint32_t bufferSize;
|
|
/** Memory offset.*/
|
|
uint32_t memoryOffset;
|
|
|
|
} inputWrite;
|
|
|
|
/** Output arguments for the Write command.*/
|
|
struct {
|
|
|
|
/** Bytes written.*/
|
|
uint32_t bytesWritten;
|
|
|
|
uint32_t bufferSize; // temp
|
|
/** Memory offset.*/
|
|
uint32_t memoryOffset; // temp
|
|
uint32_t debug1 ;
|
|
uint32_t debug2 ;
|
|
|
|
} outputWrite;
|
|
|
|
/** Input arguments for the Read command.*/
|
|
struct {
|
|
|
|
/** Buffer address. */
|
|
uint32_t bufferAddr;
|
|
/** Buffer size.*/
|
|
uint32_t bufferSize;
|
|
/** Memory offset.*/
|
|
uint32_t memoryOffset;
|
|
|
|
} inputRead;
|
|
|
|
/** Output arguments for the Read command.*/
|
|
struct {
|
|
|
|
/** Bytes read.*/
|
|
uint32_t bytesRead;
|
|
|
|
} outputRead;
|
|
|
|
/** Input arguments for the Full Erase command.*/
|
|
// NONE
|
|
|
|
/** Output arguments for the Full Erase command.*/
|
|
// NONE
|
|
/** Input arguments for the Buffer Erase command.*/
|
|
struct {
|
|
|
|
/** Memory offset to be erase.*/
|
|
uint32_t memoryOffset;
|
|
|
|
} inputBufferErase;
|
|
|
|
/** Output arguments for the Buffer Erase command.*/
|
|
struct {
|
|
|
|
/** Memory offset to be erase.*/
|
|
uint32_t bytesErased;
|
|
|
|
} outputBufferErase;
|
|
} argument;
|
|
};
|
|
|
|
/*----------------------------------------------------------------------------
|
|
* Global variables
|
|
*----------------------------------------------------------------------------*/
|
|
|
|
/** End of program space (code + data). */
|
|
extern int end;
|
|
extern int _sstack;
|
|
/*----------------------------------------------------------------------------
|
|
* Local variables
|
|
*----------------------------------------------------------------------------*/
|
|
/** Global DMA driver instance for all DMA transfers in application. */
|
|
//static sDmad dmad;
|
|
|
|
/** SPI driver instance. */
|
|
//static Spid spid;
|
|
|
|
/** Serial flash driver instance. */
|
|
//static At25 at25;
|
|
|
|
/** Pins to configure for the application. */
|
|
//static Pin pins[] = {SPI_PINS};
|
|
|
|
/** Size of one page in the serial flash, in bytes. */
|
|
static uint32_t pageSize;
|
|
|
|
/** Size of one block in the serial flash, in bytes. */
|
|
static uint32_t blockSize;
|
|
|
|
/**Size of the buffer used for read/write operations in bytes. */
|
|
static uint32_t bufferSize;
|
|
|
|
/** Depending on DYN_TRACES, dwTraceLevel is a modifable runtime variable or a define */
|
|
uint32_t dwTraceLevel;
|
|
|
|
uint32_t Spi_init_done = 0 ;
|
|
struct _Mailbox *Pmailbox = 0 ;
|
|
|
|
uint8_t Spi_tx_buf[24] ;
|
|
uint8_t Spi_rx_buf[24] ;
|
|
|
|
uint8_t Block_buffer[4096] ;
|
|
|
|
/*----------------------------------------------------------------------------
|
|
* Global functions
|
|
*----------------------------------------------------------------------------*/
|
|
|
|
|
|
uint32_t eeprom_write_one( uint8_t byte, uint8_t count ) ;
|
|
uint32_t eeprom_read_status( void ) ;
|
|
void eeprom_write_enable( void ) ;
|
|
void eeprom_wait_busy( void ) ;
|
|
uint32_t spi_operation( register uint8_t *tx, register uint8_t *rx, register uint32_t count ) ;
|
|
void init_spi( void ) ;
|
|
uint32_t unprotect_eeprom( void ) ;
|
|
uint32_t spi_PDC_action( register uint8_t *command, register uint8_t *tx, register uint8_t *rx, register uint32_t comlen, register uint32_t count ) ;
|
|
uint32_t eeprom_block_erased( register uint8_t *p) ;
|
|
uint32_t eeprom_page_erased( register uint8_t *p) ;
|
|
void AT25D_Read( uint8_t *BufferAddr, uint32_t size, uint32_t memoryOffset) ;
|
|
void AT25D_Write( uint8_t *BufferAddr, uint32_t size, uint32_t memoryOffset ) ;
|
|
uint32_t AT25D_EraseBlock( uint32_t memoryOffset ) ;
|
|
//uint32_t min( uint32_t a, uint32_t b ) ;
|
|
|
|
|
|
uint32_t eeprom_write_one( uint8_t byte, uint8_t count )
|
|
{
|
|
register Spi *spiptr ;
|
|
register uint32_t result ;
|
|
|
|
spiptr = SPI ;
|
|
spiptr->SPI_CR = 1 ; // Enable
|
|
(void) spiptr->SPI_RDR ; // Dump any rx data
|
|
|
|
spiptr->SPI_TDR = byte ;
|
|
|
|
result = 0 ;
|
|
while( ( spiptr->SPI_SR & SPI_SR_RDRF ) == 0 )
|
|
{
|
|
// wait for received
|
|
if ( ++result > 10000 )
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
if ( count == 0 )
|
|
{
|
|
spiptr->SPI_CR = 2 ; // Disable
|
|
return spiptr->SPI_RDR ;
|
|
}
|
|
(void) spiptr->SPI_RDR ; // Dump the rx data
|
|
spiptr->SPI_TDR = 0 ;
|
|
result = 0 ;
|
|
while( ( spiptr->SPI_SR & SPI_SR_RDRF ) == 0 )
|
|
{
|
|
// wait for received
|
|
if ( ++result > 10000 )
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
spiptr->SPI_CR = 2 ; // Disable
|
|
return spiptr->SPI_RDR ;
|
|
}
|
|
|
|
|
|
uint32_t eeprom_read_status()
|
|
{
|
|
return eeprom_write_one( 5, 1 ) ;
|
|
}
|
|
|
|
|
|
void eeprom_write_enable()
|
|
{
|
|
eeprom_write_one( 6, 0 ) ;
|
|
}
|
|
|
|
void eeprom_wait_busy()
|
|
{
|
|
register uint32_t x ;
|
|
register uint32_t y ;
|
|
|
|
y = 0 ;
|
|
do
|
|
{
|
|
y += 1 ;
|
|
if ( y > 1000000 )
|
|
{
|
|
break ;
|
|
}
|
|
x = eeprom_read_status() ;
|
|
} while ( x & 1 ) ;
|
|
|
|
Pmailbox->argument.outputWrite.debug2 = y ;
|
|
}
|
|
|
|
uint32_t spi_operation( register uint8_t *tx, register uint8_t *rx, register uint32_t count )
|
|
{
|
|
register Spi *spiptr ;
|
|
register uint32_t result ;
|
|
|
|
// PMC->PMC_PCER0 |= 0x00200000L ; // Enable peripheral clock to SPI
|
|
|
|
result = 0 ;
|
|
spiptr = SPI ;
|
|
spiptr->SPI_CR = 1 ; // Enable
|
|
(void) spiptr->SPI_RDR ; // Dump any rx data
|
|
while( count )
|
|
{
|
|
result = 0 ;
|
|
while( ( spiptr->SPI_SR & SPI_SR_TXEMPTY ) == 0 )
|
|
{
|
|
// wait
|
|
if ( ++result > 10000 )
|
|
{
|
|
result = 0xFFFF ;
|
|
break ;
|
|
}
|
|
}
|
|
if ( result > 10000 )
|
|
{
|
|
break ;
|
|
}
|
|
// if ( count == 1 )
|
|
// {
|
|
// spiptr->SPI_CR = SPI_CR_LASTXFER ; // LastXfer bit
|
|
// }
|
|
spiptr->SPI_TDR = *tx++ ;
|
|
result = 0 ;
|
|
while( ( spiptr->SPI_SR & SPI_SR_RDRF ) == 0 )
|
|
{
|
|
// wait for received
|
|
if ( ++result > 10000 )
|
|
{
|
|
result = 0x2FFFF ;
|
|
break ;
|
|
}
|
|
}
|
|
if ( result > 10000 )
|
|
{
|
|
break ;
|
|
}
|
|
*rx++ = spiptr->SPI_RDR ;
|
|
count -= 1 ;
|
|
}
|
|
if ( result <= 10000 )
|
|
{
|
|
result = 0 ;
|
|
}
|
|
spiptr->SPI_CR = 2 ; // Disable
|
|
|
|
// Power save
|
|
// PMC->PMC_PCER0 &= ~0x00200000L ; // Disable peripheral clock to SPI
|
|
|
|
return result ;
|
|
}
|
|
|
|
uint32_t spi_PDC_action( register uint8_t *command, register uint8_t *tx, register uint8_t *rx, register uint32_t comlen, register uint32_t count )
|
|
{
|
|
register Spi *spiptr ;
|
|
// register uint32_t result ;
|
|
register uint32_t condition ;
|
|
static uint8_t discard_rx_command[4] ;
|
|
|
|
// PMC->PMC_PCER0 |= 0x00200000L ; // Enable peripheral clock to SPI
|
|
|
|
// Spi_complete = 0 ;
|
|
if ( comlen > 4 )
|
|
{
|
|
// Spi_complete = 1 ;
|
|
return 0x4FFFF ;
|
|
}
|
|
condition = SPI_SR_TXEMPTY ;
|
|
spiptr = SPI ;
|
|
spiptr->SPI_CR = 1 ; // Enable
|
|
(void) spiptr->SPI_RDR ; // Dump any rx data
|
|
(void) spiptr->SPI_SR ; // Clear error flags
|
|
spiptr->SPI_RPR = (uint32_t)discard_rx_command ;
|
|
spiptr->SPI_RCR = comlen ;
|
|
if ( rx )
|
|
{
|
|
spiptr->SPI_RNPR = (uint32_t)rx ;
|
|
spiptr->SPI_RNCR = count ;
|
|
condition = SPI_SR_RXBUFF ;
|
|
}
|
|
spiptr->SPI_TPR = (uint32_t)command ;
|
|
spiptr->SPI_TCR = comlen ;
|
|
if ( tx )
|
|
{
|
|
spiptr->SPI_TNPR = (uint32_t)tx ;
|
|
}
|
|
else
|
|
{
|
|
spiptr->SPI_TNPR = (uint32_t)rx ;
|
|
}
|
|
spiptr->SPI_TNCR = count ;
|
|
|
|
spiptr->SPI_PTCR = SPI_PTCR_RXTEN | SPI_PTCR_TXTEN ; // Start transfers
|
|
|
|
// Wait for things to get started, avoids early interrupt
|
|
for ( count = 0 ; count < 1000 ; count += 1 )
|
|
{
|
|
if ( ( spiptr->SPI_SR & SPI_SR_TXEMPTY ) == 0 )
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
|
|
count = 0 ;
|
|
while( ( spiptr->SPI_SR & condition ) == 0 )
|
|
{
|
|
if ( ++count > 1000000 )
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
|
|
spiptr->SPI_CR = 2 ; // Disable
|
|
(void) spiptr->SPI_RDR ; // Dump any rx data
|
|
(void) spiptr->SPI_SR ; // Clear error flags
|
|
spiptr->SPI_PTCR = SPI_PTCR_RXTDIS | SPI_PTCR_TXTDIS ; // Stop tramsfers
|
|
|
|
if ( count > 1000000 )
|
|
{
|
|
return 1 ;
|
|
}
|
|
return 0 ;
|
|
}
|
|
|
|
|
|
uint32_t unprotect_eeprom()
|
|
{
|
|
register uint8_t *p ;
|
|
|
|
eeprom_write_enable() ;
|
|
|
|
p = Spi_tx_buf ;
|
|
*p = 0x39 ; // Unprotect sector command
|
|
*(p+1) = 0 ;
|
|
*(p+2) = 0 ;
|
|
*(p+3) = 0 ; // 3 bytes address
|
|
|
|
return spi_operation( p, Spi_rx_buf, 4 ) ;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void init_spi()
|
|
{
|
|
register Pio *pioptr ;
|
|
register Spi *spiptr ;
|
|
register uint32_t timer ;
|
|
register uint8_t *p ;
|
|
uint8_t spi_buf[4] ;
|
|
|
|
if ( !Spi_init_done)
|
|
{
|
|
PMC->PMC_PCER0 |= 0x00200000L ; // Enable peripheral clock to SPI
|
|
/* Configure PIO */
|
|
pioptr = PIOA ;
|
|
pioptr->PIO_ABCDSR[0] &= ~0x00007800 ; // Peripheral A bits 14,13,12,11
|
|
pioptr->PIO_ABCDSR[1] &= ~0x00007800 ; // Peripheral A
|
|
pioptr->PIO_PDR = 0x00007800 ; // Assign to peripheral
|
|
|
|
spiptr = SPI ;
|
|
timer = ( 64000000 / 3000000 ) << 8 ;
|
|
spiptr->SPI_MR = 0x14000011 ; // 0001 0100 0000 0000 0000 0000 0001 0001 Master
|
|
spiptr->SPI_CSR[0] = 0x01180009 | timer ; // 0000 0001 0001 1000 xxxx xxxx 0000 1001
|
|
// NVIC_EnableIRQ(SPI_IRQn) ;
|
|
|
|
p = spi_buf ;
|
|
|
|
// *p = 0x39 ; // Unprotect sector command
|
|
// *(p+1) = 0 ;
|
|
// *(p+2) = 0 ;
|
|
// *(p+3) = 0 ; // 3 bytes address
|
|
|
|
// spi_operation( p, spi_buf, 4 ) ;
|
|
|
|
eeprom_write_enable() ;
|
|
|
|
*p = 1 ; // Write status register command
|
|
*(p+1) = 0 ;
|
|
spi_operation( p, spi_buf, 2 ) ;
|
|
Spi_init_done = 1 ;
|
|
}
|
|
|
|
}
|
|
|
|
uint32_t eeprom_block_erased( register uint8_t *p)
|
|
{
|
|
register uint32_t x ;
|
|
register uint32_t result ;
|
|
|
|
result = 1 ;
|
|
|
|
for ( x = 0 ; x < 4096 ; x += 1 )
|
|
{
|
|
if ( *p++ != 0xFF )
|
|
{
|
|
result = 0 ;
|
|
break ;
|
|
}
|
|
}
|
|
return result ;
|
|
}
|
|
|
|
uint32_t eeprom_page_erased( register uint8_t *p)
|
|
{
|
|
register uint32_t x ;
|
|
register uint32_t result ;
|
|
|
|
result = 1 ;
|
|
|
|
for ( x = 0 ; x < 256 ; x += 1 )
|
|
{
|
|
if ( *p++ != 0xFF )
|
|
{
|
|
result = 0 ;
|
|
break ;
|
|
}
|
|
}
|
|
return result ;
|
|
}
|
|
|
|
|
|
void AT25D_Read( uint8_t *BufferAddr, uint32_t size, uint32_t memoryOffset)
|
|
{
|
|
register uint8_t *p ;
|
|
|
|
p = Spi_tx_buf ;
|
|
*p = 3 ; // Read command
|
|
*(p+1) = memoryOffset >> 16 ;
|
|
*(p+2) = memoryOffset >> 8 ;
|
|
*(p+3) = memoryOffset ; // 3 bytes address
|
|
|
|
spi_PDC_action( p, 0, BufferAddr, 4, size ) ;
|
|
}
|
|
|
|
void AT25D_Write( uint8_t *BufferAddr, uint32_t size, uint32_t memoryOffset )
|
|
{
|
|
register uint8_t *p ;
|
|
|
|
eeprom_write_enable() ;
|
|
|
|
p = Spi_tx_buf ;
|
|
*p = 2 ; // Write command
|
|
*(p+1) = memoryOffset >> 16 ;
|
|
*(p+2) = memoryOffset >> 8 ;
|
|
*(p+3) = memoryOffset ; // 3 bytes address
|
|
|
|
spi_PDC_action( p, BufferAddr, 0, 4, size ) ;
|
|
|
|
eeprom_wait_busy() ;
|
|
|
|
}
|
|
|
|
uint32_t AT25D_EraseBlock( uint32_t memoryOffset )
|
|
{
|
|
register uint8_t *p ;
|
|
register uint32_t x ;
|
|
|
|
eeprom_write_enable() ;
|
|
p = Spi_tx_buf ;
|
|
*p = 0x20 ; // Block Erase command
|
|
*(p+1) = memoryOffset >> 16 ;
|
|
*(p+2) = memoryOffset >> 8 ;
|
|
*(p+3) = memoryOffset ; // 3 bytes address
|
|
x = spi_operation( p, Spi_rx_buf, 4 ) ;
|
|
|
|
eeprom_wait_busy() ;
|
|
return x ;
|
|
}
|
|
|
|
//uint32_t min( uint32_t a, uint32_t b )
|
|
//{
|
|
// if ( a < b )
|
|
// {
|
|
// return a ;
|
|
// }
|
|
// return b ;
|
|
//}
|
|
|
|
/**
|
|
* \brief Applet main entry. This function decodes received command and executes it.
|
|
* \param argc always 1
|
|
* \param argv Address of the argument area.
|
|
*/
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
struct _Mailbox *pMailbox = (struct _Mailbox *) argv;
|
|
Pmailbox = pMailbox ;
|
|
/* Communication type with SAM-BA GUI. */
|
|
// uint8_t comType;
|
|
|
|
// uint32_t jedecId;
|
|
uint32_t bytesToWrite, bytesToRead, bufferAddr, startMemoryOffset, memoryOffset, packetSize;
|
|
/* index on read/write buffer */
|
|
uint8_t *pBuffer;
|
|
/* Temporary buffer used for non block aligned read/write */
|
|
// uint32_t tempBufferAddr;
|
|
/* Offset in destination buffer during buffer copy */
|
|
// uint32_t bufferOffset;
|
|
|
|
uint32_t block_address ;
|
|
uint32_t x ;
|
|
/* INIT */
|
|
/* Save communication link type */
|
|
// comType = pMailbox->argument.inputInit.comType;
|
|
|
|
init_spi() ;
|
|
|
|
if (pMailbox->command == APPLET_CMD_INIT)
|
|
{
|
|
#if (DYN_TRACES == 1)
|
|
dwTraceLevel = pMailbox->argument.inputInit.traceLevel;
|
|
#endif
|
|
|
|
// TRACE_INFO("-- SerialFlash AT25/AT26 applet %s --\n\r", SAM_BA_APPLETS_VERSION);
|
|
// TRACE_INFO("-- %s\n\r", BOARD_NAME);
|
|
// TRACE_INFO("-- Compiled: %s %s --\n\r", __DATE__, __TIME__);
|
|
/* Configure pins */
|
|
// PIO_Configure(pins, PIO_LISTSIZE(pins));
|
|
/* Initialize DMA driver instance with polling mode */
|
|
// DMAD_Initialize( &dmad, 1 );
|
|
|
|
/* Initialize the SPI and serial flash */
|
|
init_spi() ;
|
|
|
|
|
|
|
|
// SPID_Configure(&spid, SPI0, ID_SPI0, &dmad);
|
|
// AT25_Configure(&at25, &spid, SPI_CS, 1);
|
|
// TRACE_INFO("SPI and AT25/AT25 drivers initialized\n\r");
|
|
pMailbox->argument.outputInit.bufferAddress = (uint32_t) &end ;
|
|
/* Read the JEDEC ID of the device to identify it */
|
|
// jedecId = AT25D_ReadJedecId(&at25);
|
|
// if (AT25_FindDevice(&at25, jedecId) == 0) {
|
|
// pMailbox->status = APPLET_DEV_UNKNOWN;
|
|
// pMailbox->argument.outputInit.bufferSize = 0;
|
|
// pMailbox->argument.outputInit.memorySize = 0;
|
|
// TRACE_INFO("Device Unknown\n\r");
|
|
// goto exit;
|
|
// }
|
|
// else {
|
|
/* Get device parameters */
|
|
pMailbox->status = APPLET_SUCCESS;
|
|
pageSize = 256 ;
|
|
// pageSize = AT25_PageSize(&at25);
|
|
blockSize = 4096 ;
|
|
// blockSize = AT25_BlockSize(&at25);
|
|
|
|
/* Program page */
|
|
// if (AT25_ManId(&at25) == SST_SPI_FLASH) {
|
|
/* SST Flash write is slower, we reduce buffer size to avoid USB timeout */
|
|
// bufferSize = 10 * pageSize;
|
|
// }
|
|
// else {
|
|
bufferSize = 2 * blockSize;
|
|
// }
|
|
/* integer number of pages can be contained in each buffer */
|
|
bufferSize -= bufferSize % pageSize;
|
|
if ( bufferSize < pageSize) {
|
|
// TRACE_INFO("No enought memory to load buffer.\n\r");
|
|
goto exit;
|
|
}
|
|
pMailbox->argument.outputInit.bufferSize = bufferSize;
|
|
pMailbox->argument.outputInit.memorySize = 128*4096;
|
|
// TRACE_INFO("%s blockSize : 0x%lx bufferAddr : 0x%lx\n\r",
|
|
// "ersky9x", blockSize, pMailbox->argument.outputInit.bufferAddress);
|
|
// }
|
|
}
|
|
|
|
// ----------------------------------------------------------
|
|
// WRITE:
|
|
// ----------------------------------------------------------
|
|
else if (pMailbox->command == APPLET_CMD_WRITE)
|
|
{
|
|
startMemoryOffset = pMailbox->argument.inputWrite.memoryOffset;
|
|
memoryOffset = startMemoryOffset;
|
|
bufferAddr = pMailbox->argument.inputWrite.bufferAddr;
|
|
// tempBufferAddr = bufferAddr + bufferSize;
|
|
bytesToWrite = pMailbox->argument.inputWrite.bufferSize;
|
|
// TRACE_INFO("WRITE at offset: 0x%lx buffer at : 0x%lx of: 0x%lx Bytes\n\r",
|
|
// memoryOffset, bufferAddr, bytesToWrite);
|
|
/* Check word alignment */
|
|
if (memoryOffset % 4) {
|
|
|
|
pMailbox->status = APPLET_ALIGN_ERROR;
|
|
goto exit;
|
|
}
|
|
|
|
unprotect_eeprom() ;
|
|
// if (AT25D_Unprotect(&at25)) {
|
|
|
|
// TRACE_INFO("Can not unprotect the flash\n\r");
|
|
// pMailbox->status = APPLET_UNPROTECT_FAIL;
|
|
// goto exit;
|
|
// }
|
|
|
|
pBuffer = (uint8_t *) bufferAddr;
|
|
|
|
block_address = memoryOffset &0xFFFFF000 ; // 4k boundary
|
|
AT25D_Read( Block_buffer, 4096, block_address ) ; // read block to write to
|
|
// Check to see if it is blank
|
|
x = eeprom_block_erased( Block_buffer ) ;
|
|
if ( x == 0 )
|
|
{
|
|
AT25D_EraseBlock( block_address ) ;
|
|
}
|
|
|
|
if (bytesToWrite)
|
|
{
|
|
uint32_t bytes_to_copy ;
|
|
uint32_t i ;
|
|
uint32_t j ;
|
|
uint8_t *s ;
|
|
uint8_t *dest ;
|
|
|
|
dest = Block_buffer + (memoryOffset & 0x0FFF ) ;
|
|
s = pBuffer ;
|
|
bytes_to_copy = 4096 - ( memoryOffset - block_address ) ;
|
|
if ( bytes_to_copy > bytesToWrite )
|
|
{
|
|
bytes_to_copy = bytesToWrite ;
|
|
}
|
|
for ( i = 0 ; i < bytes_to_copy ; i += 1 )
|
|
{
|
|
*dest++ = *s++ ;
|
|
}
|
|
memoryOffset += bytes_to_copy ;
|
|
bytesToWrite -= bytes_to_copy ;
|
|
s = Block_buffer ;
|
|
j = 0 ;
|
|
for ( i = 0 ; i < 16 ; i += 1 ) // pages in block
|
|
{
|
|
j <<= 1 ;
|
|
x = eeprom_page_erased( s ) ;
|
|
if ( x == 0 ) // Not blank
|
|
{
|
|
AT25D_Write( s, 256, block_address ) ;
|
|
j |= 1 ;
|
|
}
|
|
s += 256 ;
|
|
block_address += 256 ;
|
|
}
|
|
pMailbox->argument.outputWrite.debug1 = j ;
|
|
}
|
|
|
|
|
|
// if ((memoryOffset % pageSize) != 0) {
|
|
|
|
// /* We are not page aligned, retrieve first page content to update it*/
|
|
// if (memoryOffset < writtenAddress) {
|
|
// lastErasedBlock = 0xFFFF;
|
|
// }
|
|
// /* Flush temp buffer */
|
|
// memset((uint32_t *)tempBufferAddr, 0xFF, pageSize);
|
|
|
|
// bufferOffset = (memoryOffset % pageSize);
|
|
// packetSize = pageSize - bufferOffset;
|
|
// memoryOffset -= bufferOffset;
|
|
// /* Read page to be updated*/
|
|
// AT25D_Read( (uint8_t *) tempBufferAddr, pageSize, memoryOffset);
|
|
// /* Fill retrieved page with data to be programmed */
|
|
// memcpy((uint8_t *)(tempBufferAddr + bufferOffset), pBuffer, packetSize);
|
|
|
|
// if (((memoryOffset / blockSize) > lastErasedBlock) || (lastErasedBlock == 0xFFFF)) {
|
|
// /* Erase the block to be updated */
|
|
//// AT25D_EraseBlock( memoryOffset);
|
|
// lastErasedBlock = (memoryOffset / blockSize);
|
|
// }
|
|
|
|
// /* Write the page contents */
|
|
//// AT25D_Write( (uint8_t *) tempBufferAddr, pageSize, memoryOffset);
|
|
// bytesToWrite = (bytesToWrite > packetSize) ? (bytesToWrite - packetSize) : 0;
|
|
// pBuffer += packetSize;
|
|
// memoryOffset += pageSize;
|
|
// writtenAddress = memoryOffset;
|
|
// }
|
|
|
|
// /* If it remains more than one page to write */
|
|
// while (bytesToWrite >= pageSize)
|
|
// {
|
|
// if (memoryOffset < writtenAddress)
|
|
// {
|
|
// lastErasedBlock = 0xFFFF;
|
|
// }
|
|
// if (((memoryOffset / blockSize) > lastErasedBlock) || (lastErasedBlock == 0xFFFF)) {
|
|
// /* Erase the block to be updated */
|
|
//// AT25D_EraseBlock( memoryOffset);
|
|
// lastErasedBlock = (memoryOffset / blockSize);
|
|
// }
|
|
// /* Write the page contents */
|
|
// AT25D_Write( (uint8_t *) pBuffer, pageSize, memoryOffset);
|
|
// pBuffer += pageSize;
|
|
// memoryOffset += pageSize;
|
|
// bytesToWrite -= pageSize;
|
|
// writtenAddress = memoryOffset;
|
|
// }
|
|
|
|
// /* Write remaining data */
|
|
// if (bytesToWrite > 0) {
|
|
|
|
// /* Read previous content of page */
|
|
// AT25D_Read( (uint8_t *) tempBufferAddr, pageSize, memoryOffset);
|
|
// /* Fill retrieved block with data to be programmed */
|
|
// memcpy((uint8_t *)tempBufferAddr, pBuffer, bytesToWrite);
|
|
// if (((memoryOffset / blockSize) > lastErasedBlock) || (lastErasedBlock == 0xFFFF)) {
|
|
// /* Erase the block to be updated */
|
|
//// AT25D_EraseBlock( memoryOffset);
|
|
// lastErasedBlock = (memoryOffset / blockSize);
|
|
// }
|
|
// /* Write the page contents */;
|
|
//// AT25D_Write( (uint8_t *) tempBufferAddr, pageSize, memoryOffset);
|
|
// writtenAddress = memoryOffset + bytesToWrite;
|
|
// /* No more bytes to write */
|
|
// bytesToWrite = 0;
|
|
// }
|
|
|
|
//// TRACE_INFO("WRITE return byte written : 0x%lx Bytes\n\r",
|
|
//// pMailbox->argument.inputWrite.bufferSize - bytesToWrite);
|
|
|
|
pMailbox->argument.outputWrite.bytesWritten = pMailbox->argument.inputWrite.bufferSize - bytesToWrite;
|
|
// pMailbox->argument.outputWrite.bytesWritten = bytesToWrite;
|
|
pMailbox->status = APPLET_SUCCESS;
|
|
}
|
|
|
|
// // ----------------------------------------------------------
|
|
// // READ:
|
|
// // ----------------------------------------------------------
|
|
else if (pMailbox->command == APPLET_CMD_READ)
|
|
{
|
|
memoryOffset = pMailbox->argument.inputRead.memoryOffset;
|
|
bufferAddr = pMailbox->argument.inputRead.bufferAddr;
|
|
bytesToRead = pMailbox->argument.inputRead.bufferSize;
|
|
|
|
// TRACE_INFO("READ at offset: 0x%lx buffer at : 0x%lx of: 0x%lx Bytes\n\r",
|
|
// memoryOffset, bufferAddr, bytesToRead);
|
|
/* Check word alignment */
|
|
if (memoryOffset % 4) {
|
|
|
|
pMailbox->status = APPLET_ALIGN_ERROR;
|
|
goto exit;
|
|
}
|
|
pBuffer = (uint8_t *) bufferAddr;
|
|
|
|
/* Read packet after packets */
|
|
while (((uint32_t)pBuffer < (bufferAddr + bufferSize)) && (bytesToRead > 0))
|
|
{
|
|
packetSize = bytesToRead ;
|
|
if ( packetSize > MAX_COUNT )
|
|
{
|
|
packetSize = MAX_COUNT ;
|
|
}
|
|
AT25D_Read( pBuffer, packetSize, memoryOffset);
|
|
pBuffer += packetSize;
|
|
bytesToRead -= packetSize;
|
|
memoryOffset += packetSize;
|
|
}
|
|
|
|
// TRACE_INFO("READ return byte read : 0x%lx Bytes\n\r",
|
|
// pMailbox->argument.inputRead.bufferSize - bytesToRead);
|
|
|
|
pMailbox->argument.outputRead.bytesRead = pMailbox->argument.inputRead.bufferSize - bytesToRead;
|
|
pMailbox->status = APPLET_SUCCESS;
|
|
}
|
|
|
|
// // ----------------------------------------------------------
|
|
// // FULL ERASE:
|
|
// // ----------------------------------------------------------
|
|
else if (pMailbox->command == APPLET_CMD_FULL_ERASE)
|
|
{
|
|
pMailbox->status = APPLET_ERASE_FAIL;
|
|
// TRACE_INFO("FULL ERASE\n\r");
|
|
|
|
// unprotect_eeprom() ;
|
|
|
|
// /* Unprotected the flash */
|
|
//// if (AT25D_Unprotect(&at25)) {
|
|
|
|
//// TRACE_INFO("Can not unprotect the flash\n\r");
|
|
//// pMailbox->status = APPLET_UNPROTECT_FAIL;
|
|
//// goto exit;
|
|
//// }
|
|
|
|
// TRACE_INFO("Flash unprotected\n\r");
|
|
|
|
// /* Erase the chip */
|
|
// TRACE_INFO("Chip is being erased...\n\r");
|
|
|
|
//// if (AT25D_EraseChip()) {
|
|
|
|
// // NOT IMPLEMENTED (YET?)
|
|
|
|
// TRACE_INFO("Erasing error\n\r");
|
|
// pMailbox->status = APPLET_ERASE_FAIL;
|
|
// goto exit;
|
|
//// }
|
|
|
|
// TRACE_INFO("Full Erase achieved\n\r");
|
|
// pMailbox->status = APPLET_SUCCESS;
|
|
}
|
|
// // ----------------------------------------------------------
|
|
// // BUFFER ERASE:
|
|
// // ----------------------------------------------------------
|
|
else if (pMailbox->command == APPLET_CMD_BUFFER_ERASE)
|
|
{
|
|
|
|
// TRACE_INFO("BUFFER ERASE \n\r");
|
|
/* Unprotected the flash */
|
|
unprotect_eeprom() ;
|
|
|
|
// if (AT25D_Unprotect(&at25)) {
|
|
|
|
// TRACE_INFO("Can not unprotect the flash\n\r");
|
|
// pMailbox->status = APPLET_UNPROTECT_FAIL;
|
|
// goto exit;
|
|
// }
|
|
|
|
memoryOffset = pMailbox->argument.inputBufferErase.memoryOffset;
|
|
|
|
if (AT25D_EraseBlock( memoryOffset))
|
|
{
|
|
pMailbox->status = APPLET_ERASE_FAIL;
|
|
// TRACE_INFO("Block erasing error\n\r");
|
|
goto exit;
|
|
}
|
|
pMailbox->argument.outputBufferErase.bytesErased = 4096;
|
|
|
|
// TRACE_INFO("Buffer Erase achieved\n\r");
|
|
pMailbox->status = APPLET_SUCCESS;
|
|
}
|
|
exit:
|
|
/* Acknowledge the end of command */
|
|
// TRACE_INFO("\tEnd of applet (command : %lx --- status : %lx)\n\r", pMailbox->command, pMailbox->status);
|
|
|
|
/* Notify the host application of the end of the command processing */
|
|
pMailbox->command = ~(pMailbox->command);
|
|
/* Send ACK character */
|
|
// if (comType == DBGU_COM_TYPE)
|
|
// {
|
|
/* Wait for the transmitter to be ready */
|
|
// while ( (DBGU->DBGU_SR & DBGU_SR_TXEMPTY) == 0 ) ;
|
|
/* Send character */
|
|
// DBGU->DBGU_THR= 0x06 ;
|
|
// }
|
|
return 0;
|
|
}
|
|
|