1. 程式人生 > >u-boot 環境變數引數設定

u-boot 環境變數引數設定

今天本來是燒寫核心,結果一不小心把uboot也整不能用了,無奈之下只好重新燒個uboot,等都弄好以後,發現系統還是啟動不了,原來是啟動引數設定不對,於是找到了這篇文章,//是我新增的內容。

原文:

     看到這個標題,可能覺得這個並沒有什麼的,其實不然,編好了u-boot了,但是如何來使用確不是那麼簡單的,想當初我將uboot製作出來後以為全部都搞定了,屁顛屁顛的燒到板子上後可系統就是起不來,為什麼了,折騰了很久以後才發現是u-boot的環境變數設定錯誤,在後來的工作中,我發現一些有一定經驗的人在這方面也是模模糊糊的扯不清,所以這裡要討論一下。

的環境變數是使用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,我也不自到。