ld:无法对非PE输出文件错误执行PE操作

use*_*185 5 c gcc mingw ld

我是操作系统编程的新手,我正在阅读一本书,它提供了一个简单的内核示例,如下所示:

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中的最后一个消息显示为三个奇怪的字符.我不明白会发生什么.

Ros*_*dge 9

您需要做的第一件事是更改您的功能名称.如果你调用它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)

  • 好的,我已经更改了我的答案。我没有使用将所有内容链接到一个目标文件的“-r”链接器选项,而是使用“-T NUL”告诉链接器不要使用链接器脚本。这两个选项的目的是删除链接描述文件将添加的一些额外垃圾。然而,“-r”选项导致修复程序应用不正确。请注意,这可能不是您问题的结束。您所追随的书中所采用的方法非常脆弱。 (2认同)
  • @JamJar00是的,您丢失了所有其他部分,但原始海报没有任何其他部分。至少在他改变问题之前是这样。正如我在之前的评论中提到的,整个事情非常脆弱。您可以复制所有内容,但最终会得到一个更大的文件,因为它最终包含一些无用的 4K 对齐部分(“.rdata$zzz”和“.eh_frame”)。更好的解决方案是编写一个自定义链接器脚本(即不是“NUL”),将所有内容(除了垃圾)放入“.text”中。 (2认同)