历史上的今天
返回首页

历史上的今天

今天是:2024年09月11日(星期三)

正在发生

2020年09月11日 | 解析C语言与ARM汇编与的相互调用以及与 C++ 相互调用

2020-09-11 来源:elecfans

1.汇编程序访问C语言全局变量

全局变量只能通过地址间接调用,为了访问C++++语言中全局变量,首先要通过extern伪指令引入全局变量,然后将其地址装入寄存器中。


对于unsigned char类型,使用LDRB/STRB访问;

对于unsigned short类型,使用LDRH/STRH访问;

对于unsigned int类型,使用LDR/STR访问;

对于char类型,使用LDRSB/STRSB访问;

对于short类型,使用LDRSH/STRSH访问;


例子:

.text

.global asmsubrouTIne

.extern globvar

asmsubrouTIne:

LDR R1,=globvar

LDR R0,[R1]

ADD R0,R0,#2

STR R0,[R1]

MOV PC,LR

.end
解析C语言与ARM汇编与的相互调用以及与 C++ 相互调用

2.C程序调用汇编程序

C程序调用汇编程序首先通过extern声明要调用的汇编程序模块,声明中形参个数要与汇编程序模块中需要的变量个数一致,且参数传递要满足ATPCS规则,然后在C程序中调用。


例子:

#include

extern void *strcopy(char*d,char*s);//模块声明

int main()

{

char*srcstr=“first”;

char*dststr=“second”;

strcopy(dststr,srcstr);//汇编模块调用;

}

.text

.global strcopy

Strcopy:

LDRB R2,[R1],#1

STRB R2,[R0],#1

CMP R2,#0

BNE Sstcopy

MOV PC,LR

.end


汇编程序调用C程序

在调用之前必须根据C语言模块中需要的参数个数,以及ATPCS参数规则,完成参数传递,即前四个参数通过R0-R3传递,后面的参数通过堆栈传递,然后再利用B、BL指令调用。


例子:

int g(int a,int b,int c,int d,int e)//C语言函数原型

{

return(a+b+c+d+e);

}

汇编语言完成是求i+2i+3i+4i+5i的结果;

.global _start

.text

_start:

.extern g ;引入c程序

STR LR,{SP,-#4}!;保存PC

ADD R1,R0,R0

ADD R2,R1,R0

ADD R3,R1,R2

STR R3,{SP,#-4}!

ADD R3,R1,R1

BL g ;调用C函数g

ADD SP,SP,#4

LDR PC,[SP],#4

.end

return(0);


C和C++之间库的互相调用

昨晚有个朋友问我关于在C中调用C++库的问题,今天午饭后,由于脖子痛的厉害而没有加入到我们组的“每天一战”的行列中去,所以正好将C和C++之间的库调用关系做个总结。


1.extern “C”的理解:

很多人认为“C”表示的C语言,实际并非如此,“C”表示的是一种链接约定,只是因C和C++语言之间的密切关系而在它们之间更多的应用而已。实际上Fortran和汇编语言也常常使用,因为它们也正好符合C实现的约定。


extern “C”指令描述的是一种链接约定,它并不影响调用函数的定义,即时做了该声明,对函数类型的检查和参数转换仍要遵循C++的标准,而不是C。


2.extern “C”的作用:

不同的语言链接性是不同的,那么也决定了它们编译后的链接符号的不同,比如一个函数void fun(double d),C语言会把它编译成类似_fun这样的符号,C链接器只要找到该函数符号就可以链接成功,它假设参数类型信息是正确的。而C++会把这个函数编译成类似_fun_double或_xxx_funDxxx这样的符号,在符号上增加了类型信息,这也是C++可以实现重载的原因。


那么,对于用C编译器编译成的库,用C++直接链接势必会出现不能识别符号的问题,是的,需要extern “C”的时刻来了,它就是干这个用的。extern “C” 的作用就是让编译器知道要以C语言的方式编译和连接封装函数。


3.在C++中调用C库的例子:

1)。做一个C动态库:

// hello.c:

#include

void hello()

{

printf(“hellon”);

}

编译并copy到系统库目录下(也可以自己定义库目录,man ldconfig):

[root@coredump test]# gcc --shared -o libhello.so hello.c

[root@coredump test]# cp libhello.so /lib/

2)。写个C++程序去调用它:

