欢迎来到三一办公! | 帮助中心 三一办公31ppt.com(应用文档模板下载平台)
三一办公
全部分类
  • 办公文档>
  • PPT模板>
  • 建筑/施工/环境>
  • 毕业设计>
  • 工程图纸>
  • 教育教学>
  • 素材源码>
  • 生活休闲>
  • 临时分类>
  • ImageVerifierCode 换一换
    首页 三一办公 > 资源分类 > PPT文档下载  

    汇编及嵌入式C语言-第三章.ppt

    • 资源ID:6388961       资源大小:423.50KB        全文页数:62页
    • 资源格式: PPT        下载积分:15金币
    快捷下载 游客一键下载
    会员登录下载
    三方登录下载: 微信开放平台登录 QQ登录  
    下载资源需要15金币
    邮箱/手机:
    温馨提示:
    用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)
    支付方式: 支付宝    微信支付   
    验证码:   换一换

    加入VIP免费专享
     
    账号:
    密码:
    验证码:   换一换
      忘记密码?
        
    友情提示
    2、PDF文件下载后,可能会被浏览器默认打开,此种情况可以点击浏览器菜单,保存网页到桌面,就可以正常下载了。
    3、本站不支持迅雷下载,请使用电脑自带的IE浏览器,或者360浏览器、谷歌浏览器下载即可。
    4、本站资源下载后的文档和图纸-无水印,预览文档经过压缩,下载后原文更清晰。
    5、试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。

    汇编及嵌入式C语言-第三章.ppt

    汇编及嵌入式C语言,教 师:王茜Email:,第三章 ARM程序设计基础,3.1 ARM编程器所支持的伪操作 伪指令一些特殊的指令助记符,没有相对应的操作码。伪指令在源程序中的作用是为完成汇编程序作各种准备,这些伪指令仅在汇编过程中起作用,一旦汇编结束,伪指令的使命也就完成了。伪指令所完成的操作称为伪操作。,3.1.1 符号定义伪指令 符号定义伪指令用于定义ARM 汇编程序中的变量、对变量赋值以及定义寄存器的别名等操作。常见的符号定义伪指令有如下几种:用于定义全局变量的GBLA、GBLL 和GBLS 用于定义局部变量的LCLA、LCLL 和LCLS 用于对变量赋值的SETA、SETL、SETS 为通用寄存器列表定义名称的RLIST,3.1.1 符号定义伪指令GBLA、GBLL和GBLS-GBLA 伪指令用于定义一个全局的数字变量,并初始化为0;-GBLL 伪指令用于定义一个全局的逻辑变量,并初始化为F(假);-GBLS 伪指令用于定义一个全局的字符串变量,并初始化为空;由于以上三条伪指令用于定义全局变量,因此在整个程序范围内变量名必须唯一。,3.1.1 符号定义伪指令举例如下:GBLA Test1;定义一个全局的数字变量Test1 SETA 0 xaa;将该变量赋值为0 xaa GBLL Test2;定义一个全局的逻辑变量Test2 SETL TRUE;将该变量赋值为真 GBLS Test3;定义一个全局的字符串变量Test3 SETS“Testing”;将该变量赋值为“Testing”,3.1.1 符号定义伪指令LCLA、LCLL和LCLS-LCLA 伪指令用于定义一个局部的数字变量,并初始化为0;-LCLL 伪指令用于定义一个局部的逻辑变量,并初始化为F(假);-LCLS 伪指令用于定义一个局部的字符串变量,并初始化为空;以上三条伪指令用于声明局部变量,在其作用范围内变量名必须唯一。,3.1.1 符号定义伪指令举例如下:LCLA Test4;声明一个局部的数字变量Test4 SETA 0 xaa;将该变量赋值为0 xaa LCLL Test5;声明一个局部的逻辑变量Test5 SETL TRUE;将该变量赋值为真 LCLS Test6;定义一个局部的字符串变量Test6 SETS“Testing”;将该变量赋值为“Testing”,3.1.1 符号定义伪指令SETA、SETL和SETS-SETA 伪指令用于给数字变量赋值;-SETL 给一个逻辑变量赋值;-SETS 伪指令用于给一个字符串变量赋值;,3.1.1 符号定义伪指令RLIST RLIST伪指令可用于对一个通用寄存器列表定义名称,使用该伪指令定义的名称可在ARM 指令LDM/STM 中使用。在LDM/STM 指令中,列表中的寄存器访问次序为根据寄存器的编号由低高,而与列表中的寄存器排列次序无关。举例如下:RegList RLIST R0-R5,R8,R10;将寄存器列表名称定义为RegList,可在ARM指令LDM/STM中通过该名称访问寄存器列表。,3.1.2 数据定义伪指令 数据定义伪指令一般用于为特定的数据分配存储单元,同时可完成已分配存储单元的初始化。常见的数字定义伪指令有如下几种:DCB用于分配一片连续的字节存储单元并用指定的数据初始化。DCW(DCWU)用于分配一片连续的半字存储单元并用指定的数据初始化。DCD(DCDU)用于分配一片连续的字存储单元并用指定的数据初始化。,3.1.2 数字定义伪指令DCB DCB 伪指令用于分配一片连续的字节存储单元并用伪指令中指定的表达式初始化。其中,表达式可以为0255 的数字或字符串。DCB 也可用“=”代替。举例如下:Str DCB“This is a test!”;分配一片连续的字节存储单元并初始化。,3.1.2 数字定义伪指令DCW(或DCWU)DCW(或DCWU)伪指令用于分配一片连续的半字存储单元并用伪指令中指定的表达式初始化。其中,表达式可以为程序标号或数字表达式。用DCW 分配的字存储单元是半字对齐的,而用DCWU 分配的字存储单元并不严格半字对齐。举例如下:DataTest DCW 1,2,3;分配一片连续的半字存储单元并初始化。,3.1.2 数字定义伪指令DCD(或DCDU)DCD(或DCDU)伪指令用于分配一片连续的字存储单元并用伪指令中指定的表达式初始化。其中,表达式可以为程序标号或数字表达式。DCD也可用“分配一片连续的字存储单元并初始化。,3.1.2 数据定义伪指令 LTORG用于指示汇编器立即汇编当前文字池 SPACE用于分配一片连续的存储单元 MAP用于定义一个结构化的内存表首地址 FIELD用于定义一个结构化的内存表的数据域,3.1.2 数字定义伪指令LTORG 使用 LTORG 可确保在指定范围内汇编文字池。,使用示例:AREA Example,CODE,READONLY start BL func1 func1;code LDR r1,=0 x55555555;code MOV pc,lr LTORG data SPACE 4200;从当前位置开始分配连续4200字 节的存储单元并初始化为0END,3.1.2 数字定义伪指令SPACE SPACE 伪指令用于分配一片连续的存储区域并初始化为0。其中,表达式为要分配的字节数。SPACE 也可用“”代替。举例如下:DataSpace SPACE 100;分配连续100字节的存储单元并初始化为0。,3.1.2 数字定义伪指令MAP MAP 表达式,基址寄存器 MAP 伪指令用于定义一个结构化的内存表的首地址。MAP 也可用“”代替。表达式可以为程序中的标号或数学表达式,基址寄存器为可选项,当基址寄存器选项不存在时,表达式的值即为内存表的首地址,当该选项存在时,内存表的首地址为表达式的值与基址寄存器的和。MAP 伪指令通常与FIELD 伪指令配合使用来定义结构化的内存表。,MAP 0 x100,R0;定义结构化内存表首地址的值为0 x100R0。,3.1.2 数字定义伪指令FIELD标号 FIELD 表达式 FIELD 伪指令用于定义一个结构化内存表中的数据域。FILED 也可用“#”代替。表达式的值为当前数据域在内存表中所占的字节数。FIELD 伪指令常与MAP 伪指令配合使用来定义结构化的内存表。MAP 伪指令定义内存表的首地址,FIELD 伪指令定义内存表中的各个数据域,并可以为每个数据域指定一个标号供其他的指令引用。注意MAP 和FIELD 伪指令仅用于定义数据结构,并不实际分配存储单元。,举例如下:MAP 0 x100;定义结构化内存表首地址的值为0 x100。A FIELD 16;定义A的长度为16字节,位置为0 x100 B FIELD 32;定义B的长度为32字节,位置为0 x110 S FIELD 256;定义S的长度为256字节,位置为0 x130,3.1.3 汇编控制伪指令 汇编控制伪指令用于控制汇编程序的执行流程,常用的汇编控制伪指令包括以下几条:IF、ELSE、ENDIF WHILE、WEND MACRO、MEND MEXIT,3.1.3 汇编控制伪指令 IF、ELSE、ENDIF IF、ELSE、ENDIF 伪指令能根据条件的成立与否决定是否执行某个指令序列。当IF 后面的逻辑表达式为真,则执行指令序列1,否则执行指令序列2。其中,ELSE 及指令序列2 可以没有,此时,当IF 后面的逻辑表达式为真,则执行指令序列1,否则继续执行后面的指令。IF、ELSE、ENDIF 伪指令可以嵌套使用。,举例如下:GBLL Test IF Test=TRUE 指令序列1 ELSE 指令序列2 ENDIF,GBLL THUMBCODE CONFIG=16THUMBCODE SETLTRUE|THUMBCODE SETLFALSE,3.1.3 汇编控制伪指令 WHILE、WEND WHILE、WEND 伪指令能根据条件的成立与否决定是否循环执行某个指令序列。当WHILE 后面的逻辑表达式为真,则执行指令序列,该指令序列执行完毕后,再判断逻辑表达式的值,若为真则继续执行,一直到逻辑表达式的值为假。WHILE、WEND 伪指令可以嵌套使用。,GBLA Counter Counter SETA 3 WHILE Counter 10 指令序列 WEND,3.1.3 汇编控制伪指令 MACRO、MEND MACRO、MEND伪指令可以将一段代码定义为一个整体,称为宏指令,然后就可以在程序中通过宏指令多次调用该段代码。宏指令可以使用一个或多个参数,当宏指令被展开时,这些参数被相应的值替换。宏指令的使用方式和功能与子程序有些相似,子程序可以提供模块化的程序设计、节省存储空间并提高运行速度。但在使用子程序结构时需要保护现场,从而增加了系统的开销,因此,在代码较短且需要传递的参数较多时,可以使用宏指令代替子程序。MACRO、MEND 伪指令可以嵌套使用。,;macro definition MACRO;start macro definition$label xmac$p1,$p2;code$label.loop1;code;code BGE$label.loop1$label.loop2;code BL$p1 BGT$label.loop2;code ADR$p2;code MEND;end macro definition,;macro invocation abc xmac subr1,de;invoke macro;codeabcloop1;code;code BGE abcloop1 abcloop2;code BL subr1 BGT abcloop2;code ADR de;code,3.1.3 汇编控制伪指令 MEXIT MEXIT 用于从宏定义中跳转出去。,MACRO$abc example abc$param1,$param2;code WHILE condition1;code IF condition2;code MEXIT ELSE;code ENDIF WEND;code MEND,3.1.4 其他常用的伪指令 AREA AREA 指令指示汇编器汇编新的代码段或数据段。段是不可分的已命名独立代码或数据块,由链接器处理。AREA sectionname,attr,attr.sectionname 段指定的名称,以非字母字符开头的名称必须包含在竖杠内,否则会生成缺失段名的错误。,attr表示该段的相关属性,多个属性用逗号分隔,常用属性如下:,3.1.4 其他常用的伪指令 AREA,attr表示该段的相关属性,多个属性用逗号分隔,常用属性如下:CODE:用于定义代码段,READONLY是缺省值;DATA:用于定义数据段,不包含指令,READWRITE是缺省值;READONLY:指示不应向此段写入,这是代码段的缺省值;READWRITE:指示可以读写此段,这是数据段的缺省值;ALIGN:使用方式为ALIGN表达式,表达式的取值范围为031,相应的对齐方式为2的次方。在默认时,可执行连接文件的代码段和数据段是按字对齐的;,AREA cacheable,CODE,ALIGN=3;表示按8byte对齐Note不要对 ARM 代码节使用 ALIGN=0 或 ALIGN=1。不要对 Thumb 代码节使用 ALIGN=0。,AREA Init,CODE,READONLY;定义Init代码段,属性为只读AREA RamData,DATA,READWRITE;定义RamData数据段,属性为可读写,3.1.4 其他常用的伪指令 ENTRY ENTRY伪指令用于指定汇编程序的入口点。在一个完整的汇编程序中至少要有一个ENTRY(也可以有多个,当有多个ENTRY时,程序的真正入口点由链接器指定),但在一个源文件里最多只能有一个ENTRY(可以没有)。举例如下:AREA Init,CODE,READONLY ENTRY;指定应用程序的入口点,3.1.4 其他常用的伪指令 ALIGN ALIGN 指令通过用零或 NOP 指令进行填充将当前位置对齐到指定边界。若未指定对齐方式,则将当前位置对齐到下一个字的位置。偏移量为一个数字表达式,若使用该字段表示,则当前位置的对齐方式为:2的表达式次幂+偏移量,MOV pc,lr ALIGN 8;当前位置以8字节对齐rout2;code AREA OffsetExample,CODE DCB 1 ALIGN 4,3 DCB 1;当前位置以4+3byte字节对齐,3.1.4 其他常用的伪指令 EQU EQU伪指令用于为程序中的常量、标号等定义一个 等效的字符名称,类似于C语言中的define。其中EQU可用“*”代替。举例如下:Test EQU 50;定义标号Test的值为50,3.1.4 其他常用的伪指令 引用指令 EXPORT 用于在程序中声明一个全局的标号,该标号可在其他的文件中被引用。GLOBAL与EXPORT是同义词。IMPORT 用于通知编译器要使用的标号在其他的源文件中有定义,但在当前的源文件中使用,无论当前源文件是否引用该标号,该标号均会被加入到当前源文件的符号表中。EXTERN 与EMPORT相似,只是如果当前源文件没有引用该标号,该标号就不会被加入到当前源文件的符号表中。,AREA Example,CODE,READONLY EXTERN _CPP_INITIALIZE LDR r0,=_CPP_INITIALIZE CMP r0,#0 BEQ nocplusplus,3.1.4 其他常用的伪指令 包含指令 GET 用于将一个源文件包含到当前的源文件中,并将被包含的源文件在当前位置进行汇编处理。INCLUDE与它用法相同。INCBIN 用于将一个目标文件或数据文件包含到当前的源文件中,被包含的文件不作任何变动的存放在当前的源文件中,编译器从其后开始继续处理。,AREA Example,CODE,READONLY GET file1.s GET c:projectfile2.s GET c:Program filesfile3.s,AREA Example,CODE,READONLY INCBIN file1.dat INCBIN c:projectfile2.txt,3.2 汇编语言程序设计 ARM汇编程序中每一行的通用格式为:标号 指令|伪指令;注解在ARM汇编语言源程序中,除了标号和注释外,指令和伪指令都必须有前导空格,而不能顶格书写。如果每一行的代码太长,可以使用字符“”将其分行书写,并允许有空行。指令助记符和寄存器名既可以全用大写字母,也可以全用小写字母,但不能混用。注释从“;”开始,到该行结束为止。标号代表一个地址,段内标号的地址值在汇编时确定,段外标号的地址值在链接时确定。,3.2.1 汇编语言程序中常用的符号 在汇编语言程序设计中,经常使用各种符号代替地址、变量和常量等,以增加程序的可读性。尽管符号的命名由编程者决定,但并不是任意的,必须遵循以下的约定:符号区分大小写,同名的大、小写符号会被编译器认为是两个不同的符号。符号在其作用范围内必须唯一。自定义的符号名不能与系统的保留字相同。符号名不应与指令或伪指令同名。,3.2.1 汇编语言程序中常用的符号1、程序中的变量 程序中的变量是指其值在程序的运行过程中可以改变的量。ARM(Thumb)汇编程序所支持的变量有数字变量、逻辑变量和字符串变量。在ARM(Thumb)汇编语言程序设计中,可使用GBLA、GBLL、GBLS伪指令声明全局变量,使用LCLA、LCLL、LCLS伪指令声明局部变量,并可使用SETA、SETL和SETS对其进行初始化。见,3.2.1 汇编语言程序中常用的符号2、程序中的常量 程序中的常量是指其值在程序的运行过程中不能被改变的量。ARM(Thumb)汇编程序所支持的常量有数字常量、逻辑常量和字符串常量。数字常量一般为32位的整数,当作为无符号数时,其取值范围为0232-1,当作为有符号数时,其取值范围为-231231-1。逻辑常量只有两种取值情况:真或假。字符串常量为一个固定的字符串,一般用于程序运行时的信息提示。,3.2.1 汇编语言程序中常用的符号3、程序中的变量代换 程序中的变量可通过代换操作取得一个常量。代换操作符为“$”。如果在数字变量前面有一个代换操作符“$”,编译器会将该数字变量的值转换为十六进制的字符串,并将该十六进制的字符串代换“$”后的数字变量。如果在逻辑变量前面有一个代换操作符“$”,编译器会将该逻辑变量代换为它的取值(真或假)。如果在字符串变量前面有一个代换操作符“$”,编译器会将该字符串变量的值代换“$”后的字符串变量。,使用示例:LCLS S1;定义局部字符串变量S1和S2LCLS S2 S1 SETS“Test!”S2 SETS“This is a$S1”;字符串变量S2的值为“This is a Test!”,3.2.2 汇编语言程序中的表达式和运算符 在汇编语言程序设计中,也经常使用各种表达式,表达式一般由变量、常量、运算符和括号构成。常用的表达式有数字表达式、逻辑表达式和字符串表达式,其运算次序遵循如下的优先级:优先级相同的双目运算符的运算顺序为从左到右。相邻的单目运算符的运算顺序为从右到左,且单目运算符的优先级高于其他运算符。括号运算符的优先级最高。,3.2.2 汇编语言程序中的表达式和运算符1、字符串表达式及运算符 字符串表达式一般由字符串常量、字符串变量、运算符和括号构成。编译器所支持的字符串最大长度为512字节。常用的与字符串表达式相关的运算符如下:LEN运算符 LEN运算符返回字符串的长度(字符数),以X表示字符串表达式,其语法格式如下:LEN:X 举例如下:LEN:“string”,3.2.2 汇编语言程序中的表达式和运算符1、字符串表达式及运算符 CHR运算符 CHR运算符将0255之间的整数转换为一个字符,以M表示某一个整数,其语法格式如下:CHR:M 举例如下:CHR:97,3.2.2 汇编语言程序中的表达式和运算符1、字符串表达式及运算符 STR运算符 STR运算符将将一个数字表达式或逻辑表达式转换为一个字符串。对于数字表达式,STR运算符将其转换为一个以十六进制组成的字符串;对于逻辑表达式,STR运算符将其转换为字符串T或F,其语法格式如下:STR:X 其中,X为一个数字表达式或逻辑表达式。,3.2.2 汇编语言程序中的表达式和运算符1、字符串表达式及运算符 LEFT运算符 LEFT运算符返回某个字符串左端的一个子串,其语法格式如下:X:LEFT:Y 其中:X为源字符串,Y为一个整数,表示要返回的字符个数。RIGHT运算符 与LEFT运算符相对应,RIGHT运算符返回某个字符串右端的一个子串,其语法格式如下:X:RIGHT:Y 其中:X为源字符串,Y为一个整数,表示要返回的字符个数。,使用示例:“string”:LEFT:2“string”:RIGHT:2,3.2.2 汇编语言程序中的表达式和运算符1、字符串表达式及运算符 CC运算符 CC运算符用于将两个字符串连接成一个字符串,其语法格式如下:X:CC:Y 其中:X为源字符串1,Y为源字符串2,CC运算符将Y连接到X的后面。举例如下:ha:CC:ppy,3.2.2 汇编语言程序中的表达式和运算符2、数字表达式及运算符 数字表达式一般由数字常量、数字变量、数字运算符和括号构成。与数字表达式相关的运算符如下:“”、“”、“”、“/”及“MOD”算术运算符 以上的算术运算符分别代表加、减、乘、除和取余数运算。例如,以X和Y表示两个数字表达式,则:XY 表示X与Y的和。XY 表示X与Y的差。XY 表示X与Y的乘积。X/Y 表示X除以Y的商。X:MOD:Y 表示X除以Y的余数。,3.2.2 汇编语言程序中的表达式和运算符2、数字表达式及运算符“ROL”、“ROR”、“SHL”及“SHR”移位运算符 以X和Y表示两个数字表达式,以上的移位运算符代表的运算如下:X:ROL:Y 表示将X循环左移Y位。X:ROR:Y 表示将X循环右移Y位。X:SHL:Y 表示将X左移Y位。X:SHR:Y 表示将X右移Y位。,3.2.2 汇编语言程序中的表达式和运算符2、数字表达式及运算符“AND”、“OR”、“NOT”及“EOR”按位逻辑运算符 以X和Y表示两个数字表达式,以上的按位逻辑运算符代表的运算如下:X:AND:Y 表示将X和Y按位作逻辑与的操作。X:OR:Y 表示将X和Y按位作逻辑或的操作。:NOT:Y 表示将Y按位作逻辑非的操作。X:EOR:Y 表示将X和Y按位作逻辑异或的操作。,3.2.2 汇编语言程序中的表达式和运算符3、逻辑表达式及运算符 逻辑表达式一般由逻辑量、逻辑运算符和括号构成,其表达式的运算结果为真或假。与逻辑表达式相关的运算符如下:“=”、“”、“=”、“”运算符 以X和Y表示两个逻辑表达式,以上的运算符代表的运算如下:X=Y 表示X等于Y。X Y 表示X大于Y。X=Y 表示X大于等于Y。X Y 表示X不等于Y。,3.2.2 汇编语言程序中的表达式和运算符3、逻辑表达式及运算符“LAND”、“LOR”、“LNOT”及“LEOR”运算符 以X和Y表示两个逻辑表达式,以上的逻辑运算符代表的运算如下:X:LAND:Y 表示将X和Y 作逻辑与的操作。X:LOR:Y 表示将X和Y作逻辑或的操作。:LNOT:Y 表示将Y作逻辑非的操作。X:LEOR:Y 表示将X和Y作逻辑异或的操作。,3.2.3 汇编语言的程序结构 在ARM(Thumb)汇编语言程序中,以程序段为单位组织代码。段是相对独立的指令或数据序列,具有特定的名称。段可以分为代码段和数据段,代码段的内容为执行代码,数据段存放代码运行时需要用到的数据。一个汇编程序至少应该有一个代码段,当程序较长时,可以分割为多个代码段和数据段,多个段在程序编译链接时最终形成一个可执行的映象文件。,3.2.3 汇编语言的程序结构 可执行映象文件通常由以下几部分构成:一个或多个代码段,代码段的属性为只读。零个或多个包含初始化数据的数据段,数据段的属性为可读写。零个或多个不包含初始化数据的数据段,数据段的属性为可读写。链接器根据系统默认或用户设定的规则,将各个段安排在存储器中的相应位置。因此源程序中段之间的相对位置与可执行的映象文件中段的相对位置一般不会相同。,3.2.3 汇编语言的程序结构 见书P54上的例子。,AREA Word,CODE,READONLYnum EQU 20 ENTRY LDR r0,=src LDR r1,=dst MOV r2,#numwordcopy LDR r3,r0,#4 STR r3,r1,#4 SUBS r2,r2,#1 BNE wordcopystop MOV r0,#0 x18 LDR r1,=0 x20026 AREA BlockData,DATA,READWRITEsrc DCD 1,2,3,4,5,6,7,8,1,2,3,4,5,6,7,8,1,2,3,4dst DCD 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 END,3.3 C语言和汇编语言混合程序设计3.3.1 汇编语言与C/C+的混合编程 在应用系统的程序设计中,若所有的编程任务均用汇编语言来完成,其工作量是可想而知的,同时,不利于系统升级或应用软件移植,事实上,ARM体系结构支持 C/C以及与汇编语言的混合编程,在一个完整的程序设计的中,除了初始化部分用汇编语言完成以外,其主要的编程任务一般都用C/C+完成。汇编语言与C/C+的混合编程通常有以下几种方式:在C/C代码中嵌入汇编指令。在汇编程序和C/C的程序之间进行变量的互访。汇编程序、C/C程序间的相互调用。,3.3.1 汇编语言与C/C+的混合编程 在C语言中内嵌汇编指令 注意:不能直接向PC寄存器赋值,程序跳转要使用B或BL指令,如果跳转位置为C中的标号,只能使用B指令。一般不要直接指定物理寄存器,而让编译器进行分配。内嵌汇编语言使用的标记是_asm或asm关键字。,#include void my_strcpy(const char*src,char*dst)int ch;_asm loop:LDRB ch,src,#1STRB ch,dst,#1CMP ch,#0BNE loop int main(void)const char*a=Hello World!;char b20;my_strcpy(a,b);printf(Original String:%sn,a);printf(Copied String:%sn,b);return 0;,其他嵌入式处理器中使用方法是否相同?,一个MIPS操作的实例:_asm_(li$8,0 xb00e6000);_asm_(li$8,0 xb00e4000);for(i=0;i2880;i=i+16)_asm_(pref 0,16($25);_asm_(lw$9,0($25);_asm_(sw$9,0($8);_asm_(lw$9,4($25);_asm_(sw$9,4($8);_asm_(lw$9,8($25);_asm_(sw$9,8($8);_asm_(lw$9,12($25);_asm_(sw$9,12($8);_asm_(nop);_asm_(nop);_asm_(nop);_asm_(nop);_asm_(addi$25,16);_asm_(“addi$8,16”);,3.3.1 汇编语言与C/C+的混合编程 2.在汇编语句中使用C语言定义的全局变量 内嵌汇编语言由于不用单独编译汇编文件,比较简洁,但是当汇编代码比较多时,一般放在单独的汇编文件中,汇编语言和C语言之间通过使用全局变量进行数据的传递。具体访问方法如下:使用IMPORT伪操作声明该全局变量;使用LDR伪指令读取该全局变量的内存地址;根据该数据的类型,使用相应的LDR指令读取该全局变量的值;使用相应的STR指令修改该全局变量的值。,对于无符号的char类型的变量通过指令 LDRB/STRB来读/写;对于无符号的short类型的变量通过指令 LDRH/STRH来读/写;对于int类型的变量通过指令LDR/STR来读/写;对于有符号的char类型的变量通过指令LDRSB来读取;对于有符号的char类型的变量通过指令STRB来写入;对于有符号的short类型的变量通过指令LDRSH来读取;对于有符号的short类型的变量通过指令STRH来写入。,AREA globals,CODE,READONLYEXPORT asmsubIMPORT globvlasmsubLDR r1,=globvlLDR r0,r1ADD r0,r0,#2STR r0,r1MOV pc,lrEND,3.3.1 汇编语言与C/C+的混合编程 3.在C语言中调用汇编程序 为保证程序调用时参数的正确传递,汇编程序设计要遵循ATPCS准则。具体方法如下:在汇编程序中使用EXPORT伪指令声明子程序,使其他程序可以调用此子程序;在C语言程序中使用extern关键字声明外部函数(声明要调用的汇编子程序),即可调用此汇编子程序。,#include extern void strcopy(char*d,const char*s);int main()const char*srcstr=First string-source;char dststr=Second string-destination;printf(Before copying:n);printf(%sn%sn,srcstr,dststr);strcopy(dststr,srcstr);printf(After copying:n);printf(%sn%sn,srcstr,dststr);return 0;,AREA SCopy,CODE,READONLY EXPORT strcopystrcopy;r0 points to destination string;r1 points to source string LDRB r2,r1,#1 STRB r2,r0,#1 CMP r2,#0 BNE strcopy MOV pc,lr END,根据ATPCS标准,函数前4个参数通过R0R3来传递,其它参数通过堆栈(FD)传递,3.3.1 汇编语言与C/C+的混合编程 4.在汇编语言中调用C语言的函数 在汇编语言程序中调用C语言函数,需要在汇编程序中利用IMPORT伪指令声明对应的C函数名,汇编程序的设计要遵循ATPCS规则,保证程序调用时参数的正确传递。,int g(int a,int b,int c,int d,int e)return a+b+c+d+e;,;汇编程序调用c程序g()计算5个整数i,2*i,3*i,4*i,5*i的和 EXPORT fAREA f,CODE,READONLY IMPORT g STR lr,sp,#-4!ADD r1,r0,r0 ADD r2,r1,r0 ADD r3,r1,r2 STR r3,sp,#-4!ADD r3,r1,r1 BL g ADD sp,sp,#4 LDR pc,sp,#4 END,3.3.2 ATPCS规则 ATPCSARM/Thumb Procedure Call Standard 在C程序和ARM汇编程序之间相互调用必须遵守ATPCS,使用ADS的C语言编译器编译的C语言子程序满足用户指定的ATPCS类型,而对于汇编语言来说,完全要依赖用户来保证各个子程序满足选定的ATPCS类型,汇编语言子程序必须满足以下三个条件:1.在子程序编写时必须遵守相应的ATPCS规则;2.堆栈的使用要遵守相应的ATPCS规则;3.在汇编编译器中使用-apcs选项。,3.3.2 ATPCS规则 寄存器的使用规则:1.子程序通过寄存器R0R3来传递参数.这时寄存器可以记作:A0A3,被调用的子程序在返回前无需恢复寄存器R0R3的内容.2.在子程序中,使用R4R11来保存局部变量,这时寄存器可以记作:V1V8.如果在子程序中使用到V1V8的某些寄存器,子程序进入时必须保存这些寄存器的值,在返回前必须恢复这些寄存器的值。3.寄存器R12用作子程序间的临时寄存器,记作IP,在子程序的连接代码段中常使用临时寄存器。,3.3.2 ATPCS规则 寄存器的使用规则:4.寄存器R13用作数据栈指针,记做SP,在子程序中寄存器R13不能用做其他用途.寄存器SP在进入子程序时的值和退出子程序时的值必须相等.5.寄存器R14用作连接寄存器,记作LR;它用于保存子程序的返回地址,如果在子程序中保存了返回地址,则R14可用作其它的用途.6.寄存器R15是程序计数器,记作PC;它不能用作其他用途.,3.3.2 ATPCS规则 参数传递规则:1.参数个数可变的子程序参数传递规则 对于参数个数可变的子程序,当参数不超过4个时,使用寄存器R0R3来进行参数传递,当参数超过4个时,还可以使用数据栈来传递参数.在参数传递时,将所有参数看做是存放在连续的内存单元中的字数据。然后,依次将各数据传送到寄存器R0,R1,R2,R3;如果参数多于4个,将剩余的字数据传送到数据栈中,入栈的顺序与参数顺序相反,即最后一个字数据先入栈。2.参数个数固定的子程序参数传递规则 对于参数个数固定的子程序,参数传递与参数个数可变的子程序参数传递规则不同,第一个整数参数通过寄存器R0R3来传递,其他参数通过数据栈传递。,3.3.2 ATPCS规则 子程序结果返回规则:1.结果为一个32位的整数时,可以通过寄存器R0返回.2.结果为一个64位整数时,可以通过R0和R1返回,依此类推.3.对于位数更多的结果,需要通过调用内存来传递.,3.4 ARM编译器关键词_irq:声明的函数可用作对IRQ或者FIQ异常中断的中断处理函数,可以保存被该函数破坏的寄存器,包括ATPCS标准要求的寄存器。static void _irq Eint1Int(void)ClearPending(BIT_EINT1);Uart_Printf(“EINT1 interrut is occurred.n”);,3.4 ARM编译器关键词_swi:声明的函数最多可接收4个整型变量,并最多可利用value_in_regs返回4个结果。不返回参数:void _swi(swi_num)swi_name(int arg1,int argn);返回一个参数:int _swi(swi_num)swi_name(int arg1,int argn);返回参数多于1个时:typedef struct int res1,resn;res_type;res_type _value_in_regs _swi(swi_num)swi_name(int arg1,int argn);,3.4 ARM编译器关键词register:所声明的变量在编译器处理时要尽量保存到寄存器中。volatile:所声明的变量用于告知编译器该变量可能在程序之外修改,编译器在编译时不优化对volatile变量的操作。3.5 程序设计示例 见书P64,

    注意事项

    本文(汇编及嵌入式C语言-第三章.ppt)为本站会员(小飞机)主动上传,三一办公仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知三一办公(点击联系客服),我们立即给予删除!

    温馨提示:如果因为网速或其他原因下载失败请重新下载,重复下载不扣分。




    备案号:宁ICP备20000045号-2

    经营许可证:宁B2-20210002

    宁公网安备 64010402000987号

    三一办公
    收起
    展开