[嵌入式系统-38]:龙芯1B 开发学习套件 -8-启动过程详解

2024-03-10 09:44

本文主要是介绍[嵌入式系统-38]:龙芯1B 开发学习套件 -8-启动过程详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一、启动程序的入口

二、初始化硬件浮点单元fpu

三、全局指针

四、设置系统栈

五、设置内存

六、设置Cache

七、配置协处理器

八、初始化TLB


一、启动程序的入口

  1. FRAME(_start, sp, 0, ra):这一行是一个汇编宏,用于创建一个函数框架_start 函数的入口地址sp 是栈指针,0 表示没有保存寄存器的空间,ra 是返回地址寄存器。

  2. .set noreorder:这个指令告诉汇编器不要重新排序指令。

  3. #if (__mips_hard_float):这是一个条件编译指令,检查是否启用了硬件浮点运算

  4. li v0, SR_CU1:将 SR_CU1 的值加载到寄存器 v0 中。

  5. mtc0 v0, C0_SR:将寄存器 v0 的值写入 CP0 寄存器 C0_SR 中。

  6. #else

  7. mtc0    zero, C0_SR                     /* clr IntMsks/ kernel/disabled mode */    

  8. #endif

  9. nop:空操作指令。

  10. li v0, CAUSE_DC:将 CAUSE_DC 的值加载到寄存器 v0 中。

  11. mtc0 v0, C0_CAUSE:将寄存器 v0 的值写入 CP0 寄存器 C0_CAUSE 中。

  12. nop:空操作指令。

  13. lui v0, 0xFFFF:将 0xFFFF 的高 16 位加载到寄存器 v0 中。

  14. mtc0 v0, C0_COMPARE:将寄存器 v0 的值写入 CP0 寄存器 C0_COMPARE 中。

  15. nop:空操作指令。

这段代码主要是针对 MIPS 架构的初始化设置。根据是否启用了硬件浮点运算,设置相应的寄存器值,清除软件中断和定时器中断

二、初始化硬件浮点单元fpu

  1. #if (__mips_hard_float):这是一个条件编译指令,检查是否启用了硬件浮点运算。

  2. li t3, 0xAAAA5555:将常数 0xAAAA5555 加载到寄存器 t3 中。

  3. mtc1 t3, fp0:将寄存器 t3 的值写入浮点寄存器 fp0 中。

  4. mtc1 zero, fp1:将寄存器 zero 的值(0)写入浮点寄存器 fp1 中。

  5. mfc1 t0, fp0:将浮点寄存器 fp0 的值读取到通用寄存器 t0 中。

  6. mfc1 t1, fp1:将浮点寄存器 fp1 的值读取到通用寄存器 t1 中。

  7. nop:空操作指令。

  8. bne t0, t3, 1f:如果寄存器 t0 的值不等于 t3,则跳转到标签 1f 处执行。

  9. nop:空操作指令。

  10. bne t1, zero, 1f:如果寄存器 t1 的值不等于 0,则跳转到标签 1f 处执行。

  11. nop:空操作指令。

  12. ctc1 zero, fcr31:将寄存器 zero 的值写入浮点控制寄存器 fcr31 中,即清除浮点控制状态寄存器。

  13. j 2f:无条件跳转到标签 2f 处执行。

  14. nop:空操作指令。

  15. li v0, 0x0:将常数 0x0 加载到寄存器 v0 中。

  16. mtc0 v0, C0_SR:将寄存器 v0 的值写入 CP0 寄存器 C0_SR 中,即重置状态寄存器。

  17. nop:空操作指令。

  18. 2::标签 2,用于跳转。

  19. #endif:结束条件编译指令块。

这段代码主要用于检查是否存在浮点处理器(FPU),如果存在,则清除浮点控制状态寄存器,并重置状态寄存器。

三、全局指针

  1. la gp, _gp:将全局指针 _gp 的地址加载到全局指针寄存器 gp 中,用于初始化全局指针寄存器。

  2. la v0, _fbss:将未初始化数据段(bss段)的起始地址 _fbss 加载到通用寄存器 v0 中。

  3. la v1, _end:将未初始化数据段的结束地址 _end 加载到通用寄存器 v1 中。

  4. 3::标签 3,用于循环中的跳转。

  5. sw zero, 0(v0):将寄存器 zero 的值(0)存储到地址为 v0 的内存位置,即清零未初始化数据段中的一个字。

  6. bltu v0, v1, 3b:如果 v0 的值小于 v1 的值,则跳转到标签 3 处执行。这是一个无符号数比较分支指令,用于在清零未初始化数据段的过程中循环处理直到结束。

  7. add v0, 4:将寄存器 v0 的值加上4,即增加一个字的大小,用于指向下一个未初始化数据段的地址。

这段代码的主要作用是初始化全局指针寄存器,并清零未初始化数据段(bss段)的内容,确保其在使用之前被正确初始化。

