裸机开发之汇编、寄存器

一、什么是汇编?为什么学汇编?

        在之前写控制代码的时候就在想:底层是怎么控制的?后来经过学习知道之前所编写的代码都是应用层代码,顾名思义就是在系统写好的底层之上调用系统函数。原以为底层是指写系统写好的底层函数,类似于写linux系统函数read、open、close之类的函数。但是再往下挖:read、open这类函数的下一层是什么?经过系统性查资料,总结了下面的流程:

         从上面的图可以知晓:linux系统所提供的IO函数属于应用层,并未涉及到最底层。open、read等函数通过file_operation结构体去调用驱动程序中的自定义的接口函数。所说的写驱动就是写自定义的device_open()、device_close()、device_write()、device_read()、device_ioctl()函数这些名字随便起。

1.1 那么这些自己编写的驱动函数是如何写入硬件设备中?

        众所周知机器所能识别的只有0和1二进制码,如果想要写入硬件设备,就要转换成二进制码写给机械。那么就又引入一个问题:

我们所写的C语言代码,怎么转换成二进制码?转换成机械识别的指令?

      提一下C语言编译流程:.c---->.i----->.s------>.o-------->二进制

 1.2 由此引入汇编指令:

        汇编语言是直接与机器打交道的,机器只能读懂二进制码,在不同的设备中,汇编语言对应着不同的机器语言指令集,通过汇编过程转换成机器指令。汇编语言是面向机器的,处于整个计算机语言层次结构的底层,故被视为一种低级语言。从上面的图可知,预处理部分是将程序中所引用的文件打开,汇编部分是将C语言转换成可被机器识别的代码,针对CPU中的寄存器进行操作。

        汇编是一种语言,和C语言一样,有自己的编写方式,不过指令个人感觉有点难懂。

知道汇编是什么了,但还是没有解决问题:使用汇编写入硬件设备,但是写进硬件设备的哪里去了?至此引入最重要的一块:寄存器。

 二、寄存器

2.1 为什么要用寄存器?

        一块成熟的板子笼统分两部分:CPU和内存,使用流程是从CPU从内存中搬出来数据使用(大致如此)。

        CPU 本身只负责运算,不负责储存数据。数据一般都储存在内存之中,CPU 要用的时候就去内存读写数据。但是,CPU 的运算速度远高于内存的读写速度,为了避免被拖慢,CPU 都自带一级缓存和二级缓存。(CPU 缓存可以看作是读写速度较快的内存。)

        CPU缓存还不够快,并且数据在缓存的地址不固定,导致CPU每次读写都会拖慢速度。因此,CPU还自带寄存器:用来储存最常用的数据。也就是说,那些最频繁读写的数据(比如循环变量),都会放在寄存器里面,CPU 优先读写寄存器,再由寄存器跟内存交换数据。

        寄存器不依靠地址区分数据,而依靠名称。每一个寄存器都有自己的名称,我们告诉 CPU 去具体的哪一个寄存器拿数据,这样的速度是最快的。有人比喻寄存器是 CPU 的零级缓存。

        总结一下就是寄存器是CPU用来存储频繁读写的数据(变量之类的)

        比如硬件有个固定地址0x300000,那么想要往这个地址里面写东西的时候,就需要将地址先暂存在寄存中使用。

下面说一下寄存器种类

2.2 寄存器种类

参考文档:ARM寄存器组织(常见专用寄存器、控制寄存器CPSR)_arm cpsr-CSDN博客

        寄存器分为三种:通用寄存器、专用寄存器(有特定的用途和功能,只能存放指定内容的寄存器。常见的有sp、lr、pc寄存器)、控制寄存器CPSR

2.2.1 通用寄存器

没有指定用途,可以存放任意内容的寄存器,既可以存放地址,也可以存放参与运算的数据或者运算产生的结果。在上图中:除去FIQ模式下,r0-r12是通用寄存器。

2.2.2 栈指针寄存器 r13 SP

栈指针,用于存储当前模式下的栈顶地址。假设现在CPU要将运算结果保存到栈上,这个时候SP寄存器就会告诉CPU栈顶的位置在哪,保存完毕以后,SP指向的地址会更新。

2.2.3 链接寄存器 r14 LR

链接寄存器,也是保存指令地址,一般是发生跳转的时候,事先保存跳转指令下一条指令的地址,一般有两种用途:

1、调用函数发生跳转

        假设main函数在执行程序的时候,需要调用函数func(),这个时候会跳转到func() 函数的定义。由于执行完func()函数以后还要继续运行main函数,在跳转之前LR寄存器会保存func()函数下一条指令的地址,也就是printf函数的地址。在执行完func函数以后,只需要让PC = LR 就可以回到func()函数下一条指令的地址。

2、异常发生(产生中断)

        产生异常时,异常模式下的LR会自动保存被异常打断的指令的下一条指令的地址。(也可以理解成是一种函数跳转)。比如CPU正在忙手里的任务,突然收到了网卡发来的信号,CPU就会进入FIQ或者IRQ模式,此时CPU就会停下手里的任务,转而先去执行异常处理程序。异常处理结束后将LR的值复制到PC可实现程序返回。

        没有哪个寄存器可以检测异常产生,因为能检测的就不是异常产生了。产生异常的时候LR默认自动保存下一条指令的地址,处理完异常情况再返回。

        异常情况时要在PC特定位置执行相应错误处理,由于偏移量(地址)是固定死的。所以当发生异常情况时,会自动跳转到该异常情况的地址执行处理异常,但PC寄存器是一条指令4字节并且是按照下面的顺序存放的(系统要求给的,只能这么写)。无法在此处处理异常,因为处理异常肯定不止一条指令,所以这里指的都是b跳转到相应标签位置处理异常

 异常的8种情况,这个需要在汇编启动代码中写出来,否则无法正常运行程序:

偏移量(地址)

异常情况

注释

0x1C

FIQ

快中断

0x18

IRQ

慢中断

0x14

(Reserved)

保留异常,现在没有用

0x10

Data Abort

数据访问异常

访问合法空间的数据发现数据不存在,访问了非法的空间(越界)

0x0c

Prefetch Abort

指令预取异常, 对程序指令预取时产生的异常

0x08

Software Interrupt(SWI)

软中断

即User模式级别使用代码发出的中断

系统调用就是使用软中断从用户模式切换到特权模式的

0x04

Undefined Instruction

未定义异常

0x00

Reset

重启异常, 模式切换到Supervisor模式

  • 通常说的异常主要来自cpu内部,abort,reset等;而中断主要来自外设。

  • 从高到低的顺序依次为:Reset,Data Abort, FIQ, IRQ,Prefetch Abort,SWI(软中断),undefined Instruction 

    • 高优先级的异常会终止底先级的异常

上面的异常向量表在代码中的体现

2.2.4 程序指针寄存器 r15 PC

        程序计数器,用于存储当前取址指令的地址。我们写的程序在经过预处理、编译、汇编以后,得到的二进制机器码就是指令。这些指令是被保存在内存中的,CPU接下来要执行哪一条指令都是由PC控制的。

        我们写程序时的逻辑是顺序执行,那么CPU在执行指令的时候也是如此,在ARM状态下,每一条指令都占4个字节,所以每执行完一条指令,PC的值会自动自增4个字节(地址自增4字节),为下一次取指令做准备

2.2.5 控制寄存器 CPSR

CPSR很重要,该寄存器里面的每个位都代表了各自含义,下图有一些说明

 举个例子:比如在进行加法运算的时候,CPSR的C位(进位)会变化

 adc:要计算十进制 3389+443 ,但是只会计算和记录两位数,于是先计算了 89+43=132 ,但是你只能记住两位数,所以记下了 32,然后把进位出的1扔进了CF 暂存;然后再计算 33+4=37 ,顺便加上 CF 的进位得到 38,跟刚才的 32 组合起来得到结果3832

 相对值得注意的是工作模式,由于板子在运行程序的时候肯定会遇到各种情况,那么其应对各种情况也是有不同的工作模式。ARM总共有7个基本工作模式:

那么体现在CPSR中MODE位,不同的编码值对应不同的模式

2.3 异常处理流程

 上面说完寄存器的种类后,那么他工作的流程顺序是如何的?

在没有异常的时候,也是按照C语言执行顺序一样,按顺序执行,遇到特殊标记跳转。

在有异常的时候,那么就需要对一些寄存器进行配置,流程如下图:

