ld:无法对非PE输出文件错误执行PE操作
我是操作系统编程的新手,我正在阅读一本书,它给出了一个简单的内核示例如下:
main() { char *video_memory = 0xb8000; *video_memory = 'X'; }
要编译这个名为kernel.c的文件,我在Windows 7下使用MinGW,如下所示:
gcc -ffreestanding -c kernel.c -o kernel.o
这将创建目标文件kernel.o。 但是,以下命令不起作用。
ld -o kernel.bin -Ttext 0x1000 kernel.o --oformat binary
我收到以下错误:
ld: cannot perform PE operations on non PE output file 'kernel.bin'
我无法解决问题。 请帮我。
谢谢
在罗斯的帮助下,我成功地跳到内核偏移。 但是,我无法从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
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
所有消息都正确显示,但Kernel_entry.asm中的最后一个消息显示为三个奇怪的字符。 我不明白会发生什么。
您需要做的第一件事是更改您的function名称。 如果将其称为main
则MinGW版本的GCC将插入对__main
的调用以进行初始化。 例如:
start() { char *video_memory = 0xb8000; *video_memory = 'X'; }
这意味着您还必须相应地编辑kernel_entry.asm
:
[bits 32] [extern _start] call _start jmp $
接下来,像以前一样编译和汇编这两个文件,并将它与以下命令链接:
ld -T NUL -o kernel.tmp -Ttext 0x1000 kernel_entry.o kernel.o objcopy -O binary -j .text kernel.tmp kernel.bin
第一个命令将对象链接为PECOFF可执行文件,然后第二个命令将其转换为二进制文件。 现在将二进制文件与引导加载程序组合:
copy /b boot_sect.bin+kernel.bin os-image