2014-02-02 16:48:18 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 INRIA
|
|
|
|
*
|
|
|
|
* The source code is licensed under the LGPLv2 license,
|
|
|
|
* See the file LICENSE for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup cpu
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
* @brief lpc iap driver
|
|
|
|
* @note based on iap driver for LPC2148 Controller made by Andreas Weschenfelder, 2008
|
|
|
|
*
|
|
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stddef.h>
|
2010-12-02 15:38:28 +01:00
|
|
|
/* iap driver
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
*/
|
|
|
|
|
2013-12-16 17:54:58 +01:00
|
|
|
#include "irq.h"
|
|
|
|
#include "flashrom.h"
|
|
|
|
#include "iap.h"
|
2013-04-01 00:18:07 +02:00
|
|
|
/* TODO: replace by a lpc generic header */
|
2013-12-16 17:54:58 +01:00
|
|
|
#include "lpc2387.h"
|
2010-12-02 15:38:28 +01:00
|
|
|
|
2013-07-24 00:36:06 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
2013-12-16 17:54:58 +01:00
|
|
|
#include "debug.h"
|
2010-12-02 15:38:28 +01:00
|
|
|
|
|
|
|
/* pointer to reserved flash rom section for configuration data */
|
2013-06-21 03:52:57 +02:00
|
|
|
__attribute((aligned(256))) char configmem[256] __attribute__((section(".configmem")));
|
2010-12-02 15:38:28 +01:00
|
|
|
|
|
|
|
static unsigned int iap_command[5]; // contains parameters for IAP command
|
|
|
|
static unsigned int iap_result[2]; // contains results
|
2013-06-21 03:52:57 +02:00
|
|
|
typedef void (*IAP)(unsigned int[], unsigned int[]); // typedefinition for IAP entry function
|
2010-12-02 15:38:28 +01:00
|
|
|
IAP IAP_Entry;
|
|
|
|
|
|
|
|
/* some function prototypes */
|
|
|
|
static uint32_t blank_check_sector(uint32_t tmp_sect1, uint32_t tmp_sect2);
|
|
|
|
static uint32_t prepare_sectors(uint32_t tmp_sect1, uint32_t tmp_sect2);
|
|
|
|
static uint32_t erase_sectors(uint32_t tmp_sect1, uint32_t tmp_sect2);
|
|
|
|
static uint32_t copy_ram_to_flash(uint32_t tmp_adr_dst, uint32_t tmp_adr_src, uint32_t tmp_size);
|
|
|
|
static uint32_t compare(uint32_t tmp_adr_dst, uint32_t tmp_adr_src, uint32_t tmp_size);
|
|
|
|
static uint32_t iap(uint32_t code, uint32_t p1, uint32_t p2, uint32_t p3, uint32_t p4);
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* P U B L I C F U N C T I O N S
|
|
|
|
*****************************************************************************/
|
2013-06-21 03:52:57 +02:00
|
|
|
uint8_t flashrom_write(uint8_t *dst, char *src, size_t size)
|
|
|
|
{
|
2014-04-30 20:34:14 +02:00
|
|
|
(void) size; /* unused */
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
char err;
|
|
|
|
unsigned intstate;
|
|
|
|
uint8_t sec;
|
|
|
|
|
|
|
|
sec = iap_get_sector((uint32_t) dst);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (sec == INVALID_ADDRESS) {
|
2010-12-02 15:38:28 +01:00
|
|
|
DEBUG("Invalid address\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check sector */
|
2013-06-24 22:37:35 +02:00
|
|
|
if (blank_check_sector(sec, sec) == SECTOR_NOT_BLANK) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("Warning: Sector %i not blank\n", sec);
|
|
|
|
}
|
2010-12-02 15:38:28 +01:00
|
|
|
|
|
|
|
/* prepare sector */
|
2013-06-21 03:52:57 +02:00
|
|
|
err = prepare_sectors(sec, sec);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (err) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("\n-- ERROR: PREPARE_SECTOR_FOR_WRITE_OPERATION: %u\n", err);
|
|
|
|
return 0;
|
|
|
|
}
|
2010-12-02 15:38:28 +01:00
|
|
|
/* write flash */
|
2013-06-21 03:52:57 +02:00
|
|
|
else {
|
2010-12-02 15:38:28 +01:00
|
|
|
intstate = disableIRQ();
|
2013-06-21 03:52:57 +02:00
|
|
|
err = copy_ram_to_flash((uint32_t) dst, (uint32_t) src, 256);
|
2010-12-02 15:38:28 +01:00
|
|
|
restoreIRQ(intstate);
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (err) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("ERROR: COPY_RAM_TO_FLASH: %u\n", err);
|
2010-12-02 15:38:28 +01:00
|
|
|
/* set interrupts back and return */
|
|
|
|
restoreIRQ(intstate);
|
2013-06-21 03:52:57 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2010-12-02 15:38:28 +01:00
|
|
|
/* check result */
|
2013-06-21 03:52:57 +02:00
|
|
|
else {
|
|
|
|
err = compare((uint32_t) dst, (uint32_t) src, 256);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (err) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("ERROR: COMPARE: %i (at position %u)\n", err, iap_result[1]);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG("Data successfully written!\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
uint8_t flashrom_erase(uint8_t *addr)
|
|
|
|
{
|
2010-12-03 22:22:58 +01:00
|
|
|
uint8_t sec = iap_get_sector((uint32_t) addr);
|
2010-12-02 15:38:28 +01:00
|
|
|
unsigned intstate;
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (sec == INVALID_ADDRESS) {
|
2010-12-02 15:38:28 +01:00
|
|
|
DEBUG("Invalid address\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check sector */
|
2013-06-24 22:37:35 +02:00
|
|
|
if (!blank_check_sector(sec, sec)) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("Sector already blank!\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* prepare sector */
|
2013-06-24 22:37:35 +02:00
|
|
|
if (prepare_sectors(sec, sec)) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("-- ERROR: PREPARE_SECTOR_FOR_WRITE_OPERATION --\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-12-02 15:38:28 +01:00
|
|
|
intstate = disableIRQ();
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2010-12-02 15:38:28 +01:00
|
|
|
/* erase sector */
|
2013-06-24 22:37:35 +02:00
|
|
|
if (erase_sectors(sec, sec)) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("-- ERROR: ERASE SECTOR --\n");
|
2010-12-02 15:38:28 +01:00
|
|
|
restoreIRQ(intstate);
|
2013-06-21 03:52:57 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-12-02 15:38:28 +01:00
|
|
|
restoreIRQ(intstate);
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2010-12-02 15:38:28 +01:00
|
|
|
/* check again */
|
2013-06-24 22:37:35 +02:00
|
|
|
if (blank_check_sector(sec, sec)) {
|
2013-06-21 03:52:57 +02:00
|
|
|
DEBUG("-- ERROR: BLANK_CHECK_SECTOR\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG("Sector successfully erased.\n");
|
|
|
|
return 1;
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* PRIVATE FUNCTIONS
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
static uint32_t iap(uint32_t code, uint32_t p1, uint32_t p2, uint32_t p3, uint32_t p4)
|
|
|
|
{
|
|
|
|
iap_command[0] = code; // set command code
|
|
|
|
iap_command[1] = p1; // set 1st param
|
|
|
|
iap_command[2] = p2; // set 2nd param
|
|
|
|
iap_command[3] = p3; // set 3rd param
|
|
|
|
iap_command[4] = p4; // set 4th param
|
|
|
|
|
|
|
|
((void (*)())0x7ffffff1)(iap_command, iap_result); // IAP entry point
|
|
|
|
return *iap_result;
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* Function: blank_check_sector
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
|
|
|
* Description: This command is used to blank check a sector or multiple sectors
|
2010-12-02 15:38:28 +01:00
|
|
|
* of on-chip Flash memory. To blank check a single sector use the
|
|
|
|
* same "Start" and "End" sector numbers.
|
|
|
|
* Command: 53
|
|
|
|
* Param0: Start Sector Number
|
2013-06-21 03:52:57 +02:00
|
|
|
* Param1: End Sector Number (should be greater than equal to the start
|
2010-12-02 15:38:28 +01:00
|
|
|
* sector number)
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Parameters: long tmp_sect1: Param0
|
|
|
|
* long tmp_sect2: Param1
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Return: Code CMD_SUCCESS |
|
|
|
|
* BUSY |
|
|
|
|
* SECTOR_NOT_BLANK |
|
|
|
|
* INVALID_SECTOR
|
|
|
|
* Result0: Offset of the first non blank word location if the status code is SECTOR_NOT_BLANK.
|
|
|
|
* Result1: Contents of non blank wird location.
|
|
|
|
*****************************************************************************/
|
2013-06-21 03:52:57 +02:00
|
|
|
uint32_t blank_check_sector(uint32_t tmp_sect1, uint32_t tmp_sect2)
|
|
|
|
{
|
|
|
|
return iap(BLANK_CHECK_SECTOR, tmp_sect1, tmp_sect2, 0 , 0);
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* Function: copy_ram_to_flash
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
|
|
|
* Description: This command is used to programm the flash memory. the affected should be
|
|
|
|
* prepared first by calling "Prepare Sector for Write Operation" command. the
|
2010-12-02 15:38:28 +01:00
|
|
|
* affected sectors are automatically protected again once the copy command is
|
|
|
|
* successfully executed. the boot sector cannot be written by this command.
|
|
|
|
* Command: 51
|
2013-06-21 03:52:57 +02:00
|
|
|
* Param0: (DST) Destination Flash adress where data bytes are to be written.
|
2010-12-02 15:38:28 +01:00
|
|
|
* This address should be a 512 byte boundary.
|
|
|
|
* Param1: (SRC) Source RAM adress from which data byre are to be read.
|
|
|
|
* Param2: Number of bytes to be written. Should be 512 | 1024 | 4096 | 8192.
|
|
|
|
* Param3: System Clock Frequency (CCLK) in KHz.
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Parameters: long tmp_adr_dst: Param0
|
|
|
|
* long tmp_adr_src: Param1
|
|
|
|
* long tmp_size: Param2
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Return: Code CMD_SUCCESS |
|
|
|
|
* SRC_ADDR_ERROR (Address not on word boundary) |
|
|
|
|
* DST_ADDR_ERROR (Address not on correct boundary) |
|
|
|
|
* SRC_ADDR_NOT_MAPPED |
|
|
|
|
* DST_ADDR_NOT_MAPPED |
|
|
|
|
* COUNT_ERROR (Byte count is not 512 | 1024 | 4096 | 8192) |
|
|
|
|
* SECTOR_NOT_PREPARED_FOR_WRITE_OPERATION |
|
|
|
|
* BUSY
|
|
|
|
*****************************************************************************/
|
2013-06-21 03:52:57 +02:00
|
|
|
uint32_t copy_ram_to_flash(uint32_t tmp_adr_dst, uint32_t tmp_adr_src, uint32_t tmp_size)
|
|
|
|
{
|
|
|
|
return iap(COPY_RAM_TO_FLASH, tmp_adr_dst, tmp_adr_src, tmp_size, _XTAL);
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* Function: Prepare_Sector
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Description: This command must be executed before executing "Copy RAM to Flash" or "Erase Sector(s)"
|
|
|
|
* command. Successful execution of the "Copy RAM to Flash" or "Erase Sector(s)" command causes
|
|
|
|
* relevant sectors to be protected again. The boot sector can not be prepared by this command. To
|
|
|
|
* prepare a single sector use the same "Start" and "End" sector numbers..
|
|
|
|
* Command code: 50
|
|
|
|
* Param0: Start Sector Number
|
|
|
|
* Param1: End Sector Number: Should be greater than or equal to start sector number.
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Parameters: long tmp_sect1: Param0
|
|
|
|
* long tmp_sect2: Param1
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Return: Code CMD_SUCCESS |
|
|
|
|
* BUSY |
|
|
|
|
* INVALID_SECTOR
|
|
|
|
*****************************************************************************/
|
2013-06-21 03:52:57 +02:00
|
|
|
uint32_t prepare_sectors(uint32_t tmp_sect1, uint32_t tmp_sect2)
|
|
|
|
{
|
|
|
|
return iap(PREPARE_SECTOR_FOR_WRITE_OPERATION, tmp_sect1, tmp_sect2, 0 , 0);
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* Function: erase_sectors
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Description: This command is used to erase a sector or multiple sectors of on-chip Flash memory. The boot
|
|
|
|
* sector can not be erased by this command. To erase a single sector use the same "Start" and "End"
|
|
|
|
* sector numbers.
|
|
|
|
* Command code: 52
|
|
|
|
* Param0: Start Sector Number
|
|
|
|
* Param1: End Sector Number: Should be greater than or equal to start sector number.
|
|
|
|
* Param2: System Clock Frequency (CCLK) in KHz.
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Parameters: long tmp_sect1: Param0
|
|
|
|
* long tmp_sect2: Param1
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Return: Code CMD_SUCCESS |
|
|
|
|
* BUSY |
|
|
|
|
* SECTOR_NOT_PREPARED_FOR_WRITE_OPERATION |
|
|
|
|
* INVALID_SECTOR
|
|
|
|
*****************************************************************************/
|
2013-06-21 03:52:57 +02:00
|
|
|
uint32_t erase_sectors(uint32_t tmp_sect1, uint32_t tmp_sect2)
|
|
|
|
{
|
|
|
|
return iap(ERASE_SECTOR, tmp_sect1, tmp_sect2, _XTAL, 0);
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* Function: compare
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Description: This command is used to compare the memory contents at two locations. compare result may not
|
|
|
|
* be correct when source or destination address contains any of the first 64 bytes starting
|
|
|
|
* from address zero. First 64 bytes can be re-mapped to RAM.
|
|
|
|
* Command Code: 56
|
|
|
|
* Param0(DST): Starting Flash or RAM address from where data bytes are to be
|
|
|
|
* address should be a word boundary.
|
|
|
|
* Param1(SRC): Starting Flash or RAM address from where data bytes are to be
|
|
|
|
* address should be a word boundary.
|
|
|
|
* Param2: Number of bytes to be compared. Count should be in multiple of 4.
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Parameters: long tmp_adr_dst
|
2013-06-21 03:52:57 +02:00
|
|
|
* long tmp_adr_src
|
2010-12-02 15:38:28 +01:00
|
|
|
* long tmp_size
|
2013-06-21 03:52:57 +02:00
|
|
|
*
|
2010-12-02 15:38:28 +01:00
|
|
|
* Return: Code CMD_SUCCESS |
|
|
|
|
* COMPARE_ERROR |
|
|
|
|
* COUNT_ERROR (Byte count is not multiple of 4) |
|
|
|
|
* ADDR_ERROR |
|
|
|
|
* ADDR_NOT_MAPPED
|
|
|
|
* Result0: Offset of the first mismatch if the Status Code is COMPARE_ERROR.
|
|
|
|
*****************************************************************************/
|
2013-06-21 03:52:57 +02:00
|
|
|
uint32_t compare(uint32_t tmp_adr_dst, uint32_t tmp_adr_src, uint32_t tmp_size)
|
|
|
|
{
|
|
|
|
return iap(COMPARE, tmp_adr_dst, tmp_adr_src, tmp_size, 0);
|
2010-12-02 15:38:28 +01:00
|
|
|
}
|