0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看威廉希尔官方网站 视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

printf是如何与UART外设驱动函数“勾搭”起来的?

嵌入式悦翔园 来源:痞子衡嵌入式 作者:痞子衡 2022-11-17 11:56 次阅读

大家好,我是痞子衡,是正经搞威廉希尔官方网站 的痞子。今天痞子衡给大家分享的是IAR下调试信息输出机制之硬件UART外设

嵌入式世界里,输出打印信息是一种非常常用的辅助调试手段,借助打印信息,我们可以比较容易地定位和分析程序问题。在嵌入式应用设计里实现打印信息输出的方式有很多,本系列将以 IAR 环境为例逐一介绍 ARM Cortex-M 内核 MCU 下打印信息输出方法。

本篇是第一篇,我们先介绍最常见的输出打印信息方式,即利用 MCU 芯片内的硬件 UART 外设。本篇其实并不是要具体介绍 UART 外设模块使用方法,而是重点分析 IAR 下是如何联系 C 标准头文件 stdio.h 定义的 printf() 函数与 UART 外设底层驱动函数的。

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

一、打印输出整体框图

首先简介下本文介绍的打印输出方法整体软硬件框图,硬件上主要是 PC 主机、MCU 目标板、一根连接线(连接线有两种方案:一种是 RS232 串口线、另一种是 TTL 串口转 USB 模块板)。

软件上 PC 这边需要安装一个串口调试助手软件,然后目标板 MCU 应用程序需要包含打印输出相关代码,当 MCU 程序运行起来后,驱动片内 UART 外设实现打印字符数据 (hello world.) 物理传输,在 PC 上串口调试助手软件里可以看到打印信息。

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

上图里的 MCU 应用程序是在 IAR 环境下编译链接的,因此我们的重点就是 stdio.h 头文件里的 printf() 在 IAR 下到底是如何与 UART 外设驱动函数“勾搭”起来的。

二、C 标准头文件 stdio.h

熟悉嵌入式工程的朋友应该都知道 stdio.h 头文件并不在用户工程文件夹里,无需我们手动添加该文件进工程目录,该文件是 C 标准定义的头文件,由工具链自动提供。

stdio.h 是 C 语言为输入输出提供的标准库头文件,其前身是迈克·莱斯克 20 世纪 70 年代编写的“可移植输入输出程序库”。C 语言中的所有输入和输出都由抽象的字节流来完成,对文件的访问也通过关联的输入或输出流进行。

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

大部分人学 C 语言一般都是在 Visual Studio / C++ 环境下,在这个环境里 stdio.h 定义的那些函数底层实现都由 Visual Studio 软件直接搞定,我们通常无需关心其实现细节。

在嵌入式 IAR 环境下,这些标准 C 定义的头文件大部分也都是可以被支持的,我们可以在如下 IAR 软件目录找到它们,当我们在工程代码里加入 #include 等语句时,实际上就是添加 IAR 软件目录里的文件进工程编译。

IAR SystemsEmbedded Workbench 9.10.2arminccstdio.h

但是 IAR 目录下 stdio.h 文件里定义的诸如 printf() 函数具体实现我们是需要关注的,毕竟是要编译链接生成具体机器码下载进 MCU 运行的,但是 printf() 函数原型在哪呢?我们先留个悬念。

三、UART 外设驱动函数

说到 UART 外设驱动函数,这个大家应该再熟悉不过了。我们以恩智浦 i.MXRT1060 型号(ARM Cortex-M7 内核)为例来具体介绍,在其官方 SDK 包里有相应的 LPUART 驱动文件:

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

这个 LPUART 驱动库里的 LPUART_WriteBlocking() 和 LPUART_ReadBlocking() 函数可以完成用户数据包的发送和接收,其实单纯利用 LPUART_WriteBlocking() 函数也可以实现打印信息输出,只是没有 printf() 函数那样包含格式化输出的强大功能。

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 标准 I/O 库的支持

IAR 显然是对 C 标准 I/O 库有支持的,不然我们不可能在工程里能使用 printf() 函数,只是这个支持我们如何去轻松发现呢?痞子衡今天教大家一个方法,就是看工程编译链接后生成的 .map 文件,这个 map 文件里会列出工程里所有函数的来源。