三、总结

       简单了解完寄存器之后,那么可以解决第一个问题:为什么使用汇编?

  1. 汇编语言的大部分语句直接对应着机器指令,执行速度快,效率高,代码体积小

  2. 在系统程序的核心部分,以及与系统硬件频繁打交道的部分,可以使用汇编语言。比如操作系统的核心程序段。

缺点:

  1. 不同的处理器有不同的汇编语言语法和编译器,编译的程序无法在不同的处理器上执行,缺乏可移植性

  2. 难懂且工作量大

        C语言与汇编是可以交叉使用的,C语言的好处是易懂,但是要相对寄存器直接操作是很复杂的,不如汇编直接了当,毕竟汇编指令是直接操作寄存器的,可以简单了解一些汇编指令。

        自己总结的一个图,简单一点,可能具体情况也会更复杂。

        写的文章可能还会存在缺陷,我自己是这么理解的,参考了其他博主一些文章,文章内容觉得写得好就引用了一些,勿怪勿怪。关于汇编和寄存器,弄懂就好

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

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

相关文章

VRRP(虚拟路由冗余协议)详解

VRRP-------虚拟路由冗余协议 在一个网络中,要做为一个合格的网络首先就要具备几种冗余,增加网络的可靠性。 这几种冗余分别为:线路冗余,设备冗余,网关冗余,UPS冗余 VRRP该协议就是解决网关冗余的。在二层…

面经:MapReduce编程模型与优化策略详解

作为一名专注于大数据处理与分布式计算的博主,我深知MapReduce作为一款经典的分布式计算框架,在海量数据处理领域所起的关键作用。本篇博客将结合我个人的面试经历,深入剖析MapReduce编程模型与优化策略,分享面试必备知识点&#…

Vue 移动端(H5)项目怎么实现页面缓存(即列表页面进入详情返回后列表页面缓存且还原页面滚动条位置)keep-alive缓存及清除keep-alive缓存

一、需求 产品要求:Vue移动端项目进入列表页,列表页需要刷新,而从详情页返回列表页,列表页则需要缓存并且还原页面滚动条位置 二、实现思路 1、使用Vue中的keep-alive组件,keep-alive提供了路由缓存功能 2、因为我项…

自然语言处理NLP:文本预处理Text Pre-Processing

大家好,自然语言处理(NLP)是计算机科学领域与人工智能领域中的一个重要方向,其研究能实现人与计算机之间用自然语言进行有效通信的各种理论和方法。本文将介绍文本预处理的本质、原理、应用等内容,助力自然语言处理和模型的生成使用。 1.文本…

MMU映射

MMU功能: 将虚拟地址转换物理地址 提供页属性,地址保护

K8S之Controller

我们在回顾下pod的启动流程: 用户通过kubectl,向api-server 发起请求api-server接受请求,并将数据写入etcdkube-scheduler通过watch检测到未绑定node 的pod,调度pod到某一node上,并通知给api-server,api-se…

Linux高级IO——多路转接之epoll

本章代码Gitee地址&#xff1a;EpollServer 文章目录 1. epoll接口1.1 epoll_create1.2 epoll_wait1.3 epoll_ctl 2. epoll原理3. epoll_server4. epoll两种工作模式 1. epoll接口 1.1 epoll_create #include <sys/epoll.h> int epoll_create(int size);参数int size理…

微信小程序 uniapp+vue城市公交线路查询系统dtjl3

小程序Android端运行软件 微信开发者工具/hbuiderx uni-app框架&#xff1a;使用Vue.js开发跨平台应用的前端框架&#xff0c;编写一套代码&#xff0c;可编译到Android、小程序等平台。 前端&#xff1a;HTML5,CSS3 VUE 后端&#xff1a;java(springbootssm)/python(flaskdja…

2011年认证杯SPSSPRO杯数学建模B题(第一阶段)生物多样性的评估全过程文档及程序

2011年认证杯SPSSPRO杯数学建模 B题 生物多样性的评估 原题再现&#xff1a; 2010 年是联合国大会确定的国际生物多样性年。保护地球上的生物多样性已经越来越被人类社会所关注&#xff0c;相关的大规模科研和考察计划也层出不穷。为了更好地建立国际交流与专家间的合作&…

文献学习-32-新生儿皮质表面重建的条件时间注意网络

