11.操作系统实战——第一个C函数,hal层初始化,初始化平台,

切换CPU到长模式

在调用第一个c函数之前,我们仍然要写一段汇编,切换CPU进入长模式,初始化CPU寄存器和C语言要用的栈。
因为目前代码执行流在二级引导器中,进入到 Cosmos 中这样在二级引导器中初始过的东西都不能用了
因为 CPU 进入了长模式,寄存器的位宽都变了,所以需要重新初始化。让我们一起来写这段汇编代码吧,我们先在 Cosmos/hal/x86/ 下建立一个 init_entry.asm 文件,写上后面这段代码


[section .start.text]
[BITS 32]
_start:
    cli
    mov ax,0x10
    mov ds,ax
    mov es,ax
    mov ss,ax
    mov fs,ax
    mov gs,ax
    lgdt [eGdtPtr]        
    ;开启 PAE
    mov eax, cr4
    bts eax, 5                      ; CR4.PAE = 1
    mov cr4, eax
    mov eax, PML4T_BADR             ;加载MMU顶级页目录
    mov cr3, eax  
    ;开启 64bits long-mode
    mov ecx, IA32_EFER
    rdmsr
    bts eax, 8                      ; IA32_EFER.LME =1
    wrmsr
    ;开启 PE 和 paging
    mov eax, cr0
    bts eax, 0                      ; CR0.PE =1
    bts eax, 31
    ;开启 CACHE       
    btr eax,29                    ; CR0.NW=0
    btr eax,30                    ; CR0.CD=0  CACHE
    mov cr0, eax                    ; IA32_EFER.LMA = 1
    jmp 08:entry64
[BITS 64]
entry64:
    mov ax,0x10
    mov ds,ax
    mov es,ax
    mov ss,ax
    mov fs,ax
    mov gs,ax
    xor rax,rax
    xor rbx,rbx
    xor rbp,rbp
    xor rcx,rcx
    xor rdx,rdx
    xor rdi,rdi
    xor rsi,rsi
    xor r8,r8
    xor r9,r9
    xor r10,r10
    xor r11,r11
    xor r12,r12
    xor r13,r13
    xor r14,r14
    xor r15,r15
    mov rbx,MBSP_ADR
    mov rax,KRLVIRADR
    mov rcx,[rbx+KINITSTACK_OFF]
    add rax,rcx
    xor rcx,rcx
    xor rbx,rbx
    mov rsp,rax
    push 0
    push 0x8
    mov rax,hal_start                 ;调用内核主函数
    push rax
    dw 0xcb48
    jmp $
[section .start.data]
[BITS 32]
x64_GDT:
enull_x64_dsc:  dq 0  
ekrnl_c64_dsc:  dq 0x0020980000000000   ; 64-bit 内核代码段
ekrnl_d64_dsc:  dq 0x0000920000000000   ; 64-bit 内核数据段
euser_c64_dsc:  dq 0x0020f80000000000   ; 64-bit 用户代码段
euser_d64_dsc:  dq 0x0000f20000000000   ; 64-bit 用户数据段
eGdtLen      equ  $ - enull_x64_dsc   ; GDT长度
eGdtPtr:    dw eGdtLen - 1      ; GDT界限
        dq ex64_GDT

上述代码中,1~11 行表示加载 70~75 行的 GDT,13~17 行是设置 MMU 并加载在二级引导器中准备好的 MMU 页表,19~30 行是开启长模式并打开 Cache,34~54 行则是初始化长模式下的寄存器,55~61 行是读取二级引导器准备的机器信息结构中的栈地址,并用这个数据设置 RSP 寄存器

最关键的是 63~66 行,它开始把 8 和 hal_start 函数的地址压入栈中。dw 0xcb48 是直接写一条指令的机器码——0xcb48,这是一条返回指令。这个返回指令有点特殊,它会把栈中的数据分别弹出到 RIP,CS 寄存器,这正是为了调用我们 Cosmos 的第一个 C 函数 hal_start。

由于这是第一个 C 函数,也是初始化函数,我们还是要为它单独建立一个文件,以显示对它的尊重,依然在 Cosmos/hal/x86/ 下建立一个 hal_start.c 文件。写上这样一个函数。


void hal_start()
{
   
   
    //第一步:初始化hal层
    //第二步:初始化内核层
    for(;;);
    return;
}

hal层(硬件抽象层)初始化

