• ZYNQ Linux 移植:包含petalinux移植和手动移植debian9


    参考:
    问题:
    1.ZYNQ Linux启动流程:
    BOOTROM→FSBL→Uboot→Kernel→RootFS

    环境:
    虚拟机ubuntu16.04
    petalinux2018.3
    vivado2018.3
    开发板:小熊猫z7020

    前言
    使用ZYNQ大概率会用到Linux。这里就对linux移植的整个流程做一个记录。
    移植有两种方式:
    (1)使用petalinux工具链进行移植。虚拟机需要装petalinux,自动化程度相对高。
     (2)手动移植,需安装SDK移植环境或者petalinux的环境,两者貌似是一样的。
    实际上,rootfs是可以随便用哪个的。比如,可以混搭手动移植的ubootkernel,而使用petalinux的rootfs。或者使用petalinux的image.ub,而使用debian等三方rootfs。看需要使用。
    手动移植可以更熟悉定制方面的内容,当然更为繁琐。

    • 建立BSP:
    首先我们从建立底层硬件平台开始,其定义了底层的设备树相关内容、PL端功能等。
    这里使用了网口0、SD卡、UART1,根据原理图分配管脚即可,网口和SD卡的管脚使用fast,设置好DDR信息。
    (1)配置ZYNQ:
    (2)正常综合编译生成bit导出BSP到文件夹。
    (3)打开SDK查看一下是否导入成功。

    • SD卡分区:本次移植u-boot、kernel、rootfs都在SD上。
    分两个区即可,BOOT分区和RootFS分区,前面的分区用于存储u-boot和内核和PL端的bit,后一个分区用于存储根文件系统。
    (1)在ubuntu中安装gparted。使用图形化界面简化操作,可直接在命令行中打开。

    (2)切换到sd卡,可根据容量确认。
    右键删除分区。
    新建BOOT分区:最起始保留4MB空间,这里分512M,文件格式fat32,标签设置为BOOT。
    剩下的全部分为RootFS分区。
    至此,分区完成。

    下述会讲述petalinux的移植方式和手动移植的方式。
    一、petalinux移植
    对于初学者,自然,petalinux是一种更为简便的方式,敲几个命令即可得到控制台的欢喜,而手动移植的坑还蛮多的。
    (1)导出前述bsp到虚拟机:
    (2)随便哪个位置新建个目录,并把sdk文件夹放入。这里新建个demo_linux文件夹。
    (3)source下petalinux的环境(可放在bashrc中自动source,避免手动敲)。路径根据petalinux的安装路径进行选择。
    source /home/kingstacker/petalinux2018.3/settings.sh
    (4)新建一个petalinux项目:这里新建了个demo1_linux工程,路径下会自动新建demo1_linux文件夹,模板使用zynq,zynq7系列应该都是这个。
    petalinux-create -t project -n demo1_linux --template zynq
    (5)cd到这个工程下面:
    cd demo1_linux/
    (6)导入上层的SDK文件夹内容:会自动弹出配置窗口。
    petalinux-config --get-hw-description=/home/kingstacker/demo_linux/project_1.sdk
    (7)配置使用哪个串口进行打印。这里我使用的是PS端的串口1,波特率115200,跟ZYNQ配置时保持一致即可。
    在Subsystem AUTO Hardware Settings选项下的Serial Settings设置。
    (8)设置bootargs,用于打印信息的串口指定,指定内核启动位置。注意:我这里使用了自动生成头,不使用自动的没得实验成功。
    在DTG Setting选项卡下第三行修改如下:
    键入bootargs:我这里使用了PS端的uart1(在设备树中被alias到了serial0),内核要从SD卡的第二个分区启动,分区格式为ext4。
    console=ttyPS0,115200 root=/dev/mmcblk0p2 rw earlyprintk rootfstype=ext4 rootwait
    在Image Packaging Configuration中设置根文件系统类型,这里为SD卡。

    切换到save保存主项配置,然后按两次ESC退出。

    根据需要配置u-bootkernel ootfs内容,这里不配置使用默认,直接运行build即可。
    petalinux-config -c u-boot
    petalinux-config -c kernel
    petalinux-config -c rootfs
    (9)编译工程,打把游戏回来再看:
    petalinux-build
    (10)切换到image目录下的linux目录,执行语句生成BOOT.BIN文件。
    petalinux-package --boot --format BIN --fsbl zynq_fsbl.elf --fpga system.bit --u-boot
    (11)复制BOOT.BIN和image.ub文件到SD卡的BOOT分区。
    我这里BOOT路径如下:
    cp BOOT.BIN /media/kingstacker/BOOT/
    cp image.ub /media/kingstacker/BOOT/
    (12)解压文件系统到SD卡的RootFS分区。
    sudo tar xvf rootfs.tar.gz -C /media/kingstacker/RootFS/
    至此,完成了所有内容,把SD卡放到板子上,切换板子启动模式为SD卡启动,连接CRT显示串口打印信息。

    其他:
    QEMU仿真:对于petalinux编译的系统,其提供了仿真工具,在上板之前就可以知道Uboot和Kernel是否可以启动。
    (1)在工程目录下执行下述打包命令:
    petalinux-package --prebuilt
    (2)进行第三阶段仿真:
    petalinux-boot --qemu --prebuilt 3
    可以看到在等待文件系统,则表示成功。
    也可以单独仿真uboot、kernel:
    petalinux-boot --qemu --u-boot
    petalinux-boot --qemu --kernel
    还有一些高级用法这里不表。

    二、手动移植linux:

    (1)获取u-bootkernel、device tree。
    • 下载xilinx的u-boot、kernel:
    git clone https://github.com/Xilinx/u-boot-xlnx.git
    git clone https://github.com/Xilinx/linux-xlnx.git
    git checkout 检出想使用的版本,git tag查看所有可用的版本。
    u-boot 2018.3貌似没得zynq_zc702_defconfig这个配置文件,可检出到2018.1拷贝一份。
    确保含有device tree文件,没有就要下载并导入到SDK的仓库中:
    git clone https://github.com/Xilinx/device-tree-xlnx
    (2)设备树编译:
    • 工程导入SDK生成设备树:就是新建个设备树工程。

    • 设备树信息导入linux中生成dtb文件:
    导入bsp信息:
    右键命令行中运行:
    首先source下SDK的settings.sh环境,source petalinux的sh文件也是一样的:
    归集dts文件到一个文件中:
    cpp -nostdinc -I include -I arch  -undef -x assembler-with-cpp  system-top.dts system-top.dts.preprocessed
    编译设备树,生成dtb:
    dtc -I dts -O dtb -i . -o devicetree.dtb system-top.dts.preprocessed
    可以看到文件夹下生成了dtb文件:

    (3)编译uboot:2018.3没得zc702,所以这里检出了2018.1的版本。
    这里可以使用zc702的defconfig文件,其在xil_source/u-boot-xlnx/configs路径下。
    CONFIG_ARM=y
    CONFIG_SYS_CONFIG_NAME="zynq_zc70x"
    CONFIG_ARCH_ZYNQ=y
    CONFIG_SYS_TEXT_BASE=0x4000000
    CONFIG_SYS_MALLOC_F_LEN=0x800
    CONFIG_IDENT_STRING=" Xilinx Zynq ZC702"
    CONFIG_SPL_STACK_R_ADDR=0x200000
    CONFIG_DEFAULT_DEVICE_TREE="zynq-zc702"
    CONFIG_DEBUG_UART=y
    CONFIG_DISTRO_DEFAULTS=y
    CONFIG_FIT=y
    CONFIG_FIT_SIGNATURE=y
    CONFIG_FIT_VERBOSE=y
    CONFIG_BOOTCOMMAND="run $modeboot || run distro_bootcmd"
    # CONFIG_DISPLAY_CPUINFO is not set
    CONFIG_SPL=y
    CONFIG_SPL_STACK_R=y
    CONFIG_SPL_OS_BOOT=y
    CONFIG_SYS_PROMPT="Zynq> "
    CONFIG_CMD_THOR_DOWNLOAD=y
    CONFIG_CMD_EEPROM=y
    CONFIG_CMD_MEMTEST=y
    CONFIG_CMD_DFU=y
    # CONFIG_CMD_FLASH is not set
    CONFIG_CMD_FPGA_LOADBP=y
    CONFIG_CMD_FPGA_LOADFS=y
    CONFIG_CMD_FPGA_LOADMK=y
    CONFIG_CMD_FPGA_LOADP=y
    CONFIG_CMD_GPIO=y
    CONFIG_CMD_I2C=y
    CONFIG_CMD_MMC=y
    CONFIG_CMD_SF=y
    CONFIG_CMD_USB=y
    # CONFIG_CMD_SETEXPR is not set
    CONFIG_CMD_TFTPPUT=y
    CONFIG_CMD_CACHE=y
    CONFIG_CMD_EXT4_WRITE=y
    CONFIG_OF_EMBED=y
    #CONFIG_ENV_IS_IN_SPI_FLASH=y
    CONFIG_NET_RANDOM_ETHADDR=y
    CONFIG_SPL_DM_SEQ_ALIAS=y
    CONFIG_DFU_MMC=y
    CONFIG_DFU_RAM=y
    CONFIG_FPGA_XILINX=y
    CONFIG_DM_GPIO=y
    CONFIG_MMC_SDHCI=y
    CONFIG_MMC_SDHCI_ZYNQ=y
    CONFIG_SPI_FLASH=y
    CONFIG_SPI_FLASH_BAR=y
    CONFIG_SF_DUAL_FLASH=y
    CONFIG_SPI_FLASH_ISSI=y
    CONFIG_SPI_FLASH_MACRONIX=y
    CONFIG_SPI_FLASH_SPANSION=y
    CONFIG_SPI_FLASH_STMICRO=y
    CONFIG_SPI_FLASH_WINBOND=y
    CONFIG_PHY_MARVELL=y
    CONFIG_PHY_REALTEK=y
    CONFIG_PHY_XILINX=y
    CONFIG_ZYNQ_GEM=y
    CONFIG_DEBUG_UART_ZYNQ=y
    CONFIG_DEBUG_UART_BASE=0xe0001000
    CONFIG_DEBUG_UART_CLOCK=50000000
    CONFIG_ZYNQ_SERIAL=y
    CONFIG_ZYNQ_QSPI=y
    CONFIG_USB=y
    CONFIG_USB_EHCI_HCD=y
    CONFIG_USB_ULPI_VIEWPORT=y
    CONFIG_USB_ULPI=y
    CONFIG_USB_STORAGE=y
    CONFIG_USB_GADGET=y
    CONFIG_USB_GADGET_MANUFACTURER="Xilinx"
    CONFIG_USB_GADGET_VENDOR_NUM=0x03fd
    CONFIG_USB_GADGET_PRODUCT_NUM=0x0300
    CONFIG_CI_UDC=y
    CONFIG_USB_GADGET_DOWNLOAD=y
    因为没有用到flash则并修改下述:关掉 不然启动时候会卡死在SPI flash初始化部分
    清除中间编译:
    make distclean
    使用配置文件:
    make CROSS_COMPILE=arm-linux-gnueabihf- zynq_zc702_defconfig
    通过下述指令可在界面中uboot进行进一步修改配置:改defconfig文件也可以
    make CROSS_COMPILE=arm-linux-gnueabihf- menuconfig
    工具编译:
    make CROSS_COMPILE=arm-linux-gnueabihf- tools
    编译u-boot:
    make CROSS_COMPILE=arm-linux-gnueabihf-

    最后把编译生成的u-boot后缀改成.elf。

    (4)生成FSBL文件并合成BOOT.BIN:
    • 在SDK中先生新建fsbl工程成FSBL,在src的h文件中添加debug属性,可以在控制台中打印FSBL阶段的启动信息:
    #define FSBL_DEBUG_INFO

    SDK中合成BOOT.BIN文件:uboot.elf、fsbl.elf、bit。
    复制BOOT.BIN到SD卡的BOOT分区。上电确认BOOT.BIN是否可以正常启动。
    (5)内核编译:
    切换到linux-xlnx目录。
    注意:错误的解决:

    解决 "mkimage" command not found - U-Boot images will not be built

    sudo apt-get install u-boot-tools
    或者#export PATH=${YOUR_UBOOT_DIR}/tools:$PATH  //编译内核如果要生成uImage,则需要用到mkimage工具,该工具在u-boot/tools下有提供
    或者直接复制mkimage到/bin目录即可,生成uimage会用到。
    清除老的编译文件:
    #make distclean
    配置:
    #make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- xilinx_mz7x_defconfig  
    根据需要对内核进行图形化界面的配置:暂时默认就好
    #make ARCH=arm menuconfig
    编译工具:
    #make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- prepare scripts  
    编译内核生成uimage:
    #make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- UIMAGE_LOADADDR=0x8000 uImage  
    (6)定制文件系统:
    • 文件系统:
    这里文件系统使用debian,也可用别的,无所谓。
    安装arm环境和debootstrap:
    sudo apt-get install binfmt-support qemu qemu-user-static debootstrap
    debian提取:
    sudo debootstrap --arch=armhf --foreign stretch rootfs http://cdn.debian.net/debian
    拷贝到bin路径:
    cp /usr/bin/qemu-arm-static {{刚刚rootfs目录}}/usr/bin
    在rootfs文件夹的上层执行:
    DEBIAN_FRONTEND=noninteractive DEBCONF_NONINTERACTIVE_SEEN=true LC_ALL=C LANGUAGE=C LANG=C chroot rootfs debootstrap/debootstrap --second-stage
    进入rootfs:
    chroot rootfs
    添加源到list:
    echo deb http://ftp.cn.debian.org/debian/ stretch main > /etc/apt/sources.list
    更新:
    apt-get update
    按需安装相关包:
    apt-get install vim sudo net-tools

    debian9启动后网口默认没有自动挂载,开发者可以指定rc.local(注意给权限777)文件中自动执行脚本,使用脚本配置网口,脚本注意给权限chmod +x xx.sh。
    而debian9默认不带rc.local。解决方法:https://www.cnblogs.com/flymeng/p/7901062.html
    比如我这里新建/etc/rc.local文件:放入了sh_boot.sh文件。
    #!/bin/sh -e
    #
    #rc.local
    
    #sh in there
    /home/sh_boot.sh
    #sh end 
    exit 0
    EOF
    给权限:chmod 777 /etc/rc.local
    在home路径放入sh文件,即可。(注意别放在普通用户文件夹下,否则无法上电就执行了)配置网口0使用,根据需要修改。
    echo "Welcome to use,powerd by kingstacker"
    echo "config the eth......."
    
    ifconfig eth0 192.168.0.110 netmask 255.255.255.0 up
    
    echo "config finish."
    给权限:chmod 777 /home/sh_boot.sh
    命令行执行exit 退出。

    • 打包rootfs备份:
    文件夹压缩:
    切换到rootfs路径,执行:
    tar -zcvf debian9_rootfs.tar.gz ./

    文件夹解压到SD卡的rootfs:
    tar zxvf debian9_rootfs.tar.gz -C /media/kingstacker/RootFS

    debian9 rootfs文件坚果云分享:
    https://www.jianguoyun.com/p/DdhhnukQ15CBCBjvgrcD 
    普通用户登录及密码:kingstacker
    su登录密码:123123


    (7)复制相关文件到SD卡的BOOT分区,并添加uEnv.txt文件:
    • SD卡BOOT分区内容如下:bin文件、设备树、txt、内核镜像。
    uEnv.txt:其指定了内核传参。表示uart速率115200,使用ttyPS0。根目录为SD卡的第二个分区。
    bootargs=console=ttyPS0,115200 root=/dev/mmcblk0p2 rw earlyprintk rootfstype=ext4 rootwait
    load_image=fatload mmc 0 ${kernel_load_address} ${kernel_image} && fatload mmc 0 ${devicetree_load_address} ${devicetree_image}
    uenvcmd=echo Copying Linux from SD to RAM... && mmcinfo &&  run load_image && bootm ${kernel_load_address} - ${devicetree_load_address}

    而SD卡的RootFS分区存储了根文件系统:

    至此,移植结束,插到板子上,享受劳动成果吧。

    以上。






  • 相关阅读:
    memset()函数,多用于清空数组
    Echart 词云图 上手代码 同含(echarts-wordcloud.js)最简单的教程 复制可用
    爬虫使用真实浏览器打开网页进行爬取
    jsoup 模拟登陆github网页(源代码)亲测可用 直接复制就能用
    拷贝虚拟电脑 Ubuntu 系统 含hadoop hive hbase mysql spark eclipse
    Python 连接MySQL 增删改查 直接可用(最简易,含源码)
    Python 中文词频统计,热词统计,简要分析(含上手源码)
    百度百科简介爬取(含源代码、信息领域词频数据csv格式)
    博客园博文爬取 标签爬取(含源代码)
    输入一行电报文字,将字母变成其下一字母
  • 原文地址:https://www.cnblogs.com/kingstacker/p/13571942.html
Copyright © 2020-2023  润新知