1. 程式人生 > >iMX6Q-sbreasd U-boot TFTP and NFS

iMX6Q-sbreasd U-boot TFTP and NFS

i.MX6Q-sbreasdU-bootTFTPandNFS

在介紹如何通過TFTP載入核心、NFS掛載網路系統之前,先簡單介紹一下關於u-boot引數的種類:

一、

U-boot的環境變數值得注意的有兩個: bootcmd 和bootargs。

bootcmd是自動啟動時預設執行的一些命令,因此你可以在當前環境中定義各種不同配置,不同環境的引數設定,然後設定bootcmd為你經常使用的那種引數。

     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

B. rootfstype

  這個選項需要跟root一起配合使用,一般如果根檔案系統是ext2的話,有沒有這個選項是無所謂的,但是如果是jffs2,squashfs等檔案系統的話,就需要rootfstype指明檔案系統的型別,不然會無法掛載根分割槽.

C. console

console=tty 使用虛擬串列埠終端裝置 .

console=ttyS[,options] 使用特定的串列埠,options可以是這樣的形式bbbbpnx,這裡bbbb是指串列埠的波特率,p是奇偶位(從來沒有看過使用過),n是指的bits。

console=ttySAC[,options] 同上面。

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指令碼,一般是一個連線罷了。

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’

二、

u-boot引數瞭解後,我們要簡單知道u-boot載入kernel和掛載rotfs的過程:

  首先我們開啟開發板後,從flash中將u-boot讀取到RAM中並執行,u-boot正常啟動,會按照botcmd引數的配置執行,而我們恰好設定了bootcmd的啟動方式,比如TFTP。這樣u-boot通過tftp載入了zImage核心,並執行核心,核心成功啟動了之後(同時將u-boot的bootargs引數傳遞給了核心),核心做完相應的硬體初始化等等之後,通過botargs傳遞的引數(我們設定成nfs網路掛載rootfs)進行nfs掛載根檔案系統rootfs,並啟動檔案系統的第一個程序。之後我們的開發板就成功的和我們的主機同步,可以進行相應的除錯開發了。

三、

  關於i.MX6q-sbaresd的mfgtools燒工具中已經有編譯好的u-boot和uImage,(當然還有rootfs,但是是壓縮檔案tar.bz2個格式),我們使用燒些工具將整個系統(包括u-boot、kernel、rootfs)都燒寫進emmc,成功後,開啟開發板在u-boot的botdelay的時間之內,進入u-boot。接下來,在開始配置u-boot通過TFTP載入核心zImage和通過NFS和網路檔案系統進行掛載根檔案系統rootfs之前,我們需要在主機(我用的是虛擬機器VM10+ubuntu14.04)搭建滿足TFTP和NFS的伺服器環境:

1、建立TFTP伺服器:

  ubuntu下:

  安裝:

     sudo apt-get install tftp-hpa tftpd-hpa

  建立目錄:

  sudo mkdir /tftpboot -->用於tftp傳輸的目錄,也就是這裡面放編譯好的適用於此開發板zImage,和裝置樹uImagexxx.dtb

  sudo chmod 777 /tftpboot

  更改配置檔案:

  sudo vi /etc/defailt/tftpd-hpa

  # /etc/default/tftpd-hpa

TFTP_USERNAME="tftp"

TFTP_DIRECTORY="/tftpboot" # 這裡是你的tftpd-hpa的服務目錄,這個想建立在哪裡都行

TFTP_ADDRESS="0.0.0.0:69"

TFTP_OPTIONS="-l -c -s" #這裡是選項,-c是可以上傳檔案的引數,-s是指定tftpd-hpa服務目錄,上面已經指定

  重啟服務:

  sudo /etc/init.d/tftp-hpa restart

  至此,TFTP伺服器建立完成。

2、搭建NFS伺服器:

  安裝:

  sudo apt-get install nfs-kernel-server 

  建立目錄:

  sudo mkdir /nfsdir

  sudo chmod 777 -R /nfsdir

  sudo cp rootfs /nfsdir -a

