5

我一直在研究这个网页上的教程,它逐步创建了一个显示 Hello World 的引导加载程序。

第二个教程(我们试图输出一个“A”)完美地工作,但第一个教程对我根本不起作用!(BIOS 完全忽略软盘并直接引导至 Windows)。这不是一个问题,尽管任何解释都会受到赞赏。

真正的问题是我无法让第三个教程工作。在输出“Hello World”时,我在屏幕的左下角看到了一个不寻常的字符(和闪烁的光标)。它看起来有点像圆角矩形内的笑脸。有谁知道如何让 Hello World 正常显示?

4

2 回答 2

7

您说“直接启动到 Windows”,所以我假设您使用的是物理 PC。未来注意事项:始终使用模拟器进行开发!这更容易。我喜欢用于 OSDeving 的 Bochs,因为它具有很好的调试功能。现在,讨论可能的解决方案。

有很多错误的 BIOS 破坏了 IBM PC 对 0x7C00 加载地址的非正式规范。

每当您组装时,这可能会给内存地址等带来很多问题。所以让开头看起来像这样:

[BITS 16] ;tell the assembler that its a 16 bit code
[ORG 0x7C00] ;this tells the assembler where the code will be loaded at when it runs on your machine. It uses this to compute the absolute addresses of labels and such.

jmp word 0:flush ;#FAR jump so that you set CS to 0. (the first argument is what segment to jump to. The argument(after the `:`) is what offset to jump to)
;# Without the far jmp, CS could be `0x7C0` or something similar, which will means that where the assembler thinks the code is loaded and where your computer loaded the code is different. Which in turn messes up the absolute addresses of labels.
flush: ;#We go to here, but we do it ABSOLUTE. So with this, we can reset the segment and offset of where our code is loaded.
mov BP,0 ;#use BP as a temp register
mov DS,BP ;#can not assign segment registers a literal number. You have to assign to a register first.
mov ES,BP ;#do the same here too
;#without setting DS and ES, they could have been loaded with the old 0x7C0, which would mess up absolute address calculations for data. 

看,一些负载0x07C0:0000和大多数负载(并且它被认为适合)在0x0000:7C00。它是相同的平面地址,但不同的段设置确实会搞砸绝对内存地址。所以让我们去掉汇编器的“魔法”,看看它是什么样子(注意我不保证地址完全正确。我不知道所有操作码的大小)

jmp word 0:0x7C04 ;# 0x7C04 is the address of the `flush` label 
...

所以,我们跳转到一个绝对地址。

接着。如果我们不这样做会发生什么?

以这个程序为例:

mov ax,[mydata]
hlt

mydata: dw 500 ;#just some data

这会分解成类似的东西

mov ax,[0x7C06] 

哦,它使用绝对寻址,那怎么会出错呢?好吧,如果 DS 实际上是0x7C0怎么办?0:0x7C06然后,它不会得到预期的汇编器,而是会得到0x7C0:0x7C06不同的平面地址。

我希望这可以帮助你理解。不过,这确实是一个复杂的话题,需要一段时间的低级编程才能完全理解。

于 2010-04-13T16:44:37.580 回答
1

我认为这个问题很可能与指定的来源有关。

[ORG 0x7C00]    ;Origin, tell the assembler that where the code will

根据我们一直在进行的对话,该地址似乎与某些方面的预测不符。可能只是 DS 数据段寄存器不是您所期望的。您实际上可以通过在调用之前添加 ds 的推送和弹出来从网页中获取原始列表以显示这样的字符串,

 push cs
 pop ds

如果不是,则以下代码有效。

 [ORG 0x000]    ; switched to 0 since we are going to try to correct it ourself

 call nextinstruction
 nextinstruction:    ; get the return address of the call into dx
 pop dx              ; which is essentially the start of the code + 3 (3 bytes for the call instruction)
 MOV SI, HelloString ;Store string pointer to SI
 add si, dx          ; add IP from start of program
 sub si, 3           ; subtract the 3 the call instruction probably took
 push cs
 pop ds              ; make ds the same as cs.  
 CALL PrintString   ;Call print string procedure
 JMP $      ;Infinite loop, hang it here.

此代码计算出正在运行的代码在运行时的偏移量,并确保 DS 指向同一段。除非另有说明,涉及 SI 的指令通常也使用 DS 作为其代码段来引用内存。

DS 是一个段寄存器,您可能想阅读诸如组装艺术之类的内容以了解更多信息。

Earlz 也在做同样的事情,只是确保寄存器正确,以便正确引用内存地址。只是他比我更了解引导扇区的细节。

于 2010-04-13T15:48:49.900 回答