// test.cpp

#include

#ifdef __cplusplus

extern “C” { // 告诉编译器下列代码要以C链接约定的模式进行链接

#endif

void hello();

#ifdef __cplusplus

}

#endif

int main()

{

hello();

return 0;

}

编译并运行:

[root@coredump test]# g++ test.cpp -o test -lhello

[root@coredump test]# 。/test

hello

[root@coredump test]#


3).__cplusplus宏的条件编译:

为什么要加这个条件编译呢?小沈阳有话:小妹,这是为什么呢?

因为这种技术也可能会用在由C头文件产生出的C++文件中,这样使用是为了建立起公共的C和C++文件,也就是保证当这个文件被用做C文件编译时,可以去掉C++结构,也就是说,extern “C”语法在C编译环境下是不允许的。


比如:将上面的test.cpp更名为test.c,将头文件改为stdio.h,将条件编译去掉,再用gcc编译就可以看到效果。而即使做了上面的修改,如果用g++编译就可以正常使用,这就是我上面说的“公共的C和C++文件”的意思。


4.C调用C++库:

C++调用C库看上去也不是那么困难,因为C++本身就有向前(向C)兼容的特性,再加上纯天然的extern “C”约定,使得一切都是那么自然。而让C调用C++的库似乎就没那么容易,不过也不是不可以的。


言归正传,还是要借助这纯天然的extern “C”。

1)做一个C++库:

// world.cpp

#include

void world()

{

std::cout 《《 “world” 《《 std::endl;

}

编译并copy到系统库目录下:

[root@coredump test]# g++ --shared -o libworld.so world.cpp

[root@coredump test]# cp libworld.so /lib/

2)做一个中间接口库,对C++库进行二次封装:

// mid.cpp

#include

void world();

#ifdef __cplusplus

extern “C” { // 即使这是一个C++程序,下列这个函数的实现也要以C约定的风格来搞!

#endif

void m_world()

{

world();

}

#ifdef __cplusplus

}

#endif

其中方法m_world即为libworld库中world方法的二次封装,编译并copy到系统库目录下:

[root@coredump test]# g++ --shared -o libmid.so mid.cpp -lworld

[root@coredump test]# cp libmid.so /lib/

3).C程序通过链接二次接口库去调用C++库:

// test.c

#include

int main()

{

m_world();

return 0;

}

编译并运行:

[root@coredump test]# gcc test.c -l mid -o test

[root@coredump test]# 。/test

world

[root@coredump test]#

推荐阅读

史海拾趣

Cybernetic Micro Systems公司的发展小趣事

随着业务的不断发展,CMS开始将目光投向国际市场。他们通过设立海外研发中心和生产基地,积极拓展海外市场。同时,CMS还加强了与国际知名企业的合作与交流,共同推动电子行业的发展。如今,CMS的产品已经遍布全球多个国家和地区,成为电子行业中不可忽视的一股力量。

Ercona Corp公司的发展小趣事

随着业务的不断发展,CMS开始将目光投向国际市场。他们通过设立海外研发中心和生产基地,积极拓展海外市场。同时,CMS还加强了与国际知名企业的合作与交流,共同推动电子行业的发展。如今,CMS的产品已经遍布全球多个国家和地区,成为电子行业中不可忽视的一股力量。

展恒电子(Broadic)公司的发展小趣事

2012年,展恒电子与FANSO(孚安特)签订战略合作协议,共同向智能电表厂家提供配套销售服务。这一合作不仅拓宽了展恒电子的销售渠道,也提升了公司在智能电表市场的影响力。此后,展恒电子继续与多家知名企业建立战略合作关系,通过资源共享和优势互补,实现了共赢发展。

振宝佳(DMBJ)公司的发展小趣事

作为一家具有国际视野的企业,振宝佳公司始终坚持以质量为核心的发展理念。为了满足出口欧盟的质量要求,公司不断提升产品的品质和性能,加强质量控制和检验。经过多年的努力,公司终于成功获得了出口欧盟的资格认证。这一成就的取得不仅证明了公司的实力和能力,也为其在国际市场上赢得了更多的机会和声誉。

