printf函数一般是打印到终端的,stm32芯片调试中经常需要用到串口来打印调试信息,那能不能用串口实现类似windows的Console中的printf呢?

答案是肯定的,那就是printf函数的重定向。

使用KEIL5对stm32的printf函数进行重定向,有两种方法:一种是使用微库,另一种是不使用微库。

方法1–使用微库:

1、使用微库,在KEIL5中点击options for target,在” Target标签下有个Use MicroLIB—勾选,使用微库。

2、在串口文件中添加如下代码:

#include "stdio.h"

 #ifdef __GNUC__
  #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
  #else
  #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE* f)
  #endif /* __GNUC__ */


  #ifdef __cplusplus
  extern “C” {
  #endif //__cplusplus

  PUTCHAR_PROTOTYPE 

  {
    while (USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET)
    ;
    USART_SendData(USART2, (uint8_t)ch);

    return (ch);
  }

  #ifdef __cplusplus
  }
  #endif //__cplusplus

这里的:USART_SendData(USART2, (unsigned char) ch);

    while (!(USART2->SR & USART_FLAG_TXE));

就是往串口发送一个字节的代码,修改相应的串口号,初始化,就能使用printf了。

 

方法2–不使用微库(那么就要强调不使用半主机(no semihosting)模式)

1、使用微库(平台式keil-MDK),点击“魔术棒options for target,在” Target标签下有个Use MicroLIB—取消勾选,不使用微库。

2、在串口文件中添加如下代码:

#ifdef __GNUC__
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE* f)
#endif /* __GNUC__ */

#ifdef __cplusplus
extern "C" {
#endif

//加入以下代码,支持printf函数,而不需要选择use MicroLIB	  
#pragma import(__use_no_semihosting)             
//定义_sys_exit()以避免使用半主机模式
void _sys_exit(int x)
{ 
	x = x; 
} 

void _ttywrch(int x)
{
	x = x;
}

//标准库需要的支持函数
struct __FILE 
{ 
	int handle; 
}; 

FILE __stdout;     

//重定义fputc函数
PUTCHAR_PROTOTYPE
{
	while((USART2->SR & 0X40) == 0);//循环发送,知道发送完毕
	USART2->DR = (u8)ch;      
	return ch;
}

#ifdef __cplusplus
}
#endif //__cplusplus

同样的,修改相应的串口号,初始化,就能使用printf了。

如果编译的时候出现FILE __stdout;编译不过,可以打开stdio.h文件,将typedef struct __FILE FILE; 

修改为

typedef struct __FILE
{

}FILE;

即可。

版权声明:本文为jasonchan1202原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/jasonchan1202/p/9673548.html