开发一个RISC-V上的操作系统(二)—— 系统引导程序(Bootloader)

news2024/12/28 20:30:56

目录

文章传送门

一、什么是Bootloader

二、简单的启动程序

三、上板测试


文章传送门

开发一个RISC-V上的操作系统(一)—— 环境搭建_riscv开发环境_Patarw_Li的博客-CSDN博客

开发一个RISC-V上的操作系统(二)—— 系统引导程序(Bootloader)_Patarw_Li的博客-CSDN博客

​​​​​​​开发一个RISC-V上的操作系统(三)—— 串口驱动程序(UART)_Patarw_Li的博客-CSDN博客

一、什么是Bootloader

Bootloader是cpu在上电后执行的第一段代码,用于初始化各类资源,并且跳转到主程序上执行,比如初始化sp寄存器,将rom中的数据搬运到ram上,清零bss段等等。

百度百科的词条中,这样解释Bootloader:“Bootloader是嵌入式系统在加电后执行的第一段代码,在它完成CPU和相关硬件的初始化之后,再将操作系统映像或固化的嵌入式应用程序装载到内存中然后跳转到操作系统所在的空间,启动操作系统运行”。

一般系统引导程序都是固化在flash中(因为ram断电即失),上电后先执行引导程序再跳转到主程序上执行:

引导程序大多都是使用汇编语言编写(毕竟涉及到一些寄存器操作),下面我会写一个简单的启动程序来帮助我们初始化栈指针sp、并且跳转到主程序执行。

二、简单的启动程序

可以先去我的gitee仓库下载代码,本节代码在 00_START 目录下下:

riscv_os: 一个RISC-V上的简易操作系统

代码结构如下: 

 

inc目录下存放头文件;kernel.c为主程序,引导程序最终会跳转到这里执行;start.S为引导程序;Makefile为自动化构建脚本。

先来看看start.S里的内容:

#include "inc/platform.h"

        # size of stack is 256 bytes
        .equ    STACK_SIZE, 256
        .global _start

        .text
_start:
        la      sp, RAM + STACK_SIZE     # set the initial stack pointer to 0x00001100 (0x00001000 + 256)
        j       start_kernel             # jump to kernel

        .end                             # end of file
  • .equ类似于C语言里面的宏,将STACK_SIZE设置成256。
  • .global关键字用来让一个符号对链接器可见,可以供其他链接对象模块使用;告诉编译器后续跟的是一个全局可见的名字(变量/函数名)。
  • .text指定后续内容为代码段。
  • _start是一个符号,是汇编程序默认入口标号。也是编译、链接后程序的起始地址。 由于程序是通过加载器来加载的,必然要找到 _start名字的函数,因此 _start必须定义成全局的,以便存在于编译后的全局符号表中,供其他程序(如加载器)寻找到。
  • la  sp, RAM + STACK_SIZE 将栈指针寄存器sp的值初始化为RAM + STACK_SIZE(0x00001000 + 256)。
  •  j   start_kernel 跳转到start_kernal 主程序中执行。

为什么用大写的.S后缀而不用小写的.s呢?因为使用GCC(准确说是GCC调用了as汇编器)处理汇编代码时,汇编文件的后缀有两种:.s.S。这两种文件都是汇编代码,其区别在于:

.s格式的汇编文件中,只能包含纯粹的汇编代码,汇编器只对其进行汇编操作,没有预处理操作;
.S格式的汇编文件中,还可以使用预处理命令,汇编器会先进行预处理,然后再进行汇编。

而我们的启动代码包含了头文件,所以就需要用大写的.S结尾的汇编文件了。

然后是Makefile里面的内容:

CROSS_COMPILE = riscv64-unknown-elf-
CFLAGS = -nostdlib -fno-builtin -march=rv32im -mabi=ilp32 -g -Wall

CC = ${CROSS_COMPILE}gcc
OBJCOPY = ${CROSS_COMPILE}objcopy
OBJDUMP = ${CROSS_COMPILE}objdump

SRCS_ASM = \
           start.S \

SRCS_C = \
        kernel.c \

OBJS = $(SRCS_ASM:.S=.o)
OBJS += $(SRCS_C:.c=.o)

.DEFAULT_GOAL := all
all: os.elf

# start.o must be the first in dependency!
os.elf: ${OBJS}
        ${CC} ${CFLAGS} -o os.elf $^
        ${OBJCOPY} -O binary os.elf os.bin

%.o : %.c
        ${CC} ${CFLAGS} -c -o $@ $<

%.o : %.S
        ${CC} ${CFLAGS} -c -o $@ $<

.PHONY : code
code: all
        @${OBJDUMP} -S os.elf | less

clean:
        rm -fr *.o *.bin *.elf

该脚本的工作是先把start.S和kernel.c编译成start.o和kernel.o目标文件,然后再将start.o和kernel.o目标文件链接成os.elf文件,最后再通过objcopy将os.elf文件变成二进制os.bin文件,os.bin文件就是最后我们要放到板子上跑的程序。

可能有人会问为什么不直接把elf文件放到处理器上去运行,下面对elf格式的文件做一些简单的介绍:

