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

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

3天内不再提示

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

ZLG致远电子 2023-11-25 08:24 次阅读

printf函数作为标准库定义的格式化输出方式,本文将介绍其在AWorksLP下默认适配以及重映射至热拔插设备端口的实现。

默认适配

AWorksLP中默认已经对printf函数完成相关适配工作,且默认被适配在UART设备。用户可以在图形化配置界面中使能 support the stdio functions ,并选择期望UART设备进行输出,具体配置如下图所示。

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

注:若用户未使能 stdio function 时,调用printf函数时,将不会有任何输出。

本文将使用 EPC6450-AWI 平台,选择标有丝印为DUART的调试串口(UART0设备)进行printf功能演示测试。将TTL转USB串口模块的TXD与板子的RXD丝印连接,RXD与板子的TXD丝印,将另一端的USB口接入电脑07591f5e-8b29-11ee-9788-92fbcf53809c.png启动串口调试助手,搜索并打开串口模块的设备端口号后,在工程中调用printf函数,根据下图可知,printf函数适配UART0设备成功。076d89e4-8b29-11ee-9788-92fbcf53809c.png

重定向至其他设备嵌入式的诸多应用在UART设备资源受限的情况下,可能存在将printf函数重定向到其他设备需求。为此,笔者将以EPC6450-AWI平台的USB串口设备为例进行说明。077d1c56-8b29-11ee-9788-92fbcf53809c.png1.实施步骤

与UART设备不同,USB设备为动态设备,因此重定向printf函数时,需要注意以下几个关键步骤:

1.1 支持NEWLIB标准库函数

由于AWorksLP中利用posix file相关操作接口对printf函数进行适配,故在重映射端口时,需将 support libc file operations 使能,并取消默认选择UART设备作为printf函数的适配,具体如下图所示。

08005670-8b29-11ee-9788-92fbcf53809c.png1.2 检测动态设备

USB设备为动态设备,因此需要持续检测设备的是否存在情况。可通过初始化一个动态设备检测任务,对设备的是否存在情况进行周期性检测。

while true: access (device) delay()

1.3 关联标准文件流

在检测到USB设备存在时,仅需将设备与标准文件流(stdio中的stdin、stdout、stderr,且在C库中被假定为交互设备,并约定了这些设备的文件描述符依次为0、1、2)关联起来。故在使用时,我们仅需将描述符0、1、2与USB串口设备即可,其伪代码如下所示。

while true: if access (device): 0 = open (device) duplicate 1 to 0 duplicate 2 to 0 delay()

1.4 清理文件描述符

检测到USB设备不存在时,需及时取消设备与标准文件流的关联。即根据设备的打开情况,对文件描述符进行清理,以便之后重新关联标准文件流。

while true: if access (device): 0 = open (device) duplicate 1 to 0 duplicate 2 to 0 else: close (device) delay()

2. 基础配置在EPC6450-AWI平台标有丝印为Type-C的接口处,插上Type-C线,将Type-C线的另一端USB口连接电脑。并在图形化配置界面,将USB设备选择为CDC串口设备。

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;
// 检测动态设备 find = aw_access(AW_DYNAMIC_DEV_PATH, AW_F_OK);
if(find == AW_OK) { // 关联标准文件流 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;
// 设置检测周期 AW_TASK_DELAY(100); }
aw_kprintf("hello world, ZLG\n"); printf("hello world, ZLG\n");
return 0;}

启动串口调试助手,搜索并打开DEBUG UART设备与CDC串口设备的端口号后,运行上文示例程序。根据下图可知,USB设备枚举后,printf函数成功重定向到了CDC串口设备。0825bf28-8b29-11ee-9788-92fbcf53809c.png

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


07799d06-8b29-11ee-9788-92fbcf53809c.jpg 总结实现重定向printf函数时主要关注以下两个关键点:

  1. 重写NEWLIB标准库中printf函数的底层实现;
  2. 将指定设备以标准文件流约定的文件描述符打开。


07799d06-8b29-11ee-9788-92fbcf53809c.jpg 扩展阅读

本文所演示平台使用的是GCC编译器,其对应C库为NEWLIB标准库。在AWorksLP中printf函数的底层输出接口在AWorksLP中实现为_write_r 函数,其具体代码实现如下所示。

__attribute__((__used__)) _ssize_t_write_r(struct _reent *ptr, int fd, const void *buf, size_t nbytes){ return aw_write(fd,buf,nbytes);}

需要值得注意的是,上述适配方式仅兼容NEWLIB,若是其他编译器,其实现以及接口不尽相同,下表仅给出部分以供参考,在使用时需根据实际情况进行调整。

工具链

标准库

底层接口

GCC

NEWLIB

_write_r

ARMCC

ARMCLIB

_sys_write

ARMCLANG

ARMCLIB

