• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            天衣有縫

            冠蓋滿(mǎn)京華,斯人獨(dú)憔悴~
            posts - 35, comments - 115, trackbacks - 0, articles - 0
               :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

            1課:引導(dǎo)程序

            聲明:轉(zhuǎn)載請(qǐng)保留

            譯者http://www.shnenglu.com/jinglexy

            原作者:xiaoming.mo at skelix dot org

            MSN & Email: jinglexy at yahoo dot com dot cn


            目標(biāo):使"system"從軟盤(pán)啟動(dòng),并打印"Hello World!" 下載源程序

            內(nèi)存尋址

            處理器以字節(jié)管理和訪問(wèn)內(nèi)存,每個(gè)字節(jié)都有獨(dú)立的地址,即物理地址。有兩種地址映射方式:分段和分頁(yè),skelix內(nèi)核中都用到了。


            段對(duì)于我們來(lái)說(shuō)再熟悉不過(guò)了,先回顧一下dos時(shí)期的段吧。它是一個(gè)16位的寄存器,所以最多可以直接訪問(wèn)2^16字節(jié)的內(nèi)存,即64K。這對(duì)應(yīng)用程序來(lái)說(shuō)太少了,于是Intel使用Segment:Offset結(jié)合方式來(lái)表示一個(gè)虛擬地址。段寄存器左移4位加上偏移就得到實(shí)際的物理地址了。例如,0x7c00:0x0189表示物理地址0x7c189,而不是0x7c000189。計(jì)算過(guò)程如下:

             

             7C000
            + 0189
            -------
             7C189

            現(xiàn)在我們來(lái)計(jì)算最大可以訪問(wèn)的地址:FFFF:FFFF

             FFFF0
            + FFFF
            -------
            10FFEF

            這個(gè)范圍是1M + 65519 bytes, 因?yàn)樵?span lang="EN-US">80386中使用了20位地址線,所以可以額外多訪問(wèn)65519個(gè)字節(jié)虛擬地址,例如地址0x100010被映射到地址0x10,訪問(wèn)這兩個(gè)地址是等價(jià)的。

            表示同一個(gè)物理地址有多種方式,例如07C0:00000000:7C00 就是一樣的。

             

            另一個(gè)概念是線性地址,這個(gè)是32位地址,只有當(dāng)分頁(yè)機(jī)制開(kāi)啟時(shí)才有效,文章后面會(huì)提到它。

             

            引導(dǎo)過(guò)程

             

            當(dāng)系統(tǒng)上電或RESET時(shí),處理器將執(zhí)行一些列的初始化,寄存器被設(shè)置成非預(yù)知狀態(tài),并且cpu處于實(shí)模式。也許你想知道cpu是怎樣設(shè)置segment:offset為物理地址FFFF0的(0xf000:0xfff0就是bios入口地址),這是因?yàn)?span lang="EN-US">cs寄存器有一個(gè)非可見(jiàn)部分,它保存了ffff:0000地址,并且cs在初始化時(shí)會(huì)被裝入f000值。此后以正常方式使用它。當(dāng)bois取得控制權(quán)后,根據(jù)用戶(hù)配置(從軟驅(qū),硬盤(pán),或cdrom)中讀取第一個(gè)sector00007C00,并跳轉(zhuǎn)到該地址執(zhí)行(就是引導(dǎo)程序bootstrap)。在bootstrap中我們可以使用bios中斷,但是進(jìn)入kernel后就不能再使用了。

             


            程序一:使用asld的范例

             

            你可以在下載源程序的01/first.cry/bootsect.s

             

                    .text              .text表示代碼段
                    .globl             start
            表示start可以用作外部符號(hào)
                    .code16            GCC
            默認(rèn)使用32位地址和操作數(shù),這里告訴它使用16
            start:
                    jmp      start    
            死循環(huán)

            .org    0x1fe,   0x90      .org NEW-LC, FILL說(shuō)明:這里填充0x90,是nop指令的機(jī)器碼
            .word   0xaa55

             

            講解:.org指令指示下一個(gè)數(shù)據(jù)地址,為了編譯這個(gè)程序,我們寫(xiě)了一個(gè)Makefile,總不能老是敲命令吧,呵呵。

            網(wǎng)絡(luò)上可以找到很多寫(xiě)Makefile的資料,編譯選項(xiàng)才是我們關(guān)注的焦點(diǎn)。


            01/first.cry/Makefile

            AS=as                     gcc匯編工具
            LD=ld                     gcc
            連接器

            .s.o:
                ${AS} -a $< -o $*.o >$*.map

            all: final.img

            final.img: bootsect
                mv bootsect final.img

            bootsect: bootsect.o
                ${LD} --oformat binary -N -e start -Ttext 0x7c00 -o bootsect $<

             

            講解:ld可以被配置為支持多于一種的目標(biāo)文件. binary表示沒(méi)有程序頭和其他信息,僅僅是一些裸數(shù)據(jù)。如果沒(méi)有這個(gè)選項(xiàng),將被默認(rèn)鏈接為elf格式。-Ntextdata節(jié)設(shè)置為可讀寫(xiě)。-Ttexttext節(jié)起始地址設(shè)置為0x7c00(在jmp和數(shù)據(jù)引用等重定位鏈接時(shí)會(huì)用到這個(gè)參考值),所有的引用地址都是在7c00這個(gè)地址上加出來(lái)的。-e選項(xiàng)指定程序入口點(diǎn)

            現(xiàn)在我們運(yùn)行make指令編譯一下:

            [root@root~/source/os/skelix/01/first.cry]$ ls
            bootsect.s  COPYING  Makefile
            [root@root~/source/os/skelix/01/first.cry]$ make
            as -a bootsect.s -o bootsect.o >bootsect.map
            ld --oformat binary -N -e start -Ttext 0x7c00 -o bootsect bootsect.o
            mv bootsect final.img
            [root@root~/source/os/skelix/01/first.cry]$ ls
            bootsect.map  bootsect.o  bootsect.s  COPYING  final.img  Makefile
            [root@root~/source/os/skelix/01/first.cry]$

             

            現(xiàn)在,我們啟動(dòng)vmware,運(yùn)行,載入軟驅(qū)映象文件"final.img",我們得到一個(gè)黑屏,這是正確的,因?yàn)槲覀兪裁匆矝](méi)有做。

             

            程序一:顯示 Hello World!

            好了,上面的黑屏程序并不是太好玩,現(xiàn)在我們嘗試在上面打印"Hello World!"

            01/hello.world/bootsect.s

                    .text
                    .globl  start
                    .code16
            start:
                    jmp     code
            msg:                                  
            使用jmp指令跳過(guò)該變量,這是我們?yōu)槭裁丛?span lang="EN-US">Makefile使用-N鏈接選項(xiàng)了
                    .string "Hello World!\x0"
            code:
                    movw    $0xb800,%ax
                    movw    %ax,    %es            es
            段設(shè)置成B800,如前所述,segment:offset地址映射方式,它指向B8000

                                                   這意味著第一個(gè)字節(jié)地址是0(映射到B8000),屬性字節(jié)是1(映射到B8001

                                                   B8001值設(shè)置為0x07可以將這個(gè)byte顏色設(shè)置為黑底白字。
                    xorw    %ax,    %ax
                    movw    %ax,    %ds

                    movw    $msg,   %si            movsb指令設(shè)置正確的sidi
                    xorw    %di,    %di
                    cld
                    movb    $0x07,  %al            
            字的顏色

            1:
                    cmp     $0,    (%si)
                    je      1f   
                    movsb
                    stosb
                    jmp     1b
            1:      jmp     1b

            .org    0x1fe,  0x90
            .word   0xaa55

             

             

            Feedback

            # re: 自己動(dòng)手寫(xiě)內(nèi)核(第1課:引導(dǎo)程序)(原創(chuàng))  回復(fù)  更多評(píng)論   

            2007-06-19 02:15 by 路西菲爾
            問(wèn)個(gè)弱智問(wèn)題我怎么不能加載引導(dǎo)程序.
            我用vm版本是5.5.3,找不到你文中說(shuō)的"啟動(dòng)vmware,運(yùn)行,載入軟驅(qū)映象文件".我只能在虛擬機(jī)的屬性設(shè)置中找到cd-rom -> 使用ISO鏡像.不過(guò)設(shè)置以后沒(méi)有效果.

            # re: 自己動(dòng)手寫(xiě)內(nèi)核(第1課:引導(dǎo)程序)(原創(chuàng))  回復(fù)  更多評(píng)論   

            2007-06-19 17:00 by 天衣有縫
            “編輯虛擬機(jī)設(shè)置” --> 選項(xiàng)卡:“硬件” --> “添加” --> “軟盤(pán)驅(qū)動(dòng)器” ,然后選擇設(shè)備設(shè)置,“使用軟磁盤(pán)映象”

            # re: 自己動(dòng)手寫(xiě)內(nèi)核(第1課:引導(dǎo)程序)(原創(chuàng))  回復(fù)  更多評(píng)論   

            2007-06-19 17:37 by 路西菲爾
            感謝你的回復(fù)解決了我學(xué)習(xí)道路上的第一個(gè)問(wèn)題

            # re: 自己動(dòng)手寫(xiě)內(nèi)核(第1課:引導(dǎo)程序)(原創(chuàng))  回復(fù)  更多評(píng)論   

            2007-06-19 17:52 by 天衣有縫
            不客氣,如果有什么心得可以郵件聯(lián)系,這樣回復(fù)可能會(huì)快些:)

            # re: 自己動(dòng)手寫(xiě)內(nèi)核(第1課:引導(dǎo)程序)(原創(chuàng))  回復(fù)  更多評(píng)論   

            2007-12-19 10:35 by 上海大眾搬場(chǎng)
            我也學(xué)習(xí)了.謝謝

            # re: 自己動(dòng)手寫(xiě)內(nèi)核(第1課:引導(dǎo)程序)(原創(chuàng))[未登錄](méi)  回復(fù)  更多評(píng)論   

            2009-02-06 07:42 by a
            我make沒(méi)通過(guò)!
            錯(cuò)誤信息:(環(huán)境:cygwin(您老人家的那個(gè)版本,MinGW的as和ld+MinGW的Make))
            zjs@ccb-zz /cygdrive/f/sys
            $ make
            ld --oformat binary -N -e start -Ttext 0x7c00 -o bootsect bootsect.o
            ld: PE operations on non PE file.
            f:\mingw\bin\make.exe: *** [bootsect] Error 1
            /cygdrive/f/代表F盤(pán)(/dev/hda4)
            我使用網(wǎng)上一個(gè)AS+LD(CYG編譯版)錯(cuò)誤更多!
            錯(cuò)誤:(make還是MinGWmake)
            zjs@ccb-zz /cygdrive/f/sys
            $ /usr/bin/make
            ld --oformat binary -N -e start -Ttext 0x7c00 -o bootsect bootsect.o
            usage: ld [-03Mimrstz[-]] [-llib_extension] [-o outfile] [-Ccrtfile]
            [-Llibdir] [-Olibfile] [-T textaddr] infile...
            F:\cygwin\bin\make.exe: *** [bootsect] Error 2
            精品午夜久久福利大片| 一本一本久久aa综合精品| 久久精品人成免费| 久久99精品久久久久婷婷| 久久综合国产乱子伦精品免费| 国产精品美女久久久久久2018| 久久九九有精品国产23百花影院| 亚洲乱亚洲乱淫久久| 久久久久亚洲av毛片大| 天堂久久天堂AV色综合| 精品国产综合区久久久久久| 97精品伊人久久久大香线蕉| 日本精品久久久久中文字幕| 亚洲色婷婷综合久久| 久久久久国产| 情人伊人久久综合亚洲| 亚洲AV日韩精品久久久久久| 国产精品无码久久四虎| 久久er99热精品一区二区| 伊人久久大香线蕉精品不卡| 国产精品欧美亚洲韩国日本久久 | 久久性生大片免费观看性| 亚洲中文精品久久久久久不卡| 欧美激情精品久久久久| 久久久久亚洲av无码专区喷水| 久久福利资源国产精品999| 精品久久久久久久久久久久久久久| 日韩AV无码久久一区二区 | 欧美丰满熟妇BBB久久久| 久久亚洲国产成人影院网站 | 天天做夜夜做久久做狠狠| 久久久久久久综合日本亚洲| 国产成年无码久久久久毛片| 日韩久久久久久中文人妻| 亚洲成色WWW久久网站| 国产色综合久久无码有码| 久久久久久精品久久久久| 97精品国产97久久久久久免费| 四虎国产精品成人免费久久 | 99re这里只有精品热久久| 久久久久久国产精品免费无码|