STM32 HAL库 使用printf函数 Use MicroLIB配置

问题描述

STM32 开发过程中难免需要使用串口打印一些调试信息,经常会用到printf()函数;
STM32 还不能直接使用printf()函数,需要重定向一下,这里讲解两种配置办法;

该配置基于HAL库


解决方案:

使用微库

配置MDK-keil ,点击魔术棒Target里面勾选Use MicroLIB

Use MicroLIB

然后重定向printf()函数,修改fputc()函数

# include "stdio.h"
int fputc(int ch, FILE *f)
{
    HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
    return ch;
}

不使用微库

不用配置MDK-keil

直接重定向printf()函数

//printf重定向代码,修改其底层fputc
#if 1
#include <stdio.h>

/* 告知连接器不从C库链接使用半主机的函数 */
#pragma import(__use_no_semihosting)

/* 定义 _sys_exit() 以避免使用半主机模式 */
void _sys_exit(int x)
{
    x = x;
}

/* 标准库需要的支持类型 */
struct __FILE
{
    int handle;
};

FILE __stdout;

/*  */
int fputc(int ch, FILE *stream)
{
    /* 堵塞判断串口是否发送完成 */
	/* 不同芯片的串口标志位不一定相同! */
    while((USART1->SR & 0X40) == 0);

    /* 串口发送完成,将该字符发送 */
    USART1->DR = (uint8_t) ch;

    return ch;
}

#endif

使用多串口方法

如果要使用多个串口,比如在做物联网一些项目的时候,一个串口用来连接外设或者连接WIFI模块,另一个串口用来输出调试信息;

参考原子哥的代码,在HAL库中需要修改,具体修改方法参考正点原子视频,直接给出修改后的代码:

需要在 usrat.c 文件中添加如下代码:

#include <stdarg.h>
#include <string.h>
#include <stdio.h>
void UsartPrintf(UART_HandleTypeDef USARTx, char *fmt,...)
{
 
	unsigned char UsartPrintfBuf[296];
	va_list ap;
	unsigned char *pStr = UsartPrintfBuf;
	
	va_start(ap, fmt);
	vsnprintf((char *)UsartPrintfBuf, sizeof(UsartPrintfBuf), fmt, ap);							//格式化
	va_end(ap);
	
	while(*pStr != NULL)
	{
        HAL_UART_Transmit (&USARTx ,(uint8_t *)pStr++,1,HAL_MAX_DELAY );		
	}
 
}

需要在 usrat.h文件中添加如下代码:

#define USART_DEBUG		huart1
 
void UsartPrintf(UART_HandleTypeDef USARTx, char *fmt,...);

使用办法直接调用函数:

UsartPrintf(USART_DEBUG, "The USART1 is OK!\r\n");

注意:函数参数中 USART_DEBUG 参数为在 usrat.h 中重定义的 huart1 。

如果同时打开了USART1和USART2,那么在 usrat.h 中还会有一个 huart2 ,可以像huart1 一样重定义 huart2 ,可以自定义成其它名字以和其它串口区分开来。

优缺点

第一种使用微库,需要配置keil 里面的Use MicroLIB;
第二种不用配置设置,直接使用,但是第二个比较长;
第三种适用与使用多个串口的情况,比如在使用esp8266模块时,一个串口用来调试,另外一个用于和模块通信。
为了防止忘记配置,建议使用第二个;

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值