编写一个读取用户名称的简单引导加载程序

时间:2017-04-06 20:32:23

标签: assembly x86 operating-system bootloader minix

我正在尝试了解操作系统的工作原理。这是一项简单的任务,我很难解决问题:编写一个简单的引导加载程序,提示用户输入他的名字并打印一条欢迎信息,例如" hello,>> name<<& #34; - 之后,它什么也没做。

我与minix 3一起运行qemu,如果这有任何相关性的话。我只需将asm文件和dd的前512个字节编译为/dev/c0d0minix的虚拟硬盘驱动器)。

我可以打印邮件并打印用户输入的内容。但是,我之后没有设法打印用户的名字。

这是我的汇编代码:

[bits 16]
[org 0x7c00]

mov si, HelloString
call print_string
mov di, name
call read_name
mov si, name
call print_string

read_name:
    read_char:
        mov ah, 0h  ; read character from keyboard
        mov [di], ah    ; save it in the buffer
        inc di      ; next char
        int 0x16    ; store it in AL
        cmp ah, 0x0d    ; check for enter
        je stop_reading 
        mov ah, 0eh     ; display character in AL
        int 0x10    ; echo it
        jmp read_char   ; an so on
    stop_reading:
        mov si, EoL
        call print_string
        ret

print_char:
    mov ah, 0x0e    ; one char
    mov bh, 0x00    ; page number
    mov bl, 0x07    ; font color
    int 0x10
    ret

print_string:
    next_char:
        mov al, [si]
        inc si
        or al, al
        jz exit_function
        call print_char
        jmp next_char
    exit_function:
        ret

;data
HelloString db 'Enter your name', 0xd, 0xa, 0
name times 20 db 0
EoL db 0xd, 0xa, 0

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

我做错了什么?我会很感激一些提示和文献

1 个答案:

答案 0 :(得分:4)

您的代码存在许多问题。罗斯和我在评论中指出了一些内容。你应该阅读我的General Bootloader Tips。虽然与您的实际问题无关,但您应将 DS (以及 ES ,如果您最终需要它)设置为0,因为您使用的原点为0x7c00({{1} })。您还应该在您知道代码不会破坏的地方设置堆栈。我之前将此代码添加到顶部:

org 0x7c00

将其更改为:

    mov si, HelloString
    call print_string

代码运行完毕后,应将CPU置于无限循环中,以便通过在主代码下执行功能来继续运行。所以在标签 xor ax, ax ; AX=0 mov ds, ax mov es, ax mov ss, ax ; SS=ES=DS=0 mov sp, 0x7c00 ; Place stack before the bootloader. Grows down from 0x0000:0x7c00 mov si, HelloString call print_string 之前放置一个无限循环。这样的事情是典型的:

read_name:

您的 cli ; Turn off interrupts endloop: hlt ; Halt processor until next interrupt encountered jmp endloop ; Jump back just in case we get an MNI (non-maskable interrupt) 功能中存在一些错误。 BIOS中断信息的最佳位置之一是Ralph Brown's Interrupt ListInt 0x16/AH=0记录为:

read_char

您应该使用 AL 中的ASCII字符存储到字符串缓冲区中。您还应该将 AL 与0x0d进行比较,而不是 AH (这是键盘扫描码,而不是ASCII字符)。在使用AH = 00h Return: AH = BIOS scan code AL = ASCII character 读取字符之前,您还要将数据存储到字符串缓冲区中。你需要把它们放在缓冲区之后。当你到达int 0x16时,你会想要在缓冲区的末尾放置一个NUL(0x00)字符。

stop_reading:的代码可能如下:

read_name

修改后的引导加载程序可能如下所示:

read_name:
read_char:
    mov ah, 0h  ; read character from keyboard
    int 0x16    ; store it in AL
    cmp al, 0x0d    ; check for enter
    je stop_reading
    mov [di], al    ; save it in the buffer
    inc di      ; next char
    mov ah, 0eh     ; display character in AL
    int 0x10    ; echo it
    jmp read_char   ; an so on
stop_reading:
    mov byte [di], 0x00    ; NUL terminate buffer
    mov si, EoL
    call print_string
    ret

我强烈建议使用BOCHS来调试引导加载程序。它有一个内置的调试器,可以理解实模式和实模式寻址,并且比 QEMU

更适合调试引导加载程序