一篇文章带你使用C语言编写内核

内核是操作系统最核心的内容,主要提供硬件抽象层、磁盘及文件系统控制、多任务等功能,由于其涉及非常广泛的计算机知识,很少被人们所熟悉,因而披上了一层神秘的面纱

gcc 命令

  • 使用 gcc 编译 c语言

-c 编译、汇编到目标代码,不进行链接,也就是直接生成目标文件

-o 将输出的文件以指定文件名来储存,有同名文件存在时直接覆盖

gcc -c -o kernel/main.o kernel/main.c

编译:编译号之后只是个目标文件,也称为待重定位文件,重定位指的是文件里面所用的符号还没有安排地址,这些符号的地址需要将来与其他目标文件“组成”一个可执行文件时再重新定位(编排地址〉,这里的符号就是指该目标文件中所调用的函数或使用的变量,而这里的“组成”就是指链接。需要在所有目标文件都到齐了,将它们链接到 起时再重新定位(编排地址)

  • 使用 gcc 链接

-Ttext指定虚拟地址

-e 用来指定程序的起始地址(默认为_start)

gcc kernel/main.o -Ttext 0xc0001500 -e main -o kernel/kernel.bin

  • 编译链接

生成的test.bin不再是目标文件,而是可执行文件

gcc -o ./kernel/test.bin ./kernel/main.c

  • main 函数不是第一个执行的代码,它一定是被其它代码调用的,main函数在运行库代码初始化完环境后才被调用

文件头

二进制文件的运行方法

  • 在文件头中写入和程序属性有关的信息
  • 将这种具有程序头格式的程序文件从外存读入到内存后,从该程序文件的程序头中读出入口地址, 要直接跳进入口地址执行,跨过程序头才行。

在这里插入图片描述

  • header.S

编译后生成的文件是 header.binnams -o header.bin header.S

在这里插入图片描述

  • 调用方的执行过程

在这里插入图片描述

在这里插入图片描述

  • 在实际中,程序头和程序体相分离的文件叫 elf 格式

将内核载入内存

将内核写入磁盘

dd if=./test/kernel/kernel.bin of=hd60M.img bs=512 count=200 seek=9 conv=notrunc

可以将编译、链接、写入硬盘写成一个脚本

gcc -c -o test/kernel/main.o test/kernel/main.c && gcc test/kernel/main.o -Ttext 0xc0001500 -e main -o test/kernel/kernel.bin && dd if=./test/kernel/kernel.bin of=hd60M.img bs=512 count=200 seek=9 conv=notrunc

修改 loader.S

加载内核:需要把内核文件加载到内存缓冲区。

初始化内核:需要在分页后,将加载进来的 elf 内核文件安置到相应的虚拟内存地址,然后跳过去执行,从此 loader 的工作结束。

把内核文件从硬盘上加载到内存中

 mov eax, KERNEL_START_SECTOR               ; kernel.bin所在的扇区号 mov ebx, KERNEL_BIN_BASE_ADDR              ; 从磁盘读出后,写入到ebx指定的地址。加载到的内存地址 mov ecx, 200			                      ; 读入的扇区数 call rd_disk_m_32 ; 创建页目录及页表并初始化页内存位图 call setup_page 

初始化内核:初始化内核就是根据 elf 规范将内核文件中的段( segment )展开到(复制到)内存中的相应位置

总结

本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注html中文网的更多内容!

以上就是一篇文章带你使用C语言编写内核的详细内容,更多请关注0133技术站其它相关文章!

赞(0) 打赏
未经允许不得转载:0133技术站首页 » C语言