以上五个故事框架均基于振宝佳(DMBJ)公司在电子行业发展的实际情况进行编写,旨在展示公司在技术、市场、生产、质量等方面的努力和成就。请注意,这些故事仅为概述,具体细节和数据可能需要根据实际情况进行补充和完善。

Chip Quik公司的发展小趣事

Chip Quik公司在电子行业初创时,以其独特的芯片焊接技术脱颖而出。该公司研发了一种新型的低温焊接材料,能够在不损坏周围元件的情况下快速、准确地修复或更换芯片。这一创新技术迅速吸引了电子维修市场的关注,公司借此机会成功进入市场。

Digital Equipment Corp公司的发展小趣事

随着业务的不断扩展和产品的不断创新,DEC在1966年决定公开上市。这次上市不仅为DEC带来了大量的资金,还提升了其在业界的知名度。借助资本市场的力量,DEC能够进一步扩大生产规模,加强研发能力,并推出更多具有竞争力的产品。在上市后的几年里,DEC的销售收入快速增长,成为计算机行业的重要参与者。

问答坊 | AI 解惑

直流电机PWM调速,C51、Proteus环境

分享一个利用AT89C51产生PWM信号控制直流电机调速的模拟模型。 软件环境:Keil C51,Proteus 已有文件:Keil C51工程文件,C51源文件,Proteus模型。已经产生的hex文件…

查看全部问答>

传感器产业发展的新动向

传感器技术是现代科技的前沿技术,传感器产业也是国内外公认的具有发展前途的高技术产业,它以其技术含量高、经济效益好、渗透能力强、市场前景广等特点为世人瞩目。     我国自动化方面的专家呼吁:目前复杂系统越来越复杂,自动化已经 ...…

查看全部问答>

【Labview】labview学习网站有哪些

大家学习labview都在哪些网站和论坛上混啊?     本人常去的有两个,www.ni.com这是NI的网站,学习labview怎么能不上呢!     其次是www.gsdzone.net 这是我常去的Labview论坛,里面有资深NI工程师潜伏哦!…

查看全部问答>

WHY???

在下面代码中发现产生INTSRC_CAM中断后若BIT_SUB_CAM_P和BIT_SUB_CAM_C都没触发 那就执行 “//有疑问的代码”标记的代码 即屏蔽掉了Camera所有中断并返回了SYSINTR_NOP 这样相应的IST得不到执行也就不会调用InterruptDone来打开屏蔽的中断那以后 ...…

查看全部问答>

windows mobile 6.0 下ras拨号的问题

我在mobile5下使用ras函数RasDial拨号连接没有问题 同样的程序以及拨号配置在mobile 6下却无法拨号成功 不知道5跟6在这方面有什么不一样吗 有经验的大虾帮忙看看,谢谢!!!…

查看全部问答>

请教版主,stm32的抗静电等级与潮敏是多少?

您好,版主! 我想请教下stm32的抗静电等级与潮敏各是多少呢? 静电等级:J8000 J5000 J2000 J1000 J500 J200 是那个级别的 是j2000吗? 潮敏:C1 C2 C3 C4 C5是那个级别的呢?…

查看全部问答>

2013年电子设计大赛是不是9月7号晚上结束后就不能在做了

本帖最后由 paulhyde 于 2014-9-15 03:13 编辑 同题目    …

查看全部问答>

这里有----Multisim电子电路仿真教程

想学习模电的同学,也可以试试仿真软件哦,,这里有Multisim电子电路仿真教程,但动手还是主要的,,,,, 通过对该软件的学习和使用,读者可以轻松地拥有一个元件设备非常完善的虚拟电子实验室,进而可以完成电子电路的各种实验和设计。全书共9章 ...…

查看全部问答>

为什么输出电压的反馈取自于最终的输出电压后变压器就发出响声?

我刚开始的反馈是取自于辅助绕组,发现输出电压随着输入电压变化大约1.2V。现在用面包板搭了一个从输出电压反馈的电路,用了PC817+TL431的方法,上电后变压器就开始发出声音了,下面是我的电路图,就问,这个为什么会响呢?该怎么解决呢? …

查看全部问答>