RISC-V FreeRTOS启动过程分析(基于qemu+gdb调试)

目录

启动代码分析

RISC-V系统定时器初始化

任务创建以及栈帧初始化

启动第一个任务

使用GDB跟踪任务启动过程


        最近在学RISC-V架构,有幸找到了一个非常好的课程《循序渐进,学习开发一个RISC-V上的操作系统》,学完后受益匪浅,但是课程上开发的操作系统毕竟只是教学用,想对比学习一下实际商用的RTOS的实现。FreeRTOS以前也用过,是一个非常流行的开源RTOS,所以选择了qume+FreeRTOS分析学习相关代码。本文先分析RISC-V + qume + FreeRTOS的启动流程。

环境搭建可以参考

基于RISC-V的QEMU + FreeRTOS开发环境构建_qemu riscv_吹角连营G的博客-CSDN博客q

启动代码分析

启动文件入口FreeRTOS\Demo\RISC-V-Qemu-virt_GCC\start.S

        启动代码很简单,本人加了中文注释,上面的代码中51-61行,将data段从ROM加载到RAM中_data位置,打开编译生成的RTOSDemo.map文件,可以看到_data的地址为0x80080000(实际就是链接脚本指定的RAM的起始地址), _data_lma地址为0x8000b338。

以上_data和_data_lma都是链接脚本里定义的,链接脚本如下:

FreeRTOS\Demo\RISC-V-Qemu-virt_GCC\fake_rom.lds

        这里和Linux有点不一样,Linux会把代码段数据段等...,整个内核都加载到内存中(一般为DDR)。RTOS一般用在单片机上,单片机一般没有DDR,程序直接片内Flash和片内SRAM运行。代码段、只读数据段不用从Flash加载到RAM,直接在片内Flash上(对应着上面的rom)取址执行,只把可读写数据段加载到内存,所以上面的代码只加载的.date(读写数据段),没有加载代码段和.rodata(只读数据段)。

        对于_data和_data_lma的值,我们可以使用命令riscv64-unknown-elf-objdump -S -d build/RTOSDemo.axf > RTOSDemo.dsi,将编译后的RTOSDemo.axf 反汇编到RTOSDemo.dsi。使用gdb单步调试,对照反汇编文件,停在start.s 55行处,打印a0和a1寄存器的值也是8000b338和80080000。

        加载完date后清BSS,之后跳转到main函数了(实在有点快了,连异常入口都没有配置)

FreeRTOS\FreeRTOS\Demo\RISC-V-Qemu-virt_GCC\main.c

main函数中,先配置异常入口寄存器mtvec,mainVECTOR_MODE_DIRECT=0没有启用向量中断。所有异常和中断的入口地址都为freertos_risc_v_trap_handler。

关于RISC-V mtvec寄存器,功能类似armv8的VBAR_Elx寄存器,多了MODE向量中断配置:

接下来的main_blinky()函数,创建两个任务,之后调用vTaskStartScheduler(),系统就进入了Task的世界,vTaskStartScheduler函数不会退出。

vTaskStartScheduler中创建了idleTask(最低优先级的任务,没有其它活干时,总的有一个任务在运行,就轮到idleTask了,可以在其中做省电或者睡眠等处理),之后调用xTimerCreateTimerTask创建了软件定时器任务。函数最后调用xPortStartScheduler(和架构相关的调度器启动代码)。

xPortStartScheduler函数在FreeRTOS\Source\portable\GCC\RISC-V\port.c文件中

xPortStartScheduler中通过vPortSetupTimerInterrupt启动了系统定时器,之后配置mie(Machine Interrupt Enable)寄存器,最后xPortStartFirstTask启动了第一个任务,之后就进入Task世界了。

RISC-V系统定时器初始化

RISC-V系统定时器简单介绍下:

定时器初始化代码如下:

FreeRTOS\Source\portable\GCC\RISC-V\port.c

上面代码中加了详细注释,这里就不在啰嗦了。

任务创建以及栈帧初始化

        之前代码创建任务函数没有展开,直接跳过了。现在启动第一个任务需要涉及到创建任务的代码,这里对创建任务过程简单分析一下:

        创建任务时,创建一个任务控制块pxNewTCB(tskTaskControlBlock类型),任务控制块的第一个成员变量pxTopOfStack,保存着该任务的栈顶指针。创建任务过程中调用,pxPortInitialiseStack为任务初始化了任务栈。

