Ubuntu 应用层支持 视频硬件编解码支持 Mpp 是 Rockchip 为 RK3399 提供的一套视频编解码的 API, 并且基于 mpp, Rockchip 提供了一套 gstreamer 的编解码插件。用户可以根据自己的需求,基 于 gstreamer 来做视频编解码的应用,或者直接调用 Mpp,来实现硬件的编解 码加速。 Firefly 发布的 Ubuntu 系统, 都已经提供了完整的 gstremaer 和 Mpp 支持 ,并且提供了相应的例子,供用户开发参考。 Gstreamer Ubuntu 16.04 下,gstreamer 1.12 已经安装在 /opt/ 目录下。 Ubuntu 18.04 下,gstreamer 1.12 已经安装到系统中。 /usr/local/bin/h264dec.sh 测试硬件 H264 解码。 /usr/local/bin/h264enc.sh 测试硬件 H264 编码。 用户可以参照这两个脚本,配置自己的 gstreamer 应用。 Mpp Ubunut 系统下, Mpp 相关 dev 包都已经安装到系统中。 /opt/mpp/ 下分别 是 Mpp 编解码的相关例子和源文件。 OpenGL-ES RK3399 支持 OpenGL ES1.1/2.0/3.0/3.1。 Firefly 发布的 Ubuntu 系统, 都已经提供了完整的 OpenGL-ES 支持。运行 "glmark2-es2" 可以测试 OpenGL-ES 支持。 如果要避免屏幕刷新率对测试结果 的影响,可以在串口终端上使用以下命令测试。 # systemctl stop lightdm # export DISPLAY=:0 # Xorg & # glmark2-es2 –off-screen 在 Chromium 浏览器中, 在地址栏输入:"chrome://gpu" 可以查看 chromium 下硬件加速的支持。 注意: EGL 是用 ARM 平台上 OpenGL 针对 x window system 的扩展,功能等效于 x86 下的 glx 库。 由于 Xorg 使用的 Driver modesettings 默认会加载 libglx.so(禁用 glx 会 导致某些通过检测 glx 环境的应用启动失败),libglx.so 会搜索系统中的 dri实现库。但是 rk3399 Xorg 2D 加速是直接基于 DRM 实现, 并未实现 dri 库,所以启动过程中,libglx.so 会报告如下的错误 。 (EE) AIGLX error: dlopen of /usr/lib/aarch64-linux-gnu/dri/rockchip_dri.so failed 这个 对系统运行没有任何影响,不需要处理。 基于同样的道理,某些应用启动过程中,也会报告如下错误,不用处理,对应用 的运行不会造成影响。 libGL error: unable to load driver: rockchip_dri.so libGL error: driver pointer missing libGL error: failed to load driver: rockchip Firefly 之前发布的某些版本的 Ubuntu 软件,默认关闭了加载 libglx.so,在 某些情况下,运行某些应用程序会出现下述错误: GdkGLExt-WARNING **: Window system doesn't support OpenGL. 修正的方法如下: 删除 /etc/X11/xorg.conf.d/20-modesetting.conf 中一下三行配置。 Section "Module" Disable "glx" EndSection OpenCL Firefly 发布的 Ubuntu 系统,已经添加了opencl1.2 支持,可以运行系统内置 的 "clinfo" 获取平台 opencl 相关参数。 firefly@firefly:~$ clinfo Platform #0 Name: ARM Platform Version: OpenCL 1.2 v1.r14p0-01rel0-git(966ed26).f44c85cb3d2ceb87e8be88e7592755c3 Device #0 Name: Mali-T860 Type: GPU Version: OpenCL 1.2 v1.r14p0-01rel0-git(966ed26).f44c85cb3d2ceb87e8be88e7592755c3 Global memory size: 1 GB 935 MB 460 kB Local memory size: 32 kB Max work group size: 256 Max work item sizes: (256, 256, 256) … OpenCV Face-RK3399 发布的 Ubuntu 系统中,已经添加了 OpenCV-3.4.5 的支持,相关 的测试文件为 "/usr/loacl/bin/camera.py" 和 "/usr/local/bin/test_opencv_rkisp.sh"。摄像头设备注册成功后,直接运行 "test_opencv_rkisp.sh" 即可预览摄像头。 文件 "/usr/loacl/bin/camera.py" 内容如下: import numpy as np import cv2 #preview /dev/video0 #cap = cv2.VideoCapture("rkisp device=/dev/video0 io-mode=1 analyzer=1 enable-3a=1 path-iqf=/etc/cam_iq.xml ! video/x-raw,format=NV12,width=640,height=480, framerate=30/1 ! videoconvert ! appsink", cv2.CAP_GSTREAMER) #preview /dev/video5 cap = cv2.VideoCapture("rkisp device=/dev/video5 io-mode=1 analyzer=1 enable-3a=1 path-iqf=/etc/cam_iq.xml ! video/x-raw,format=NV12,width=640,height=480, framerate=30/1 ! videoconvert ! appsink", cv2.CAP_GSTREAMER) if not cap.isOpened(): print('VideoCapture not opened') exit(0) while(True): # Capture frame-by-frame ret, frame = cap.read() # Display the resulting frame cv2.imshow('Hello RKISP-OpenCv',frame) if cv2.waitKey(1) & 0xFF == ord('q'): break # When everything done, release the capture cap.release() cv2.destroyAllWindows() 文件 "/usr/loacl/bin/test_opencv_rkisp.sh" 内容如下: #!/bin/sh export DISPLAY=:0.0 export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/lib/aarch64-linux-gnu/gstreamer-1.0 #export GST_DEBUG=ispsrc:5 #export GST_DEBUG_FILE=/tmp/2.txt echo "Start RKISP Camera Preview!" #echo 7 > /sys/module/video_rkisp1/parameters/debug su root -c "python3 /usr/local/bin/camera.py" 运行脚本,结果如图所示: TensorFlow Lite RK3399 支持神经网络的 GPU 加速方案 LinuxNN, Firefly 发布的 Ubuntu 系 统,已经添加了 LinuxNN 的支持。 在 "opt/tensorflowbin/" 下,运行 "test.sh", 即可测试 MobileNet 模型图 像分类器的 Demo 和 MobileNet-SSD 模型的目标检测 Demo firefly@firefly:/opt/tensorflowbin$ ./test.sh Loaded model mobilenet_ssd.tflite resolved reporter nn version: 1.0.0 findAvailableDevices filename:libarmnn-driver.so d_info:40432 d_reclen:40s [D][ArmnnDriver]: Register Service: armnn (version: 1.0.0)! first invoked time: 1919.17 ms invoked average time: 108.4 ms validCount: 26 car @ (546, 501) (661, 586) car @ (1, 549) (51, 618) person @ (56, 501) (239, 854) person @ (332, 530) (368, 627) person @ (391, 541) (434, 652) person @ (418, 477) (538, 767) person @ (456, 487) (602, 764) car @ (589, 523) (858, 687) person @ (826, 463) (1034, 873) bicycle @ (698, 644) (1128, 925) write out.jpg succ! 屏幕旋转 Firefly 发布的 Ubuntu 系统,如果需要默认对系统的显示方向做旋转,可以在 "/etc/default/xrandr" 中修改对应的显示设备的方向即可。 firefly@firefly:~$ cat /etc/default/xrandr #!/bin/sh # Rotation can be one of 'normal', 'left', 'right' or 'inverted'. # xrandr --output HDMI-1 --rotate normal # xrandr --output LVDS-1 --rotate normal # xrandr --output EDP-1 --rotate normal # xrandr --output MIPI-1 --rotate normal # xrandr --output VGA-1 --rotate normal # xrandr --output DP-1 --rotate normal 对于配有触摸屏的平台,如果需要对触摸屏的方向做旋转,可以在 "/etc/X11/xorg.conf.d/05-gslX680.conf" 中修改 SwapAxes / InvertX / InvertY 三个值。 firefly@firefly:~$ cat /etc/X11/xorg.conf.d/05-gslX680.conf Section "InputClass" Identifier "gslX680" MatchIsTouchscreen "on" MatchProduct "gslX680" Driver "evdev" Option "SwapAxes" "off" # Invert the respective axis. Option "InvertX" "off" Option "InvertY" "off" EndSection 屏幕键盘 官方的 Ubuntu 系统中自带屏幕键盘,可以在菜单栏中点击打开: 声音配置 开发板一般都有2个或以上的音频设备。常见的是耳机和 HDMI 这两个设备的音 频输出。下面是音频设置的例子,供用户参考。 从命令行指定音频设备 系统自带的音频文件在 "/usr/share/sound/alsa/" 目录中,播放前请先查看声 卡设备: root@firefly:~# aplay -l **** List of PLAYBACK Hardware Devices **** card 0: rockchiprt5640c [rockchip,rt5640-codec], device 0: ff890000.i2s-rt5640-aif1 rt5640-aif1-0 [] Subdevices: 1/1 Subdevice #0: subdevice #0 card 1: rkhdmidpsound [rk-hdmi-dp-sound], device 0: HDMI-DP multicodec-0 [] Subdevices: 1/1 Subdevice #0: subdevice #0 然后指定声卡播放音频,其中声卡 card0 的设备0对应的是耳机口,card1 的设 备 0 对应的是 HDMI。一般播放音频命令是 "aplay -Dhw:0,0 Fornt_Center.wav",但系统自带的音频文件是单声道的,所以为了防止播放失 败,可以按照下面的命令进行播放: #选择耳机口输出音频文件 root@firefly:/usr/share/sounds/alsa# aplay -Dplughw:0,0 Front_Center.wav Playing WAVE 'Front_Center.wav' : Signed 16 bit Little Endian, Rate 48000 Hz, Mono #选择 HDMI 输出音频文件 root@firefly:/usr/share/sounds/alsa# aplay -Dplughw:1,0 Front_Center.wav Playing WAVE 'Front_Center.wav' : Signed 16 bit Little Endian, Rate 48000 Hz, Mono 在图形界面中选择音频设备 在图形界面中,播放准备好的音频文件,然后点击声音图标,打开 "Sound Setting",选择到 "Configuration" 。可以看到两个声卡设备,例如设置为 HDMI 输出音频,则把 HDMI 的声卡设备选择为 "Output",另一个声卡设置为 "Off"。(如果 HDMI 无声或者声音小,可以试试按 HDMI 屏上的物理按键调高 音量) 开机启动应用程序 在系统中,可以根据用户需求自行设置应用程序开机自动启动。 在系统菜单栏中依次选择:"Preferences" -> "Default applications for LXSession",打开设置界面。 在设置界面的 "Launching applocations" 一栏中可以设置应用默认的打开方式 : 在 "Autostart" 一栏中就可以选择用户需要开机启动的应用。例如蓝牙默认是 开机不启动的,如果需要让它开机自动启动,就把 "Blueman Applet" 的勾打上 ,就可以实现开机自动启动蓝牙: ADB 调试 ADB,全称 Android Debug Bridge,是 Android 的命令行调试工具,可以完成 多种功能,如跟踪系统日志,上传下载文件,安装应用等。以 Firefly-RK3399 为例,为了在 Ubuntu 系统也能使用 ADB 工具进行调试,我们移植了 ADB 服务 。但由于并非 Android 系统,很多 ADB 命令类似 "adb logcat"、"adb install" 等不能使用,仅作为普通的调试辅助工具,可以进行 shell 交互、上 传下载文件等操作。同样的网络远程 ADB 调试也不能使用。 下载 ADB 在 Ubuntu 系统中,执行: sudo apt-get install android-tools-adb ADB 连接 安装了 ADB 后,用 Micro USB OTG 线连接开发板和 PC 机。然后通过命令 "adb devices" 查看是否有设备连接: firefly@Desktop:~$ adb devices List of devices attached 0123456789ABCDEF device 从返回的信息可以看到已经查找到了设备,表明设备已经成功连接。 设备连接成功后,输入命令 "adb shell" 即可进入命令行模式: firefly@Desktop:~$ adb shell # 此状态下是看不到命令行当前的路径,同时 Tab 键补全功能失效,需要进入一 个用户中才可以正常操作。 firefly@Desktop:~$ adb shell # su firefly To run a command as administrator (user "root"), use "sudo ". See "man sudo_root" for details. firefly@firefly:/$ #到这里命令行就可以正常使用了 用户也可以使用命令 "adb shell /bin/bash",也可以进入正常的命令行模式。 可以输入 "adb help" 查看命令行帮助信息,注意并不是所有命令都可以使用, 帮助信息只做参考。 串口 开发板上的串口分两种,一种是普通串口另外一种是通过转换芯片把 SPI 转换 而成的串口。转换而成的串口和普通串口功能完全一致但是需要注意它们的设备 文件名是不一样的。下面是两者的区别: //普通串口 root@firefly:~# ls /dev/ttyS* ttyS0 ttyS1 ttyS2 ttyS3 //SPI转串口 root@firefly:~# ls /dev/ttysWK* ttysWK0 ttysWK1 ttysWK2 ttysWK3 设置波特率 以 "ttyS4" 为例,查看串口波特率命令: root@firefly:~# stty -F /dev/ttyS4 speed 9600 baud; line = 0; -brkint -imaxbel 设置波特率命令: //设置波特率为115200 root@firefly:~# stty -F /dev/ttyS4 ospeed 115200 ispeed 115200 cs8 //查看确认是否已经修改 root@firefly:~# stty -F /dev/ttyS4 speed 115200 baud; line = 0; -brkint -imaxbel root@firefly:~# 关闭回显 在做环回收发测试的时候回显功能会影响到我们的测试结果,所以在环回测试或 者其他特殊情况下需要关闭回显。以下是关闭回显的命令: //关闭回显示 root@firefly:~# stty -F /dev/ttyS4 -echo -echoe -echok //查看所有功能的配置,检查是否已经关闭。“-”号代表该功能已经关闭 root@firefly:~# stty -F /dev/ttyS4 -a | grep echo isig icanon iexten -echo -echoe -echok -echonl -noflsh -xcase -tostop -echoprt echoctl echoke -flusho -extproc 发送接收原始数据 在实际应用中实际发送数据和我们希望发送的数据可能存在差别,例如多了回车 或者其他特殊字符。利用 "stty" 可以把发送接收设置成 "raw" 模式确保发送 接收的是原始数据,以下是设置命令: root@firefly:~# stty -F /dev/ttyS4 raw 工作模式 串口有 2 种工作模式分别是中断模式和 DMA 模式。 中断模式 内核默认把串口配置成中断模式所以不需要做任何修改。中断模式下传输速率比 较快但是在传大量数据的时候很容易丢包或者出错,所以当数据量比较大的时候 请不要使用中断模式。 DMA 模式 DMA 模式主要是传输大量数据的时候使用。内核会为串口提供一个缓存空间接收 数据来尽量降低串口传输的丢包率。 注意: 缓存空间限默认大小为 "8K" 如果一次传输超过缓存大小就会丢包,所以使用 DMA 模式的话需要发送端需要分包发送。 设备树文件配置 &uart4 { status = "okay"; + dmas = <&dmac_peri 8>, <&dmac_peri 9>; + dma-names = "tx", "rx"; }; DMA 模式并不能提高传输速率,相反因为需要缓存,传输速率会有所下降,所以 如果不是需要传输大量数据的话不要使用 DMA 模式。 流控 不管是中断模式还有 DMA 模式都无法保证数据传输万无一失,因为长时间传输 大量数据的时候 DDR、CPU 变频或者占用率过高都可能导致上层处理数据不及时 导致丢包,这个时候就需要是用流控了。流控分两种,一种是软件流控另一种为 硬件流控,下面只介绍硬件流控的使用。 硬件支持 硬件流控需要有硬件支持,开发板的串口 "CTX" 和 "RTX" 脚需要和设备相连。 注意: 开发板不是所有串口都支持硬件流控,请先从原理图上确认硬件是否支持 设备树文件配置 uart3: serial@ff1b0000 { compatible = "rockchip,rk3399-uart", "snps,dw-apb-uart"; reg = <0x0 0xff1b0000 0x0 0x100>; clocks = <&cru SCLK_UART3>, <&cru PCLK_UART3>; clock-names = "baudclk", "apb_pclk"; interrupts = ; reg-shift = <2>; reg-io-width = <4>; pinctrl-names = "default"; + pinctrl-0 = <&uart3_xfer &uart3_cts &uart3_rts>; status = "disabled"; }; 应用层设置 上层也需要打开流控的设置,这里介绍 stty 如何打开流控如果你使用的是其他 应用程序请从应用中打开流控。 //打开流控 root@firefly:~# stty -F /dev/ttyS4 crtscts //检测流控是否打开,“-”号代表该功能已经关闭 root@firefly:~# stty -F /dev/ttyS4 -a | grep crtscts -parenb -parodd -cmspar cs8 hupcl -cstopb cread clocal crtscts USB 以太网 USB 以太网,主要实现的是将开发板的 OTG 接口做外设模式,模拟成一个网络 接口,然后主机通过 USB 连接开发板并通过开发板访问互联网。以下是基于 Firefly-RK3399 开发板进行的具体操作。 操作环境: Ubuntu 系统的 PC 机 Firefly-RK3399 开发板 内核设置 在内核目录下,打开内核配置选项菜单: make firefly_linux_defconfig make menuconfig 进入内核配置菜单后依次选择:"Device Drivers" -> "USB Support" -> "USB Gadget Support" 在 "USB Gadget Driver" 选项中选择到 "Ethernet Gadget (with CDC Ethernet support)" 同时再选择上 "RNDIS support" 。 <*> USB Gadget Drivers (Ethernet Gadget (with CDC Ethernet support)) ---> Ethernet Gadget (with CDC Ethernet support) [*] RNDIS support (NEW) 接着在 "kernel" 目录下编译内核: make rk3399-firefly.img -j12 编译完成后将内核烧录到开发板中,烧录过程请参考维基教程: 升级固件 中的分区镜像烧写部分。 IP 地址设置 用数据线连接 PC 机和开发板的 OTG 接口,在 PC 机中执行 "lsusb" 命令可以 查看到 USB 以太网设备,即说明连接成功。 firefly@Desktop:~$ lsusb Bus 002 Device 003: ID 09da:5814 A4Tech Co., Ltd. Bus 002 Device 002: ID 8087:0024 Intel Corp. Integrated Rate Matching Hub Bus 002 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub Bus 001 Device 005: ID 04f2:b2ea Chicony Electronics Co., Ltd Integrated Camera [ThinkPad] Bus 001 Device 004: ID 0a5c:21e6 Broadcom Corp. BCM20702 Bluetooth 4.0 [ThinkPad] Bus 001 Device 003: ID 147e:1002 Upek Biometric Touchchip/Touchstrip Fingerprint Sensor Bus 001 Device 002: ID 8087:0024 Intel Corp. Integrated Rate Matching Hub Bus 001 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub Bus 004 Device 001: ID 1d6b:0003 Linux Foundation 3.0 root hub Bus 003 Device 003: ID 0525:a4a2 Netchip Technology, Inc. Linux-USB Ethernet/RNDIS Gadget Bus 003 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub 输出信息中 ID 0525:a4a2 Netchip Technology, Inc. Linux-USB Ethernet/RNDIS Gadget 即为 USB 网卡设备。 开发板插入网线,使开发板可以连接外网。 在开发板中 IP 的设置: 输入执行 "ifconfig -a" 命令,可以查看到以下信息: root@firefly:~# ifconfig -a # eth0 是有线网卡 eth0: flags=4163 mtu 1500 inet 168.168.100.48 netmask 255.255.0.0 broadcast 168.168.255.255 inet6 fe80::1351:ae2f:442e:e436 prefixlen 64 scopeid 0x20 ether 8a:4f:c3:77:94:ac txqueuelen 1000 (Ethernet) RX packets 9759 bytes 897943 (897.9 KB) RX errors 0 dropped 0 overruns 0 frame 0 TX packets 236 bytes 35172 (35.1 KB) TX errors 0 dropped 0 overruns 0 carrier 0 collisions 0 device interrupt 42 base 0x8000 ... # usb0 是虚拟的 usb 网卡 usb0: flags=4098 mtu 1500 ether 4a:81:b1:34:d2:ad txqueuelen 1000 (Ethernet) RX packets 0 bytes 0 (0.0 B) RX errors 0 dropped 0 overruns 0 frame 0 TX packets 0 bytes 0 (0.0 B) TX errors 0 dropped 0 overruns 0 carrier 0 collisions 0 然后给 usb0 网卡自定义一个适当的 IP: 注意要设置 usb0 的 IP 和有线网卡 eth0 的 IP 不在同一网段!! ifconfig usb0 192.168.1.101 在 PC 机中 IP 的设置: # 先查看 USB 虚拟网卡 firefly@Desktop:~$ ifconfig enp0s20u2i1: flags=4163 mtu 1500 inet 192.168.2.90 netmask 255.255.255.0 broadcast 192.168.2.255 inet6 fe80::871c:b87e:1327:7fd4 prefixlen 64 scopeid 0x20 ether 46:fe:6e:97:ee:a6 txqueuelen 1000 (以太网) RX packets 0 bytes 0 (0.0 B) RX errors 0 dropped 0 overruns 0 frame 0 TX packets 1 bytes 54 (54.0 B) TX errors 0 dropped 0 overruns 0 carrier 0 collisions 0 ... # 设置 USB 网卡的 IP 地址和开发板的 usb0 的 IP 地址在同一网段 firefly@Desktop:~$ sudo ifconfig enp0s20u2i1 192.168.1.100 #设置默认网关:要设置为开发板 usb0 的 ip 地址,因为后面要通过 usb0 来进行流量的转发 firefly@Desktop:~$ sudo route add default gw 192.168.1.101 设置完开发板和 PC 机的 IP 后,互相是可以 ping 通的,PC 机也可以用 "ssh" 命令登录到开发板。 网络共享实现 PC 机上网 在开发板上:首先打开 IPv4 的转发功能: echo 1 > /proc/sys/net/ipv4/ip_forward 如果希望每次重启开发板后都自动打开转发功能,请直接修改 "/etc/sysctl.conf" 文件的 "net.ipv4.ip_forward" 值为1。修改文件参数后 要执行 "sysctl -p" 命令重新载入 "/etc/sysctl.conf" 文件,使 IPv4 转发 功能生效。 然后清空 iptables 默认规则,并添加规则进行流量转发: #清空默认规则 iptables -F iptables -X iptables -Z #流量转发规则设置 iptables -t nat -A POSTROUTING -s 192.168.1.0/24 -o eth0 -j SNAT --to-source 168.168.100.48 现在 PC 主机就可以访问网络了,配置过程中要注意以下几点: 对应好上述步骤中自己设备上的各个 IP 地址,注意开发板上的 USB 虚拟网卡 IP 和有线网络 IP 不在同一网段; PC 机虚拟 USB 网卡的网关要设置为开发板虚拟 USB 网卡的 IP 地址; 开发板上的虚拟网卡 IP 地址、IP 转发功能、流量转发规则等设置会在设备重 启后恢复,用户可以自行查找资料如何开机自动设置这些配置。 外部存储设备 rootfs 根文件系统除了可以使用在内部的 eMMC 中的,还可以使用外部存储设备的根文 件系统,如 SD 卡,U 盘等。以下是以 SD 卡为例,在 Firefly-RK3399 开发板 上实现挂载外部存储设备的根文件系统。 在 SD 卡上建立分区 在 PC 机上插入 SD 卡,用 "gdisk" 工具分出一个装载根文件系统的分区: #用户请用 `fdisk -l` 查询 SD 卡设备名是什么,在用 gdisk 命令进入对应的设备 sudo gdisk /dev/sdb GPT fdisk (gdisk) version 1.0.3 Partition table scan: MBR: protective BSD: not present APM: not present GPT: present Found valid GPT with protective MBR; using GPT. Command (? for help): ------------------------------------------------------------------- #输入 ? 查看帮助信息 # 按 p 显示 SD 卡当前已有分区,由于使用的 SD 卡暂未创建分区,所以没有查询到分区信息 Command (? for help): p Disk /dev/sdb: 15278080 sectors, 7.3 GiB Model: CARD-READER Sector size (logical/physical): 512/512 bytes Disk identifier (GUID): 5801AE61-92ED-42FB-A144-E522E8E15827 Partition table holds up to 128 entries Main partition table begins at sector 2 and ends at sector 33 First usable sector is 34, last usable sector is 15278046 Partitions will be aligned on 2048-sector boundaries Total free space is 15278013 sectors (7.3 GiB) Number Start (sector) End (sector) Size Code Name ------------------------------------------------------------------- #创建新的分区,请用户根据自身实际情况创建合适的分区大小给根文件系统 Command (? for help): n #输入 n 创建新分区 Partition number (1-128, default 1): 1 #输入1创建第一个分区 First sector (34-15278046, default = 2048) or {+-}size{KMGTP}: #直接按回车,使用默认值 Last sector (2048-15278046, default = 15278046) or {+-}size{KMGTP}: +3G #创建分区大小为3G Current type is 'Linux filesystem' Hex code or GUID (L to show codes, Enter = 8300): #按回车使用默认值 Changed type of partition to 'Linux filesystem' ------------------------------------------------------------------- #创建完成后,输入 i,可以查看分区的 unique GUID Command (? for help): i Using 1 Partition GUID code: 0FC63DAF-8483-4772-8E79-3D69D8477DE4 (Linux filesystem) Partition unique GUID: 6278AF5D-BC6B-4213-BBF6-47D333B5CB53 #使用的是该分区的 unique GUID,记下前12位即可 First sector: 2048 (at 1024.0 KiB) Last sector: 6293503 (at 3.0 GiB) Partition size: 6291456 sectors (3.0 GiB) Attribute flags: 0000000000000000 Partition name: 'Linux filesystem' ------------------------------------------------------------------- #输入 wq 保存修改并退出 Command (? for help): wq Final checks complete. About to write GPT data. THIS WILL OVERWRITE EXISTING PARTITIONS!! Do you want to proceed? (Y/N): y OK; writing new GUID partition table (GPT) to /dev/sdb. Warning: The kernel is still using the old partition table. The new table will be used at the next reboot or after you run partprobe(8) or kpartx(8) The operation has completed successfully. 把新创建的分区进行格式化。 # 注意 sdb1 表示 SD 卡对应分区设备名,请用户根据实际情况填入 sudo mkfs.ext4 /dev/sdb1 格式化完成后,使用 "dd" 命令,将制作好的根文件系统烧写到 SD 卡刚新建的 分区中。(根文件系统的制作可以参考 Ubuntu 根文件系统的制作 ) # 用户请根据实际情况填写根文件系统镜像的路径和 SD 卡对应分区设备名 dd if=/rootfs_path/rootfs.img of=/dev/sdb1 挂载 SD 卡根文件系统 首先要修改设备树文件,打开对应的 dts 文件,在根节点下重写 "chosen" 节 点下的 "bootargs" 参数的 "root" 值为写入了根文件系统的 SD 卡分区的 unique GUID: #将 root 的值改为获取到的 unique GUID 值的前12位 chosen { bootargs = "earlycon=uart8250,mmio32,0xff1a0000 swiotlb=1 console=ttyFIQ0 rw root=PARTUUID=6278AF5D-BC6B rootfstype=ext4 rootwait"; }; 修改完成后编译并烧录至开发板中。 SD 卡的根文件系统烧录完成后,插入开发板的 TF 卡槽中,开机即可进入到 SD 的根文件系统中。 注意事项: 在操作前请注意备份 U 盘或者 SD 卡内的重要文件,避免丢失; 操作时请确认自己的 SD 卡对应的设备名; dts 文件中 root 参数的值使用的是 unique GUID,记下前12位即可。用户也可 以根据 gdisk 的帮助信息自行修改这个值。 网络启动 网络启动,是用 TFTP 在服务器下载内核、dtb 文件到目标机的内存中,同时可 以用 NFS 挂载网络根文件系统到目标机上,实现目标机的无盘启动。以下基于 Firefly-RK3399 作出一个示例,提供用户参考。 准备工作: Firefly-RK3399 开发板; 路由器、网线; 安装有 NFS 和 TFTP 的服务器; 一份制作好的根文件系统。 注:示例中使用的是 Ubuntu 系统的 PC 机作为服务器,通过路由器和网线实现 与开发板的连接。用户可以根据自己的实际情况作调整,但如果是 PC 机直连开 发板,请使用交叉网线。请确保服务器和目标机在同一局域网内。 服务器部署 1、在服务器上部署 TFTP 服务: 安装 TFTP 服务: sudo apt-get install tftpd-hpa 创建 "/tftpboot" 目录并赋予权限: mkdir /tftpboot sudo chmod 777 /tftpboot 然后修改 TFTP 服务器的配置文件 "/etc/default/tftpd-hpa",修改内容为: # /etc/default/tftpd-hpa TFTP_USERNAME="tftp" TFTP_DIRECTORY="/tftpboot" #这个根据用户实际的 tftp 目录设置 TFTP_ADDRESS="0.0.0.0:69" TDTP_OPTIONS="-c -s -l" 退出修改后重启 TFTP 服务器: sudo service tftpd-hpa restart 在 "/tftpboot" 目录中创建一个文件,测试 TFTP 服务是否可用: firefly@Desktop:~$ cd /tftpboot/ firefly@Desktop:/tftpboot$ touch test firefly@Desktop:/tftpboot$ cd /tmp firefly@Desktop:/tmp$ tftp 127.0.0.1 tftp> get test tftp> q 查看 "/tmp" 目录中的文件,如果看到了 "test" 文件表示 TFTP 服务器是可用 的。 2、在服务器上部署 NFS 服务: 安装 NFS 服务器: sudo apt-get install nfs-kernel-server 创建共享目录: sudo mkdir /nfs sudo chmod 777 /nfs cd /nfs sudo mkdir rootfs sudo chmod 777 rootfs 然后将制作好的根文件系统复制到 "/nfs/rootfs" 目录中。(根文件系统的制 作可以参考: Ubuntu 根文件系统的制作 ) 接着配置 NFS,在 "/etc/exports" 中添加共享目录位置: /nfs/rootfs *(rw,sync,no_root_squash,no_subtree_check) 共享的目录根据用户实际情况设置,其中的"*"代表的是所有用户可访问。 重启 NFS 服务器: sudo /etc/init.d/nfs-kernel-server restart 本地挂载共享目录,测试 NFS 服务器是否可用: sudo mount -t nfs 127.0.0.1:/nfs/rootfs/ /mnt 在 "/mnt" 目录下查看到了与 "/nfs/rootfs" 一致的内容,则说明 NFS 服务器 部署成功。然后可以解除挂载: sudo umount /mnt 内核的配置 如果要做到挂载网络根文件系统,需要在内核中做相关配置,并在 dts 中修改 相关挂载根文件系统的配置。 首先进行内核配置,在内核目录中执行 "make menuconfig",选择相关配置: [*] Networking support ---> Networking options ---> [*] IP: kernel level autoconfiguration [*] IP: DHCP support [*] IP: BOOTP support [*] IP: RARP support File systems ---> [*] Network File Systems ---> [*] Root file system on NFS 修改 "rk3399-firefly.dts" 配置,在 dts 文件中修改 "chosen" 节点下的 "bootargs" 参数,选择使用 NFS 挂载远程根文件系统,内容如下。 chosen { bootargs = "earlycon=uart8250,mmio32,0xff1a0000 swiotlb=1 console=ttyFIQ0 root=/dev/nfs rootfstype=ext4 rootwait"; }; #主要是修改 root 的值,root=/dev/nfs 表示通过 NFS 挂载网络根文件系统 编译内核: make ARCH=arm64 rk3399-firefly.img -j12 编译完成后,将编译好的内核文件 "boot.img" 和 "rk3399-firefly.dtb" 文件 复制到 "/tftpboot" 目录中: cp boot.img /tftpboot cp arch/arm64/boot/dts/rockchip/rk3399-firefly.dtb /tftpboot 详细说明可以参考内核目录中的 "kernel/Documentation/filesystems/nfs/nfsroot.txt" U-Boot设置 请先确保目标机网线已插入,接入到服务器的局域网内。 目标机启动进入 U-Boot 命令行模式,设置以下参数: => setenv ipaddr 192.168.31.101 #设置目标机 IP 地址 => setenv serverip 192.168.31.106 #设置 serverip 为服务器 IP 地址 #设置从 TFTP 下载内核和 dtb 文件到相应地址,用户请根据自己实际的目标机修改相应地址 => setenv bootcmd tftpboot 0x0027f800 boot.img \; tftpboot 0x08300000 rk3399-firefly.dtb \; bootm 0x0027f800 - 0x08300000 #设置挂载网络根文件系统,IP 参数依次为:目标机 IP:服务器 IP:网关:网络掩码:设备名:off,可以更简单的设置 ip=dhcp,通过 DHXP 自动分配 IP => setenv bootargs root=/dev/nfs rw nfsroot=192.168.31.106:/nfs/rootfs,v3 ip=192.168.31.101:192.168.31.106:192.168.31.1:255.255.255.0::eth0:off #如果不想每次开机都设置一遍,可以保存上面配置的变量 => saveenv Saving Environment to MMC... Writing to MMC(0)... done #启动目标机 => boot ethernet@fe300000 Waiting for PHY auto negotiation to complete. done Speed: 100, full duplex Using ethernet@fe300000 device TFTP from server 192.168.31.106; our IP address is 192.168.31.101 Filename 'boot.img'. Load address: 0x27f800 Loading: ################################################################# ################################################################# ################################################################# ################################################################# ################################################################# ################################################## 475.6 KiB/s done Bytes transferred = 20072448 (1324800 hex) Speed: 100, full duplex Using ethernet@fe300000 device TFTP from server 192.168.31.106; our IP address is 192.168.31.101 Filename 'rk3399-firefly.dtb'. Load address: 0x8300000 Loading: ####### 645.5 KiB/s done Bytes transferred = 97212 (17bbc hex) ## Booting Android Image at 0x0027f800 ... Kernel load addr 0x00280000 size 19377 KiB ## Flattened Device Tree blob at 08300000 Booting using the fdt blob at 0x8300000 XIP Kernel Image ... OK Loading Device Tree to 0000000073edc000, end 0000000073ef6bbb ... OK Adding bank: 0x00200000 - 0x08400000 (size: 0x08200000) Adding bank: 0x0a200000 - 0x80000000 (size: 0x75e00000) Total: 912260.463 ms Starting kernel ... ... 在开机内核日志中可见: [ 12.146297] VFS: Mounted root (nfs filesystem) on device 0:16. 说明已经挂载上了网络根文件系统。 注意事项 确保 TFTP 服务器、NFS 服务器可用; 确保目标机先插入网线后在开机,且和服务器在同一局域网内,如果是直连目标 机和服务器,请使用交叉网线; 内核配置中,Root file system on NFS 依赖于 IP: kernel level autoconfiguration 选项,请先选择 IP: kernel level autoconfiguration, 之后才可以找到 Root file system on NFS 选项; 在 U-Boot 命令行中,请确认 boot.img 烧录地址和 dtb 文件烧录地址。(提 示:boot.img 的文件结构中,开头有2k的头文件,然后才是 kernel。所以在 TFTP 下载内核到目标机时,要下载到对应 kernel 地址减去0x0800的地址上); 在设置挂载远程根文件系统时,nfsroot=192.168.31.106:/nfs/rootfs,v3 中的 v3 代表 NFS 版本信息,请添加上以避免出现挂载不成功的问题。 在线更新内核和 U-Boot 本小节介绍了在线更新的一个简单的流程。将内核、U-Boot 或者其他需要更新 的文件打包成 deb 安装包,然后导入到本地包仓库,实现在开发板上下载并自 动更新。仅供用户参考。 准备 deb 安装包 操作中需要升级内核和 U-Boot,事先已经准备好了修改好的相关文件: "uboot.img" 、"trust.img" 、"boot.img" 。 deb 是 Debian Linux 的软件包格式,打包最关键的是在 "DEBIAN" 目录下创建 一个 "control" 文件。首先创建 deb 工作目录,然后在 deb 目录中创建相应 的目录和文件: mkdir deb cd deb mkdir firefly-firmware #创建 firefly-firmware 目录 cd firefly-firmware mkdir DEBIAN #创建 DEBIAN 目录,这个目录是必须要有的。 mkdir -p usr/share/{kernel,uboot} mv ~/boot.img ~/deb/firefly-firmware/usr/share/kernel #将相应文件放到相应的目录 mv ~/uboot.img ~/deb/firefly-firmware/usr/share/uboot mv ~/trust.img ~/deb/firefly-firmware/usr/share/uboot "DEBIAN" 目录下存放的文件是 deb 包安装的控制文件以及相应的脚本文件。 在 "DEBIAN" 目录下创建控制文件 "control" 和脚本文件 "postinst"。 "control" 文件内容如下,用于记录软件标识,版本号,平台,依赖信息等数据 。 Package: firefly-firmware #文件目录名 Version: 1.0 #版本号 Architecture: arm64 #架构 Maintainer: neg #维护人员,自定义即可 Installed-Size: 1 Section: test Priority: optional Descriptionon: This is a deb test "postinst" 文件内容如下,就是将需要更新的内核和 U-Boot 文件用 "dd" 命令 写进对应分区的脚本: echo "-----------uboot updating------------" dd if=/usr/share/uboot/uboot.img of=/dev/disk/by-partlabel/uboot echo "-----------trust updating------------" dd if=/usr/share/uboot/trust.img of=/dev/disk/by-partlabel/trust echo "-----------kernel updating------------" dd if=/usr/share/kernel/boot.img of=/dev/disk/by-partlabel/boot 说明:"postinst" 脚本,是在解包数据后运行的脚本,对应的还有: preinst:在解包数据前运行的脚本; prerm:卸载时,在删除文件之前运行的脚本; postrm:在删除文件之后运行的脚本; 其他相关知识点用户可以自行了解。这里只用到了 "preinst" 脚本。 以下是创建好的目录树: deb └── firefly-firmware ├── DEBIAN │   ├── control │   └── postinst └── usr └── share ├── kernel │   └── boot.img └── uboot ├── trust.img └── uboot.img 进入 deb 目录,用 "dpkg" 命令生成 deb 包: dpkg -b firefly-firmware firefly-firmware_1.0_arm64.deb 生成 deb 包时,一般按照这种规范进行命名:"package_version- reversion_arch.deb" 。 创建本地仓库 首先安装需要的包: sudo apt-get install reprepro gnupg 然后用 GnuPG 工具生成一个 GPG 密匙,执行命令后请根据提示操作: gpg --gen-key 执行 "sudo gpg --list-keys" 可以查看刚刚生成的密匙信息: sudo gpg --list-keys gpg: WARNING: unsafe ownership on homedir '/home/firefly/.gnupg' gpg: 正在检查信任度数据库 gpg: marginals needed: 3 completes needed: 1 trust model: pgp gpg: 深度:0 有效性: 4 已签名: 0 信任度:0-,0q,0n,0m,0f,4u gpg: 下次信任度数据库检查将于 2021-05-29 进行 /home/firefly/.gnupg/pubring.kbx -------------------------------- pub rsa3072 2019-05-31 [SC] [有效至:2021-05-30] BCB65788541D632C057E696B8CBC526C05417B76 uid [ 绝对 ] firefly sub rsa3072 2019-05-31 [E] [有效至:2021-05-30] 接下来创建包仓库,首先创建目录: cd /var/www mkdir apt #包仓库目录 mkdir -p ./apt/incoming mkdir -p ./apt/conf mkdir -p ./apt/key 把前面生成的密匙导出到仓库文件夹,请用户对应好自己创建的用户名和邮箱地 址。 gpg --armor --export firefly firefly@t-chip.com > /var/www/apt/key/deb.gpg.key 在 "conf" 目录下创建 "distributions" 文件,其内容如下: Origin: Neg #你的名字 Label: Mian #库的名字 Suite: stable #(stable 或 unstable) Codename: bionic #发布代码名,可自定义 Version: 1.0 #发布版本 Architectures: arm64 #架构 Components: main #组件名,如main,universe等 Description: Deb source test #相关描述 SignWith: BCB65788541D632C057E696B8CBC526C05417B76 #上面步骤中生成的 GPG 密匙 建立仓库树: reprepro --ask-passphrase -Vb /var/www/apt export 将第2步做好的 deb 包加入到仓库中: reprepro --ask-passphrase -Vb /var/www/apt includedeb bionic ~/deb/firefly-firmware_1.0_arm64.deb 可以查看库中添加的文件: root@Desktop:~# reprepro -b /var/www/apt/ list bionic bionic|main|arm64: firefly-firmware 1.0 你的包已经加入了仓库,如果要移除它的话采用如下命令: reprepro --ask-passphrase -Vb /var/www/apt remove bionic firefly-firmware 安装 nginx 服务器: sudo apt-get install nginx 修改nginx的配置文件 "/etc/nginx/sites-available/default" 为: server { listen 80 default_server; listen [::]:80 default_server; root /var/www/apt; //本地包仓库的路径 access_log /var/log/nginx/repo.access.log; error_log /var/log/nginx/repo.error.log; location ~ /(db|conf) { deny all; return 404; } } 重启 nginx 服务器: sudo service nginx restart 客户端更新安装 在客户端开发板中,首先要添加本地包仓库的源,在目录 "/etc/apt/sources.list.d" 下添加一个新的配置文件 "bionic.list",内容如 下: deb http://192.168.31.106 bionic main IP 地址是服务器地址, "bionic" 是仓库发布代码名, "main" 是组件名。 从服务器中获取并添加 GPG 密匙: wget -O - http://192.168.31.106/key/deb.gpg.key | apt-key add - 更新后即可安装自定义软件源里的 "firefly-firmware_1.0_arm64" 包: root@firefly:/home/firefly# apt-get update Hit:1 http://192.168.31.106 bionic InRelease Hit:2 http://wiki.t-firefly.com/firefly-rk3399-repo bionic InRelease Hit:3 http://ports.ubuntu.com/ubuntu-ports bionic InRelease Hit:4 http://archive.canonical.com/ubuntu bionic InRelease Hit:5 http://ports.ubuntu.com/ubuntu-ports bionic-updates InRelease Hit:6 http://ports.ubuntu.com/ubuntu-ports bionic-backports InRelease Hit:7 http://ports.ubuntu.com/ubuntu-ports bionic-security InRelease Reading package lists... Done root@firefly:/home/firefly# apt-get install firefly-firmware Reading package lists... Done Building dependency tree Reading state information... Done The following package was automatically installed and is no longer required: libllvm6.0 Use 'apt autoremove' to remove it. The following NEW packages will be installed: firefly-firmware 0 upgraded, 1 newly installed, 0 to remove and 3 not upgraded. Need to get 0 B/6982 kB of archives. After this operation, 1024 B of additional disk space will be used. Selecting previously unselected package firefly-firmware. (Reading database ... 117088 files and directories currently installed.) Preparing to unpack .../firefly-firmware_1.0_arm64.deb ... Unpacking firefly-firmware (1.0) ... Setting up firefly-firmware (1.0) ... -----------uboot updating------------ 8192+0 records in 8192+0 records out 4194304 bytes (4.2 MB, 4.0 MiB) copied, 0.437281 s, 9.6 MB/s -----------trust updating------------ 8192+0 records in 8192+0 records out 4194304 bytes (4.2 MB, 4.0 MiB) copied, 0.565762 s, 7.4 MB/s -----------kernel updating------------ 39752+0 records in 39752+0 records out 20353024 bytes (20 MB, 19 MiB) copied, 0.1702 s, 120 MB/s 可以看到安装过程中执行了 deb 包中的 "poinst" 脚本。安装完成后,重启开 发板,更新完成。 在 "/usr/share" 目录中还可以看到 "kernel" 和 "uboot" 目录,分别存放了 "boot.img","uboot.img" 和 "trust.img" 。 注意事项 在制作 deb 包中,与 DEBIAN 同级的目录视为根目录,即放在与 DEBIAN 同目 录下的文件,在客户端安装 deb 包后,可以在客户端的根目录下找到; deb 包中的文件和脚本,用户要根据自己的实际情况做调整; 每次在仓库中修改了配置文件后,都要重新导入仓库目录树 ; nginx 服务器配置中,root 参数配置的是仓库的地址,请用户根据自己的实际 情况修改; 客户端添加的新下载源的文件时,注意核对正确的服务器地址,包仓库代码名和 组件名。注意客户端要连通服务器; 客户端要用 apt-key add 命令添加 GPG 密匙之后,才可以获取本地的仓库的信 息。 用Docker搭建分布式编译环境 "distcc" 是个通过网络上的若干台机器用来分布式编译 C、C++、Objective-C 或 Objective-C++ 代码的程序。"distcc" 不要求所有的机器共享文件系统,有 同步的时钟,或者安装有同样的库和头文件,只要作为服务器的机器有合适的编 译工具即可进行编译。本例在两块 Firefly-RK3399 开发板 (arm64) 和一台 PC 机 (x86_64) 上利用 Docker 技术来布署 distcc 分布式编译服务,进而实现在 其中一块 Firefly-RK3399 开发板上利用 distcc 的分布式编译特性来加速内核 的编译过程。 准备工作 两块 Firefly-RK3399 开发板; 路由器、网线; PC 机。 将两块开发板和 PC 机都连接到同一个局域网内。连接好后对应的 IP 地址别是 : PC 机:192.168.1.100 开发板 A:192.168.1.101 开发板 B:192.168.1.102 PC 机部署 Docker 使用脚本安装 Docker : wget -qO- https://get.docker.com/ | sh 为了可以使当前普通用户可以执行 docker 相关命令,需要将当前用户添加到 dokcer 用户组中: sudo groupadd docker #添加 docker 用户组 sudo gpasswd -a $USER docker #将当前用户添加到 docker 用户组中 newgrp docker #更新 docker 用户组 启动 Docker 后台服务: sudo service docker start 新建一个 "Dockerfile_distcc.x86_64" 文件。其文件内容如下: FROM ubuntu:bionic MAINTAINER Firefly ARG DEBIAN_FRONTEND=noninteractive RUN apt-get update \ && apt-get install -y --no-install-recommends --no-install-suggests\ gcc-aarch64-linux-gnu distcc\ && apt-get clean \ && rm -rf /var/lib/apt/lists RUN ln -s aarch64-linux-gnu-gcc /usr/bin/gcc &&\ ln -s aarch64-linux-gnu-gcc /usr/bin/cc EXPOSE 3632 ENTRYPOINT ["/usr/bin/distccd"] CMD ["--no-detach" , "--log-stderr" , "--log-level", "debug", "--allow" , "0.0.0.0/0"] 生成 Docker 镜像: docker build -t distcc_server:x86_64 -f Dockerfile_distcc.x86_64 . 可以用命令 "docker images" 查看生成的 Docker 镜像: docker images REPOSITORY TAG IMAGE ID CREATED SIZE distcc_server x86_64 138c0b7e3801 9 minutes ago 66.1MB 用新建的镜像启动 Docker 容器,在主机网络的 3632 TCP 端口提供 distcc 服 务: docker run -d -p 3632:3632 distcc_server:x86_64 用命令 "docker ps" 可以查看正在运行中的容器: docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES fa468d068185 distcc_server:x86_64 "/usr/bin/distccd --…" 9 minutes ago Up 9 minutes 0.0.0.0:3632->3632/tcp epic_chatterjee 开发板部署 Docker Firefly-RK3399 内核默认是不支持 Docker 的,需要进行相关内核的配置。可 以在 Firefly Github 中查看修改后的 "/kernel/arch/arm64/configs/firefly_linux_defconfig" 文件。 修改好后编译内核并更新到所有用到的开发板。 内核更新完成后,在开发板上使用脚本安装 Docker : apt-get update wget -qO- https://get.docker.com/ | sh 将当前用户添加到 dokcer 用户组中: sudo groupadd docker sudo gpasswd -a $USER docker newgrp docker #更新 docker 启动 Docker 后台服务: service docker start 新建 "Dockerfile_distcc.arm64" 文件,内容如下: FROM ubuntu:bionic MAINTAINER Firefly ARG DEBIAN_FRONTEND=noninteractive RUN apt-get update \ && apt-get install -y --no-install-recommends --no-install-suggests \ gcc distcc\ && apt-get clean \ && rm -rf /var/lib/apt/lists EXPOSE 3632 ENTRYPOINT ["/usr/bin/distccd"] CMD ["--no-detach" , "--log-stderr" , "--log-level", "debug", "--allow" , "0.0.0.0/0"] 生成 Docker 镜像: docker build -t distcc_server:arm64 -f Dockerfile_distcc.arm64 . 用新建的镜像启动 Docker 容器,在主机网络的 3632 TCP 端口提供 distcc 服 务: docker run -d -p 3632:3632 distcc_server:arm64 用 "docker save" 命令导出新建的镜像: docker save -o distcc_server.tar distcc_server:arm64 将生成的 "distcc_server.tar" 文件复制到另一块开发板上,然后导入镜像: docker load -i distcc_server.tar 导入镜像后,在这个开发板上也有了 "distcc_server:arm64" 镜像,然后可以 运行一个容器: docker run -d -p 3632:3632 distcc_server:arm64 提示: 用户如果注册了 Docker Hub 帐号,可以用 "docker push" 推送镜像到远程仓 库,在另一开发板上用 "docker pull" 拉取远程仓库的镜像。用户可以自行了 解相关操作。 客户端用 distcc 编译内核 到这里,三个机器都部署了分布式编译环境。可以选择任意一台机器作为客户端 ,剩下两个机器作为服务器。这里选择其中一个开发板作为客户端。 在客户端中创建 "Dockerfile_compile.arm64" 文件,生成一个编译内核的 Docker 镜像,文件内容如下: FROM ubuntu:bionic MAINTAINER Firefly ARG DEBIAN_FRONTEND=noninteractive RUN apt-get update \ && apt-get install -y --no-install-recommends --no-install-suggests \ bc make python sed libssl-dev binutils build-essential distcc\ liblz4-tool gcc \ && apt-get clean \ && rm -rf /var/lib/apt/lists 然后生成镜像: docker build -t compile:arm64 -f Dockerfile_compile.arm64 . 在启动容器前先将内核文件拉到客户端中。然后创建 "/etc/distcc/hosts" 文 件,其内容是列举出所有提供 distcc 服务的机器的 IP 地址。内容如下: # As described in the distcc manpage, this file can be used for a global # list of available distcc hosts. # # The list from this file will only be used, if neither the # environment variable DISTCC_HOSTS, nor the file $HOME/.distcc/hosts # contains a valid list of hosts. # # Add a list of hostnames in one line, seperated by spaces, here. 192.168.1.100 192.168.1.101 193.168.1.102 为了测试结果更准确,先清除客户端开发板上的缓存: echo 3 > /proc/sys/vm/drop_caches 用 "compile:arm64" 镜像启动一个 Docker 容器,同时将主机当前内核目录挂 载到容器中的 "/mnt" 目录,将主机的 "/etc/distcc/hosts" 文件挂载到容器 的 "/etc/distcc/hosts" 中,并用参数 "-it" 以交互模式启动容器: docker run -it --rm -v $(pwd):/mnt -v /etc/distcc/hosts:/etc/distcc/hosts compile:arm64 /bin/bash root@f4415264351b:/# ls bin boot dev etc home lib media mnt opt proc root run sbin srv sys tmp usr var root@f4415264351b:/# cd mnt/ 进入到容器中的 "/mnt" 目录,然后开始用 distcc 编译内核,加入 "time" 命 令可以查看编译命令执行耗时,"CC" 参数指明用 "distcc" 进行编译: time make ARCH=arm64 rk3399-firefly.img -j32 CC="distcc" 注意: 如果用 PC 机作为客户端,则需要用以下命令进行编译: time make ARCH=arm64 CROSS_COMPILE=aarch64-linux-gnu- rk3399-firefly.img -j32 CC="distcc aarch64-linux-gnu-gcc" 编译过程中,可以在用于编译的容器内新的窗口中用命令 "distccmon-text 1" 查看编译情况: distccmon-text 1 ... 15713 Compile perf_regs.c 192.168.1.101[0] 15327 Compile fork.c 192.168.1.101[2] 15119 Compile dma-mapping.c 192.168.1.101[3] 15552 Compile signal32.c 192.168.1.100[0] 15644 Compile open.c 192.168.1.100[2] 15112 Compile traps.c 192.168.1.100[3] 15670 Compile arm64ksyms.c 192.168.1.102[0] 15629 Compile mempool.c 192.168.1.102[2] 15606 Compile filemap.c 192.168.1.102[3] 15771 Preprocess localhost[0] 15573 Preprocess localhost[1] 15485 Preprocess localhost[2] ... 最后编译命令完成后可以看到编译所用时间: real 15m44.809s user 16m0.029s sys 6m22.317s 下边是单独使用一块开发板进行内核编译所耗费的时间: real 23m33.002s user 113m2.615s sys 9m29.348s 对比可见,用采用 distcc 实现的分布式编译可以有效提高编译速度。 注意: 平台不同,所需的编译器也不同。如在 x86_64 平台上,需要安装对应的交叉编 译工具 gcc-aarch64-linux-gnu。在 arm64 平台则只需要安装 gcc 编译工具即 可。用户需要根据实际情况在对应的 Dockerfile 文件里指定安装正确的工具。 如果用 PC 机做客户端,则在用于编译的容器中编译内核的时候,要用参数 CC="distcc aarch64-linux-gnu-gcc" 指明用 aarch64-linux-gnu-gcc 交叉编 译工具编译。