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

715 lines
19 KiB
C
Raw Permalink Normal View History

2024-11-10 10:44:17 +00:00
#include "update.h"
#include "update_loader_download.h"
#include "asm/crc16.h"
#include "asm/wdt.h"
#include "os/os_api.h"
#include "app_config.h"
#include "cpu.h"
#include "syscfg_id.h"
#include "vm.h"
#include "btcontroller_modules.h"
#include "system/includes.h"
#include "uart_update.h"
#include "dual_bank_updata_api.h"
#if (SMART_BOX_EN && JL_SMART_BOX_EXTRA_FLASH_OPT)
#include "smartbox_extra_flash_opt.h"
#endif
#if TCFG_UI_ENABLE
#include "ui/ui_api.h"
#endif
#if RCSP_BTMATE_EN
#include "rcsp_user_update.h"
#elif RCSP_ADV_EN
#include "rcsp_adv_user_update.h"
#endif
#include "custom_cfg.h"
#include "update_tws.h"
#define LOG_TAG "[APP-UPDATE]"
#define LOG_INFO_ENABLE
#define LOG_ERROR_ENABLE
#include "system/debug.h"
extern void *dev_update_get_parm(int type);
extern u8 get_ota_status();
extern void get_nor_update_param(void *buf);
extern s32 vm_write(vm_hdl hdl, u8 *data_buf, u16 len);
extern const int support_norflash_update_en;
extern u32 config_update_features;
#ifdef DEV_UPDATE_SUPPORT_JUMP
extern void __JUMP_TO_MASKROM();
extern void save_spi_port();
extern s32 sd1_unmount(void);
extern void usb_sie_close_all(void);
#endif //endif DEV_UPDATE_SUPPORT_JUMP
#ifdef UPDATE_VOICE_REMIND
#include "tone_player.h"
#include "audio_config.h"
#endif
#ifdef UPDATE_LED_REMIND
#include "asm/pwm_led.h"
#endif
#define DEVICE_UPDATE_KEY_ERR BIT(30)
#define DEVICE_FIRST_START BIT(31)
//升级文件路径必须是短文件名8+3结构仅支持层目录
/* const char updata_file_name[] = "/UPDATA/JL_692X.BFU"; */
const char updata_file_name[] = "/*.UFW";
u32 g_updata_flag = 0;
u16 update_result_get(void)
{
u16 ret = UPDATA_NON;
if (!UPDATE_SUPPORT_DEV_IS_NULL()) {
UPDATA_PARM *p = UPDATA_FLAG_ADDR;
u16 crc_cal;
crc_cal = CRC16(((u8 *)p) + 2, sizeof(UPDATA_PARM) - 2); //2 : crc_val
if (crc_cal && crc_cal == p->parm_crc) {
ret = p->parm_result;
}
g_updata_flag = ret;
g_updata_flag |= ((u32)(p->magic)) << 16;
memset(p, 0x00, sizeof(UPDATA_PARM));
}
return ret;
}
void update_result_set(u16 result)
{
if (!UPDATE_SUPPORT_DEV_IS_NULL()) {
UPDATA_PARM *p = UPDATA_FLAG_ADDR;
memset(p, 0x00, sizeof(UPDATA_PARM));
p->parm_result = result;
p->parm_crc = CRC16(((u8 *)p) + 2, sizeof(UPDATA_PARM) - 2);
}
}
bool update_success_boot_check(void)
{
if ((0 == config_update_features & BIT(CONFIG_UPDATE_FEATRUES_CONTENT_COMPARE_EN))
&& !UPDATE_SUPPORT_DEV_IS_NULL()) {
u16 result = g_updata_flag & 0xffff;
u16 up_tpye = g_updata_flag >> 16;
if ((UPDATA_SUCC == result) && ((SD0_UPDATA == up_tpye) || (USB_UPDATA == up_tpye))) {
return true;
}
}
return false;
}
bool device_is_first_start()
{
log_info("g_updata_flag=0x%x\n", g_updata_flag);
if ((g_updata_flag & DEVICE_FIRST_START) || (g_updata_flag & DEVICE_UPDATE_KEY_ERR) || (g_updata_flag == UPDATA_SUCC)) {
puts("\n=================device_is_first_start=========================\n");
return true;
}
return false;
}
void led_update_start(void)
{
#ifdef UPDATE_LED_REMIND
puts("led_update_start\n");
pwm_led_mode_set(PWM_LED_ALL_OFF);
#endif
}
void led_update_finish(void)
{
#ifdef UPDATE_LED_REMIND
puts("led_update_finish\n");
pwm_led_mode_set(PWM_LED0_LED1_FAST_FLASH);
#endif
}
static inline void dev_update_close_ui()
{
#if (TCFG_UI_ENABLE&&(CONFIG_UI_STYLE == STYLE_JL_LED7))
u8 count = 0;
UI_SHOW_WINDOW(ID_WINDOW_POWER_OFF);
__retry:
if (UI_GET_WINDOW_ID() != ID_WINDOW_POWER_OFF) {
os_time_dly(10);//增加延时防止没有关显示
if (count < 3) {
goto __retry;
}
count++;
}
#endif
}
extern void delay_2ms(int cnt);
extern void app_audio_set_wt_volume(s16 volume);
int update_result_deal()
{
#ifdef CONFIG_FPGA_ENABLE
return 0;
#endif
u8 key_voice_cnt = 0;
u16 result = 0;
result = (g_updata_flag & 0xffff);
log_info("<--------update_result_deal=0x%x %x--------->\n", result, g_updata_flag >> 16);
#ifdef CONFIG_DEBUG_ENABLE
#if TCFG_APP_BT_EN
u8 check_update_param_len(void);
ASSERT(check_update_param_len(), "UPDATE_PARAM_LEN ERROR");
#endif
#endif
if (result == UPDATA_NON || 0 == result) {
return 0;
}
#ifdef UPDATE_VOICE_REMIND
#endif
if (result == UPDATA_SUCC) {
#if(JL_EARPHONE_APP_EN && RCSP_UPDATE_EN)
u8 clear_update_flag = 0;
vm_write(VM_UPDATE_FLAG, &clear_update_flag, 1);
#endif
#if(JL_SMART_BOX_EXTRA_FLASH_OPT && RCSP_UPDATE_EN)
smartbox_eflash_update_flag_set(0);
smartbox_eflash_flag_set(0);
#endif
#ifdef UPDATE_LED_REMIND
led_update_finish();
#endif
}
extern u8 get_max_sys_vol(void);
while (1) {
wdt_clear();
key_voice_cnt++;
#ifdef UPDATE_VOICE_REMIND
if (result == UPDATA_SUCC) {
puts("<<<<<<UPDATA_SUCC");
app_audio_set_volume(APP_AUDIO_STATE_WTONE, get_max_sys_vol() / 2, 1);
//tone_play_index(IDEX_TONE_NORMAL, 1);
tone_play(TONE_SIN_NORMAL, 1);
/* os_time_dly(25); */
/* tone_stop(); */
//delay_2ms(500);
os_time_dly(25);
puts(">>>>>>>>>>>\n");
} else {
log_info("!!!!!!!!!!!!!!!updata waring !!!!!!!!!!!=0x%x\n", result);
app_audio_set_volume(APP_AUDIO_STATE_WTONE, get_max_sys_vol() / 2, 1);
tone_play(TONE_SIN_NORMAL, 1);
//tone_play_index(IDEX_TONE_NORMAL, 1);
/* os_time_dly(50); */
/* tone_stop(); */
os_time_dly(25);
//delay_2ms(500);
}
#endif
if (key_voice_cnt > 5) {
key_voice_cnt = 0;
//delay_2ms(500);
//os_time_dly(100);
puts("enter_sys_soft_poweroff\n");
break;
//注:关机要慎重,要设置开机键
//enter_sys_soft_poweroff();
}
}
return 1;
}
void clr_update_ram_info(void)
{
UPDATA_PARM *p = UPDATA_FLAG_ADDR;
memset(p, 0x00, sizeof(UPDATA_PARM));
}
static u32 loader_start_addr = 0;
void set_loader_start_addr(u32 addr)
{
loader_start_addr = addr;
}
void update_close_hw(void)
{
const struct update_target *p;
list_for_each_update_target(p) {
p->driver_close();
printf("close Hw Name : %s\n", p->name);
}
}
//write(buf,len);
void updata_parm_set(UPDATA_TYPE up_type, void *priv, u32 len)
{
UPDATA_PARM *p = UPDATA_FLAG_ADDR;
u8 addr[6];
#ifdef UPDATE_LED_REMIND
led_update_start();
#endif
memset((u8 *)p, 0x00, sizeof(UPDATA_PARM));
p->parm_type = (u16)up_type;
p->parm_result = (u16)UPDATA_READY;
memcpy(p->file_path, updata_file_name, strlen(updata_file_name));
if (priv) {
if (support_norflash_update_en) {
printf("p->parm_priv:0x%x\n len:%d\n", p->parm_priv, len);
memcpy(p->parm_priv + 12, priv, len);
} else {
printf("p->parm_priv:0x%x priv:0x%x len:%d\n", p->parm_priv, priv, len);
memcpy(p->parm_priv, priv, len);
}
} else {
memset(p->parm_priv, 0x00, sizeof(p->parm_priv));
if ((up_type == BLE_TEST_UPDATA) && UPDATE_MODULE_IS_SUPPORT(UPDATE_BLE_TEST_EN)) {
extern int le_controller_get_mac(void *addr);
le_controller_get_mac(addr);
memcpy(p->parm_priv, addr, 6);
puts("ble addr:\n");
put_buf(p->parm_priv, 6);
}
}
#if (USE_SDFILE_NEW == 1)
p->magic = UPDATE_PARAM_MAGIC;
p->ota_addr = loader_start_addr;
#endif
#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) {
log_info("usb updata ");
int usb_start = (u32)p + sizeof(UPDATA_PARM);
memset((void *)usb_start, 0, UPDATE_PRIV_PARAM_LEN);
}
#endif
/* #if RCSP_UPDATE_EN */
/* extern u32 ex_cfg_get_start_addr(void); */
/* if (support_norflash_update_en) { */
/* *((u32 *)(p->parm_priv + 12)) = ex_cfg_get_start_addr();//12个字节是Norflash升级参数占用 */
/* } else { */
/* *((u32 *)(p->parm_priv)) = ex_cfg_get_start_addr(); //为了兼容之前的程序升级 */
/* } */
/* #endif */
if (support_norflash_update_en) {
get_nor_update_param(p->parm_priv);
p->parm_type = NORFLASH_UPDATA;
*((u16 *)((u8 *)p + sizeof(UPDATA_PARM) + 32)) = up_type; //将实际的升级类型保存到ram
}
p->parm_crc = CRC16(((u8 *)p) + 2, sizeof(UPDATA_PARM) - 2); //2 : crc_val
printf("UPDATA_PARM_ADDR = 0x%x\n", p);
printf_buf((void *)p, sizeof(UPDATA_PARM));
printf("exif_addr:0x%x\n", *(u32 *)(p->parm_priv));
}
//reset
void updata_enter_reset(UPDATA_TYPE up_type)
{
log_info("updata_enter_reset\n\n\n");
cpu_reset();
//reset
//JL_POWER->CON |= BIT(4);
/* JL_CLOCK->PWR_CON |= BIT(4); */
}
#define LOADER_NAME "LOADER.BIN"
const u8 loader_file_path[] = "mnt/norflash/C/"LOADER_NAME"";
extern void ll_hci_destory(void);
extern void hci_controller_destory(void);
extern const int support_norflash_update_en;
extern void ram_protect_close(void);
extern void hwi_all_close(void);
static void update_before_jump_common_handle(UPDATA_TYPE up_type)
{
dev_update_close_ui();
#if TCFG_AUDIO_ANC_ENABLE
extern void audio_anc_hw_close();
audio_anc_hw_close();
#endif
local_irq_disable();
hwi_all_close();
/* local_irq_disable();
for (i = 0; i < 64; i++) {
bit_clr_ie(i);
} */
}
extern void __bt_updata_reset_bt_bredrexm_addr(void);
extern int __bt_updata_save_connection_info(void);
void update_mode_api(UPDATA_TYPE up_type, ...)
{
u8 i;
u32 addr;
dev_update_close_ui();
#if RCSP_ADV_EN || RCSP_BTMATE_EN || SMART_BOX_EN
if (up_type == BLE_APP_UPDATA || up_type == SPP_APP_UPDATA) {
addr = ex_cfg_fill_content_api();
}
#endif
//step 1: disable irq
#if TCFG_USER_TWS_ENABLE || TCFG_USER_BLE_ENABLE
if (up_type == BT_UPDATA) {
log_info("close ble hw\n");
ll_hci_destory();
}
#endif
#if TCFG_AUDIO_ANC_ENABLE
extern void audio_anc_hw_close();
audio_anc_hw_close();
#endif
local_irq_disable();
for (i = 0; i < 64; i++) {
bit_clr_ie(i);
}
//step 2: prepare parm
printf("update_type:0x%x\n", up_type);
switch (up_type) {
#ifdef CONFIG_SD_UPDATE_ENABLE
case SD0_UPDATA:
case SD1_UPDATA:
updata_parm_set(up_type, (u8 *)loader_file_path, sizeof(loader_file_path));
break;
#endif
#ifdef CONFIG_USB_UPDATE_ENABLE
case USB_UPDATA:
updata_parm_set(up_type, (u8 *)loader_file_path, sizeof(loader_file_path));
break;
#endif
case UART_UPDATA:
/* uart_updata_io_ctrl(&parm); */
//log_info("up_io = %x\nup_baud = %d\nup_timeout = %dms\n", ((UPDATA_UART *)parm)->control_io, ((UPDATA_UART *)parm)->control_baud, ((UPDATA_UART *)parm)->control_timeout * 10);
/* updata_parm_set(up_type, (u8 *)loader_file_path, sizeof(loader_file_path)); */
{
va_list argptr;
va_start(argptr, up_type);
u32 baudrate = va_arg(argptr, int);
u32 uart_update_io_tx = va_arg(argptr, int);
u32 uart_update_io_rx = va_arg(argptr, int);
va_end(argptr);
UPDATA_UART uart_param = {.control_baud = baudrate, .control_io_tx = uart_update_io_tx, .control_io_rx = uart_update_io_rx};
updata_parm_set(up_type, (u8 *)&uart_param, sizeof(UPDATA_UART));
}
break;
case BT_UPDATA:
if (UPDATE_MODULE_IS_SUPPORT(UPDATE_BT_LMP_EN)) {
if (__bt_updata_save_connection_info()) {
log_error("bt save conn info fail!\n");
break;
}
updata_parm_set(up_type, (u8 *)loader_file_path, sizeof(loader_file_path));
ram_protect_close();
__bt_updata_reset_bt_bredrexm_addr();
//note:last func no return;
}
break;
case BLE_TEST_UPDATA:
if (BT_MODULES_IS_SUPPORT(BT_MODULE_LE)) {
ll_hci_destory();
updata_parm_set(up_type, NULL, 0);
}
break;
#if RCSP_UPDATE_EN
case BLE_APP_UPDATA:
updata_parm_set(up_type, (u8 *)&addr, sizeof(addr));
break;
case SPP_APP_UPDATA:
updata_parm_set(BLE_APP_UPDATA, (u8 *)&addr, sizeof(addr)); //暂时不支持SPP
break;
#endif
default:
updata_parm_set(up_type, NULL, 0);
break;
}
#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
//step 3: enter updata
updata_enter_reset(up_type);
/* updata_enter_jump(up_type); */
#endif //DEV_UPDATE_SUPPORT_JUMP
}
/* void update_param_setting()
{
u32 saddr
void (*
} */
void update_parm_set_and_get_buf(int type, u32 loader_saddr, void **buf_addr, u16 *len)
{
u32 exif_addr;
int total_len = sizeof(UPDATA_PARM);
if ((type == SD0_UPDATA) || (type == SD1_UPDATA) || (type == USB_UPDATA)) {
total_len += UPDATE_PRIV_PARAM_LEN;
}
set_loader_start_addr(loader_saddr);
if ((BLE_APP_UPDATA == type) || (SPP_APP_UPDATA == type)) {
#if RCSP_UPDATE_EN
extern u32 ex_cfg_get_start_addr(void);
exif_addr = ex_cfg_get_start_addr();
printf("exif_addr:0x%x\n", exif_addr);
updata_parm_set(type, (u8 *)&exif_addr, sizeof(exif_addr));
#endif
} else if (UART_UPDATA == type) {
#if(USER_UART_UPDATE_ENABLE) && (UART_UPDATE_ROLE == UART_UPDATE_SLAVE)
sava_uart_update_param();
#endif
} else if (BLE_TEST_UPDATA == type) {
updata_parm_set(type, NULL, 0);
} else {
updata_parm_set(type, (u8 *)loader_file_path, sizeof(loader_file_path));
}
*buf_addr = UPDATA_FLAG_ADDR;
*len = total_len;
}
//fill common content
//fill private content
//fill crc16
static succ_report_t succ_report;
/* static void (*update_param_private_fill_handle)(UPDATA_PARM *p) = NULL;
static void (*update_before_jump_private_handle)(void) = NULL;
void update_param_private_fill_handle_register(void (*handle)(UPDATA_PARM *p))
{
update_param_private_fill_handle = handle;
}
void update_before_jump_private_handle_register(void (*handle)(u16 type))
{
update_before_jump_private_handle = handle;
} */
static void update_param_content_fill(int type, UPDATA_PARM *p, void (*priv_param_fill_hdl)(UPDATA_PARM *P))
{
memset((u8 *)p, 0x00, sizeof(UPDATA_PARM));
p->parm_type = (u16)type;
p->parm_result = (u16)UPDATA_READY;
p->magic = UPDATE_PARAM_MAGIC;
p->ota_addr = succ_report.loader_saddr;
if (0 == p->ota_addr) {
log_error("ota addr err\n");
return;
}
if (priv_param_fill_hdl) {
priv_param_fill_hdl(p);
}
p->parm_crc = CRC16(((u8 *)p) + 2, sizeof(UPDATA_PARM) - 2); //2 : crc_val
}
static void update_param_ram_set(u8 *buf, u16 len)
{
u8 *update_ram = UPDATA_FLAG_ADDR;
memcpy(update_ram, (u8 *)buf, len);
}
void update_mode_api_v2(UPDATA_TYPE type, void (*priv_param_fill_hdl)(UPDATA_PARM *p), void (*priv_update_jump_handle)(int type))
{
u16 update_param_len = sizeof(UPDATA_PARM) + UPDATE_PRIV_PARAM_LEN;
UPDATA_PARM *p = malloc(update_param_len);
if (p) {
update_param_content_fill(type, p, priv_param_fill_hdl);
if (succ_report.update_param_write_hdl) {
succ_report.update_param_write_hdl(succ_report.priv_param, p, update_param_len);
}
#ifdef UPDATE_LED_REMIND
led_update_start();
#endif
update_before_jump_common_handle(type);
update_param_ram_set((u8 *)p, update_param_len);
if (priv_update_jump_handle) {
priv_update_jump_handle(type);
}
free(p);
} else {
ASSERT(p, "malloc update param err \n");
}
}
int update_check_sniff_en(void)
{
if (!UPDATE_SUPPORT_DEV_IS_NULL()) {
#if (OTA_TWS_SAME_TIME_ENABLE && RCSP_ADV_EN && !OTA_TWS_SAME_TIME_NEW)
if (tws_ota_control(OTA_STATUS_GET) != OTA_OVER) {
return 0;
}
#endif
if (get_ota_status()) {
log_info("ota ing...");
return 0;
} else {
return 1;
}
}
return 1;
}
static volatile u8 ota_status = 0;
u8 get_ota_status()
{
return ota_status;
}
static u8 ota_idle_query(void)
{
return !ota_status;
}
REGISTER_LP_TARGET(ota_lp_target) = {
.name = "ota",
.is_idle = ota_idle_query,
};
extern void tws_sync_update_api_register(const update_op_tws_api_t *op);
extern update_op_tws_api_t *get_tws_update_api(void);
extern const int support_dual_bank_update_en;
extern int tws_ota_init(void);
extern void sys_auto_shut_down_disable(void);
extern void sys_auto_shut_down_enable(void);
extern void tws_api_auto_role_switch_disable();
extern void tws_api_auto_role_switch_enable();
static void update_init_common_handle(int type)
{
ota_status = 1;
if (UPDATE_DUAL_BANK_IS_SUPPORT()) {
#if TCFG_AUTO_SHUT_DOWN_TIME
sys_auto_shut_down_disable();
#endif
#if OTA_TWS_SAME_TIME_ENABLE
tws_api_auto_role_switch_disable();
tws_sync_update_api_register(get_tws_update_api());
tws_ota_init();
#endif
}
}
static void update_exit_common_handle(int type, void *priv)
{
update_ret_code_t *ret_code = (update_ret_code_t *)priv;
#if TCFG_AUTO_SHUT_DOWN_TIME
sys_auto_shut_down_enable();
#endif
#if OTA_TWS_SAME_TIME_ENABLE
if (UPDATE_DUAL_BANK_IS_SUPPORT()) {
tws_api_auto_role_switch_enable();
}
#endif
ota_status = 0;
}
static void update_common_state_cbk(update_mode_info_t *info, u32 state, void *priv)
{
int type = info->type;
log_info("type:%x state:%x code:%x\n", type, state, priv);
switch (state) {
case UPDATE_CH_INIT:
memset((u8 *)&succ_report, 0x00, sizeof(succ_report_t));
update_init_common_handle(info->type);
break;
case UPDATE_CH_SUCESS_REPORT:
log_info("succ report stored\n");
memcpy((u8 *)&succ_report, (u8 *)priv, sizeof(succ_report_t));
break;
}
if (info->state_cbk) {
info->state_cbk(type, state, priv);
}
switch (state) {
case UPDATE_CH_EXIT:
update_exit_common_handle(info->type, priv);
break;
}
}
extern void update_module_init(void (*cbk)(update_mode_info_t *, u32, void *));
extern void testbox_update_init(void);
static void app_update_init(void)
{
update_module_init(update_common_state_cbk);
testbox_update_init();
}
__initcall(app_update_init);