当前位置: 首页 > news >正文

汇编程序调用 C 程序详解

文章目录

1. ATPCS 规则

2. 汇编和C程序传递参数

汇编程序向 C 程序的函数传递参数

C 程序返回结果给汇编程序

代码示例

3. C 函数使用栈

4. C 语言中读写寄存器


在嵌入式开发中,经常需要在 C 程序和 ARM 汇编程序之间进行相互调用。为了保证这些调用的正确性和兼容性,ARM 提出了 ATPCS(ARM-Thumb Procedure Call Standard)规范。该规范定义了函数调用时的基本规则和寄存器使用约定。

1. ATPCS 规则

ATPCS 是 ARM 和 THUMB 指令集过程序调用的规范,它规定了函数调用时如何传递参数、如何获取参数以及如何返回值。

寄存器使用规则:

  • 在函数中,通过寄存器 R0-R3 传递参数,被调用的函数在返回前无须恢复寄存器 R0-R3 的内容。
  • 在函数中,通过寄存器 R4~R11 保存局部变量。
  • 寄存器 R12 用作函数间的 scratch 寄存器,即临时寄存器。
  • 寄存器 R13 用作堆栈指针,即 SP(Stack Pointer),在函数中寄存器 R13 不能用于其他用途。寄存器 SP 在进入函数时的值和退出函数时的值必须相等。
  • 寄存器 R14 用于存放返回地址,即 LR(Link Register),它用于存放调用函数的返回地址。函数返回时,CPU 会跳转到 LR 指向的地址继续执行调用函数。
  • 寄存器 R15 是程序计数器,即 PC(Program Counter),用于指向当前指令的地址,指令执行时自动递增。

示例代码

假设一个 C 函数 add,它接收两个整数参数并返回它们的和:

// add.c
int add(int a, int b) {return a + b;
}

汇编代码示例:

.global main.extern add  // 声明外部函数 addmain:MOV R0, #5       // 第一个参数 a = 5MOV R1, #10      // 第二个参数 b = 10BL add           // 调用 C 函数 addMOV R7, #1       // syscall: exitSWI 0            // 软件中断,退出程序
  • .global main:声明 main 函数为全局符号,以便链接器能够识别和连接它。
  • .extern add:声明外部符号 add,表明 add 函数是在外部文件中定义的。
  • MOV R0, #5MOV R1, #10:将值 5 和 10 分别加载到寄存器 R0 和 R1 中,这两个寄存器用于传递参数 a 和 b。
  • BL add:调用 C 函数 add。BL 指令(Branch with Link)会将当前 PC 值存储到 LR 中,并跳转到 add 函数的地址。
  • MOV R7, #1SWI 0:用于执行软件中断,退出程序。

函数调用前准备:

  • 在调用 add 函数之前,需要将参数准备好。根据 ATPCS 规则,前四个参数依次存放在 R0、R1、R2 和 R3 寄存器中。

函数调用:

  • 使用 BL add 指令调用 add 函数。BL 指令会保存返回地址到 LR 寄存器,并跳转到 add 函数的入口地址。

函数返回:

  • add 函数执行完毕后,返回值存放在 R0 寄存器中,CPU 会从 LR 寄存器中读取返回地址并跳转回调用函数继续执行。

2. 汇编和C程序传递参数

汇编程序向 C 程序的函数传递参数

在汇编程序中调用 C 函数时,通常需要传递参数给 C 函数。ATPCS 规范规定了如何传递参数的方法:

  • 参数少于等于 4 个时:
    • 使用寄存器 R0 至 R3 来进行参数传递。第一个参数传递到 R0,第二个参数传递到 R1,依此类推。如果只有一个参数,它将存储在 R0 中;如果有两个参数,第一个参数在 R0 中,第二个参数在 R1 中,以此类推,直到 R3。
  • 参数大于 4 个时:
    • 前四个参数按照上述方法传递。剩余的参数通过堆栈传递。调用函数时,调用者需要将这些多余参数压入堆栈。入栈的顺序是从最后一个参数开始,即最后一个参数先入栈。

