Mpp是Rockchip為RK3399提供的一套視頻編解碼的api, 并且基于mpp,Rockchip提供了一套gstreamer的編解碼插件。用戶可以根據(jù)自己的需求,基于gstreamer來做視頻編解碼的應(yīng)用,或者直接調(diào)用mpp,來實現(xiàn)硬件的編解碼加速。
Firefly 發(fā)布的Ubuntu 系統(tǒng), 都已經(jīng)提供了完整的gstremaer 和 mpp支持,并且提供了相應(yīng)的demo,供用戶開發(fā)參考。
-
Ubuntu 16.04 下,gstreamer 1.12 已經(jīng)安裝在/opt/目錄下。
-
Ubuntu 18.04下, gstreamer 1.12 已經(jīng)安裝到系統(tǒng)中。
/usr/local/bin/h264dec.sh 測試硬件H264解碼。
/usr/local/bin/h264enc.sh 測試硬件H264編碼。
用戶可以參照這兩個腳本,配置自己的gstreamer應(yīng)用。
-
Ubunut 系統(tǒng)下, mpp 相關(guān)dev包都已經(jīng)安裝到系統(tǒng)中。
/opt/mpp/下分別是mpp 編解碼的相關(guān)demo 和 源文件。
RK3399 支持 OpenGL ES1.1/2.0/3.0/3.1。
Firefly 發(fā)布的Ubuntu 系統(tǒng), 都已經(jīng)提供了完整的OpenGL-ES支持。運行glmark2-es2
可以測試openGL-ES支持。 如果要避免屏幕刷新率對測試結(jié)果的影響,可以在串口終端上使用以下命令測試。
# systemctl stop lightdm # export DISPLAY=:0 # Xorg & # glmark2-es2 –off-screen
在Chromium瀏覽器中, 在地址欄輸入:chrome://gpu
可以查看chromium下硬件加速的支持。
Note:
-
EGL 是用arm 平臺上OpenGL針對x window system的擴展,功能等效于x86下的glx庫。
-
由于Xorg使用的Driver modesettings 默認(rèn)會加載libglx.so(禁用glx會導(dǎo)致某些通過檢測glx環(huán)境的應(yīng)用啟動失敗), libglx.so會搜索系統(tǒng)中的dri實現(xiàn)庫。但是rk3399 Xorg 2D加速是直接基于DRM實現(xiàn), 并未實現(xiàn)dri庫,所以啟動過程中,libglx.so會報告如下的錯誤 。
這個對系統(tǒng)運行沒有任何影響,不需要處理。
-
基于同樣的道理,某些應(yīng)用啟動過程中,也會報告如下錯誤,不用處理,對應(yīng)用的運行不會造成影響。
libGL error: unable to load driver: rockchip_dri.so libGL error: driver pointer missing libGL error: failed to load driver: rockchip
-
Firefly之前發(fā)布的某些版本的Ubuntu軟件,默認(rèn)關(guān)閉了加載libglx.so,在某些情況下,運行某些應(yīng)用程序會出現(xiàn)下述錯誤:
GdkGLExt-WARNING **: Window system doesn't support OpenGL.
修正的方法如下:
刪除
/etc/X11/xorg.conf.d/20-modesetting.conf
中一下三行配置。Section "Module" Disable "glx" EndSection
Firefly發(fā)布的Ubuntu系統(tǒng),已經(jīng)添加了opencl1.2支持,可以運行系統(tǒng)內(nèi)置的clinfo
獲取平臺opencl相關(guān)參數(shù)。
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) …
RK3399 支持神經(jīng)網(wǎng)絡(luò)的GPU加速方案LinuxNN, Firefly發(fā)布的Ubuntu系統(tǒng),已經(jīng)添加了LinuxNN的支持。
在opt/tensorflowbin/下,運行test.sh, 即可測試MobileNet 模型圖像分類器的 Demo和MobileNet-SSD 模型的目標(biāo)檢測 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 發(fā)布的 Ubuntu 系統(tǒng),如果需要默認(rèn)對系統(tǒng)的顯示方向做旋轉(zhuǎn),可以在
/etc/default/xrandr 中修改對應(yīng)的顯示設(shè)備的方向即可。
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
對于配有觸摸屏的平臺,如果需要對觸摸屏的方向做旋轉(zhuǎn),可以在/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 系統(tǒng)中自帶屏幕鍵盤,可以在菜單欄中點擊打開:
開發(fā)板一般都有2個或以上的音頻設(shè)備。常見的是耳機和 HDMI 這兩個設(shè)備的音頻輸出。下面是音頻設(shè)置的例子,供用戶參考。
系統(tǒng)自帶的音頻文件在 /usr/share/sound/alsa/
目錄中,播放前請先查看聲卡設(shè)備:
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 的設(shè)備0對應(yīng)的是耳機口,card1 的設(shè)備0對應(yīng)的是 HDMI。一般播放音頻命令是 aplay -Dhw:0,0 Fornt_Center.wav
,但系統(tǒng)自帶的音頻文件是單聲道的,所以為了防止播放失敗,可以按照下面的命令進行播放:
#選擇耳機口輸出音頻文件 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
在圖形界面中,播放準(zhǔn)備好的音頻文件,然后點擊聲音圖標(biāo),打開 Sound Setting
,選擇到 Configuration
??梢钥吹絻蓚€聲卡設(shè)備,例如設(shè)置為 HDMI 輸出音頻,則把 HDMI 的聲卡設(shè)備選擇為 Output
,另一個聲卡設(shè)置為 Off
。(如果 HDMI 無聲或者聲音小,可以試試按 HDMI 屏上的物理按鍵調(diào)高音量)
在系統(tǒng)中,可以根據(jù)用戶需求自行設(shè)置應(yīng)用程序開機自動啟動。
在系統(tǒng)菜單欄中依次選擇:Preferences
-> Default applications for LXSession
,打開設(shè)置界面。
在設(shè)置界面的 Launching applocations
一欄中可以設(shè)置應(yīng)用默認(rèn)的打開方式:
在 Autostart
一欄中就可以選擇用戶需要開機啟動的應(yīng)用。例如藍(lán)牙默認(rèn)是開機不啟動的,如果需要讓它開機自動啟動,就把 Blueman Applet
的勾打上,就可以實現(xiàn)開機自動啟動藍(lán)牙:
ADB,全稱 Android Debug Bridge,是 Android 的命令行調(diào)試工具,可以完成多種功能,如跟蹤系統(tǒng)日志,上傳下載文件,安裝應(yīng)用等。以 Firefly-RK3399 為例,為了在 Ubuntu 系統(tǒng)也能使用 ADB 工具進行調(diào)試,我們移植了 ADB 服務(wù)。但由于并非 Android 系統(tǒng),很多 ADB 命令類似 adb logcat
、adb install
等不能使用,僅作為普通的調(diào)試輔助工具,可以進行 shell 交互、上傳下載文件等操作。同樣的網(wǎng)絡(luò)遠(yuǎn)程 ADB 調(diào)試也不能使用。
在 Ubuntu 系統(tǒng)中,執(zhí)行:
sudo apt-get install android-tools-adb
在內(nèi)核目錄下,打開內(nèi)核配置選項菜單:
make firefly_linux_defconfig make menuconfig
進入內(nèi)核配置菜單后依次選擇:Device Drivers
-> USB Support
-> USB Gadget Support
。
在 USB Gadget Driver
選項中選擇到 USB functions configurable through configfs
。
同時再選擇上 Function filesystem (FunctionFS)
。
<*> USB Gadget Drivers (USB functions configurable through configfs) ---> USB functions configurable through configfs [*] Function filesystem (FunctionFS)
接著在 kernel
目錄下編譯內(nèi)核:
make rk3399-firefly.img -j12
編譯完成后將內(nèi)核燒錄到開發(fā)板中,燒錄過程請參考維基教程:升級固件中的分區(qū)鏡像燒寫部分。
安裝了 ADB 后,用 Micro USB OTG 線連接開發(fā)板和 PC 機。然后通過命令 adb devices
查看是否有設(shè)備連接:
firefly@Desktop:~$ adb devices List of devices attached 0123456789ABCDEF device
從返回的信息可以看到已經(jīng)查找到了設(shè)備,表明設(shè)備已經(jīng)成功連接。
設(shè)備連接成功后,輸入命令 adb shell
即可進入命令行模式:
firefly@Desktop:~$ adb shell #
此狀態(tài)下是看不到命令行當(dāng)前的路徑,同時 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
查看命令行幫助信息,注意并不是所有命令都可以使用,幫助信息只做參考。
開發(fā)板上的串口分兩種,一種是普通串口另外一種是通過轉(zhuǎn)換芯片把SPI轉(zhuǎn)換而成的串口。轉(zhuǎn)換而成的串口和普通串口功能完全一致但是需要注意它們的設(shè)備文件名是不一樣的。下面是兩者的區(qū)別:
//普通串口 root@firefly:~# ls /dev/ttyS* ttyS0 ttyS1 ttyS2 ttyS3 //SPI轉(zhuǎn)串口 root@firefly:~# ls /dev/ttysWK* ttysWK0 ttysWK1 ttysWK2 ttysWK3
以 ttyS4
為例,查看串口波特率命令:
root@firefly:~# stty -F /dev/ttyS4 speed 9600 baud; line = 0; -brkint -imaxbel
設(shè)置波特率命令:
//設(shè)置波特率為115200 root@firefly:~# stty -F /dev/ttyS4 ospeed 115200 ispeed 115200 cs8 //查看確認(rèn)是否已經(jīng)修改 root@firefly:~# stty -F /dev/ttyS4 speed 115200 baud; line = 0; -brkint -imaxbel root@firefly:~#
在做環(huán)回收發(fā)測試的時候回顯功能會影響到我們的測試結(jié)果,所以在環(huán)回測試或者其他特殊情況下需要關(guān)閉回顯。以下是關(guān)閉回顯的命令:
//關(guān)閉回顯示 root@firefly:~# stty -F /dev/ttyS4 -echo -echoe -echok //查看所有功能的配置,檢查是否已經(jīng)關(guān)閉?!?”號代表該功能已經(jīng)關(guān)閉 root@firefly:~# stty -F /dev/ttyS4 -a | grep echo isig icanon iexten -echo -echoe -echok -echonl -noflsh -xcase -tostop -echoprt echoctl echoke -flusho -extproc
在實際應(yīng)用中實際發(fā)送數(shù)據(jù)和我們希望發(fā)送的數(shù)據(jù)可能存在差別,例如多了回車或者其他特殊字符。利用 stty
可以把發(fā)送接收設(shè)置成 raw
模式確保發(fā)送接收的是原始數(shù)據(jù),以下是設(shè)置命令:
root@firefly:~# stty -F /dev/ttyS4 raw
串口有 2 種工作模式分別是中斷模式和 DMA 模式。
中斷模式
內(nèi)核默認(rèn)把串口配置成中斷模式所以不需要做任何修改。中斷模式下傳輸速率比較快但是在傳大量數(shù)據(jù)的時候很容易丟包或者出錯,所以當(dāng)數(shù)據(jù)量比較大的時候請不要使用中斷模式。
DMA 模式
DMA 模式主要是傳輸大量數(shù)據(jù)的時候使用。內(nèi)核會為串口提供一個緩存空間接收數(shù)據(jù)來盡量降低串口傳輸?shù)膩G包率。
注意: 緩存空間限默認(rèn)大小為 8K
如果一次傳輸超過緩存大小就會丟包,所以使用 DMA 模式的話需要發(fā)送端需要分包發(fā)送。
設(shè)備樹文件配置
&uart4 { status = "okay"; + dmas = <&dmac_peri 8>, <&dmac_peri 9>; + dma-names = "tx", "rx"; };
DMA 模式并不能提高傳輸速率,相反因為需要緩存,傳輸速率會有所下降,所以如果不是需要傳輸大量數(shù)據(jù)的話不要使用 DMA 模式。
不管是中斷模式還有 DMA 模式都無法保證數(shù)據(jù)傳輸萬無一失,因為長時間傳輸大量數(shù)據(jù)的時候 DDR、CPU 變頻或者占用率過高都可能導(dǎo)致上層處理數(shù)據(jù)不及時導(dǎo)致丟包,這個時候就需要是用流控了。流控分兩種,一種是軟件流控另一種為硬件流控,下面只介紹硬件流控的使用。
硬件支持
硬件流控需要有硬件支持,開發(fā)板的串口 CTX
和 RTX
腳需要和設(shè)備相連。
注意: 開發(fā)板不是所有串口都支持硬件流控,請先從原理圖上確認(rèn)硬件是否支持
設(shè)備樹文件配置
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 = <GIC_SPI 101 IRQ_TYPE_LEVEL_HIGH 0>; reg-shift = <2>; reg-io-width = <4>; pinctrl-names = "default"; + pinctrl-0 = <&uart3_xfer &uart3_cts &uart3_rts>; status = "disabled"; };
應(yīng)用層設(shè)置
上層也需要打開流控的設(shè)置,這里介紹 stty 如何打開流控如果你使用的是其他應(yīng)用程序請從應(yīng)用中打開流控。
//打開流控 root@firefly:~# stty -F /dev/ttyS4 crtscts //檢測流控是否打開,“-”號代表該功能已經(jīng)關(guān)閉 root@firefly:~# stty -F /dev/ttyS4 -a | grep crtscts -parenb -parodd -cmspar cs8 hupcl -cstopb cread clocal crtscts
帶有 MIPI CSI
接口的 RK3399 板子都添加了雙 MIPI 攝像頭 OV13850
的支持,應(yīng)用中也添加了攝像頭的例子。下面介紹一下相關(guān)配置。
以 rk3399-firefly-aiojd4.dts
為例,這些都是攝像頭需要打開的配置。由于內(nèi)核已經(jīng)添加了雙 MIPI 攝像頭的支持,所以這里配置了兩個攝像頭,如果只使用一個攝像只需要打開其中一個攝像頭就可以了。
//電源管理 &vcc_mipi { status = "okay"; }; &dvdd_1v2 { status = "okay"; }; //MIPI 攝像頭1 &ov13850 { status = "okay"; }; &rkisp1_0 { status = "okay"; }; &mipi_dphy_rx0 { status = "okay"; }; &isp0_mmu { status = "okay"; }; //MIPI 攝像頭2 &ov13850_1 { status = "okay"; }; &rkisp1_1 { status = "okay"; }; &mipi_dphy_tx1rx1 { status = "okay"; }; &isp1_mmu { status = "okay"; };
注意: 如果使用的是 core-3399
核心板加上 DIY 底板,可能需要修改相應(yīng)的 GPIO 屬性。
在運行腳本之前先確認(rèn)一下 OV13850
設(shè)備是否注冊成功,下面是成功的內(nèi)核日志:
root@firefly:~# dmesg | grep ov13850 //MIPI 攝像頭1 [ 1.276762] ov13850 1-0036: GPIO lookup for consumer reset [ 1.276771] ov13850 1-0036: using device tree for GPIO lookup [ 1.276803] of_get_named_gpiod_flags: parsed 'reset-gpios' property of node '/i2c@ff110000/ov13850@36[0]' - status (0) [ 1.276855] ov13850 1-0036: Looking up avdd-supply from device tree [ 1.277034] ov13850 1-0036: Looking up dovdd-supply from device tree [ 1.277170] ov13850 1-0036: Looking up dvdd-supply from device tree [ 1.277535] ov13850 1-0036: GPIO lookup for consumer pwdn [ 1.277544] ov13850 1-0036: using device tree for GPIO lookup [ 1.277575] of_get_named_gpiod_flags: parsed 'pwdn-gpios' property of node '/i2c@ff110000/ov13850@36[0]' - status (0) [ 1.281862] ov13850 1-0036: Detected OV00d850 sensor, REVISION 0xb2 //MIPI 攝像頭2 [ 1.284442] ov13850 1-0046: GPIO lookup for consumer pwdn [ 1.284461] ov13850 1-0046: using device tree for GPIO lookup [ 1.284523] of_get_named_gpiod_flags: parsed 'pwdn-gpios' property of node '/i2c@ff110000/ov13850@46[0]' - status (0) [ 1.288235] ov13850 1-0046: Detected OV00d850 sensor, REVISION 0xb2
在 /dev
下應(yīng)該生成相關(guān)設(shè)備文件:
root@firefly:~# ls /dev/video //MIPI 攝像頭1 video0 video1 video2 video3 //MIPI 攝像頭2 video4 video5 video6 video7
注意: 同理,如果只使用一個攝像頭,只需要注冊一個攝像頭設(shè)備就可以了。
在官方提供的 Ubuntu 系統(tǒng)中已經(jīng)添加了攝像頭測試腳本 /usr/local/bin/dual-camera-rkisp.sh
,其內(nèi)容為:
#!/bin/bash export DISPLAY=:0 export XAUTHORITY=/home/firefly/.Xauthority export PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin export LD_LIBRARY_PATH=/usr/lib/gstreamer-1.0 WIDTH=640 HEIGHT=480 SINK=gtksink /etc/init.d/S50set_pipeline start # camera closed to the border of the board gst-launch-1.0 rkisp device=/dev/video0 io-mode=1 analyzer=1 enable-3a=1 path-iqf=/etc/cam_iq.xml ! video/x-raw,format=NV12,width=${WIDTH},hei ght=${HEIGHT}, framerate=30/1 ! videoconvert ! $SINK & # the other camera gst-launch-1.0 rkisp device=/dev/video4 io-mode=1 analyzer=1 enable-3a=1 path-iqf=/etc/cam_iq.xml ! video/x-raw,format=NV12,width=${WIDTH},hei ght=${HEIGHT}, framerate=30/1 ! videoconvert ! $SINK & wait
運行腳本即可,結(jié)果如圖所示:
USB 以太網(wǎng),主要實現(xiàn)的是將開發(fā)板的 OTG 接口做外設(shè)模式,模擬成一個網(wǎng)絡(luò)接口,然后主機通過 USB 連接開發(fā)板并通過開發(fā)板訪問互聯(lián)網(wǎng)。以下是基于 Firefly-RK3399 開發(fā)板進行的具體操作。
操作環(huán)境:
-
Ubuntu 系統(tǒng)的 PC 機
-
Firefly-RK3399 開發(fā)板
在內(nèi)核目錄下,打開內(nèi)核配置選項菜單:
make firefly_linux_defconfig make menuconfig
進入內(nèi)核配置菜單后依次選擇:Device Drivers
-> USB Support
-> USB Gadget Support
。
將 USB Gadget Driver
設(shè)置成編譯成模塊,然后可以在下邊找到 Ethernet Gadget (with CDC Ethernet support)
選項,同樣選擇編譯成模塊。同時要選擇上 RNDIS support
。
<M> USB Gadget Drivers <M> USB functions configurable through configfs <M> Ethernet Gadget (with CDC Ethernet support) [*] RNDIS support (NEW)
接著在 kernel
目錄下編譯內(nèi)核:
make rk3399-firefly.img -j12
編譯完成后將內(nèi)核燒錄到開發(fā)板中,燒錄過程請參考維基教程:升級固件中的分區(qū)鏡像燒寫部分。然后將內(nèi)核目錄下生成的下列模塊復(fù)制到開發(fā)板中:
-
drivers/usb/gadget/function/u_ether.ko
-
drivers/usb/gadget/function/usb_f_ecm_subset.ko
-
drivers/usb/gadget/function/usb_f_ecm.ko
-
drivers/usb/gadget/function/usb_f_rndis.ko
-
drivers/usb/gadget/function/usb_f_eem.ko
-
drivers/usb/gadget/legacy/g_ether.ko
-
drivers/usb/gadget/libcomposite.ko
然后在開發(fā)板上,依次加載上述模塊:
insmod libcomposite.ko insmod u_ether.ko insmod usb_f_ecm_subset.ko insmod usb_f_rndis.ko insmod usb_f_ecm.ko insmod usb_f_eem.ko insmod g_ether.ko
注意: 要先加載 libcomposite.ko
和 u_ether.ko
,后面的模塊才可以加載進去。
用數(shù)據(jù)線連接 PC 機和開發(fā)板的 OTG 接口,在 PC 機中執(zhí)行 lsusb
命令可以查看到 USB 以太網(wǎng)設(shè)備,即說明連接成功。
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 網(wǎng)卡設(shè)備。
開發(fā)板插入網(wǎng)線,使開發(fā)板可以連接外網(wǎng)。
-
在開發(fā)板中 IP 的設(shè)置:
輸入執(zhí)行 ifconfig -a
命令,可以查看到以下信息:
root@firefly:~# ifconfig -a # eth0 是有線網(wǎng)卡 eth0: flags=4163<UP,BROADCAST,RUNNING,MULTICAST> 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<link> 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 網(wǎng)卡 usb0: flags=4098<BROADCAST,MULTICAST> 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 網(wǎng)卡自定義一個適當(dāng)?shù)?IP:
注意要設(shè)置 usb0 的 IP 和有線網(wǎng)卡 eth0 的 IP 不在同一網(wǎng)段?。?/p>
ifconfig usb0 192.168.1.101
-
在 PC 機中 IP 的設(shè)置:
# 先查看 USB 虛擬網(wǎng)卡 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 (以太網(wǎng)) 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 ... # 設(shè)置 USB 網(wǎng)卡的 IP 地址和開發(fā)板的 usb0 的 IP 地址在同一網(wǎng)段 firefly@Desktop:~$ sudo ifconfig enp0s20u2i1 192.168.1.100 #設(shè)置默認(rèn)網(wǎng)關(guān):要設(shè)置為開發(fā)板 usb0 的 ip 地址,因為后面要通過 usb0 來進行流量的轉(zhuǎn)發(fā) firefly@Desktop:~$ sudo route add default gw 192.168.1.101
設(shè)置完開發(fā)板和 PC 機的 IP 后,互相是可以 ping 通的,PC 機也可以用 ssh
命令登錄到開發(fā)板。
在開發(fā)板上:首先打開 IPv4 的轉(zhuǎn)發(fā)功能:
echo 1 > /proc/sys/net/ipv4/ip_forward
如果希望每次重啟開發(fā)板后都自動打開轉(zhuǎn)發(fā)功能,請直接修改 /etc/sysctl.conf
文件的 net.ipv4.ip_forward
值為1。修改文件參數(shù)后要執(zhí)行 sysctl -p
命令重新載入 /etc/sysctl.conf
文件,使 IPv4 轉(zhuǎn)發(fā)功能生效。
添加規(guī)則進行流量轉(zhuǎn)發(fā):
iptables -t nat -A POSTROUTING -s 192.168.1.0/24 -o eth0 -j SNAT --to-source 168.168.100.48
現(xiàn)在 PC 主機就可以訪問網(wǎng)絡(luò)了。如果 PC 機可以 ping 通開發(fā)板的 usb0 和 eth0,但是無法上外網(wǎng),則需要修改 PC 機的 DNS,在 /etc/resolv.conf
中做以下添加:
nameserver 8.8.8.8 nameserver 8.8.4.4
配置過程中要注意以下幾點:
-
對應(yīng)好上述步驟中自己設(shè)備上的各個 IP 地址,注意開發(fā)板上的 USB 虛擬網(wǎng)卡 IP 和有線網(wǎng)絡(luò) IP 不在同一網(wǎng)段;
-
PC 機虛擬 USB 網(wǎng)卡的網(wǎng)關(guān)要設(shè)置為開發(fā)板虛擬 USB 網(wǎng)卡的 IP 地址;
-
開發(fā)板上的虛擬網(wǎng)卡 IP 地址、IP 轉(zhuǎn)發(fā)功能、流量轉(zhuǎn)發(fā)規(guī)則等設(shè)置會在設(shè)備重啟后恢復(fù),用戶可以自行查找資料如何開機自動設(shè)置這些配置。
根文件系統(tǒng)除了可以使用在內(nèi)部的 eMMC 中的,還可以使用外部存儲設(shè)備的根文件系統(tǒng),如 SD 卡,U 盤等。以下是以 SD 卡為例,在 Firefly-RK3399 開發(fā)板上實現(xiàn)掛載外部存儲設(shè)備的根文件系統(tǒng)。
在 PC 機上插入 SD 卡,用 gdisk
工具分出一個裝載根文件系統(tǒng)的分區(qū):
#用戶請用 `fdisk -l` 查詢 SD 卡設(shè)備名是什么,在用 gdisk 命令進入對應(yīng)的設(shè)備 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 卡當(dāng)前已有分區(qū),由于使用的 SD 卡暫未創(chuàng)建分區(qū),所以沒有查詢到分區(qū)信息 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 ------------------------------------------------------------------- #創(chuàng)建新的分區(qū),請用戶根據(jù)自身實際情況創(chuàng)建合適的分區(qū)大小給根文件系統(tǒng) Command (? for help): n #輸入 n 創(chuàng)建新分區(qū) Partition number (1-128, default 1): 1 #輸入1創(chuàng)建第一個分區(qū) First sector (34-15278046, default = 2048) or {+-}size{KMGTP}: #直接按回車,使用默認(rèn)值 Last sector (2048-15278046, default = 15278046) or {+-}size{KMGTP}: +3G #創(chuàng)建分區(qū)大小為3G Current type is 'Linux filesystem' Hex code or GUID (L to show codes, Enter = 8300): #按回車使用默認(rèn)值 Changed type of partition to 'Linux filesystem' ------------------------------------------------------------------- #創(chuàng)建完成后,輸入 i,可以查看分區(qū)的 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 #使用的是該分區(qū)的 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.
把新創(chuàng)建的分區(qū)進行格式化。
# 注意 sdb1 表示 SD 卡對應(yīng)分區(qū)設(shè)備名,請用戶根據(jù)實際情況填入 sudo mkfs.ext4 /dev/sdb1
格式化完成后,使用 dd
命令,將制作好的根文件系統(tǒng)燒寫到 SD 卡剛新建的分區(qū)中。(根文件系統(tǒng)的制作可以參考Ubuntu 根文件系統(tǒng)的制作)
# 用戶請根據(jù)實際情況填寫根文件系統(tǒng)鏡像的路徑和 SD 卡對應(yīng)分區(qū)設(shè)備名 dd if=/rootfs_path/rootfs.img of=/dev/sdb1
首先要修改設(shè)備樹文件,打開對應(yīng)的 dts 文件,在根節(jié)點下重寫 chosen
節(jié)點下的 bootargs
參數(shù)的 root
值為寫入了根文件系統(tǒng)的 SD 卡分區(qū)的 unique GUID:
#將 root 的值改為獲取到的 unique GUID 值的前12位 chosen { bootargs = "earlycon=uart8250,mmio32,0xff1a0000 swiotlb=1 console=ttyFIQ0 rw root=PARTUUID=6278AF5D-BC6B rootfstype=ext4 rootwait"; };
修改完成后編譯并燒錄至開發(fā)板中。
SD 卡的根文件系統(tǒng)燒錄完成后,插入開發(fā)板的 TF 卡槽中,開機即可進入到 SD 的根文件系統(tǒng)中。
注意事項:
-
在操作前請注意備份 U 盤或者 SD 卡內(nèi)的重要文件,避免丟失;
-
操作時請確認(rèn)自己的 SD 卡對應(yīng)的設(shè)備名;
-
dts 文件中
root
參數(shù)的值使用的是 unique GUID,記下前12位即可。用戶也可以根據(jù)gdisk
的幫助信息自行修改這個值。
網(wǎng)絡(luò)啟動,是用 TFTP 在服務(wù)器下載內(nèi)核、dtb 文件到目標(biāo)機的內(nèi)存中,同時可以用 NFS 掛載網(wǎng)絡(luò)根文件系統(tǒng)到目標(biāo)機上,實現(xiàn)目標(biāo)機的無盤啟動。以下基于 Firefly-RK3399 作出一個示例,提供用戶參考。
準(zhǔn)備工作:
-
Firefly-RK3399 開發(fā)板;
-
路由器、網(wǎng)線;
-
安裝有 NFS 和 TFTP 的服務(wù)器;
-
一份制作好的根文件系統(tǒng)。
注:示例中使用的是 Ubuntu 系統(tǒng)的 PC 機作為服務(wù)器,通過路由器和網(wǎng)線實現(xiàn)與開發(fā)板的連接。用戶可以根據(jù)自己的實際情況作調(diào)整,但如果是 PC 機直連開發(fā)板,請使用交叉網(wǎng)線。請確保服務(wù)器和目標(biāo)機在同一局域網(wǎng)內(nèi)。
1、在服務(wù)器上部署 TFTP 服務(wù):
安裝 TFTP 服務(wù):
sudo apt-get install tftpd-hpa
創(chuàng)建 /tftpboot
目錄并賦予權(quán)限:
mkdir /tftpboot sudo chmod 777 /tftpboot
然后修改 TFTP 服務(wù)器的配置文件 /etc/default/tftpd-hpa
,修改內(nèi)容為:
# /etc/default/tftpd-hpa TFTP_USERNAME="tftp" TFTP_DIRECTORY="/tftpboot" #這個根據(jù)用戶實際的 tftp 目錄設(shè)置 TFTP_ADDRESS="0.0.0.0:69" TDTP_OPTIONS="-c -s -l"
退出修改后重啟 TFTP 服務(wù)器:
sudo service tftpd-hpa restart
在 /tftpboot
目錄中創(chuàng)建一個文件,測試 TFTP 服務(wù)是否可用:
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 服務(wù)器是可用的。
2、在服務(wù)器上部署 NFS 服務(wù):
安裝 NFS 服務(wù)器:
sudo apt-get install nfs-kernel-server
創(chuàng)建共享目錄:
sudo mkdir /nfs sudo chmod 777 /nfs cd /nfs sudo mkdir rootfs sudo chmod 777 rootfs
然后將制作好的根文件系統(tǒng)復(fù)制到 /nfs/rootfs
目錄中。(根文件系統(tǒng)的制作可以參考:Ubuntu 根文件系統(tǒng)的制作)
接著配置 NFS,在 /etc/exports
中添加共享目錄位置:
/nfs/rootfs *(rw,sync,no_root_squash,no_subtree_check)
共享的目錄根據(jù)用戶實際情況設(shè)置,其中的”*”代表的是所有用戶可訪問。
重啟 NFS 服務(wù)器:
sudo /etc/init.d/nfs-kernel-server restart
本地掛載共享目錄,測試 NFS 服務(wù)器是否可用:
sudo mount -t nfs 127.0.0.1:/nfs/rootfs/ /mnt
在 /mnt
目錄下查看到了與 /nfs/rootfs
一致的內(nèi)容,則說明 NFS 服務(wù)器部署成功。然后可以解除掛載:
sudo umount /mnt
如果要做到掛載網(wǎng)絡(luò)根文件系統(tǒng),需要在內(nèi)核中做相關(guān)配置,并在 dts 中修改相關(guān)掛載根文件系統(tǒng)的配置。
首先進行內(nèi)核配置,在內(nèi)核目錄中執(zhí)行 make menuconfig
,選擇相關(guān)配置:
[*] 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
節(jié)點下的 bootargs
參數(shù),選擇使用 NFS 掛載遠(yuǎn)程根文件系統(tǒng),內(nèi)容如下。
chosen { bootargs = "earlycon=uart8250,mmio32,0xff1a0000 swiotlb=1 console=ttyFIQ0 root=/dev/nfs rootfstype=ext4 rootwait"; }; #主要是修改 root 的值,root=/dev/nfs 表示通過 NFS 掛載網(wǎng)絡(luò)根文件系統(tǒng)
編譯內(nèi)核:
make ARCH=arm64 rk3399-firefly.img -j12
編譯完成后,將編譯好的內(nèi)核文件 boot.img
和 rk3399-firefly.dtb
文件復(fù)制到 /tftpboot
目錄中:
cp boot.img /tftpboot cp arch/arm64/boot/dts/rockchip/rk3399-firefly.dtb /tftpboot
詳細(xì)說明可以參考內(nèi)核目錄中的 kernel/Documentation/filesystems/nfs/nfsroot.txt
請先確保目標(biāo)機網(wǎng)線已插入,接入到服務(wù)器的局域網(wǎng)內(nèi)。
目標(biāo)機啟動進入 U-Boot 命令行模式,設(shè)置以下參數(shù):
=> setenv ipaddr 192.168.31.101 #設(shè)置目標(biāo)機 IP 地址 => setenv serverip 192.168.31.106 #設(shè)置 serverip 為服務(wù)器 IP 地址 #設(shè)置從 TFTP 下載內(nèi)核和 dtb 文件到相應(yīng)地址,用戶請根據(jù)自己實際的目標(biāo)機修改相應(yīng)地址 => setenv bootcmd tftpboot 0x0027f800 boot.img \; tftpboot 0x08300000 rk3399-firefly.dtb \; bootm 0x0027f800 - 0x08300000 #設(shè)置掛載網(wǎng)絡(luò)根文件系統(tǒng),IP 參數(shù)依次為:目標(biāo)機 IP:服務(wù)器 IP:網(wǎng)關(guān):網(wǎng)絡(luò)掩碼:設(shè)備名:off,可以更簡單的設(shè)置 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 #如果不想每次開機都設(shè)置一遍,可以保存上面配置的變量 => saveenv Saving Environment to MMC... Writing to MMC(0)... done #啟動目標(biāo)機 => 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 ... ...
在開機內(nèi)核日志中可見:
[ 12.146297] VFS: Mounted root (nfs filesystem) on device 0:16.
說明已經(jīng)掛載上了網(wǎng)絡(luò)根文件系統(tǒng)。
注意事項
-
確保 TFTP 服務(wù)器、NFS 服務(wù)器可用;
-
確保目標(biāo)機先插入網(wǎng)線后在開機,且和服務(wù)器在同一局域網(wǎng)內(nèi),如果是直連目標(biāo)機和服務(wù)器,請使用交叉網(wǎng)線;
-
內(nèi)核配置中,
Root file system on NFS
依賴于IP: kernel level autoconfiguration
選項,請先選擇IP: kernel level autoconfiguration
,之后才可以找到Root file system on NFS
選項; -
在 U-Boot 命令行中,請確認(rèn)
boot.img
燒錄地址和 dtb 文件燒錄地址。(提示:boot.img
的文件結(jié)構(gòu)中,開頭有2k的頭文件,然后才是 kernel。所以在 TFTP 下載內(nèi)核到目標(biāo)機時,要下載到對應(yīng) kernel 地址減去0x0800的地址上); -
在設(shè)置掛載遠(yuǎn)程根文件系統(tǒng)時,
nfsroot=192.168.31.106:/nfs/rootfs,v3
中的v3
代表 NFS 版本信息,請?zhí)砑由弦员苊獬霈F(xiàn)掛載不成功的問題。
本小節(jié)介紹了在線更新的一個簡單的流程。將內(nèi)核、U-Boot 或者其他需要更新的文件打包成 deb 安裝包,然后導(dǎo)入到本地包倉庫,實現(xiàn)在開發(fā)板上下載并自動更新。僅供用戶參考。
操作中需要升級內(nèi)核和 U-Boot,事先已經(jīng)準(zhǔn)備好了修改好的相關(guān)文件:uboot.img
、trust.img
、boot.img
。
deb 是 Debian Linux 的軟件包格式,打包最關(guān)鍵的是在 DEBIAN
目錄下創(chuàng)建一個 control
文件。首先創(chuàng)建 deb 工作目錄,然后在 deb 目錄中創(chuàng)建相應(yīng)的目錄和文件:
mkdir deb cd deb mkdir firefly-firmware #創(chuàng)建 firefly-firmware 目錄 cd firefly-firmware mkdir DEBIAN #創(chuàng)建 DEBIAN 目錄,這個目錄是必須要有的。 mkdir -p usr/share/{kernel,uboot} mv ~/boot.img ~/deb/firefly-firmware/usr/share/kernel #將相應(yīng)文件放到相應(yīng)的目錄 mv ~/uboot.img ~/deb/firefly-firmware/usr/share/uboot mv ~/trust.img ~/deb/firefly-firmware/usr/share/uboot
DEBIAN
目錄下存放的文件是 deb 包安裝的控制文件以及相應(yīng)的腳本文件。
在 DEBIAN
目錄下創(chuàng)建控制文件 control
和腳本文件 postinst
。
control
文件內(nèi)容如下,用于記錄軟件標(biāo)識,版本號,平臺,依賴信息等數(shù)據(jù)。
Package: firefly-firmware #文件目錄名 Version: 1.0 #版本號 Architecture: arm64 #架構(gòu) Maintainer: neg #維護人員,自定義即可 Installed-Size: 1 Section: test Priority: optional Descriptionon: This is a deb test
postinst
文件內(nèi)容如下,就是將需要更新的內(nèi)核和 U-Boot 文件用 dd
命令寫進對應(yīng)分區(qū)的腳本:
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
腳本,是在解包數(shù)據(jù)后運行的腳本,對應(yīng)的還有:
-
preinst:在解包數(shù)據(jù)前運行的腳本;
-
prerm:卸載時,在刪除文件之前運行的腳本;
-
postrm:在刪除文件之后運行的腳本;
其他相關(guān)知識點用戶可以自行了解。這里只用到了 preinst
腳本。
以下是創(chuàng)建好的目錄樹:
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 包時,一般按照這種規(guī)范進行命名:package_version-reversion_arch.deb
。
首先安裝需要的包:
sudo apt-get install reprepro gnupg
然后用 GnuPG 工具生成一個 GPG 密匙,執(zhí)行命令后請根據(jù)提示操作:
gpg --gen-key
執(zhí)行 sudo gpg --list-keys
可以查看剛剛生成的密匙信息:
sudo gpg --list-keys gpg: WARNING: unsafe ownership on homedir '/home/firefly/.gnupg' gpg: 正在檢查信任度數(shù)據(jù)庫 gpg: marginals needed: 3 completes needed: 1 trust model: pgp gpg: 深度:0 有效性: 4 已簽名: 0 信任度:0-,0q,0n,0m,0f,4u gpg: 下次信任度數(shù)據(jù)庫檢查將于 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]
接下來創(chuàng)建包倉庫,首先創(chuàng)建目錄:
cd /var/www mkdir apt #包倉庫目錄 mkdir -p ./apt/incoming mkdir -p ./apt/conf mkdir -p ./apt/key
把前面生成的密匙導(dǎo)出到倉庫文件夾,請用戶對應(yīng)好自己創(chuàng)建的用戶名和郵箱地址。
gpg --armor --export firefly firefly@t-chip.com > /var/www/apt/key/deb.gpg.key
在 conf
目錄下創(chuàng)建 distributions
文件,其內(nèi)容如下:
Origin: Neg #你的名字 Label: Mian #庫的名字 Suite: stable #(stable 或 unstable) Codename: bionic #發(fā)布代碼名,可自定義 Version: 1.0 #發(fā)布版本 Architectures: arm64 #架構(gòu) Components: main #組件名,如main,universe等 Description: Deb source test #相關(guān)描述 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
你的包已經(jīng)加入了倉庫,如果要移除它的話采用如下命令:
reprepro --ask-passphrase -Vb /var/www/apt remove bionic firefly-firmware
安裝 nginx 服務(wù)器:
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 服務(wù)器:
sudo service nginx restart
在客戶端開發(fā)板中,首先要添加本地包倉庫的源,在目錄 /etc/apt/sources.list.d
下添加一個新的配置文件 bionic.list
,內(nèi)容如下:
deb http://192.168.31.106 bionic main
IP 地址是服務(wù)器地址, bionic
是倉庫發(fā)布代碼名, main
是組件名。
從服務(wù)器中獲取并添加 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
可以看到安裝過程中執(zhí)行了 deb 包中的 poinst
腳本。安裝完成后,重啟開發(fā)板,更新完成。
在 /usr/share
目錄中還可以看到 kernel
和 uboot
目錄,分別存放了 boot.img
,uboot.img
和 trust.img
。
注意事項
-
在制作 deb 包中,與
DEBIAN
同級的目錄視為根目錄,即放在與DEBIAN
同目錄下的文件,在客戶端安裝 deb 包后,可以在客戶端的根目錄下找到; -
deb 包中的文件和腳本,用戶要根據(jù)自己的實際情況做調(diào)整;
-
每次在倉庫中修改了配置文件后,都要重新導(dǎo)入倉庫目錄樹 ;
-
nginx 服務(wù)器配置中,
root
參數(shù)配置的是倉庫的地址,請用戶根據(jù)自己的實際情況修改; -
客戶端添加的新下載源的文件時,注意核對正確的服務(wù)器地址,包倉庫代碼名和組件名。注意客戶端要連通服務(wù)器;
-
客戶端要用
apt-key add
命令添加 GPG 密匙之后,才可以獲取本地的倉庫的信息。
distcc
是個通過網(wǎng)絡(luò)上的若干臺機器用來分布式編譯 C、C++、Objective-C 或 Objective-C++ 代碼的程序。distcc
不要求所有的機器共享文件系統(tǒng),有同步的時鐘,或者安裝有同樣的庫和頭文件,只要作為服務(wù)器的機器有合適的編譯工具即可進行編譯。本例在兩塊 Firefly-RK3399 開發(fā)板 (arm64) 和一臺 PC 機 (x86_64) 上利用 Docker 技術(shù)來布署 distcc 分布式編譯服務(wù),進而實現(xiàn)在其中一塊 Firefly-RK3399 開發(fā)板上利用 distcc 的分布式編譯特性來加速內(nèi)核的編譯過程。
-
兩塊 Firefly-RK3399 開發(fā)板;
-
路由器、網(wǎng)線;
-
PC 機。
將兩塊開發(fā)板和 PC 機都連接到同一個局域網(wǎng)內(nèi)。連接好后對應(yīng)的 IP 地址別是:
-
PC 機:192.168.1.100
-
開發(fā)板 A:192.168.1.101
-
開發(fā)板 B:192.168.1.102
使用腳本安裝 Docker :
wget -qO- https://get.docker.com/ | sh
為了可以使當(dāng)前普通用戶可以執(zhí)行 docker 相關(guān)命令,需要將當(dāng)前用戶添加到 dokcer 用戶組中:
sudo groupadd docker #添加 docker 用戶組 sudo gpasswd -a $USER docker #將當(dāng)前用戶添加到 docker 用戶組中 newgrp docker #更新 docker 用戶組
啟動 Docker 后臺服務(wù):
sudo service docker start
新建一個 Dockerfile_distcc.x86_64
文件。其文件內(nèi)容如下:
FROM ubuntu:bionic MAINTAINER Firefly <service@t-firefly.com> 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 容器,在主機網(wǎng)絡(luò)的 3632 TCP 端口提供 distcc 服務(wù):
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
Firefly-RK3399 內(nèi)核默認(rèn)是不支持 Docker 的,需要進行相關(guān)內(nèi)核的配置。可以在 Firefly Github 中查看修改后的 /kernel/arch/arm64/configs/firefly_linux_defconfig
文件。
修改好后編譯內(nèi)核并更新到所有用到的開發(fā)板。
內(nèi)核更新完成后,在開發(fā)板上使用腳本安裝 Docker :
apt-get update wget -qO- https://get.docker.com/ | sh
將當(dāng)前用戶添加到 dokcer 用戶組中:
sudo groupadd docker sudo gpasswd -a $USER docker newgrp docker #更新 docker
啟動 Docker 后臺服務(wù):
service docker start
新建 Dockerfile_distcc.arm64
文件,內(nèi)容如下:
FROM ubuntu:bionic MAINTAINER Firefly <service@t-firefly.com> 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 容器,在主機網(wǎng)絡(luò)的 3632 TCP 端口提供 distcc 服務(wù):
docker run -d -p 3632:3632 distcc_server:arm64
用 docker save
命令導(dǎo)出新建的鏡像:
docker save -o distcc_server.tar distcc_server:arm64
將生成的 distcc_server.tar
文件復(fù)制到另一塊開發(fā)板上,然后導(dǎo)入鏡像:
docker load -i distcc_server.tar
導(dǎo)入鏡像后,在這個開發(fā)板上也有了 distcc_server:arm64
鏡像,然后可以運行一個容器:
docker run -d -p 3632:3632 distcc_server:arm64
提示: 用戶如果注冊了 Docker Hub 帳號,可以用 docker push
推送鏡像到遠(yuǎn)程倉庫,在另一開發(fā)板上用 docker pull
拉取遠(yuǎn)程倉庫的鏡像。用戶可以自行了解相關(guān)操作。
到這里,三個機器都部署了分布式編譯環(huán)境。可以選擇任意一臺機器作為客戶端,剩下兩個機器作為服務(wù)器。這里選擇其中一個開發(fā)板作為客戶端。
在客戶端中創(chuàng)建 Dockerfile_compile.arm64
文件,生成一個編譯內(nèi)核的 Docker 鏡像,文件內(nèi)容如下:
FROM ubuntu:bionic MAINTAINER Firefly <service@t-firefly.com> 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 .
在啟動容器前先將內(nèi)核文件拉到客戶端中。然后創(chuàng)建 /etc/distcc/hosts
文件,其內(nèi)容是列舉出所有提供 distcc 服務(wù)的機器的 IP 地址。內(nèi)容如下:
# 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
為了測試結(jié)果更準(zhǔn)確,先清除客戶端開發(fā)板上的緩存:
echo 3 > /proc/sys/vm/drop_caches
用 compile:arm64
鏡像啟動一個 Docker 容器,同時將主機當(dāng)前內(nèi)核目錄掛載到容器中的 /mnt
目錄,將主機的 /etc/distcc/hosts
文件掛載到容器的 /etc/distcc/hosts
中,并用參數(shù) -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 編譯內(nèi)核,加入 time
命令可以查看編譯命令執(zhí)行耗時,CC
參數(shù)指明用 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"
編譯過程中,可以在用于編譯的容器內(nèi)新的窗口中用命令 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
下邊是單獨使用一塊開發(fā)板進行內(nèi)核編譯所耗費的時間:
real 23m33.002s user 113m2.615s sys 9m29.348s
對比可見,用采用 distcc 實現(xiàn)的分布式編譯可以有效提高編譯速度。
注意:
-
平臺不同,所需的編譯器也不同。如在 x86_64 平臺上,需要安裝對應(yīng)的交叉編譯工具
gcc-aarch64-linux-gnu
。在 arm64 平臺則只需要安裝gcc
編譯工具即可。用戶需要根據(jù)實際情況在對應(yīng)的Dockerfile
文件里指定安裝正確的工具。 -
如果用 PC 機做客戶端,則在用于編譯的容器中編譯內(nèi)核的時候,要用參數(shù)
CC="distcc aarch64-linux-gnu-gcc"
指明用aarch64-linux-gnu-gcc
交叉編譯工具編譯。
-
嵌入式主板
+關(guān)注
關(guān)注
7文章
6086瀏覽量
35475 -
安卓
+關(guān)注
關(guān)注
5文章
2134瀏覽量
57366 -
可制造性設(shè)計
+關(guān)注
關(guān)注
10文章
2065瀏覽量
15642 -
Firefly
+關(guān)注
關(guān)注
2文章
538瀏覽量
7071 -
華秋DFM
+關(guān)注
關(guān)注
20文章
3494瀏覽量
4613
發(fā)布評論請先 登錄
相關(guān)推薦
評論