时间:2022-04-15 09:30:27 | 栏目:C代码 | 点击:次
-c 编译、汇编到目标代码,不进行链接,也就是直接生成目标文件
-o 将输出的文件以指定文件名来储存,有同名文件存在时直接覆盖
gcc -c -o kernel/main.o kernel/main.c
编译:编译号之后只是个目标文件,也称为待重定位文件,重定位指的是文件里面所用的符号还没有安排地址,这些符号的地址需要将来与其他目标文件“组成”一个可执行文件时再重新定位(编排地址〉,这里的符号就是指该目标文件中所调用的函数或使用的变量,而这里的“组成”就是指链接。需要在所有目标文件都到齐了,将它们链接到 起时再重新定位(编排地址)
-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
二进制文件的运行方法
编译后生成的文件是 header.bin
:nams -o header.bin header.S
将内核写入磁盘
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 )展开到(复制到)内存中的相应位置