四、设置系统栈

  1. la t0, _stack_end: 将系统栈的结束地址 _stack_end 加载到临时寄存器 t0 中。

  2. sub t0, t0, (4 * 4): 从系统栈的结束地址减去4个字(即16个字节)。这可能是为了留出一些空间用于栈帧或其他用途。“XXX overhead” 注释可能表示这个操作有一定的额外开销。

  3. move sp, t0: 将临时寄存器 t0 中的值移动到栈指针寄存器 sp 中,从而设置系统栈。

  4. la a0, _RamSize: 将 RAM 的大小 _RamSize 加载到寄存器 a0 中。

五、设置内存

  1. jal set_memory_size: 调用函数 set_memory_size,这可能是设置内存大小的函数。

  2. nop: 空操作指令,用于填充延时槽或优化。

六、设置Cache

  1. jal config_cache: 调用函数 config_cache,用于确定 D & I caches 的大小。

  2. nop: 空操作指令。

  3. la a0, memory_cfg_struct: 将内存配置结构体 memory_cfg_struct 的地址加载到寄存器 a0 中。

  4. jal get_memory_conf: 调用函数 get_memory_conf,可能用于填充内存配置结构体。

  5. nop: 空操作指令。

  6. jal flush_cache: 调用函数 flush_cache,用于初始化缓存。

  7. nop: 空操作指令。

七、配置协处理器

  1. mfc0 v0, C0_CONFIG, 0: 从协处理器0的配置寄存器中(特权级0)读取数据,将其存储到通用寄存器 v0 中。

  2. and v0, v0, ~CFG0_K0: 将 v0 的值与 CFG0_K0 的按位取反值进行按位与操作,可能是为了清除特定位的设置。

  3. or v0, v0, CFG_C_CACHABLE: 将 CFG_C_CACHABLE 的值与 v0 进行按位或操作,可能是为了设置缓存为可缓存的模式。

  4. mtc0 v0, C0_CONFIG, 0: 将寄存器 v0 的值写入到协处理器0的配置寄存器中(特权级0)。

  5. nop: 空操作指令。

这段代码的主要作用是设置系统栈、内存大小、缓存配置等,并进行一些与内存和缓存相关的初始化操作。

八、初始化TLB

/* Clear Translation Lookaside Buffer (TLB)
 */
    jal     init_tlb                        /* clear the tlb */
    nop

/* End of CPU initialization, ready to start kernel
 */
    move    a0, zero                        /* Set argc passed to main */
    jal     bsp_start                        //C语言启动代码,不返回,
    nop

/* Kernel has been shutdown, jump to the "exit" routine
 */
    jal     _sys_exit
    move    a0, v0                          # pass through the exit code

1:
    beq     zero, zero, 1b
    nop

    .set    reorder
ENDFRAME(_start)

这篇关于[嵌入式系统-38]:龙芯1B 开发学习套件 -8-启动过程详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

SpringBoot 中 CommandLineRunner的作用示例详解

《SpringBoot中CommandLineRunner的作用示例详解》SpringBoot提供的一种简单的实现方案就是添加一个model并实现CommandLineRunner接口,实现功能的... 目录1、CommandLineRunnerSpringBoot中CommandLineRunner的作用

Python使用smtplib库开发一个邮件自动发送工具

《Python使用smtplib库开发一个邮件自动发送工具》在现代软件开发中,自动化邮件发送是一个非常实用的功能,无论是系统通知、营销邮件、还是日常工作报告,Python的smtplib库都能帮助我们... 目录代码实现与知识点解析1. 导入必要的库2. 配置邮件服务器参数3. 创建邮件发送类4. 实现邮件

Java死锁问题解决方案及示例详解

《Java死锁问题解决方案及示例详解》死锁是指两个或多个线程因争夺资源而相互等待,导致所有线程都无法继续执行的一种状态,本文给大家详细介绍了Java死锁问题解决方案详解及实践样例,需要的朋友可以参考下... 目录1、简述死锁的四个必要条件:2、死锁示例代码3、如何检测死锁?3.1 使用 jstack3.2

详解Linux中常见环境变量的特点与设置

《详解Linux中常见环境变量的特点与设置》环境变量是操作系统和用户设置的一些动态键值对,为运行的程序提供配置信息,理解环境变量对于系统管理、软件开发都很重要,下面小编就为大家详细介绍一下吧... 目录前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变

Java日期类详解(最新推荐)

《Java日期类详解(最新推荐)》早期版本主要使用java.util.Date、java.util.Calendar等类,Java8及以后引入了新的日期和时间API(JSR310),包含在ja... 目录旧的日期时间API新的日期时间 API(Java 8+)获取时间戳时间计算与其他日期时间类型的转换Dur

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

Oracle修改端口号之后无法启动的解决方案

《Oracle修改端口号之后无法启动的解决方案》Oracle数据库更改端口后出现监听器无法启动的问题确实较为常见,但并非必然发生,这一问题通常源于​​配置错误或环境冲突​​,而非端口修改本身,以下是系... 目录一、问题根源分析​​​二、保姆级解决方案​​​​步骤1:修正监听器配置文件 (listener.

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地

Java中常见队列举例详解(非线程安全)

《Java中常见队列举例详解(非线程安全)》队列用于模拟队列这种数据结构,队列通常是指先进先出的容器,:本文主要介绍Java中常见队列(非线程安全)的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一.队列定义 二.常见接口 三.常见实现类3.1 ArrayDeque3.1.1 实现原理3.1.2