6

对于一个项目,我想直接从 DOS 调用第一个硬盘上的 MBR。我编写了一个小型汇编程序,它在 0:7c00h 将 MBR 加载到内存中,并跳到了很远的地方。我已将我的实用程序放在(DOS)可引导软盘上。我正在尝试引导的磁盘(HD0,0x80)上有一个 TrueCrypt 引导加载程序。当我在此设置中运行该工具时,它会显示 TrueCrypt 屏幕,但在输入密码后它会导致系统崩溃。当我在普通的 WinXP 机器上运行我的小实用程序 (w00t.com) 时,它似乎立即崩溃了。

显然我忘记了 BIOS 通常会做的一些关键事情,我猜这是微不足道的。有更好的裸机 DOS 和 BIOS 经验的人可以帮助我吗?

这是我的代码:

.MODEL tiny
.386
_TEXT SEGMENT USE16

INCLUDE BootDefs.i

ORG 100h

start:
    ; http://vxheavens.com/lib/vbw05.html
    ; Before DOS has booted the BIOS stores the amount of usable lower memory 
    ; in a word located at 0:413h in memory. We going to erase this value because
    ; we have booted dos before loading the bootsector, and dos is fat (and ugly).

    ; fake free memory  
    ;push ds
    ;push   0
    ;pop        ds
    ;mov        ax, TC_BOOT_LOADER_SEGMENT / 1024 * 16 + TC_BOOT_MEMORY_REQUIRED
    ;mov    word ptr ds:[413h], ax  ;ax = memory in K
    ;pop ds
    ;lea si, memory_patched_msg
    ;call print

    ;mov ax, cs
    mov ax, 0
    mov es, ax

    ; read first sector to es:7c00h (== cs:7c00)
    mov  dl, 80h
    mov  cl, 1
    mov  al, 1
    mov  bx, 7c00h ;load sector to es:bx
    call read_sectors

    lea si, mbr_loaded_msg
    call print

    lea si, jmp_to_mbr_msg
    call print

    ;Set BIOS default values in environment
    cli
    mov dl, 80h ;(drive C)
    xor ax, ax
    mov ds, ax
    mov es, ax
    mov ss, ax
    mov sp, 0ffffh
    sti

    push es
    push 7c00h
    retf            ;Jump to MBR code at 0:7c00h


    ; Print string
print:
    xor bx, bx
    mov ah, 0eh
    cld

@@: lodsb
    test al, al
    jz print_end

    int 10h
    jmp @B

print_end:
    ret

    ; Read sectors of the first cylinder
read_sectors:
    mov ch, 0           ; Cylinder
    mov dh, 0           ; Head
                        ; DL = drive number passed from BIOS
    mov ah, 2
    int 13h
    jnc read_ok

    lea si, disk_error_msg
    call print
read_ok:
    ret

memory_patched_msg      db 'Memory patched', 13, 10, 7, 0
mbr_loaded_msg          db 'MBR loaded', 13, 10, 7, 0
jmp_to_mbr_msg          db 'Jumping to MBR code', 13, 10, 7, 0
disk_error_msg          db 'Disk error', 13, 10, 7, 0

_TEXT ENDS
END start
4

3 回答 3

2

已编辑——新答案:

好的,好像我首先误解了你的问题。我能给出的唯一进一步建议是:

  • 检查您是否不加载HIMEM.SYS和/或EMM386.EXE(或任何其他内存管理器)。引导加载程序执行时CPU必须处于实模式。

  • 看看 Ralf Brown 的中断列表。如果我没记错的话,那里有一些关于启动过程的技术信息。它可能会给你一个提示。

  • 查看其他加载程序实用程序的源代码,例如loadlin. (它与您的实用程序不完全相同,但可能会给您一些见解。)


上一个答案:

ORG 100h在引导加载程序中做正确的事情真的是正确的吗?

我认为这仅与 DOS.com可执行文件相关,因为 DOS 将使用程序段前缀 (PSP) 初始化前 256 个字节。如果您编写引导加载程序,则没有 DOS,也没有 PSP 之类的东西。我想这必须是ORG 0

于 2010-04-14T17:43:12.000 回答
2

好的,我的 DOS 知识非常生疏,我没有时间测试/验证我的答案,但我想你的问题如下:

当启动 DOS 或任何其他操作系统时,它们将更改中断表。DOS 将更改中断表,例如,中断 20 可用于向 DOS“内核”发送命令。他们通过保存原始中断处理程序,用他们自己的处理程序替换它来做到这一点,然后,作为默认的后备,如果他们不知道如何处理中断,则链接到原始中断处理程序。通过这种方式,他们将新功能“添加”到现有的 bios 功能中,并且在 DOS 下运行的每个程序都可以通过设置一些寄存器然后调用中断来使用系统调用。

但是,当您启动一个新操作系统时,这个新操作系统将假定 a) 所有中断都由 bios 处理并且 b) 所有内存都是空闲/未使用的,除非该 bios 报告正在使用。

因此,新操作系统将覆盖旧操作系统当前正在使用的内存,然后它会在某个时候调用其中一个中断,并在无效内存中执行某些操作,然后您的计算机将崩溃。

因此,将您的中断表重置为原始的 bios 版本,您应该没问题...

于 2010-10-31T23:14:47.333 回答
0

我不认为这是一个引导加载程序,它是加载引导扇区并尝试执行它的 .com 文件。所以它在 DOS 初始化后运行。

于 2010-04-14T18:30:06.243 回答