Bootstrap

王爽《汇编语言》课程设计二续(包含软盘操作)

之前完成了课程设计的第一部分(CMOS的读取与修改等):王爽《汇编语言》课程设计二(只包含CMOS RAM操作不包含软盘操作)
但是由于没有软盘做不了第二部分(脱离操作系统的引导程序),今天我们完善一下,将第二部分补上去,首先得解决软盘的问题:关于VMWare虚拟机中软盘问题

1、题目描述与分析:

开机后,CPU自动进入到FFFF:0单元处执行,此处有一条跳转指令。CPU执行该指令以后,转去BIOS中的硬件系统检测和初始化程序。初始化程序建立BIOS所支持的终端向量,即将BIOS提供的中断例程的入口地址登记在中断向量表中。
硬件检测和初始化完成以后,调用int 19H进行操作系统的引导。
如果设为从软盘启动操作系统,则int 19H将主要完成以下工作:
(1)、控制0号软驱(A:),读取软盘0面0道1扇区的内容到0:7C00
(2)、将CS:IP指向0:7C00处。
即软盘的0面0道1扇区中装有操作系统的引导程序,则int 19H将其装载到0:7C00处后(程序要执行就必须装载到内存,引导程序也不列外),设置CPU从0:7C00开始执行此处的引导程序,操作系统被激活,控制计算机。
如果0号软驱中没有软盘,或者发生软盘I/O错误,则int 19H将主要完成以下工作:
(1)、读取硬C的盘0面0道1扇区的内容到0:7C00
(2)、将CS:IP指向0:7C00处。
即直接从C盘装载引导程序到内存,然后开始引导激活操作系统。

但是我们知道(在虚拟机中装软盘时知道),如果软盘存在,但是软盘中没有引导程序,则发生如下问题(需要把软盘去掉后重启):
这里写图片描述

在解决软盘的问题以后,我们来分析如何实现编写一个可以自行启动计算机,不需要在现有操作系统环境中运行的程序,但是此时BIOS的中断例程已经映射到中断向量表中(即该程序不是由操作系统装载的而是BIOS的int 19H中断例程装载的),要编写的程序处于一个中间状态。

2、问题解决方案分析:

3.5英寸软盘分为上下两面,每个面有80个磁道,每个磁道又分为18个扇区,每个扇区的大小为512字节:
2面*80磁道*18扇区*512Byte=1440KB≈1.44MB(=1.40625MB)

我们用已经映射的BIOS中断例程int 13H来操作磁盘细节。而本来int 19H读取软盘0面0道1扇区到内存中去然后引导操作系统启动,现在我们要将自己写的小型系统放到0面0道1扇区中去,那么int 19H在装载0面0道1扇区的内容时,就将我们的程序装载进去然后执行。但是有一个问题:若果程序量大于512字节,一个扇区放不下怎么办?那我们就在软哦按0面0道1扇区中放复制磁盘中占多个扇区的程序到内存的复制程序程序,而int 19H将其装载到内存,其又将目的程序从多个扇区装载到内存,并设置CS:IP在目的程序的入口处即可。至于要将代码复制到磁盘上去就是为了在断电重启后代码能重新加载进内存运行(存盘的重要性:所有系统程序不是本来就在内存中,而是在磁盘上,开机后才一个一个加载进内存的)。

那么整个思路如图所示:
这里写图片描述

与具体代码配合参详:
这里的代码就是将只含有CMOS的第一部分添加上上图的五个部分,并补全进入键盘中断选择后的两个功能的代码(重启与引导操作系统),并且将原来的复制boot~boot_end的代码到0:7E00的功能去掉(其实是代换了:原来是直接加载进内存,现在先存盘再在启动时从磁盘上加载进内存)。

;201751412:42:24
;Mail:mailbox_krj@163.com
;Author:KangRuoJin
;Version:v2.1

;int 19H负责将软盘001扇区的512字节复制到0:7C00H处,并且设置CS:IP=0:7C00
;而我们在软盘001扇区中写入我们的代码
;其功能为将软盘002扇区开始的几个扇区复制到0:7E00H,并设置CS:IP=0:7E00
;注意:7E00H-7C00H=200H=2*16*16=512Byte即一个扇区的大小

;先将offset boot_end - offset boot复制到软盘上002扇区开始的几个扇区
;再将“复制几个扇区到0:7E00H”的功能代码复制到软盘001扇区
;注意:如果功能代码一个扇区就可以装下,就不必用001扇区做“中介”,而直接存放功能代码

