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

发布者:草木知秋最新更新时间:2018-05-31 来源: eefocus关键字: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 require 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 transmission */
    while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET)
    {
    }

    return ch;
}


关键字:stm32  printf函数  重定向 引用地址:关于stm32中printf函数重定向问题

上一篇:stm32 IAR printf函数重定义
下一篇:IAR for ARM 中printf使用方法

推荐阅读最新更新时间:2024-03-16 16:03

STM32一键下载电路导致的单片机不能正常运行
使用的正点原子精英版的电路,直接使用那种4线的stlink给单片机供电+下载程序,发现能够正常下载,但死活不运行程序。 一键下载电路如图~ 当时没有使用USB串口功能,所以没有接入5V供电,导致了RTS和DTR引脚悬空,测量电压均为2.69V,即BOOT0被拉高,单片机的启动模式变成了从ISP下载。 插上5V供电一切正常~
[单片机]
<font color='red'>STM32</font>一键下载电路导致的单片机不能正常运行
STM32学习笔记之定时器的配置
实验目的: 让蜂鸣器每隔一秒响一次; 实验步骤: 实验程序: /************************led.c***********************/ #include stm32f4xx.h //在SYSTEM目录下可以找到 #include sys.h void LED_Init(void){ RCC- AHB1ENR |= 1 5; //使能GPIO端口的F时钟 GPIO_Set(GPIOF,PIN9|PIN10,GPIO_MODE_OUT,GPIO_OTYPE_PP,GPIO_SPEED_25M,GPIO_PUPD_PU);
[单片机]
<font color='red'>STM32</font>学习笔记之定时器的配置
一文解析STM32启动流程
可执行程序 - cpu执行第一条用户代码 这个流程中着重讲述的是 HEX 文件如何被烧写到 STM32 内部的指定地址处。(烧写到 STM32 中的可执行文件不仅只有 HEX 格式,还有 axf、bin。针对不同格式的可执行文件,用不同的工具进行烧写)。 而本篇文章将要详细地描述一个流程: cpu执行第一条用户代码 - 调用 __main 函数- __rt_entry - main函数 这里需要注意一下,__main 是 c 库中的一个函数,和 main 函数是有区别的!!! 启动文件内容描述 上图中的汇编关键字最好记住,因为比较常用。 在此基础上,我们继续深入一点。 DCD指令 STM32 启动文件中使用 DCD 指令的
[单片机]
一文解析<font color='red'>STM32</font>启动流程
stm32专题五:时钟树(一)寄存器
详细介绍stm32系统时钟配置: 1 时钟来源:外部高速晶振HSE和内部RC振荡器HSI。通过RCC_CR(时钟控制寄存器来配置)。 HSEON为1时,打开外部时钟;当位HSERDY为1时,表示HSE已经准备就绪。同理,HSION和HSIRDY配置内部时钟。当HSE故障时,系统时钟自动切到HSI,直到HSE启动成功。注意:由于HSI不太精确,系统启动时会自动校准,HSICAL 这些位被自动初始化,HSITRIM 用于内部时钟HSI的调整,由软件写入,这些位会叠加到HSICAL 的数值上。 问题:具体怎么调整?HSITRIM 的默认初始值为16,即10000。中文参考手册中说到,可以把HSI调整到8MHz±1
[单片机]
<font color='red'>stm32</font>专题五:时钟树(一)寄存器
stm32 移植uip
1 stm32 定义常量不要code 关键字 // 定义全1 地址常量 static const uip_ipaddr_t code all_ones_addr = #if UIP_CONF_IPV6 {0xffff,0xffff,0xffff,0xffff,0xffff,0xffff,0xffff,0xffff}; #else /* UIP_CONF_IPV6 */ {0xffff,0xffff}; #endif /* UIP_CONF_IPV6 */ //定义全0地址常量 static const uip_ipaddr_t code all_zeroes_addr = #if UIP_CONF_IP
[单片机]
STM32程序跳转需要注意的问题
Note: keil生成的.axf或者.elf都是可以执行的镜像文件。 IAP - User Program 需要注意的问题 要设置好好中断向量的问题,假设User Program 开始地址是(0x08008000),那么中断向量就必须设在在0x08008000处,中断向量要设置program的开始的位置。 User Program到IAP主要是要设置好中断向量,跳转前要注意设置中断向量在IAP程序(0x08000000)的开始地方即(0x08000000)。还要关闭所有中断,如下操作: NVIC_SetVectorTable(0x08000000, 0x00); __disable_irq(); 问题:如果这样
[单片机]
STM32封装ESP8266一键配置函数:实现实现AP模式和STA模式切换、服务器与客户端创建
工程源码下载: https://download.csdn.net/download/xiaolong1126626497/15805037 一、硬件环境介绍 1. ESP8266 :采用安信可的模组,型号是ESP12F 2. STM32 : 采用STM32F103C8T6 3. 编程软件 : 采用Keil5 ESP8266编程调试过程中用到的相关软件下载地址: 物联网网调试终端:https://download.csdn.net/download/xiaolong1126626497/12624530 TCP调试助手:https://download.csdn.net/download/xiaolong
[单片机]
<font color='red'>STM32</font>封装ESP8266一键配置<font color='red'>函数</font>:实现实现AP模式和STA模式切换、服务器与客户端创建
基于STM32的实时心率检测仪设计
一、开发环境介绍 主控芯片: STM32F103ZET6 代码编程软件: keil5 心率检测模块: PulseSensor WIFI模块: ESP8266 --可选的。直接使用串口有线传输给上位机也可以。 上位机: C++(QT) 设计的。 支持PC机电脑、Android手机显示。 与上位机的传输协议: 支持串口传输、WIFI网络传输两种。 如果是PC就可以直接连接串口传输数据,如果不方便可以直接通过WIFI---TCP协议传输。 二、PulseSensor心率模块介绍 PulseSensor 是一款用于脉搏心率测量的光电反射式模拟传感器。 可以将其佩戴于手指、耳垂、手腕等处,通过杜邦线--导线将引脚连接到单片机,可将采集到
[单片机]
基于<font color='red'>STM32</font>的实时心率检测仪设计
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved