我是操作系统编程的新手,我正在阅读一本书,它提供了一个简单的内核示例,如下所示:
main() {
char *video_memory = 0xb8000;
*video_memory = 'X';
}
Run Code Online (Sandbox Code Playgroud)
要编译这个名为kernel.c的文件,我在Windows 7下使用MinGW,如下所示:
gcc -ffreestanding -c kernel.c -o kernel.o
Run Code Online (Sandbox Code Playgroud)
这将创建目标文件kernel.o.但是,以下命令不起作用.
ld -o kernel.bin -Ttext 0x1000 kernel.o --oformat binary
Run Code Online (Sandbox Code Playgroud)
我收到以下错误:
ld: cannot perform PE operations on non PE output file 'kernel.bin'
Run Code Online (Sandbox Code Playgroud)
我无法解决问题.请帮我.
谢谢
在Ross的帮助下,我成功地跳到内核偏移.但是,我无法从Kernel_entry.asm调用C函数.此外,当我从我的kernel.bin中删除C函数并更改下面的代码时,屏幕上会显示三个奇怪的字符.
Kernel_entry.asm如下:
[bits 32]
;[extern _start]
mov ebx, MSG_KERNEL_ENTRY
call print_string_pm
;call _start
jmp $
%include "print_string_pm.asm"
MSG_KERNEL_ENTRY db "Kernel entry is invoked", 0
Run Code Online (Sandbox Code Playgroud)
bootsec.asm如下:
[org 0x7c00]
KERNEL_OFFSET equ 0x1000
mov [BOOT_DRIVE], dl
mov bp, 0x9000
mov sp, bp
mov bx, MSG_REAL_MODE
call print_string
call load_kernel
call switch_to_pm
jmp $
%include "print_string.asm"
%include "disk_load.asm"
%include "gdt.asm"
%include "print_string_pm.asm"
%include "switch_to_pm.asm"
%include "clear_screen.asm"
[bits 16]
load_kernel:
mov bx, MSG_LOAD_KERNEL
call print_string
mov bx, KERNEL_OFFSET
mov dh, 15
mov dl, [BOOT_DRIVE]
call disk_load
ret
[bits 32]
BEGIN_PM:
;call clear_screen
mov ebx, MSG_PROT_MODE
call print_string_pm
call KERNEL_OFFSET
jmp $
BOOT_DRIVE db 0
MSG_REAL_MODE db "Started in 16-Bit Real Mode", 0
MSG_PROT_MODE db "Successfully switched to 32-Bit Protected Mode", 0
MSG_LOAD_KERNEL db "Loading Kernel into memory", 0
times 510 - ($ - $$) db 0
dw 0xaa55
Run Code Online (Sandbox Code Playgroud)
所有消息都正确显示,但Kernel_entry.asm中的最后一个消息显示为三个奇怪的字符.我不明白会发生什么.
您需要做的第一件事是更改您的功能名称.如果你调用它main,MinGW版本的GCC将插入一个调用来__main进行初始化.例如:
start() {
char *video_memory = 0xb8000;
*video_memory = 'X';
}
Run Code Online (Sandbox Code Playgroud)
这意味着您还必须进行相应的编辑kernel_entry.asm:
[bits 32]
[extern _start]
call _start
jmp $
Run Code Online (Sandbox Code Playgroud)
接下来,像以前一样编译和汇编这两个文件,并将它与这些命令链接:
ld -T NUL -o kernel.tmp -Ttext 0x1000 kernel_entry.o kernel.o
objcopy -O binary -j .text kernel.tmp kernel.bin
Run Code Online (Sandbox Code Playgroud)
第一个命令将对象链接为PECOFF可执行文件,然后第二个命令将其转换为二进制文件.现在结合二进制文件和引导加载程序:
copy /b boot_sect.bin+kernel.bin os-image
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
3915 次 |
| 最近记录: |