4.1 引出底层接口 __write()

我们以 SDK_2.11.0_EVK-MIMXRT1060oardsevkmimxrt1060demo_appshello_worldiar 工程为例来介绍,需要简单改造一下工程里 hello_world.c 文件里的 main() 函数,将原来代码全部删掉(原来的打印输出涉及恩智浦 SDK 封装,本文没必要关心其实现),只要如下一句打印即可:

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

然后注意工程选项里跟 Library 实现相关的如下三处设置。其中 Library 选项配置的是 runtime lib 的功能,有 Normal 和 Full 两个选项(可按需选择);Printf formatter 选项决定格式化输出功能细节,分 Full、Large、Small、Tiny 四个选项(可按需选择)。Library low-level interface implementation 选项决定低层 I/O 实现,这里我们选 None,即由用户来实现。

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

配置好 Library 后编译工程会发现有如下报错,根据这个报错我们可以猜到 dl7M_tln.a 是 IAR 编译好的 C/C++ 库,库里面实现了 printf() 函数及其所依赖的 putchar() 函数,而 puchar() 函数对外提供了底层 I/O 接口函数,这个 I/O 函数名字叫 __write(),它就是需要用户结合芯片 UART 外设去实现的发送函数。

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

在 IAR 目录下我们可以找到 dl7M_tln.a 文件路径,经过测试,工程 Library 设置里 Normal 和 Full 选项其实就是选 dl7M_tln.a 还是 dl7M_tlf.a 进用户工程去链接。

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

4.2 DLIB底层 I/O 接口设计

找到了 __write() 函数,但是它的原型到底是什么?我们该如何实现它?这时候需要去查万能的 IAR SystemsEmbedded Workbench 9.10.2armdocEWARM_DevelopmentGuide.ENU 手册,在里面搜索 __write 字样可以找到如下设计,原来我们在代码里调用的 C 标准 I/O 接口均是由 IAR 底层预编译好的 DLIB 去具体实现的,这个 DLIB 库也留下了给用户实现的最底层与硬件相关的接口函数。

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

IAR 为 DLIB 里那些最底层的 I/O 接口函数都创建了模板源文件,在这些模板文件里我们可以找到它们的原型,所以我们在 write.c 文件里找到了 __write() 原型及其示例实现。

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

4.3 DLIB库 I/O 相关源码实现

有了 __write() 原型及示例代码,我们很容易便能用 LPUART_WriteBlocking() 函数去实现它,将这个代码添加进 hello_world 工程编译,这时候就不会报错了(当然要想真正在板子上测试打印功能,main 函数里还得加入 LPUART 初始化代码)。

