arm export 汇编_C/C++与汇编混合编程有什么好处?

1.导语

当需要C/C++与汇编混合编程时,可以有以下两种处理策略:

若汇编代码较短,则可在C/C++源文件中直接内嵌汇编语言实现混合编程。

若汇编代码较长,可以单独写成汇编文件,最后以汇编文件的形式加入项目中,通过ATPCS规定与C程序相互调用及访问。

2. 内嵌汇编语言指令

用C/C++程序嵌入汇编程序中可以实现一些高级语言没有的功能,提高程序执行效率。armcc编译器的内嵌汇编器支持ARM指令集,tcc编译器的内嵌汇编器支持Thumb指令集。

2.1 内嵌汇编指令的语法格式

在ARM的C语言程序中可以使用关键字__asm来加入一段汇编语言的程序,格式如下:

[cpp] view plain copy

1. __asm  

2. {  

3.     指令 [;指令]      /* comments */  

4.     ...  

5.   

6.     指令  

7. }  

其中,{ }中的指令都为汇编指令,一行允许写多条汇编指令语句,指令语句之间要用分号隔开。在汇编指令段中,注释语句采用C语言的注释格式。ARM C++程序中除了可以使用关键字__asm来标识一段内嵌汇编指令程序外,还可以使用关键词asm来表示一段内嵌汇编指令。

格式如下:asm ("指令");

其中,asm后面的括号中必须是一条汇编指令语句,并且不能包含注释语句。

2.2 使能/禁止IRQ中断实例

[cpp] view plain copy

1. void enable_IRQ(void) //使能中断程序  

2. {  

3.     int tmp;              //定义临时变量,后面使用  

4.     __asm                 //内嵌汇编程序的关键词  

5.     {  

6.         MRS tmp, CPSR     //把状态寄存器加载给tmp  

7.         BIC tmp, tmp, #80 //将IRQ控制位清0  

8.         MSR CPSR_c, tmp   //加载程序状态寄存器  

9.     }  

10. }  

11.   

12. void disable_IRQ(void) //禁止中断程序  

13. {  

14.     int tmp;              //定义临时变量,后面使用  

15.     __asm                 //内嵌汇编程序的关键词  

16.     {  

17.         MRS tmp, CPSR     //把状态寄存器加载给tmp  

18.         ORR tmp, tmp, #80 //将IRQ控制位置1  

19.         MSR CPSR_c, tmp   //加载程序状态寄存器  

20.     }  

21. }  

2.3 内嵌汇编注意事项      

后缀.S文件中的汇编指令是用armasm汇编器进行汇编的,而C语言程序中的内嵌汇编指令则是用内嵌汇编器进行汇编的。这两种汇编器存在一定的差异,所以在内嵌汇编时要注意以下几点。

2.3.1 小心使用物理寄存器

必须小心使用物理寄存器,如R0~R3、IP(R12)、LR(R14)和CPSR中的N、Z、C、V标志位。因为计算汇编代码中的C表达式时,可能使用这些物理寄存器,并会修改N、Z、C、V标志位。

如计算:

y=x+x/y;

[cpp] view plain copy

1. __asm  

2. {  

3.     MOV R0, x         //把x的值给R0  

4.     ADD y, R0, x/y    //计算x/y时R0的值会被修改  

5. }  

2.3.2 内嵌汇编程序中允许使用C变量        

在计算x/y时R0会被修改,从而影响R0+x/y的结果。内嵌汇编程序中允许使用C变量,用C变量来代替寄存器R0可以解决上述问题。这时内嵌汇编器将会为变量var分配合适的存储单元,从而避免冲突的发生。如果内嵌汇编器不能分配合适的存储单元,它将会报告错误。

[cpp] view plain copy

1. int var;  

2. __asm  

3. {  

4.     MOV var, x      //把x的值给R0  

5.     ADD y, var, x/y //计算x/y时R0的值会被修改  

6. }  

2.3.3 不需要保存和恢复用到的寄存器

对于在内嵌汇编语言程序中用到的寄存器,编译器在编译时会自动保存和恢复这些寄存器,用户不用保存和恢复这些寄存器。除了CPSR和SPSR寄存器外,其他物理寄存器在读之前必须先赋值,否则编译器会报错。

[cpp] view plain copy

1. int fun (int x)  

2. {  

3.     __asm  

4.     {  

5.         STMFD SP!, {R0}   //保存R0,先读后写,汇编出错  

6.         ADD R0, x, #1  

7.         EOR x, R0, x  

8.         LDMFD SP!, {R0}   //多余的  

9.     }  

10.     return x;  

11. }  

3. 汇编与C/C++程序的变量相互访问

3.1 汇编程序访问C/C++程序变量