平台初始化,hal 层的内存初始化,中断初始化,最后进入到内核层的初始化。
为了分离硬件的特性,我们设计了 hal 层,把硬件相关的操作集中在这个层,并向上提供接口,目的是让内核上层不用关注硬件相关的细节,也能方便以后移植和扩展。
下面我们在 Cosmos/hal/x86/ 下建立一个 halinit.c 文件,写出 hal 层的初始化函数


void init_hal()
{
   
   
    //初始化平台
    //初始化内存
    //初始化中断
    return;
}

这个函数也是一个调用者,没怎么干活。不过根据代码的注释能看出,它调用的函数多一点,但主要是完成初始化平台、初始化内存、初始化中断的功能函数。

初始化平台

我们先来写好平台初始化函数,因为它需要最先被调用。这个函数主要负责完成两个任务,
一是把二级引导器建立的机器信息结构复制到 hal 层中的一个全局变量中,方便内核中的其它代码使用里面的信息,之后二级引导器建立的数据所占用的内存都会被释放。
二是要初始化图形显示驱动,内核在运行过程要在屏幕上输出信息。
下面我们在 Cosmos/hal/x86/ 下建立一个 halplatform.c 文件,写上如下代码。


void machbstart_t_init(machbstart_t *initp)
{
   
   
    //清零
    memset(initp, 0, sizeof(machbstart_t));
    return;
}

void init_machbstart()
{
   
   
    machbstart_t *kmbsp = &kmachbsp;
    machbstart_t *smbsp = MBSPADR;//物理地址1MB处
    machbstart_t_init(kmbsp);
    //复制,要把地址转换成虚拟地址
    memcopy((void *)phyadr_to_viradr((adr_t)smbsp), (void *)kmbsp, sizeof(machbstart_t));
    return;
}
//平台初始化函数
void init_halplaltform()
{
   
   
    //复制机器信息结构
    init_machbstart();
    //初始化图形显示驱动
    init_bdvideo();
    return;
}

kmachbsp 你可能会有点奇怪,它是个结构体变量,结构体类型是 machbstart_t,这个结构和二级引导器所使用的一模一样。同时,它还是一个 hal 层的全局变量,我们想专门有个文件定义所有 hal 层的全局变量,于是我们在 Cosmos/hal/x86/ 下建立一个 halglobal.c 文件,写上如下代码。


//全局变量定义变量放在data段
#define HAL_DEFGLOB_VARIABLE(vartype,varname) \
EXTERN  __attribute__((section(".data"))) vartype varname

HAL_DEFGLOB_VARIABLE(machbstart_t,kmachbsp);

初始化图形显示驱动

下面,我们在 Cosmos/hal/x86/ 下的 bdvideo.c 文件中,写好 init_bdvideo 函数。


