关于stm32中printf函数重定向问题

学习stm32过程中,对printf函数进行了重定向,但是不能够在串口调试助手中显示出来,并且debug不能全速运行,printf函数阻碍了程序的运行。

后来求教了度娘。


标准库函数的默认输出设备是显示器,要实现在串口或LCD输出,必须重定义标准库函数里调用的与输出设备相关的函数.

例如:printf输出到串口,需要将fputc里面的输出指向串口(重定向),方法如下:

#ifdef __GNUC__


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


PUTCHAR_PROTOTYPE
{
USART_SendData(USART1, (uint8_t) ch);

while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
return ch;
}


因printf()之类的函数,使用了半主机模式。使用标准库会导致程序无法运行,以下是解决方法:


方法1.使用微库,因为使用微库的话,不会使用半主机模式.



方法2.仍然使用标准库,在主程序添加下面代码:


#pragma import(__use_no_semihosting) 
_sys_exit(int x) 

x = x; 

struct __FILE 

int handle; 



}; 

FILE __stdout;
另一个高手的讲解
刚开始学stm32,顺着gpio、 uart 。。。的顺序慢慢爬
初始化的方法学习了马老师的 STM32 _Init.h****,自己英文还可以,加上 avr 的基础还不错,所以gpio和时钟配置都很顺利
碰到uart就头大了,看到各种例程里都是printf()函数,自己也想用,毕竟是avr想用却开销不了的东西。但是我自己写的程序里一旦出现printf, 单片机 的不干活了。查论坛首先发现要重定义fputc函数,照做了,还是不行。
后来怀疑是uart1初始化问题,用自己写的put_c函数却没问题。
后来又发现一种说法,需要避免使用semihosting(半主机模式),我也把代码加进去了(改fputc去掉了),还是不行。
再一想,重定义fputc是绝对必须的,加上了之后问题解决,成功使用printf("(敏感词0373) \n");输出了,哈哈
***************************************************************************************************

以上废话,可以不看。
简单地说:想在mdk 3.80a中用printf,需要同时重定义fputc函数和避免使用semihosting(半主机模式), 
论坛里应该有完整介绍这个的帖子,但是我没搜到,也许是沉了。重发出来希望能帮上像我这样的菜鸟们。

需要添加以下代码


#pragma import(__use_no_semihosting) 
/****************************************************************************** 
*标准库需要的支持函数 
******************************************************************************/ 
struct __FILE 

int handle; 
/* Whatever you r equ ire here. If the only file you are using is */ 
/* standard output using printf() for debugging, no file handling */ 
/* is required. */ 
}; 
/* FILE is typedef’ d in stdio.h. */ 
FILE __stdout; 

/// 
/// 定义_sys_exit()以避免使用半主机模式 
///  
///   
///   
_sys_exit(int x) 

x = x; 




int fputc(int ch, FILE *f)
{
    //USART_SendData(USART1, (u8) ch);
    USART1->DR = (u8) ch;
    
    /* Loop until the end of transm issi on */
    while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET)
    {
    }

    return ch;
}

在使用STM32printf函数时,可以通过重定向来将其输出重定向到串口或其他输出设备上。这样可以方便地在开发过程通过串口输出调试信息。 首先,需要在代码定义一个文件描述符,并重写该文件描述符的_write函数。在这个函数,可以将要输出的字符通过串口发送出去。 以下是一个使用USART1作为串口输出的示例代码: ```c #include "stdio.h" // 重写文件描述符的 _write 函数 int _write(int file, char *ptr, int len) { for (int i = 0; i < len; i++) { // 将字符发送到串口 USART_SendData(USART1, (uint8_t) *ptr++); // 等待发送完毕 while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); } return len; } int main(void) { // 初始化串口 USART_InitTypeDef USART_InitStructure; USART_InitStructure.USART_BaudRate = 9600; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Tx; USART_Init(USART1, &USART_InitStructure); USART_Cmd(USART1, ENABLE); // 使用printf输出 printf("Hello, world!\n"); while (1) { // 其他操作 } } ``` 在这个例子,我们将USART1作为串口输出设备,并将printf函数的输出重定向到USART1。需要注意的是,在使用重定向后,printf函数会比较耗费资源,因此在实际应用要谨慎使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值