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

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

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

printf是如何與UART外設(shè)驅(qū)動函數(shù)“勾搭”起來的?

嵌入式悅翔園 ? 來源:痞子衡嵌入式 ? 作者:痞子衡 ? 2022-11-17 11:56 ? 次閱讀

大家好,我是痞子衡,是正經(jīng)搞技術(shù)的痞子。今天痞子衡給大家分享的是IAR下調(diào)試信息輸出機制之硬件UART外設(shè)。

嵌入式世界里,輸出打印信息是一種非常常用的輔助調(diào)試手段,借助打印信息,我們可以比較容易地定位和分析程序問題。在嵌入式應(yīng)用設(shè)計里實現(xiàn)打印信息輸出的方式有很多,本系列將以 IAR 環(huán)境為例逐一介紹 ARM Cortex-M 內(nèi)核 MCU 下打印信息輸出方法。

本篇是第一篇,我們先介紹最常見的輸出打印信息方式,即利用 MCU 芯片內(nèi)的硬件 UART 外設(shè)。本篇其實并不是要具體介紹 UART 外設(shè)模塊使用方法,而是重點分析 IAR 下是如何聯(lián)系 C 標(biāo)準(zhǔn)頭文件 stdio.h 定義的 printf() 函數(shù)與 UART 外設(shè)底層驅(qū)動函數(shù)的。

Note:本文使用的 IAR EWARM 軟件版本是 v9.10.2。

一、打印輸出整體框圖

首先簡介下本文介紹的打印輸出方法整體軟硬件框圖,硬件上主要是 PC 主機、MCU 目標(biāo)板、一根連接線(連接線有兩種方案:一種是 RS232 串口線、另一種是 TTL 串口轉(zhuǎn) USB 模塊板)。

軟件上 PC 這邊需要安裝一個串口調(diào)試助手軟件,然后目標(biāo)板 MCU 應(yīng)用程序需要包含打印輸出相關(guān)代碼,當(dāng) MCU 程序運行起來后,驅(qū)動片內(nèi) UART 外設(shè)實現(xiàn)打印字符數(shù)據(jù) (hello world.) 物理傳輸,在 PC 上串口調(diào)試助手軟件里可以看到打印信息。

fc53fb66-662a-11ed-8abf-dac502259ad0.png

上圖里的 MCU 應(yīng)用程序是在 IAR 環(huán)境下編譯鏈接的,因此我們的重點就是 stdio.h 頭文件里的 printf() 在 IAR 下到底是如何與 UART 外設(shè)驅(qū)動函數(shù)“勾搭”起來的。

二、C 標(biāo)準(zhǔn)頭文件 stdio.h

熟悉嵌入式工程的朋友應(yīng)該都知道 stdio.h 頭文件并不在用戶工程文件夾里,無需我們手動添加該文件進工程目錄,該文件是 C 標(biāo)準(zhǔn)定義的頭文件,由工具鏈自動提供。

stdio.h 是 C 語言為輸入輸出提供的標(biāo)準(zhǔn)庫頭文件,其前身是邁克·萊斯克 20 世紀(jì) 70 年代編寫的“可移植輸入輸出程序庫”。C 語言中的所有輸入和輸出都由抽象的字節(jié)流來完成,對文件的訪問也通過關(guān)聯(lián)的輸入或輸出流進行。

stdio.h 原型:https://cplusplus.com/reference/cstdio/

大部分人學(xué) C 語言一般都是在 Visual Studio / C++ 環(huán)境下,在這個環(huán)境里 stdio.h 定義的那些函數(shù)底層實現(xiàn)都由 Visual Studio 軟件直接搞定,我們通常無需關(guān)心其實現(xiàn)細節(jié)。

在嵌入式 IAR 環(huán)境下,這些標(biāo)準(zhǔn) C 定義的頭文件大部分也都是可以被支持的,我們可以在如下 IAR 軟件目錄找到它們,當(dāng)我們在工程代碼里加入 #include 等語句時,實際上就是添加 IAR 軟件目錄里的文件進工程編譯。

IAR SystemsEmbedded Workbench 9.10.2arminccstdio.h

但是 IAR 目錄下 stdio.h 文件里定義的諸如 printf() 函數(shù)具體實現(xiàn)我們是需要關(guān)注的,畢竟是要編譯鏈接生成具體機器碼下載進 MCU 運行的,但是 printf() 函數(shù)原型在哪呢?我們先留個懸念。

三、UART 外設(shè)驅(qū)動函數(shù)

