您的位置 首页 设计

DSP编程技巧–详解cmd文件

cmd文件是编译完成之后链接各个目标文件时,用来指示各个数据、符号等是如何划分到各个段,以及每个段所使用的存储空间的。许多筒子对cmd文件有畏难情绪,不容易理解各个段的含义,特别是在程序编译没有问题,

  cmd文件是编译完结之后链接各个方针文件时,用来指示各个数据、符号等是怎么区分到各个段,以及每个段所运用的存储空间的。许多筒子对cmd文件有畏难情绪,不简单了解各个段的意义,特别是在程序编译没有问题,可是在链接生成可执行的.out遇到过错时更简单不知所措,所以咱们就来详细解读一下cmd文件的详细意义。

  C28x的编译器把存储空间区分为两个部分进行办理,包含:

  1. 程序存储空间:包含可执行的代码,初始化的记载和switch-case运用的表。

  2. 数据存储空间:包含外部变量,静态变量以及体系的栈;一般情况下,各个寄存器对应的存储空间也归类在数据空间里。

  为了便利办理,不同品种的代码、变量等往往又被别离分配到不同的段(section)之中,然后对存储空间的区分就变成了对段的地址分配问题了。例如,鄙人面的代码中,就规则了.text这个段会寄存在RAM中Page0下面的RAML1中,RAML1的开始地址是0x009000,长度是0x001000。

  MEMORY

  {

  /* 省掉不在此显现的代码 */

  PAGE 0 :

  RAML1 : origin = 0x009000, length = 0x001000

  RAML2 : origin = 0x00A000, length = 0x001000

  /* 省掉不在此显现的代码 */

  }

  SECTIONS

  {

  /* 省掉不在此显现的代码 */

  .text : > RAML1, PAGE = 0

  /* 省掉不在此显现的代码 */

  }

  一般情况下,咱们的代码不会大到无法存储,可是也有或许由于代码特别多导致无法存储,发生.text的实践巨细是size xxx,可是RAML1的size只要yyy这样的链接过错,以至于无法生成输出文件。此刻咱们能够把上面对应的RAML1的长度,即length增大,使得.text段所分配的地址空间变多。可是RAML1地址空间扩展之后,挤占了RAML2的空间,导致地址堆叠,此刻RAML2的开始方位要后移,其长度也要相应地减缩,才干不发生地址掩盖过错;修正之后能够为:

  RAML1 : origin = 0x009000, length = 0x001500

  RAML2 : origin = 0x00A500, length = 0x000500

  还有一个解决办法则是把.text给分配到其它更长的地址空间里去;假如没有现成的地址规模比较长的段,也能够兼并现有的段,修正办法比方把RAML2删去,把它的地址悉数兼并到RAML1中去,而.text仍是分配在RAML1,就没有问题了。删去RAML2的时分要留意,它在没有被任何段运用的情况下才干操作,不然编译、链接的时分又提示其它的段找不到对应的存储单元了。

  下面咱们就解释一下各个段的意义:

  一.初始化的段

  其间包含了数据和可执行代码,通常情况下是只读的。它们包含:

  1 .cinit和.pinit

  包含了初始化变量和常量所用的表格,是只读的。

  C28x .cinit被约束在16bit规模内,即低64K规模。

  2 .const

  包含了字符串常量、字符串文字、挑选表以及运用const关键字界说(可是不包含volatile类型,并假定运用小内存模型)的只读型变量。

  3 .econst

  包含了字符串常量,以及运用far关键字界说的大局变量和静态变量。

  4 .switch

  寄存switch-case指令所运用的挑选表。

  5 .text

  通常是只读的,包含一切可执行的代码,以及编译器编译发生的常量。

  二.无初始化的段

  无初始化的段尽管不会被初始化,可是依然需求在存储单元(一般是RAM)中保存相关的地址空间。它们包含:

  1 .bss

  为大局和静态变量保存存储空间。在发动或许程序加载的时分,C/C++的发动程序会把.cinit段中的数据(一般寄存在ROM中)复制到.bss段中。

  2 .ebss

  为far关键字界说(仅适用于C代码)的大局和静态变量保存存储空间。在发动或许程序加载的时分,C/C++的发动程序会把.cinit段中的数据(一般寄存在ROM中)复制到.ebss段中。

  3 .stack

  默许情况下,栈(stack)保存在.stack段中(参阅boot.asm),这个段用来为栈保存存储空间。栈(stack)的效果主要有:

  1) 保存存储空间用于存储传递给函数的参数;

  2) 为局部变量分配相关的地址空间;

  3) 保存处理器的状况;

  4) 保存函数的回来地址;

  5) 保存某些暂时变量的值。

  需求留意的是,.stack段只能运用低64K地址的数据存储单元,由于CPU的SP寄存器是16位的,它无法读取超越64K的地址规模。此外,编译器无法查看栈的溢出过错(除非咱们自己编写某些代码来检测),这将导致过错的输出成果,所以要为栈分配一个相对较大的存储空间,它的默许值是1K字。改动栈的巨细的操作能够经过编译器选项–stack_size来完结。

  4 .sysmem

