ZYNQ生成uboot的时候和正常的ARM设备不太同样,ZYNQ属于二次辅助启动uboot而后由uboot启动内核,大概意思就是 ZYNQ内部有一个机制,该机制不可修改,能够经过拨码开关控制启动方式,好比从SD卡启动仍是从QSPI启动,SD卡中要包含uboot的镜像信息。最大的不一样就是,uboot编译完还不能够直接使用,还须要使用Vivado设计PL,再用SDK将uboot和设计PL的文件进行合成,最终合成后的文件拷贝到SD卡,由其启动。html
我不会FPGA,本文也只概述在Linux端,SD卡如何作,如何制做一个全新的Linux系统。linux
映像文件BOOT.BIN通常包括:FSBL,Bitstream和SSBL这三个文件,其中Bitstream是配置PL端程序,是可选项,在咱们制做Linaro系统的时候并不须要。FSBL是first stage boot loader,文件的制做须要使用Vivado环境;SSBL是Second Stage Boot Loader,这里使用的是Xilinx公司提供的u-boot。ios
来自参考文献1git
软件环境:Vivado 2017.02 Linux版本github
系统环境:Ubuntu 16.04 amd64shell
交叉编译器: gcc-linaro-7.3-2018.05.tar.xzubuntu
个人交叉编译环境放在/opt/toolschain/linaro/bin/arm-linux-gnueabihf-下,我编译的时候喜欢指定绝对编译器路径学习
获取xilinx的uboot源码:git clone https://github.com/Xilinx/u-boot-xlnx.git
ui
清除编译:make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm clean
.net
配置板级信息:make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm zynq_zc702_defconfig
板级信息在如图所示位置,个人是zc701的板子,可是没有,我就选择一个和这个最相近的。
menuconfig写入配置信息:make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm menuconfig
编译uboot:make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm -j8
编译成功后生产的是uboot,因此须要重命名uboot: mv uboot uboot.elf
拿到uboot.elf后,留存备用,再合成最终的boot程序须要这个uboot.elf文件。
大致流程就是:用Vivado这个软件新建工程,而后添加ip设计,配置时钟、配置一些Linux须要的基本外设(SD卡卡、串口、以太网等),使用wrap HDL功能生成顶层设计.v文件,而后编译.v文件生成.bit文件,再生成硬件描述文件,launch SDK软件,会自动生成一个工程,编译后拿到fsbl文件。
具体过程不少博友都已经写的很清楚了,我这里贴出一个讲的比较好的,能够按照这个方法作:在将来的多核通讯机制里面,PS和PL的通讯,则PL文件就是这样设计好以后而后咱们从新合成uboot文件。
https://blog.csdn.net/long_fly/article/details/78643258
咱们经过这样的方式拿到vivado编译生成的bit文件,而且在SDK里面创建了工程,生成了一个硬件平台,接下来咱们获取fsbl这个文件。fsbl文件须要在SDK里面创建一个FSBL工程,而且基于刚才咱们生成的硬件平台。
创建完以后直接编译,就能够拿到fsbl文件。
到目前位置,拿到了:
能够开始合成BOOT.bin文件了
这个操做仍是在sdk软件里面进行。
使用create boot image功能:
到此完成BOOT.bin的合成。
还有一个很是重要的事情,我试了不少次,zynq平台就是不启动,uboot也不输出任何的信息。这个小小的问题卡了我好久,不过在今天早上洗漱的时候,忽然想到,Xilinx Vivado和SDK都是在root状况下启动,生成BOOT.bin也是可能有权限问题。因此....
我拿到板子,而后在SD卡里面,给定sudo chmod 777 BOOT.bin
而后弹出SD卡,把SD放在ZYNQ上,竟然成功启动了。若是你是在Linux系统下,不要忘记给定BOOT.bin权限。
git clone https://github.com/Xilinx/linux-xlnx.git
make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm clean
make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm xilinx_zynq_defconfig
make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm menuconfig
进来以后退出就行。make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm -j8
make CROSS_COMPILE=/opt/toolschain/linaro/bin/arm-linux-gnueabihf- ARCH=arm uImage LOADADDR=0x00008000
在linux-xlnx/arch/arm/boot/dts目录内新建zynq-7010.dts文件,文件内容:
/dts-v1/; /include/ "zynq-7000.dtsi" / { model = "HLF"; compatible = "ALINX,zynq", "xlnx,zynq-7000"; aliases { ethernet0 = &gem0; serial0 = &uart1; spi0 = &qspi; mmc0 = &sdhci0; }; memory@0 { device_type = "memory"; reg = <0x0 0x20000000>; }; chosen { bootargs = ""; stdout-path = "serial0:115200n8"; }; usb_phy0: phy0 { compatible = "usb-nop-xceiv"; #phy-cells = <0>; reset-gpios = <&gpio0 46 1>; }; }; &clkc { ps-clk-frequency = <50000000>; }; &gem0 { status = "okay"; phy-mode = "rgmii-id"; phy-handle = <ðernet_phy>; ethernet_phy: ethernet-phy@0 { reg = <0>; }; }; &qspi { u-boot,dm-pre-reloc; status = "okay"; };
切换到内核的主目录里面:./scripts/dtc/dtc -I dts -O dtb -o ./arch/arm/boot/devicetree.dtb ./arch/arm/boot/dts/zynq-7010.dts
而后在linux-xlnx/arch/arm/boot/目录下便可发现devicetree.dtb文件,一样留着备用。
随便找个位置新建一个uEnv.txt 文件,文件内写入boot的配置信息:
uenvcmd=run linaro_sdboot linaro_sdboot=echo Copying Linux from SD to RAM... && \ fatload mmc 0 0x3000000 ${kernel_image} && \ fatload mmc 0 0x2A00000 ${devicetree_image} && \ if fatload mmc 0 0x2000000 ${ramdisk_image}; \ then bootm 0x3000000 0x2000000 0x2A00000; \ else bootm 0x3000000 - 0x2A00000; fi bootargs=console=ttyPS0,115200 root=/dev/mmcblk0p2 rw earlyprintk rootfstype=ext4 rootwait
保存,留着备用。
准备一张空白的超过8G的SD卡,读卡器读取该卡,咱们使用Linux系统进行格式化,Windows用户能够经过diskgen等格式化分区的软件制做也好。
sudo fdisk -l
假如查看到的SD卡是/dev/sde分区,(不要格式化错了,在我年轻的时候我曾经把整个硬盘都格式化了,很危险的操做,看清楚是/dev/sd* 后面是c 仍是d仍是e仍是f)。进入分区管理:sudo fdisk /dev/sde
如下步骤按照这个OMAPL138制做SD卡启动盘及重装Linux系统,个人这个博客来。注意不一样的是,咱们创建启动分区的大小是100M便可,建立boot的分区的类型也为Linux。
而后格式化boot分区:sudo mkfs.vfat -F 32 -n "boot" /dev/sde1
格式化rootfs分区:sudo mkfs.ext4 -L "rootfs" /dev/sde2
到此咱们完成了SD卡制做。
sd卡的boot分区:使用命令将 BOOT.bin / devicetree.dtb / uImage / uEnv.txt 四个文件拷贝到boot分区。
解压Linaro的文件系统: 在第一章写的 ARM端的Linaro文件系统:linaro-precise-ubuntu-desktop-20120723-305.tar.gz 解压到SD卡的root分区
sudo tar --strip-components=3 -C /media/delvis/rootfs -xzpf linaro-precise-ubuntu-desktop-20120723-305.tar.gz binary/boot/filesystem.dir
最好找一个带知识灯的读卡器,解压命令执行完了,不表明SD卡写入完毕,若是有指示灯,指示灯不闪烁以后弹出SD卡。
到此,一个完整的Linaro系统就写入了SD卡,将FPGA板子的boot拨码开关拨到SD卡启动位置,就能够看到Linaro系统启动了。
[1] long_fly, ZYNQ跑系统 系列(一) 传统方式移植linux, 2017年11月28日
[2] 雅可, Zedboard上运行Linaro系统(二):生成BOOT.BIN, 2016年07月26日
[3] 带你高飞, 03-ZYNQ学习(启动篇)之程序的固化, 2018年05月22日