创建 x86 引导程序

Creating x86 bootloader

我正在写一个bootloader如下:

bits 16
[org 0x7c00]
KERN_OFFSET equ 0x1000
mov [BOOTDISK], dl
mov dl, 0x0 ;0 is for floppy-disk
mov ah, 0x2 ;Read function for the interrupt
mov al, 0x15 ;Read 15 sectors conating kernel
mov ch, 0x0 ;Use cylinder 0
mov cl, 0x2 ;Start from the second sector which contains kernel
mov dh, 0x0 ;Read head 0 
mov bx, KERN_OFFSET
int 0x13
jc disk_error
cmp al, 0x15
jne disk_error
jmp KERN_OFFSET:0x0
jmp $

disk_error:
jmp $


BOOTDISK: db 0

times 510-($-$$) db 0
dw 0xaa55

内核是一个简单的 C 程序,它在 VGA 显示器上打印 "e"(在 QEmu 上看到):

void main()
{
 extern void put_in_mem();
 char c = 'e';
 put_in_mem(c, 0xA0);
}

我在 QEmu 中以 16 位(实模式)使用这段代码,所以我使用编译器 bcc 来处理这段代码,使用:

bcc -ansi -c -o kernel.o kernel.c

我有以下问题:
1. 当我尝试反汇编这段代码时,使用

objdump -D -b binary -mi386 kernel.o

我得到这样的输出(只有输出的初始部分):

kernel.o:     file format binary

Disassembly of section .data:

00000000 <.data>:
   0:   a3 86 01 00 2a          mov    %eax,0x2a000186
   5:   3e 00 00                add    %al,%ds:(%eax)
   8:   00 22                   add    %ah,(%edx)
   a:   00 00                   add    %al,(%eax)
   c:   00 19                   add    %bl,(%ecx)
   e:   00 00                   add    %al,(%eax)
   10:  00 55 55                add    %dl,0x55(%ebp)
   13:  55                      push   %ebp
   14:  55                      push   %ebp
   15:  00 00                   add    %al,(%eax)
   17:  00 02                   add    %al,(%edx)
   19:  22 00                   and    (%eax),%al

此输出似乎与我制作的kernel.c 文件不对应。例如,我看不到 'e' 存储为 ASCII 0x65 的位置或对 put_in_mem 的调用在哪里。是我反汇编代码的方式有问题吗?

  1. 为了制作 QEmu 内核的目标文件,我使用了以下命令:
ld86 -o 内核 -d kernel.o put_in_mem.o

此处put_in_mem.o是汇编put_in_mem.asm文件后创建的目标文件,其中包含kernel.c.
中使用的函数put_in_mem()的定义 然后使用以下方法制作 QEmu 的软盘映像:

cat boot.o kernel > floppy_img

但是当我尝试查看地址 0x10000(使用 GDB)时,内核应该在加载后存在(使用 boot.asm 程序),但它不存在。
为什么会这样?
此外,在 ld 命令中我们使用了 -Ttext 选项来指定二进制文件的加载地址,我们是否应该在此处使用与 ld86 类似的选项?

您的 kernel.o 是 objdump 无法理解的 object 文件格式,因此它会尝试反汇编其中的所有内容,包括 headers 等等。尝试反汇编链接的输出 kernel。 objdump 也可能不理解 16 位代码。如果可以的话,最好尝试 objdump86

至于为什么它不存在:你看错地方了。您正在加载它以偏移 0x1000(3 个零),但您正在查看 0x10000(4 个零)。另请注意,您没有设置 ES 这是不好的做法。也许您打算将 ES 设置为 0x1000 并将 BX 设置为 0x0000 然后您会在 0x10000 物理地址找到您的内核。

-Ttext 不影响加载,它只指定代码期望找到自己的位置。