gpt4 book ai didi

assembly - BIOS int 10h 在 QEMU 上打印垃圾

转载 作者:行者123 更新时间:2023-12-01 01:59:06 25 4
gpt4 key购买 nike

我在编写一个在 QEMU 中作为引导加载程序运行的 x86 实模式汇编程序时遇到问题。我正在尝试通过 BIOS 中断 0x10 打印文本。我的代码是:

print:
pusha
.loop:
mov AL, [SI]
cmp AL, 0
je .end
call printChar
inc SI
jmp .loop
.end:
popa
ret

printChar:
pusha
mov AH, 0x0E
mov BH, 0
mov BL, 0x0F
int 0x10
popa
ret

我使用 [ORG 0x7c00] 作为原点。我测试了 printChar 标签并用 AL 中的一些字母调用它,它工作正常。当我尝试将内存地址加载到这样的消息时:

loadMsg      db "Loading",0
mov SI, loadMessage
call print

我在 QEMU 模拟器上得到像“U”这样的垃圾输出。昨天,我写了一段和这个非常相似的代码,一点问题都没有。是什么导致了我的问题,如何解决?

最佳答案

我最近写了一些General Bootloader Tips在可能对您有用的 Stackoverflow 答案中。可能提示 #1 适用于您的问题:

When the BIOS jumps to your code you can't rely on CS,DS,ES,SS,SP registers having valid or expected values. They should be set up appropriately when your bootloader starts. You can only be guaranteed that your bootloader will be loaded and run from physical address 0x00007c00 and that the boot drive number is loaded into the DL register.

基于 printChar 有效的事实,并且写出整个字符串并不意味着 DS:SI 没有指向内存中的正确位置你的字符串驻留。这通常是因为当 BIOS 跳转到引导加载程序时,开发人员错误地假设 CS 和/或 DS 寄存器已正确设置。它必须手动设置。在原点为 0x7c00 的情况下,DS 需要设置为 0。在 16 位实模式中,物理内存地址是从 segment:offset pairs 计算得出的。使用公式 (segment<<4)+offset .在您的情况下,您使用的是偏移量 0x7C00。 DS 中的值为 0 将产生 (0<<4)+0x7c00 = 0x07c00 的正确物理地址。

您可以在程序开始时将 DS 设置为 0,方法如下:

xor ax, ax       ; set AX to zero
mov ds, ax ; DS = 0

对于 QEMU,BIOS 跳转到 0x07c0:0x0000。这也代表相同的物理内存位置 (0x07c0<<4)+0 = 0x07c00 。这样的跳转将设置 CS=0x07c0(不是 CS=0)。由于有许多段:偏移量对映射到相同的物理内存位置,因此您需要适本地设置 DS。您不能指望 CS 是您期望的值。所以在 QEMU 中,这样的代码甚至不会正确设置 DS(使用 ORG 0x7c00 时):

mov ax, cs
mov ds, ax ; Make DS=CS

这可能适用于一些模拟器,如 DOSBOX 和一些物理硬件,但不是全部。这段代码可以工作的环境是当 BIOS 跳转到 0x0000:0x7c00 时。在那种情况下,CS 在到达您的引导加载程序代码时将为零,并且将 CS 复制到 DS 会起作用。不要假设 CS 在所有环境中都为零是我的要点。始终将段寄存器设置为您明确想要的内容。

应该工作的代码示例是:

    BITS  16
ORG 0x7c00
GLOBAL main

main:
xor ax, ax ; AX = 0
mov ds, ax ; DS = 0
mov bx, 0x7c00

cli ; Turn off interrupts for SS:SP update
; to avoid a problem with buggy 8088 CPUs
mov ss, ax ; SS = 0x0000
mov sp, bx ; SP = 0x7c00
; We'll set the stack starting just below
; where the bootloader is at 0x0:0x7c00. The
; stack can be placed anywhere in usable and
; unused RAM.
sti ; Turn interrupts back on

mov SI, loadMsg
call print

cli
.endloop:
hlt
jmp .endloop ; When finished effectively put our bootloader
; in endless cycle

print:
pusha
.loop:
mov AL, [SI] ; No segment on [SI] means implicit DS:[SI]
cmp AL, 0
je .end
call printChar
inc SI
jmp .loop
.end:
popa
ret

printChar:
pusha
mov AH, 0x0E
mov BH, 0
mov BL, 0x0F
int 0x10
popa
ret

; Place the Data after our code
loadMsg db "Loading",0

times 510 - ($ - $$) db 0 ; padding with 0 at the end
dw 0xAA55 ; PC boot signature

关于assembly - BIOS int 10h 在 QEMU 上打印垃圾,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33974115/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com