?? 1.html
?? 介紹linux下文件和設備編程
?? HTML
?? 第 1 頁 / 共 5 頁
字號:
??
SETUPSECS = 4 ! 默認的setup程序扇區數(setup-sectors)的默認值;<p>BOOTSEG = 0x7C0 ! bootsect的原始地址;<p>INITSEG = DEF_INITSEG ! 將bootsect程序移到這個段處(0x9000) - 避開;<br>SETUPSEG = DEF_SETUPSEG ! 設置程序(setup)從這里開始(0x9020);<br>SYSSEG = DEF_SYSSEG ! 系統加載至0x1000(65536)(64k)段處;<br>SYSSIZE = DEF_SYSSIZE ! 系統的大小(0x7F00): 要加載的16字節為一節的數;<br>!! 以上4個DEF_參數定義在boot.h中:<br>!! DEF_INITSEG 0x9000<br>!! DEF_SYSSEG 0x1000<br>!! DEF_SETUPSEG 0x9020<br>!! DEF_SYSSIZE 0x7F00 (=32512=31.75k)*16=508k<p>! ROOT_DEV & SWAP_DEV 現在是由"build"中編制的;<br>ROOT_DEV = 0<br>SWAP_DEV = 0<br>#ifndef SVGA_MODE<br>#define SVGA_MODE ASK_VGA<br>#endif<br>#ifndef RAMDISK<br>#define RAMDISK 0<br>#endif<br>#ifndef CONFIG_ROOT_RDONLY<br>#define CONFIG_ROOT_RDONLY 1<br>#endif<p>! ld86 需要一個入口標識符,這和通常的一樣;<br>.globl _main<br>_main:<br>#if 0 /* 調試程序的異常分支,除非BIOS古怪(比如老的HP機)否則是無害的 */<br>int 3<br>#endif<br>mov ax,#BOOTSEG !! 將ds段寄存器置為0x7C0;<br>mov ds,ax<br>mov ax,#INITSEG !! 將es段寄存器置為0x9000;<br>mov es,ax<br>mov cx,#256 !! 將cx計數器置為256(要移動256個字, 512字節);<br>sub si,si !! 源地址 ds:si=0x07C0:0x0000;<br>sub di,di !! 目的地址es:di=0x9000:0x0000;<br>cld !! 清方向標志;<br>rep !! 將這段程序從0x7C0:0(31k)移至0x9000:0(576k)處;<br>movsw !! 共256個字(512字節)(0x200長);<br>jmpi go,INITSEG !! 間接跳轉至移動后的本程序go處;<p>! ax和es現在已經含有INITSEG的值(0x9000);<p>go: mov di,#0x4000-12 ! 0x4000(16k)是>=bootsect + setup 的長度 +<br>! + 堆棧的長度 的任意的值;<br>! 12 是磁盤參數塊的大小 es:di=0x94000-12=592k-12;<p>! bde - 將0xff00改成了0x4000以從0x6400處使用調試程序(bde)。如果<br>! 我們檢測過最高內存的話就不用擔心這事了,還有,我的BIOS可以被配置為將wini驅動<br>表<br>! 放在內存高端而不是放在向量表中。老式的堆棧區可能會搞亂驅動表;<p>mov ds,ax ! 置ds數據段為0x9000;<br>mov ss,ax ! 置堆棧段為0x9000;<br>mov sp,di ! 置堆棧指針INITSEG:0x4000-12處;<br>/*<br>* 許多BIOS的默認磁盤參數表將不能<br>* 進行扇區數大于在表中指定<br>* 的最大扇區數( - 在某些情況下<br>* 這意味著是7個扇區)后面的多扇區的讀操作。<br>*<br>* 由于單個扇區的讀操作是很慢的而且當然是沒問題的,<br>* 我們必須在RAM中(為第一個磁盤)創建新的參數表。<br>* 我們將把最大扇區數設置為36 - 我們在一個ED 2.88驅動器上所能<br>* 遇到的最大值。<br>*<br>* 此值太高是沒有任何害處的,但是低的話就會有問題了。<br>*<br>* 段寄存器是這樣的: ds=es=ss=cs - INITSEG,(=0X9000)<br>* fs = 0, gs沒有用到。<br>*/<p>! 上面執行重復操作(rep)以后,cx為0;<p>mov fs,cx !! 置fs段寄存器=0;<br>mov bx,#0x78 ! fs:bx是磁盤參數表的地址;<br>push ds<br>seg fs<br>lds si,(bx) ! ds:si是源地址;<br>!! 將fs:bx地址所指的指針值放入ds:si中;<br>mov cl,#6 ! 拷貝12個字節到0x9000:0x4000-12開始處;<br>cld<br>push di !! 指針0x9000:0x4000-12處;<p>rep<br>movsw<p>pop di !! di仍指向0x9000:0x4000-12處(參數表開始處);<br>pop si !! ds => si=INITSEG(=0X9000);<p>movb 4(di),*36 ! 修正扇區計數值;<p>seg fs<br>mov (bx),di !! 修改fs:bx(0000:0x0078)處磁盤參數表的地址為0x9000:0x4000-12;<br>seg fs<br>mov 2(bx),es<p>! 將setup程序所在的扇區(setup-sectors)直接加載到boot塊的后面。!! 0x90200開始處<br>;<br>! 注意,es已經設置好了。<br>! 同樣經過rep循環后cx為0<p>load_setup:<br>xor ah,ah ! 復位軟驅(FDC);<br>xor dl,dl<br>int 0x13<p>xor dx,dx ! 驅動器0, 磁頭0;<br>mov cl,#0x02 ! 從扇區2開始,磁道0;<br>mov bx,#0x0200 ! 置數據緩沖區地址=es:bx=0x9000:0x200;<br>! 在INITSEG段中,即0x90200處;<br>mov ah,#0x02 ! 要調用功能號2(讀操作);<br>mov al,setup_sects ! 要讀入的扇區數SETUPSECS=4;<br>! (假釋所有數據都在磁頭0、磁道0);<br>int 0x13 ! 讀操作;<br>jnc ok_load_setup ! ok則繼續;<p>push ax ! 否則顯示出錯信息。保存ah的值(功能號2);<br>call print_nl !! 打印換行;<br>mov bp,sp !! bp將作為調用print_hex的參數;<br>call print_hex !! 打印bp所指的數據;<br>pop ax<p>jmp load_setup !! 重試!<p>!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!<br>!!INT 13 - DISK - READ SECTOR(S) INTO MEMORY<br>!! AH = 02h<br>!! AL = number of sectors to read (must be nonzero)<br>!! CH = low eight bits of cylinder number<br>!! CL = sector number 1-63 (bits 0-5)<br>!! high two bits of cylinder (bits 6-7, hard disk only)<br>!! DH = head number<br>!! DL = drive number (bit 7 set for hard disk)<br>!! ES:BX -> data buffer<br>!! Return: CF set on error<br>!! if AH = 11h (corrected ECC error), AL = burst length<br>!! CF clear if successful<br>!! AH = status (see #00234)<br>!! AL = number of sectors transferred (only valid if CF set for some<br>!! BIOSes)<br>!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!<p><br>ok_load_setup:<p>! 取得磁盤驅動器參數,特別是每磁道扇區數(nr of sectors/track);<p>#if 0<p>! bde - Phoenix BIOS手冊中提到功能0x08只對硬盤起作用。<br>! 但它對于我的一個BIOS(1987 Award)不起作用。<br>! 不檢查錯誤碼是致命的錯誤。<p>xor dl,dl<br>mov ah,#0x08 ! AH=8用于取得驅動器參數;<br>int 0x13<br>xor ch,ch<p>!!!!!!!!!!!!!!!!!!!!!!!!!!!<br>!! INT 13 - DISK - GET DRIVE PARAMETERS (PC,XT286,CONV,PS,ESDI,SCSI)<br>!! AH = 08h<br>!! DL = drive (bit 7 set for hard disk)<br>!!Return: CF set on error<br>!! AH = status (07h) (see #00234)<br>!! CF clear if successful<br>!! AH = 00h<br>!! AL = 00h on at least some BIOSes<br>!! BL = drive type (AT/PS2 floppies only) (see #00242)<br>!! CH = low eight bits of maximum cylinder number<br>!! CL = maximum sector number (bits 5-0)<br>!! high two bits of maximum cylinder number (bits 7-6)<br>!! DH = maximum head number<br>!! DL = number of drives<br>!! ES:DI -> drive parameter table (floppies only)<br>!!!!!!!!!!!!!!!!!!!!!!!!!!!!<p>#else<p>! 好象沒有BIOS調用可取得扇區數。如果扇區36可以讀就推測是36個扇區,<br>! 如果扇區18可讀就推測是18個扇區,如果扇區15可讀就推測是15個扇區,<br>! 否則推測是9. [36, 18, 15, 9]<p>mov si,#disksizes ! ds:si->要測試扇區數大小的表;<p>probe_loop:<br>lodsb !! ds:si所指的字節 =>al, si=si+1;<br>cbw ! 擴展為字(word);<br>mov sectors, ax ! 第一個值是36,最后一個是9;<br>cmp si,#disksizes+4<br>jae got_sectors ! 如果所有測試都失敗了,就試9;<br>xchg ax,cx ! cx = 磁道和扇區(第一次是36=0x0024);<br>xor dx,dx ! 驅動器0,磁頭0;<br>xor bl,bl !! 設置緩沖區es:bx = 0x9000:0x0a00(578.5k);<br>mov bh,setup_sects !! setup_sects = 4 (共2k);<br>inc bh<br>shl bh,#1 ! setup后面的地址(es=cs);<br>mov ax,#0x0201 ! 功能2(讀),1個扇區;<br>int 0x13<br>jc probe_loop ! 如果不對,就試用下一個值;<p>#endif<p>got_sectors:<p>! 恢復es<p>mov ax,#INITSEG<br>mov es,ax ! es = 0x9000;<p>! 打印一些無用的信息(換行后,顯示Loading)<p>mov ah,#0x03 ! 讀光標位置;<br>xor bh,bh<br>int 0x10<p>mov cx,#9<br>mov bx,#0x0007 ! 頁0,屬性7 (normal);<br>mov bp,#msg1<br>mov ax,#0x1301 ! 寫字符串,移動光標;<br>int 0x10<p>! ok, 我們已經顯示出了信息,現在<br>! 我們要加載系統了(到0x10000處)(64k處)<p>mov ax,#SYSSEG<br>mov es,ax ! es=0x01000的段;<br>call read_it !! 讀system,es為輸入參數;<br>call kill_motor !! 關閉驅動器馬達;<br>call print_nl !! 打印回車換行;<p>! 這以后,我們來檢查要使用哪個根設備(root-device)。如果已指定了設備(!=0)<br>! 則不做任何事而使用給定的設備。否則的話,使用/dev/fd0H2880 (2,32)或/dev/PS0<br>(2,28)<br>! 或者是/dev/at0 (2,8)之一,這取決于我們假設我們知道的扇區數而定。<br>!! |__ ps0?? (x,y)--表示主、次設備號?<p>seg cs<br>mov ax,root_dev<br>or ax,ax<br>jne root_defined<br>seg cs<br>mov bx,sectors !! sectors = 每磁道扇區數;<br>mov ax,#0x0208 ! /dev/ps0 - 1.2Mb;<br>cmp bx,#15<br>je root_defined<br>mov al,#0x1c ! /dev/PS0 - 1.44Mb !! 0x1C = 28;<br>cmp bx,#18<br>je root_defined<br>mov al,0x20 ! /dev/fd0H2880 - 2.88Mb;<br>cmp bx,#36<br>je root_defined<br>mov al,#0 ! /dev/fd0 - autodetect;<br>root_defined:<br>seg cs<br>mov root_dev,ax !! 其中保存由設備的主、次設備號;<p>! 這以后(所有程序都加載了),我們就跳轉至<br>! 被直接加載到boot塊后面的setup程序去:<p>jmpi 0,SETUPSEG !! 跳轉到0x9020:0000(setup程序的開始位置);<p><br>! 這段程序將系統(system)加載到0x10000(64k)處,<br>! 注意不要跨越64kb邊界。我們試圖以最快的速度<br>! 來加載,只要可能就整個磁道一起讀入。<br>!<br>! 輸入(in): es - 開始地址段(通常是0x1000)<br>!<br>
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -