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