网上最好的printf? 移植和例程!

本文介绍了如何将一个号称网上最佳的printf函数移植到STM32F4ZG和CC2530F256芯片上,详细阐述了在Keil和IAR环境下进行移植的步骤。作者强调了printf在线程环境中的安全性问题,并提出了一种直接发送参数的线程安全解决方案。移植过程中涉及的关键步骤包括修改_putchar函数以适应不同串口,以及处理IAR的编译错误。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

网上最好的printf? 移植和例程!

最近在忙活搞别的事情(太难受了),严重影响了硪那一颗自由飞翔的芯~~
所以今天打算分享一个麻绳理工小伙写的printf家族的函数~说是号称目前网上嵌入式最好的printf喔… ~&*…;’;l]] .{.

在嵌入式中printf 这种功能强大的函数可谓是c语言库函数的中的一股清流!也就是太好用了吧!
今天分享的例程有stm32f4ZGcc2530f256,这个两款芯片的移植例程和移植教程!相信你看完后也可以移植到别的芯片去!
使用的keil版本为:5.21a
IAR for 8051 version 为 10.10.1
可能会因为有些版本不同配置就略有所不同!
这个和我前边的博客printf的方式原理会有所区别!
前来看看结果(f4)串口1:
在这里插入图片描述
串口2(stm32f4):
在这里插入图片描述
下面这个是TI 的cc2530(就先跑着寄存器版本的吧,协议栈里头配置差不太多!):
在这里插入图片描述

为啥要移植呢!相信学过实时操作系统的都有所了解有个叫信号互斥量的东西!就是为了防止同一时间内有两个寄存器(那十来个寄存器,不是指外设!)在访问同一块内存!这个很严重的,可能会导致程序死机,或者卡在了某个死循环里面!

我们所使用的printf就是过首先vsprinf 通过对我们传进来的参数进行格式化,我们传进去多少数据他按照我们传进来的格式,格式化就统统存放在一个buf里,如果我们定义了重定向,它就会把这个buf发送至你要发送的地方!

想象一下如果在某个系统中有一个线程在vsprintf里面运行,突然有一个任务级别比他高的任务把它运行的时间抢了过去!而任务优先级高的任务也在printf里面使用了那个buf,那就会产生上述的问题!

那就意味着我们所使用的printf不安全问题!当然我们在系统中可以通过临界保护区来处理,也可以通过信号量等等处理!

但是目前有一个线程安全的函数,摆在你面前,就问你用不用?

咳咳,那我们的上边printf_u1它是怎么处理的呢?毫无疑问的看代码就能知道,它是直接发送的,也就是说你传进来的每一个参数它都会顺手就发送出去(当然要看你配置),它的数据将不会存放在一个buf里边,在发送。这就是他给力的地方了!

keil移植printf

首先我们到文章的末尾获取到源代码后,打开keil软件,
点击 project > Options for…就能打开一下页面
在这里插入图片描述
点击 C/C++,选择c99标准。到此为止编译器配置完毕!

下面来配置代码部分,打开头文件为printf.h
后面会放上百度云链接!
在这里插入图片描述

1,添加这个两个文件的路径还有头文件,如果不懂可以搜搜!
2,点击图片上面的printf_ 去它定义的地方,我们要做出一定的修改。

在这里插入图片描述
来到这里之后我们可以看到_vsprintf
这个代码里面最复杂的就是这一串函数了!

//我们只需要关心第一个参数即可!它是一个函数指针
static int _vsnprintf(out_fct_type out, char* buffer, const size_t maxlen, const char* format, va_list va)
//指向了一个参数为下图的函数,返回值为void
typedef void (*out_fct_type)(char character, void* buffer, size_t idx, size_t maxlen);

我们只要修改这_putchar 里面的内容即可。参数二是我加上去的为了区分不同的串口类型,有多少个串口就的写多少个_out_char 这样的函数(这个是我的方法)

static inline void _out_char(char character, void* buffer, size_t idx, size_t maxlen)
{
  (void)buffer; (void)idx; (void)maxlen;
  if (character) {
    _putchar(character,1);
  }
}

上图是串口1的

再来看看串口2的

int printf_u2(const char* format, ...)
{
  va_list va;
  va_start(va, format);
  char buffer[1];
  const int ret = _vsnprintf(u2_out_char, buffer, (size_t)-1, format, va);
  va_end(va);
  return ret;
}
//注意vsprintf第一参数
static inline void u2_out_char(char character, void* buffer, size_t idx, size_t maxlen)
{
  (void)buffer; (void)idx; (void)maxlen;
  if (character) {
    _putchar(character,2);
  }
}

而_putchar 就是我们数据最终流向的地方了!
我是这样写的。

void _putchar(char character,char sw)
{
  // send char to console etc.
  if(sw == 1)
  {
    while((USART1->SR&0X40)==0);//循环发送,直到发送完毕
    USART1->DR = (u8) character;
  }
  if(sw == 2)
  {
	 while((USART2->SR&0X40)==0);//循环发送,直到发送完毕
     USART2->DR = (u8) character;
  }
  if(sw == 3)
  {
	 while((USART3->SR&0X40)==0);//循环发送,直到发送完毕
     USART3->DR = (u8) character;
  }

}

USART3->DR 这个为stm32串口的寄存器!读者可参照自己的芯片来配置!

简单就完了!

IAR 配置方法

首先添加文件:
在这里插入图片描述

我在这里起名字为 C_Library
然后在添加文件
在这里插入图片描述

选择printf.c(反正最是添加这个文件,放到哪里看你自己)在这里插入图片描述
接着就是添加编译器头文件路径!

在这里插入图片描述
在这里插入图片描述
在选择第一个大框框右边的那个…

在这里插入图片描述
点击Click to add

选择头文件路径所在的文件夹后 select,接着就会回到这里!
在这里插入图片描述
上面是配置头文件路径的,第一个是绝对路径,也就是说你工程文件移动了,就回到不到了,第二个是相对路径!选第二个吧!

接着编译就会发现报错了!!!

不要慌!我们首先选择Options > General Options
在这里插入图片描述
此时在编译一次,就会发现也还是错了!说什么没定义。
在这里插入图片描述
经研究,哦,原来是stdint.h这个头文件在for 8051 这款工具上,没有定义64为的无符号整形,那个怎么办啊!

经发现,哦 ,原来注释这个宏就行!
在这里插入图片描述
程序编译通过!printf_u0 和stm32 配置类型,无非就是修改一下数据的流向!读者自己研究一下吧!!
在这里插入图片描述
转载请标明出处,谢谢!

百度云源码下载链接: https://pan.baidu.com/s/16oOTmbRHKohpYMLnunK1sQ
提取码:1234

评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值