分區只是內核的概念,就是說A~B地址放內核,C~D地址放文件系統,(也就是規定哪一個地址區間放內核或者文件系統)等等。linux
1:在內核MTD中能夠定義分區A~B,C~D。。。。。。並予以絕對的地址賦值給每一個分區。咱們能夠來看看在內核中是怎樣來對MTD進行分區的:arch\arm\plat-s3c24xx\common-smdk.c
app
static struct mtd_partition smdk_default_nand_part[] = {
[0] = {
.name = "Boot",
.size = SZ_16K,
.offset = 0,
},
[1] = {
.name = "S3C2410 flash partition 1",
.offset = 0,
.size = SZ_2M,
},
[2] = {
.name = "S3C2410 flash partition 2",
.offset = SZ_4M,
.size = SZ_4M,
},
[3] = {
.name = "S3C2410 flash partition 3",
.offset = SZ_8M,
.size = SZ_2M,
},
[4] = {
.name = "S3C2410 flash partition 4",
.offset = SZ_1M * 10,
.size = SZ_4M,
},ide
......
};測試
通常咱們只須要分3-4個區,第一個爲boot區,一個爲boot參數區(傳遞給內核的參數),一個爲內核區,一個爲文件系統區。spa
而對於bootloader中只要能將內核下載到A~B區的A地址開始處就能夠,C~D區的C起始地址下載文件系統。。。這些起始地址在MTD的分區信息中能找到。因此bootloader對分區的概念不重要,只要它能把內核燒到A位置,把文件系統燒到C位置。
因此,在bootloader對Flash進行操做時,哪塊區域放什麼是之內核爲主。命令行
而爲了方便操做,bootloader相似也引入分區的概念,如,可使用「nand write 0x3000000 kernel 200000」命令將uImage燒到kernel分區,而沒必要寫那麼長:nand write 3000000 A 200000,也就是用分區名來代替具體的地址。server
這要對bootloader對內核從新分區:這須要從新設置一下bootloader環境參數,就能夠同步更新內核分區信息htm
如:ip
setenv bootargs 'noinitrd console=ttySAC0 root=/dev/mtdblock3 rootfstype=jffs2內存
mtdparts=nand_flash:128k(u-boot)ro,64k(u-boot envs),3m(kernel),30m(root.jffs2),30m(root.yaffs)'
內核配置時選上Device Drivers ---> Memory Technology Device (MTD) support ---> Command line partition table parsing
在設置了mtdparts變量以後,就能夠在nand read/write/erase命令中直接使用分區的名字而沒必要指定分區的偏移位置.而這須要內核MTD最好沒有規劃分區。
若是你是經過uboot的內核命令行給MTD層傳遞MTD分區信息,這種狀況下,內核讀取到的分區信息始終和u-boot中的保持一致(推薦的作法)
若是你是把分區信息寫在內核源代碼MTD裏定義好的方法,那最好保證它和u-boot中的保持一致,即同步修改uboot及內核的相關部分。
內核經過bootargs找到文件系統,bootargs中的mtdblockx即表明分區,block1,2,3表明哪一個分區。
事實上,bootargs中的"root=/dev/mtdblockx"只是告訴內核,root fs從第x個(x=0,1,2...)MTD分區掛載,mtdblock0對應第一個分區,mtdblock1對應第二個分區,以此類推.
3:分區方法
1) MTD層的分區
2) 經過U-boot傳遞給內核的命令行中的mtdparts=...
3) 其餘可讓內核知道分區信息的任何辦法,(內核默認的命令參數)
下面說到mtdparts,及它的用法:
mtdparts
mtdparts=fc000000.nor_flash:1920k(linux),128k(fdt),20M(ramdisk),4M(jffs2),38272k(user),256k(env),384k(uboot)
要想這個參數起做用,內核中的mtd驅動必需要支持,即內核配置時須要選上Device Drivers ---> Memory Technology Device (MTD) support ---> Command line partition table parsing
mtdparts的格式以下:
mtdparts=<mtddef>[;<mtddef]
<mtddef> := <mtd-id>:<partdef>[,<partdef>]
<partdef> := <size>[@offset][<name>][ro]
<mtd-id> := unique id used in mapping driver/device
<size> := standard linux memsize OR "-" to denote all remaining space
<name> := (NAME)
所以你在使用的時候須要按照下面的格式來設置:
mtdparts=mtd-id:<size1>@<offset1>(<name1>),<size2>@<offset2>(<name2>)
這裏面有幾個必需要注意的:
a. mtd-id 必需要跟你當前平臺的flash的mtd-id一致,否則整個mtdparts會失效 怎樣獲取到當前平臺的flash的mtd-id?
在bootargs參數列表中能夠指定當前flash的mtd-id,如指定 mtdids:nand0=gen_nand.1,前面的nand0則表示第一個flash
b. size在設置的時候能夠爲實際的size(xxM,xxk,xx),也能夠爲'-'這表示剩餘的全部空間。
相關信息能夠查看drivers/mtd/cmdlinepart.c中的註釋找到相關描述。
U-boot的環境變量值得注意的有兩個: bootcmd 和bootargs。
引用:
u bootcmd
前面有說過bootcmd是自動啓動時默認執行的一些命令,所以你能夠在當前環境中定義各類不一樣配置,不一樣環境的參數設置,而後設置bootcmd爲你常用的那種參數。
u bootargs
bootargs是環境變量中的重中之重,甚至能夠說整個環境變量都是圍繞着bootargs來設置的。bootargs的種類很是很是的多,咱們日常只是使用了幾種而已,感興趣的能夠看看這篇文章說的很全:http://www.linuxidc.com/Linux/2011-03/33599p4.htm。bootargs很是的靈活,內核和文件系統的不一樣搭配就會有不一樣的設置方法,甚至你也能夠不設置bootargs,而直接將其寫到內核中去(在配置內核的選項中能夠進行這樣的設置),正是這些緣由致使了bootargs使用上的困難。
下面介紹一下bootargs經常使用參數,bootargs的種類很是的多,並且隨着kernel的發展會出現一些新的參數,使得設置會更加靈活多樣。
A. root
用來指定rootfs的位置, 常見的狀況有:
root=/dev/ram rw
root=/dev/ram0 rw
請注意上面的這兩種設置狀況是通用的,我作過測試甚至root=/dev/ram1 rw和root=/dev/ram2 rw也是能夠的,網上有人說在某些狀況下是不通用的,即必須設置成ram或者ram0,可是目前尚未遇到,還須要進一步確認,遇到不行的時候能夠逐一嘗試。
root=/dev/mtdx rw
root=/dev/mtdblockx rw
root=/dev/mtdblock/x rw
root=31:0x
上面的這幾個在必定狀況下是通用的,固然這要看你當前的系統是否支持,不過mtd是字符設備,而mtdblock是塊設備,有時候你的挨個的試到底當前的系統支持上面那種狀況下,不過root=/dev/mtdblockx rw比較通用。此外,若是直接指定設備名能夠的話,那麼使用此設備的設備號也是能夠的。
root=/dev/nfs
在文件系統爲基於nfs的文件系統的時候使用。固然指定root=/dev/nfs以後,還須要指定nfsroot=serverip:nfs_dir,即指明文件系統存在那個主機的那個目錄下面。
B. rootfstype
這個選項須要跟root一塊兒配合使用,通常若是根文件系統是ext2的話,有沒有這個選項是無所謂的,可是若是是jffs2,squashfs等文件系統的話,就須要rootfstype指明文件系統的類型,否則會沒法掛載根分區.
C. console
console=tty<n> 使用虛擬串口終端設備 <n>.
console=ttyS<n>[,options] 使用特定的串口<n>,options能夠是這樣的形式bbbbpnx,這裏bbbb是指串口的波特率,p是奇偶校驗位,n是指的bits。
console=ttySAC<n>[,options] 同上面。
看你當前的環境,有時用ttyS<n>,有時用ttySAC<n>,網上有人說,這是跟內核的版本有關,2.4用ttyS<n>,2.6用ttySAC<n>,但實際狀況是官方文檔中也是使用ttyS<n>,因此應該是跟內核版本沒有關聯的。能夠查看Documentation/serial-console.txt找到相關描述。
D. mem
mem=xxM 指定內存的大小,不是必須的
E. ramdisk_size
ramdisk=xxxxx 不推薦
ramdisk_size=xxxxx 推薦
上面這兩個均可以告訴ramdisk 驅動,建立的ramdisk的size,默認狀況下是4m(s390默認8M),你能夠查看Documentation/ramdisk.txt找到相關的描述,不過ramdisk=xxxxx在新版的內核都已經沒有提了,不推薦使用。
F. initrd, noinitrd
當你沒有使用ramdisk啓動系統的時候,你須要使用noinitrd這個參數,可是若是使用了的話,就須要指定initrd=r_addr,size, r_addr表示initrd在內存中的位置,size表示initrd的大小。
G. init
init指定的是內核啓起來後,進入系統中運行的第一個腳本,通常init=/linuxrc, 或者init=/etc/preinit,preinit的內容通常是建立console,null設備節點,運行init程序,掛載一些文件系統等等操做。請注意,不少初學者覺得init=/linuxrc是固定寫法,其實否則,/linuxrc指的是/目錄下面的linuxrc腳本,通常是一個鏈接罷了。
H. ip
指定系統啓動以後網卡的ip地址,若是你使用基於nfs的文件系統,那麼必需要有這個參數,其餘的狀況下就看你本身的喜愛了。設置ip有兩種方法:
ip = ip addr
ip=ip addr:server ip addr:gateway:netmask::which netcard:off
這兩種方法能夠用,不過很明顯第二種要詳細不少,請注意第二種中which netcard 是指開發板上的網卡,而不是主機上的網卡。
說完常見的幾種bootargs,那麼咱們來討論日常我常用的幾種組合:
1). 假設文件系統是ramdisk,且直接就在內存中,bootargs的設置應該以下:
setenv bootargs ‘initrd=0x32000000,0xa00000 root=/dev/ram0 console=ttySAC0 mem=64M init=/linuxrc’
2). 假設文件系統是ramdisk,且在flash中,bootargs的設置應該以下:
setenv bootargs ‘mem=32M console=ttyS0,115200 root=/dev/ram rw init=/linuxrc’
注意這種狀況下你應該要在bootm命令中指定ramdisk在flash中的地址,如bootm kernel_addr ramdisk_addr (fdt_addr)
3). 假設文件系統是jffs2類型的,且在flash中,bootargs的設置應該以下
setenv bootargs ‘mem=32M console=ttyS0,115200 noinitrd root=/dev/mtdblock2 rw rootfstype=jffs2 init=/linuxrc’
4). 假設文件系統是基於nfs的,bootargs的設置應該以下
setenv bootargs ‘noinitrd mem=64M console=ttySAC0 root=/dev/nfs nfsroot=192.168.0.3:/nfs ip=192.168.0.5:192.168.0.3:192.168.0.3:255.255.255.0::eth0:off’
或者
setenv bootargs ‘noinitrd mem=64M console=ttySAC0 root=/dev/nfs nfsroot=192.168.0.3:/nfs ip=192.168.0.5’