說到 UART 外設(shè)驅(qū)動函數(shù),這個大家應(yīng)該再熟悉不過了。我們以恩智浦 i.MXRT1060 型號(ARM Cortex-M7 內(nèi)核)為例來具體介紹,在其官方 SDK 包里有相應(yīng)的 LPUART 驅(qū)動文件:

SDK_2.11.0_EVK-MIMXRT1060devicesMIMXRT1062driversfsl_lpuart.h
SDK_2.11.0_EVK-MIMXRT1060devicesMIMXRT1062driversfsl_lpuart.c

這個 LPUART 驅(qū)動庫里的 LPUART_WriteBlocking() 和 LPUART_ReadBlocking() 函數(shù)可以完成用戶數(shù)據(jù)包的發(fā)送和接收,其實單純利用 LPUART_WriteBlocking() 函數(shù)也可以實現(xiàn)打印信息輸出,只是沒有 printf() 函數(shù)那樣包含格式化輸出的強大功能。

status_tLPUART_Init(LPUART_Type*base,constlpuart_config_t*config,uint32_tsrcClock_Hz)
status_tLPUART_WriteBlocking(LPUART_Type*base,constuint8_t*data,size_tlength)
status_tLPUART_ReadBlocking(LPUART_Type*base,uint8_t*data,size_tlength)

四、IAR 對 C 標(biāo)準(zhǔn) I/O 庫的支持

IAR 顯然是對 C 標(biāo)準(zhǔn) I/O 庫有支持的,不然我們不可能在工程里能使用 printf() 函數(shù),只是這個支持我們?nèi)绾稳ポp松發(fā)現(xiàn)呢?痞子衡今天教大家一個方法,就是看工程編譯鏈接后生成的 .map 文件,這個 map 文件里會列出工程里所有函數(shù)的來源。

4.1 引出底層接口 __write()

我們以 SDK_2.11.0_EVK-MIMXRT1060oardsevkmimxrt1060demo_appshello_worldiar 工程為例來介紹,需要簡單改造一下工程里 hello_world.c 文件里的 main() 函數(shù),將原來代碼全部刪掉(原來的打印輸出涉及恩智浦 SDK 封裝,本文沒必要關(guān)心其實現(xiàn)),只要如下一句打印即可:

