RM新时代网站-首页

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

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

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

使用MicroLIB+fputc的方式實(shí)現(xiàn)串口打印功能

GReq_mcu168 ? 來源:玩轉(zhuǎn)單片機(jī) ? 2020-08-05 10:52 ? 次閱讀

常規(guī)打印方法

STM32的應(yīng)用中,我們常常對printf進(jìn)行重定向的方式來把打印信息printf到我們的串口助手。

在MDK環(huán)境中,我們常常使用MicroLIB+fputc的方式實(shí)現(xiàn)串口打印功能,即:

要實(shí)現(xiàn)fputc函數(shù)的原因是:printf函數(shù)依賴于fputc函數(shù),重新實(shí)現(xiàn)fputc內(nèi)部從串口發(fā)送數(shù)據(jù)即可間接地實(shí)現(xiàn)printf打印輸出數(shù)據(jù)到串口。

不知道大家有沒有看過正點(diǎn)原子裸機(jī)串口相關(guān)的例程,他們的串口例程里不使用MicroLIB,而是使用標(biāo)準(zhǔn)庫+fputc的方式。相關(guān)代碼如:

#if1 #pragmaimport(__use_no_semihosting) //標(biāo)準(zhǔn)庫需要的支持函數(shù) struct__FILE { inthandle; }; FILE__stdout; /** *@brief定義_sys_exit()以避免使用半主機(jī)模式 *@paramvoid *@returnvoid */ void_sys_exit(intx) { x=x; } intfputc(intch,FILE*f) { while((USART1->ISR&0X40)==0);//循環(huán)發(fā)送,直到發(fā)送完畢 USART1->TDR=(u8)ch; returnch; } #endif

關(guān)于這兩種方法的一些說明可以查看Mculover666兄的《重定向printf函數(shù)到串口輸出的多種方法》這篇文章。這篇文章中不僅包含上面的兩種方法,而且也包含著在GCC中使用標(biāo)準(zhǔn)庫重定向printf的方法。

自己實(shí)現(xiàn)一個打印函數(shù)

以上的幾種方法基本上是改造C庫的printf函數(shù)來實(shí)現(xiàn)串口打印的功能。其實(shí)我們也可以自己實(shí)現(xiàn)一個串口打印的功能。

printf本身就是一個變參函數(shù),其原型為:

intprintf(constchar*__format,...);

所以,我們要重新封裝的一個串口打印函數(shù)自然也應(yīng)該是一個變參函數(shù)。具體實(shí)現(xiàn)如下:

1、基于STM32的HAL庫

左右滑動查看全部代碼>>>

#defineTX_BUF_LEN256/*發(fā)送緩沖區(qū)容量,根據(jù)需要進(jìn)行調(diào)整*/ uint8_tTxBuf[TX_BUF_LEN];/*發(fā)送緩沖區(qū)*/ voidMyPrintf(constchar*__format,...) { va_listap; va_start(ap,__format); /*清空發(fā)送緩沖區(qū)*/ memset(TxBuf,0x0,TX_BUF_LEN); /*填充發(fā)送緩沖區(qū)*/ vsnprintf((char*)TxBuf,TX_BUF_LEN,(constchar*)__format,ap); va_end(ap); intlen=strlen((constchar*)TxBuf); /*往串口發(fā)送數(shù)據(jù)*/ HAL_UART_Transmit(&huart1,(uint8_t*)&TxBuf,len,0xFFFF); }

因為我們使用printf函數(shù)基本不使用其返回值,所以這里直接用void類型了。

自定義變參函數(shù)需要用到va_start、va_end等宏,需要包含頭文件stdarg.h。關(guān)于變參函數(shù)的一些學(xué)習(xí)可以查看網(wǎng)上的一些博文,如:

https://www.cnblogs.com/wulei0630/p/9444062.html

這里我們使用的是STM32的HAL庫,其給我們提供HAL_UART_Transmit接口可以直接把整個發(fā)送緩沖區(qū)的內(nèi)容給一次性發(fā)出去。

2、基于STM32標(biāo)準(zhǔn)庫

若是基于STM32的標(biāo)準(zhǔn)庫,就需要一字節(jié)一字節(jié)的循環(huán)發(fā)送出去,具體代碼如:

左右滑動查看全部代碼>>>

#defineTX_BUF_LEN256/*發(fā)送緩沖區(qū)容量,根據(jù)需要進(jìn)行調(diào)整*/ uint8_tTxBuf[TX_BUF_LEN];/*發(fā)送緩沖區(qū)*/ voidMyPrintf(constchar*__format,...) { va_listap; va_start(ap,__format); /*清空發(fā)送緩沖區(qū)*/ memset(TxBuf,0x0,TX_BUF_LEN); /*填充發(fā)送緩沖區(qū)*/ vsnprintf((char*)TxBuf,TX_BUF_LEN,(constchar*)__format,ap); va_end(ap); intlen=strlen((constchar*)TxBuf); /*往串口發(fā)送數(shù)據(jù)*/ for(inti=0;i

測試結(jié)果:

我們也可以使用我們的MyPrintf函數(shù)按照上一篇文章:《C語言、嵌入式中幾個非常實(shí)用的宏技巧》的方式封裝一個宏打印函數(shù):

以上就是我們自定義方式實(shí)現(xiàn)的一種串口打印函數(shù)。

但是,我想說:對于串口打印的使用,我們沒必要自己創(chuàng)建一個打印函數(shù)。

看到這,是不是有人想要打我了。。。??戳税胩?,你卻跟我說沒必要用。。。

哈哈,別急,我們不應(yīng)用在串口打印調(diào)試方面,那可以用在其它方面呀。

(1)應(yīng)用一:

比如最近我在實(shí)際應(yīng)用中:我們的MCU跑的是我們老大自己寫的一個小的操作系統(tǒng)+我們公司自己開發(fā)的上位機(jī)。

我們MCU端與上位機(jī)使用的是串口通訊,MCU往上位機(jī)發(fā)送的數(shù)據(jù)有兩種類型,一種是HEX格式數(shù)據(jù),一種是字符串?dāng)?shù)據(jù)。

但是我們下位機(jī)的這兩種數(shù)據(jù),在通過串口發(fā)送之前都得統(tǒng)一把數(shù)據(jù)封包交給那個系統(tǒng)通信任務(wù),然后再由通信任務(wù)發(fā)出去。

在這里,就不能用printf了。老大也針對他的這個系統(tǒng)實(shí)現(xiàn)了一個deb_printf函數(shù)用于打印調(diào)試。

但是,那個函數(shù)既復(fù)雜又很雞肋,稍微復(fù)雜一點(diǎn)的數(shù)據(jù)就打印不出來了。

因此我利用上面的思路給它新封裝了一個打印調(diào)試函數(shù),很好用,完美地兼容了老大的那個系統(tǒng)。具體代碼就不分享了,大體代碼、思路如上。

(2)應(yīng)用二:

我們在使用串口與ESP8266模塊通訊時,可利用類似這樣的方式封裝一個發(fā)送數(shù)據(jù)的函數(shù),這個函數(shù)的使用可以像printf一樣簡單。

可以以很簡單的方式把數(shù)據(jù)透傳至服務(wù)端,比如我以前的畢設(shè)中就有這么應(yīng)用:

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

    關(guān)注

    2270

    文章

    10895

    瀏覽量

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

    關(guān)注

    3

    文章

    4327

    瀏覽量

    62569
  • 串口打印
    +關(guān)注

    關(guān)注

    0

    文章

    10

    瀏覽量

    3092

原文標(biāo)題:串口打印知多少?

