You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
251 lines
6.3 KiB
C
251 lines
6.3 KiB
C
/*
|
|
* This file is part of the EasyFlash Library.
|
|
*
|
|
* Copyright (c) 2015, Armink, <armink.ztl@gmail.com>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Function: IAP(In-Application Programming) operating interface.
|
|
* Created on: 2015-01-05
|
|
*/
|
|
|
|
#include "flash.h"
|
|
|
|
/* IAP section backup application section start address in flash */
|
|
static uint32_t bak_app_start_addr = NULL;
|
|
|
|
static uint32_t get_bak_app_start_addr(void);
|
|
|
|
/**
|
|
* Flash IAP function initialize.
|
|
*
|
|
* @param start_addr IAP section backup application section start address in flash
|
|
*
|
|
* @return result
|
|
*/
|
|
FlashErrCode flash_iap_init(uint32_t start_addr) {
|
|
FlashErrCode result = FLASH_NO_ERR;
|
|
|
|
FLASH_ASSERT(start_addr);
|
|
|
|
bak_app_start_addr = start_addr;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Erase backup area application data.
|
|
*
|
|
* @param app_size application size
|
|
*
|
|
* @return result
|
|
*/
|
|
FlashErrCode flash_erase_bak_app(size_t app_size) {
|
|
FlashErrCode result = FLASH_NO_ERR;
|
|
|
|
result = flash_erase(get_bak_app_start_addr(), app_size);
|
|
switch (result) {
|
|
case FLASH_NO_ERR: {
|
|
FLASH_INFO("Erased backup area application OK.\n");
|
|
break;
|
|
}
|
|
case FLASH_ERASE_ERR: {
|
|
FLASH_INFO("Warning: Erase backup area application fault!\n");
|
|
/* will return when erase fault */
|
|
return result;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
/**
|
|
* Erase user old application
|
|
*
|
|
* @param user_app_addr application entry address
|
|
* @param app_size application size
|
|
*
|
|
* @return result
|
|
*/
|
|
FlashErrCode flash_erase_user_app(uint32_t user_app_addr, size_t app_size) {
|
|
FlashErrCode result = FLASH_NO_ERR;
|
|
|
|
result = flash_erase(user_app_addr, app_size);
|
|
switch (result) {
|
|
case FLASH_NO_ERR: {
|
|
FLASH_INFO("Erased user application OK.\n");
|
|
break;
|
|
}
|
|
case FLASH_ERASE_ERR: {
|
|
FLASH_INFO("Warning: Erase user application fault!\n");
|
|
/* will return when erase fault */
|
|
return result;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Erase old bootloader
|
|
*
|
|
* @param bl_addr bootloader entry address
|
|
* @param bl_size bootloader size
|
|
*
|
|
* @return result
|
|
*/
|
|
FlashErrCode flash_erase_bl(uint32_t bl_addr, size_t bl_size) {
|
|
FlashErrCode result = FLASH_NO_ERR;
|
|
|
|
result = flash_erase(bl_addr, bl_size);
|
|
switch (result) {
|
|
case FLASH_NO_ERR: {
|
|
FLASH_INFO("Erased bootloader OK.\n");
|
|
break;
|
|
}
|
|
case FLASH_ERASE_ERR: {
|
|
FLASH_INFO("Warning: Erase bootloader fault!\n");
|
|
/* will return when erase fault */
|
|
return result;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Write data of application to backup area.
|
|
*
|
|
* @param data a part of application
|
|
* @param size data size
|
|
* @param cur_size current write application size
|
|
* @param total_size application total size
|
|
*
|
|
* @return result
|
|
*/
|
|
FlashErrCode flash_write_data_to_bak(uint8_t *data, size_t size, uint32_t *cur_size,
|
|
size_t total_size) {
|
|
FlashErrCode result = FLASH_NO_ERR;
|
|
|
|
/* make sure don't write excess data */
|
|
if (*cur_size + size > total_size) {
|
|
size = total_size - *cur_size;
|
|
}
|
|
|
|
result = flash_write(get_bak_app_start_addr() + *cur_size, (uint32_t *) data, size);
|
|
switch (result) {
|
|
case FLASH_NO_ERR: {
|
|
*cur_size += size;
|
|
FLASH_INFO("Write data to backup area OK.\n");
|
|
break;
|
|
}
|
|
case FLASH_WRITE_ERR: {
|
|
FLASH_INFO("Warning: Write data to backup area fault!\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Copy backup area application to application entry.
|
|
*
|
|
* @param user_app_addr application entry address
|
|
* @param app_size application size
|
|
*
|
|
* @return result
|
|
*/
|
|
FlashErrCode flash_copy_app_from_bak(uint32_t user_app_addr, size_t app_size) {
|
|
size_t cur_size;
|
|
uint32_t app_cur_addr, bak_cur_addr;
|
|
FlashErrCode result = FLASH_NO_ERR;
|
|
/* 32 words size buffer */
|
|
uint32_t buff[32];
|
|
|
|
/* cycle copy data */
|
|
for (cur_size = 0; cur_size < app_size; cur_size += sizeof(buff)) {
|
|
app_cur_addr = user_app_addr + cur_size;
|
|
bak_cur_addr = get_bak_app_start_addr() + cur_size;
|
|
flash_read(bak_cur_addr, buff, sizeof(buff));
|
|
result = flash_write(app_cur_addr, buff, sizeof(buff));
|
|
if (result != FLASH_NO_ERR) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch (result) {
|
|
case FLASH_NO_ERR: {
|
|
FLASH_INFO("Write data to application entry OK.\n");
|
|
break;
|
|
}
|
|
case FLASH_WRITE_ERR: {
|
|
FLASH_INFO("Warning: Write data to application entry fault!\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Copy backup area bootloader to bootloader entry.
|
|
*
|
|
* @param bl_addr bootloader entry address
|
|
* @param bl_size bootloader size
|
|
*
|
|
* @return result
|
|
*/
|
|
FlashErrCode flash_copy_bl_from_bak(uint32_t bl_addr, size_t bl_size) {
|
|
size_t cur_size;
|
|
uint32_t bl_cur_addr, bak_cur_addr;
|
|
FlashErrCode result = FLASH_NO_ERR;
|
|
/* 32 words buffer */
|
|
uint32_t buff[32];
|
|
|
|
/* cycle copy data by 32bytes buffer */
|
|
for (cur_size = 0; cur_size < bl_size; cur_size += sizeof(buff)) {
|
|
bl_cur_addr = bl_addr + cur_size;
|
|
bak_cur_addr = get_bak_app_start_addr() + cur_size;
|
|
flash_read(bak_cur_addr, buff, sizeof(buff));
|
|
result = flash_write(bl_cur_addr, buff, sizeof(buff));
|
|
if (result != FLASH_NO_ERR) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch (result) {
|
|
case FLASH_NO_ERR: {
|
|
FLASH_INFO("Write data to bootloader entry OK.\n");
|
|
break;
|
|
}
|
|
case FLASH_WRITE_ERR: {
|
|
FLASH_INFO("Warning: Write data to bootloader entry fault!\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Get IAP section start address in flash.
|
|
*
|
|
* @return size
|
|
*/
|
|
static uint32_t get_bak_app_start_addr(void) {
|
|
FLASH_ASSERT(bak_app_start_addr);
|
|
return bak_app_start_addr;
|
|
}
|