#include
intmain(void)
{
printf("helloworld.
");
while(1);
}

然后注意工程選項里跟 Library 實現(xiàn)相關(guān)的如下三處設(shè)置。其中 Library 選項配置的是 runtime lib 的功能,有 Normal 和 Full 兩個選項(可按需選擇);Printf formatter 選項決定格式化輸出功能細節(jié),分 Full、Large、Small、Tiny 四個選項(可按需選擇)。Library low-level interface implementation 選項決定低層 I/O 實現(xiàn),這里我們選 None,即由用戶來實現(xiàn)。

fc80f8f0-662a-11ed-8abf-dac502259ad0.png

配置好 Library 后編譯工程會發(fā)現(xiàn)有如下報錯,根據(jù)這個報錯我們可以猜到 dl7M_tln.a 是 IAR 編譯好的 C/C++ 庫,庫里面實現(xiàn)了 printf() 函數(shù)及其所依賴的 putchar() 函數(shù),而 puchar() 函數(shù)對外提供了底層 I/O 接口函數(shù),這個 I/O 函數(shù)名字叫 __write(),它就是需要用戶結(jié)合芯片 UART 外設(shè)去實現(xiàn)的發(fā)送函數(shù)。

 Error[Li005]: no definition for "__write" [referenced from putchar.o(dl7M_tln.a)]

在 IAR 目錄下我們可以找到 dl7M_tln.a 文件路徑,經(jīng)過測試,工程 Library 設(shè)置里 Normal 和 Full 選項其實就是選 dl7M_tln.a 還是 dl7M_tlf.a 進用戶工程去鏈接。

fc98ef5a-662a-11ed-8abf-dac502259ad0.png

4.2 DLIB底層 I/O 接口設(shè)計

找到了 __write() 函數(shù),但是它的原型到底是什么?我們該如何實現(xiàn)它?這時候需要去查萬能的 IAR SystemsEmbedded Workbench 9.10.2armdocEWARM_DevelopmentGuide.ENU 手冊,在里面搜索 __write 字樣可以找到如下設(shè)計,原來我們在代碼里調(diào)用的 C 標(biāo)準(zhǔn) I/O 接口均是由 IAR 底層預(yù)編譯好的 DLIB 去具體實現(xiàn)的,這個 DLIB 庫也留下了給用戶實現(xiàn)的最底層與硬件相關(guān)的接口函數(shù)。

fcbaa050-662a-11ed-8abf-dac502259ad0.png

IAR 為 DLIB 里那些最底層的 I/O 接口函數(shù)都創(chuàng)建了模板源文件,在這些模板文件里我們可以找到它們的原型,所以我們在 write.c 文件里找到了 __write() 原型及其示例實現(xiàn)。

size_t__write(inthandle,constunsignedchar*buffer,size_tsize)
fccf118e-662a-11ed-8abf-dac502259ad0.png

4.3 DLIB庫 I/O 相關(guān)源碼實現(xiàn)

有了 __write() 原型及示例代碼,我們很容易便能用 LPUART_WriteBlocking() 函數(shù)去實現(xiàn)它,將這個代碼添加進 hello_world 工程編譯,這時候就不會報錯了(當(dāng)然要想真正在板子上測試打印功能,main 函數(shù)里還得加入 LPUART 初始化代碼)。

#include"fsl_lpuart.h"
size_t__write(inthandle,constunsignedchar*buf,size_tsize)
{
//假設(shè)使用LPUART1去做輸出
(void)LPUART_WriteBlocking(LPUART1,buf,size);

return0;
}

工程編譯完成后,查看生成的 hello_world.map 文件,找到 dl7M_tln.a 部分的信息,可以看到其由很多個 .o 文件組成(功能比較豐富),這些 .o 文件都是可以在 IAR 安裝目錄下找到其源碼的。

*******************************************************************************
*** MODULE SUMMARY
***

    Module                ro code  ro data  rw data
    ------                -------  -------  -------
dl7M_tln.a: [10]
    abort.o                     6
    exit.o                      4
    low_level_init.o            4
    printf.o                   40
    putchar.o                  32
    xfail_s.o                  64                 4
    xprintfsmall_nomb.o     1'281
    xprout.o                   22
    -----------------------------------------------
    Total:                  1'453                 4

DLIB 庫中關(guān)于 I/O 相關(guān)的源碼放在了如下目錄里,感興趣的可以去查看其具體實現(xiàn),這里特別提一下 formatter 文件夾下 xprintf 有很多種不同的源文件實現(xiàn),其實就對應(yīng)了工程選項 Printf formatter 里的不同配置。

IAR SystemsEmbedded Workbench 9.10.2armsrclibdlibfile
IAR SystemsEmbedded Workbench 9.10.2armsrclibdlibformatters
fcecdb38-662a-11ed-8abf-dac502259ad0.png

至此,IAR下調(diào)試信息輸出機制之硬件UART外設(shè)痞子衡便介紹完畢了,掌聲在哪里~~~

審核編輯:湯梓紅

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

    關(guān)注

    146

    文章

    16802

    瀏覽量

    349375
  • uart
    +關(guān)注

    關(guān)注

    22

    文章

    1214

    瀏覽量

    101004
  • Printf
    +關(guān)注

    關(guān)注

    0

    文章

    81

    瀏覽量

    13588

原文標(biāo)題:printf是如何與UART外設(shè)驅(qū)動函數(shù)“勾搭”起來的?

文章出處:【微信號:嵌入式悅翔園,微信公眾號:嵌入式悅翔園】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    怎么讓printf函數(shù)在GB110上與UART 4一起工作

    我試圖讓printf函數(shù)在GB110上與UART 4一起工作。我試圖創(chuàng)建一個本地putch函數(shù),它應(yīng)該覆蓋其他函數(shù),但是它不能工作。所以,在
    發(fā)表于 09-05 09:36

    怎樣使用rttthread studio寫uart外設(shè)驅(qū)動

    本次練習(xí)是使用rttthread studio寫uart外設(shè)驅(qū)動,f103的sdk版本是0.1.9(在studio的包管理器那查看),導(dǎo)入我的例程需參考如何導(dǎo)入工程,祝君順利。圖中的控制臺就是調(diào)試
    發(fā)表于 05-19 15:50

    使用printf()函數(shù)時默認UART0輸出,請教如何重定向printf()到UART1?

    芯片:N76E003; 編譯環(huán)境:keil5 使用printf()函數(shù)時默認UART0輸出,請教如何重定向printf()到UART1?
    發(fā)表于 06-25 07:12

    printf函數(shù)用法示例

    《OpenCV3編程入門》書本配套源代碼:printf函數(shù)用法示例
    發(fā)表于 06-06 15:20 ?10次下載

    實現(xiàn)重定向printf()和scanf() 函數(shù)案例分析

    要想printf()和scanf() 函數(shù)工作,我們需要把printf()和scanf() 重新定向到串口中。重定向是指用戶可以自己重寫C 的庫函數(shù),當(dāng)連接器檢查到用戶編寫了與C 庫
    發(fā)表于 06-23 08:26 ?9050次閱讀
    實現(xiàn)重定向<b class='flag-5'>printf</b>()和scanf() <b class='flag-5'>函數(shù)</b>案例分析

    如何使用單片機系統(tǒng)重寫printf函數(shù)

    本文檔的主要內(nèi)容詳細介紹的是如何使用單片機系統(tǒng)重寫printf函數(shù)
    發(fā)表于 07-23 17:37 ?1次下載
    如何使用單片機系統(tǒng)重寫<b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>

    基于STM32的多種printf用法

    在調(diào)試代碼的時候,最常用的就是使用printf函數(shù)來輸出一些打印信息,提示自己代碼的執(zhí)行情況。 如果你的UART串口不夠用,還要用printf,此時該怎么辦? ? ? ? ?解決方法:
    的頭像 發(fā)表于 07-23 11:12 ?4747次閱讀

    《51單片機筆記》keilC51軟件中printf函數(shù)內(nèi)部機制詳解,單片機中怎么使用printf函數(shù),printf函數(shù)編寫程序例子及

    printf函數(shù)的講解
    發(fā)表于 11-20 16:51 ?11次下載
    《51單片機筆記》keilC51軟件中<b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>內(nèi)部機制詳解,單片機中怎么使用<b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>,<b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>編寫程序例子及

    華大HC32L110 printf重映射UART

    華大HC32L110 printf重映射UART在使用printf時官方工程默認使用的端口是UART0。找到dll.c文件 fputc函數(shù)
    發(fā)表于 11-23 18:06 ?9次下載
    華大HC32L110 <b class='flag-5'>printf</b>重映射<b class='flag-5'>UART</b>

    【CC2530授課筆記】⑨ UART串口通信 printf

    摘要此篇文章介紹了CC2530寄存器的描述,通過具體的示例,實現(xiàn)了串口通信,115200波特率,并重寫Putchar函數(shù),實現(xiàn)了printf功能。printf 函數(shù)的實現(xiàn)要實現(xiàn)
    發(fā)表于 11-30 09:51 ?11次下載
    【CC2530授課筆記】⑨ <b class='flag-5'>UART</b>串口通信 <b class='flag-5'>printf</b>

    STM32F103串口1 printf函數(shù)的實現(xiàn)

    怎么使用呢?能不能將這個函數(shù)和串口1對應(yīng)起來,當(dāng)然是有方法的。??下面就通過代碼來演示一下如何在串口1上使用printf()函數(shù)的功能。void u
    發(fā)表于 12-20 19:37 ?1次下載
    STM32F103串口1 <b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>的實現(xiàn)

    通過串口利用printf函數(shù)輸出數(shù)據(jù)

    一。printf函數(shù)格式printf函數(shù)具有強大的輸出功能%表示格式化字符串輸出目前printf支持以下格式的輸出,例如:
    發(fā)表于 12-28 19:11 ?11次下載
    通過串口利用<b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>輸出數(shù)據(jù)

    使用printf函數(shù)的安全隱患

    程序員都知道,也都會使用printf函數(shù),但你知道它也有“安全隱患”嗎?
    的頭像 發(fā)表于 10-09 09:49 ?1854次閱讀

    如何實現(xiàn)Printf()接口重定向到UART

    車規(guī)級MCU開發(fā),不像PC端,包含stdio.h頭文件就可以使用Printf()函數(shù)。為了實現(xiàn)PC端Printf()接口功能,需要在MCU端實現(xiàn)Printf()接口的"重定向"
    的頭像 發(fā)表于 10-01 10:13 ?1171次閱讀
    如何實現(xiàn)<b class='flag-5'>Printf</b>()接口重定向到<b class='flag-5'>UART</b>

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

    printf函數(shù)作為標(biāo)準(zhǔn)庫定義的格式化輸出方式,本文將介紹其在AWorksLP下默認適配以及重映射至熱拔插設(shè)備端口的實現(xiàn)。默認適配AWorksLP中默認已經(jīng)對printf函數(shù)完成相關(guān)適
    的頭像 發(fā)表于 11-25 08:24 ?542次閱讀
    AWorksLP應(yīng)用筆記:重定向<b class='flag-5'>printf</b><b class='flag-5'>函數(shù)</b>