用C语言设计TMS320C2X/C5X应用程序

2019-08-03 16:03发布

一、存储器模式
        TMS320C2X/C5X定点处理器有两种类型的存储器:程序存储器和数据存储器。在程序存储器中主要包含可执行的程序代码,在数据存储器中,则主要包含外部变量、静态变量和系统堆栈。由C程序生成的每一块程序或数据存放于存储空间的一个连续块中。
        (一)C编译器生成的块
        由TMS320C2X/C5X编译器编译生成的块共有六种,可分为两大类。在六种块中,除了具有浮点C编译器生成的五种块之外,还有一个.switch块,它是一个已初始化块,包含为 .switoh语句建立的表格。
        (二)C系统堆栈
        定点C系统堆栈的作用与浮点c编译器的C系统堆栈的作用完全相同,管理堆栈的方法也基本类似。但管理C相同堆栈所用的寄存器是不一样的。定点C编译器采用下面两个寄存器来管理这个堆栈:
AR1——是堆栈指针(SP)。它指向堆栈的顶部。
AR0——帧指针(FP)。指向当前帧的开始。
激活每个函数时,都在堆栈中建立一个新的帧,以用于分配局部变量和临时变量。C环境能够自动管理这些寄存器。如果需要编写用到运行堆栈的汇编程序,必须正确使用这些寄存器。
        与浮点C编译器一样,定点C编译器的堆栈长度也由链接器确定,全局符号_STACK_SIZE的值等于堆栈长度,单位为字节,缺省值为1K字节。同样,需要改变堆栈长度时,在链接时用-stack选项,并在其后指定一个数值。
        (三)动态存储器分配
        在运行支持库中,有几个允许在运行时进行动态存储器分配的函数,如malloc、calloc、realloc,动态存储器分配的方法与浮点C编译器的动态存储器分配完全相同。
        (四)静态和全局变量的存储器分配
        在C程序中说明的每一个外部或静态变量被分配给一个唯一的连续空间。空间的地址
由链接器确定。编译器保证这些变量的空间分配在多个字中以使每个变量按字边界对准。
        (五)域/结构的对准
        编译器为结构分配空间时,它分配足够的字以包含所有的结构成员,在一组结构中,每个结构开始于字边界。
        所有的非域类型对准于字的边界。对域分配足够多的比特,相邻域组装进一个字的相邻比特,但不跨越两个字。如果一个域要跨越两个字,则整个域分配到下一个字中。
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
6条回答
huangfeng33
2019-08-03 22:51
  三、函数调用规则
        定点C编译器也规定了一组严格的函数调用规则。除了特殊的运行支持函数外,任何调用C函数或被C函数所调用的函数都必须遵循这些规则,否则就会破坏C环境,造成不可预测的后果。
        (一)参数传递
        将参数传递给一个C函数时,必须遵循下列规则:
        (1)函数调用前,将多数压入运行堆栈。。
        (2)以逆序传递参数。也就是说。第一个参数(最左边)最后压栈,而最后一个参数(最右边)最先压栈。
        (3)若参数是浮点数或长整型数,则低位字先压栈,高位字后压栈。
        (4)传递结构时,采用多字方式。
        (二)局部帧的产生
        函数被调用时,编译器在运行栈中建立一个帧以存储信息。当前函数帧称为局部帧。C环境利用局部帧来保护调用者的有关信息、传递参数和生成局部变量。每调用一个函数,就建立一个新的帧。
        上面已经介绍,寄存器AR1为SP,AR0为FP,SP指向栈顶,FP指向局部帧。编译器在建立局部帧时完成如下工作:
        (1)从TMS320C2X/C5X的内部堆栈中弹出返回地址,并压人C运行堆栈。
        (2)将旧的FP的内容压人C运行堆栈,并将FP设置为当前的SP。
        (3)增加SP。增加的值等于需要保存的局部变量的字的个数加1,其中,额外的一个字位于帧的开始,用于存储临时变量。
        (4)若函数使用AR6和AN7作为寄存器变量,则将它们的内容压人堆栈,然后装人相应局部变量的地址。
        下面是完成上述工作的TMS320C2X的汇编程序,其中SIZE是局部帧的长度。
        例2.1  调用函数时的初始处理
POPD * +    ;将返回地址压人C堆栈
SAR AR0, * +    ;保护旧FP
SAR AR1, *
LARK AR0, SIZE
LAR ARO, * 0 +    ;FP= 旧的SP,SP=SP+ SIZE
SAR AR6,* +    ;保护AR6
SAR AR7, * +    ;保护AR7
  下面是一些产生局部帧时的注意事项:
        (1)函数进入时,编译器认为ARP为1。
        (2)没有独立的参数表指针。指向参数时,帧指针用负偏移,指向局部变量时,用正偏移;
        (3)帧指针AR0指向一个独立的字,这个字在局部变量前分配,用于存储临存值,通过AR0直接访问。
        (4)编译器用AR2来计算局部变量的地址。一般来说,局部变量的偏移值放在AR2中,然后加上AR0。
        (5)对TMS320C5X来说上面的程序稍有不同,但作用是一样的。
        (三)函数结束
        函数结束返回时,必须完成如下工作以恢复C调用环境:
        (1)处理要传递给调用者的返回值;
        (2)如果使用了AR6和AR7,则必须予以恢复;
        (3)撤销为局部变量和临存值分配的空间;
        (4)恢复原来的帧指针;
        (5)将返回地址压人TMS320C2X/C5X的堆栈并返回调用程序。
        下面是完成上述工作的TMS320C2X汇编程序。
        例2.2  TMS320C2X  C函数的结束处理
LAR AR7,* -     ;恢复AR7
LAR AR6, * -     ;恢复0柏
SBRK SIZE    ;撤销局部帧
LAR AR0,* -     ;恢复FP
PSHD*     ;返回地址压人内部堆栈
RET    ;返回
需要注意的是:
        (1)函数在ACC中返回函数值。整数和指针在ACC的低16位中返回,浮点数和长整型数使用ACC全部32位返回。
        (2)由于用ACC返回函数值,因此,必须保证ACC不被结束程序所修改。
        (3)参数不是由被调用的函数弹出堆栈,而是必须由调用函数弹出。因此,调用者可以传递任意数目的参数至函数,同时,函数也不必知道有多少个参数传递。
        (4)从函数返回时,ARP指向AR1

一周热门 更多>