;关于扇区个数(或字节数),可以在start之后设置栈寄存器以后,mov bx,offset boot_end - offset boot
;然后debug -u查看一下mov到bx的值 

assume cs:code,ds:data,ss:stack

data segment
    db 256 dup (0)
data ends
stack segment
    db 128 dup (0)
stack ends

code segment
start:
    mov bx,stack
    mov ss,bx
    mov sp,128

    call copylead_to_softdisk   ;将复制程序复制到一扇区
    call copyboot_to_softdisk   ;将功能程序复制到二扇区开始的几个扇区

mov ax,4C00H
int 21H
;===========================复制程序的内容======================
lead:
    ;设置栈
    mov bx,0
    mov ss,bx
    mov sp,7C00H

    ;将磁盘中002扇区开始的内容复制到0:7E00H
    call copyboot_from_softdisk

    ;从内存0:7E00H开始启动
    mov bx,0
    push bx
    mov bx,7E00H
    push bx
    retf    
ret
;================================================================
copyboot_from_softdisk:
    mov bx,cs
    mov es,bx
    mov bx,7E00H    ;设置数据流动的目标地址es:bx

    mov al,2    ;扇区个数
    mov ah,2    ;读取
    mov cl,2    ;扇区起始号
    mov ch,0    ;磁道
    mov dl,0    ;驱动器A
    mov dh,0    ;面号
    int 13H
ret
;================================================================
copylead_to_softdisk:
    mov bx,cs
    mov es,bx
    mov bx,offset lead  ;设置数据流动的源地址es:bx

    mov al,1    ;1个扇区
    mov ah,3    ;写
    mov ch,0    ;0磁道
    mov cl,1    ;1扇区
    mov dh,0    ;0面
    mov dl,0    ;软驱A
    int 13H 
ret
;================================================================
copyboot_to_softdisk:
    mov bx,cs
    mov es,bx
    mov bx,offset boot  ;设置数据流动的源地址es:bx

    mov al,2    ;2个扇区
    mov ah,3    ;写
    mov ch,0    ;0磁道
    mov cl,2    ;2扇区
    mov dh,0    ;0面
    mov dl,0    ;软驱A
    int 13H 
ret
;================================================================
boot:
    jmp boot_start
;*******************************************************************************************
    MENU_1  db 'Weclome to System of Mr.Kang',0
    MENU_2  db '(1) Reset PC',0
    MENU_3  db '(2) Start System',0
    MENU_4  db '(3) ShowClock',0
    MENU_5  db '(4) Set Clock',0
    MENU_6  db 'Please Enter Your Choose(1~4):',0

    MENU_ADDRESS    dw offset MENU_1 - offset boot + 7E00H
                    dw offset MENU_2 - offset boot + 7E00H
                    dw offset MENU_3 - offset boot + 7E00H
                    dw offset MENU_4 - offset boot + 7E00H
                    dw offset MENU_5 - offset boot + 7E00H
                    dw offset MENU_6 - offset boot + 7E00H
    time    db 'YY/MM/DD hh:mm:ss',0
    cmos    db 9,8,7,4,2,0
    clockone    db 'press F1 to change the color, press ESC to return',0
    clock2  db 'Please input Date and Time,(YY MM DD hh mm ss):',0
    change  db 12 dup ('*'),0
;*******************************************************************************************
boot_start:
    call init_reg
    call clear_screen
    call show_menu
    jmp choose_option
mov ax,4C00H
int 21H
;================================================================
choose_option:
    call clear_keyboard_buffer  ;清除键盘缓冲区
    mov ah,0
    int 16H

    cmp al,'1'
    je choose_option_one
    cmp al,'2'
    je choose_option_two
    cmp al,'3'
    je choose_option_three
    cmp al,'4'
    je choose_option_four
    jmp choose_option
;================================
choose_option_one:    
    call ResetPC           
jmp choose_option                

choose_option_two: 
    call StartSystem
jmp choose_option                

choose_option_three:             
    call ShowClock               
jmp boot_start          ;跳回到boot_start清屏,并重新显示                

choose_option_four:    
    call SetClock
jmp boot_start   
;=================================================================
ResetPC:
    mov bx,0FFFFH   ;重新引导
    push bx
    mov bx,0
    push bx
    retf
ret
;=================================================================
StartSystem:
    mov bx,0            ;将C盘(驱动80H)的000扇区复制到0:7C00H处
    mov es,bx
    mov bx,7C00H

    mov al,1    ;1个扇区
    mov ah,2    ;读
    mov ch,0    ;0磁道
    mov cl,1    ;1扇区
    mov dh,0    ;0面
    mov dl,80H  ;C盘驱动
    int 13H 

    mov bx,0            ;从0:7C00开始执行,启动引导操作系统
    push bx 
    mov bx,7C00H
    push bx
    retf