Conditional Temporal Attention Networks for Neonatal Cortical Surface Reconstruction Authors: Qiang Ma, Liu Li, Vanessa Kyriakopoulou, Joseph V. Hajnal, Emma C. Robinson, Bernhard Kainz, and Daniel Rueckert Source: MICCAI 2023 Abstract 皮层表面重建在模拟…

数学杂谈之四:学习数学的方法

数学杂谈之四&#xff1a;学习数学的方法 数学杂谈之一&#xff1a;数学的形态 https://blog.csdn.net/cnds123/article/details/137437208 数学杂谈之二&#xff1a;数学中的概念和理解 https://blog.csdn.net/cnds123/article/details/137500537 数学杂谈之三&#xff1a;…

UTONMOS元宇宙游戏特点

在元宇宙的世界里&#xff0c;游戏不再只是一种娱乐方式&#xff0c;而是一种全新的生活体验。UTONMOS元宇宙游戏带你穿越虚拟与现实的边界&#xff0c;开启一段前所未有的冒险之旅。 在这个充满无限可能的UTONMOS元宇宙游戏中&#xff0c;你将成为自己游戏世界的主角。可以自…

如何恢复未保存或删除/丢失的Word文件?

关闭 Word 应用程序而不保存文档&#xff1f;误删Word文档&#xff1f;许多用户会在不同的情况下丢失Word文档。如果不幸遇到此类问题&#xff0c;如何恢复已删除或未保存的 Word 文档&#xff1f;有一些方法可以恢复未保存/删除的文档。此外&#xff0c;您还可以使用Word文件恢…

程序员如何搞副业?——程序员的副业建议

目录 前言 一、个人项目开发 二、在线教育和培训 三、技术博客和内容创作 总结 前言 程序员不仅拥有将抽象概念转化为实际应用的能力&#xff0c;还通常具备强大的逻辑思维和问题解决能力。然而&#xff0c;许多程序员并不满足于仅仅在一家公司工作&#xff0c;他们渴望通…

ES6: promise对象与回调地狱

ES6&#xff1a; promise对象与回调地狱 一、回调地狱二、Promise概述三、Promise的组成四、用函数封装Promise读取文件操作 一、回调地狱 在js中大量使用回调函数进行异步操作&#xff0c;而异步操作什么时候返回结果是不可控的&#xff0c;所以希望一段程序按我们制定的顺序执…

自定义注解(一)——统一请求拦截

文章目录 一、为什么会用到自定义注解二、关键参数说明三、应用场景示例&#xff1a;统一token认证1. 背景2. 自定义Token注解3. AOP上定义切面方法4. 方法上应用5. 总结 一、为什么会用到自定义注解 自定义注解可以帮助我们更好地组织和管理代码&#xff0c;提高代码的可读性和…

IIS服务器更换即将过期的SSL证书

公司IIS服务器证书快要过期&#xff0c;替换证书的步骤&#xff1a; Winr输入mstsc命令&#xff0c;显示远程登录&#xff1b;输入服务器IP以及密码&#xff0c;进行远程登陆登陆IIS服务器&#xff0c;winr输入inetmgr命令显示IIS操控器&#xff1b;选择服务器证书--点击服务器…

Springboot引入swagger

讲在前面&#xff1a;在spring引入swagger时&#xff0c;由于使用的JDK、Spring、swagger 的版本不匹配&#xff0c;导致启动报错&#xff0c;一直存在版本依赖问题。所以在此声明清楚使用版本。JDK 1.8、Spring boot 2.6.13、 Swagger 2.9.2。 引入maven依赖 <dependency&…

神经射频脉冲术,破解疼痛之锁

一位十余年糖尿病病史的患者&#xff0c;右足开始出现疼痛和麻木的症状三个多月&#xff0c;给他的生活带来了极大的困扰。他曾在多家医院就诊&#xff0c;但治疗效果并不理想。直到他走进了北京精诚博爱医院&#xff0c;这里为他带来了希望和转机。 经过详细的检查&#xff0c…

VR紧急情况模拟|V R体验中心加盟|元宇宙文旅

通过VR技术实现紧急情况模拟&#xff0c;提升安全应急能力&#xff01; 简介&#xff1a;面对突发紧急情况&#xff0c;如火灾、地震、交通事故等&#xff0c;正确的反应和应对能够有效减少伤害和损失。为了提高人们在紧急情况下的应急能力&#xff0c;我们借助先进的虚拟现实…