下面是elf文件的格式,可以看到除了中间一部分正文段和数据段以外,还有一些其他的段,比如ELF Header,里面描述了体系结构和操作系统等基本信息,并指出Section Header Table和Program Header Table在文件中的什么位置;Program Header Table在汇编和链接过程中没有用到,所以是可有可无的,Section Header Table中保存了所有Section的描述信息。

但是cpu并不能识别这些信息,只有一些特定的操作系统才能识别这些信息,所以这些信息对处理器来说是没有用的,而objcopy指令正是帮我们去掉这些处理器无法识别的内容,留下的内容即为处理器可以识别的内容。

Makefile脚本的用法:

1. 生成二进制.bin文件,执行make即可:

make

生成的os.bin即为我们要烧录到板子上运行的程序。

2. 查看二进制文件的os.elf的汇编代码:

make code

使用这个指令可以查看每条C语句对应的汇编代码以及每条指令的地址。 

3. 清除所有生成的文件:

make clean

最后是kernel.c里面的内容,这里面即可存放我们要运行的内容,还是以我们的流水灯程序为例子:

void start_kernel(void){
        uint8_t *gpio_data = (uint8_t *)0x20000004;
        while(1){
                // 第一个灯亮起
                *gpio_data = 1;
                for(int i = 0; i < 1000000; i++); // delay

                // 第二个灯亮起
                *gpio_data = 2;
                for(int i = 0; i < 1000000; i++); // delay

                // 第三个灯亮起
                *gpio_data = 4;
                for(int i = 0; i < 1000000; i++); // delay

                // 第四个灯亮起
                *gpio_data = 8;
                for(int i = 0; i < 1000000; i++); // delay
        }

        while(1){}; // stop here!
}

这样引导程序和主程序都准备完毕了,我们接下来就可以上板实验了。

三、上板测试

要进行上板测试,首先得按照我前面的文章烧录riscv处理器程序到板子上:

RISC-V处理器的设计与实现(三)—— 上板验证(基于野火征途Pro开发板)_Patarw_Li的博客-CSDN博客

项目仓库地址:cpu_prj: 一个基于RISC-V指令集的CPU实现

然后执行make生成os.bin文件后,通过python串口发送程序(serial_utils目录下)将os.bin文件烧录到处理器的memory上(按住key1不动,烧录完后松开),烧录后即可看到流水灯现象。 

遇到问题欢迎加群 892873718 交流~

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

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

相关文章

SpringBoot中接口幂等性实现方案-自定义注解+Redis+拦截器实现防止订单重复提交

场景 SpringBootRedis自定义注解实现接口防刷(限制不同接口单位时间内最大请求次数)&#xff1a; SpringBootRedis自定义注解实现接口防刷(限制不同接口单位时间内最大请求次数)_redis防刷_霸道流氓气质的博客-CSDN博客 以下接口幂等性的实现方式与上面博客类似&#xff0c;…

实战攻防Demo|如何轻松形成自动响应的安全闭环?

从威胁阻断角度来说&#xff0c;拦住黑客的第一步攻击尤为重要。同样&#xff0c;对于攻击者来说&#xff0c;第一步攻击的成本也往往是最高的。日常工作中人们会遇到很多类型的攻击&#xff0c;但暴力破解或者撞库攻击往往被作为黑客的第一步攻击。这主要源于其技术含量低&…

express编写一个简单的get接口

/01编写get接口.jsconst express require(express) const app express()// 创建路由 const useRouter require(./router/user.js) // 注册路由 app.use(/api,useRouter)app.listen(8080, (req, res) > {console.log(8080监听) }) ./02编写post接口 // 注意&#xff1a;如…

ESP32(MicroPython) 两轮差速五自由度机械臂小车

这次的项目在软件上没多少调整&#xff0c;但本人希望分享一下硬件上的经验。 小车使用两轮差速底盘&#xff0c;驱动轮在小车中间&#xff0c;前后都要万向轮。这种形式可以实现0转弯半径&#xff0c;但受万向轮及用于加高的铜柱的规格限制&#xff0c;两个万向轮难以调到相同…

3ds MAX 洗菜池

在家居中我们显然离不开这个对吧 首先绘制一个长方体作为基础 注意设置长宽高的网格大小&#xff0c;方便后续调整 俯视图网格线如下&#xff1a; 长方形变换为可编辑网络&#xff0c;并在【多边形】界面选择底面的所有多边形&#xff0c;按delete删除&#xff0c;形成一个壳体…

AtcoderABC233场

A - 10yen StampA - 10yen Stamp 题目大意 Takahashi已有X日元邮票&#xff0c;计算出Takahashi至少需要再贴多少个10日元的邮票才能使得信封上的邮票总价值达到Y日元。 思路分析 我们只需要计算Y - X&#xff0c;即Y日元减去X日元的差值。如果该差值大于等于0&#xff0c;则…

MyBatisPlus分页功能实现

MyBatisPlus分页功能实现 1. MyBatisPlus分页使用1.1 设置分页拦截器作为Spring管理的bean1.2 执行分页查询 2. 开启MyBatisPlus日志3. 解决日志打印过多问题3.1 取消初始化spring日志打印3.2 取消SpringBoot启动banner图标3.3 取消MybatisPlus启动banner图标 1. MyBatisPlus分…

