363 lines
9.4 KiB
C
363 lines
9.4 KiB
C
#ifdef SUPPORT_MS_EXTENSIONS
|
|
#pragma bss_seg(".dev_update.data.bss")
|
|
#pragma data_seg(".dev_update.data")
|
|
#pragma const_seg(".dev_update.text.const")
|
|
#pragma code_seg(".dev_update.text")
|
|
#endif
|
|
#include "dev_update.h"
|
|
#include "dev_manager.h"
|
|
#include "update/update.h"
|
|
#include "update/update_loader_download.h"
|
|
#include "app_config.h"
|
|
#include "btcontroller_modules.h"
|
|
#include "trim.h"
|
|
|
|
#if TCFG_DEV_UPDATE_EN
|
|
|
|
#if defined(CONFIG_SD_UPDATE_ENABLE) || defined(CONFIG_USB_UPDATE_ENABLE)
|
|
#define DEV_UPDATE_EN 1
|
|
#else
|
|
#define DEV_UPDATE_EN 0
|
|
#endif
|
|
|
|
#define STR(x) #x
|
|
#define STRCHANGE(x) STR(x)
|
|
|
|
|
|
extern bool uart_update_send_update_ready(char *file_path);
|
|
extern bool get_uart_update_sta(void);
|
|
extern void latch_reset(void);
|
|
|
|
static char update_path[48] = {0};
|
|
extern const char updata_file_name[];
|
|
|
|
struct __update_dev_reg {
|
|
char *logo;
|
|
int type;
|
|
union {
|
|
UPDATA_SD sd;
|
|
UPDATE_UDISK udisk;
|
|
} u;
|
|
};
|
|
|
|
|
|
#if TCFG_SD0_ENABLE
|
|
static struct __update_dev_reg sd0_update = {
|
|
.logo = "sd0",
|
|
.type = SD0_UPDATA,
|
|
.u.sd.control_type = SD_CONTROLLER_0,
|
|
#ifdef TCFG_SD0_PORT_CMD
|
|
.u.sd.control_io_clk = TCFG_SD0_PORT_CLK,
|
|
.u.sd.control_io_cmd = TCFG_SD0_PORT_CMD,
|
|
.u.sd.control_io_dat = TCFG_SD0_PORT_DA0,
|
|
#else
|
|
#if (TCFG_SD0_PORTS=='A')
|
|
.u.sd.control_io = SD0_IO_A,
|
|
#elif (TCFG_SD0_PORTS=='B')
|
|
.u.sd.control_io = SD0_IO_B,
|
|
#elif (TCFG_SD0_PORTS=='C')
|
|
.u.sd.control_io = SD0_IO_C,
|
|
#elif (TCFG_SD0_PORTS=='D')
|
|
.u.sd.control_io = SD0_IO_D,
|
|
#elif (TCFG_SD0_PORTS=='E')
|
|
.u.sd.control_io = SD0_IO_E,
|
|
#elif (TCFG_SD0_PORTS=='F')
|
|
.u.sd.control_io = SD0_IO_F,
|
|
#endif
|
|
#endif
|
|
.u.sd.power = 1,
|
|
};
|
|
#endif//TCFG_SD0_ENABLE
|
|
|
|
#if TCFG_SD1_ENABLE
|
|
static const struct __update_dev_reg sd1_update = {
|
|
.logo = "sd1",
|
|
.type = SD1_UPDATA,
|
|
.u.sd.control_type = SD_CONTROLLER_1,
|
|
#if (TCFG_SD1_PORTS=='A')
|
|
.u.sd.control_io = SD1_IO_A,
|
|
#else
|
|
.u.sd.control_io = SD1_IO_B,
|
|
#endif
|
|
.u.sd.power = 1,
|
|
|
|
};
|
|
#endif//TCFG_SD1_ENABLE
|
|
|
|
#if TCFG_UDISK_ENABLE
|
|
static const struct __update_dev_reg udisk_update = {
|
|
.logo = "udisk0",
|
|
.type = USB_UPDATA,
|
|
};
|
|
#endif//TCFG_UDISK_ENABLE
|
|
|
|
|
|
static const struct __update_dev_reg *update_dev_list[] = {
|
|
#if TCFG_UDISK_ENABLE
|
|
&udisk_update,
|
|
#endif//TCFG_UDISK_ENABLE
|
|
#if TCFG_SD0_ENABLE
|
|
&sd0_update,
|
|
#endif//
|
|
#if TCFG_SD1_ENABLE
|
|
&sd1_update,
|
|
#endif//TCFG_SD1_ENABLE
|
|
};
|
|
|
|
void *dev_update_get_parm(int type)
|
|
{
|
|
struct __update_dev_reg *parm = NULL;
|
|
for (int i = 0; i < ARRAY_SIZE(update_dev_list); i++) {
|
|
if (update_dev_list[i]->type == type) {
|
|
parm = (struct __update_dev_reg *)update_dev_list[i];
|
|
}
|
|
}
|
|
|
|
if (parm == NULL) {
|
|
return NULL;
|
|
}
|
|
if (parm->type == USB_UPDATA) {
|
|
return (void *)&parm->u.udisk;
|
|
}
|
|
return (void *)&parm->u.sd;
|
|
}
|
|
|
|
|
|
struct strg_update {
|
|
void *fd;
|
|
char *update_path;
|
|
};
|
|
static struct strg_update strg_update = {0};
|
|
#define __this (&strg_update)
|
|
|
|
static u16 strg_f_open(void)
|
|
{
|
|
if (!__this->update_path) {
|
|
printf("file path err ");
|
|
return false;
|
|
}
|
|
|
|
if (__this->fd) {
|
|
return true;
|
|
/* fclose(__this->fd);
|
|
__this->fd = NULL; */
|
|
}
|
|
__this->fd = fopen(__this->update_path, "r");
|
|
if (!__this->fd) {
|
|
printf("file open err ");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static u16 strg_f_read(void *fp, u8 *buff, u16 len)
|
|
{
|
|
if (!__this->fd) {
|
|
return (u16) - 1;
|
|
}
|
|
|
|
len = fread(buff, len, 1, __this->fd);
|
|
return len;
|
|
}
|
|
|
|
static int strg_f_seek(void *fp, u8 type, u32 offset)
|
|
{
|
|
if (!__this->fd) {
|
|
return (int) - 1;
|
|
}
|
|
|
|
int ret = fseek(__this->fd, offset, type);
|
|
/* return 0; // 0k */
|
|
return ret;
|
|
}
|
|
static u16 strg_f_stop(u8 err)
|
|
{
|
|
if (__this->fd) {
|
|
fclose(__this->fd);
|
|
__this->fd = NULL;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int strg_update_set_file_path_and_hdl(char *update_path, void *fd)
|
|
{
|
|
__this->update_path = update_path;
|
|
__this->fd = fd;
|
|
|
|
return true;
|
|
}
|
|
|
|
static const update_op_api_t strg_dev_update_op = {
|
|
.f_open = strg_f_open,
|
|
.f_read = strg_f_read,
|
|
.f_seek = strg_f_seek,
|
|
.f_stop = strg_f_stop,
|
|
};
|
|
|
|
void update_param_ext_fill(UPDATA_PARM *p, u8 ext_type, u8 *ext_data, u8 ext_len);
|
|
static void dev_update_param_private_handle(UPDATA_PARM *p)
|
|
{
|
|
// cppcheck-suppress unreadVariable
|
|
u16 up_type = p->parm_type;
|
|
|
|
#ifdef CONFIG_SD_UPDATE_ENABLE
|
|
if ((up_type == SD0_UPDATA) || (up_type == SD1_UPDATA)) {
|
|
int sd_start = (u32)p->parm_priv;
|
|
void *sd = NULL;
|
|
sd = dev_update_get_parm(up_type);
|
|
if (sd) {
|
|
#if defined(TCFG_SD0_POWER_PORT)
|
|
sprintf((char *)&sd0_update.u.sd.io_det_func, "P%c%02d", TCFG_SD0_POWER_PORT / 16 + 'A', TCFG_SD0_POWER_PORT % 16);
|
|
sd0_update.u.sd.power = 0;
|
|
printf(">>>[test]:sd0_update.u.sd.io_det_func = %s, power = %d\n", (char *)&sd0_update.u.sd.io_det_func, sd0_update.u.sd.power);
|
|
#endif
|
|
memcpy((void *)sd_start, sd, UPDATE_PRIV_PARAM_LEN);
|
|
} else {
|
|
memset((void *)sd_start, 0, UPDATE_PRIV_PARAM_LEN);
|
|
}
|
|
|
|
char io_port_stirng[4 * 3 + 1] = {0};
|
|
sprintf(&io_port_stirng[0], "P%c%02d", TCFG_SD0_PORT_CLK / 16 + 'A', TCFG_SD0_PORT_CLK % 16);
|
|
sprintf(&io_port_stirng[4], "P%c%02d", TCFG_SD0_PORT_CMD / 16 + 'A', TCFG_SD0_PORT_CMD % 16);
|
|
sprintf(&io_port_stirng[8], "P%c%02d", TCFG_SD0_PORT_DA0 / 16 + 'A', TCFG_SD0_PORT_DA0 % 16);
|
|
update_param_ext_fill(p, EXT_SD_IO_INFO, (u8 *)io_port_stirng, sizeof(io_port_stirng));
|
|
|
|
}
|
|
#endif
|
|
|
|
#ifdef CONFIG_USB_UPDATE_ENABLE
|
|
if (up_type == USB_UPDATA) {
|
|
printf("usb updata ");
|
|
UPDATE_UDISK *usb_start = (UPDATE_UDISK *)p->parm_priv;
|
|
memset((void *)usb_start, 0, UPDATE_PRIV_PARAM_LEN);
|
|
void *usb = dev_update_get_parm(up_type);
|
|
if (usb) {
|
|
memcpy((void *)usb_start, usb, UPDATE_PRIV_PARAM_LEN);
|
|
#ifdef CONFIG_CPU_BR27
|
|
usb_start->u.param.rx_ldo_trim = trim_get_usb_rx_ldo();
|
|
usb_start->u.param.tx_ldo_trim = trim_get_usb_tx_ldo();
|
|
#endif
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#ifdef CONFIG_UPDATE_JUMP_TO_MASK
|
|
memcpy(p->file_patch, STRCHANGE(CONFIG_SD_LATCH_IO), 32);
|
|
#else
|
|
memcpy(p->file_patch, updata_file_name, strlen(updata_file_name));
|
|
#endif
|
|
}
|
|
|
|
static void dev_update_before_jump_handle(int up_type)
|
|
{
|
|
#if CONFIG_UPDATE_JUMP_TO_MASK
|
|
y_printf(">>>[test]:latch reset update\n");
|
|
latch_reset();
|
|
#else
|
|
cpu_reset();
|
|
#endif
|
|
}
|
|
|
|
static void dev_update_state_cbk(int type, u32 state, void *priv)
|
|
{
|
|
update_ret_code_t *ret_code = (update_ret_code_t *)priv;
|
|
|
|
switch (state) {
|
|
case UPDATE_CH_EXIT:
|
|
if ((0 == ret_code->stu) && (0 == ret_code->err_code)) {
|
|
update_mode_api_v2(type,
|
|
dev_update_param_private_handle,
|
|
dev_update_before_jump_handle);
|
|
} else {
|
|
printf("update fail, cpu reset!!!\n");
|
|
cpu_reset();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
u16 dev_update_check(char *logo)
|
|
{
|
|
if ((update_success_boot_check() == true) || !CONFIG_UPDATE_STORAGE_DEV_EN) {
|
|
return UPDATA_NON;
|
|
}
|
|
struct __dev *dev = dev_manager_find_spec(logo, 0);
|
|
if (dev) {
|
|
#if DEV_UPDATE_EN
|
|
//<查找设备升级配置
|
|
struct __update_dev_reg *parm = NULL;
|
|
for (int i = 0; i < ARRAY_SIZE(update_dev_list); i++) {
|
|
if (0 == strcmp(update_dev_list[i]->logo, logo)) {
|
|
parm = (struct __update_dev_reg *)update_dev_list[i];
|
|
}
|
|
}
|
|
if (parm == NULL) {
|
|
printf("dev update without parm err!!!\n");
|
|
return UPDATA_PARM_ERR;
|
|
}
|
|
|
|
//最新设备升级只支持fat32(省flash空间),此处截断
|
|
struct vfs_partition *part = fget_partition((const char *)dev_manager_get_root_path(dev));
|
|
if (part) {
|
|
// fs_type: 1为fat12 、2 为fat16 、3为fat32 、4为exfat
|
|
if (part->fs_type != 3) {
|
|
printf(">>>[test]:dev update only support fat32 !!!\n");
|
|
return UPDATA_PARM_ERR;
|
|
}
|
|
} else {
|
|
printf(">>>[test]:dev part err!!!\n");
|
|
return UPDATA_PARM_ERR;
|
|
}
|
|
|
|
//<尝试按照路径打开升级文件
|
|
char *updata_file = (char *)updata_file_name;
|
|
if (*updata_file == '/') {
|
|
updata_file ++;
|
|
}
|
|
memset(update_path, 0, sizeof(update_path));
|
|
sprintf(update_path, "%s%s", dev_manager_get_root_path(dev), updata_file);
|
|
printf("update_path: %s\n", update_path);
|
|
FILE *fd = fopen(update_path, "r");
|
|
if (!fd) {
|
|
//没有升级文件, 继续跑其他解码相关的流程
|
|
printf("open update file err!!!\n");
|
|
return UPDATA_DEV_ERR;
|
|
}
|
|
|
|
#if(TCFG_UPDATE_UART_IO_EN) && (TCFG_UPDATE_UART_ROLE)
|
|
uart_update_send_update_ready(update_path);
|
|
while (get_uart_update_sta()) {
|
|
os_time_dly(10);
|
|
}
|
|
#else
|
|
//进行升级
|
|
strg_update_set_file_path_and_hdl(update_path, (void *)fd);
|
|
|
|
update_mode_info_t info = {
|
|
.type = parm->type,
|
|
.state_cbk = dev_update_state_cbk,
|
|
.p_op_api = &strg_dev_update_op,
|
|
.task_en = 0,
|
|
};
|
|
app_active_update_task_init(&info);
|
|
|
|
#endif
|
|
#endif//DEV_UPDATE_EN
|
|
}
|
|
return UPDATA_READY;
|
|
}
|
|
|
|
#else // TCFG_dev_UPDATE_EN
|
|
|
|
u16 dev_update_check(char *logo)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void *dev_update_get_parm(int type)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
#endif // TCFG_dev_UPDATE_EN
|
|
|