在 zybo board 開發記錄: Zynq 與 LED 閃爍控制 一文中我們談到了如何透過 C 語言撰寫獨立的程式,讓它控制 Zynq 的處理器系統 (Processing System, PS) 去閃爍 LED 的亮暗。既然 Zynq 的處理器系統 (Processing System, PS) 使用的是 ARM Cortex-A9 的處理器,那當然也可以讓我們跑 Linux 在 Zybo Board 上。
(本文以 Vivado 2016.2 進行開發)
本文將簡述如何自行編譯 u-boot 以及 Linux Kernel,並搭配 Busybox 製作簡單的 RootFS 執行於 Zybo Board 上。
開發目標
在這次的開發中,我們要編譯 u-boot、Linux,並使用 Busybox 製作簡單的 Rootfs 後,透過製作 SD 卡來讓 Zybo Board 透過 SD 卡開機。
要注意到的是,由於我們要透過 SD 卡開機進入到 Linux 系統,因此我們要透過 JP5 去更改開機模式。
了解開機流程
既然我們要讓 Zybo board 執行 Linux 系統,就要先來了解一下開機流程,才知道我們大概需要準備哪些東西。從 Zynq-7000 All Programmable SoC: Embedded Design Tutorial - A Hands-On Guide to Effective Embedded System Design (UG1165) 可以看到 Zynq-7000 執行 Linux 系統的開機流程圖。
也就是說,當開始提供電源給 Zynq 處理器系統 (Processing System, PS) 並完成重置(reset) 後,Zynq 內建的 Boot ROM 會去載入 第一階段開機程式 (First Stage Boot Loader, FSBL) ,接著載入 位元流 (bitstream) 去初始化整個 可程式邏輯(Programmable Logic, PL) 。 完成後,接下來就是透過 U-Boot 去載入 Linux Kernel、Device Tree 以及 Root File System。
了解了這個,我們就知道我們大概要準備哪些東西了。
設定好環境
在安裝玩 Vivado 與 Xilinx SDK 後,實際上包含 Zynq 在用的 ARM toolchain 亦同時被安裝到系統中,我們只要使用 source 命令即可讓當前的環境知道 xilinx-arm toolchain 的路徑。這邊以 Viavdo 2016.2 作為範例。
coldnew@gentoo ~ $ source /opt/Xilinx/Vivado/2016.2/settings64.sh
這樣就可以獲得 arm-xilinx-* toolchain 的命令,實際上有哪些呢? 輸入個 arm-xilinx- 按下 TAB 看看
coldnew@gentoo ~ $ arm-xilinx- arm-xilinx-eabi-addr2line arm-xilinx-eabi-gcc-4.9.2 arm-xilinx-eabi-objcopy arm-xilinx-linux-gnueabi-ar arm-xilinx-linux-gnueabi-gcc-ar arm-xilinx-linux-gnueabi-objdump arm-xilinx-eabi-ar arm-xilinx-eabi-gcc-ar arm-xilinx-eabi-objdump arm-xilinx-linux-gnueabi-as arm-xilinx-linux-gnueabi-gcc-nm arm-xilinx-linux-gnueabi-ranlib arm-xilinx-eabi-as arm-xilinx-eabi-gcc-nm arm-xilinx-eabi-ranlib arm-xilinx-linux-gnueabi-c++ arm-xilinx-linux-gnueabi-gcc-ranlib arm-xilinx-linux-gnueabi-readelf arm-xilinx-eabi-c++ arm-xilinx-eabi-gcc-ranlib arm-xilinx-eabi-readelf arm-xilinx-linux-gnueabi-c++filt arm-xilinx-linux-gnueabi-gcov arm-xilinx-linux-gnueabi-size arm-xilinx-eabi-c++filt arm-xilinx-eabi-gcov arm-xilinx-eabi-size arm-xilinx-linux-gnueabi-cpp arm-xilinx-linux-gnueabi-gdb arm-xilinx-linux-gnueabi-sprite arm-xilinx-eabi-cpp arm-xilinx-eabi-gdb arm-xilinx-eabi-sprite arm-xilinx-linux-gnueabi-elfedit arm-xilinx-linux-gnueabi-gprof arm-xilinx-linux-gnueabi-strings arm-xilinx-eabi-elfedit arm-xilinx-eabi-gprof arm-xilinx-eabi-strings arm-xilinx-linux-gnueabi-g++ arm-xilinx-linux-gnueabi-ld arm-xilinx-linux-gnueabi-strip arm-xilinx-eabi-g++ arm-xilinx-eabi-ld arm-xilinx-eabi-strip arm-xilinx-linux-gnueabi-gcc arm-xilinx-linux-gnueabi-nm arm-xilinx-eabi-gcc arm-xilinx-eabi-nm arm-xilinx-linux-gnueabi-addr2line arm-xilinx-linux-gnueabi-gcc-4.9.2 arm-xilinx-linux-gnueabi-objcopy
如果你系統上已經有其他的 ARM toolchain 的話,可以考慮跳過這一步驟,接下來要格式化 Micro SD 卡。
格式化 MicroSD 卡
在這次的開發中,我們要設定 MicroSD 卡片成兩個分區,第一個是 fat32 格式,第二個則使用 ext4 格式,若不會使用 fdisk 命令的話,可以透過 gparted 來進行格式化,以下是我格式化卡片的範例 (8GB 卡片)。
(實際上在本文的範例中,只會用到第一個分區,第二個分區是為了往後文章要開機到大一點的 rootfs 準備的。)
編譯 u-boot
我們首先去 GitHub 下載 DigilentInc 加入 zybo board 後的 u-boot 版本,要注意這邊要選擇 master-next 分支。
git clone https://github.com/DigilentInc/u-boot-Digilent-Dev.git -b master-next
完成後進入到該資料夾
coldnew@gentoo ~ $ cd u-boot-Digilent-Dev
編譯 u-boot,記得指派編譯目標為 zynq_zybo_config
coldnew@gentoo ~/u-boot-Digilent-Dev $ CROSS_COMPILE=arm-xilinx-linux-gnueabi- make zynq_zybo_config coldnew@gentoo ~/u-boot-Digilent-Dev $ CROSS_COMPILE=arm-xilinx-linux-gnueabi- make
編譯完成後,注意一下 u-boot 這個檔案,他就是我們等等要用到的 u-boot 執行檔,不過由於 Xilinx Tool 要找有 .elf 副檔名的檔案,因此我們把它複製成 u-boot.elf 。
coldnew@gentoo ~/u-boot-Digilent-Dev $ cp u-boot u-boot.elf
編譯 Linux kernel
編譯好 u-boot 後,接下來就是編譯 Linux Kernel 了,我們一樣選擇 DigilentInc 加入 zybo board 後的 Linux Kernel 版本,記得要選 master-next 分支。
git clone https://github.com/DigilentInc/Linux-Digilent-Dev.git -b master-next
接著,當然就是編譯了,不過在這之前請先確定你有裝 u-boot-tools 這套件,我們需要裡面的 mkimage 指令,Gentoo Linux 可以直接用以下命令來安裝。
coldnew@gentoo ~ $ sudo emerge dev-embedded/u-boot-tools
完成後進入 Linux Kernel 資料夾
coldnew@gentoo ~ $ cd Linux-Digilent-Dev
編譯我們需要的 uImage 文件,記得要指定 config 為 xilinx_zynq_defconfig 以及設定 UIMAGE_LOADADDR 為 0x8000 。
coldnew@gentoo ~/Linux-Digilent-Dev $ ARCH=arm CROSS_COMPILE=arm-xilinx-linux-gnueabi- make xilinx_zynq_defconfig coldnew@gentoo ~/Linux-Digilent-Dev $ ARCH=arm CROSS_COMPILE=arm-xilinx-linux-gnueabi- make coldnew@gentoo ~/Linux-Digilent-Dev $ ARCH=arm CROSS_COMPILE=arm-xilinx-linux-gnueabi- make UIMAGE_LOADADDR=0x8000 uImage coldnew@gentoo ~/Linux-Digilent-Dev $ ARCH=arm CROSS_COMPILE=arm-xilinx-linux-gnueabi- make zynq-zybo.dtb
編譯完後,我們會需要 arch/arm/boot/uImage 以及 arch/arm/boot/dts/zynq-zybo.dtb 這兩個檔案,後者就是 device tree 編譯出來的資料檔。
由於放入到 SD 卡上的 device tree 檔案名稱為 devicetree.dtb ,因此這邊將 zynq-zybo.dtb 改一下名。
coldnew@gentoo ~/Linux-Digilent-Dev $ cp arch/arm/boot/dts/zynq-zybo.dtb devicetree.dtb
如果你想手動修改 Device Tree 並再重新編譯的話,也可以這樣去產生我們要的 devicetree.dtb 。
coldnew@gentoo ~/Linux-Digilent-Dev $ ./scripts/dtc/dtc -I dts -O dtb -o devicetree.dtb arch/arm/boot/dts/zynq-zybo.dts
編譯 BusyBox
Busybox 是一個非常有趣的程式,舉凡我們在 Linux 下最常用的命令如 ls、cd 等到 sed、vi 他都具有相對應的簡單實現,此外,這些命令實際上都只是一個軟連結 (symlink) 連結到名為 busybox 的執行檔,也就是說,如果我們將 busybox 進行靜態編譯 (static link),則製作出來的系統整體大小大約為 2 MB (kernel) + 1.4 MB (busybox),而這個系統卻又可以具有許多 UN*X 下的常用命令,也因此 busybox 很常用於空間有限的系統。
我們在這個開發過程中,由於只是驗證執行 Linux 系統的功能,因此選用 Busybox 來作為我們的 rootfs。
首先先下載 Busybox 的原始碼,這裡選用 1_25_stable 這個穩定分支
git clone git://git.busybox.net/busybox -b 1_25_stable
進行我們自己的設定
coldnew@gentoo ~/busybox $ ARCH=arm CROSS_COMPILE=arm-xilinx-linux-gnueabi- make menuconfig
在進行設定時有以下幾點要確實注意,我們要將 busybox 編譯為靜態連結,並且增加 init 功能,主要設定如下:
Busybox Settings --->
Build Options --->
[*] Build BusyBox as a static binary (no shared libs)
Init Utilities --->
[*] init
Login/Password Management Utilities --->
[*] getty
Shells --->
[*] ash
設定完成後開始進行編譯
coldnew@gentoo ~/busybox $ ARCH=arm CROSS_COMPILE=arm-xilinx-linux-gnueabi- make
編譯完成後透過 make install 命令,會將編譯出來的 busybox 與軟連結(symlink)產生在 _install 資料夾內
coldnew@Rosia ~/busybox $ ARCH=arm CROSS_COMPILE=arm-xilinx-linux-gnueabi- make install
建立一些缺少的資料夾 (/dev、/sys …etc)
coldnew@gentoo ~/busybox $ cd _install && mkdir -p proc sys dev etc/init.d root
建立 etc/init.d/rcS 作為啟動腳本,並添加以下內容
coldnew@gentoo ~/busybox/_install $ vim etc/init.d/rcS #!/bin/sh mount -t proc none /proc mount -t sysfs none /sys /sbin/mdev -s
將 etc/init.d/rcS 加入可執行權限
coldnew@gentoo ~/busybox/_install $ chmod +x etc/init.d/rcS
建立 etc/inittab ,這會讓我們可以透過 UART 登入 zybo board
coldnew@gentoo ~/busybox/_install $ vim etc/inittab #!/bin/sh # Init script ::sysinit:/etc/init.d/rcS # Start shell on the serial ports ::respawn:/sbin/getty -L ttyPS0 115200 vt100 # What to do when restarting the init process ::restart:/sbin/init # What to do before rebooting ::shutdown:/bin/umount -a -r
設定預設的 /etc/passwd 檔案,我們要讓 root 用戶登入時不用輸入密碼
coldnew@gentoo ~/busybox/_install $ vim etc/passwd root::0:0:root:/root:/bin/sh
建立 /init 並軟連結到 /sbin/init ,避免 Linux Kernel 開機時找不到 rootfs 的 init。
coldnew@gentoo ~/busybox/_install $ ln -s /sbin/init init
接下來,由於這次我們只是要開機到 ramdisk 上的 rootfs, 因此將 busybox 做出的 rootfs 打包成 cpio 格式。
coldnew@gentoo ~/busybox/_install $ find . | sudo cpio -H newc -o | gzip -9 > ../uramdisk.cpio.gz
再透過 mkimage 將這個 uramdisk.cpio.gz 檔案轉成 uboot 用的 uramdisk.image.gz
coldnew@gentoo ~/busybox/_install $ mkimage -A arm -T ramdisk -C gzip -d ../uramdisk.cpio.gz ../uramdisk.image.gz Image Name: Created: Sun Jul 17 19:02:08 2016 Image Type: ARM Linux RAMDisk Image (gzip compressed) Data Size: 1042106 Bytes = 1017.68 kB = 0.99 MB Load Address: 00000000 Entry Point: 00000000
在這邊的這個 uramdisk.image.gz 就是我們開機會進入到的 rootfs,也是我們等等要放到 SD 卡第一個磁區的檔案。