ret  
;=================================================================
SetClock:
    call init_reg           ;寄存器初始化
    call clear_screen       ;清屏并重新画边框
    call clear_change_stack ;清空存放修改内容的栈缓冲区

    ;设置显示提示信息的位置为1015列开始
    mov di,160*10+15*2
    mov si,offset clock2 - offset boot + 7E00H
    call show_line
    ;显示修改缓冲区中的内容
    mov di,160*12+30*2      
    mov si,offset change - offset boot + 7E00H
    call show_line

    call get_string
    call set_CMOSRAM
ret
;=================================================================
set_CMOSRAM:
    mov bx,offset cmos - offset boot + 7E00H
    mov si,offset change - offset boot + 7E00H
    mov cx,6

    ChangeMessage:   ;GetMessage的逆过程
        push cx

        mov dx,ds:[si]
        sub dx,3030H
        mov cl,4
        shl dl,cl
        and dh,00001111B
        or dl,dh

        mov al,ds:[bx]
        out 70H,al    ;将al送入地址端口70h
        mov al,dl
        out 71H,al ;    将数据写入CMOSRAM时钟

        inc bx
        add si,2
        pop cx
    loop ChangeMessage
ret
;================================================================= 
get_string:
    mov si,offset change - offset boot + 07E00H
    mov bx,0
getSting:
    mov ah,0
    int 16H
    cmp al,'0'  ;判断是否是小于‘0’
    jb noNumber
    cmp al,'9'  ;判断是否大于'9'
    ja noNumber
    call char_push
    mov di,160*12+30*2  ;与SetClock中,显示修改缓冲区中的内容位置相对应
    call show_line
    jmp getSting
noNumber:
    cmp ah,0EH  ;退格通码
    je BackspaceKey
    cmp ah,1CH  ;回车通码
    je EnterKey
    jmp getSting
BackspaceKey:
    call char_pop
    mov di,160*12+30*2  ;与SetClock中,显示修改缓冲区中的内容位置相对应    
    call show_line
    jmp getSting
EnterKey:

ret
;=================================================================
char_push:
    cmp bx,12
    ja char_push_over
    mov ds:[si+bx],al
    inc bx
    char_push_over:
ret
char_pop:
    cmp bx,0
    je char_pop_over
    dec bx
    mov byte ptr ds:[si+bx],'*'
    char_pop_over:
ret
;=================================================================      
clear_change_stack:
    push bx
    push cx

    mov bx,offset change - offset boot + 7E00H
    mov cx,6            
    ClearChangeStack:
        mov word ptr ds:[bx],'**'
        add bx,2
    loop ClearChangeStack

    pop cx
    pop bx
ret
;=================================================================
ShowClock:
    call init_reg           ;寄存器初始化
    call clear_screen       ;清屏并重新画边框
    mov di,14*160+15*2      ;设置显示提示信息的位置为1415列开始
    mov si,offset clockone - offset boot + 7E00H
    call show_line  ;显示

    ShowClockStart:
        call GetTimeMessage     ;获取时间信息放到time中去
        mov di,10*160+30*2      ;设置显示时间与日期的位置为1030列开始
        mov si,offset time - offset boot + 7E00H
        call show_line          ;按规则显示

        ;这里的读取键盘缓冲区是自己完成,我们亦可以在
        mov ah,1        ;非阻塞式读取,由于是非阻塞读取,当读取到之后,缓冲区不会清空
        int 16H
        jz ShowClockStart
        cmp ah,3BH
        je ChangeColor
        cmp al,1BH      ;ESC扫描码为1,与设定的功能号有冲突,所以用ASCII码
        je ReturnToMainMenu
        cmp al,0        ;能到达这一步说明不为F1也不为ESC
        jne ClearKeyBuffer  ;但是不为0说明一定是有键盘中断产生,就要清除无用的键盘中断
    jmp ShowClockStart
;================================
ChangeColor:                    ;改变颜色
    call ChangeShowColor
ClearKeyBuffer:
    call clear_keyboard_buffer  ;清除非阻塞读取时未清空的缓冲区
    jmp ShowClockStart