void init_bdvideo()
{
   
   
    dftgraph_t *kghp = &kdftgh;
    //初始化图形数据结构,里面放有图形模式,分辨率,图形驱动函数指针
    init_dftgraph();
    //初始bga图形显卡的函数指针
    init_bga();
    //初始vbe图形显卡的函数指针
    init_vbe(
<think>嗯,用户这次的问题是关于HAL_MspInit函数在STM32HAL库中的初始化调用顺序,看起来是在深入调试STM32底驱动时遇到的。用户之前的问题是关于I2C中断未触发的问题,现在转向了更基础的初始化流程,可能是想从根本上理解HAL库的启动机制。从用户提供的引用来看,用户应该已经查阅过STM32F1的HAL库资料,注意到了stm32f1xx_hal_msp.c和stm32f1xx_hal.h之间的关系。特别是引用[2]中提到的GD32替换STM32时CAN初始化顺序问题,说明用户对硬件初始化顺序的敏感性很高——这点很专业,很多工程师都会忽略这种细节。用户真正想知道的是整个初始化链条中HAL_MspInit的准确定位。考虑到用户上次调试I2C中断时遇到的困难,这次的问题可能是想排查系统初始化面的问题。用户似乎是个嵌入式开发经验者,但可能在HAL库的抽象实现细节上需要更清晰的解释。需要重点说明三个关键初始化阶段的关系:HAL_Init()→hal库全局初始化HAL_MspInit()外设硬件初始化→外设驱动初始化。特别要强调CubeMX生成的代码结构里,HAL_MspInit是在HAL_Init内部被自动调用的这个重要特性。用户可能没有意识到的是:很多时候工程师会错误地在main函数里手动调用HAL_MspInit,这会导致重复初始化。或者更隐蔽的问题——在SystemInit()之前调用HAL_Init(),这时时钟都还没配置好。这些细节都要在回答中点明。从技术深度来看,这个问题涉及:1.启动文件(startup_stm32f1xxxx.s)中的复位序列2.SystemInit()函数的位置(在启动文件中调用)3.HAL库的三初始化模型最好用流程图配合初始化函数的调用栈来展示,这样才能让用户真正理解各个初始化函数的执行时机。用户可能下一步会问如何自定义MspInit回调,或者遇到外设初始化失败时如何调试MspInit函数,所以要在相关问题里提前埋下伏笔。考虑到用户之前遇到的问题都是中断相关的,相关问题的设计要侧重初始化对中断的影响。</think> 在 STM32 HAL 库的初始化流程中,`HAL_MspInit()` 函数的调用位置至关重要,它由系统自动触发,**不应手动调用**。其正确执行顺序和逻辑关系如下: --- ### 一、关键初始化函数的调用顺序 1. **系统级初始化 (`SystemInit()`)** - 由启动文件(如 `startup_stm32f1xxxx.s`)在 `main()` 前自动调用 - 负责初始化 FPU、配置系统时钟源(内部/外部晶振)、设置中断向量表等 - **调用位置**:`Reset_Handler → SystemInit → main` 2. **HAL 库全局初始化 (`HAL_Init()`)** - 在 `main()` 函数中**显式调用**的第一个 HAL 函数 - 初始化 Flash 预取、配置 SysTick 计时器、设置 NVIC 优先级分组 ```c int main(void) { HAL_Init(); // 第一步:初始化 HAL 库 // ... } ``` 3. **硬件抽象 MSP 初始化 (`HAL_MspInit()`)** - **由 `HAL_Init()` 内部自动调用** - 负责 MCU 级别的硬件初始化(GPIO、时钟、DMA 等) - 实现位置:`stm32f1xx_hal_msp.c` 中用户重写的函数 ```c // stm32f1xx_hal_msp.c void HAL_MspInit(void) { __HAL_RCC_AFIO_CLK_ENABLE(); // 例如:启用 AFIO 时钟 __HAL_RCC_PWR_CLK_ENABLE(); // 启用 PWR 时钟 } ``` 4. **外设初始化(如 `HAL_I2C_Init()`)** - 在 `HAL_MspInit()` **执行完成后**手动调用 - 初始化具体外设(I2C、UART 等),并自动触发外设级的 `HAL_I2C_MspInit()` ```c int main(void) { HAL_Init(); SystemClock_Config(); // 配置系统时钟(需用户实现) MX_I2C1_Init(); // 初始化 I2C(内部调用 HAL_I2C_Init()) } ``` --- ### 二、关键调用逻辑图 ```mermaid graph TD A[启动文件] -->|自动调用| B[SystemInit] B -->|设置时钟/中断| C[main函数] C --> D[HAL_Init] D -->|内部调用| E[HAL_MspInit] E -->|初始化MCU级硬件| F[HAL_Init返回] F --> G[SystemClock_Config] G --> H[MX_I2C1_Init] H -->|调用| I[HAL_I2C_Init] I -->|自动调用| J[HAL_I2C_MspInit] ``` --- ### 三、常见问题分析 1. **手动调用 `HAL_MspInit()` 导致重复初始化** - 错误示例:在 `main()` 中重复调用 `HAL_MspInit()` - 结果:可能破坏 `HAL_Init()` 建立的上下文,引发硬件冲突 2. **`HAL_MspInit()` 未正确实现** - 缺失关键时钟使能(如 `__HAL_RCC_GPIOA_CLK_ENABLE()`) - 未配置复用功能(GPIO 的 `AF_PP` 模式)[^2] 3. **初始化顺序错误** - 在 `HAL_Init()` 前调用外设初始化函数 → 外设依赖的时钟/中断未就绪 - 在 `SystemClock_Config()` 前调用 `HAL_MspInit()` → 时钟频率错误 --- ### 四、初始化顺序总结 | 步骤 | 函数 | 调用方式 | 实现位置 | |------|---------------------|---------------|---------------------| | 1 | `SystemInit()` | 自动(启动文件)| 系统库(通常不修改)| | 2 | `HAL_Init()` | 手动(main) | HAL 库 | | 3 | `HAL_MspInit()` | 自动(HAL内部)| 用户重写(hal_msp.c)| | 4 | `SystemClock_Config()`| 手动(main) | 用户实现 | | 5 | `MX_I2C1_Init()` 等 | 手动(main) | CubeMX 生成 | > **核心原则**:`HAL_MspInit()` 只能通过 `HAL_Init()` 自动触发,**禁止**在 `main()` 或其他位置显式调用[^1]。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值