在C/C++程序中声明的全局变量可以被汇编程序通过地址间接访问。具体访问方法/步骤如下:

     1) 在C/C++程序中声明全局变量。

     2) 在汇编程序中使用IMPORT/EXTERN伪指令声明引用该全局变量。

     3) 使用LDR伪指令读取该全局变量的内存地址。

     4) 根据该数据的类型,使用相应的LDR指令读取该全局变量;使用相应的STR指令存储该全局变量的值。对于不同类型的变量,需要采用不同选项的LDR和STR指令,如下表所示。

对于结构,如果知道各个数据项的偏移量,可以通过存储/加载指令访问。如果结构所占空间小于8个字,可以使用LDM和STM一次性读写。

读取C的一个全局变量,并进行修改,然后保存新的值到全局变量中:

[cpp] view plain copy

1. AREA Example4, CODE, READONLY  

2.      EXPORT AsmAdd  

3.      IMPORT g_cVal      @声明外部变量g_cVal,在C中定义的全局变量  

4. Add  

5.      LDR R1, =g_cVal    @装载变量地址  

6.      LDR R0, [R1]       @从地址中读取数据到R0  

7.      ADD R0, R0, #1     @加1操作  

8.      STR R0, [R1]       @保存变量值  

9.      MOV PC, LR         @程序返回  

10. END  

3.2 C/C++程序访问汇编程序数据

在汇编程序中声明的数据可以被C/C++程序所访问。具体访问方法/步骤如下:

     1) 在汇编程序中用EXPORT/GLOBAL伪指令声明该符号为全局标号,可以被其他文件应用。

     2) C/C++程序中定义相应数据类型的指针变量。

     3) 对该指针变量赋值为汇编程序中的全局标号,利用该指针访问汇编程序中的数据。

假设在汇编程序中定义了一块内存区域,并保存一串字符,汇编代码如下: 

[cpp] view plain copy

1. EXPORT Message        @声明全局标号  

2. Message DCB "HELLO$"  @定义了5个有效字符,$为结束符

[cpp] view plain copy

1. extern char* Message;  

2. int MessageLength()  

3. {  

4.     int Length = 0;  

5.     char *pMessage;         //定义字符指针变量  

6.     pMessage = Message;     //指针指向Message 内存块的首地址  

7.       

8.     /*while循环,统计字符串的长度*/  

9.     while(*pMessage != '$') //$为字符串的结束符  

10.     {  

11.         Length++;  

12.         pMessage++;  

13.     }  

14.     return(Length); //返回字符串的长度  

15. }  

4. 汇编与C/C++程序的函数相互调用

C/C++程序和ARM汇编程序之间相互调用必须遵守ATPCS(ARM/Thumb Procedure Call Standard)规则。使用ADS的C语言编译器编译的C语言子程序会自动满足用户指定的ATPCS类型。而对于汇编语言来说,完全要依赖用户来保证各个子程序满足选定的ATPCS类型。具体来说,汇编程序必须满足以下3个条件才能实现与C语言的相互调用:

1) 在子程序编写时必须遵守相应的ATPCS规则。

2) 堆栈的使用要遵守相应的ATPCS规则。

3) 在汇编编译器中使用-atpcs选项。

4.1 ATPCS基本规则

ATPCS基本规则见ATPCS。

4.2 C程序调用汇编程序

汇编程序的设置要遵循ATPCS规则,保证程序调用时参数的正确传递,在这种情况下,C程序可以调用汇编子函数。C程序调用汇编程序的方法如下:

1) 汇编程序中使用EXPORT伪指令声明本子程序可外部使用,使其他程序可调用该子程序。

2) 在C语言程序中使用extern关键字声明外部函数(声明要调用的汇编子程序),才可调用此汇编的子程序。

[cpp] view plain copy

1. #include  

2. extern void strcopy(char *d, const char *s); 

//声明外部函数,即要调用的汇编子程序  

3. int main(void)  

4. {  

5.     const char *srcstr = "First ource";          //定义字符串常量  

6.     char dststr[] = "Second string-destination"; //定义字符串变量  

7.     printf("Before copying:\n");  

8.     printf("src=%s, dst=%s\n", srcstr, dststr);  

//显示源字符串和目标字符串的内容  

9.     strcopy(dststr, srcstr);  //调用汇编子程序R0=dststr, R1=srcstr  

10.     printf("After copying:\n");  

11.     printf("src=%s, dst=%s\n", srcstr, dststr);  //显示复制后的结果  

12.     return(0);  

13. }  

strcopy实现代码如下:

[cpp] view plain copy

1.       AREA Example, CODE, READONLY @声明代码段Example  

2.       EXPORT strcopy      @声明strcopy,以便外部函数调用  

3.   

4. strcopy     @ R0为目标字符串的地址, R1为源字符串的地址  

5.   

6.       LDRB R2, [R1], #1    @读取字节数据,源地址加1  

