IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> 【嵌入式06】STM32的C与汇编语言混合编程 -> 正文阅读

[嵌入式]【嵌入式06】STM32的C与汇编语言混合编程


一、题目要求

修改代码,要求将原汇编语言 Init_1函数的类型改为 int Init_1(init) ,此函数功能修改为传入一个整型数x,函数运行后返回整型数 x+100。 请编程实现,并仿真跟踪调试;
如果要求在汇编函数中调用一个 C语言写的函数,应该如何修改汇编代码?

二、项目创建

打开Keil ARM,Project->New μVision Project->选择STM32F103ZE
在这里插入图片描述

添加main.cFunc.s文件
在这里插入图片描述

在这里插入图片描述
Func.s


	AREA	MY_FUNCTION,CODE,READONLY
	EXPORT 	Init_1  ; 与在c文件中定义的Init_1函数关联起来


; 高级语言中的声明和使用变量其实是对板子寄存器的使用,所以我们只需要直接使用寄存器即可

Init_1

	MOV R1,#0     ; 设R1寄存器为i
	MOV R2,#0	  ; 设R2寄存器为j
	
LOOP	; 写在最左边的是程序段的段名,执行跳转程序时用到
	CMP R1,#10	  ; 比较R1和10的大小
	BHS LOOP_END  	  ; 如果R1大于等于10,则跳转到LOOP_END程序段,反之忽略该语句,直接执行下面的语句
	ADD R2,#1	  ; j++
	ADD R1,#1     ; i++
	B LOOP		  ; 循环
	
LOOP_END
	NOP	
	
	END  ; 必须空格后再写END,不然会被认为是段名,表示程序结束

main.c

# include<stdio.h>

extern void	Init_1(void);

int main(){
	
	Init_1();
	
	return 0;
}

三、C语言调用汇编-无参数调用

1、仿真设置

打开魔法棒,选择Debug,勾选Use Simulator
将左侧的Dialog DLL中的内容改为DARMSTM.DLL,将Parameter的内容改为-pSTM32F103C8
在这里插入图片描述

2、设置断点

在目标行前侧点击,生成小红点,即完成断点设置
在这里插入图片描述
在这里插入图片描述

3、编译并调试

编译buildrebuild后,进行调试
在这里插入图片描述
在左上角点击stepstep overRun To Cursor Line
观察R1、R2的寄存器值的变化
可以发现其由0逐步加到10
在这里插入图片描述
在这里插入图片描述

四、C语言调用汇编语言-有参数调用

1、修改 三 中代码

Func.s

 
	AREA	MY_Function,CODE,READONLY
	EXPORT 	Init_1  ; 与在c文件中定义的Init_1函数关联起来


; 高级语言中的声明和使用变量其实是对板子寄存器的使用,所以我们只需要直接使用寄存器即可

Init_1
	ADD R0,#100     ; 将传入的值+100
	MOV PC,LR		; 返回R0
	
	
LOOP	; 写在最左边的是程序段的段名,执行跳转程序时用到
	CMP R1,#10	  ; 比较R1和10的大小
	BHS LOOP_END  	  ; 如果R1大于等于10,则跳转到LOOP_END程序段,反之忽略该语句,直接执行下面的语句
	ADD R2,#1	  ; j++
	ADD R1,#1     ; i++
	B LOOP		  ; 循环
	
LOOP_END
	NOP	
	
	
	END  ; 必须空格后再写END,不然会被认为是段名,表示程序结束

main.c

# include<stdio.h>

extern int Init_1(int x);

int main(){
	
	int xx = Init_1(10);
	printf("%d", xx);
	
	return 0;
}

在Keil中,子函数的参数值传递按顺序存放到了R0R1R2R3中,超过四个参数值传递放栈帧里。

由此,Init_1(10)传入的10放到了R0中,由MOV PC,LR返回110.

2、设置断点

在这里插入图片描述
在这里插入图片描述

3、编译并调试

操作同上
在这里插入图片描述
在这里插入图片描述
此时值为6E110,调用成功!

五、汇编语言调用C语言函数

1、修改 四 中代码

Func.s


	AREA	MY_Function,CODE,READONLY
	EXPORT 	Init_1  ; 与在c文件中定义的Init_1函数关联起来
	IMPORT  get5    ; 声明get5 为外部引用


; 高级语言中的声明和使用变量其实是对板子寄存器的使用,所以我们只需要直接使用寄存器即可

Init_1

	MOV R1,#0     ; 设R1寄存器为i
	MOV R2,#0	  ; 设R2寄存器为j
	
LOOP	; 写在最左边的是程序段的段名,执行跳转程序时用到
	CMP R1,#10	  ; 比较R1和10的大小
	BHS LOOP_END  	  ; 如果R1大于等于10,则跳转到LOOP_END程序段,反之忽略该语句,直接执行下面的语句
	ADD R2,#1	  ; j++
	ADD R1,#1     ; i++
	BL get5  	  ; 调用get5,返回的值传入R0
	B LOOP		  ; 循环
	
