嵌牛 9

姓名:龚树焱

学号:19029100008

学院:电子信息工程

转载自:https://mp.weixin.qq.com/s?__biz=MzI0ODU0NDI1Mg==&mid=2247494945&idx=2&sn=4f3e2463150ad44853a4b5fd1aaff99f&scene=19#wechat_redirect

【嵌牛导读】本文介绍了将Flash中的程序转移到RAM中运行的过程

【嵌牛鼻子】Flash,RAM

【嵌牛提问】如何实现将Flash中的程序转移到RAM中运行

【嵌牛正文】

一.将DSP的Flash里面的函数转移到RAM中

对于独立的嵌入式系统,需要把程序存入non-volitale存储单元中,常用的也就是flash。但是程序在flash中运行相对在RAM中行,速度会变慢很多,具体有多慢,拿28335来说吧,假设系统时钟为150MHz,在RAM中运行时频率还是150MHz,而放在flash中,频率会降到90-95MHz,参照Ti手册SPRA958L,这对于有些对实时性要求较高的函数功能,是不可接受的。所以在系统上电时,把对实时性要求高的函数转移到RAM中去。

下面以initflash函数为例,具体步骤如下:

(1)、将函数定位到section:

#pragma CODE_SECTION(InitFlash, "secureRamFuncs")

当遇到InitFlash(),就到段secureRamFuncs去运行。

当有多个函数需要转移时,重复使用#pragma CODE_SECTION(“函数名", "secureRamFuncs")即可。

即使有多个#pragma CODE_SECTION,后面的步骤只需要一次。

(2)、section分配到memory(红色为memory)。

意思是到FLASH去下载InitFlash(),下载到SECURE_RAM,然后要到SECURE_RAM去运行程序,这个过程给出了下载地址和目标地址。注意此时SECURE_RAM中还没有代码。

SECTIONS{/*** User Defined Sections ***/secureRamFuncs: LOAD = FLASH,PAGE = 0RUN =SECURE_RAM, PAGE = 0//定义FLASH和SECURE_RAM的首地址secureRamFuncs_loadstart和secureRamFuncs_loadstart以代替绝对地址LOAD_START(_secureRamFuncs_loadstart),LOAD_SIZE(_secureRamFuncs_loadsize),RUN_START(_secureRamFuncs_runstart),}

(3)、用memcpy()将经过#pragmaCODE_SECTION设定的函数从FLASH弄到SECURE_RAM中去。注意不是将FLASH的东西全部弄到SECURE_RAM中。

#include<string.h>//实际应用中这一部分声明可有可无externunsignedintsecureRamFuncs_loadstart;externunsignedintsecureRamFuncs_loadsize;externunsignedintsecureRamFuncs_runstart;voidmain(void){/* Copy the secureRamFuncs section */memcpy(&secureRamFuncs_runstart,&secureRamFuncs_loadstart,(Uint32)&secureRamFuncs_loadsize);/* Initialize the on-chip flash registers*/InitFlash();}

二.将MCU的内嵌Flash里的部分代码运行在 RAM 中

MCU 异于资源丰富的linux 平台。MCU(如:基于Cortex V6M 的Cortex M0+ 等) Code通常运行在内嵌Flash中。在某些特定应用场合,需要将部分函数运行于RAM 中。为解决次问题,笔者实现了一种解法,具体做法如下:

1. 实现要运行在RAM的 routine, 本routine 使用纯汇编实现, 如:

__asmvoidprogram_word2addr(uint32_taddr,uint32_tdata){        push {r3, r4, r5, lr}  ;save some regsiters/*your code for this routine*/      pop {r3, r4, r5, pc}  }

2.编译时,采用code 与运行位置无关的编译选项 如 (Keil --apcs /ropi/rwpi), 生成 *.axf;

3.通过fromelf -c 将生成 *.axf 反汇编,找到对应program_word2addr 实现部分, 并将routine 对应的binary code Copy 到所要应用的 Code 中,以只读数组的形式出现:

      如:

conststaicuint16_ts_flashProg2AddressCode[16] = {...., ....}

4.定义 一个全局数组, 如 static uint16_t g_code[16], size正好等于 s_flashProg2AddressCode的长度;

5. 定义一个函数指针, 如  static void (*callFlashPrg2Address)(uint32_t addr, uint32_t data)

6.定义一个函数实现将Code 运行与 RAM如:

voidrun_prgcode_onram(uint32_taddr,uint32_tdata){memcpy(g_code,s_flashProg2AddressCode,32);callFlashPrg2Address = (void(*)(uint32_taddr,uint32_tdata))((uin32_t)g_code +1);            callFlashPrg2Address (address, data);    }

     run_prgcode_onram, 便可以将program_word2addr 运行于RAM中。 

callFlashPrg2Address = (void (*)(uint32_t addr, uint32_t data))((uin32_t)g_code + 1); +1 的目的,时由于运行平台为 Cortex V6M , 采用的thumb指令集,根据ARM Spec 要求完成。

callFlashPrg2Address (address, data); 则是实现RAM运行program_word2addr 的关键所在。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 关键时刻,第一时间送达! 问题种类 时间复杂度 在集合里数据量小的情况下时间复杂度对于性能的影响看起来微乎其微。但...
    C9090阅读 932评论 0 1
  • 三:Sensor SLPI层代码分析 在学习SLPI侧代码前我们先了解下SEE的registry&config。 ...
    seuwt阅读 2,175评论 0 50
  • STM32 启动流程是怎样的? 根据启动模式,从不同的存储器加载程序运行。STM32的启动代码在startup_s...
    我在等你回复可你没回阅读 1,157评论 0 0
  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 126,303评论 2 7
  • 16宿命:用概率思维提高你的胜算 以前的我是风险厌恶者,不喜欢去冒险,但是人生放弃了冒险,也就放弃了无数的可能。 ...
    yichen大刀阅读 6,122评论 0 4