#include"fsl_lpuart.h"
size_t__write(inthandle,constunsignedchar*buf,size_tsize)
{
//假设使用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 库中关于 I/O 相关的源码放在了如下目录里,感兴趣的可以去查看其具体实现,这里特别提一下 formatter 文件夹下 xprintf 有很多种不同的源文件实现,其实就对应了工程选项 Printf formatter 里的不同配置。

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

至此,IAR下调试信息输出机制之硬件UART外设痞子衡便介绍完毕了,掌声在哪里~~~

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • mcu
    mcu
    +关注

    关注

    146

    文章

    17077

    浏览量

    350730
  • uart
    +关注

    关注

    22

    文章

    1232

    浏览量

    101302
  • Printf
    +关注

    关注

    0

    文章

    82

    浏览量

    13637

原文标题:printf是如何与UART外设驱动函数“勾搭”起来的?

文章出处:【微信号:嵌入式悦翔园,微信公众号:嵌入式悦翔园】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    怎么让printf函数在GB110上与UART 4一起工作

    我试图让printf函数在GB110上与UART 4一起工作。我试图创建一个本地putch函数,它应该覆盖其他函数,但是它不能工作。所以,在
    发表于 09-05 09:36

    怎样使用rttthread studio写uart外设驱动

    本次练习是使用rttthread studio写uart外设驱动,f103的sdk版本是0.1.9(在studio的包管理器那查看),导入我的例程需参考如何导入工程,祝君顺利。图中的控制台就是调试
    发表于 05-19 15:50

    使用printf()函数时默认UART0输出,请教如何重定向printf()到UART1?

    芯片:N76E003; 编译环境:keil5 使用printf()函数时默认UART0输出,请教如何重定向printf()到UART1?
    发表于 06-25 07:12

    printf函数用法示例

    《OpenCV3编程入门》书本配套源代码:printf函数用法示例
    发表于 06-06 15:20 10次下载

    实现重定向printf()和scanf() 函数案例分析

    要想printf()和scanf() 函数工作,我们需要把printf()和scanf() 重新定向到串口中。重定向是指用户可以自己重写C 的库函数,当连接器检查到用户编写了与C 库
    发表于 06-23 08:26 9176次阅读
    实现重定向<b class='flag-5'>printf</b>()和scanf() <b class='flag-5'>函数</b>案例分析

    如何使用单片机系统重写printf函数

    本文档的主要内容详细介绍的是如何使用单片机系统重写printf函数
    发表于 07-23 17:37 1次下载
    如何使用单片机系统重写<b class='flag-5'>printf</b><b class='flag-5'>函数</b>

    基于STM32的多种printf用法

    在调试代码的时候,最常用的就是使用printf函数来输出一些打印信息,提示自己代码的执行情况。 如果你的UART串口不够用,还要用printf,此时该怎么办?        解决方法:
    的头像 发表于 07-23 11:12 4846次阅读

    《51单片机笔记》keilC51软件中printf函数内部机制详解,单片机中怎么使用printf函数printf函数编写程序例子及

    printf函数的讲解
    发表于 11-20 16:51 11次下载
    《51单片机笔记》keilC51软件中<b class='flag-5'>printf</b><b class='flag-5'>函数</b>内部机制详解,单片机中怎么使用<b class='flag-5'>printf</b><b class='flag-5'>函数</b>,<b class='flag-5'>printf</b><b class='flag-5'>函数</b>编写程序例子及

    华大HC32L110 printf重映射UART

    华大HC32L110 printf重映射UART在使用printf时官方工程默认使用的端口是UART0。找到dll.c文件 fputc函数
    发表于 11-23 18:06 9次下载
    华大HC32L110 <b class='flag-5'>printf</b>重映射<b class='flag-5'>UART</b>

    【CC2530授课笔记】⑨ UART串口通信 printf

    摘要此篇文章介绍了CC2530寄存器的描述,通过具体的示例,实现了串口通信,115200波特率,并重写Putchar函数,实现了printf功能。printf 函数的实现要实现
    发表于 11-30 09:51 11次下载
    【CC2530授课笔记】⑨ <b class='flag-5'>UART</b>串口通信 <b class='flag-5'>printf</b>

    STM32F103串口1 printf函数的实现

    怎么使用呢?能不能将这个函数和串口1对应起来,当然是有方法的。  下面就通过代码来演示一下如何在串口1上使用printf()函数的功能。void u
    发表于 12-20 19:37 1次下载
    STM32F103串口1 <b class='flag-5'>printf</b><b class='flag-5'>函数</b>的实现

    通过串口利用printf函数输出数据

    一。printf函数格式printf函数具有强大的输出功能%表示格式化字符串输出目前printf支持以下格式的输出,例如:
    发表于 12-28 19:11 11次下载
    通过串口利用<b class='flag-5'>printf</b><b class='flag-5'>函数</b>输出数据

    使用printf函数的安全隐患

    程序员都知道,也都会使用printf函数,但你知道它也有“安全隐患”吗?
    的头像 发表于 10-09 09:49 1919次阅读

    如何实现Printf()接口重定向到UART

    车规级MCU开发,不像PC端,包含stdio.h头文件就可以使用Printf()函数。为了实现PC端Printf()接口功能,需要在MCU端实现Printf()接口的"重定向"
    的头像 发表于 10-01 10:13 1301次阅读
    如何实现<b class='flag-5'>Printf</b>()接口重定向到<b class='flag-5'>UART</b>

    AWorksLP应用笔记:重定向printf函数

    printf函数作为标准库定义的格式化输出方式,本文将介绍其在AWorksLP下默认适配以及重映射至热拔插设备端口的实现。默认适配AWorksLP中默认已经对printf函数完成相关适
    的头像 发表于 11-25 08:24 621次阅读
    AWorksLP应用笔记:重定向<b class='flag-5'>printf</b><b class='flag-5'>函数</b>