_sys_write

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

    关注

    2

    文章

    4526

    浏览量

    70744
  • uart
    +关注

    关注

    22

    文章

    1241

    浏览量

    101510
  • 函数
    +关注

    关注

    3

    文章

    4338

    浏览量

    62787
收藏 人收藏

    评论

    相关推荐

    全面掌握Linux重定向技巧:让命令行操作效率翻倍的秘密

    #1 linux输出重定向 $ command > file 2>1 这里的 ,表示重定向的目标不是一个 文件 ,而是一个 文件描述符 ,内置的文件描述符如下 1 => stdout2
    的头像 发表于 01-08 10:19 148次阅读

    stdio.h实现了printf函数?

    我们平时包含的 stdio.h 头文件,里面是不是实现了 printf 函数? 为什么会有这个疑问?因为每次使用 printf,就得包含 stdio.h ,这就导致很多同学误以为,stdio.h
    的头像 发表于 12-18 10:28 184次阅读

    重定向了fputc及putchar函数,但printf没有输出,为什么?

    重定向了fputc及putchar函数,但printf没有输出 删除了drivers/drv_uart.c drv_uart.h 删除了文件rt-thread\\components
    发表于 07-18 07:44

    esp32c3能不能将串口打印函数重定向到自己软件模拟的uart上?

    模拟uart,用于打印日志;请问这种做法可以实现吗?可以将打印函数重定向到该模拟uart吗?如果可以,麻烦指导一下重定向需要修改哪几个函数? ps:打印的
    发表于 06-20 06:32

    请问如何将printf使用的stdout重定向到asclin?

    /v4.9.3.0-infineon-1.0/docs/userguide.pdf 第 406 页中关于 printf 的唯一说明是 stdout 重定向到模拟 io 调试器。 当然,您可以使用 Ifx_Console_print,但我想知道是否可以通过标准库来实现。
    发表于 06-03 06:48

    芯海通用 MCU 应用笔记:在 MDK 开发环境下代码重定向到 RAM 执行的几种方法

    为 V5.37.0.0。本文档介绍方法适用于芯海科技 MCU。*附件:应用笔记:在MDK开发环境下代码重定向到RAM执行的几种方法.pdf
    发表于 05-16 11:58

    芯海通用 MCU应用笔记 :在 IAR 及 MDK 开发环境下使用 printf 函数重定向移植差异指南

    printf 函数支持的差异。并在本应用笔记结尾处给出可以一键移植到 Keil、IAR8.x、IAR9.x 版本下的通用重定向代码。 Keil 和 IAR 都是常用的开发工具 I
    发表于 05-16 11:56

    求助,关于HAL下的printf重定向输出疑问求解

    (1)网上查了使用printf重定向,串口输出,函数如下,使用HAL_UART_Transmit函数。 int fputc(int ch,FILE *f) { uint8_t tem
    发表于 05-10 06:04

    STM8S如何在STVD环境下重定向printf函数实现UART简化输出?

    printf(),网上IAR环境的重定向调通了,求STVD环境下的code。 3. 用STM8的官方例程UART1_Printf在STVD环境下编译OK,但是debug时候报错下载不了程序 4. 针对3
    发表于 04-30 08:29

    STM8L051片子使用重定向printf函数时总是报错,没办法引用printf函数是什么原因?

    STM8L051片子使用重定向printf函数时总是报错,没办法引用printf函数,不知道是什么原因,使用的是IAR编译器,总是报内存不足
    发表于 04-28 08:05

    用keil仿真无法用printf打印怎么解决?

    用keil仿真 无法用printf打印.想用printf重定向到Usart1的方法,使printf打印信息能够在UART#1的窗口打印出来. 硬件连接用的是SW(没有连SWO引脚,只
    发表于 04-11 08:20

    STM32CubeIDE中打印重定向报错怎么解决?

    HAL_UART_Transmit( huart1 , (uint8_t *) ch, 1, 0xFFFF);报错 printf重定向 报错../Core/Src/main.c:42:21: error: \'huart1\' undeclared 已添加stdio.h头
    发表于 04-03 07:33

    H7平台如何重定向sqrtf函数到RAM中运行?

    如题,H7平台如何重定向sqrtf函数到RAM中运行,这个函数是库函数,不能使用__ramfunc前缀,有什么方法或是例程,网上找了很多icf配置的方法,都不行,要不就是只放置到RAM
    发表于 03-27 06:40

    将动态指示段重定向到单独的LED

    有时,除了数字指示器之外,还需要单独的 LED。它们可以连接到微控制器的单独引脚,但您也可以节省引脚。如果数字指示器具有从未使用过的段(例如,右边数字中的一个点),则可以将它们重定向到外部 LED。
    发表于 02-02 16:49 601次阅读
    将动态指示段<b class='flag-5'>重定向</b>到单独的LED

    2分钟搞懂输出重定向

    视频最后我们通过重定向把标准输出写到了文件中,但是错误输出还是留在了屏幕上。
    的头像 发表于 01-15 16:41 580次阅读