pxPortInitialiseStack是一个汇编函数

FreeRTOS\Source\portable\GCC\RISC-V\portASM.S

        函数第一个参数pxTopOfStack是task栈顶指针对应a0寄存器,第二个参数pxCode为Task的入口函数对应a1寄存器,第三个参数pvParameters为pxCode入口函数的参数。该函数注释写的很明白,pxPortInitialiseStack就是为任务创建栈帧,并且返回新的栈顶指针,而且压栈顺序注释上也列出来了:先入mstatus -> xCriticalNesting -> x31 -> x30 ->x29 .... -> [chip specific registers go here] -> pxCode,其中[chip specific registers go here],Qume没有specific registers需要保存,portasmADDITIONAL_CONTEXT_SIZE=0,代码中chip_specific_stack_frame跳过了。

        这里有特别说明一下,pxCode最后入栈也就是放在栈顶(sp偏移0位置),pvParameters放在x11和x9之前,直白一点就是应该放x10的位置,对照RISC-V寄存器表,x10就是a0,a0-a7根据RISC-V的ABI规范是用来传递参数的,a0是函数第一个参数,同时也做函数返回值。pvParameters是任务入口函数的参数,所以放在a0应该放的位置;pxCode实际上也是对应的放在了x1也就是ra寄存器的位置。这里没有保存x2、x3、x4,x2是sp放在TCB(任务控制块)的pxTopOfStack中(任务栈顶指针);x3是gp指针在start.S中第一个初始化了,代码中不再修改,不保存;x4 tp线程指针,在Linux这种支持进程和线程的系统中才用到(可以搜索该项目的反汇编文件,没有使用x4或者tp,我们有需要可以自定义其它用途),所以也不保存。至于mstatus的值有详细注释,可以对照寄存器文档分析,这里略过。

启动第一个任务

        现在回到xPortStartFirstTask,该函数也在文件FreeRTOS\Source\portable\GCC\RISC-V\portASM.S中,而且就在pxPortInitialiseStack的下面,其实就这两个就是一对,一个先压栈,为任务准备栈帧;另一个出栈,使用栈帧为任务准备运行环境,并且最终启动任务(跳转到任务入口函数中执行)。

xPortStartFirstTask中首先从TCB中获取任务栈顶指针到sp,sp偏移0的位置保存着任务的入口函数,将任务入口函数恢复到x1(ra);pvParameters恢复到了x10(a0)中,上面已经详细解释了。最后调用ret指令,cpu使用ra恢复到pc也就跳转到了任务入口函数中执行。

        到这里,cpu进入了Task世界,整个启动过程就结束了。

使用GDB跟踪任务启动过程

        启动gdb调试,加上3个断点xTaskCreate、pxPortInitialiseStack、xPortStartFirstTask。

        在断点xTaskCreate、pxPortInitialiseStack处打印处,a0、a1寄存器对比反汇编代码(或者对比map文件也一样)可以看到,和main_blinky()函数中创建任务时传递的参数是一致的,处理main_blinky中用户创建的task,系统还会自动创建IdleTask和TimerTask(config了软件定时器功能)。

        创建TimerTask的代码如下,入口函数prvTimerTask并且设置了最高优先级(毕竟是软件定时器可以看着rtos的软中断)。

       继续运行到xPortStartFirstTask时可以看到第一个启动的任务是TimerTask,将入口函数prvTimerTask写入到了ra寄存器,继续单步,等函数最后执行ret后就跳入到了prvTimerTask中,就切换到Task世界了,start.S 47行初始化的stack也就用不到了。

