你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

stm32f1使用IAP升级后STemwin显示图片速度变慢

[复制链接]
山竹君 提问时间:2022-1-14 16:17 / 未解决
我使用STM32F103VET6进行IAP升级。
现在将512K的flash划分为以下四个区域:
0x8000000——0x800B400的45K为bootloader区,存放IAP升级程序。
0x800B400——0x8030C00的150K为APP1区,存放第一个app程序。
0x8030C00——0x8056400的150K为APP2区,存放第二个app程序。
0x8056400往后的区域为IAP升级标志区,存放IAP升级标志。

app程序中使用了ucos操作系统和STemwinGUI。
我遇到的问题是,同一个app程序烧录在APP1区和APP2区后,STemwin的图片显示函数:"GUI_DrawBitmap()"显示图片的速度不一样。APP1区的显示速度很快,但APP2区的显示速度就变得很慢,通常刷一张图片要730ms左右。而且经过我的测试,APP2在flash的烧录位置越靠后,这个刷图函数就越慢。
在app程序中除了这个刷图慢的问题外其他功能都正常。
可以确保正常的app程序中配置没问题,应为已经测试过了。两个APP程序的唯一区别就是在main函数开头是用了SCB->VTOR = FLASH_BASE | 0xB400;来偏移中断向量表。
如果不使用IAP升级,把APP程序从0x8000000地址开始烧,APP程序就正常运行STemwin也没有任何问题。

想问一下大家有没有遇到过类似的问题或者能帮我提供一些解决思路,谢谢

以下是IAP的升级函数

void app_updata (uint16_t app_flg)        {//传参为APP固件上次烧写标志

        u16 j=0;
        u16 temp;
        uint16_t app_num = 0;
        u32 fwaddr = FLASH_APP1_ADDR;//当前写入的地址
        u16 u;
        int i = 0;
        char pathbuf[30];
       
       
        if (app_flg == APP_1)        {        //上次从一区起的,这次烧二区
                fwaddr = FLASH_APP2_ADDR;
                strcpy(pathbuf, APP2_NAME);
        }
        else if (app_flg == APP_2)        {        //上次从二区起的,这次烧一区
                fwaddr = FLASH_APP1_ADDR;
                strcpy(pathbuf, APP1_NAME);
        }
       
        mount_flash();
        res_flash = f_open(&fnew, pathbuf, FA_OPEN_EXISTING | FA_READ );//打开外部flash中的固件
        if (res_flash == FR_NO_FILE)        {
               
                if (app_flg == APP_1)        {
                        printf("无固件升级包,跳转到APP一区执行\r\n");
                        iap_load_app(FLASH_APP1_ADDR);//执行FLASH APP代码
                }
                else {
                        printf("无固件升级包,跳转到APP二区执行\r\n");
                        iap_load_app(FLASH_APP2_ADDR);//执行FLASH APP代码
                }
        }
        else if (res_flash == FR_OK) {
                recordsize = f_size(&fnew);
                app_num = recordsize / READ_SIZE;
                if (recordsize != 0)        {
                        printf( "打开固件%s成功,字节数为:%ld\r\n", pathbuf, recordsize);
                        printf ("准备开始固件升级!\r\n");
                        for (i = 0; i < app_num; i++)        {                //读固件数据
                                res_flash = f_lseek(&fnew, i * READ_SIZE);
                                memset(app_buf, 0, READ_SIZE);
                                res_flash = f_read(&fnew, app_buf, READ_SIZE, &fnum);                                                       
                                u8 *dfu = app_buf;
                                for(u = 0; u < READ_SIZE; u += 2)
                                {                                                    
                                        temp = (u16)dfu[1] << 8;
                                        temp += (u16)dfu[0];          
                                        dfu += 2;//偏移2个字节
                                        iapbuf[j++] = temp;            
                                        if(j == 1024)
                                        {
                                                j = 0;
                                                STMFLASH_Write(fwaddr, iapbuf, 1024);       
                                                fwaddr += 2048;//偏移2048  16=2*8.所以要乘以2.
                                        }
                                }
                                printf("固件升级中:读到%d字节,还剩%ld字节\r\n", READ_SIZE, (recordsize - (i + 1) * READ_SIZE));
                        }
                        //读剩余的不满1024的字节
                        res_flash = f_lseek(&fnew, app_num * READ_SIZE);
                        memset(app_buf, 0, READ_SIZE);
                        res_flash = f_read(&fnew, app_buf, recordsize % READ_SIZE, &fnum);
                        u8 *dfu = app_buf;
                        for(u = 0; u < recordsize % READ_SIZE; u += 2)
                        {                                                    
                                temp = (u16)dfu[1] << 8;
                                temp += (u16)dfu[0];          
                                dfu += 2;//偏移2个字节
                                iapbuf[j++] = temp;            
                        }
                        STMFLASH_Write(fwaddr, iapbuf, j);
                       
                        printf ("最后读到%ld字节\r\n", recordsize % READ_SIZE);
                        if (app_flg == APP_1)                                       
                                f_unlink("0:/iCasa_multimeter_2.bin");        //删除固件包
                        else f_unlink("0:/iCasa_multimeter_1.bin");       
//                                                unmount_flash();
                        f_close(&fnew);
                       
                        beep_en();                //蜂鸣器响一声代表固件升级完成
                        delay_ms(100);
                        beep_dis();
                        if (app_flg == APP_1)        {
                                Test_Write(FLASH_BAK_ADDR, APP_2);                //往flash写入标志,表示当前写入的是那个APP区
                                printf("固件升级完成,准备跳转至APP二程序!\r\n");
                                iap_load_app(FLASH_APP2_ADDR);//执行FLASH APP代码
                        }
                        else {
                                Test_Write(FLASH_BAK_ADDR, APP_1);                //往flash写入标志,表示当前写入的是那个APP区
                                printf("固件升级完成,准备跳转至APP一程序!\r\n");
                                iap_load_app(FLASH_APP1_ADDR);//执行FLASH APP代码
                        }
                }
                else {        //打开文件为空
                       
                        if (app_flg == APP_1)        {
                                printf("固件包为空,跳转到APP一区执行\r\n");
                                iap_load_app(FLASH_APP1_ADDR);//执行FLASH APP代码
                        }
                        else {
                                printf("固件包为空,跳转到APP二区执行\r\n");
                                iap_load_app(FLASH_APP2_ADDR);//执行FLASH APP代码
                        }
                }
        }        //f_open出错
        else
                printf("打开固件包失败");
}


收藏 评论3 发布时间:2022-1-14 16:17

举报

3个回答
liuxq80 回答时间:2022-1-14 16:44:53

先学习一下。同问。

butterflyspring 回答时间:2022-1-17 10:50:40
这个一下子很难辨别哪里有差异。建议楼主跟踪一下代码,看看具体是画图部分还是刷图部分的代码执行变慢。不具体测试很难有线索的
山竹君 回答时间:2022-1-18 08:32:04

butterflyspring 发表于 2022-1-17 10:50
这个一下子很难辨别哪里有差异。建议楼主跟踪一下代码,看看具体是画图部分还是刷图部分的代码执行变慢。不 ...

跟踪过了,就是GUI_DrawBitmap();这个函数调用慢,画一次图要大概730ms

关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版