大家好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给大家分享的是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 上串口调试助手软件里可以看到打印信息。
上图里的 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.hsdk_2.11.0_evk-mimxrt1060devicesmimxrt1062driversfsl_lpuart.c
这个 lpuart 驱动库里的 lpuart_writeblocking() 和 lpuart_readblocking() 函数可以完成用户数据包的发送和接收,其实单纯利用 lpuart_writeblocking() 函数也可以实现打印信息输出,只是没有 printf() 函数那样包含格式化输出的强大功能。
status_t lpuart_init(lpuart_type *base, const lpuart_config_t *config, uint32_t srcclock_hz)status_t lpuart_writeblocking(lpuart_type *base, const uint8_t *data, size_t length)status_t lpuart_readblocking(lpuart_type *base, uint8_t *data, size_t length)
四、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 int main(void){ printf(hello world.); while (1);}
然后注意工程选项里跟 library 实现相关的如下三处设置。其中 library 选项配置的是 runtime lib 的功能,有 normal 和 full 两个选项(可按需选择);printf formatter 选项决定格式化输出功能细节,分 full、large、small、tiny 四个选项(可按需选择)。library low-level interface implementation 选项决定低层 i/o 实现,这里我们选 none,即由用户来实现。
配置好 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 进用户工程去链接。
4.2 dlib底层 i/o 接口设计
找到了 __write() 函数,但是它的原型到底是什么?我们该如何实现它?这时候需要去查万能的 iar systemsembedded workbench 9.10.2armdocewarm_developmentguide.enu 手册,在里面搜索 __write 字样可以找到如下设计,原来我们在代码里调用的 c 标准 i/o 接口均是由 iar 底层预编译好的 dlib 去具体实现的,这个 dlib 库也留下了给用户实现的最底层与硬件相关的接口函数。
iar 为 dlib 里那些最底层的 i/o 接口函数都创建了模板源文件,在这些模板文件里我们可以找到它们的原型,所以我们在 write.c 文件里找到了 __write() 原型及其示例实现。
size_t __write(int handle, const unsigned char * buffer, size_t size)
4.3 dlib库 i/o 相关源码实现
有了 __write() 原型及示例代码,我们很容易便能用 lpuart_writeblocking() 函数去实现它,将这个代码添加进 hello_world 工程编译,这时候就不会报错了(当然要想真正在板子上测试打印功能,main 函数里还得加入 lpuart 初始化代码)。
#include fsl_lpuart.hsize_t __write(int handle, const unsigned char *buf, size_t size){ // 假设使用 lpuart1 去做输出 (void)lpuart_writeblocking(lpuart1, buf, size); return 0;}
工程编译完成后,查看生成的 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.2armsrclibdlibfileiar systemsembedded workbench 9.10.2armsrclibdlibformatters
至此,iar下调试信息输出机制之硬件uart外设痞子衡便介绍完毕了,掌声在哪里~~~
信步科技SV1-H2726C规格英文版
无线遥控器在安防防盗中的工作原理分析
ph值和orp的关系
教大家如何选择一款好的混波段UVLED固化机
电阻柜接地保护的优势
printf是如何与UART外设驱动函数“勾搭”起来的?
半导体超表面光学技术
谷景科普直流共模电感电感量的选择小技巧
力压Intel、Nvidia,AMD采用7nm工艺打造新GPU、CPU,预计7月问世
两类高压变频器的技术性能分析对比概述
自制CPU(二)多周期
导入工位ANDON呼叫拉绳终端的效果
边缘计算概论及优缺点介绍
湘潭大学微纳传感器加工实验室落成,与华为和京东方合作
香港大学机器鱼创下吉尼斯世界纪录
在中国的扎根之旅困难重重,特斯拉还有底牌吗?
京信通信发布700M龙伯透镜天线 助力运营商打造高质量海域5G网络
人工智能为刺激消费能力做了那些事?
AI芯片是无人车领域的一个重要战场
各大手机芯片的性能排名是怎样的