ARM 汇编写启动代码之设置栈和调用C语言

news2025/1/13 3:13:52

一、C语言运行时需要和栈的意义

“C语言运行时(runtime)”需要一定的条件,这些条件由汇编来提供。C语言运行时主要是需要栈。

C语言与栈的关系:C语言中的局部变量都是用栈来实现的。如果我们汇编部分没有给 C 部分预先设置合理合法的栈地址,那么 C 代码中定义的局部变量就会落空,整个程序就死掉了。

我们平时在编写单片机程序(譬如 51 单片机)或者编写应用程序时并没有去设置栈,但是 C 程序还是可以运行的。原因是:在单片机中由硬件初始化时提供了一个默认可用的栈,在应用程序中我们编写的 C 程序其实并不是全部,编译器(gcc)在链接的时候会帮我们自动添加一个头,这个头就是一段引导我们的 C 程序能够执行的一段汇编实现的代码,这个代码中就帮我们的 C 程序设置了栈及其他的运行时需要。


二、CPU模式和各种模式下的栈

在ARM中 37 个寄存器中,每种模式下都有自己的独立的 SP 寄存器(r13),为什么这么设计?

如果各种模式都使用同一个 SP,那么就意味着整个程序(操作系统内核程序、用户自己编写的应用程序)都是用一个栈的。你的应用程序如果一旦出错(譬如栈溢出),就会连累操作系统的栈也损坏,整个操作系统的程序就会崩溃。这样的操作系统设计是非常脆弱的,不合理的。

解决方案就是各种模式下用不同的栈。我的操作系统内核使用自己的栈,每个应用程序也使用自己独立的栈,这样各是各的,一个损坏不会连累其他人。

我们现在要设置栈,不可能也懒的而且也没有必要去设置所有的栈,我们先要找到自己的模式,然后设置自己的模式下的栈到合理合法的位置,即可。

注意:系统在复位后默认是进入SVC模式的。
我们如何访问SVC模式下的SP呢?很简单,先把模式设置为SVC,再直接操作SP。
但是因为我们复位后就已经是SVC模式了,所以直接设置SP即可。

三、查阅文档并设置栈指针至合法位置

栈必须是当前一段可用的内存(可用的意思是这个地方必须有被初始化过可以访问的内存,而且这个内存只会被我们用作栈,不会被其他程序征用)。

当前 CPU 刚复位(刚启动),外部的 DRRAM 尚未初始化,目前可用的内存只有内部的 SRAM(因为它不需初始化即可使用)。因此我们只能在 SRAM 中找一段内存来作为 SVC 的栈。

栈有四种:满减栈 满增栈 空减栈 空增栈
满栈:进栈:先移动指针再存; 出栈:先出数据再移动指针
空栈:xxx
减栈:进栈:指针向下移动;	 出栈:指针向上移动
增栈:xxx
在 ARM 中,ATPCS(ARM 关于程序应该怎么实现的一个规范)要求使用满减栈,所以不出意外都是用满减栈
结合 iROM_application_note 中的 memory map ,可知 SVC 栈应该设置为 0xd0037D80。

在这里插入图片描述


四、汇编程序和C程序互相调用

bl cfuncion

1. C 函数的编写和被汇编调用

在工程中新建并且添加一个C语言源文件(led.c),注意添加时要修改 Makefile。
在汇编启动代码中设置好栈后,使用 bl xxx 的方式来调用 C 中的函数 xxx。

2. 使用 C 语言来访问寄存器的语法

寄存器的地址类似于内存地址(IO与内存统一编址的),所以这里的问题是用C语言读写寄存器,就是用C语言
来读写内存地址。用C语言来访问内存,就要用到指针:
unsigned int *p = (unsigned int *)0x0xE0200240;
*p = 0x11111111;

上面这两句其实可以简化为1句:*((unsigned int *)0x0xE0200240) = 0x11111111;

3. 源代码

