使用arm混合汇编计算两个64位的和_混合使用C、C++和汇编语之: C、C++ 和 ARM 汇编语言之间的调用...
12.4C' target='_blank' style='cursor:pointer;color:#D05C38;text-decoration:underline;'>C、C++和ARM汇编语言之间的调用
本节提供一些示例,显示如何从C++调用C和汇编语言代码,以及从C和汇编语言调用C++代码。其中包括调用约定和数据类型。主要包括下面内容:
·相互调用的一般规则;
·C++语言的特定信息;
·调用示例。
只要遵循正确的过程调用标准AAPCS,就可以混合调用C、C++和汇编语言例程。有关AAPCS的更多信息,请参阅ARM相关文档。
12.4.1相互调用的一般规则
以下一般规则适用于C、C++和汇编语言之间的调用。有关的详细信息,请参阅ARM开发相关文档。
嵌入式汇编程序以及其与ARM嵌入式应用程序二进制接口(BSABI,ApplicationBinaryInterfacefortheARMArchitecture)的兼容使得混合语言编程更易于实现。它们可提供以下功能:
·使用__cpp关键字进行名称延伸;
·传递隐含this参数的方式;
·调用虚函数的方式;
·引用的表示;
·具有基类或虚成员函数的C++类的类型布局;
·非POD(PlainOldData)结构的类对象传递。
以下一般规则适用于混合语言编程:
·使用C调用约定。
·在C++中,非成员函数可以声明为extern"C",以指定它们有C链接。带有C链接意味着定义函数的符号未延伸。C链接可以用于以一种语言实现函数,然后用另一种语言调用它。
·汇编语言模块所必须符合的AAPCS调用标准,应当适合于应用程序所使用的存储器模型。
以下规则适用于从C和汇编语言调用C++函数:
·要调用全局(非成员)C++函数,应将它声明为extern"C",以提供C链接。
·成员函数(静态和非静态)总是有已延伸的名称。使用嵌入式汇编程序的__cpp关键字,可以不必手工寻找已延伸的名称。
·不能从C调用C++内联函数,除非确保C++编译器生成了函数的外联副本。例如,取得函数地址将导致生成外联副本。
·非静态成员函数接受隐含this参数作为r0中的第一个自变量,或作为r1中第二个自变量(如果函数返回非int类结构)。静态成员函数不接受隐含this参数。
12.4.2C++的特定信息
本节主要介绍一些专门适用于C++的内容。
(1)C++调用约定
ARMC++使用与ARMC相同的调用约定,但在下面的情况下,调用规则有所不同:
·调用非静态成员函数时,隐含的this参数是第一个自变量,或者是第二个自变量(如果被调用函数返回非int类的struct)。这可能在将来的版本中有所变化。
(2)C++数据类型
ARMC++使用与ARMC相同的数据类型,但在以下几种情况下,情况有所不同:
·如果struct或class类型的C++对象没有基类或虚函数,则它们的布局与ARMC相同。如果这样的struct没有用户定义的复制赋值运算符或用户定义的析构函数,则它是POD结构。
·引用表示为指针。
·C函数指针和C++(非成员)函数指针没有区别。
(3)符号名称延伸
链接程序将取消信息中符号名称的延伸。
在C++程序中,C名称必须声明为extern"C"。ARMISOC头文件已经完成此操作。详细信息请参阅ARM相关文档。
12.4.3混合编程调用举例
汇编程序、C程序以及C++程序相互调用时,要特别注意遵守相应的AAPCS。下面一些例子具体说明了在这些混合调用中应注意遵守的AAPCS规则。这些示例程序默认为使用非软件栈检查的ATPCS规则,因为它们执行栈操作时不检查栈溢出。
(1)从C调用汇编语言
下面的程序显示如何在C程序中调用汇编语言子程序,该段代码实现了将一个字符串复制到另一个字符串。
#include
externvoidstrcopy(char*d,constchar*s);
intmain()
{constchar*srcstr="Firststring-source";
chardststr[]="Secondstring-destination";
/*下面将dststr作为数组进行操作*/
printf("Beforecopying:\n");
printf("%s\n%s\n",srcstr,dststr);
strcopy(dststr,srcstr);
printf("Aftercopying:\n");
printf("%s\n%s\n",srcstr,dststr);
return(0);
}
下面为调用的汇编程序。
PRESERVE8
AREASCopy,CODE,READONLY
EXPORTstrcopy
Strcopy ;r0指向目的字符串
;r1指向源字符串
LDRBr2,[r1],#1 ;加载字节并更新源字符串指针地址
STRBr2,[r0],#1 ;存储字节并更新目的字符串指针地址
CMPr2,#0 ;判断是否为字符串结尾
BNEstrcopy ;如果不是,程序跳转到strcopy继续拷贝
MOVpc,lr ;程序返回
END
按以下步骤从命令行编译该示例:
①输入armasm-gscopy.s编译汇编语言源代码。
②输入armcc-c-gstrtest.c编译C源代码。
③输入armlinkstrtest.oscopy.o-ostrtest链接目标文件。
④将ELF/DWARF2兼容调试器与相应调试目标配合使用,运行映像。
(2)汇编语言调用C程序
下面的例子显示了如何从汇编语言调用C程序。
下面的子程序段定义了C语言函数。
intg(inta,intb,intc,intd,inte)
{
returna+b+c+d+e;
}
下面的程序段显示了汇编语言调用。假设程序进入f时,r0中的值为i。
;intf(inti){returng(i,2*i,3*i,4*i,5*i);}
PRESERVE8
EXPORTf
AREAf,CODE,READONLY
IMPORTg //声明C程序g()
STRlr,[sp,#-4]! //保存返回地址lr
ADDr1,r0,r0 //计算2*i(第2个参数)
ADDr2,r1,r0 //计算3*i(第3个参数)
ADDr3,r1,r2 //计算5*i
STRr3,[sp,#-4]! //第五个参数通过堆栈传递
ADDr3,r1,r1 //计算4*i(第4个参数)
BLg //调用C程序
ADDsp,sp,#4 //从堆栈中删除第5个参数
LDRpc,[sp],#4 //返回
END
(3)从C++调用C
下面的例子显示了如何从C++程序中调用C函数。
下面的C++程序调用了C程序。
structS{ //本结构没有基类和虚函数
S(ints):i(s){}
inti;
};
extern"C"voidcfunc(S*);
//被调用的C函数使用extern“C”声明
intf(){
Ss(2); //初始化's'
cfunc(&s); //调用C函数'cfunc'将改变's'
returnsi*3;
}
下面显示了被调用的C程序代码。
structS{
inti;
};
voidcfunc(structS*p){
/*定义被调用的C功能*/
p->i+=5;
}
(4)从C++中调用汇编
下面的例子显示了如何从C++中调用汇编程序。
下面的例子为调用汇编程序的C++代码。
structS{ //本结果没有基类和虚拟函数
//
S(ints):i(s){}
inti;
};
extern"C"voidasmfunc(S*); //声明被调用的汇编函数
intf(){
Ss(2); //初始化结构体's'
asmfunc(&s); //调用汇编子程序'asmfunc'
returns.i*3;
}
下面是被调用的汇编程序。
PRESERVE8
AREAAsm,CODE
EXPORTasmfunc
asmfunc//被调用的汇编程序定义
LDRr1,[r0]
ADDr1,r1,#5
STRr1,[r0]
MOVpc,lr
END
(5)从C中调用C++
下面的例子显示了如何从C++代码中调用C程序。
下面的代码显示了被调用C++代码。
structS{//本结构没有基类和虚拟函数
S(ints):i(s){}
inti;
};
extern"C"voidcppfunc(S*p){
//定义被调用的C++代码
//连接了C功能
p->i+=5;//
}
调用了C++代码的C函数。
structS{
inti;
};
externvoidcppfunc(structS*p);
/*声明将会被调用的C++功能*/
intf(void){
structSs;
s.i=2;/*初始化S*/
cppfunc(&s);/*调用cppfunc函数,该函数可能改变S的值*/
returns.i*3;
}
(6)从汇编中调用C++程序
下面的代码显示了如何从汇编中调用C++程序。
下面是被调用的C++程序。
structS{//本结构没有基类和虚拟函数
S(ints):i(s){}
inti;
};
extern"C"voidcppfunc(S*p){
//定义被调用的C++功能
//功能函数体
p->i+=5;
}
在汇编语言中,声明要调用的C++功能,使用带连接的跳转指令调用C++功能。
AREAAsm,CODE
IMPORTcppfunc ;声明被调用的C++函数名
EXPORTf
f
STMFDsp!,{lr}
MOVr0,#2
STRr0,[sp,#-4]! ;初始化结构体
MOVr0,sp ;调用参数为指向结构体的指针
BLcppfunc ;调用C++功能'cppfunc'
LDRr0,[sp],#4
ADDr0,r0,r0,LSL#1
LDMFDsp!,{pc}
END
(7)在C和C++函数间传递参数
下面的例子显示了如何在C和C++函数间传递参数。
下面的代码为C++函数。
extern"C"intcfunc(constint&);
//声明被调用的C函数
extern"C"intcppfunc(constint&r){
//定义将被C调用的C++函数
return7*r;
}
intf(){
inti=3;
returncfunc(i); //相C函数传参
}
下面为C函数。
externintcppfunc(constint*);
/*声明将被调用的C++函数*/
intcfunc(constint*p){
/*定义被C++调用的C函数*/
intk=*p+4;
returncppfunc(&k);
}
(8)从C或汇编语言调用C++
下面的例子综合显示了如何从C或汇编语言中调用非静态、非虚的C++成员函数。可以使用编译器编译出的汇编程序查找已延伸的函数名。
下面是被调用的C++成员函数。
structT{
T(inti):t(i){}
intt;
intf(inti);
};
intT::f(inti){returni+t;}
//定义将被C调用的C++功能函数
extern"C"intcfunc(T*);
//声明将被C++调用的C函数
intf(){
Tt(5);//createanobjectoftypeT
returncfunc(&t);
}
下面为调用C++的C语言函数。
structT;
externint_ZN1T1fEi(structT*,int);
/*被调用的C++函数名*/
intcfunc(structT*t){
/*定义被C++调用的C函数*/
return3*_ZN1T1fEi(t,2);/*实现3乘以t->f(2)功能*/
}
下面为调用C++的汇编函数。
EXPORTcfunc
AREAfoo,CODE
IMPORT_ZN1T1fEi
cfunc
STMFDsp!,{lr} ;此时r0已经包含了指向对象的指针
MOVr1,#2
BL_ZN1T1fEi
ADDr0,r0,r0,LSL#1 ;r0乘以3
LDMFDsp!,{pc}
END
下面的例子显示了如何用嵌入式汇编语言实现上面的例子。在此例中,使用__cpp关键字引用该函数。因此,用户不必了解已延伸的函数名。
structT{
T(inti):t(i){}
intt;
intf(inti);
};
intT::f(inti){returni+t;}
//定义被C++调用的汇编功能
__asmintasm_func(T*){
STMFDsp!,{lr}
MOVr1,#2;
BL__cpp(T::f);
ADDr0,r0,r0,LSL#1;r0乘以3
LDMFDsp!,{pc}
}
intf(){
Tt(5);//创建T类型的对象
returnasm_func(&t);
}
联系方
使用arm混合汇编计算两个64位的和_混合使用C、C++和汇编语之: C、C++ 和 ARM 汇编语言之间的调用...相关推荐
- C语言 请使用short int型的变量计算两个7位十进制整数的和,要求使用尽可能少的变量
请使用short int型的变量计算两个7位十进制整数的和,要求使用尽可能少的变量. (提示:需要输入四个数,分别是第一个数的前三位和后四位,第二个数的前三位和后四位.分别求和,注意后四位产生的进位. ...
- C语言练习题,short int型的变量计算两个7位十进制整数的和
请使用short int型的变量计算两个7位十进制整数的和,要求使用尽可能少的变量. **输入格式要求:"a = %3hd%4hd, b = %3hd%4hd" **输出格式要求: ...
- 高通发布两款64位处理器骁龙Snapdragon 810/808
尽管过去高通已经发布了数款64位处理器,但都集中在中低端领域.而现在,高通终于正式推出了两款面向高端市场的64位处理器,它们的型号分别为骁龙810和骁龙808,其主要特色是基于20纳米工艺制程,拥有更 ...
- mysql 64位整型_高性能MySQL笔记精简(整数和实数优化)
一.优化数据类型的几个原则: 1. 更小的通常更好 a) 一般情况下应该尽量使用可以正确存储数据的最小值数据类型 b) 更小的通常更快,因为它占用更少的磁盘内存和cpu缓存,处理时需要的cpu周期更少 ...
- Mac OS X下64位汇编与Linux下64位汇编的一些不同
1 首先系统调用号大大的不同:mac64和linux32的系统调用号也不同(虽然局部可能有相同) 2 mac64的系统调用号在: /usr/include/sys/syscall.h 可以查到,但是调 ...
- javascript 计算两个坐标的距离 米_土方全面应用计算
各种土方量的计算方法汇总 8.2.1 DTM法土方计算 由DTM模型来计算土方量是根据实地测定的地面点坐标(X,Y,Z)和设计高程,通过生成三角网来计算每一个三棱锥的填挖方量,最后累计得到指定范围内 ...
- javascript 计算两个坐标的距离 米_斜交桥梁桩基坐标复核计算
理解下面计算步骤必须看懂桥梁各部位细部图纸,具体参见视频课程内部桥梁识图课程. ①桥梁桩号核对及各墩台中心桩号计算 一般桥梁图纸都会已知桥梁中心桩号,有的图纸会给出桥梁起终点桩号,对于桥梁起终点桩号各 ...
- 计算两个向量间的欧氏距离_计算不同长度的两个向量之间的距离
欧几里德距离公式找出了欧几里德空间中任意两点之间的距离. 欧几里德空间中的一点也被称为欧几里德矢量. 您可以使用欧氏距离公式来计算两个不同长度的向量之间的距离. 对于不同维度的矢量,适用相同的原理. ...
- 64位 java 数据类型_全面解析Java支持的数据类型及Java的常量和变量类型
基本数据类型变量就是用来储存值而保留的内存位置.这就意味着当你创建一个变量时就会在内存中占用一定的空间. 基于变量的数据类型,操作系统会进行内存分配并且决定什么将被储存在保留内存中.因此,通过给变量分 ...
最新文章
- linux 调试利器gdb, strace, pstack, pstree, lsof
- 关于面向对象的总结和疑惑(转载可乐冰
- excel数据输入模型前的转换
- Python之禅 by Tim Peters
- 51NOD 2072 装箱问题 背包问题 01 背包 DP 动态规划
- java 和 区别_java 和 =的区别
- 梅露可物语虚拟服务器,【图片】【萌新】主界面的使用方法(零基础版)【梅露可物语日服吧】_百度贴吧...
- 小米笔记本引导修复_知识帖,笔记本电脑不能开机?你可以试试这些办法
- Servlet的介绍
- 统计学习中感知机的C++代码
- python爬虫练习--爬取猫眼top100电影信息
- SQOOP——MySQL 和 HDFS 的桥梁
- C语言实现简单计算器。
- AM3352的I2C驱动与传感器sht20的应用
- 固态硬盘数据恢复教您如何操作
- Tao Admin免费开源后台管理系统
- web开发前台,懒人建站资源库
- ng-class和ng-show的使用
- JS初中段考:袖珍西历
- 招聘季如何面试软件测试岗位?超全面试题分享
热门文章
- 浙江中医药大学第十一届程序设计竞赛题解
- RabbitMQ快速安装配置指南
- 实现物联网项目,你需要提前知道的6件事情
- 【12期 3月期刊 自荐】
- 《淘宝网开店 拍摄 修图 设计 装修 实战150招》一一2.7 横式构图和竖式构图...
- 方程式漏洞之复现window2008/win7 远程命令执行漏洞
- PowerShell实现批量收集SCVMM中虚拟机IP-续
- Programming C#.Classes and Objects.只读字段
- php:兄弟连之面向对象版图形计算器1
- MATLAB 批量改文件名称