操作系统实现-boot.asm实现

博客网址:www.shicoder.top
微信:18223081347
欢迎加群聊天 :452380935

这一次我们进入操作系统实现的真实编码, 这一次主要是完善对boot.asm文件的全部实现,开始吧。。。

首先我们先来理一下boot.asm需要干什么

  • 打印出Booting System...
  • 实现磁盘读写
  • 将后续的loader.asm所在的区域读入到0x1000处,然后跳转进入loader.asm程序
  • 开始执行loader.asm程序(这一节我们下次实现)

实模式下的print

在我们平时编写c语言时候,可以直接使用,但是在boot.asm中,完全就没有可以用库函数,因此为了在开始打印处start boot,我们需要自己实现print

先来看下代码把

mov si, booting
call print

print:
    mov ah, 0x0e
.next:
    mov al, [si]
    cmp al, 0
    jz .done
    int 0x10
    inc si
    jmp .next
.done:
    ret
booting:
    db "Booting System...", 10, 13, 0; \n\r

这段程序主要使用使用BIOS的int 10h来实现一个print功能,al寄存器存储要显示的字符串

磁盘读写

因为boot.asm在主引导扇区,磁盘内存太小,不能在boot.asm中实现loader.asm的功能,因此我们将loader.asm保存在磁盘的一个地方,在boot.asm中利用磁盘读的方式,将代码读入到内存的一个区域,然后跳转到那个地方

先来看下磁盘读的功能实现

; 函数参数
; edi 将磁盘内容读到哪里
; ecx 从磁盘哪一个扇区开始 
; bl 要读多少个扇区
read_disk:
    ; 设置读写扇区的数量
    ; 0x1f2 是硬盘控制端口,表示读写扇区的数量
    mov dx, 0x1f2
    mov al, bl
    ; 写端口用OUT指令 将al的值写入到dx端口
    out dx, al

    inc dx; 0x1f3 起始扇区前8位端口
    ; 因为ecx为起始扇区 
    ; ecx中的cl就是0-7位
    mov al, cl; 起始扇区前8位
    out dx, al

    inc dx; 0x1f4 起始扇区中8位端口
    shr ecx, 8 ;右移8位
    mov al, cl; 起始扇区中8位
    out dx, al

    inc dx; 0x1f5 起始扇区高8位端口
    shr ecx, 8 ;右移8位
    mov al, cl; 起始扇区高8位
    out dx, al

    inc dx ;0x1f6
    shr ecx, 8
    and cl, 0b1111 ;将高4位置为0,对应起始扇区的24-27位
    mov al,0b1110_0000 ;第4位为0,表示主盘,第6位为1,表示LBA,5-7位必须为1
    ; 将al和cl合二为一,放在al中
    or al, cl
    out dx, al

    inc dx ;0x1f7
    mov al, 0x20 ;表示读硬盘
    out dx, al

    xor ecx, ecx ;清空ecx
    mov cl, bl ;得到写扇区的数量

    ; loop指令会检查ecx是否为0 cl在ecx里面
    .read:
        push cx ;保存下,因为函数里面使用了
        call .waits ;等待数据准备完毕
        call .reads ;读取一个扇区
        pop cx ;恢复
        loop .read
    ret

    .waits:
        mov dx, 0x1f7 ;读0x1f7端口
        .check:
            in al, dx ;将dx端口的值放入al中
            jmp $+2 ;直接跳转到下一行 其实什么都没做,就是为了延迟一下
            jmp $+2
            jmp $+2

            and al, 0b1000_1000 ;获得al的第3位和第7位
            cmp al, 0b0000_1000 ;测试是否第7位为0,第3位为1 硬盘不繁忙,数据准备完毕
            jnz .check ;数据没准备好
        ret

    .reads:
        mov dx, 0x1f0 ;用于读写数据
        mov cx, 256 ;一个扇区256字节
        ; loop指定会检查ecx cx在ecx里面
        .readw:
            in ax, dx
            jmp $+2 ;直接跳转到下一行 其实什么都没做,就是为了延迟一下
            jmp $+2
            jmp $+2

            ; edi表示读取的目标内存
            mov [edi], ax
            ; 因为ax是16bit,2个字节,所以edi+2
            add edi, 2
            loop .readw
        ret

