7.18. 资源区配置(RES和预留区)
Overview
资源区为用户存放资源文件的区域,其中分为两个区域:RES区域和预留区(RESERVED) (FLASH的区域分布请参考: UPDATE ).
RES区:res资源区为app code中的一部分,在flash空间允许的范围内,res资源区大小没有限制。在双备份升级中,res资源区也会进行双备份处理,因此res资源在flash中会占用两倍的空间。
预留区: 预留区为独立于app code的一个区域,是客户可以自行配置和使用的区域,即系统会根据客户的配置文件在flash中预留出对应的空间大小由客户自行操作。
7.18.1. RES资源区配置
添加res区资源目录:
修改脚本文件 ``download.c``,在“-res” 选项增加资源目录,如:添加资源目录audlogo isd_download.exe isd_config.ini -tonorflash -dev wl82 -boot 0x1c02000 -div1 -wait 300 -uboot uboot.boot -app app.bin cfg_tool.bin -res cfg audlogo source %AUDIO_RES% %UI_RES% %CFG_FILE%
添加资源文件
只需要在已添加的资源目录中加入资源文件,如audlogo目录中增加提示音
资源文件读取
可以通过文件系统接口fread对res区资源进行读取,其路径格式为:mnt/sdfile/res/$(user_dir)/$(filename); 其中user_dir为用户自定义资源目录,filename为用户添加的资 char name[128]; char read_buf[BUF_SIZE]; FILE *f = fopen("mnt/sdfile/res/audlogo/test.mp3", "r"); if (!f) { printf("fopen err!\n") }else{ fget_name(f, name, sizeof(name)); printf("file_name: %s\n", name); int len = fread(read_buf, 1, sizeof(read_buf), f); if(len > 0) { put_buf(read_buf, sizeof(read_buf)); } }
7.18.2. 预留区配置
预留区配置
修改配置文件cpu/wl82/tools/isd_config_rule.c,[RESERVED_EXPAND_CONFIG]或[RESERVED_EXPAND_CONFIG]下添加配置项 注:具体配置说明请参考 doc/stuff/ISD_CONFIG.INI配置文件说明.pdf ,如: #预留区扩展 [RESERVED_EXPAND_CONFIG] fixed.mp3_FILE=fixed_res/fixed.mp3; #fixed_res目录下存放fixed.mp3文件 fixed.mp3_ADR=AUTO; #由工具自动分配起始地址 fixed.mp3_LEN=0x4000; #fixed.mp3文件的大小,需要4K对齐 fixed.mp3_OPT=1; #0:下载代码时擦除指定区域;1:下载代码时不操作指定区域;2:下载代码时给指定区域加上保护; 或 #预留区扩展 [RESERVED_CONFIG] fixed.mp3_FILE=fixed_res/fixed.mp3; #fixed_res目录下存放fixed.mp3文件 fixed.mp3_ADR=AUTO; #由工具自动分配起始地址 fixed.mp3_LEN=0x4000; #fixed.mp3文件的大小,需要4K对齐 fixed.mp3_OPT=1; #0:下载代码时擦除指定区域;1:下载代码时不操作指定区域;2:下载代码时给指定区域加上保护;
预留区路径格式
(1)[RESERVED_EXPAND_CONFIG]预留区使用方法可以通过文件系统接口fread对预留区数据进行读取,其路径格式为:mnt/sdfile/EXT_RESERVED/XXX(其中XXX为预留区文件名),如: char name[128]; char read_buf[BUF_SIZE]; FILE *f = fopen("mnt/sdfile/EXT_RESERVED/fixed.mp3", "r"); if (!f) { printf("fopen err!\n") }else{ fget_name(f, name, sizeof(name)); printf("file_name: %s\n", name); int len = fread(read_buf, 1, sizeof(read_buf), f); if(len > 0) { put_buf(read_buf, sizeof(read_buf)); } } (2)[RESERVED_CONFIG]预留区使用方法可以通过文件系统接口fread对预留区数据进行读取,其路径格式为:mnt/sdfile/app/XXX(其中XXX为预留区文件名),如: char name[128]; char read_buf[BUF_SIZE]; FILE *f = fopen("mnt/sdfile/app/fixed.mp3", "r"); if (!f) { printf("fopen err!\n") }else{ fget_name(f, name, sizeof(name)); printf("file_name: %s\n", name); int len = fread(read_buf, 1, sizeof(read_buf), f); if(len > 0) { put_buf(read_buf, sizeof(read_buf)); } }
预留区读写操作:可以通过fread来读取数据,但是不能使用fwrite接口来写入数据。一般采用Flash接口来对预留区进行操作,示例如下:
#include "app_config.h" #include "system/includes.h" #include "fs/fs.h" #include "asm/sfc_norflash_api.h" #define USER_FLASH_SPACE_PATH "mnt/sdfile/app/exif" static u32 user_get_flash_exif_addr(void) { u32 addr; //打开预留区 FILE *profile_fp = fopen(USER_FLASH_SPACE_PATH, "r"); if (profile_fp == NULL) { puts("user_get_flash_addr ERROR!!!\r\n"); return 0; } struct vfs_attr file_attr; fget_attrs(profile_fp, &file_attr); //获取预留区的flash地址 addr = sdfile_cpu_addr2flash_addr(file_attr.sclust); fclose(profile_fp); printf("user_get_flash_exif_addr = 0x%x, size = 0x%x \r\n", addr,file_attr.fsize); return addr; } static int c_main(void) { printf("\r\n\r\n\r\n\r\n\r\n ----------- USER_FLASH_EXIF example run %s-------------\r\n\r\n\r\n\r\n\r\n", __TIME__); char buf[256]; u32 flash_exif_addr = user_get_flash_exif_addr(); if(flash_exif_addr==0) return -1; //擦除一个扇区 puts("USER_FLASH_EXIF ERASE_SECTOR...\r\n"); norflash_ioctl(NULL, IOCTL_ERASE_SECTOR, flash_exif_addr); puts("USER_FLASH_EXIF READ\r\n"); memset(buf,0,sizeof(buf)); //读取预留区数据 norflash_read(NULL, buf, sizeof(buf), flash_exif_addr); put_buf(buf,sizeof(buf)); puts("\r\n USER_FLASH_EXIF WRITE\r\n"); for(int i=0;i<sizeof(buf);i++) buf[i] = i; //写入数据 norflash_write(NULL, buf, sizeof(buf), flash_exif_addr); puts("USER_FLASH_EXIF READ\r\n"); memset(buf,0,sizeof(buf)); //再次读出来,查看数据是否写入成功 norflash_read(NULL, buf, sizeof(buf), flash_exif_addr); put_buf(buf,sizeof(buf)); return 0; } late_initcall(c_main);
7.18.3. 文件打包工具使用
打包工具位于 cpu\wl82\tools\packres
, 打开packrec.bat文件, 按照说明添加需要打包的文件,如:
REM packres.exe -n $(dir) -o $(output) $(file1) $(file2) ... REM 其中$(dir)为打包后的文件目录入口,$(output)打包后的输出文件,$(file1) $(file2) ...为需要打包的输入文件 REM 放置在res资源区时,其搜索路径为mnt/sdfile/res/$(output)/$(dir)/$(file),如:mnt/sdfile/res/tone/res1.txt REM 放置在[RESERVED_CONFIG]预留区时,其搜索路径为:mnt/sdfile/app/$(output)/$(dir)/$(file),如:mnt/sdfile/app/update/tone/res1.txt REM 放置在[RESERVED_EXPAND_CONFIG]预留区时,其搜索路径为:mnt/sdfile/EXT_RESERVED/$(output)/$(dir)/$(file),如:mnt/sdfile/EXT_RESERVED/update/tone/res1.txt packres.exe -n tone -o UPDATE res1.txt res2.txt res3.txt ::pause
7.18.4. 资源文件打包放在预留区或者扩展预留区
若资源区使用双备份升级时,会占用flash两倍的资源空间,为了节省双备份升级时的固件空间,现支持统一把提示音和UI资源统一放在预留区或者扩展预留区, 资源升级采用的是断点可恢复升级的单备份升级方法(若放在扩展预留区,需要在应用层上由客户手动强制升级,一般来说放在扩展预留区的资源是不升级的)。
资源打包放在预留区的配置
//app_config.h配置对应的宏 #if defined CONFIG_UI_ENABLE && !defined CONFIG_SDFILE_EXT_ENABLE #define CONFIG_UI_FILE_SAVE_IN_RESERVED_ZONE //UI资源打包后放在预留区,可以通过升级预留区更新此资源,一般用于双备份时UI资源小于代码大小的方案 #endif #if defined CONFIG_AUDIO_ENABLE && !defined CONFIG_SDFILE_EXT_ENABLE #define CONFIG_VOICE_PROMPT_FILE_SAVE_IN_RESERVED_ZONE //提示音资源打包后放在预留区,可以通过升级预留区更新此资源,一般用于双备份时提示音资源小于代码大小的方案 #endif //在代码编译前,需要根据实际打包后的资源文件空间大小填写tools/isd_config_rule.c里相应的flash空间区域配置 [RESERVED_CONFIG] //#升级之后需要保留VM数据,在生成升级文件时需要设置VM_OPT=1 VM_ADR=0; [设置VM] VM_LEN=32K; #if CONFIG_DOUBLE_BANK_ENABLE VM_OPT=0; #else VM_OPT=1;//单备份升级VM在升级时候默认VM不需要擦除,选择擦除会在ota_loader第二阶段擦除比较长,而且可能会造成VM丢失 #endif BTIF_ADR=AUTO; [设置资源] BTIF_LEN=0x1000; BTIF_OPT=1; #if defined CONFIG_AUDIO_ENABLE && defined CONFIG_VOICE_PROMPT_FILE_SAVE_IN_RESERVED_ZONE AUPACKRES_FILE=packres/AUPACKRES; [打包后的资源文件的相对路径] AUPACKRES_ADR=0xXXXXXX; [请根据编译后FLASH INFO打印的实际地址填写,比如0x59a000,烧录后不支持升级更改] AUPACKRES_LEN=0xXXXXXX; [更新提示音资源打包后必须更新此实际长度,比如0x141000,建议根据后续资源升级的需求预留好足够的空间,烧录后不支持升级更改] AUPACKRES_OPT=1; #endif #if defined CONFIG_UI_ENABLE && defined CONFIG_UI_FILE_SAVE_IN_RESERVED_ZONE UIPACKRES_FILE=packres/UIPACKRES; UIPACKRES_ADR=0xXXXXXX; [请根据编译后FLASH INFO打印的实际地址填写,比如0x6db000,烧录后不支持升级更改] UIPACKRES_LEN=0xXXXXXX; [更新UI资源打包后必须更新此实际长度,比如0x123000, 建议根据后续资源升级的需求预留好足够的空间,烧录后不支持升级更改] UIPACKRES_OPT=1; #endif PRCT_ADR=0; PRCT_LEN=CODE_LEN; PRCT_OPT=2;
资源打包放在扩展预留区的配置
//app_config.h配置对应的宏 #if defined CONFIG_UI_ENABLE && !defined CONFIG_SDFILE_EXT_ENABLE #define CONFIG_UI_FILE_SAVE_IN_RESERVED_EXPAND_ZONE //UI资源打包后放在扩展预留区,不可以通过常规升级更新此资源,一般用于UI不需要更新的方案 #endif #if defined CONFIG_AUDIO_ENABLE && !defined CONFIG_SDFILE_EXT_ENABLE #define CONFIG_VOICE_PROMPT_FILE_SAVE_IN_RESERVED_EXPAND_ZONE //提示音资源打包后放在扩展预留区,不可以通过常规升级更新此资源,一般用于提示音不需要更新的方案 #endif //在代码编译前,需要根据实际打包后的资源文件空间大小填写tools/isd_config_rule.c里相应的flash空间区域配置 [RESERVED_EXPAND_CONFIG] USER_ADR=AUTO; [固定预留给客户,避免客户量产后,想通过升级新增重要信息的保存却没有预先预留空间] USER_LEN=0x1000; USER_OPT=1; //packers文件夹下会生成AUPACKRES,UIPACKRES这两个包是资源文件打包好的包需要 //根据实际大小填写AUPACKRES_ADR,AUPACKRES_LEN, //AUPACKRES_ADR 配置 AUTO 第一次下载后会下载不进 需要根据FLASH INFO打印的实际地址填写 //例如AUPACKRES 378KB = 0x5e800 实际填大小可以填大一些 AUPACKRES_LEN = 0x5f000 #if defined CONFIG_AUDIO_ENABLE && defined CONFIG_VOICE_PROMPT_FILE_SAVE_IN_RESERVED_EXPAND_ZONE AUPACKRES_FILE=packres/AUPACKRES; [烧录后不可升级的资源] AUPACKRES_ADR=0xXXXXXX; [请根据编译后FLASH INFO打印的实际地址填写,比如0x59b000,建议预留好足够的空间,烧录后不支持升级更改] AUPACKRES_LEN=0xXXXXXX; [更新提示音资源打包后必须更新此实际长度,比如0x141000,,建议预留好足够的空间,烧录后不支持升级更改] AUPACKRES_OPT=1; #endif #if defined CONFIG_UI_ENABLE && defined CONFIG_UI_FILE_SAVE_IN_RESERVED_EXPAND_ZONE UIPACKRES_FILE=packres/UIPACKRES; [烧录后不可升级的资源] UIPACKRES_ADR=0xXXXXXX; [请根据编译后FLASH INFO打印的实际地址填写,比如0x6dc000,建议预留好足够的空间,烧录后不支持升级更改] UIPACKRES_LEN=0xXXXXXX; [更新UI资源打包后必须更新此实际长度,比如0x123000,建议预留好足够的空间,烧录后不支持升级更改] UIPACKRES_OPT=1; #endif
非常规方法升级扩展预留区资源
//因为扩展预留区不能通过常规的OTA升级更新,需要用户在应用层自行获取升级资源后,直接通过flash擦写的方法去更新扩展预留区资源,具体参考apps/common/update/expand_zone_file_update.c //测试时需要修改isd_config.ini文件把UIPACKRES_LEN设置成AUPACKRES_LEN同一个值 static int expand_zone_packres_file_test(void) { u32 au_update_addr, ui_update_addr; void *data = malloc(SDFILE_SECTOR_SIZE); check_expand_zone_packres_file_crc("mnt/sdfile/EXT_RESERVED/aupackres"); check_expand_zone_packres_file_crc("mnt/sdfile/EXT_RESERVED/uipackres"); au_update_addr = expand_zone_packres_file_update_start("mnt/sdfile/EXT_RESERVED/aupackres"); ui_update_addr = expand_zone_packres_file_update_start("mnt/sdfile/EXT_RESERVED/uipackres"); for (u32 offset = 0; offset < 0x141000; offset += SDFILE_SECTOR_SIZE) { sdfile_reserve_zone_read(data, au_update_addr + offset, SDFILE_SECTOR_SIZE, 0); //把UI资源擦写成提示音资源 expand_zone_packres_file_update_write(ui_update_addr + offset, data, SDFILE_SECTOR_SIZE); } free(data); //升级完成后重新校验文件完整性 check_expand_zone_packres_file_crc("mnt/sdfile/EXT_RESERVED/aupackres"); check_expand_zone_packres_file_crc("mnt/sdfile/EXT_RESERVED/uipackres"); return 0; }
7.18.5. 常见问题
(1) [RESERVED_CONFIG]与[RESERVED_EXPAND_CONFIG]的区别?
答:[RESERVED_CONFIG]
区域最多可以设置6个配置项,其中VM、BTIF和PRCT为系统默认使用,用户不能占用。
用户可使用的只有3个配置项(如:配置文件中PROFILE、AISP和EXIF),可以根据实际需要进行使用和修改。当超出使用个数时,可以使用 [RESERVED_EXPAND_CONFIG]
区域。
当使用预留区作为资源升级区时,只能使用 [RESERVED_CONFIG]
; [RESERVED_EXPAND_CONFIG]
区域暂时不支持作为资源升级区使用。 [RESERVED_EXPAND_CONFIG]
区域在flash空间足够的情况下没有个数限制。
(2) 预留区中指定文件,但是没有填写绝对地址导致编译器报错问题,如“错误:在配置文件(isd_config.ini)中发现非标准的预留区域配置。”?
答:当预留区指定文件时,只有编译后才知道工具实际给文件分配的地址,因此需要对isd_config_rule.c 文件进行修改。如上图:在FLASH INFO中获知AUPACKRES_RESERVED_SIZE : 0x141000, AUPACKRES_RESERVED_START : 0x6bd000,在isd_config_rule.c修改AUPACKRES_ADR=0x6bd000, AUPACKRES_LEN=0x141000。再编译工程即可。