【问题标题】:ld: cannot perform PE operations on non PE output file errorld: 无法对非 PE 输出文件执行 PE 操作错误
【发布时间】:2014-09-27 12:16:01
【问题描述】:

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

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

【问题讨论】:

    标签: c gcc mingw ld


    【解决方案1】:

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

    【讨论】:

    • 非常感谢。当我从汇编代码中调用内核偏移时,这有效。但是这次我不能将汇编内核入口代码与 c 代码合并。我尝试了以下操作:
    • 非常感谢。当我从汇编代码中调用内核偏移时,这有效。但是这次我不能将汇编内核入口代码与 c 代码合并。我尝试了以下操作:nasm kernel_entry.asm -f win -o kernel_entry.o 然后 ld -r -o kernel.tmp -Ttext 0x1000 kernel_entry.o kernel.o 最后使用 objcopy 命令创建 kernel.bin。一切正常,但屏幕上没有 X。我没有收到错误消息。你能帮帮我吗?
    • 您是否将boot_sect.binkernel.bin 结合在一起?
    • 是的,因为我使用的是 Windows 机器,所以我使用了“type”命令:输入 bootsec.bin kernel.bin empty.bin(充满 0 字节来说服 bochs 它是一个磁盘)> os -图像.img。我在引导扇区代码中放入了一些消息。所有消息都会出现,但不会在屏幕上显示 X。我不明白。
    • @JamJar00 是的,您丢失了所有其他部分,但原始海报没有任何其他部分。至少在他改变问题之前不会。正如我在之前的评论中提到的,这整个事情非常脆弱。您可以只复制所有内容,但最终会得到一个更大的文件,因为它最终包含几个无用的 4K 对齐部分(.rdata$zzz.eh_frame)。更好的解决方案是编写一个自定义链接器脚本(即不是NUL),将所有内容(除了垃圾)放入.text
    猜你喜欢
    • 2015-09-05
    • 2012-11-25
    • 1970-01-01
    • 1970-01-01
    • 2016-11-14
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多