【嵌入式笔记】简要谈一谈嵌入式开发中重写printf的几种方法
admin
2024-03-30 10:34:17
0

文章目录

  • 抛砖引玉
    • 1 宏定义替换
    • 2 直接重写printf函数的实现
    • 3 重写fputc/putchar接口
    • 4 仿照POSIX的实现,从文件描述符0/1/2上面做文章
  • 总结
  • 更多分享

摘要

简单总结了一下几种重写printf的方法,有兴趣的可以参与讨论讨论。

抛砖引玉

我大概查阅并考证了一下,在嵌入式开发中重写printf大概有以下几种方法:

1 宏定义替换

比如将printf强制替换成rt_kprintf:

#define printf rt_kprintf

有些已经实现了类似printf打印接口的平台,我们就可以简单地这么干!

2 直接重写printf函数的实现

这种就比较生硬,直接自定义一个printf函数,类似这样:

char buf[256];
int printf(const char *format,...)
{va_list args;va_start(args, format);vsprintf(buf, format, args);va_end(args);usrt_send(buf, strlen(buf));return 0;
}

先用vsprintf之类的函数把参数格式化成字符串数组,然后直接调用串口之类的驱动发送接口发出去。

有些小平台的芯片,不想引入太多的依赖,就会这么干!

3 重写fputc/putchar接口

这种做法就是在bsp层重写fputc/putchar接口,在接口里面调用串口驱动发送出去,类似这样:

可以参考:bsp/stm32/stm32mp157a-st-discovery/board/CubeMX_Config/CM4/Src/openamp_log.c

#if defined ( __CC_ARM) || (__ARMCC_VERSION) && (__ARMCC_VERSION >= 6010050)
#define PUTCHAR_PROTOTYPE int stdout_putchar(int ch)
#elif __GNUC__
/* With GCC/RAISONANCE, small log_info (option LD Linker->Libraries->Small log_infoset to 'Yes') calls __io_putchar() */
#define PUTCHAR_PROTOTYPE int __attribute__(( weak )) __io_putchar(int ch)
#else
#define PUTCHAR_PROTOTYPE int __attribute__(( weak )) fputc(int ch, FILE *f)
#endif /* __GNUC__ */#if defined (__LOG_UART_IO_) || defined (__LOG_TRACE_IO_)
PUTCHAR_PROTOTYPE
{/* Place your implementation of fputc here *//* e.g. write a character to the USART1 and Loop until the end of transmission */
#if defined (__LOG_UART_IO_)
extern UART_HandleTypeDef huart;HAL_UART_Transmit(&huart, (uint8_t *)&ch, 1, HAL_MAX_DELAY);
#endif
#if defined (__LOG_TRACE_IO_)log_buff(ch);
#endifreturn ch;
}

4 仿照POSIX的实现,从文件描述符0/1/2上面做文章

这种实现方法,在rt-thread中也可以见到,例如: bsp/qemu-vexpress-a9

我们可以通过反编译跟踪一下printf的调用轨迹:

341970 60086be4 :
341971 printf():
341972 60086be4:       b40f            push    {r0, r1, r2, r3}
341973 60086be6:       f24a 4288       movw    r2, #42120      ; 0xa488
341974 60086bea:       b500            push    {lr}
341975 60086bec:       f2c6 0209       movt    r2, #24585      ; 0x6009
341976 60086bf0:       b083            sub     sp, #12
341977 60086bf2:       ab04            add     r3, sp, #16
341978 60086bf4:       6810            ldr     r0, [r2, #0]
341979 60086bf6:       f853 2b04       ldr.w   r2, [r3], #4
341980 60086bfa:       6881            ldr     r1, [r0, #8]
341981 60086bfc:       9301            str     r3, [sp, #4]
341982 60086bfe:       f002 fbdd       bl      600893bc <_vfprintf_r>
341983 60086c02:       b003            add     sp, #12
341984 60086c04:       f85d eb04       ldr.w   lr, [sp], #4
341985 60086c08:       b004            add     sp, #16
341986 60086c0a:       4770            bx      lr

最终调用的是 _vfprintf_r

同时,我们在 build_v2/kernel/components/libc/posix/io/stdio/libc.c 中可以看到:

int libc_system_init(void)
{
#ifdef RT_USING_POSIX_STDIOrt_device_t dev_console;dev_console = rt_console_get_device();if (dev_console){libc_stdio_set_console(dev_console->parent.name, O_RDWR);}
#endif /* RT_USING_POSIX_STDIO */return 0;
}
INIT_COMPONENT_EXPORT(libc_system_init);

通过这个libc_stdio_set_console接口,就通过将console设备驱动(比如串口)跟文件句柄输入、输出、错误绑在一块,这就回到了标准C库的实现逻辑了。

值得注意的是,这里调用printf还会调用编译器里面的c库:

compiler/gcc-arm-none-eabi-9-2019-q4-major/Linux64/bin/../lib/gcc/arm-none-eabi/9.2.1/../../../../arm-none-eabi/lib/thumb/v7/nofp/libc.a(lib_a-printf.o)
rtthread.map:7287:printf 

总结

以上就是我总结的几种实现方法,也可可能不一定全对,欢迎大家指正。

如果你有更好的思路或实现方法,也欢迎拿出来讨论。

更多分享

架构师李肯

架构师李肯全网同名),一个专注于嵌入式IoT领域的架构师。有着近10年的嵌入式一线开发经验,深耕IoT领域多年,熟知IoT领域的业务发展,深度掌握IoT领域的相关技术栈,包括但不限于主流RTOS内核的实现及其移植、硬件驱动移植开发、网络通讯协议开发、编译构建原理及其实现、底层汇编及编译原理、编译优化及代码重构、主流IoT云平台的对接、嵌入式IoT系统的架构设计等等。拥有多项IoT领域的发明专利,热衷于技术分享,有多年撰写技术博客的经验积累,连续多月获得RT-Thread官方技术社区原创技术博文优秀奖,荣获CSDN博客专家、CSDN物联网领域优质创作者、2021年度CSDN&RT-Thread技术社区之星、2022年RT-Thread全球技术大会讲师、RT-Thread官方嵌入式开源社区认证专家、RT-Thread 2021年度论坛之星TOP4、华为云云享专家(嵌入式物联网架构设计师)等荣誉。坚信【知识改变命运,技术改变世界】!


欢迎关注我的gitee仓库01workstation ,日常分享一些开发笔记和项目实战,欢迎指正问题。

同时也非常欢迎关注我的CSDN主页和专栏:

【CSDN主页-架构师李肯】

【RT-Thread主页-架构师李肯】

【C/C++语言编程专栏】

【GCC专栏】

【信息安全专栏】

【RT-Thread开发笔记】

【freeRTOS开发笔记】

有问题的话,可以跟我讨论,知无不答,谢谢大家。

相关内容

热门资讯

贷款也“拼团” 银行抢单忙 购物能“拼团”,贷款也能! 近日,一场“拼团融资”的银企对接活动在省工业和信息化厅拉开帷幕。 “贷款...
逛花展、赶市集、嗨直播!202... 5月23日 “2026北京直播电商购物月” 在丰台区丽泽金融商务区·2026北京国际花展 正式拉开帷...
2026中关村毕业季|AI“吃... “上帝会掷骰子吗?” 在联想未来中心的“与智者同场”展区,一位海淀学子对着屏幕问道。 爱因斯坦微微前...
原创 今... 今日为5月23日,国际现货黄金价格在4500美元/盎司整数关口附近徘徊不前,日内最低触及4480美元...
三连亏后变为“无主”状态,农尚... 从吴亮手中接盘农尚环境(300536)不足三年后,林峰如今让出了公司控制权,上市公司进入“无主”状态...
55岁湖南女首富出手!豪掷13... 快科技5月24日消息,与马斯克、库克并肩而坐,刚参加完国宴的湖南女首富周群飞就买了家上市企业。 近日...
外资加仓A股,岂是跟风这么简单... 熬过忙碌的交易日,在周末安静时段,理清接下来布局方向。本篇为大家准备了5条要闻,涵盖市场动态、行业变...
原创 俄... 在全球能源的残酷牌桌上,手里攥着石油,腰杆子才能硬气。长期以来,中东的沙漠、俄罗斯的冰原、美国的页岩...
喜力啤酒有产品将涨价,华润啤酒... 来源:红星新闻 红星资本局5月22日消息,今日,红星资本局从雪花啤酒(厦门)有限公司、华润啤酒方面获...
原创 金... 心理预期调整刻不容缓,五月二十二日,黄金价格或将重现十五年前的历史性低迷。 近期若您密切关注着黄金市...
原创 马... 埃隆·马斯克如果能让SpaceX实现“科幻小说”级别的目标,他可能获得1万亿美元的收入。 埃隆·马斯...
涨涨涨!放开限制、可加杠杆!这... 韩国股市站在风口上! 据最新消息,为吸引更多海外资金进入股市,韩国政府计划放开限制,允许境外投资者直...
下周9家上会丨科创板首单IPO... IPO及再融资上会预告 据交易所官网审核动态信息,下周(5.25-5.29)IPO上会审核6家企业,...
富途、老虎市值蒸发1/4!或被... 来源:金融时报 5月22日,中国证监会宣布依法对Tiger Brokers (NZ) Limited...
马爸爸的好兄弟钱多多搞了杀猪盘... *此图由AI生成 作者| 史大郎&猫哥 来源| 是史大郎&大猫财经Pro 上周四,港股经纬天地大崩盘...
原创 壳... 编辑:XL 国际能源圈最近炸开了锅,壳牌这家百年石油巨头在2026年3月与委内瑞拉政府正式签署多项油...
存储热潮愈演愈烈!奖金拿到手软... 财联社5月24日讯(编辑 卞纯)在席卷全球的存储芯片热潮中,韩国“存储芯片双雄”SK海力士和三星无疑...
揽牌、合作、生态,跨境支付头部... 近日,国内头部跨境支付机构密集落地海外重要布局,一方面,连连数字、PingPong两家公司相继在中东...
原创 帮... 老铁们,周末好!我是帮主郑重。刚扫了一眼下周的财经日历,好家伙,事件一个接一个,堪称“消息面轰炸周”...
海南省住建厅与中国石化海南石油... 5月22日,中国石化海南石油分公司代表、党委书记李新强、总经理蔡文东一行赴海南省住建厅拜访交流。省住...