KT24-1110_65E-HA-651B/apps/common/dev_manager/dev_update.c

379 lines
9.1 KiB
C
Raw Permalink Normal View History

2024-11-10 10:44:17 +00:00
#include "dev_update.h"
#include "dev_manager.h"
#include "update/update.h"
#include "update/update_loader_download.h"
#include "app_config.h"
#include "tone_player.h"
#include "app_task.h"
#if defined(CONFIG_SD_UPDATE_ENABLE) || defined(CONFIG_USB_UPDATE_ENABLE)
#define DEV_UPDATE_EN 1
#else
#define DEV_UPDATE_EN 0
#endif
#ifdef DEV_UPDATE_SUPPORT_JUMP
extern void __JUMP_TO_MASKROM();
extern void save_spi_port();
extern void update_close_hw(void);
extern void ram_protect_close(void);
#endif //endif DEV_UPDATE_SUPPORT_JUMP
extern bool uart_update_send_update_ready(char *file_path);
extern bool get_uart_update_sta(void);
extern void storage_update_loader_download_init_with_file_hdl(
int type,
char *update_path,
void *fd,
void (*cb)(void *priv, int type, u8 cmd),
void *cb_priv,
u8 task_en
);
static char update_path[48] = {0};
extern const char updata_file_name[];
struct __update_dev_reg {
char *logo;
int type;
union {
UPDATA_SD sd;
} u;
};
#if TCFG_SD0_ENABLE
static const struct __update_dev_reg sd0_update = {
.logo = "sd0",
.type = SD0_UPDATA,
.u.sd.control_type = SD_CONTROLLER_0,
#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
.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
};
static void dev_update_callback(void *priv, int type, u8 cmd)
{
struct __update_dev_reg *parm = (struct __update_dev_reg *)priv;
if (cmd == UPDATE_LOADER_OK) {
update_mode_api(type);
} else {
printf("update fail, cpu reset!!!\n");
cpu_reset();
}
}
static 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;
}
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(__this->fd, buff, len);
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,
};
static void dev_update_param_private_handle(UPDATA_PARM *p)
{
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 + sizeof(UPDATA_PARM);
void *sd = NULL;
sd = dev_update_get_parm(up_type);
if (sd) {
memcpy((void *)sd_start, sd, UPDATE_PRIV_PARAM_LEN);
} else {
memset((void *)sd_start, 0, UPDATE_PRIV_PARAM_LEN);
}
}
#endif
#ifdef CONFIG_USB_UPDATE_ENABLE
if (up_type == USB_UPDATA) {
printf("usb updata ");
int usb_start = (u32)p + sizeof(UPDATA_PARM);
memset((void *)usb_start, 0, UPDATE_PRIV_PARAM_LEN);
}
#endif
memcpy(p->file_patch, updata_file_name, strlen(updata_file_name));
}
static void dev_update_before_jump_handle(u16 up_type)
{
#ifdef DEV_UPDATE_SUPPORT_JUMP
#if TCFG_BLUETOOTH_BACK_MODE //后台模式需要把蓝牙关掉
if (BT_MODULES_IS_SUPPORT(BT_MODULE_LE)) {
ll_hci_destory();
}
hci_controller_destory();
#endif
update_close_hw();
ram_protect_close();
save_spi_port();
printf("update jump to mask...\n");
/* JL_UART0->CON0 = 0; */
/* JL_UART1->CON0 = 0; */
__JUMP_TO_MASKROM();
#else
cpu_reset();
#endif //DEV_UPDATE_SUPPORT_JUMP
}
static void before_update_prompt_opt(u8 flag)
{
#if DEV_BEFORE_UPDATE_PROMPT
if (APP_POWERON_TASK != app_get_curr_task()) {
static OS_SEM update_prompt_sem;
switch (flag) {
case 0:
// 创建
os_sem_create(&update_prompt_sem, 0);
break;
case 1:
// 播放提示音post
tone_play_index(IDEX_TONE_NORMAL, 1);
while (tone_get_dec_status()) {
os_time_dly(1);
}
os_sem_post(&update_prompt_sem);
break;
case 2:
// pend
os_sem_pend(&update_prompt_sem, 100);
break;
}
}
#endif
}
static void dev_update_state_cbk(int type, u32 state, void *priv)
{
static u8 update_file_same_flag = 0;
update_ret_code_t *ret_code = (update_ret_code_t *)priv;
switch (state) {
case UPDATE_CH_INIT:
before_update_prompt_opt(1);
break;
case UPDATE_CH_SAME:
printf("update file is same\n");
update_file_same_flag = 1;
break;
case UPDATE_CH_EXIT:
if (update_file_same_flag) {
update_file_same_flag = 0;
break;
}
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) {
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;
}
//<尝试按照路径打开升级文件
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;
}
before_update_prompt_opt(0);
#if(USER_UART_UPDATE_ENABLE) && (UART_UPDATE_ROLE == UART_UPDATE_MASTER)
uart_update_send_update_ready(update_path);
while (get_uart_update_sta()) {
os_time_dly(10);
}
#else
///进行升级
/* storage_update_loader_download_init_with_file_hdl(
parm->type,
update_path,
(void *)fd,
dev_update_callback,
(void *)parm,
0); */
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// USER_UART_UPDATE_ENABLE
before_update_prompt_opt(2);
#endif//DEV_UPDATE_EN
}
return UPDATA_READY;
}