0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

AWorksLP應(yīng)用筆記:重定向printf函數(shù)

ZLG致遠(yuǎn)電子 ? 2023-11-25 08:24 ? 次閱讀

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)行輸出,具體配置如下圖所示。

0745adc0-8b29-11ee-9788-92fbcf53809c.png

注:若用戶未使能 stdio function 時(shí),調(diào)用printf函數(shù)時(shí),將不會(huì)有任何輸出。

本文將使用 EPC6450-AWI 平臺(tái),選擇標(biāo)有絲印為DUART的調(diào)試串口(UART0設(shè)備)進(jìn)行printf功能演示測試。將TTL轉(zhuǎn)USB串口模塊的TXD與板子的RXD絲印連接,RXD與板子的TXD絲印,將另一端的USB口接入電腦07591f5e-8b29-11ee-9788-92fbcf53809c.png啟動(dòng)串口調(diào)試助手,搜索并打開串口模塊的設(shè)備端口號(hào)后,在工程中調(diào)用printf函數(shù),根據(jù)下圖可知,printf函數(shù)適配UART0設(shè)備成功。076d89e4-8b29-11ee-9788-92fbcf53809c.png

重定向至其他設(shè)備嵌入式的諸多應(yīng)用在UART設(shè)備資源受限的情況下,可能存在將printf函數(shù)重定向到其他設(shè)備需求。為此,筆者將以EPC6450-AWI平臺(tái)的USB串口設(shè)備為例進(jìn)行說明。077d1c56-8b29-11ee-9788-92fbcf53809c.png1.實(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ù)的適配,具體如下圖所示。

08005670-8b29-11ee-9788-92fbcf53809c.png1.2 檢測動(dòng)態(tài)設(shè)備

USB設(shè)備為動(dòng)態(tài)設(shè)備,因此需要持續(xù)檢測設(shè)備的是否存在情況??赏ㄟ^初始化一個(gè)動(dòng)態(tài)設(shè)備檢測任務(wù),對(duì)設(shè)備的是否存在情況進(jìn)行周期性檢測。

while true: access (device) delay()

1.3 關(guān)聯(lián)標(biāo)準(zhǔn)文件流

在檢測到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 清理文件描述符

檢測到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è)備。

0812e6e6-8b29-11ee-9788-92fbcf53809c.png

3.簡單示例

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;
// 檢測動(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è)置檢測周期 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è)備。0825bf28-8b29-11ee-9788-92fbcf53809c.png

0831239a-8b29-11ee-9788-92fbcf53809c.png


07799d06-8b29-11ee-9788-92fbcf53809c.jpg 總結(jié)實(shí)現(xiàn)重定向printf函數(shù)時(shí)主要關(guān)注以下兩個(gè)關(guān)鍵點(diǎn):

  1. 重寫NEWLIB標(biāo)準(zhǔn)庫中printf函數(shù)的底層實(shí)現(xiàn);
  2. 將指定設(shè)備以標(biāo)準(zhǔn)文件流約定的文件描述符打開。