7.       STRB R2, [R0], #1    @保存读取的1字节数据,目标地址加1  

8.       CMP R2, #0           @判断字符是否复制完毕  

9.       BNE strcopy          @没有复制完,继续循环复制  

10.     MOV PC, LR   

4.3 汇编程序调用C程序

汇编程序设置要遵循APTCS规则,保证程序调用时参数的正确传递。汇编程序调用C程序的方法如下:

  • 在汇编程序中使用IMPORT伪指令声明将要调用的C程序函数。

  • 在调用C程序时,要正确设置入口参数,然后使用BL指令调用。

[cpp] view plain copy

1. int sum(int a, int b, int c, int d, int e)  

2. {  

3.     return(a+b+c+d+e); //返回5个变量的和  

4. }  

[cpp] view plain copy

1.  AREA Example, CODE, READONLY  

2.   IMPORT sum      @ 声明外部标号sum,即C函数sum()  

3.   EXPORT CALLSUM  

4.   UM  

5.      STMFD SP!, {LR} @LR寄存器入栈  

6.      MOV R0, #1         @设置sum函数入口参数,R0为参数a  

7.      MOV R1, #2         @R1为参数b  

8.      MOV R2, #3         @R2为参数c  

9.      MOV R3, #5         @参数 e=5,保存到堆栈中  