本文引用地

  为动态内存分配保存存储空间,从而为malloc,calloc,realloc和 new等动态内存分配程序服务。假如这几个动态内存办理函数没有在C/C++代码中用到的话,则不需求创立.sysmem段。

  此外,咱们常常说到“仓库”,在这里咱们只讲了栈,那堆(heap)是干啥的呢?堆便是是用来做动态内存分配的,由于在DSP上RAM资源依然是相对名贵的,所以堆占用的存储空间不能无限扩展,关于near关键字润饰的堆,其占用的地址空间最大只能到32K字;关于far关键字润饰的堆,它运用的存储空间由编译器主动设置,默许只要1K字。

  5 .esysmem

  为far malloc函数分配动态存储空间。假如没有用到这个函数,则编译器不会主动创立.esysmem段。

  关于汇编器,它会主动创立.text, .bss和.data三个段。咱们能够运用#pragma CODE_SECTION和#pragma DATA_SECTION来创立更多的段。

  默许情况下,各个段所分配的存储空间装备如下(可根据需求进行更改):

  最终,以一个ADC寄存器对应的内存地址分配的比方,来看看完结的cmd文件是怎么完结的(事实上一切寄存器的内存地址分配在TI的外设和头文件包中现已帮咱们做好了,这里是个演示)。

  首要,在运用寄存器(或许自界说的变量)的头文件或许源程序里,为寄存器(或许自界说的变量)指定一个自界说的段:

  #ifdef__cplusplus

  #pragmaDATA_SECTION(“AdcRegsFile“)

  #else

  #pragmaDATA_SECTION(AdcRegs,”AdcRegsFile“);

  #endif

  volatile structADC_REGS AdcRegs; //使得结构体被分配在指定的段中

  然后,在cmd文件中,在SECTIONS下把AdcRegsFile这个段分配到ADC这块内存区域中,并在MEMORY中界说ADC这块内存区域的开始方位和长度。

  MEMORY

  {

  PAGE 0:/* Program Memory */

  /* 省掉不相关内容的显现 */

  PAGE 1: /* Data Memory */

  /* 省掉不相关内容的显现 */

  ADC : origin = 0x007100, length = 0x000020 /* ADC registers */

  /* 省掉不相关内容的显现 */

  }

  SECTIONS

  {

  /* 省掉不相关内容的显现 */

  AdcRegsFile : > ADC, PAGE = 1

  /* 省掉不相关内容的显现 */

  }

  以上是一个自界说段并拟定内存区域的完好比方。假如不需求这样的自界说,则能够不去管它,运用现有的,比方某个比方中能够运用的cmd文件就能够了。

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/ziliao/sheji/226847.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部