07799d06-8b29-11ee-9788-92fbcf53809c.jpg 擴(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

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 設(shè)備
    +關(guān)注

    關(guān)注

    2

    文章

    4413

    瀏覽量

    70442
  • uart
    +關(guān)注

    關(guān)注

    22

    文章

    1214

    瀏覽量

    100995
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4256

    瀏覽量

    62223
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    飛凌嵌入式ElfBoard ELF 1板卡-shell腳本編寫之輸入輸出重定向

    shell輸出重定向通常是指將執(zhí)行命令的輸出信息從默認(rèn)的標(biāo)準(zhǔn)輸出(即當(dāng)前終端)重新定向到指定文件中。輸入重定向通常是指將命令所需的輸入數(shù)據(jù)的來源,從標(biāo)準(zhǔn)輸入(即當(dāng)前終端)更改為從指定文件中獲取。輸出
    發(fā)表于 09-23 10:23

    重定向了fputc及putchar函數(shù),但printf沒有輸出,為什么?

    重定向了fputc及putchar函數(shù),但printf沒有輸出 刪除了drivers/drv_uart.c drv_uart.h 刪除了文件rt-thread\\components
    發(fā)表于 07-18 07:44

    esp32c3能不能將串口打印函數(shù)重定向到自己軟件模擬的uart上?

    模擬uart,用于打印日志;請(qǐng)問這種做法可以實(shí)現(xiàn)嗎?可以將打印函數(shù)重定向到該模擬uart嗎?如果可以,麻煩指導(dǎo)一下重定向需要修改哪幾個(gè)函數(shù)? ps:打印的
    發(fā)表于 06-20 06:32

    請(qǐng)問如何將printf使用的stdout重定向到asclin?

    /v4.9.3.0-infineon-1.0/docs/userguide.pdf 第 406 頁中關(guān)于 printf 的唯一說明是 stdout 重定向到模擬 io 調(diào)試器。 當(dāng)然,您可以使用 Ifx_Console_print,但我想知道是否可以通過標(biāo)準(zhǔn)庫來實(shí)現(xiàn)。
    發(fā)表于 06-03 06:48

    芯海通用 MCU 應(yīng)用筆記:在 MDK 開發(fā)環(huán)境下代碼重定向到 RAM 執(zhí)行的幾種方法

    為 V5.37.0.0。本文檔介紹方法適用于芯??萍?MCU。*附件:應(yīng)用筆記:在MDK開發(fā)環(huán)境下代碼重定向到RAM執(zhí)行的幾種方法.pdf
    發(fā)表于 05-16 11:58

    芯海通用 MCU應(yīng)用筆記 :在 IAR 及 MDK 開發(fā)環(huán)境下使用 printf 函數(shù)重定向移植差異指南

    對(duì) printf 函數(shù)支持的差異。并在本應(yīng)用筆記結(jié)尾處給出可以一鍵移植到 Keil、IAR8.x、IAR9.x 版本下的通用重定向代碼。 Keil 和 IAR 都是常用的開發(fā)工具 I
    發(fā)表于 05-16 11:56

    求助,關(guān)于HAL下的printf重定向輸出疑問求解

    (1)網(wǎng)上查了使用printf重定向,串口輸出,函數(shù)如下,使用HAL_UART_Transmit函數(shù)。 int fputc(int ch,FILE *f) { uint8_t tem
    發(fā)表于 05-10 06:04

    STM8S如何在STVD環(huán)境下重定向printf函數(shù)實(shí)現(xiàn)UART簡化輸出?

    printf(),網(wǎng)上IAR環(huán)境的重定向調(diào)通了,求STVD環(huán)境下的code。 3. 用STM8的官方例程UART1_Printf在STVD環(huán)境下編譯OK,但是debug時(shí)候報(bào)錯(cuò)下載不了程序 4. 針對(duì)3
    發(fā)表于 04-30 08:29

    STM8L051片子使用重定向printf函數(shù)時(shí)總是報(bào)錯(cuò),沒辦法引用printf函數(shù)是什么原因?

    STM8L051片子使用重定向printf函數(shù)時(shí)總是報(bào)錯(cuò),沒辦法引用printf函數(shù),不知道是什么原因,使用的是IAR編譯器,總是報(bào)內(nèi)存不足
    發(fā)表于 04-28 08:05

    用keil仿真無法用printf打印怎么解決?

    用keil仿真 無法用printf打印.想用printf重定向到Usart1的方法,使printf打印信息能夠在UART#1的窗口打印出來. 硬件連接用的是SW(沒有連SWO引腳,只
    發(fā)表于 04-11 08:20

    STM32CubeIDE中打印重定向報(bào)錯(cuò)怎么解決?

    HAL_UART_Transmit( huart1 , (uint8_t *) ch, 1, 0xFFFF);報(bào)錯(cuò) printf重定向 報(bào)錯(cuò)../Core/Src/main.c:42:21: error: \'huart1\' undeclared 已添加stdio.h頭
    發(fā)表于 04-03 07:33

    H7平臺(tái)如何重定向sqrtf函數(shù)到RAM中運(yùn)行?

    如題,H7平臺(tái)如何重定向sqrtf函數(shù)到RAM中運(yùn)行,這個(gè)函數(shù)是庫函數(shù),不能使用__ramfunc前綴,有什么方法或是例程,網(wǎng)上找了很多icf配置的方法,都不行,要不就是只放置到RAM
    發(fā)表于 03-27 06:40

    將動(dòng)態(tài)指示段重定向到單獨(dú)的LED

    有時(shí),除了數(shù)字指示器之外,還需要單獨(dú)的 LED。它們可以連接到微控制器的單獨(dú)引腳,但您也可以節(jié)省引腳。如果數(shù)字指示器具有從未使用過的段(例如,右邊數(shù)字中的一個(gè)點(diǎn)),則可以將它們重定向到外部 LED。
    發(fā)表于 02-02 16:49 ?480次閱讀
    將動(dòng)態(tài)指示段<b class='flag-5'>重定向</b>到單獨(dú)的LED

    2分鐘搞懂輸出重定向

    視頻最后我們通過重定向把標(biāo)準(zhǔn)輸出寫到了文件中,但是錯(cuò)誤輸出還是留在了屏幕上。
    的頭像 發(fā)表于 01-15 16:41 ?490次閱讀

    C語言printf函數(shù)族學(xué)習(xí)筆記

    printf 應(yīng)該是學(xué)習(xí)C語言的第一個(gè)函數(shù),我們都知道它是用于字符串格式化輸出的。但是它的用法你真正了解了嗎?
    的頭像 發(fā)表于 11-24 11:15 ?834次閱讀
    C語言<b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>族學(xué)習(xí)<b class='flag-5'>筆記</b>