ReturnToMainMenu:               ;退回到choose_option时会清除缓冲区,所以这里不需要清除缓冲区
ret
;================================================================
GetTimeMessage:
    mov cx,6
    mov bx,offset cmos - offset boot + 7E00H
    mov si,offset time - offset boot + 7E00H
    GetMessage:   
        push cx
        mov al,ds:[bx]

        out 70H,al    ;将al送入地址端口70h
        in al,71H ;从数据端口71h处读出单元内容
        mov ah,al
        mov cl,4
        shr al,cl ;右移4and ah,0FH    ;al分成两个表示BCD码值的数据
        add ax,3030H  ;BCD码+30h=10进制数对应的ASCII码

        mov word ptr ds:[si],ax     ;ASCII码写入time段
        inc bx
        add si,3
        pop cx
    loop GetMessage
ret
;================================================================
ChangeShowColor:
    push bx
    push cx

    mov cx,2000
    mov bx,1
    loopChangeShowColor:
        add byte ptr es:[bx],1
        or byte ptr es:[bx],00001000B       ;高亮不闪烁设置
        and byte ptr es:[bx],10001111B      ;背景一直为黑色
        add bx,2
    loop loopChangeShowColor

    pop cx
    pop bx
ret
;================================================================
clear_keyboard_buffer:
    mov ah,1
    int 16H         ;修改了ax,所以ZF可能会改变

    jz ClearKeyboardBufferOver  ;如果非阻塞读取的数据为0则结束
    mov ah,0        ;否则,则继续用阻塞的方式清除一次缓冲区字节
    int 16H
    jmp clear_keyboard_buffer
    ClearKeyboardBufferOver:
ret
;================================================================
init_reg:
    mov bx,0B800H
    mov es,bx
    mov bx,0
    mov ds,bx
ret
;================================================================
clear_screen:
    mov bx,0
    mov cx,2000
    mov dx,1F00H    ;黑屏白字
    ClearScreen:
        mov es:[bx],dx  
        add bx,2
    loop ClearScreen

    mov bx,6*160+10*2
    mov cx,60
    clear_showframe_UpandDown:
        mov byte ptr es:[bx],'#'
        mov byte ptr es:[bx+160*9],'#'
        add bx,2
    loop clear_showframe_UpandDown

    mov bx,7*160+10*2
    mov cx,8
    clear_showframe_LeftandRight:
        mov byte ptr es:[bx],'#'
        mov byte ptr es:[bx+59*2],'#'
        mov byte ptr es:[bx+2],'#'
        mov byte ptr es:[bx+59*2-2],'#'
        add bx,160
    loop clear_showframe_LeftandRight

ret
;================================================================
show_menu:
    mov bx,offset MENU_ADDRESS - offset boot + 7E00H
    mov di,160*8 + 25*2
    mov cx,6
    ShowMenu:   
        mov si,ds:[bx]  ;获取输出行信息的偏移地址
        call show_line  ;给定显示的位置显示一行
        add bx,2        ;获取下一条内容的地址
        add di,160      ;输出到下一行
    loop ShowMenu
ret
;================================================================
show_line:
    push ax
    push si
    push di
ShowLineStart:
    mov al,ds:[si]  ;获取一个字符
    cmp al,0        ;遇到0则输出结束
    je ShowLineOver
    mov es:[di],al  ;遇到的不是0则输出到显示屏上
    add di,2
    inc si
    jmp ShowLineStart
ShowLineOver:
    pop di
    pop si
    pop ax
ret
;================================================================
boot_end:nop
;================================================================
code ends
end start

我们把功能一(重启)与功能二(从C盘0面0道1扇区引导现有操作系统)的代码单独拿出来分析:

;===================重启:只需要让CS:IP从FFFF:0重新执行即可==================
ResetPC:
    mov bx,0FFFFH   ;重新引导
    push bx
    mov bx,0
    push bx
    retf
ret
;======引导操作系统:将C盘(80H)的引导程序加载进内存0:7C00处后设置CS:IP从0:7C00处开始执行=======
StartSystem:
    mov bx,0            ;将C盘(驱动80H)的0面0道0扇区复制到0:7C00H处
    mov es,bx
    mov bx,7C00H

    mov al,1    ;1个扇区
    mov ah,2    ;读
    mov ch,0    ;0磁道
    mov cl,1    ;1扇区
    mov dh,0    ;0面
    mov dl,80H  ;C盘驱动
    int 13H 

    mov bx,0            ;从0:7C00开始执行,启动引导操作系统
    push bx 
    mov bx,7C00H
    push bx
    retf
ret  

结果:开机后直接进入我们写的界面(撒花~~),并能实现相关功能(缺点:正确按键组合没有问题,但是有个别键的穿插输入会影响正确的按键输入)。
这里写图片描述

;