printf函數(shù)作為標(biāo)準(zhǔn)庫定義的格式化輸出方式,本文將介紹其在AWorksLP下默認(rèn)適配以及重映射至熱拔插設(shè)備端口的實(shí)現(xiàn)。
默認(rèn)適配
AWorksLP中默認(rèn)已經(jīng)對(duì)printf函數(shù)完成相關(guān)適配工作,且默認(rèn)被適配在UART設(shè)備。用戶可以在圖形化配置界面中使能 support the stdio functions ,并選擇期望UART設(shè)備進(jìn)行輸出,具體配置如下圖所示。
注:若用戶未使能 stdio function 時(shí),調(diào)用printf函數(shù)時(shí),將不會(huì)有任何輸出。
本文將使用 EPC6450-AWI 平臺(tái),選擇標(biāo)有絲印為DUART的調(diào)試串口(UART0設(shè)備)進(jìn)行printf功能演示測(cè)試。將TTL轉(zhuǎn)USB串口模塊的TXD與板子的RXD絲印連接,RXD與板子的TXD絲印,將另一端的USB口接入電腦。啟動(dòng)串口調(diào)試助手,搜索并打開串口模塊的設(shè)備端口號(hào)后,在工程中調(diào)用printf函數(shù),根據(jù)下圖可知,printf函數(shù)適配UART0設(shè)備成功。
重定向至其他設(shè)備嵌入式的諸多應(yīng)用在UART設(shè)備資源受限的情況下,可能存在將printf函數(shù)重定向到其他設(shè)備需求。為此,筆者將以EPC6450-AWI平臺(tái)的USB串口設(shè)備為例進(jìn)行說明。1.實(shí)施步驟
與UART設(shè)備不同,USB設(shè)備為動(dòng)態(tài)設(shè)備,因此重定向printf函數(shù)時(shí),需要注意以下幾個(gè)關(guān)鍵步驟:
1.1 支持NEWLIB標(biāo)準(zhǔn)庫函數(shù)
由于AWorksLP中利用posix file相關(guān)操作接口對(duì)printf函數(shù)進(jìn)行適配,故在重映射端口時(shí),需將 support libc file operations 使能,并取消默認(rèn)選擇UART設(shè)備作為printf函數(shù)的適配,具體如下圖所示。
1.2 檢測(cè)動(dòng)態(tài)設(shè)備
USB設(shè)備為動(dòng)態(tài)設(shè)備,因此需要持續(xù)檢測(cè)設(shè)備的是否存在情況??赏ㄟ^初始化一個(gè)動(dòng)態(tài)設(shè)備檢測(cè)任務(wù),對(duì)設(shè)備的是否存在情況進(jìn)行周期性檢測(cè)。
while true: access (device) delay()
1.3 關(guān)聯(lián)標(biāo)準(zhǔn)文件流
在檢測(cè)到USB設(shè)備存在時(shí),僅需將設(shè)備與標(biāo)準(zhǔn)文件流(stdio中的stdin、stdout、stderr,且在C庫中被假定為交互設(shè)備,并約定了這些設(shè)備的文件描述符依次為0、1、2)關(guān)聯(lián)起來。故在使用時(shí),我們僅需將描述符0、1、2與USB串口設(shè)備即可,其偽代碼如下所示。
while true: if access (device): 0 = open (device) duplicate 1 to 0 duplicate 2 to 0 delay()
1.4 清理文件描述符
檢測(cè)到USB設(shè)備不存在時(shí),需及時(shí)取消設(shè)備與標(biāo)準(zhǔn)文件流的關(guān)聯(lián)。即根據(jù)設(shè)備的打開情況,對(duì)文件描述符進(jìn)行清理,以便之后重新關(guān)聯(lián)標(biāo)準(zhǔn)文件流。
while true: if access (device): 0 = open (device) duplicate 1 to 0 duplicate 2 to 0 else: close (device) delay()
2. 基礎(chǔ)配置在EPC6450-AWI平臺(tái)標(biāo)有絲印為Type-C的接口處,插上Type-C線,將Type-C線的另一端USB口連接電腦。并在圖形化配置界面,將USB設(shè)備選擇為CDC串口設(shè)備。
3.簡(jiǎn)單示例
static int __dynamic_stdin_fd = -1;static aw_err_t __dynamic_stdout_ret = -AW_EBADF;static aw_err_t __dynamic_stderr_ret = -AW_EBADF;
aw_err_t aw_printf_redirect_dynamic_dev(void){ int find = -AW_ENODEV;
// 檢測(cè)動(dòng)態(tài)設(shè)備 find = aw_access(AW_DYNAMIC_DEV_PATH, AW_F_OK);
if(find == AW_OK) { // 關(guān)聯(lián)標(biāo)準(zhǔn)文件流 if(__dynamic_stdin_fd < 0) ? ? ? ?{ ? ? ? ? ? ?__dynamic_stdin_fd = \ ? ? ? ? ? ? aw_open_at(AW_DYNAMIC_DEV_PATH,AW_O_RDWR,0,0); ? ? ? ? ? ?__dynamic_stdout_ret = aw_dup2(0, 1); ? ? ? ? ? ?__dynamic_stderr_ret = aw_dup2(0, 2); ? ? ? ? ? ?return AW_OK; ? ? ? ?} ? ?} ? ?else { ? ? ? ?// 清理文件描述符 ? ? ? ?if(__dynamic_stdin_fd >= 0) { aw_close(0); __dynamic_stdin_fd = -1; } if (__dynamic_stdout_ret == AW_OK) { aw_close(1); __dynamic_stdout_ret = -AW_EBADF; } if (__dynamic_stderr_ret == AW_OK) { aw_close(2); __dynamic_stderr_ret = -AW_EBADF; } }
return -AW_ENODEV;}
int aw_main(void){ int ret;
aw_kprintf("hello world\n"); printf("hello world\n");
while(1) { ret = aw_printf_redirect_dynamic_dev(); if (AW_OK == ret) break;
// 設(shè)置檢測(cè)周期 AW_TASK_DELAY(100); }
aw_kprintf("hello world, ZLG\n"); printf("hello world, ZLG\n");
return 0;}
啟動(dòng)串口調(diào)試助手,搜索并打開DEBUG UART設(shè)備與CDC串口設(shè)備的端口號(hào)后,運(yùn)行上文示例程序。根據(jù)下圖可知,USB設(shè)備枚舉后,printf函數(shù)成功重定向到了CDC串口設(shè)備。
總結(jié)實(shí)現(xiàn)重定向printf函數(shù)時(shí)主要關(guān)注以下兩個(gè)關(guān)鍵點(diǎn):
- 重寫NEWLIB標(biāo)準(zhǔn)庫中printf函數(shù)的底層實(shí)現(xiàn);
- 將指定設(shè)備以標(biāo)準(zhǔn)文件流約定的文件描述符打開。
擴(kuò)展閱讀
本文所演示平臺(tái)使用的是GCC編譯器,其對(duì)應(yīng)C庫為NEWLIB標(biāo)準(zhǔn)庫。在AWorksLP中printf函數(shù)的底層輸出接口在AWorksLP中實(shí)現(xiàn)為_write_r 函數(shù),其具體代碼實(shí)現(xiàn)如下所示。
__attribute__((__used__)) _ssize_t_write_r(struct _reent *ptr, int fd, const void *buf, size_t nbytes){ return aw_write(fd,buf,nbytes);}
需要值得注意的是,上述適配方式僅兼容NEWLIB,若是其他編譯器,其實(shí)現(xiàn)以及接口不盡相同,下表僅給出部分以供參考,在使用時(shí)需根據(jù)實(shí)際情況進(jìn)行調(diào)整。
工具鏈 | 標(biāo)準(zhǔn)庫 | 底層接口 |
GCC | NEWLIB | _write_r |
ARMCC | ARMCLIB | _sys_write |
ARMCLANG | ARMCLIB | _sys_write |
-
設(shè)備
+關(guān)注
關(guān)注
2文章
4554瀏覽量
70918 -
uart
+關(guān)注
關(guān)注
22文章
1245瀏覽量
101829 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4349瀏覽量
63027
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論