10.      STR R3, {SP, #-4}!  

11.      MOV R3, #4         @R3为参数d, d=4  

12.      BL sum             @调用C程序中的sum函数,结果放在R0中  

13.      ADD SP, SP, #4     @调整堆栈指针  

14.      LDMFD SP, {PC}     @程序返回  

15. END  

以上程序使用了5个参数,分别使用寄存器R0存储第1个参数,R1存储第2个参数,R2存储第3个参数,R3存储第4个参数,第5个参数利用堆栈传送。由于利用了堆栈传递参数,在程序调用结束后要调整堆栈指针。汇编程序中调用了C程序的sum子函数,实现了1+2+3+4+5,最后相加结果保存在R0寄存器中。

免责声明:本文转自网络,版权归原作者,如果您觉得不好,请联系我们删除!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/506265.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

html调整文字位在基线显示,html – 将标题对齐到相同的基线,无论后续文字是什么?...

有没有办法将不同大小的多个标题的第一行与同一基线对齐?也不管后面的文本,也应该对齐.编辑:重新上传:在我看来,唯一的解决方案是将每个标题和每个正文文本放入单独的DIV,然后使用标题来填充顶部或边缘顶部以对齐它们(例如H1将是36px,0px边距…

论文速递|Management Science 11月文章合集(上)

编者按 在本系列文章中,我们梳理了运筹学顶刊Management Science11月份发布的47篇文章的基本信息,旨在帮助读者快速洞察行业最新动态。本文为第一部分。 文章1 ● 题目:Discrimination and Economic Expectations 歧视与经济期望 ● 原文链…

不属于微型计算机特点的是什么,2017计算机应用基础模拟题及答案

2017计算机应用基础模拟题及答案一、选择题:1.过程控制的特点是( D )A.计算量大,数值范围广 B.数据输入输出量大,计算相对简单C.进行大量的图形交互操作 D.具有良好的实时性和高可靠性2.世界第一台电子计算机ENIAC诞生于(B )A.1945 B.1946 C.…

怎么修剪_幸福树怎么修剪——武汉花卉租摆

幸福树,一种寓意美好的观赏型植物,它生长非常迅速,稍不注意就长的非常茂盛。而要想保证幸福树的美貌,跟人的头发一样,我们要给它适当的修剪,那幸福树怎么修剪呢?为了大家能养出美丽的幸福树来&a…

mybatis传递多个参数_MyBatis 映射器

ps 一个用于生成MyBatis配置文件的插件 mybatis-generator使用方法呢, 是加入maven插件中 然后执行相关命令可以实现自动生成MyBatis配置文件自动映射首先编写无参的javabeanpackage com.ming.MyBatis.POJO;/** * author ming */public class Role { private int id; private S…

redis将散裂中某个值自增_Redis总结

一 初识Redis1,Redis 使用内存存储的非关系型数据库,除了各种数据类型的专有命令,还支持批量操作(bulk operation)和不完全(partial)的事务。Memcached 使用内存存储的键值对缓存。MongoDB使用硬盘存储的非关系性数据库.2,Redis可以存储键与5种不同数据结…

计算机软件水平考试什么题型,计算机软考考什么内容

原标题:计算机软考考什么内容计算机软考考试内容有哪些?软考包含三个级别,各级别有多个考试项目,不同的考试项目考试内容也是不同的。软考考试内容大家可以参考各考试项目的考试大纲,包括新版的系统分析师考试大纲、系…

python rest api_Python调用REST API接口的几种方式汇总

相信做过自动化运维的同学都用过REST API接口来完成某些动作。API是一套成熟系统所必需的接口,可以被其他系统或脚本来调用,这也是自动化运维的必修课。 本文主要介绍python中调用REST API的几种方式,下面是python中会用到的库。 - urllib2 -…

mac os 开启redis_关于Redis,学会这8点就够了

一、redis是什么redis是一种支持Key-Value等多种数据结构的存储系统。可用于缓存、事件发布或订阅、高速队列等场景。该数据库使用ANSI C语言编写,支持网络,提供字符串、哈希、列表、队列、集合结构直接存取,基于内存,可持久化。二…

python第八周小测验_Python语言程序设计第2周测验+练习题复盘

嘿,这里是目录! 练习网址 https://python123.io/index 测验2 单项选择题 1. 哪个选项不能正确引用turtle库进而使用setup()函数?‪‬‪‬‪‬‪‬‪‬‮‬‫‬‪‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‭‬‪‬‪‬…

计算机设备管理器驱动,驱动技巧:解决设备管理器中声卡驱动安装不正确的问题...

分类:声卡驱动问题:设备管理器中声卡驱动安装不正确描述:电脑没有声音,有部分朋友是因为声卡驱动没有正确安装,除了我们常见到的设备管理器出现黄色感叹号之外,另一种情况就是让一般人很难发现的问题&#…

lisp对excel其他行列写入_【极简Python 自动化办公】Python写入Excel表格

【极简Python 自动化办公】Python写入Excel表格【极简Python 自动化办公】专栏是介绍如何利用python办公,减少工作负荷。篇幅精炼,内容易懂,无论是否有编程基础,都非常适合。在上次文章中,我们学习了【用python读取exc…

全国大学生计算机应用大赛有什么好处,我校学生喜获第十一届全国大学生计算机应用能力与信息素养大赛冠军...

2021年6月,第十一届全国大学生计算机应用能力与信息素养大赛通过腾讯会议视频全程监控在线作答的形式圆满结束。本届大赛共有来自117所院校932名选手参加全国总决赛。经过校园赛选拔,考前集训,我校共选出3名40本科班同学参加了该赛项&#xf…

java8 lambda map排序_Android兼容Java 8语法特性的原理分析

本文主要阐述了Lambda表达式及其底层实现(invokedynamic指令)的原理、Android第三方插件RetroLambda对其的支持过程、Android官方最新的dex编译器D8对其的编译支持。通过对这三个方面的跟踪分析,以Java 8的代表性特性——Lambda表达式为着眼点,将Android…

lrange是取出所有值并移除么_部落冲突:兵营容量提升,移除超级部队、英雄防御状态冷却时间...

爱生活,爱游戏,大家好,我是阿呆!部落冲突已经好长时间没更新了,这次秋天不再等待,部落冲突第二弹更新来袭,被称之为今年最赞的一次更新,感觉和我一起来看看吧!防御建筑升…

python爬取百度迁徙数据_python爬虫-动态爬取百度迁徙

#1.模拟浏览器发送请求 importrequestsimportjsonimportpandas as pd city_name[] province_name[] value[] url https://huiyan.baidu.com/migration/cityrank.jsonp?dtcountry&id0&typemove_in&date20200315&callbackjsonp_1590404076900_7186798rqrequests…

计算机系统硬盘内存主频,内存时序和频率_内存硬盘-中关村在线

内存时序和频率内存时序是描述内存条性能的一种参数,一般存储在内存条的SPD中。一般数字“A-B-C-D”分别对应的参数是“CL-tRCD-tRP-tRAS”,它们的含义依次为:CAS Latency(简称CL值)内存CAS延迟时间,它是内存的重要参数之一&#…

ubuntun系统mysql数据库同步_Canal 实现 Mysql数据库实时数据同步

简介1.1 canal介绍​ Canal是一个基于MySQL二进制日志的高性能数据同步系统。Canal广泛用于阿里巴巴集团(包括https://www.taobao.com),以提供可靠的低延迟增量数据管道,github地址:https://github.com/alibaba/canalC…

int数组初始化_Java数组

###理解数组Java的数组要求所有的元素为相同数据类型。既可以存储基本类型的数据,也可以存储引用类型的数据,只要所有元素的类型相同就可以。一旦数组的初始化完成,数组在内存中占用的空间就被固定下来,因此数组长度不可变&#x…

pdf无法启动因计算机丢失,解决打开 PDF 文件时出现的文档损坏错误

如果您是客户或企业 IT 专业人员,可以通过设置相应的首选项在计算机上禁用标头验证。您可以在 HKCU 中为单个用户设置此首选项,或者在 HKLM 中在计算机级别为所有用户设置。您可以通过修改相同的 plist 首选项在 Mac OS 上使用相似的方法。如果 AVGenera…