新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > 嵌入式Linux: uClinux操作系統(tǒng)移植

嵌入式Linux: uClinux操作系統(tǒng)移植

作者: 時間:2012-05-11 來源:網(wǎng)絡(luò) 收藏

1.簡介

本文引用地址:http://2s4d.com/article/149092.htm

這個英文單詞中u表示Micro,小的意思,C表示Control,控制的意思,所以就是Micro-Control-,字面上的理解就是針對微控制領(lǐng)域而設(shè)計的系統(tǒng).

uclinux是一個源碼開放的,面向沒有MMU(MemoryManagementUnit)的硬件平臺。它是linux的一個變種,主要的區(qū)別在于兩者的內(nèi)存管理機制和進程調(diào)度管理機制,同時為了適應(yīng)應(yīng)用的需求,它的采用了romfs文件系統(tǒng),并對linux上的c語言庫glibc做了簡化。

2.硬件體系結(jié)構(gòu)簡介

運行uClinux的硬件平臺主要包括如下幾個部分:cpu(ARMv4指令集兼容)、uart、memorycontroller、定時器、flash存儲器,sdram存儲器,中斷控制器和DMA.

3.編譯環(huán)境和編譯工具

uclinux源碼絕大部分是用c語言開發(fā)的,有一些與硬件直接相關(guān)的代碼則用特定于某一CPU體系結(jié)構(gòu)的匯編來實現(xiàn)。這些源碼只能用GNU的gcc編譯工具來進行編譯、鏈接。

GNUgcc可以運行于/Unix上。如果要在Windows平臺上運行g(shù)cc,則必須安裝Cygwin.Cygwin可以在Windows中安裝一個linux的運行環(huán)境,這樣就可以在windows下運行原本只能在linux下運行的程序。

為了在PC上編譯得到運行于目標CPU上的操作系統(tǒng)內(nèi)核,還必須安裝一個合適的交叉編譯器。Gcc提供了現(xiàn)成的針對MIPS、ARM、M68K、Sharc、PowerPC的交叉編譯器。如果沒有現(xiàn)成的交叉編譯器,則需要自行設(shè)計。GNU網(wǎng)站提供了一些如何開發(fā)新的交叉編譯器的文章。開發(fā)一個新的編譯器,一般需要如下幾個步驟:

(1)、編寫機器描述腳本。采用gcc的RTL(RegisterTansferLanguage)語言描述針對某一CPU體系結(jié)構(gòu)的機器指令與尋址方式、CPU浮點處理方式、endianess、c語言中各種數(shù)據(jù)類型的位寬、寄存器的個數(shù)和使用規(guī)則、堆棧和函數(shù)調(diào)用規(guī)則等體系結(jié)構(gòu)的細節(jié)。

(2)、設(shè)計代碼生成器。Gcc在對c語言源文件進行了詞法和語法分析后,將產(chǎn)生一種中間格式文件(intermediaterepresentation)。為了把這種中間格式文件轉(zhuǎn)化為針對具體CPU體系結(jié)構(gòu)的機器碼,需要自行設(shè)計一個代碼生成器。

(3)、設(shè)計匯編器

(4)、設(shè)計鏈接器

4.uClinux啟動過程

uClinux系統(tǒng)的啟動可以分為兩個步驟:

(1)。運行bootloader初始化程序

SRAM、SDRAM等存儲設(shè)備屬于揮發(fā)性的存儲器,掉電以后其中的內(nèi)容就會全部丟失,所以必須把操作系統(tǒng)的內(nèi)核鏡像存放在Flash等不揮發(fā)性存儲介質(zhì)上。但是操作系統(tǒng)在運行時,需要動態(tài)的創(chuàng)建一些如數(shù)據(jù)段、堆棧、頁表(針對使用虛擬地址的操作系統(tǒng))等內(nèi)容,所以需要在RAM中運行操作系統(tǒng)。因此,就需要一個引導程序把操作系統(tǒng)的內(nèi)核鏡像從Flash存儲器拷貝到RAM中,然后再從RAM中執(zhí)行操作系統(tǒng)的內(nèi)核。Bootloader就是可以完成這樣一種功能的程序。

從本質(zhì)上來講,bootloader不屬于操作系統(tǒng)內(nèi)核。它采用匯編語言編寫,因此針對不同的cpu體系結(jié)構(gòu),這一部分代碼不具有可性。在操作系統(tǒng)時,這部分代碼必須加以改寫。

具體來講,bootloader在系統(tǒng)啟動時主要完成以下幾項工作:

。將操作系統(tǒng)內(nèi)核從flash拷貝到sdram中,如果是壓縮格式的內(nèi)核,還要將之解壓縮。

。改寫系統(tǒng)的memorymap,原先flash起始地址映射為0地址,這時需要將RAM的起始地址映射為0.

。設(shè)置堆棧指針并將bss段清零。將來執(zhí)行c語言程序和調(diào)用子函數(shù)時要用到。

。改變pc值,使得cpu開始執(zhí)行真正的操作系統(tǒng)內(nèi)核。

(2)運行操作系統(tǒng)內(nèi)核

bootloader程序執(zhí)行完上述的各項工作后,通過一條跳轉(zhuǎn)指令,轉(zhuǎn)而執(zhí)行ini目錄下c語言源文件main.c中的函數(shù)start_kernel()。因為在此之前bootloader已經(jīng)創(chuàng)建好一個初始化環(huán)境,

c函數(shù)可以開始執(zhí)行了。整個操作系統(tǒng)內(nèi)核的初始化工作從這里才算是真正開始。這個函數(shù)的長度比較短,代碼如下:asmlinkagevoid__initstart_kernel(void)

{

char*command_line;

unsignedlongmempages;

externcharsaved_command_line[];

/*

*Interruptsarestilldisabled.Donecessarysetups,then

*enablethem

*/

lock_kernel();

printk(linux_banner);

setup_arch(command_line);

printk(Kernelcommandline:%sn,saved_command_line);

parse_options(command_line);

trap_init();

init_IRQ();

sched_init();

softirq_init();

time_init();

/*

*HACKALERT!Thisisearly.We'reenablingtheconsolebefore

*we'vedonePCIsetupsetc,andconsole_init()mustbeawareof

*this.Butwedowantoutputearly,incasesomethinggoeswrong.

*/

console_init();

#ifdefCONFIG_MODULES

init_modules();

#endif

if(prof_shift){

unsignedintsize;

/*onlytextisprofiled*/

prof_len=(unsignedlong)_etext-(unsignedlong)_stext;

prof_len>>=prof_shift;

size=prof_len*sizeof(unsignedint)+PAGE_SIZE-1;

prof_buffer=(unsignedint*)alloc_bootmem(size);

}

kmem_cache_init();

sti();

calibrate_delay();

#ifdefCONFIG_BLK_DEV_INITRD

if(initrd_start!initrd_below_start_ok

initrd_start

printk(KERN_CRITinitrdoverwritten(0x%08lx0x%08lx)-

disablingit.n,initrd_start,min_low_pfn

initrd_start=0;

}

#endif

mem_init();

kmem_cache_sizes_init();

pgtable_cache_init();

mempages=num_physpages;

linux操作系統(tǒng)文章專題:linux操作系統(tǒng)詳解(linux不再難懂)

上一頁 1 2 下一頁

評論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