ARM汇编与C言语的混合编程

news2025/1/11 13:00:59

1. C言语如何与汇编进行交互

有些时候,我们需要在汇编代码中调用C代码,或者说C代码中调用汇编代码。

那么,汇编调用C代码,或者C代码调用汇编函数,他们的函数参数、返回值是如何传递的?

对应ARM架构来说,有一个 ATPCS (ARM-Thumb Produce Call Standard),ARM-Thumb过程调用标准。ATPCS 标准规定了下面几个方面:

  • 1、函数调用过程中的寄存器用法。比如说 R4-R11 、 R13(SP) 、R14(LR) 这些寄存器的值,应该由子程序保证不改变。也就是说在执行子程序时,如果这些寄存器需要被修改的话,那么应该先压栈,然后才能再去调用子程序。
  • 2、规定函数参数和返回值的传递。一般来说 R0-R3 这4个寄存器用于函数的输入参数,然后函数的返回值是放在 R0 寄存器的。当然,如果输入参数大于 4 个参数是,寄存器就不够用了,那么这时需要使用到栈。
  • 3、规定栈对齐。如果汇编代码调用C函数,就要确保当前的栈指针指向了双字对齐的地址(如0x20000000、0x20000008、0x20002010 等)。

对于在函数调用时的参数传递过程,可归纳为如下表格:

寄存器输入参数返回值
R0第 1 个输入参数函数返回值
R1第 2 个输入参数不使用,或者返回值(返回值是64位时)
R2第 3 个输入参数不使用
R3第 4 个输入参数不使用

函数调用过程中,寄存器的用法如下表:

在这里插入图片描述

2. 汇编调用C函数

汇编调用C函数,是汇编与C语言混合编程遇见最多的场合了。

前面说过了,ATPCS标准,函数调用时,使用 R0-R3 传递参数,R0 作为返回值。

比如,下面的一个C函数:

int my_add_c(int x1, int x2, int x3, int x4)
{
    return (x1 + x2 + x3 + x4);
}

根据ATPCS标准,x1=R0, x2=R1, x3=R2, x4=R3, 然后返回值是存放在 R0 寄存器的。另外,还需要注意调用者保存的寄存器(R0-R3, R12)这些寄存器可能会被C函数修改。

下面的汇编代码调用C函数示例:

MOVS R0, #0x1 ; First parameter (x1)
MOVS R1, #0x2 ; Second parameter (x2)
MOVS R2, #0x3 ; Third parameter (x3)
MOVS R3, #0x4 ; Fourth parameter (x4)
IMPORT my_add_c
BL my_add_c ; Call “my_add_c” function. Result store in R0

3. C函数调用汇编函数

使用汇编代码实现函数,然后通过C代码调用汇编函数。

在汇编代码中,我们需要确保,如果C代码要修改被调用者保存的寄存器(如R4-R11等),那么在汇编代码中要先对这些寄存器进行压栈处理,然后退出函数之前将这些寄存器恢复。

如果这个C函数还可能会调用到另一个函数,那么在汇编代码中,还需要保存LR寄存器的值。

汇编函数的实现:

EXPORT My_Add

my_add_asm FUNCTION
    ADDS R0, R0, R1
    ADDS R0, R0, R2
    ADDS R0, R0, R3
    BX LR ; Return result in R0
    ENDFUNC

当然,如果汇编代码需要访问C代码中的变量,则可以使用 IMPORT 关键字(不同的工具链可能不一样,gcc工具链是 .global 关键字)

然后,在C代码中调用这个汇编函数。

extern int my_add_asm(int x1, int x2, int x3, int x4);
...
int y;
...
y = my_add_asm(1, 2, 3, 4); // call the my_add_asm function

4. C代码嵌入汇编

很多编译器都支持,在C代码中,实现汇编函数。

对应 ARM CC 编译器来说,需要在函数前面加上 __asm 关键字修饰。

比如,下面在C代码中,实现一个加法函数:

__asm int My_Add1(int x1, int x2, int x3, int x4)
{
    ADDS R0, R0, R1
    ADDS R0, R0, R2
    ADDS R0, R0, R3
    BX LR ; Return result in R0
}

当然,如果你需要在嵌入的汇编函数中,需要访问变量或者地址值,可以使用 __cpp 关键字,如下:

__asm void function_A(void)
{
    PUSH {R0-R2, LR}
    BL __cpp(LCD_clr_screen) ; 方法1调用C函数
    
    LDR R0,=__cpp(&pos_x)    ; 获取C代码变量地址,存放到R0寄存器
    LDR R0, [R0]			 ; 把 R0 的值,写到以R0的值作为地址的内存中
    
    LDR R1,=__cpp(&pos_y)
    LDR R1, [R1]
    
    LDR R2, =__cpp(LCD_pixel_set) ; Import the address of a function
    BLX R2 ; 方法2调用C函数
    POP {R0-R2, PC}
}

5. C代码内联汇编

在一个C函数中,我们可以使用内联汇编的语法,把一些汇编指令,插入到C函数里面。

比如下面的一个C函数:

int My_Add2(int i, int j)
{
    int ret;
    __asm
    {
        ADD ret, i, j
    }
    return ret;
}

这个函数,本身是一个C函数,但是在C函数内部插入了汇编指令。

对于GCC编译器,支持的内联汇编语法如下:

__asm (“ inst1 op1, op2, ... \n”
“ inst2 op1, op2, ... \n”
...
“ instN op1, op2, . \n”
: output_operands /* optional */
: input_operands /* optional */
: clobbered_operands /* optional */
);

关于内联汇编语法的详细介绍,可以参考 GUN 工具链的帮助文档 GCC-Inline-Assembly-HOWTO 。

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

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

相关文章