root@ubuntu:/home/aston/workspace/git_xxx# ls
led.c  Makefile  mkgcc.sh  mkv210_image.c  readme.txt  start.S  write2sd  说明.txt
root@ubuntu:/home/aston/workspace/git_xxx# cat Makefile 
led.bin: start.o led.o
	arm-linux-ld -Ttext 0x0 -o led.elf $^
	arm-linux-objcopy -O binary led.elf led.bin
	arm-linux-objdump -D led.elf > led_elf.dis
	gcc mkv210_image.c -o mkx210
	./mkx210 led.bin 210.bin
	
%.o : %.S
	arm-linux-gcc -o $@ $< -c -nostdlib

%.o : %.c
	arm-linux-gcc -o $@ $< -c -nostdlib

clean:
	rm *.o *.elf *.bin *.dis mkx210 -f

	
	
root@ubuntu:/home/aston/workspace/git_xxx# cat start.S 
/*
 * 文件名: led.S
 * 作者: xxx
 * 描述: 演示汇编设置栈并且调用 C 语言程序来点亮 LED 
 */


#define WTCON       0xE2700000

#define SVC_STACK   0xD0037D80  //满减栈

.global _start  //解决 make 编译警告: arm-linux-ld: warning: cannot find entry symbol _start; defaulting to 00000000
                // 把 _start 链接属性改为外部,这样其他文件就可以看见 _start 了
_start:
	//第 1 步,关看门狗(向 WTCON 的 bit5 写入 0 即可)
	ldr r0, =WTCON
	ldr r1, =0x0
	str r1, [r0]

	
	//第 2 步, 设置 SVC 栈(复位之后, 自动进入 SVC 模式)
	ldr sp, =SVC_STACK

	//从这里之后, 就可以开始调用 C 程序了
    bl  led_blink

//汇编最后的这个死循环不能丢
	b   .
	

root@ubuntu:/home/aston/workspace/git_xxx# cat led.c 
#define GPJ0CON     0xE0200240
#define GPJ0DAT     0xE0200244 

#define rGPJ0CON    *((volatile unsigned int*)GPJ0CON)
#define rGPJ0DAT    *((volatile unsigned int*)GPJ0DAT) 

void delay(void);

//该函数要实现 led 闪烁效果
void led_blink(void)
{
    //led 初始化,也就是把 GPJ0CON 中设置为输出模式
    //unsigned int* p   =  (unsigned int*)GPJ0CON;
    //unsigned int* p1  =  (unsigned int*)GPJ0DAT;

    //*p = 0x11111111;
    rGPJ0CON = 0x11111111;

    while (1)
    {
	rGPJ0DAT = (0 << 3) | (0 << 4) | (0 << 5);
	delay();

	rGPJ0DAT = (1 << 3) | (1 << 4) | (1 << 5);
	delay();
    }
}

void delay(void)
{
    volatile unsigned int i = 1000000;  // volatile 让编译器不要优化,这样才能真正的减
    while (i--);                        //才能消耗时间, 实现 delay
}
root@ubuntu:/home/aston/workspace/git_xxx# make
arm-linux-gcc -o start.o start.S -c -nostdlib
arm-linux-gcc -o led.o led.c -c -nostdlib
arm-linux-ld -Ttext 0x0 -o led.elf start.o led.o
arm-linux-objcopy -O binary led.elf led.bin
arm-linux-objdump -D led.elf > led_elf.dis
gcc mkv210_image.c -o mkx210
./mkx210 led.bin 210.bin
root@ubuntu:/home/aston/workspace/git_xxx# 

4. 神奇的 volatile

volatile 的作用是让程序在编译时,编译器不对程序做优化。优化有时候是 ok 的,但是有时候是自作聪明会造成
程序不对。如果你的一个变量是易变的,不希望编译器帮我们做优化,就在这个变量定义时加 volatile。

加不加有没有差别,取决于编译器。如果编译器做了优化则有差异;如果编译器本身没做优化,那就没有差别。

在我们这里(编译器是arm-2009q3),实际测试加不加效果是一样的。

