file.obj//子目标文件名1
file2.obj//子目标文件名2
file3.obj//子目标文件名3
- o prog.out //连接器 *** 作指令,用来指定输出文件
- m prog.m//用来指定MAP文件
MEMORY
{ 略 }
SECTIONS
{ 略 }
otherlink.cmd
本命令文件link.cmd要调用的otherlink.cmd等其他命令文件,则文件的名字要放到本命令文件最后一行,因为放开头的话,链接器是不会从被调用的其他命令文件中返回到本命令文件。
2 MEMORY伪指令
MEMORY用来建立目标存储器的模型,SECTIONS指令就可以根据这个模型来安排各个段的位置,MEMORY指令可以定义目标系统的各种类型的存储器,及容量。MEMORY的语法如下:
MEMORY
{
PAGE 0 : name1[(attr)] : origin = constant,length = constant
name1n[(attr)] : origin = constant,length = constant
PAGE 1 : name2[(attr)] : origin = constant,length = constant
name2n[(attr)] : origin = constant,length = constant
PAGE n : namen[(attr)] : origin = constant,length = constant
namenn[(attr)] : origin = constant,length = constant
}
PAGE关键词对独立的存储空间进行标记,页号n的最大值为255,实际应用中一般分为两页,PAGE 0程序存储器和PAGE 1数据存储器。
name存储区间的名字,不超过8个字符,不同的PAGE上可以出现相同的名字(最好不用,免的搞混),一个PAGE内不许有相同的name。
attr的属性标识,为R表示可读;W可写X表示区间可以装入可执行代码;I表示存储器可以进行初始话,什么属性代码也不写,表示存储区间具有上述的四种属性,基本上我们都选择这种写法。
origin:略。
length:略。
下面是我经常用的2407的简单写法大家参考,程序从0x060是要避开加密位,不从0x0044开始更可靠一点,此例中的同名的页可以只写第一个,其后省略,但写上至少安全一点:
MEMORY
{
PAGE 0: VECS: origin = 0x0000, length 0x40
PAGE 0: PROG: origin = 0x0060, length 0x6000
PAGE 1: B0 : origin = 0x200, length 0x100
PAGE 1: B1 : origin = 0x300, length 0x100
PAGE 1: DATA: origin = 0x0860, length 0x0780
}
3 SECTIONS伪指令
SECTIONS指令的语法如下:
SECTIONS
{
.text: {所有.text输入段名} load=加载地址 run =运行地址
.data: {所有.data输入段名} load=加载地址 run =运行地址
.bss: {所有.bss输入段名}load=加载地址 run =运行地址
.other: {所有.other输入段名} load=加载地址 run =运行地址
}
配置好主程序的CMD文件,才能将FLASH成功烧录,并且将FLASH中的文件拷贝到RAM中运行。关于CMD文件的配置:
首先在F2812.CMD文件中,可以看到有关于加载FLASH到RAM的内容:
ramfuncs: LOAD = FLASHD,
以及在C文件中调用FLASH 到RAM的函数memcpy,将它放在系统初始化(InitSystem())之后运行。
InitSystem()
memcpy(&RamfuncsRunStart,
Initflash()
关于ramfuncs,则在系统初始化中定义即可。如:sysctrl.c中
在组织数据段和程序段的时候可以根据需要来分配内存,从而要更改cmd。比如通常DMA访问的内存块,和程序块的内存不要在同一块,这就需要组织好。再如有些数据可能为了块计算而需要
align
8,而有的变量只是
byte,这样也要在cmd中指定ram安排,以便更有效率。最后的mem分配,可以打开编译后的.map文件查看,以决定是否还需进一步优化mem分配。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)