更改配置檔案:

  sudo vi /etc/export

  在最後一行新增/nfsdir/rootfs *(rw,async,no_root_squash,no_subtree_check)

  *:允許所有的網段訪問,也可以使用具體的IP
rw:掛接此目錄的客戶端對該共享目錄具有讀寫許可權
sync:資料同步寫入記憶體和硬碟
no_root_squash:root使用者具有對根目錄的完全管理訪問許可權。
no_subtree_check:不檢查父目錄的許可權

  重新開啟伺服器:

  sudo /etc/init.d/nfs-kernel-server restart

  *測試:

  sudo mount -t nfs 192.168.1.110:/nfsdir/rootfs /tmp

  檢視/tmp目錄: ls /tmp 會看到和你rootfs內容相同的目錄結構

  解除掛載:sudo umount /tmp

附錄:NFS常用引數如下:
ro 只讀訪問
rw 讀寫訪問sync 所有資料在請求時寫入共享
asyncnfs在寫入資料前可以響應請求
secure nfs通過1024以下的安全TCP/IP埠傳送
insecure nfs通過1024以上的埠傳送
wdelay 如果多個使用者要寫入nfs目錄,則歸組寫入(預設)
no_wdelay如果多個使用者要寫入nfs目錄,則立即寫入,當使用async時,無需此設定。
hide 在nfs共享目錄中不共享其子目錄
no_hide 共享nfs目錄的子目錄
subtree_check 如果共享/usr/bin之類的子目錄時,強制nfs檢查父目錄的許可權(預設)
no_subtree_check和上面相對,不檢查父目錄許可權
all_squash 共享檔案的UID和GID對映匿名使用者anonymous,適合公用目錄。
no_all_squash 保留共享檔案的UID和GID(預設)
root_squash root使用者的所有請求對映成如anonymous使用者一樣的許可權(預設)
no_root_squasroot使用者具有根目錄的完全管理訪問許可權
anonuid=xxx 指定nfs伺服器/etc/passwd檔案中匿名使用者的UID
anongid=xxx 指定nfs伺服器/etc/passwd檔案中匿名使用者的GID

  至此,NFS主機伺服器建立完成。

四、

  配置u-boot引數:

前面已經成功啟動開發板,並進入u-boot,下面來配置u-boot的botargs和bootcmd引數:

  關於i.mx6q-sbaresd的mfgtools自帶的u-boot,引數已經進行了相應的配置,如果自己進行過改動,可以用命令:env default-a -f (之後save)來初始化配置,在這裡我們不需要改動太多,因為這個u-boot已經配置的很全面。下面進行更改引數:

  setenv serverip 192.168.1.110 <--主機的ip地址

  setenv ipaddr 192.168.1.121 <--開發板的ip地址

  setenv ip_dyn no

  setenv nfsroot /nfsdir/rootfs <--共享目錄的路徑

  setenv image zImage <-- 編譯好的可用的核心

  setenv fdt_file uImage-imx6q-sabresd.dtb <-- 裝置樹,在mfgtools工具中可以找到

  setenv netargs 'setenv bootargs console=${console},${baudrate} ${smp} root=/dev/nfs nfsroot=${serverip}:${nfsroot} ip=${ipaddr}:${serverip}:192.168.1.1:255.255.255.0::eth0:on,v3,tcp'

  這其中會看到,配置了很多的變數,其實這些變數的呼叫都始於bootcmd,我們在設定好這些後,save之後,執行 runnetboot ,成功載入核心zImage,和掛載rootfs。

  但是這樣不能再開啟u-boot後直接啟動核心和rootfs,原因是在啟動u-boot後執行的bootcmd不是我們所期望的方式,解決這個辦法只需重新設定bootcmd,

  setenv bootcmd ‘run netboot’

  save

  這樣,重啟就可以實現自動載入kernel和掛載rootfs了。

在看u-boot引數配置時會發現,netboot 在經過一些判斷後,執行了tftp zImage並bootm執行kernel,之後執行netargs引數,而netargs的內容正是我們的所配置的nfs的配置方法。

簡單的實現的u-boot通過tftp載入kernel和nfs掛載rootfs,希望能有所幫助。