0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

編寫一個可以用GRUB來引導(dǎo)的簡單x86內(nèi)核

C語言專家集中營 ? 2018-01-21 09:12 ? 次閱讀

在這篇文章中,我們將從零開始,動手編寫一個可以用GRUB來引導(dǎo)的簡單x86內(nèi)核,該內(nèi)核會在屏幕上打印一條信息,然后——掛起!

one-does-not-kernel

一個人寫一個內(nèi)核是一件簡單的事情

X86機(jī)器是怎樣啟動的?

在我們思考怎樣寫一個內(nèi)核之前,讓我們先看一下x86機(jī)器從啟動到把控制權(quán)交給內(nèi)核的過程是怎樣的:

x86CPU在機(jī)器啟動之后就會從地址[0xFFFFFFF0]處開始執(zhí)行,這個地址就是在32位尋址空間中的最后16個字節(jié)處,這里存放了一條跳轉(zhuǎn)指令,會跳轉(zhuǎn)到內(nèi)存中BIOS代碼起始處。

接著,cpu就開始開始執(zhí)行BIOS代碼塊了,BIOS首先會在我們配置好的啟動設(shè)備序列中,通過檢查一個特定的魔數(shù),找到第一個可以引導(dǎo)的設(shè)備。

一旦BIOS找到一個可以引導(dǎo)的設(shè)備后,它就會把該設(shè)備第一個扇區(qū)的代碼復(fù)制到物理內(nèi)存的[0x7c00]的位置,然后跳轉(zhuǎn)到這個地址開始執(zhí)行這一段代碼,我們習(xí)慣把這一段代碼叫作bootloader。

Bootloader會將內(nèi)核代碼加載到物理內(nèi)存[0x100000]的位置,[0x100000]這個地址是所有x86機(jī)器宏內(nèi)核代碼的起始地址。

我們需要哪一些工具?

* 一個x86構(gòu)架的計算機(jī)

*Linux

*NASM匯編

*GCC

*LD(GNU連接器)

*GRUB

源碼

源代碼可以在我的Github上找到:Githubrepository-mkernel

用匯編代碼來編寫內(nèi)核入口

我們喜歡用c來做所有的事情,但是我們無可避免地需要用到一點兒匯編,我們將會寫一小段x86的匯編代碼來作為內(nèi)核入口,這一段匯編代碼會在調(diào)用我們的c代碼后停止整個程序流程。

我們怎樣確認(rèn)匯編代碼會作為內(nèi)核的起始點呢?

我們將用一個連接器腳本將這些目標(biāo)文件鏈接成我們最終的內(nèi)核程序(稍后解釋更多),在連接器腳本里,我們指定了這段二進(jìn)制代碼會被加載到內(nèi)存[0x100000]處。這個地址就是我之前說過的,內(nèi)核所希望的起始地址。

匯編代碼如下:

1. ;;kernel.asm2. bits 32 ;nasm directive -32 bit3. section .text4.5. global start6. extern kmain ;kmain isdefinedin the c file7.8. start:9. cli ;block interrupts10. call kmain11. hlt ;halt the CPU

第一行指令bit32不是x86匯編指令,它是一條NASM指令,指定nasm匯編器產(chǎn)生32位的程序,這條語句并不是必不可少的,但加上它是一個好的編程習(xí)慣。

第二行是text段(代碼段)的開始,在這里存放著我們的代碼塊。

global是另外一個NASM指令,用將一個符號設(shè)置為全局符號。這樣做連接器才會知道符號start在哪兒開始,start是我們程序的入口地址。

kmain是我們定義在kernel.c文件中的函數(shù),extern關(guān)鍵字聲明了該函數(shù)定義在別的文件中。

到這里,我們的函數(shù)start調(diào)用kmian函數(shù)之后就會使用hlt指令將CPU掛起,中斷會cpu從hlt指令中喚醒,我們要在掛起之前用cli指令來關(guān)閉系統(tǒng)的中斷響應(yīng),cli指令是清除中斷(clear-interrupts)的縮寫。

用C實現(xiàn)的內(nèi)核

在kernle.asm中,我們調(diào)用了kmain()函數(shù),所以我們的c代碼將會在kmain()中開始運行:

1. /*2.* kernel.c3.*/4.void kmain(void)5.{6. char*str ="my first kernel";7. char*vidptr =(char*)0xb8000; //video mem begins here.8. unsignedint i =0;9. unsignedint j =0;10. //clear all11. while(j <80*25*2){12.??????? //blank character13.???????????????? ????????????vidptr[j]=' ';14.??//attribute-byte: light grey on black screen???????15.???????????????? ??????????????vidptr[j+1]=0x07; ?????? ???????????16.???????????????? ??????????????? j = j +2;17.???????????????? ???? }18.???????????????? ???? j =0;19.?????????????? ? while(str[j]!='