学习开发一个RISC-V上的操作系统(汪辰老师) — unrecognized opcode `csrr t0,mhartid‘报错问题

前言 (1)此系列文章是跟着汪辰老师的RISC-V课程所记录的学习笔记。 (2)该课程相关代码gitee链接; (3)PLCT实验室实习生长期招聘:招聘信息链接 正文 (1)在跟着…

【重拾C语言】五、模块化程序设计——函数(定义、调用、参数传递、结果返回、函数原型;典例:打印字符图形、验证哥德巴赫猜想)

目录 前言 五、模块化程序设计——函数 5.1 计算三角形的重心 5.2 函数 5.2.1 函数定义 5.2.2 函数调用 a. 函数调用的形式和过程 b. 参数传递 值传递 指针传递 c. 函数结果返回 5.2.3 函数原型(先调用后定义) 5.3 程序设计实例 5.3.1 打印…

【14】c++设计模式——>工厂模式

简单工厂模式的弊端 简单工厂模式虽然简单,但是违反了设计模式中的开放封闭原则,即工厂类在数据增加时需要被修改,而我们在设计时对于已经设计好的类需要避免修改的操作,而选用扩展的方式。 工厂模式设计 简单工厂模式只有一个…

【Java】接口 interface

目录 概述 示例代码: 接口成员访问特点 示例代码: 概述 什么是接口 接口就是一种公共的规范标准,只要符合规范标准,大家都可以调用。 Java 中的接口更多的体现在对行为的抽象! 1. 接口 用关键字 interface 修饰 pub…

AtCoder Beginner Contest 231(D-F,H)

D - Neighbors (atcoder.jp) (1)题意 给出M组关系,问是否有一个排列,能表示A[i]和B[i]相邻 (2)思路 考虑如果有环,显然不能满足排列,因为排列中度数最多为2,若有超过2的显…

JavaScript操作CSS样式

上节课我们基本完成了游戏的主体,这节课我们来学习如果使用JavaScript去操作CSS样式 ● 例如,我们现在想当玩家输入对的数字之后,我们讲背景改为绿色,并且把number的框宽度变大 const secretnumber Math.trunc(Math.random() * …

第十六章 类和对象——运算符重载

运算符重载概念:对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型。 一、加号运算符重载 作用:实现两个自定义数据类型相加的运算 class Person {public:Person() {};Person(int a, int b){this->m_A a;t…

5个适合初学者的初级网络安全工作,网络安全就业必看

前言 网络安全涉及保护计算机系统、网络和数据免受未经授权的访问、破坏和盗窃 - 防止数字活动和数据访问的中断 - 同时也保护用户的资产和隐私。鉴于公共事业、医疗保健、金融以及联邦政府等行业的网络犯罪攻击不断升级,对网络专业人员的需求很高,这并…

【异常错误】WSL2设置为全核cpu和全部内存

今天偶尔发现 WSL占用的内存是真实内存的一半(通过htop命令查看即可) 现在需要修改配置到使用全部的CPU资源: Windows R 键, 输入 %UserProfile% 并运行进入用户文件夹, 新建文件 .wslconfig,文件内容如下&#xf…

light client轻节点简介

1. 引言 前序博客: Helios——a16z crypto构建的去中心化以太坊轻节点 去中心化和自我主权对于Web3的未来至关重要,但是这些理想并不总适用于每个项目或应用程序。在非托管钱包和bridges等工具中严格优先考虑安全性而不是便利性的用户,可选择运行全节…

【JavaEE】多线程(五)- 基础知识完结篇

多线程(五) 文章目录 多线程(五)volatile关键字保证内存可见性JMM(Java Memory Model) 不保证原子性 wait 和 notifywait()notify()线程饿死 上文我们主要讲了 synchronized以及线程安全的一些话题 可重入…

【Unity】3D贪吃蛇游戏制作/WebGL本地测试及项目部署

本文是Unity3D贪吃蛇游戏从制作到部署的相关细节 项目开源代码:https://github.com/zstar1003/3D_Snake 试玩链接:http://xdxsb.top/Snake_Game_3D 效果预览: 试玩链接中的内容会和该效果图略有不同,后面会详细说明。 游戏规则 …

图像分割中的色块的提取

一 色块提取方法: ①首先是色彩模型的转换 由RGB颜色空间转到HSV颜色空间 原因:RGB颜色空间适合显示系统,但是各分量间相关性很强,比如当图像亮度发生变化时,RGB三个分量都会发生相应改变 但是HSV颜色空间更能感知颜色…

【Java 进阶篇】JDBC 数据库连接池 C3P0 详解

数据库连接池是数据库编程中常用的一种技术,它可以有效地管理数据库连接,提高数据库访问的性能和效率。在 Java 编程中,有多种数据库连接池可供选择,其中之一就是 C3P0。本文将详细介绍 C3P0 数据库连接池的使用,包括原…

LabVIEW使用ZigBee无线传感器开发住宅负载电力应用

LabVIEW使用ZigBee无线传感器开发住宅负载电力应用 长期以来,住宅客户的需求一直是电力行业的一部分。由于公用事业需要建设基础设施以满足即时和长期需求,因此公用事业账单既包含能源费用,其中衡量客户随时间消耗的总电量,也包含…

网络攻击常见手段总结

网络攻击常见手段总结 IP 欺骗 IP 欺骗技术是什么? IP 欺骗技术就是伪造某台主机的 IP 地址的技术。通过 IP 地址的伪装使得某台主机能够伪装另外的一台主机,而这台主机往往具有某种特权或者被另外的主机所信任。 攻击时,伪造大量的 IP 地…

文件操作 和 IO - 详解

一,认识文件 1.1 树形结构组织和目录 文件是对于"硬盘"数据的一种抽象,在一台计算机上,有非常多的文件,这些文件是通过 "文件系统" 来进行组织的,本质上就是通过 "目录"(文件夹) 这样…

PyTorch实例:简单线性回归的训练和反向传播解析

文章目录 🥦引言🥦什么是反向传播?🥦反向传播的实现(代码)🥦反向传播在深度学习中的应用🥦链式求导法则🥦总结 🥦引言 在神经网络中,反向传播算法…

第八章 排序 四、冒泡排序

目录 一、算法思想 二、例子 三、代码实现 四、验证 五、算法性能分析 注意:要分清楚交换次数和移动次数 六、总结 一、算法思想 从后往前,两两比较相邻元素的值,若为逆序,则交换它们的值,直到全部比较完。 二…

typescript: Builder Pattern

/*** file: CarBuilderts.ts* TypeScript 实体类 Model* Builder Pattern* 生成器是一种创建型设计模式, 使你能够分步骤创建复杂对象。* https://stackoverflow.com/questions/12827266/get-and-set-in-typescript* https://github.com/Microsoft/TypeScript/wiki/…