C 程序返回结果给汇编程序

C 函数执行完毕后,需要将结果返回给调用它的汇编程序。返回值的传递方法也有一定的规范:

  • 结果为一个 32 位的整数时:

    • 通过寄存器 R0 返回。函数的返回值会存储在 R0 寄存器中。
  • 结果为一个 64 位整数时:

    • 通过寄存器 R0 和 R1 返回。64 位整数的低 32 位存储在 R0 中,高 32 位存储在 R1 中。
  • 结果为一个浮点数时:

    • 通过浮点寄存器 f0、d0 或 s0 返回。浮点数结果存储在浮点寄存器中。
  • 结果为一个复杂的浮点数时:

    • 通过寄存器 f0fN 或者 d0dN 返回。复杂浮点数的每一部分可能存储在不同的浮点寄存器中。
  • 对于传递更多位数的结果:

    • 通过调用者的不同寄存器来传递,具体依据实际情况和编译器的规定。

代码示例

假设一个 C 函数 multiply,它接收两个整数参数并返回它们的乘积:

// multiply.c
int multiply(int x, int y) {return x * y;
}

在汇编程序中调用这个 C 函数并传递参数 67

.global main
.extern multiply  // 声明外部函数 multiplymain:MOV R0, #6       // 第一个参数 x = 6MOV R1, #7       // 第二个参数 y = 7BL multiply      // 调用 C 函数 multiplyMOV R7, #1       // syscall: exitSWI 0            // 软件中断,退出程序
  • MOV R0, #6MOV R1, #7:将值 6 和 7 分别加载到寄存器 R0 和 R1 中。这两个寄存器用于传递参数 x 和 y。
  • BL multiply:调用 C 函数 multiply。BL 指令会将当前 PC 值存储到 LR 中,并跳转到 multiply 函数的地址。
  • MOV R7, #1SWI 0:用于执行软件中断,退出程序。

调用完 multiply 函数后,乘积结果会存储在 R0 中。通过这种方式,可以在汇编程序和 C 程序之间进行参数传递和结果返回。

3. C 函数使用栈

在 C 程序和 ARM 汇编程序的函数调用过程中,使用栈(stack)是非常重要的一部分。栈的主要作用有两个:保存现场/上下文,传递参数。

保存现场/上下文

  1. 保存现场: 在程序执行过程中,当发生函数调用时,需要暂时保存当前的执行现场,以便函数执行完毕后可以恢复到调用前的状态。这个保存的过程通常称为“保存现场”。

    例如,当 CPU 运行到某些寄存器时(如 R0~R3,LR 等),这些寄存器中可能存有重要的数据。如果直接跳转到函数去执行,而不保存这些寄存器的数据,函数执行过程中对这些寄存器的操作就会破坏原有的数据。因此,需要先将这些寄存器的数据暂时存放到栈中。

  2. 保存上下文: 保存上下文的过程和保存现场类似。上下文指的是当前程序执行的状态,包括寄存器内容、程序计数器等信息。当函数调用发生时,需要将这些上下文信息保存到栈中,以便函数执行完毕后能够准确地恢复到调用前的状态。

    因此,在函数调用之前,应该将这些寄存器的数据临时保存到栈中,等待函数执行完毕返回后,再恢复现场。这样 CPU 就可以正确地继续执行后续的指令。

传递参数

  1. 传递参数: 当参数数量大于 4 个时(不包括第 4 个参数),第 4 个参数后的参数就保存在栈中。

    传递参数的过程通常如下:

    • 当被调用函数的参数超过 4 个时,前四个参数通过寄存器 R0~R3 传递,剩余的参数依次压入栈中。
    • 在函数调用过程中,调用者会将这些参数按顺序压入栈中,函数被调用时通过从栈中读取这些参数来进行处理。

代码示例

假设一个 C 函数 func,它接收五个整数参数并返回它们的和:

// func.c
int func(int a, int b, int c, int d, int e) {return a + b + c + d + e;
}

在汇编程序中调用这个 C 函数并传递参数 1, 2, 3, 4, 5

.global main
.extern func  // 声明外部函数 funcmain:MOV R0, #1       // 第一个参数 a = 1MOV R1, #2       // 第二个参数 b = 2MOV R2, #3       // 第三个参数 c = 3MOV R3, #4       // 第四个参数 d = 4PUSH {R4}        // 保存现场,R4 用于传递第 5 个参数MOV R4, #5       // 第五个参数 e = 5BL func          // 调用 C 函数 funcPOP {R4}         // 恢复现场MOV R7, #1       // syscall: exitSWI 0            // 软件中断,退出程序
  • MOV R0, #1MOV R3, #4:将前四个参数加载到寄存器 R0~R3 中。
  • PUSH {R4}:将寄存器 R4 的内容压入栈中,以便在使用 R4 传递第 5 个参数时,不破坏原有的内容。
  • MOV R4, #5:将第 5 个参数加载到寄存器 R4 中。
  • BL func:调用 C 函数 func。BL 指令会将当前 PC 值存储到 LR 中,并跳转到 func 函数的地址。
  • POP {R4}:从栈中恢复寄存器 R4 的内容,恢复现场。
  • MOV R7, #1SWI 0:用于执行软件中断,退出程序。

通过使用栈,可以确保在函数调用过程中,参数能够正确传递,并且在函数调用完毕后,能够正确恢复程序的执行现场,从而保证程序的正常运行。

4. C 语言中读写寄存器

在嵌入式系统开发中,经常需要访问芯片上某些特定模块的寄存器。这些寄存器并不位于 CPU 内部,而是在芯片的外部模块中。我们通过指针访问这些寄存器的方式与访问普通内存一样。

寄存器地址与指针

每个寄存器都有一个固定的地址。通过定义一个指向这个地址的指针,可以对寄存器进行读写操作。

例如,假设有一个寄存器 CCM_CCGR1,它的地址为 0x20C406C,我们可以定义一个指向该地址的指针,并使用 volatile 关键字告知编译器该寄存器的值可能会被外部硬件或其他程序修改,不应进行优化。

示例代码

volatile unsigned int *CCM_CCGR1 = (volatile unsigned int *)(0x20C406C);

这里的 volatile unsigned int * 表示一个指向 unsigned int 类型的指针,并且这个指针指向的数据是 volatile 类型,即数据可能会被外部因素修改。

读写寄存器

通过这个指针,我们可以进行读写操作:

  • 读寄存器
val = *CCM_CCGR1;

这行代码将读取 CCM_CCGR1 寄存器的值,并存储到变量 val 中。

  • 写寄存器
*CCM_CCGR1 |= (3 << 30);
  • *CCM_CCGR1 通过指针访问寄存器 CCM_CCGR1
  • |= (3 << 30) 是一个位操作,将 3 左移 30 位,得到一个值,这个值在二进制中表示为 11(即第 31 位和第 30 位都为 1),然后将这个值与 CCM_CCGR1 的当前值进行按位或操作,结果将这两位设置为 1。

通过上述方法可以在 C 语言中方便地对寄存器进行读写操作,确保对硬件寄存器的正确访问和修改。这种方法广泛应用于嵌入式系统开发中,特别是在处理器需要与外部设备进行通信或控制时。

相关文章:

汇编程序调用 C 程序详解

文章目录 1. ATPCS 规则 2. 汇编和C程序传递参数 汇编程序向 C 程序的函数传递参数 C 程序返回结果给汇编程序 代码示例 3. C 函数使用栈 4. C 语言中读写寄存器 在嵌入式开发中&#xff0c;经常需要在 C 程序和 ARM 汇编程序之间进行相互调用。为了保证这些调用的正确性…...

代码随想三刷图论篇1