5. 编译报错(实际上是连接阶段报错):undefined reference to `__aeabi_unwind_cpp_pr1’

解决:在编译时添加-nostdlib这个编译选项即可解决。nostdlib就是不使用标准函数库。标准函数库就是编译器中
自带的函数库,用-nostdlib可以让编译器链接器优先选择我程序内自己写的函数库。

源自朱有鹏老师.

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

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

相关文章

PAT甲级考试知识点总结和一些看法

0 引言 本人今年PAT甲级考了95分&#xff0c;平时力扣也有再刷&#xff08;大概有360题&#xff09;&#xff0c;感觉PAT主要还是面向考研党的&#xff0c;里面的题目其实难度是小于力扣的&#xff0c;但这种难度的题目浙大去年考研机试居然有20%的0分我其实不是很理解。 PAT…

【计算机网络】计算机网络复习总结 ------ 物理层

计算机网络 内容管理物理层 physical layer相关概念术语信息数据 data信号 signal码元 code cell 【波特率B --- 信号v】比特率R ---- 数据v基带信号 baseband带通&#xff08;频带&#xff09;信号单工 simplex 半双工 全双工失真理想信道奈奎斯特定理 &#xff08;理想&#…

[附源码]Python计算机毕业设计Django求职招聘网站

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

室内温度控制仿真(Simulink+PLC)

本篇博客将会和大家一起一步步解读Simulink自带的仿真模型(Thermal Model of a House),之后再讨论PLC控制系统控制室内环境温度的一些经验方法。温度控制的大部分控制方法都是采用PID控制,有关PLC的PID控制相关内容可以参看专栏的其它文章,链接如下: 博途PLC 1200/1500P…

CN_计算机网络性能指标@信道利用率@信道吞吐率

文章目录性能指标带宽(Bandwidth)&#x1f388;时延(Dely)发送时延&#x1f388;传播时延处理时延排队时延时延带宽积往返时延(Round-Trip Time,RTT)吞吐量(Throughput)速率(Speed)带宽(Bandwidth)信道利用率补充利用率信道利用率发送周期发送时间(传输时间)信道利用率计算&…

(附源码)springboot《升学日》日本大学信息及院校推荐网站 毕业设计 251949

基于springboot《升学日》日本大学信息及院校推荐网站 摘 要 随着科学技术的飞速发展&#xff0c;各行各业都在努力与现代先进技术接轨&#xff0c;通过科技手段提高自身的优势&#xff1b;对于《升学日》日本大学信息及院校推荐网站当然也不能排除在外&#xff0c;随着网络…

面向对象中的继承

面向对象中的继承 封装 低耦合&#xff0c;高内聚 多态 重载&重写 重载 其实这是后台的知识&#xff0c;这么做的原因是&#xff1a;涉及到服务器的承压能力。减轻并发数 重写 子类重写父类中的方法 怎么理解面向对象&#xff1f; 一切皆对象……学院派的答法&#xff0c;尽…

Some App Tech Support 一些应用技术支持

Some App Tech Support 一些应用技术支持 Getting Support: mail: qiudi7323gmail.com or leave comment below. 获得支持&#xff1a; 邮件&#xff1a;qiudi7323gmail.com 或者在下面留下评论。

深入理解Nginx线程池【内附原理讲解以及源码分析】

文章目录&#x1f680;前言❓什么是并发编程⭐多进程和多线程并发编程的比较&#x1f34e;线程池⭐线程池组成⭐线程池的核心组件⭐源码分享⭐线程池关键结构体刨析⭐线程池关键函数刨析&#x1f330;总结&#x1f680;前言 因为前段时间项目需要所以阅读分析了Nginx线程池源码…

[每周一更]-(第23期):Docker 逻辑图及常用命令汇总

Docker是一种轻量级的虚拟化技术&#xff0c;同时是一个开源的应用容器运行环境搭建平台&#xff0c;可以让开发者以便捷方式打包应用到一个可移植的容器中&#xff0c;然后安装至任何运行Linux或Windows等系统的服务器上。相较于传统虚拟机&#xff0c;Docker容器提供轻量化的…

认识哈希表

作者&#xff1a;~小明学编程 文章专栏&#xff1a;Java数据结构 格言&#xff1a;目之所及皆为回忆&#xff0c;心之所想皆为过往 目录 为什么我们需要哈希表&#xff1f; 哈希表的原理 什么是哈希值 冲突 负载因子 解决冲突 闭散列 开散列/哈希桶 代码实现 不考虑…

我们为什么需要调用InitCommonControls?

很多第一次使用外壳通用控件 (Shell common controls) 的新手碰到的问题之一是&#xff1a;他们经常忘记调用 InitCommonControls 。 但是如果你有机会查看这个函数的反汇编代码&#xff0c;则你会发现&#xff1a;这个函数实际上不做任何事情&#xff0c;就像另外一个函数 Flu…

python--pip常用命令、国内PyPI镜像、使用pip安装第三方库

让我们来看看具体内容&#xff1a; 一. pip常用命令 列出已安装的包&#xff1a; pip freeze or pip list 导出requirements.txt&#xff1a; pip freeze ><目录>/requirements.txt 在线安装包&#xff08;模块库&#xff09;&#xff1a; pip install <包名>…

论文解读:为了数据中心的未来,存算这对CP竟然又离了

古语有云&#xff1a;天下大势分久必合、合久必分。 同样&#xff0c;在数据中心多年的发展历程中&#xff0c;计算与存储也经历了多次分分合合。从大型机的计算与存储紧耦合&#xff0c;到小型机经典的IOE存算分离架构&#xff0c;再到随云兴起的超融合让存算再次融合&#x…

达梦数据库的名词解释

达梦数据库的名词解释 C/S、客户端、服务器、物理结构、逻辑结构、文件系统、文件、数据库、数据库实例、表空间、表、段、簇、页、用户、模式、角色、 一、数据库的组成 客户端连接服务器&#xff0c;通过数据库实例&#xff08;共享内存后台进程及线程&#xff09;将磁盘内…

关于修复预制体上UnityEngine.UI引用丢失的一种思路

在开发项目过程中&#xff0c;如果出现了Unity版本变化&#xff0c;有可能会导致一些预制体上的UI组件丢失&#xff0c;特别是大量UI脚本&#xff0c;明明一看就知道这个是Text组件&#xff0c;但是一个大大的missing出现在预制体上&#xff0c;让人产生了莫名的恐慌。 一、根…

头歌计算机组成原理MIPS寄存器文件设计

全部答案点击底部 <?xml version"1.0" encoding"UTF-8" standalone"no"?> <project source"2.15.0.2.exe" version"1.0"> This file is intended to be loaded by Logisim http://logisim.altervista.org &…

Linux | 二级页表的虚拟地址是怎么转换的?

文章目录页的概念可执行文件的虚拟地址二级页表的转换二级页表的优点页的概念 在聊文件系统时&#xff0c;我提到操作系统是以块为基本单位进行IO的&#xff0c;一个块的大小为4KB&#xff0c;在文件系统中它的名字叫做块&#xff0c;在内存系统中它的名字叫做页&#xff0c;p…

并发编程十 定时任务定时线程池

一 ScheduledThreadPoolExecutor 定时线程池类的类结构图 它接收SchduledFutureTask类型的任务&#xff0c;是线程池调度任务的最小单位&#xff0c;有三种提交任务的方式&#xff1a; schedulescheduledAtFixedRatescheduledWithFixedDelay 它采用DelayQueue存储等待的任务…

带你玩转序列模型之Bleu得分注意力模型语音识别

目录 一.Bleu得分 二.注意力模型直观理解 三.注意力模型 四.语音识别 五.触发字检测 一.Bleu得分 先跳过&#xff0c;等回头用得到了再来补。 二.注意力模型直观理解 在本周大部分时间中&#xff0c;你都在使用这个编码解码的构架&#xff08;a Encoder-Decoder archit…