LOOP_END
	NOP	
	
	
	END  ; 必须空格后再写END,不然会被认为是段名,表示程序结束

main.c

# include<stdio.h>

extern void	Init_1(void);

int get5(void);

int main(){
	
	printf("Begin...\n");
	Init_1();

	
	return 0;
}

int get5(){
	return 5;
}


2、设置断点

在这里插入图片描述

3、编译并调试

执行get5后,R0变为了5,调用成功!

六、其他

1、C与汇编之间函数调用ATPCS简介

  • ARM-Thumb 过程调用标准 ATPCS(ARM-Thumb Procedure Call Standard)
  • ATPCS 标准既是ARM 编译器使用的函数调用规则,也是设计可被 C 程序调用的汇编函数的编写规则
  • ATPCS 规定,ARM 的数据堆栈为 FD 型堆栈,即满递减堆栈
  • 函数是通过寄存器和堆栈来传递参数和返回函数值的,形参和返回值都应定义在具有暂存性质的寄存器和堆栈中

2、参数如何传递

汇编程序调用C函数时,函数的入口参数使用栈来传送,参数的传递顺序是从右到左。即函数最后(最右边的)一个参数先入栈,而最左边的第一个参数最后入栈,然后执行 CALL 指令去调用C函数。

3、参数的清除

在C函数返回后,汇编程序需要把先前压入栈中的函数参数清除掉,即调用者负责清除参数占用的栈空间。

4、C语言与汇编语言混合程序

在C程序中内联或嵌入式汇编代码,以提高程序的效率

内联汇编

  • 在 C 程序中直接编写汇编程序段而形成一个语句块,这个语句块可以使用除了 BX 和 BLX之外的全部ARM指令来编写
  • 可以使程序实现一些不能从C获得的底层功能
  • 汇编语句块中,如果有两条指令占据了同一行,那么必须用分号“ ;”将它们分隔
  • 如果一条指令需要占用多行,那么必须用反斜线符号“ \ ”作为续行符
  • 可以在内联汇编语言块内的任意位置使用C/C++格式的注释
  • 内联汇编代码中定义的标号可被用作跳转或C/C++ goto 语句的目标,同样,在C/C++代码中定义的标号,也可被用作内联汇编代码跳转指令的目标
  • 在内联汇编语句块中最好使用 C 或 C++ 变量作为操作数
  • 状态寄存器 PSR ,任何对 PSR 的引用总是执行指向物理 PSR

void enable_IRQ(void)
{
int tmp;
_ _asm //声名内联汇编代码
{
MRS tmp, CPSR
BIC tmp, tmp, #0x80
MSR CPSR_c, tmp
}
}

  • 内联汇编的限制
    • 它不支持 Thumb 指令;除了程序状态寄存器 PSR 之外,不能直接访问其他任何物理寄存器
    • 如果在内联汇编程序指令中出现了以某个寄存器名称命名的操作数,那么它被叫做虚拟寄存器,而不是实际的物理寄存器。编译器在生成和优化代码的过程中,会给每个虚拟寄存器分配实际的物理寄存器,但这个物理寄存器可能与在指令中指定的不同。
    • 在内联汇编代码中不能使用寄存器 PC(R15)、LR(R14)和SP(R13)
    • 处理器模式会禁止使用 C 操作数或对已编译 C 代码的调用,直到将处理器模式恢复为原设置之后

嵌入式汇编

  • 嵌入式汇编程序是一个编写在C程序外的单独汇编程序,该程序段可以像函数那样被 C 程序调用
  • 嵌入式汇编具有真实汇编的所有特性,数据交换符合 ATPCS 标准,同时支持 ARM 和Thumb,所以它可以对目标处理器进行不受限制的低级访问
  • 不能直接引用 C/C++ 的变量
  • 参数名只允许使用在参数列表中,不能用在嵌入式汇编函数体内
  • 在 C 程序中调用嵌入式汇编程序的方法与调用 C 函数的方法相同

定义一个嵌入式汇编函数的语法格式为:
_ _asm return–type function–name(parameter-list)
{
汇编程序段
}

return–type:函数返回值类型,C语言中的数据类型
function–name:函数名
parameter-list:函数参数列表

项目内联汇编嵌入式汇编
指令集仅限于ARMARM和Thumb
ARM汇编程序命令不支持支持
C表达式支持仅支持常量表达式
优化代码支持不支持
内联可能从不
寄存器访问不使用物理寄存器使用物理寄存器
返回指令自动生成显式编写

七、总结

ARM在C语言中调用汇编函数的方法
在C中调用汇编文件中的函数,要做的主要工作有两个:
一是在C中声明函数原型,并加extern关键字;
二是在汇编中用EXPORT导出函数名,并用该函数名作为汇编代码段的标识,最后用mov pc, lr返回。然后,就可以在C中使用该函数了。


参考:
[1] https://blog.csdn.net/longintchar/article/details/79511747
[2] https://blog.csdn.net/sinat_27421407/article/details/78829508

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-10-13 11:36:15  更:2021-10-13 11:37:07 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/2 0:52:51-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码