代码随想三刷图论篇1 98. 所有可达路径题目代码99. 岛屿数量题目代码100. 岛屿的最大面积题目代码101. 孤岛的总面积题目代码102. 沉没孤岛题目代码103. 水流问题题目代码98. 所有可达路径 题目 链接 代码 import java.util.*;class Main{public static void main(String […...

Windows 快捷键汇总

Windows 快捷键汇总 前言进阶快捷键【最好用】Chrome 常用快捷键【跟 Windows 快捷键不搭杆&#xff0c;但常用】基础快捷键扩展快捷键 前言 Coder 苦鼠标久已&#xff0c;整理汇总 Windows 快捷键包括一些常用的快捷键&#xff0c;比如“浏览器”相关的快捷键内容分为四小节&…...

微服务有哪些组件?

1.注册中心&#xff1a;用于服务的注册和发现&#xff0c;管理微服务的地址 Nacos&#xff0c;Eureka 2.配置中心&#xff1a;集中管理微服务的配置中心 Nacos config 3.远程调用&#xff1a;用于不同微服务间的通信和协作 RESTful API&#xff08;RestTemplate&#xff0…...

camera-qsc-crosstalk校准数据XTALK回写

问题背景 手机越做越紧凑&#xff0c;需要模组和芯片尺寸越做越小&#xff0c;在尺寸一定的基础上&#xff0c;高像素和大像素&#xff0c;对于手机摄像头来说&#xff0c;一直是一对矛盾的存在。 高像素&#xff1a;带来高分辨率画质大像素&#xff1a;带来暗态下高感光度和…...

混合贪心算法求解地铁线路调度

一、问题描述 城市轨道交通的繁荣发展&#xff0c;带来了车辆资源需求的日益增加。如何兼顾运营服务水平和运营成本&#xff0c;以最少的车底优质地完成运输任务成为一大严峻问题。本题在后续的描述中将由多辆动车和拖车组合而成的车组称为车底。在日常的运营组织中&#xff0…...

vue项目:关闭页面,删除本地登录信息

vue项目&#xff1a;关闭页面&#xff0c;删除本地登录信息 代码 代码 import { removeToken } from /utils/auth //区分关闭和刷新页面&#xff0c;关闭时退出登录 window.onload ()>{if(!window.sessionStorage["tempFlag"]){removeToken();location.reload()…...

获奖案例回顾|基于卫星遥感和无人机的水稻全流程风险减量项目

引言 在现代农业保险领域&#xff0c;技术创新是推动行业进步的关键。珈和科技与太平财险的合作&#xff0c;旨在利用先进的卫星遥感和无人机技术&#xff0c;解决传统农业保险面临的诸多挑战&#xff0c;从而提升保险效率和服务质量。本次分享的项目案例获得了《金融电子化》…...

全栈 Discord 克隆:Next.js 13、React、Socket.io、Prisma、Tailwind、MySQL笔记(一)

前言 阅读本文你需要有 Next.js 基础 React 基础 Prisma 基础 tailwind 基础 MySql基础 准备工作 打开网站 https://ui.shadcn.com/docs 这不是一个组件库。它是可重用组件的集合&#xff0c;您可以将其复制并粘贴到应用中。 打开installation 选择Next.js 也就是此页面…...

【Unity】制作简易计时器

一、创建计时器相关的变量 我们需要创建三个变量&#xff0c;分别是&#xff1a;计时时长、计时剩余时长、是否处于计时状态。 public float duration;//计时时长 public float remain; //计时剩余时长 public bool isCount; //是否处于计时状态 二、初始化变量 我们可以直…...

TDesign组件库日常应用的一些注意事项

【前言】Element&#xff08;饿了么开源组件库&#xff09;在国内使用的普及率和覆盖率高于TDesign-vue&#xff08;腾讯开源组件库&#xff09;&#xff0c;这也导致日常开发遇到组件使用上的疑惑时&#xff0c;网上几乎搜索不到其文章解决方案&#xff0c;只能深挖官方文档或…...

51单片机7(点亮第一个LED)

一、LED简介 1、LED&#xff0c;它是一个发光二极管&#xff0c;它具有单向导电性&#xff0c;那么通过5毫安的一个电流&#xff0c;就可以使它发光&#xff0c;那么电流越大&#xff0c;它的发光也就越强&#xff0c;但是电流不能过大&#xff0c;过大会把这个发光二极管给烧…...

基于Vue和UCharts的前端组件化开发:实现高效、可维护的词云图与进度条组件

基于Vue和UCharts的前端组件化开发&#xff1a;实现高效、可维护的词云图与进度条组件 摘要 随着前端技术的迅速发展和业务场景的日益复杂&#xff0c;传统的整块应用开发方式已无法满足现代开发的需求。组件化开发作为一种有效的解决方案&#xff0c;能够将系统拆分为独立、…...

CentOS 系统监控项

在维护和优化 CentOS 系统时&#xff0c;实时监控硬件和资源的使用情况非常重要。为了满足工作需要&#xff0c;可以定时采集 CentOS 系统相关的监控数据&#xff0c;并将其推送到 Prometheus 进行集中监控和管理。以下是日常采集项及对应的 shell 命令&#xff0c;并附上每项命…...

连锁直营店小程序赋能多店如何管理

如商超便利店卖货线下场景&#xff0c;也有不少品牌以同城多店和多地开店经营为主&#xff0c;获取店铺周围客户和散流&#xff0c;如今线上重要性凸显&#xff0c;品牌电商发展是经营的重要方式之一&#xff0c;也是完善同城和外地客户随时便捷消费的方式之一。 多个门店管理…...

决策树算法入门到精通:全面解析与案例实现

1. 介绍决策树算法 决策树的基本概念和原理 决策树是一种基于树形结构的分类和回归方法&#xff0c;通过对数据集进行递归地划分&#xff0c;每个内部节点表示一个属性上的判断&#xff0c;每个叶节点代表一种类别或者数值。 决策树在机器学习中的应用场景 分类问题&#xf…...

LangChain —— 多模态大模型的 prompt template

文章目录 一、如何直接将多模态数据传输给模型二、如何使用 mutimodal prompts 一、如何直接将多模态数据传输给模型 在这里&#xff0c;我们演示了如何将多模式输入直接传递给模型。对于其他的支持多模态输入的模型提供者&#xff0c;langchain 在类中提供了内在逻辑来转化为期…...

ssh升级

文章目录 ssh升级一、解包ssh、ssl二、更新安装ssl三、手动更新手动复制库文件四、创建符号链接五、更新库路径六、验证库文件七、设置库路径环境变量八、配置、编译、安装OpenSSH&#xff1a;意外&#xff1a;缺少 zlib 的开发库解决方法&#xff1a; 九、刷新ssh服务、查看ss…...

51单片机10(蜂鸣器介绍)

一、蜂鸣器介绍&#xff1a; 1、蜂鸣器是一种一体化结构的电子讯响器&#xff0c;采用直流电压供电&#xff0c;广泛应用于电子产品中作为发声器件。蜂鸣器主要分为压电式蜂鸣器和电磁式蜂鸣器。 &#xff08;1&#xff09;压电式蜂鸣器&#xff0c;它主要由多谐的一个增胀器…...

Python爬虫:基础爬虫架构及爬取证券之星全站行情数据!

爬虫成长之路&#xff08;一&#xff09;里我们介绍了如何爬取证券之星网站上所有A股数据&#xff0c;主要涉及网页获取和页面解析的知识。爬虫成长之路&#xff08;二&#xff09;里我们介绍了如何获取代理IP并验证&#xff0c;涉及了多线程编程和数据存储的知识。此次我们将在…...

基于大模型的 UI 自动化系统

基于大模型的 UI 自动化系统 下面是一个完整的 Python 系统,利用大模型实现智能 UI 自动化,结合计算机视觉和自然语言处理技术,实现"看屏操作"的能力。 系统架构设计 #mermaid-svg-2gn2GRvh5WCP2ktF {font-family:"trebuchet ms",verdana,arial,sans-…...

鱼香ros docker配置镜像报错:https://registry-1.docker.io/v2/

使用鱼香ros一件安装docker时的https://registry-1.docker.io/v2/问题 一键安装指令 wget http://fishros.com/install -O fishros && . fishros出现问题&#xff1a;docker pull 失败 网络不同&#xff0c;需要使用镜像源 按照如下步骤操作 sudo vi /etc/docker/dae…...

微信小程序云开发平台MySQL的连接方式

注&#xff1a;微信小程序云开发平台指的是腾讯云开发 先给结论&#xff1a;微信小程序云开发平台的MySQL&#xff0c;无法通过获取数据库连接信息的方式进行连接&#xff0c;连接只能通过云开发的SDK连接&#xff0c;具体要参考官方文档&#xff1a; 为什么&#xff1f; 因为…...

Python ROS2【机器人中间件框架】 简介

销量过万TEEIS德国护膝夏天用薄款 优惠券冠生园 百花蜂蜜428g 挤压瓶纯蜂蜜巨奇严选 鞋子除臭剂360ml 多芬身体磨砂膏280g健70%-75%酒精消毒棉片湿巾1418cm 80片/袋3袋大包清洁食品用消毒 优惠券AIMORNY52朵红玫瑰永生香皂花同城配送非鲜花七夕情人节生日礼物送女友 热卖妙洁棉…...

BLEU评分:机器翻译质量评估的黄金标准

BLEU评分&#xff1a;机器翻译质量评估的黄金标准 1. 引言 在自然语言处理(NLP)领域&#xff0c;衡量一个机器翻译模型的性能至关重要。BLEU (Bilingual Evaluation Understudy) 作为一种自动化评估指标&#xff0c;自2002年由IBM的Kishore Papineni等人提出以来&#xff0c;…...

人工智能--安全大模型训练计划:基于Fine-tuning + LLM Agent

安全大模型训练计划&#xff1a;基于Fine-tuning LLM Agent 1. 构建高质量安全数据集 目标&#xff1a;为安全大模型创建高质量、去偏、符合伦理的训练数据集&#xff0c;涵盖安全相关任务&#xff08;如有害内容检测、隐私保护、道德推理等&#xff09;。 1.1 数据收集 描…...

软件工程 期末复习

瀑布模型&#xff1a;计划 螺旋模型&#xff1a;风险低 原型模型: 用户反馈 喷泉模型:代码复用 高内聚 低耦合&#xff1a;模块内部功能紧密 模块之间依赖程度小 高内聚&#xff1a;指的是一个模块内部的功能应该紧密相关。换句话说&#xff0c;一个模块应当只实现单一的功能…...

算法打卡第18天

从中序与后序遍历序列构造二叉树 (力扣106题) 给定两个整数数组 inorder 和 postorder &#xff0c;其中 inorder 是二叉树的中序遍历&#xff0c; postorder 是同一棵树的后序遍历&#xff0c;请你构造并返回这颗 二叉树 。 示例 1: 输入&#xff1a;inorder [9,3,15,20,7…...

OCR MLLM Evaluation

为什么需要评测体系&#xff1f;——背景与矛盾 ​​ 能干的事&#xff1a;​​ 看清楚发票、身份证上的字&#xff08;准确率>90%&#xff09;&#xff0c;速度飞快&#xff08;眨眼间完成&#xff09;。​​干不了的事&#xff1a;​​ 碰到复杂表格&#xff08;合并单元…...

基于 HTTP 的单向流式通信协议SSE详解

SSE&#xff08;Server-Sent Events&#xff09;详解 &#x1f9e0; 什么是 SSE&#xff1f; SSE&#xff08;Server-Sent Events&#xff09; 是 HTML5 标准中定义的一种通信机制&#xff0c;它允许服务器主动将事件推送给客户端&#xff08;浏览器&#xff09;。与传统的 H…...