可以看到FreeRTOS一直运行在M模式,不像Linux有区分用户态和内核态。针对自己的SoC我们可以自己修改代码,将用户任务切换到U或者S模式,rtos自动创建的任务以及异常或者中断处理放在M模式中。

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 RISC-V 平台上使用 GDB 进行调试,需要先安装 RISC-V 版本的 GDB。可以通过以下命令安装: ``` sudo apt-get install gdb-multiarch ``` 安装完成后,可以使用以下命令启动 GDB: ``` riscv64-unknown-elf-gdb <executable> ``` 其中 `<executable>` 是要调试的可执行文件的路径。接下来,可以使用 GDB 的各种命令进行调试。 以下是一些常用的 GDB 命令: - `break <function>`:在指定函数处设置断点。 - `break <line>`:在指定行处设置断点。 - `run <args>`:运行程序并传入参数。 - `step`:单步执行程序。 - `next`:执行下一条语句,不进入函数内部。 - `continue`:继续执行程序,直到遇到下一个断点。 - `print <variable>`:打印变量的值。 - `backtrace`:查看函数调用栈。 此外,还可以使用 `-g` 选项编译程序时生成调试信息,以便在 GDB 中进行调试。例如: ``` riscv64-unknown-elf-gcc -g -o <executable> <source files> ``` ### 回答2: gdb是一个功能强大的调试工具,适用于多种不同的处理器架构,其中包括risc-v。使用gdb调试risc-v程序可以帮助我们定位并解决程序中的bug。 具体步骤如下: 1. 安装gdb:首先需要确保在计算机上安装了gdb。可以通过包管理器或者从gdb的官方网站上下载并安装。 2. 编译程序:在使用gdb之前,需要将C/C++程序编译为可执行文件。例如,我们可以使用risc-v的交叉编译工具链来编译程序。 3. 运行gdb:在命令行中输入"gdb"命令来启动gdb调试器。 4. 加载可执行文件:在gdb中输入"file <可执行文件路径>"命令来加载要调试的可执行文件。 5. 设置断点:可以使用"break <行号或函数名>"命令在程序中设置断点。这将使程序暂停在指定的位置,以便进行调试。 6. 启动程序:在gdb中输入"run"命令来启动程序的执行。当程序到达设置的断点时,它将暂停执行。 7. 检查程序状态:在程序暂停时,可以使用各种gdb命令来检查当前变量的值、堆栈跟踪等信息,以了解程序的当前状态。 8. 单步执行:可以使用"next"或"step"命令逐行执行程序。这将允许我们逐步跟踪程序的执行,以查找错误。 9. 查看变量值:使用"gdb"命令可以查看当前变量的值。可以为每个变量设置监视点来观察其变化。 10. 修复错误:一旦发现了bug,可以在gdb中进行相应的更改或修复。然后可以重新编译程序并再次使用gdb进行调试,以确保问题已经解决。 使用gdb调试risc-v程序可以更加高效地定位问题并进行调试。通过将gdb与其他调试技术,如断言和日志输出,结合使用,可以帮助开发人员更轻松地编写和维护risc-v程序。 ### 回答3: gdb是一款强大的调试器,可用于调试RISC-V程序。以下是使用gdb调试RISC-V程序的步骤: 首先,确保已经安装了RISC-V架构的gdb工具。如果没有安装,可以通过下载源代码并手动编译安装来获取。 1. 将需要调试的RISC-V可执行文件加载到gdb中。可以使用以下命令: ``` gdb <可执行文件名> ``` 2. 设置断点来在程序的特定位置中断执行。使用`break`命令来设置断点。例如,要在第30行设置断点,可以使用以下命令: ``` break main.c:30 ``` 3. 启动调试会话。可以直接使用`run`命令启动程序,或者使用`start`命令以暂停的方式启动程序(这样可以在设置完断点后再启动)。 4. 当程序在断点处中断时,可以使用`next`命令逐过程执行,或者使用`step`命令逐语句执行。还可以使用`continue`命令使程序继续执行直到下一个断点或程序结束。 5. 在程序执行过程中,可以使用`print`命令打印变量或表达式的值。例如,要打印变量`x`的值,可以使用以下命令: ``` print x ``` 6. 若要查看当前执行的源代码行,可以使用`list`命令。 7. 将gdb作为交互式调试器使用时,还可以使用一些其他命令来查看寄存器的值、改变变量的值等。 在调试RISC-V程序时,gdb是一个非常有用的工具。它提供了一系列命令和功能,帮助我们理解程序执行的过程,查找错误并进行调试。以上是使用gdb调试RISC-V程序的基本步骤。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值