零信任安全解决方案

什么是零信任 零信任网络架构 &#xff08;ZTNA&#xff09; 或零信任安全是一种新的组织网络安全方法。它旨在修复传统基于边界的安全性中的缺陷并简化网络设计。 它以“永不信任&#xff0c;始终验证”的原则运作。这意味着&#xff0c;无论用户或设备位于何处&#xff0c;…

基于自适应运动补偿的双向运动估计算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 matlab2022a 3.部分核心程序 ..........................................................................% 单向运动法 …

繁體標楷體 如何安装使用?

中文繁體標楷體。 ——这个地方下载&#xff1a;http://www.downcc.com/font/316365.html ——下载下来被叫做“台湾标楷体DFKai-SB.ttf” ——安装到“C:\Windows\Fonts”以后名字叫 ——在WPS中使用&#xff0c;被称为这个

使用node内置test runner,和 Jest say 拜拜

参考 https://nodejs.org/dist/latest-v20.x/docs/api/test.html#test-runner 在之前&#xff0c;我们写单元测试&#xff0c;必须安装第三方依赖包&#xff0c;而从node 20.0.0 版本之后&#xff0c;可以告别繁琐的第三方依赖包啦&#xff0c;可直接使用node的内置test runner…

【搜索引擎Solr】Apache Solr 神经搜索

Sease[1] 与 Alessandro Benedetti&#xff08;Apache Lucene/Solr PMC 成员和提交者&#xff09;和 Elia Porciani&#xff08;Sease 研发软件工程师&#xff09;共同为开源社区贡献了 Apache Solr 中神经搜索的第一个里程碑。 它依赖于 Apache Lucene 实现 [2] 进行 K-最近邻…

龙芯iTOP-2K1000开发板制作启动U盘

我们准备一个 U 盘&#xff08;最小不要小于 4G&#xff0c;最大不要大于 32G&#xff09;&#xff0c;U 盘有且只有一个分区&#xff0c;U 盘格式化成 FAT32 分区&#xff0c;&#xff0c;如不满足要求&#xff0c;请格式化您的 U 盘&#xff0c;准备完成如下图所示 格式化软…

推荐系统(十)用户行为序列建模-Pooling 路线

对推荐系统而言&#xff0c;准确捕捉用户兴趣是其面临的核心命题。不管是样本、特征还是模型结构等方面的优化&#xff0c;本质上做的事情都是在提高推荐系统对用户兴趣的捕捉能力&#xff0c;因此如何提高这种能力&#xff0c;对推荐效果的提升有重要作用&#xff0c;也是算法…

性能优化问题

提升首屏的加载速度&#xff0c;是前端性能优化中「最重要」的环节&#xff0c;这里笔者梳理出一些 常规且有效 的首屏优化建议 1、路由懒加载 SPA 项目&#xff0c;一个路由对应一个页面&#xff0c;如果不做处理&#xff0c;项目打包后&#xff0c;会把所有页面打包成一个文…

使用lua脚本操作redis

redis中实现事务有两种方法&#xff1a; 1.WATCH监视键的变动&#xff0c;然后MULTI开始事务&#xff0c;EXEC提交事务 WATCH key [key…]&#xff1a;监视一个或多个键&#xff0c;如果在事务执行之前被修改&#xff0c;则事务被打断。 MULTI&#xff1a;标记一个事务的开始。…

Redis原理篇(二)

Redis原理 Redis数据结构 Redis网络模型 RESP协议 Redis内存回收 Redis原理篇 一、原理篇-Redis数据结构 1.1 Redis数据结构-动态字符串 我们都知道Redis中保存的Key是字符串&#xff0c;value往往是字符串或者字符串的集合。可见字符串是Redis中最常用的一种数据结构。 不…

JVM理论(六)执行引擎--垃圾回收

概述 垃圾: 指的是在运行程序中没有任何指针指向的对象垃圾回收目的: 为了及时清理空间使得程序可以正常运行垃圾回收机制: JVM采取的是自动内存管理,即JVM负责对象的创建以及回收,将程序员从繁重的内存管理释放出来,更加专注业务的开发垃圾回收区域: 频繁收集Young区(新生代)…

【前端知识】React 基础巩固(三十二)——Redux的三大原则、使用流程及实践

React 基础巩固(三十二)——Redux的三大原则 一、Redux的三大原则 单一数据源 整个应用程序的state被存储在一颗object tree 中&#xff0c;并且这个object tree 只存储在一个store中&#xff1b;Redux并没有强制让我们不能创建多个Store&#xff0c;但是那样做不利于数据维护…

Java网络编程(一)基本网络概念

一、网络 网络(network) 是几乎可以实时相互发送和接收数据的计算机和其他设备的集合。网络通常用线缆连接&#xff0c;数据位转换为电磁波&#xff0c;通过线缆移动。不过&#xff0c;无线网络会通过无线电波传输数据&#xff0c;许多长距离的传输现在会用通过玻璃纤维发送可见…