STM32支持printf打印

							STM32支持printf打印
#include "sys.h"
#include "delay.h"
#include "stdio.h"


void RCC_Configuration(void);
void GPIO_Configuration(void);
void USART_Configuration(u32 Baudrate);
int fputc(int ch, FILE *f)//重定向,让printf输出到串口
{
    USART_SendData(USART1,ch);

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

int main(void)
{
    u8 i,data;


    RCC_Configuration();
    GPIO_Configuration();
    USART_Configuration(9600);
    data = 'A';
    for(i=0;i<30;i++)
    {
        USART_SendData(USART1, data);
        data++;
        while(USART_GetFlagStatus(USART1,USART_FLAG_TC)==RESET);
    }
    printf("yuanbao");
}

void RCC_Configuration(void)
{
        /*---------------使用外部RC晶振----------*/
        RCC_DeInit();           //设置时钟为缺省值
        RCC_HSEConfig(RCC_HSE_ON);  //使能外部高速晶振
        while(RCC_GetFlagStatus(RCC_FLAG_HSERDY) == RESET);//等待HSE准备就绪

        FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);   //使能指令预取
        FLASH_SetLatency(FLASH_Latency_2);                      //等待2个周期

        RCC_HCLKConfig(RCC_SYSCLK_Div1);    //HCLK = SYSCLK
        RCC_PCLK2Config(RCC_HCLK_Div1);     //PCLK2 = HCLK
        RCC_PCLK1Config(RCC_HCLK_Div2);     //PCLK1 = HCLK/2
        RCC_PLLConfig(RCC_PLLSource_HSE_Div1,RCC_PLLMul_9); //PLLCLK = 72MHZ
        RCC_PLLCmd(ENABLE);                                 //Enable PLLCLK
        while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); //Wait PLL is ready

       RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);         //SYSCLK = PLLCLK
       while(RCC_GetSYSCLKSource()!= 0x08);                   //Wait PLLCLK as system clock 


        //---------打开相应外设时钟--------------------
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);    //使能APB2外设的GPIOA的时钟 
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);    
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);     
} 

void GPIO_Configuration(void)
{
    GPIO_InitTypeDef    GPIO_InitStructure;     //声明一个结构体变量
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;   //
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;    //管脚频率为50MHZ
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; 

    GPIO_Init(GPIOA,&GPIO_InitStructure);                //初始化GPIOA寄存器

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;  //选择
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;    
    GPIO_Init(GPIOA,&GPIO_InitStructure);                //初始化GPIOA寄存器      
} 

void USART_Configuration(u32 Baudrate)
{
    USART_InitTypeDef USART_InitStructure; 
    USART_InitStructure.USART_BaudRate = Baudrate;      
    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_Mode_Rx; 
    USART_Init(USART1, &USART_InitStructure);

    USART_Cmd(USART1,ENABLE);
}

注意:`/********** 禁用半主机模式 **********/
#pragma import(__use_no_semihosting)

struct __FILE
{
int a;
};

FILE __stdout;

void _sys_exit(int x)
{

}
`
需要关闭半主机模式
上面的配置似乎有点麻烦,要加入这么一堆难懂的代码,难道没有更简便点的方法吗?有,但不推荐。方法是使用微库(MicroLIB),只要在Keil的“Options for Target -> Target ->Use MicroLIB”上打钩,即可使用串口打印(fputc()函数还是要改,但上述代码不用加)。微库是区别于C标准库的另一个库,当使用微库时,就默认关闭了半主机模式,也就不用添加上面的代码。这样虽然方便,但个人建议能不用就不用,原因:第一,微库是为小内存嵌入式设备而设计的,使用它可以减少代码所占空间,但对现在STM32等单片机来说,内存一般都够用,微库并非必需;第二,微库相对于C标准库而言,支持的功能更少,主要体现在对操作系统的支持上。总的来说,标准的东西总是相对更可靠,所以不必要的掉坑,还是用C标准库,不用微库。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32F407系列微控制器中,printf是一个常用的调试输出函数,通常用于打印变量值、字符串等信息。然而,由于STM32的硬件资源有限,特别是没有集成标准的UART或USART,直接使用printf可能会遇到问题。为了解决这个限制,开发者通常会选择重定向printf输出到其他可用的通信接口,如HAL库提供的USB CDC、UART、I2C或SPI。 1. **重定向到USB CDC(通用串行控制器)**:如果你的项目支持USB调试,可以通过配置USB设备为虚拟COM口,然后在HAL库中设置相应的USB CDC驱动,使其成为printf的输出目的地。 ```c HAL_UART_Init(&huart1); // 初始化一个用于printf的串口 hal_UART_SetCallback(&huart1, &usart_rx HAL_UART_Callback); // 设置接收数据的回调函数 ``` 2. **重定向到HAL UART**:如果选择传统的UART,你可以先初始化一个HAL UART结构体,然后注册一个回调函数来捕获打印信息并发送到指定的UART端口。 ```c void usart_rx(UART_HandleTypeDef *huart, uint8_t Data) { // 捕获并处理printf格式的数据 if (Data == '\n') { // 发送换行符到UART HAL_UART_Transmit(huart, "\r", 1, HAL_MAX_DELAY); } else { // 发送其他字符 HAL_UART_Transmit(huart, &Data, 1, HAL_MAX_DELAY); } } ``` 3. **使用HAL I/O Stream Interface(IStream)**:一些第三方库如mbed或FreeRTOS可能会提供一个I/O流抽象层,你可以将printf连接到这个层,然后由该层决定如何将数据发送出去。 在进行重定向时,确保正确初始化通信接口,并且处理好数据格式的解析和传输延迟。此外,可能还需要考虑printf的缓冲机制,确保数据能够被完整且及时地发送出去。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值