下面是磁盘的相关端口

Primary 通道 Secondary 通道 in 操作 out 操作
0x1F0 0x170 Data Data
0x1F1 0x171 Error Features
0x1F2 0x172 Sector count Sector count
0x1F3 0x173 LBA low LBA low
0x1F4 0x174 LBA mid LBA mid
0x1F5 0x175 LBA high LBA high
0x1F6 0x176 Device Device
0x1F7 0x177 Status Command
  • 0x1F0:16bit 端口,用于读写数据
  • 0x1F1:检测前一个指令的错误
  • 0x1F2:读写扇区的数量
  • 0x1F3:起始扇区的 0 ~ 7 位
  • 0x1F4:起始扇区的 8 ~ 15 位
  • 0x1F5:起始扇区的 16 ~ 23 位
  • 0x1F6:
    • 0 ~ 3:起始扇区的 24 ~ 27 位
    • 4: 0 主盘, 1 从片
    • 6: 0 CHS, 1 LBA
    • 5 ~ 7:固定为1
  • 0x1F7: out
    • 0xEC: 识别硬盘
    • 0x20: 读硬盘
    • 0x30: 写硬盘
  • 0x1F7: in / 8bit
    • 0 ERR
    • 3 DRQ 数据准备完毕
    • 7 BSY 硬盘繁忙

注意上面的out和in指令

读端口用IN指令,写端口用OUT指令

out a,b 将b的值写入到a端口

in a,b 将b端口的值读到a中

先来看4个起始扇区的寄存器 :0x1F30x1F40x1F50x1F6,假如此时的起始扇区ecx=123456789 ,即32位bit为00000111010110111100110100010101

  • 0-7位:00010101 => 0x1F3

  • 8-15位:11001101 => 0x1F4

  • 16-23位:01011011 => 0x1F5

  • 24-31位:00000111

    • 24-27位:0111 => 0x1F6(0-3)
    • mov al,0b1110_0000
      • 0 => 0x1F6(4) 表示主盘
      • 111 => ox1F6(5-7) 固定为1

再来看0x1F7,值为0x20,表示读磁盘

然后通过mov cl,bl,将扇区数量放在cl中,后面进行循环,汇编中循环的次数和ecx有关。因为是要读磁盘,因此需要先等待磁盘数据处理好,然后才进行读取,.wait便是这个作用,其余的相关解析可以通过代码注释看懂,这里就不赘述了

jmp $+2

可以通过反汇编看到

0000:jmp $+2
0002:xxx

所以这行代码就是跳到下一行,起到等待的作用

经过编写这个函数,我们就可以从磁盘中得到我们想要的代码啦,前面说过,我们本身就想将loader.asm代码放在磁盘的一个地方,然后再读进来,那怎么放呢,这样,我们先简单写一个loader.asm

loader.asm

代码如下

[org 0x1000]

; 打印字符串
mov si, loading
call print

; 阻塞
jmp $

print:
    mov ah, 0x0e
.next:
    mov al, [si]
    cmp al, 0
    jz .done
    int 0x10
    inc si
    jmp .next
.done:
    ret

loading:
    db "Loading System...", 10, 13, 0; \n\r

同样的,我们只是打印出一句话即可,那我们怎么将这些代码复制到磁盘中去呢,下面两行命令

nasm -f bin loader.bin loader.asm
dd if=loader.bin of=master.img bs=512 count=4 seek=2 conv=notrunc

利用dd命令,将bin文件从偏移为2的地方,写入4个到master.img中,这样就可以知道loader.bin在磁盘哪里,就可以读入了

boot.asm中代码如下

; 因为loader.bin是从第2个扇区开始写入,写了4个扇区
mov edi, 0x1000;读取的目标内存
mov ecx, 2 ;起始扇区
mov bl, 4 ;扇区数量
call read_disk

经过上面一番折腾,终于从boot跳转到loader中了,后续我们将对loader.asm进行完善,实现loader所需要的功能,下次见啦。。。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容