文章出處:【微信號:mcu168,微信公眾號:硬件攻城獅】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    【敏矽微ME32G070開發(fā)板免費(fèi)體驗】使用JLINK的RTT功能實(shí)現(xiàn)類似串口printf打印功能

    是壞的,非常影響我的測評使用。 我這個串口是壞的,非常影響我的測評使用。 我這個串口是壞的,非常影響我的測評使用。 為了先代替串口打印功能,
    發(fā)表于 12-15 20:30

    【RA-Eco-RA4E2-64PIN-V1.0開發(fā)板試用】串口打印功能

    剛剛點(diǎn)燈完畢,現(xiàn)在來實(shí)現(xiàn)RA4E2的串口打印功能,先打開原理圖,我們看到 串口使用的是P109和P110兩個IO口,然后我們來進(jìn)行配置 先打
    發(fā)表于 12-13 13:08

    打印針控制機(jī)構(gòu)實(shí)現(xiàn)打印針的什么和什么動作

    打印針控制機(jī)構(gòu)實(shí)現(xiàn)打印針的 出針 和 收針 動作。這是針式打印機(jī)打印過程中的關(guān)鍵環(huán)節(jié)。打印針控制
    的頭像 發(fā)表于 10-14 15:45 ?336次閱讀

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

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

    單獨(dú)打印UART_test時串口可以打印,添加以下代碼后串口打印,為什么?

    單獨(dú)打印UART_test時串口可以打印,添加以下代碼后串口打印。代碼看附件截圖
    發(fā)表于 06-27 06:21

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

    軟件環(huán)境:esp-idf v4.3 硬件環(huán)境:esp32c3 問題描述:因為項目里面需要使用到3個uart:2個其他功能規(guī)劃+1個日志打印調(diào)試;由于c3只有兩個串口資源,所以我們打算用一個io引腳來
    發(fā)表于 06-20 06:32

    串口屏的幾種安裝方式

    串口屏的幾種安裝方式
    的頭像 發(fā)表于 05-10 11:28 ?1550次閱讀

    有誰知道如何在熱敏打印機(jī)中實(shí)現(xiàn)圖片的灰階打印效果嗎?

    有誰知道如何在熱敏打印機(jī)中實(shí)現(xiàn)圖片的灰階打印效果。 現(xiàn)在基本上文字打印,圖片打印功能都已開發(fā)結(jié)束
    發(fā)表于 04-24 15:43

    串口屏的安裝方式方法

    串口屏的安裝方式方法
    的頭像 發(fā)表于 04-02 16:25 ?1466次閱讀

    如何添加microLib庫?cubeIDE是否支持添加microLib庫?

    如何添加microLib庫?cubeIDE是否支持添加microLib庫?
    發(fā)表于 03-08 08:21

    SPWM調(diào)制方式是怎樣實(shí)現(xiàn)變壓功能的?又是怎樣實(shí)現(xiàn)變頻功能的?

    SPWM調(diào)制方式是怎樣實(shí)現(xiàn)變壓功能的?又是怎樣實(shí)現(xiàn)變頻功能的? SPWM是一種常見的調(diào)制方式,它
    的頭像 發(fā)表于 02-06 11:09 ?1809次閱讀

    SPWM調(diào)制方式是怎樣實(shí)現(xiàn)變壓功能的?

    SPWM調(diào)制方式是怎樣實(shí)現(xiàn)變壓功能的? SPWM調(diào)制是一種常用的調(diào)制方式,用于將直流電壓轉(zhuǎn)換為交流電壓,同時也可以實(shí)現(xiàn)變壓
    的頭像 發(fā)表于 02-06 11:08 ?1261次閱讀

    TLE9854 printf函數(shù)無法輸出是怎么回事?

    TLE9854 的串口能正常輸出,現(xiàn)在想配置成printf,使能了STDOUT和STDIN,選擇了MicroLIB,可是沒法輸出,懷疑是程序里這個函數(shù)沒被調(diào)用,這個函數(shù)在哪里調(diào)用的? void
    發(fā)表于 02-02 16:05

    xmc7200使用cy_retarget_io_init_fc功能初始化串口6,為什么printf功能CAN不打印出數(shù)據(jù)?

    我使用的是xmc7200開發(fā)板,使用cy_retarget_io_init_fc功能初始化串口6,為什么printf功能 CAN 不打印出數(shù)據(jù)?
    發(fā)表于 01-30 06:55

    M487JIDAE如何使用ITM功能實(shí)現(xiàn)printf打印?

    M487JIDAE如何使用ITM功能實(shí)現(xiàn)printf打印?
    發(fā)表于 01-16 08:03
    RM新时代网站-首页