u-boot 環境變數引數設定
原文:
看到這個標題,可能覺得這個並沒有什麼的,其實不然,編好了u-boot了,但是如何來使用確不是那麼簡單的,想當初我將uboot製作出來後以為全部都搞定了,屁顛屁顛的燒到板子上後可系統就是起不來,為什麼了,折騰了很久以後才發現是u-boot的環境變數設定錯誤,在後來的工作中,我發現一些有一定經驗的人在這方面也是模模糊糊的扯不清,所以這裡要討論一下。
的環境變數是使用u-boot的關鍵,它
環境變數 |
描述 |
bootdelay |
執行自動啟動的等候秒數 |
baudrate |
串列埠控制檯的波特率 |
netmask |
乙太網介面的掩碼 |
ethaddr |
乙太網卡的網絡卡實體地址 |
bootfile |
預設的下載檔案 |
bootargs |
傳遞給核心的啟動引數 |
bootcmd |
自動啟動時執行的命令 |
serverip |
伺服器端的ip地址 |
ipaddr |
本地ip地址 |
stdin |
標準輸入裝置 |
stdout |
標準輸出裝置 |
stderr |
標準出錯裝置 |
上面只是一些最基本的環境變數,請注意,板子裡原本是沒有環境變數的,u-boot的預設情況下會有一些基本的環境變數,在你執行了saveenv之後,環境變數會第一次儲存到flash中,之後你對環境變數的修改,儲存都是基於儲存在flash中的環境變數的操作。
U-boot的環境變數值得注意的有兩個: bootcmd 和bootargs。
bootcmd
前面有說過bootcmd是自動啟動時預設執行的一些命令,因此你可以在當前環境中定義各種不同配置,不同環境的引數設定,然後設定bootcmd為你經常使用的那種引數。//如果要輸入分號,記得是這樣:‘\;’,zImage用的是go addr,uImage用的是bootm addr,我是這樣理解的。
bootargs
bootargs是環境變數中的重中之重,甚至可以說整個環境變數都是圍繞著bootargs來設定的。bootargs的種類非常非常的多,我們平常只是使用了幾種而已,感興趣的可以看看這篇文章說的很全:http://blog.chinaunix.net/u2/79570/showart_1675071.html。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 使用虛擬串列埠終端裝置 .
console=ttyS[,options] 使用特定的串列埠,options可以是這樣的形式bbbbpnx,這裡bbbb是指串列埠的波特率,p是奇偶校驗位,n是指的bits。
console=ttySAC[,options] 同上面。
看你當前的環境,有時用ttyS,有時用ttySAC,網上有人說,這是跟核心的版本有關,2.4用ttyS,2.6用ttySAC,但實際情況是官方文件中也是使用ttyS,所以應該是跟核心版本沒有關聯的。可以檢視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. 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]< p="" style="word-wrap: break-word;">
:= :[,]
:= [@offset][][ro]
:= unique id used in mapping driver/device
:= standard linux memsize OR "-" to denote all remaining space
:= (NAME)
因此你在使用的時候需要按照下面的格式來設定:
mtdparts=mtd-id:@(),@()
這裡面有幾個必須要注意的:
a. mtd-id 必須要跟你當前平臺的flash的mtd-id一致,不然整個mtdparts會失效
b. size在設定的時候可以為實際的size(xxM,xxk,xx),也可以為'-'這表示剩餘的所有空間。
舉例:
假設flash 的mtd-id是sa1100,那麼你可以使用下面的方式來設定:
mtdparts=sa1100:- → 只有一個分割槽
mtdparts=sa1100:256k(ARMboot)ro,-(root) → 有兩個分割槽
可以檢視drivers/mtd/cmdlinepart.c中的註釋找到相關描述。
I. 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’
上面就是我們經常使用的幾種bootargs的組合,老實說,bootargs非常非常的靈活,所以設定的方法有很多中形式,具體的還應該根據你的平臺具體的情況來設定。
//mini2400執行OpenWrt:setenv bootargs noinitrd root=/dev/mtdblock3 rootfstype=jffs2 console=ttySAC0,115200 師兄給我解釋mtdblock3:0是uboot,1是uboot環境變數,2是核心,3是檔案系統。至於為什麼是ttySAC0而不是ttyS0,我也不自到。