458 lines
12 KiB
C
458 lines
12 KiB
C
|
#include "app_config.h"
|
||
|
#if(USER_UART_UPDATE_ENABLE) && (UART_UPDATE_ROLE == UART_UPDATE_MASTER)
|
||
|
#include "typedef.h"
|
||
|
#include "update_loader_download.h"
|
||
|
#include "os/os_api.h"
|
||
|
#include "system/task.h"
|
||
|
#include "update.h"
|
||
|
#include "gpio.h"
|
||
|
#include "uart_update.h"
|
||
|
#include "asm/uart_dev.h"
|
||
|
#include "asm/clock.h"
|
||
|
#include "timer.h"
|
||
|
#include "system/fs/fs.h"
|
||
|
|
||
|
static volatile u32 uart_to_cnt = 0;
|
||
|
static volatile u32 uart_file_offset = 0;
|
||
|
static volatile u16 rx_cnt; //收数据计数
|
||
|
|
||
|
typedef struct _uart_updte_ctl_t {
|
||
|
OS_SEM sem;
|
||
|
OS_SEM rx_sem;
|
||
|
volatile u16 timemax;
|
||
|
volatile u16 timeout;
|
||
|
volatile u8 flag;
|
||
|
volatile u8 err_code;
|
||
|
u8 update_sta;
|
||
|
u8 update_percent;
|
||
|
u32 update_total_size;
|
||
|
u32 update_send_size;
|
||
|
u8 rx_cmd[0x30];
|
||
|
u16 cmd_len;
|
||
|
u16 uart_timer_hdl;
|
||
|
} uart_update_ctl_t;
|
||
|
|
||
|
static uart_update_ctl_t uart_update_ctl;
|
||
|
#define __this (&uart_update_ctl)
|
||
|
|
||
|
#define LOG_TAG "[UART_UPDATE]"
|
||
|
#define LOG_ERROR_ENABLE
|
||
|
#define LOG_DEBUG_ENABLE
|
||
|
#define LOG_INFO_ENABLE
|
||
|
#define LOG_CLI_ENABLE
|
||
|
#include "debug.h"
|
||
|
|
||
|
#define RETRY_TIME 4//重试n次
|
||
|
#define PACKET_TIMEOUT 200//ms
|
||
|
|
||
|
#define FILE_READ_UNIT 512 //
|
||
|
#define UART_DEFAULT_BAUD 9600
|
||
|
#define UART_UPDATE_BAUD (50*10000L)
|
||
|
#define TIME_TICK_UNIT (10) //unit:ms
|
||
|
//命令
|
||
|
#define CMD_UPDATE_START 0x01
|
||
|
#define CMD_UPDATE_READ 0x02
|
||
|
#define CMD_UPDATE_END 0x03
|
||
|
#define CMD_SEND_UPDATE_LEN 0x04
|
||
|
#define CMD_KEEP_ALIVE 0x05
|
||
|
|
||
|
|
||
|
#define READ_LIT_U16(a) (*((u8*)(a)) + (*((u8*)(a)+1)<<8))
|
||
|
#define READ_LIT_U32(a) (*((u8*)(a)) + (*((u8*)(a)+1)<<8) + (*((u8*)(a)+2)<<16) + (*((u8*)(a)+3)<<24))
|
||
|
|
||
|
#define WRITE_LIT_U16(a,src) {*((u8*)(a)+1) = (u8)(src>>8); *((u8*)(a)+0) = (u8)(src&0xff); }
|
||
|
#define WRITE_LIT_U32(a,src) {*((u8*)(a)+3) = (u8)((src)>>24); *((u8*)(a)+2) = (u8)(((src)>>16)&0xff);*((u8*)(a)+1) = (u8)(((src)>>8)&0xff);*((u8*)(a)+0) = (u8)((src)&0xff);}
|
||
|
|
||
|
#define THIS_TASK_NAME "uart_update"
|
||
|
|
||
|
|
||
|
static protocal_frame_t protocal_frame __attribute__((aligned(4)));
|
||
|
u32 update_baudrate = 9600; //初始波特率
|
||
|
static uart_update_cfg update_cfg;
|
||
|
void *fd = NULL;
|
||
|
|
||
|
u32 uart_dev_receive_data(void *buf, u32 relen, u32 addr);
|
||
|
void uart_set_dir(u8 mode);
|
||
|
void uart_update_write(u8 *data, u32 len);
|
||
|
void uart_update_set_baud(u32 baudrate);
|
||
|
void uart_close_deal(void);
|
||
|
void uart_hw_init(uart_update_cfg update_cfg, void (*cb)(void *, u32));
|
||
|
void uart_data_decode(u8 *buf, u16 len);
|
||
|
|
||
|
/* enum { */
|
||
|
/* SEEK_SET = 0x0, */
|
||
|
/* SEEK_CUR = 0x1, */
|
||
|
/* SEEK_END = 0X2, */
|
||
|
/* }; */
|
||
|
|
||
|
enum {
|
||
|
CMD_UART_UPDATE_START = 0x1,
|
||
|
CMD_UART_UPDATE_READ,
|
||
|
CMD_UART_UPDATE_END,
|
||
|
CMD_UART_UPDATE_UPDATE_LEN,
|
||
|
CMD_UART_JEEP_ALIVE,
|
||
|
CMD_UART_UPDATE_READY,
|
||
|
};
|
||
|
|
||
|
enum {
|
||
|
RX_DATA_READY = 0,
|
||
|
RX_DATA_TIMEOUT,
|
||
|
RX_DATA_SUCC,
|
||
|
};
|
||
|
|
||
|
void uart_data_decode(u8 *buf, u16 len)
|
||
|
{
|
||
|
u16 crc, crc0, i, ch;
|
||
|
/* printf("decode_len:%d\n", len); */
|
||
|
/* put_buf(buf, len); */
|
||
|
for (i = 0; i < len; i++) {
|
||
|
ch = buf[i];
|
||
|
__recheck:
|
||
|
if (rx_cnt == 0) {
|
||
|
if (ch == SYNC_MARK0) {
|
||
|
protocal_frame.raw_data[rx_cnt++] = ch;
|
||
|
}
|
||
|
} else if (rx_cnt == 1) {
|
||
|
protocal_frame.raw_data[rx_cnt++] = ch;
|
||
|
if (ch != SYNC_MARK1) {
|
||
|
rx_cnt = 0;
|
||
|
goto __recheck;
|
||
|
}
|
||
|
} else if (rx_cnt < 4) {
|
||
|
protocal_frame.raw_data[rx_cnt++] = ch;
|
||
|
} else {
|
||
|
protocal_frame.raw_data[rx_cnt++] = ch;
|
||
|
if (rx_cnt == (protocal_frame.data.length + SYNC_SIZE)) {
|
||
|
rx_cnt = 0;
|
||
|
extern u16 CRC16(void *ptr, u32 len);
|
||
|
crc = CRC16(protocal_frame.raw_data, protocal_frame.data.length + SYNC_SIZE - 2);
|
||
|
memcpy(&crc0, &protocal_frame.raw_data[protocal_frame.data.length + SYNC_SIZE - 2], 2);
|
||
|
if (crc0 == crc) {
|
||
|
__this->timemax = 0;
|
||
|
if (protocal_frame.data.length <= sizeof(__this->rx_cmd)) {
|
||
|
memcpy(__this->rx_cmd, &(protocal_frame.data.data), protocal_frame.data.length);
|
||
|
__this->flag = RX_DATA_SUCC;
|
||
|
os_sem_post(&__this->rx_sem);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static bool uart_send_packet(u8 *buf, u16 length)
|
||
|
{
|
||
|
bool ret;
|
||
|
u16 crc;
|
||
|
u8 *buffer;
|
||
|
|
||
|
buffer = (u8 *)&protocal_frame;
|
||
|
protocal_frame.data.mark0 = SYNC_MARK0;
|
||
|
protocal_frame.data.mark1 = SYNC_MARK1;
|
||
|
protocal_frame.data.length = length;
|
||
|
memcpy((char *)&buffer[4], buf, length);
|
||
|
crc = CRC16(buffer, length + SYNC_SIZE - 2);
|
||
|
memcpy(buffer + 4 + length, &crc, 2);
|
||
|
uart_set_dir(0);//设为输出
|
||
|
uart_update_write((u8 *)&protocal_frame, length + SYNC_SIZE);
|
||
|
uart_set_dir(1);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
//read file by file operation handle
|
||
|
u32 ufw_data_read_api(u8 *buff, u32 addr, u32 size)
|
||
|
{
|
||
|
//To do...
|
||
|
if (fd) {
|
||
|
fseek(fd, addr, SEEK_SET);
|
||
|
return fread(fd, buff, size);
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
//open file and get file operation handle
|
||
|
bool ufw_file_op_init(char *update_path)
|
||
|
{
|
||
|
if (fd) {
|
||
|
fclose(fd);
|
||
|
}
|
||
|
//To do
|
||
|
fd = fopen(update_path, "r");
|
||
|
if (!fd) {
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//close the file and release resource
|
||
|
void ufw_file_op_close(void)
|
||
|
{
|
||
|
if (fd) {
|
||
|
fclose(fd);
|
||
|
fd = NULL;
|
||
|
}
|
||
|
//To do
|
||
|
}
|
||
|
|
||
|
static u32 update_data_read_from_file(void *p, u32 addr, u32 len)
|
||
|
{
|
||
|
u8 *buffer;
|
||
|
struct file_info *p_file_info;
|
||
|
u32 read_len = 0;
|
||
|
|
||
|
if (len > FILE_READ_UNIT) {
|
||
|
return (u32) - 1;
|
||
|
}
|
||
|
|
||
|
buffer = malloc(len + sizeof(struct file_info));
|
||
|
if (buffer) {
|
||
|
p_file_info = (struct file_info *)buffer;
|
||
|
p_file_info->cmd = CMD_UART_UPDATE_READ;
|
||
|
p_file_info->addr = addr;
|
||
|
p_file_info->len = len;
|
||
|
read_len = ufw_data_read_api(buffer + sizeof(struct file_info), addr, len);
|
||
|
} else {
|
||
|
return (u32) - 2;
|
||
|
}
|
||
|
|
||
|
uart_send_packet(buffer, len + sizeof(struct file_info));
|
||
|
|
||
|
if (buffer) {
|
||
|
free(buffer);
|
||
|
}
|
||
|
|
||
|
return read_len;
|
||
|
}
|
||
|
|
||
|
enum {
|
||
|
UPDATE_STA_NONE = 0,
|
||
|
UPDATE_STA_READY,
|
||
|
UPDATE_STA_START,
|
||
|
UPDATE_STA_TIMEOUT,
|
||
|
UPDATE_STA_LOADER_DOWNLOAD_FINISH,
|
||
|
UPDATE_STA_SUCC,
|
||
|
UPDATE_STA_FAIL,
|
||
|
};
|
||
|
|
||
|
enum {
|
||
|
UPDATE_ERR_NONE = 0,
|
||
|
UPDATE_ERR_KEY,
|
||
|
UPDATE_ERR_VERIFY,
|
||
|
|
||
|
UPDATE_ERR_LOADER_DOWNLOAD_SUCC = 0x80,
|
||
|
};
|
||
|
|
||
|
static void uart_update_err_code_handle(u8 code)
|
||
|
{
|
||
|
if (code) {
|
||
|
if (UPDATE_ERR_LOADER_DOWNLOAD_SUCC == code) {
|
||
|
__this->update_percent = 100;
|
||
|
__this->update_send_size = 0;
|
||
|
__this->update_total_size = 0;
|
||
|
__this->update_sta = UPDATE_STA_LOADER_DOWNLOAD_FINISH;
|
||
|
log_info("loader dn succ\n");
|
||
|
} else {
|
||
|
__this->update_sta = UPDATE_STA_FAIL;
|
||
|
log_error("update_err:%x\n", code);
|
||
|
}
|
||
|
} else {
|
||
|
__this->update_percent = 100;
|
||
|
__this->update_sta = UPDATE_STA_SUCC;
|
||
|
log_info("update all succ\n");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static int uart_update_wait_rev_data(timeout)
|
||
|
{
|
||
|
u32 err;
|
||
|
__this->flag = RX_DATA_READY;
|
||
|
__this->timeout = 0;
|
||
|
__this->timemax = (timeout + 1) / TIME_TICK_UNIT;
|
||
|
|
||
|
err = os_sem_pend(&__this->rx_sem, 2000);
|
||
|
|
||
|
if (OS_NO_ERR != err) {
|
||
|
log_info("wait tm out\n");
|
||
|
}
|
||
|
|
||
|
__this->timemax = __this->timeout = 0;
|
||
|
|
||
|
if (__this->flag == RX_DATA_SUCC) {
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
int uart_update_api_write_then_read(u8 *buf, u8 length, u8 timeout)
|
||
|
{
|
||
|
int ret = FALSE;
|
||
|
|
||
|
uart_send_packet(buf, length);
|
||
|
if (timeout) {
|
||
|
ret = uart_update_wait_rev_data(timeout);
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
bool uart_update_send_update_ready(char *file_update_path)
|
||
|
{
|
||
|
u8 ut_cmd[1];
|
||
|
ut_cmd[0] = CMD_UART_UPDATE_READY;
|
||
|
|
||
|
if (ufw_file_op_init(file_update_path)) {
|
||
|
__this->update_sta = UPDATE_STA_READY;
|
||
|
uart_send_packet(ut_cmd, sizeof(ut_cmd));
|
||
|
log_info("uart_update_send_update_ready\n");
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
bool get_uart_update_sta(void)
|
||
|
{
|
||
|
return (__this->update_sta == UPDATE_STA_READY || __this->update_sta == UPDATE_STA_START) ? TRUE : FALSE;
|
||
|
}
|
||
|
|
||
|
static void update_process_run(void)
|
||
|
{
|
||
|
update_baudrate = UART_DEFAULT_BAUD;
|
||
|
uart_update_set_baud(update_baudrate);
|
||
|
|
||
|
__this->update_total_size = 0;
|
||
|
__this->update_percent = 0;
|
||
|
__this->update_send_size = 0;
|
||
|
|
||
|
u8 *pbuf = &__this->rx_cmd;
|
||
|
while (1) {
|
||
|
if (OS_NO_ERR != os_sem_pend(&__this->rx_sem, 800)) {
|
||
|
log_info("uart_timeout\n");
|
||
|
__this->update_sta = UPDATE_STA_TIMEOUT;
|
||
|
update_baudrate = 9600;
|
||
|
uart_update_set_baud(update_baudrate);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
//os_time_dly(1);
|
||
|
|
||
|
switch (pbuf[0]) {
|
||
|
case CMD_UPDATE_START:
|
||
|
log_info("CMD_UPDATE_START\n");
|
||
|
__this->update_sta = UPDATE_STA_START;
|
||
|
update_baudrate = UART_UPDATE_BAUD;
|
||
|
WRITE_LIT_U32(pbuf + 1, update_baudrate);
|
||
|
uart_send_packet(pbuf, 1 + sizeof(u32));
|
||
|
log_info("use baud:%x\n", update_baudrate);
|
||
|
uart_update_set_baud(update_baudrate);
|
||
|
break;
|
||
|
|
||
|
case CMD_UPDATE_READ: {
|
||
|
u32 addr = READ_LIT_U32(&pbuf[1]);
|
||
|
u32 len = READ_LIT_U32(&pbuf[1 + sizeof(u32)]);
|
||
|
if (__this->update_total_size) {
|
||
|
__this->update_send_size += len;
|
||
|
__this->update_percent = (__this->update_send_size * 100) / __this->update_total_size;
|
||
|
if (__this->update_percent >= 99) {
|
||
|
__this->update_percent = 99;
|
||
|
}
|
||
|
log_info("send data process:%x\n", __this->update_percent);
|
||
|
}
|
||
|
log_info("CMD_UPDATE_READ\n");
|
||
|
update_data_read_from_file(NULL, addr, len);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case CMD_UPDATE_END:
|
||
|
log_info("CMD_UPDATE_END\n");
|
||
|
uart_update_err_code_handle(pbuf[1]);
|
||
|
uart_send_packet(pbuf, 1);
|
||
|
break;
|
||
|
|
||
|
case CMD_SEND_UPDATE_LEN:
|
||
|
__this->update_total_size = READ_LIT_U32(&pbuf[1]);
|
||
|
__this->update_percent = 0;
|
||
|
__this->update_send_size = 0;
|
||
|
log_info("update_total_size:%x\n", __this->update_total_size);
|
||
|
uart_send_packet(pbuf, 1);
|
||
|
break;
|
||
|
|
||
|
case CMD_KEEP_ALIVE:
|
||
|
uart_send_packet(pbuf, 1);
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void uart_timeout_handler(void *priv)
|
||
|
{
|
||
|
if (__this->timemax) {
|
||
|
__this->timeout++;
|
||
|
if (__this->timeout > __this->timemax) {
|
||
|
__this->timemax = 0;
|
||
|
__this->flag = RX_DATA_TIMEOUT;
|
||
|
os_sem_post(&__this->rx_sem);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void update_loader_download_task(void *p)
|
||
|
{
|
||
|
log_info("create %s task\n", THIS_TASK_NAME);
|
||
|
os_sem_create(&__this->sem, 0);
|
||
|
os_sem_create(&__this->rx_sem, 0);
|
||
|
|
||
|
|
||
|
while (1) {
|
||
|
update_process_run();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void uart_update_init(uart_update_cfg *cfg)
|
||
|
{
|
||
|
memcpy(&update_cfg, cfg, sizeof(uart_update_cfg));
|
||
|
task_create(update_loader_download_task, NULL, THIS_TASK_NAME);
|
||
|
uart_hw_init(update_cfg, uart_data_decode);
|
||
|
|
||
|
}
|
||
|
|
||
|
void uart_update_exit(void)
|
||
|
{
|
||
|
log_info("uart update exit\n");
|
||
|
|
||
|
if (__this->uart_timer_hdl) {
|
||
|
sys_timer_del(__this->uart_timer_hdl);
|
||
|
}
|
||
|
|
||
|
os_sem_del(&__this->sem, 0);
|
||
|
os_sem_del(&__this->rx_sem, 0);
|
||
|
|
||
|
ufw_file_op_close();
|
||
|
|
||
|
uart_close_deal();
|
||
|
|
||
|
task_kill(THIS_TASK_NAME);
|
||
|
}
|
||
|
|
||
|
static void clock_critical_enter(void)
|
||
|
{
|
||
|
|
||
|
}
|
||
|
|
||
|
static void clock_critical_exit(void)
|
||
|
{
|
||
|
uart_update_set_baud(update_baudrate);
|
||
|
}
|
||
|
CLOCK_CRITICAL_HANDLE_REG(uart_update, clock_critical_enter, clock_critical_exit)
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
#endif
|