資料介紹
描述
介紹
本文中的“app”是指預(yù)編譯的二進(jìn)制文件,無(wú)需使用 Arduino IDE,即可直接在 Arduino 板卡上運(yùn)行。
并且因?yàn)樗且粋€(gè)文件,“應(yīng)用程序”可以通過(guò) SD 卡、以太網(wǎng)、WiFi 或任何合適的方法分發(fā)。
標(biāo)題圖顯示了執(zhí)行 Arduino 應(yīng)用程序RTT-QRCode的 MKR ZERO 板。
你有興趣嗎?
(本文基于Arduino RT-Thread庫(kù)v0.6.0。)
動(dòng)態(tài)模塊
在 RT-Thread 架構(gòu)中,“app”被稱為動(dòng)態(tài)模塊,構(gòu)建為動(dòng)態(tài)共享庫(kù),擴(kuò)展名為“ .mo
”或“ .so
”。(什么是 RT-Thread?=> Arduino 上的多任務(wù)處理)
RT-Thread 提供 API 來(lái)訪問(wèn)動(dòng)態(tài)模塊。更有趣的是,MSH(一個(gè)微型外殼)能夠.mo
直接執(zhí)行“”文件(詳細(xì)信息在以下部分中)。
RT-Thread 的原始動(dòng)態(tài)鏈接器似乎不適用于 ARM Cortex-M。所以我修改了 Arduino RT-Thread 庫(kù)的代碼。
多發(fā)性硬化癥
Module SHell (MSH) 是默認(rèn)啟用的一項(xiàng)新功能(從 v0.5.1 開(kāi)始),它構(gòu)建在 FinSH 之上。(什么是 FinSH?=> Arduino 上的多任務(wù)處理)
由于 Arduino 應(yīng)用程序是由 MSH 執(zhí)行的,讓我們簡(jiǎn)單介紹一下。
相比 FinSH,MSH 更符合 Unix shell 的使用習(xí)慣:
- 在 FinSH 中發(fā)出命令
led(0, 1)
copy("datalog.txt", "copy.txt")
- 在 MSH 中發(fā)出命令
led 0 1
cp datalog.txt copy.txt
但是,MSH 不支持像 FinSH 提供的那樣的 shell 變量。
另一個(gè)限制是用戶定義的 MSH 命令的原型是固定的:
int my_msh_cmd(int argc, char **argv)
MSH 執(zhí)行用戶命令時(shí),參數(shù)argc
為參數(shù)個(gè)數(shù)加一,參數(shù)列表argv
為參數(shù)列表(firstentryiscommandname)。您可能已經(jīng)猜到了,所有參數(shù)只能是char
數(shù)組類型。
以下是 MSH 命令格式的“l(fā)ed”示例。
int led(int argc, char **argv) {
// argc - the number of arguments
// argv[0] - command name, e.g. "led"
// argv[n] - nth argument in the type of char array
rt_uint32_t id;
rt_uint32_t state;
if (argc != 3) {
rt_kprintf("Usage: led \n");
return 1;
}
rt_kprintf("led%s=%s\n", argv[1], argv[2]);
// convert arguments to their specific types
sscanf(argv[1], "%u", &id);
sscanf(argv[2], "%u", &state);
if (id != 0) {
rt_kprintf("Error: Invalid led ID\n");
return 1;
}
if (state) {
digitalWrite(LED_BUILTIN, HIGH);
} else {
digitalWrite(LED_BUILTIN, LOW);
}
return 0;
}
制作 Arduino 應(yīng)用程序
首先,CONFIG_USING_MODULE
在“rtconfig.h”中啟用,因?yàn)槟J(rèn)情況下它是禁用的。
構(gòu)建可執(zhí)行文件
讓我們?cè)?Arduino IDE 中打開(kāi)“HelloMo”示例,然后按“驗(yàn)證”。(該示例也可以在下面的“代碼”部分中找到。)代碼現(xiàn)在被構(gòu)建到一個(gè)包含草圖和庫(kù)的單個(gè)可執(zhí)行文件中。我們可以使用 GCC 工具readelf
(Arduino IDE 提供)來(lái)驗(yàn)證。
{path_to_gcc_tools}\arm-none-eabi-readelf -h {path_to_output}\HelloMo.ino.elf
ELF Header:
Magic: 7f 45 4c 46 01 01 01 00 00 00 00 00 00 00 00 00
Class: ELF32
Data: 2's complement, little endian
Version: 1 (current)
OS/ABI: UNIX - System V
ABI Version: 0
Type: EXEC (Executable file)
Machine: ARM
Version: 0x1
Entry point address: 0xf7fd
Start of program headers: 52 (bytes into file)
Start of section headers: 798052 (bytes into file)
Flags: 0x5000002, has entry point, Version5 EABI
Size of this header: 52 (bytes)
Size of program headers: 32 (bytes)
Number of program headers: 2
Size of section headers: 40 (bytes)
Number of section headers: 18
Section header string table index: 15
如果您不確定 GCC 工具和編譯輸出的位置,請(qǐng)?jiān)?File-> Preferences 中啟用以下選項(xiàng)。
再次單擊“驗(yàn)證”,您將在輸出窗口中觀察到信息:
...
Compiling sketch...
"C:\\Users\\onelife\\AppData\\Local\\Arduino15\\packages\\arduino\\tools\\arm-none-eabi-gcc\\4.8.3-2014q1/bin/arm-none-eabi-gcc" -mcpu=cortex-m0plus -mthumb -c -g -Os -Wall -Wextra -std=gnu11 -ffunction-sections -fdata-sections -nostdlib --param max-inline-insns-single=500 -MMD -DF_CPU=48000000L -DARDUINO=10809 -DARDUINO_SAMD_MKRZERO -DARDUINO_ARCH_SAMD -DUSE_ARDUINO_MKR_PIN_LAYOUT -D__SAMD21G18A__ -DUSB_VID=0x2341 -DUSB_PID=0x804f -DUSBCON "-DUSB_MANUFACTURER="Arduino LLC"" "-DUSB_PRODUCT="Arduino MKRZero"" "-IC:\\Users\\onelife\\AppData\\Local\\Arduino15\\packages\\arduino\\tools\\CMSIS\\4.5.0/CMSIS/Include/" "-IC:\\Users\\onelife\\AppData\\Local\\Arduino15\\packages\\arduino\\tools\\CMSIS-Atmel\\1.1.0/CMSIS/Device/ATMEL/" "-IC:\\Users\\onelife\\AppData\\Local\\Arduino15\\packages\\arduino\\hardware\\samd\\1.6.21\\cores\\arduino" "-IC:\\Users\\onelife\\AppData\\Local\\Arduino15\\packages\\arduino\\hardware\\samd\\1.6.21\\variants\\mkrzero" "-IC:\\Users\\onelife\\Documents\\Arduino\\libraries\\RT-Thread\\src" "-IC:\\Users\\onelife\\AppData\\Local\\Arduino15\\packages\\arduino\\hardware\\samd\\1.6.21\\libraries\\SPI" "C:\\Users\\onelife\\AppData\\Local\\Temp\\arduino_build_508434\\sketch\\hello_mo.c" -o "C:\\Users\\onelife\\AppData\\Local\\Temp\\arduino_build_508434\\sketch\\hello_mo.c.o"
...
就我而言,GCC 工具位于“ C:\\Users\\onelife\\AppData\\Local\\Arduino15\\packages\\arduino\\tools\\arm-none-eabi-gcc\\4.8.3-2014q1/bin/
”,編譯輸出位于“ C:\\Users\\onelife\\AppData\\Local\\Temp\\arduino_build_508434\\
”。
構(gòu)建應(yīng)用程序(動(dòng)態(tài)共享庫(kù))
但是,我們要構(gòu)建的目標(biāo)“應(yīng)用程序”是一種共享庫(kù)。它必須與位置無(wú)關(guān),因此可以加載到任何 RAM 地址中。為了讓它更小(因?yàn)槲覀兊?RAM 大小是有限的),最終的二進(jìn)制文件將不包含其他庫(kù)的任何功能。(所有外部功能都應(yīng)由固件端提供。)
壞消息是 Arduino IDE 不提供這些選項(xiàng)。好消息是 Arduino IDE 確實(shí)提供了我們需要的所有工具。我們開(kāi)始做吧。
第一步是編譯。
我們必須將選項(xiàng)“ -mlong-calls -fPIC
”添加到原始編譯命令中(在輸出窗口中查找“正在編譯草圖...”)。
{path_to_gcc_tools}\arm-none-eabi-gcc -mlong-calls -fPIC ... {path_to_output}\sketch\hello_mo.c -o {path_to_output}\sketch\hello_mo.c.o
{path_to_gcc_tools}\arm-none-eabi-gcc -mlong-calls -fPIC ... {path_to_output}\sketch\load_mo.c -o {path_to_output}\sketch\load_mo.c.o
第二步是鏈接。
在這一步中,我們選擇將目標(biāo)文件構(gòu)建為“ app ”(帶有入口點(diǎn)的“.mo”文件)或?qū)⑵錁?gòu)建為庫(kù)(不帶入口點(diǎn)的“.so”文件)。在以下示例中,我們將“ load_mo.c.o
”構(gòu)建為“app”,并將“ hello_mo.c.o
”構(gòu)建為庫(kù)。
我們通過(guò)以下方式修改鏈接命令(尋找“將所有內(nèi)容鏈接在一起......”)
-
只保留目標(biāo)目標(biāo)文件,例如“
load_mo.c.o
”,并刪除其他 -
刪除選項(xiàng)“
-Wl,--unresolved-symbols=report-all
” -
刪除選項(xiàng)“
-L{path_to_output}
” -
刪除選項(xiàng)“
-T.../flash_with_bootloader.ld
” -
刪除選項(xiàng)“
-Wl,--start-group ... -Wl,--end-group
” -
添加選項(xiàng)“
-shared -fPIC -nostdlib -Wl,-marmelf -Wl,-z,max-page-size=0x4
” -
添加入口點(diǎn)選項(xiàng)(例如“
-Wl,-eload_hello
”或“-Wl,-e0
”表示無(wú))
{path_to_gcc_tools}\arm-none-eabi-g++ -shared -fPIC -nostdlib -Wl,-e0 -Wl,-marmelf -Wl,-z,max-page-size=0x4 ... -o {path_to_output}\hello_mo.elf {path_to_output}\hello_mo.c.o
{path_to_gcc_tools}\arm-none-eabi-g++ -shared -fPIC -nostdlib -Wl,-eload_hello -Wl,-marmelf -Wl,-z,max-page-size=0x4 ... -o {path_to_output}\load_mo.elf {path_to_output}\load_mo.c.o
第三步是分條。
為了進(jìn)一步減小文件大小,我們必須去掉 ELF 文件中不必要的部分。
{path_to_gcc_tools}\arm-none-eabi-strip -R .hash -R .comment -R .ARM.attributes {path_to_output}\hello_mo.elf -o {path_to_output}\hello.so
{path_to_gcc_tools}\arm-none-eabi-strip -R .hash -R .comment -R .ARM.attributes {path_to_output}\load_mo.elf -o {path_to_output}\load.mo
第四步是檢查大小(可選)。
{path_to_gcc_tools}\arm-none-eabi-size {path_to_output}\hello.so
{path_to_gcc_tools}\arm-none-eabi-size {path_to_output}\load.mo
恭喜!您剛剛構(gòu)建了一個(gè) Arduino 應(yīng)用程序。讓我們檢查一下輸出。
{path_to_gcc_tools}\arm-none-eabi-readelf -h {path_to_output}\hello.so
ELF Header:
Magic: 7f 45 4c 46 01 01 01 00 00 00 00 00 00 00 00 00
Class: ELF32
Data: 2's complement, little endian
Version: 1 (current)
OS/ABI: UNIX - System V
ABI Version: 0
Type: DYN (Shared object file)
Machine: ARM
Version: 0x1
Entry point address: 0x0
Start of program headers: 52 (bytes into file)
Start of section headers: 896 (bytes into file)
Flags: 0x5000000, Version5 EABI
Size of this header: 52 (bytes)
Size of program headers: 32 (bytes)
Number of program headers: 3
Size of section headers: 40 (bytes)
Number of section headers: 9
Section header string table index: 8
{path_to_gcc_tools}\arm-none-eabi-readelf -h {path_to_output}\load.mo
ELF Header:
Magic: 7f 45 4c 46 01 01 01 00 00 00 00 00 00 00 00 00
Class: ELF32
Data: 2's complement, little endian
Version: 1 (current)
OS/ABI: UNIX - System V
ABI Version: 0
Type: DYN (Shared object file)
Machine: ARM
Version: 0x1
Entry point address: 0x285
Start of program headers: 52 (bytes into file)
Start of section headers: 1060 (bytes into file)
Flags: 0x5000002, has entry point, Version5 EABI
Size of this header: 52 (bytes)
Size of program headers: 32 (bytes)
Number of program headers: 3
Size of section headers: 40 (bytes)
Number of section headers: 9
Section header string table index: 8
它表明“ .so
”和“ .mo
”文件都是“ DYN
”(動(dòng)態(tài))類型。不同之處在于“ .so
”文件沒(méi)有入口點(diǎn),而“ .mo
”文件有。
我們還沒(méi)有完成。
最后一步是公開(kāi)應(yīng)用程序所需的功能。
在文件“ ”中,如果啟用mo_sym.h
,所有內(nèi)核 API 都已經(jīng)公開(kāi)。CONFIG_USING_MODULE
如有必要,您可以添加自己的。
發(fā)出 MSH 命令“ lsym
”將列出所有暴露的符號(hào):
運(yùn)行 Arduino 應(yīng)用程序
hello.so
讓我們將“ ”和“ ”復(fù)制load.mo
到具有以下文件結(jié)構(gòu)的SD卡。
SD_ROOT/
├── lib/
│ └── hello.so
└── mo/
└── load.mo
規(guī)則是,如果我們將相對(duì)路徑傳遞給dlopen()
或 MSH,它將分別在和中查找“ .so
”和“ ” 。.mo
/lib/
/mo/
現(xiàn)在我們將卡插入 Arduino 板,在本例中為 MRKZERO ,上傳“HelloMo”草圖(草圖什么都不做),然后發(fā)出命令“ load
”。
為了顯示有關(guān)應(yīng)用程序執(zhí)行過(guò)程的更多詳細(xì)信息,我們可以在“ dlmodule.c
”中啟用調(diào)試消息:
#define LOG_LVL LOG_LVL_DBG
結(jié)果揭示了以下過(guò)程:
-
MSH 線程 ("tshell") 將 "
load.mo
" 加載到 RAM 并創(chuàng)建一個(gè)新線程 ("load") 以執(zhí)行入口點(diǎn)函數(shù) "load_hello()
" -
“
load_hello()
”然后加載“hello.so
”,調(diào)用它的“module_init()
”函數(shù),調(diào)用它的“say_hello()
”函數(shù)(不是入口點(diǎn)) -
在 "
say_hello()
" 返回后, "load_hello()
" 關(guān)閉 "hello.so
" (調(diào)用它的 "module_cleanup()
" 函數(shù)然后銷毀它的 RAM 副本) -
“加載”線程標(biāo)記以銷毀“”的RAM副本,
load.mo
然后退出 -
" " 的 RAM 副本
load.mo
最終被空閑線程 ("tidle0") 銷毀
" module_init()
" 和 " module_cleanup()
" 是特殊函數(shù)。如果定義,前者.mo
在將應(yīng)用程序加載到 RAM 后由 MSH 線程(在“”文件的情況下)調(diào)用,后者.mo
在銷毀 RAM 副本之前由空閑線程(在“”文件的情況下)調(diào)用。
讓我們將“hello_mo.c”重建為一個(gè)應(yīng)用程序(入口點(diǎn)是“ say_hello()
”,例如-Wl,-esay_hello
)并執(zhí)行。
結(jié)果清楚地表明“ module_init()
”被MSH線程(“tshell”)module_cleanup()
調(diào)用,“ ”被空閑線程(“tidle0”)調(diào)用。順便說(shuō)一句,傳遞給這兩個(gè)函數(shù)的參數(shù)是指向模塊描述符的指針。
優(yōu)點(diǎn)缺點(diǎn)
優(yōu)點(diǎn)
(我們感興趣的是)Arduino 應(yīng)用程序可以構(gòu)建一次并在許多板上運(yùn)行。根據(jù) Wiki ,“可用于 Cortex-M0 / Cortex-M0+ / Cortex-M1 的二進(jìn)制指令無(wú)需修改即可在 Cortex-M3 / Cortex-M4 / Cortex-M7 上執(zhí)行??捎糜?Cortex-M3 的二進(jìn)制指令無(wú)需修改即可執(zhí)行在 Cortex-M4 / Cortex-M7 / Cortex-M33 / Cortex-M35P 上進(jìn)行修改。”
因此,為 MKR Zero 板(SAMD 架構(gòu))構(gòu)建的應(yīng)用程序應(yīng)該在 Arduino Due(SAM 架構(gòu))上運(yùn)行而不會(huì)出現(xiàn)問(wèn)題。
此功能可以實(shí)現(xiàn)遠(yuǎn)程添加或更新功能而無(wú)需重新啟動(dòng)(與 OTA 固件更新相比)等。
缺點(diǎn)
與 MSH 命令相比,Arduino 應(yīng)用程序需要更多 RAM。另一個(gè)主要缺點(diǎn)是在固件方面,應(yīng)用程序所需的所有外部功能都必須在那里待機(jī)(盡管固件可能不會(huì)使用它們)并暴露(在“ mo_sym.h
”中)。
動(dòng)態(tài)模塊狀態(tài)
Arduino RT-Thread 庫(kù) v0.6.0 中的功能仍處于 beta 階段。
在原始代碼(RT-Thread 項(xiàng)目)中,除了DYN
ELF 文件的類型,動(dòng)態(tài)鏈接器還支持REL
類型。但是,經(jīng)過(guò)一些測(cè)試,我發(fā)現(xiàn)至少對(duì)于 ARM Cortex-M 架構(gòu),只有“ .o
”(對(duì)象)文件類型為REL
. 所以目前REL
Arduino RT-Thread 庫(kù)不支持 ELF 文件類型。
此外,僅測(cè)試了兩種重新分配類型:
-
R_ARM_JUMP_SLOT
-
R_ARM_RELATIVE
我需要一些代碼來(lái)測(cè)試其他類型。因此,如果您遇到其他類型的錯(cuò)誤,請(qǐng)幫助提出問(wèn)題。
最后,并非所有“l(fā)ibgcc”函數(shù)都默認(rèn)公開(kāi)。例如,開(kāi)關(guān)助手功能沒(méi)有公開(kāi)。您可以將它們添加到“ mo_sym.h
”或在您的應(yīng)用程序中將“ switch...case...
”替換為“ ” 。if...else...
RTT-QRCode 應(yīng)用程序
有一個(gè)更復(fù)雜的示例RTT-QRCode ,可以構(gòu)建為 MSH 命令或 Arduino 應(yīng)用程序。請(qǐng)查看代碼并玩得開(kāi)心!
下一步
- Arduino上的多任務(wù)處理
- 帶有 RT-Thread 的更好的 SD 庫(kù)
- RT-Thread Primer(即將推出)
- 帶有RT-Thread的更好的SD庫(kù)
- RT-Thread文檔_workqueue
- RT-Thread文檔_ringbuffer
- RT-Thread文檔_completion
- RT-Thread文檔_RT-Thread SMP 介紹與移植
- RT-Thread文檔_內(nèi)核基礎(chǔ)
- RT-Thread文檔_RT-Thread 潘多拉 STM32L475 上手指南
- RT-Thread文檔_RT-Thread 簡(jiǎn)介
- RT-Thread Smart 上手指南
- 【RT-Thread開(kāi)源作品秀】基于RT-Thread的星務(wù)平臺(tái)研究
- RT-Thread AI kit開(kāi)源:輕松實(shí)現(xiàn)一鍵部署AI模型至 RT-Thread
- 嵌入式RT-Thread應(yīng)用與開(kāi)發(fā) 71次下載
- 嵌入式實(shí)時(shí)操作系統(tǒng)RT-Thread的特點(diǎn)與體系結(jié)構(gòu)及移植方法詳細(xì)說(shuō)明 27次下載
- RT-Thread用戶手冊(cè) 0次下載
- RT-Thread編程指南 0次下載
- 基于 RT-Thread專業(yè)版的EtherCAT主站方案 429次閱讀
- RT-Thread qemu mps2-an385 bsp移植制作 :系統(tǒng)運(yùn)行篇 675次閱讀
- i.MX RT1170:VGLite移植RT-Thread Nano過(guò)程講解(下) 738次閱讀
- i.MX RT1170:VGLite移植RT-Thread Nano過(guò)程講解(上) 2094次閱讀
- rt-studio潘多拉開(kāi)發(fā)板最新rt-thread不能運(yùn)行解決辦法 1064次閱讀
- RT-Thread自動(dòng)初始化機(jī)制 2362次閱讀
- RT-Thread 4.1.0的CMake構(gòu)建教程 3129次閱讀
- 如何使用xmake工具來(lái)編譯rt-thread工程 2090次閱讀
- RT-Thread v4.1.0中FAL介紹 3523次閱讀
- 如何創(chuàng)建RT-Thread Nano工程 3572次閱讀
- 如何在rt-smart簡(jiǎn)化應(yīng)用程序開(kāi)發(fā) 1832次閱讀
- 簡(jiǎn)要分析Thread的通用GPIO設(shè)備驅(qū)動(dòng) 1413次閱讀
- RT-Thread NetUtils的使用方法 7848次閱讀
- 如何創(chuàng)建標(biāo)準(zhǔn)的RT-Thread項(xiàng)目工程?詳細(xì)過(guò)程分析概述 9102次閱讀
- RT-Thread軟件包定義和使用 1w次閱讀
下載排行
本周
- 1山景DSP芯片AP8248A2數(shù)據(jù)手冊(cè)
- 1.06 MB | 532次下載 | 免費(fèi)
- 2RK3399完整板原理圖(支持平板,盒子VR)
- 3.28 MB | 339次下載 | 免費(fèi)
- 3TC358743XBG評(píng)估板參考手冊(cè)
- 1.36 MB | 330次下載 | 免費(fèi)
- 4DFM軟件使用教程
- 0.84 MB | 295次下載 | 免費(fèi)
- 5元宇宙深度解析—未來(lái)的未來(lái)-風(fēng)口還是泡沫
- 6.40 MB | 227次下載 | 免費(fèi)
- 6迪文DGUS開(kāi)發(fā)指南
- 31.67 MB | 194次下載 | 免費(fèi)
- 7元宇宙底層硬件系列報(bào)告
- 13.42 MB | 182次下載 | 免費(fèi)
- 8FP5207XR-G1中文應(yīng)用手冊(cè)
- 1.09 MB | 178次下載 | 免費(fèi)
本月
- 1OrCAD10.5下載OrCAD10.5中文版軟件
- 0.00 MB | 234315次下載 | 免費(fèi)
- 2555集成電路應(yīng)用800例(新編版)
- 0.00 MB | 33566次下載 | 免費(fèi)
- 3接口電路圖大全
- 未知 | 30323次下載 | 免費(fèi)
- 4開(kāi)關(guān)電源設(shè)計(jì)實(shí)例指南
- 未知 | 21549次下載 | 免費(fèi)
- 5電氣工程師手冊(cè)免費(fèi)下載(新編第二版pdf電子書(shū))
- 0.00 MB | 15349次下載 | 免費(fèi)
- 6數(shù)字電路基礎(chǔ)pdf(下載)
- 未知 | 13750次下載 | 免費(fèi)
- 7電子制作實(shí)例集錦 下載
- 未知 | 8113次下載 | 免費(fèi)
- 8《LED驅(qū)動(dòng)電路設(shè)計(jì)》 溫德?tīng)栔?/a>
- 0.00 MB | 6656次下載 | 免費(fèi)
總榜
- 1matlab軟件下載入口
- 未知 | 935054次下載 | 免費(fèi)
- 2protel99se軟件下載(可英文版轉(zhuǎn)中文版)
- 78.1 MB | 537798次下載 | 免費(fèi)
- 3MATLAB 7.1 下載 (含軟件介紹)
- 未知 | 420027次下載 | 免費(fèi)
- 4OrCAD10.5下載OrCAD10.5中文版軟件
- 0.00 MB | 234315次下載 | 免費(fèi)
- 5Altium DXP2002下載入口
- 未知 | 233046次下載 | 免費(fèi)
- 6電路仿真軟件multisim 10.0免費(fèi)下載
- 340992 | 191187次下載 | 免費(fèi)
- 7十天學(xué)會(huì)AVR單片機(jī)與C語(yǔ)言視頻教程 下載
- 158M | 183279次下載 | 免費(fèi)
- 8proe5.0野火版下載(中文版免費(fèi)下載)
- 未知 | 138040次下載 | 免費(fèi)
評(píng)論
查看更多