[应用相关] STM32 printf函数打印到串口

[复制链接]
1618|8
 楼主| 梅花香自123 发表于 2021-9-28 23:07 | 显示全部楼层 |阅读模式
学习STM32过程中,经常打交道的莫过于串口,你可以将任何信息,当然重要的是调试信息打印到串口中输出,总是用一个字节发送函数或者字符串发送函数总是有些不放便,之前编程中熟悉的莫过于printf了,下面就给出了用printf打印到串口的方案,当然方案不止一个,仅供参考。
 楼主| 梅花香自123 发表于 2021-9-28 23:08 | 显示全部楼层
1、 添加printf的头文件 #include
  1. int fputc(int ch, FILE *f)
  2. {
  3. USART_SendData(USART1, (uint8_t) ch);

  4. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  5. return ch;
  6. }
 楼主| 梅花香自123 发表于 2021-9-28 23:09 | 显示全部楼层
3、修改一下选中Use MicroLIB ,在 Target/Code Generation选中Use MicroLIB
这样就可以像以前那样使用printf了。
不过现在的USART已经在程序中加了这个重定向;
编译会出错:..\OBJ\164串口转并口.axf: Error: L6200E: Symbol fputc multiply defined (by usart.o and main.o)。
 楼主| 梅花香自123 发表于 2021-9-28 23:10 | 显示全部楼层
  1. /************************************************
  2. 程序实现3 printf实现
  3. **************************************************/
  4. #include "sys.h"
  5. #include "delay.h"
  6. #include "stdio.h"


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

  13.     while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  14.     return ch;
  15. }

  16. int main(void)
  17. {
  18.     u8 i,data;


  19.     RCC_Configuration();
  20.     GPIO_Configuration();
  21.     USART_Configuration(9600);
  22.     data = 'A';
  23.     for(i=0;i<30;i++)
  24.     {
  25.         USART_SendData(USART1, data);
  26.         data++;
  27.         while(USART_GetFlagStatus(USART1,USART_FLAG_TC)==RESET);
  28.     }
  29.     printf("yuanbao");
  30. }

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

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

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

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


  47.         //---------打开相应外设时钟--------------------
  48.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);    //使能APB2外设的GPIOA的时钟
  49.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);   
  50.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
  51.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);     
  52. }

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

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

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

  64. void USART_Configuration(u32 Baudrate)
  65. {
  66.     USART_InitTypeDef USART_InitStructure;
  67.     USART_InitStructure.USART_BaudRate = Baudrate;      
  68.     USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  69.     USART_InitStructure.USART_StopBits = USART_StopBits_1;
  70.     USART_InitStructure.USART_Parity = USART_Parity_No;
  71.     USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  72.     USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;
  73.     USART_Init(USART1, &USART_InitStructure);

  74.     USART_Cmd(USART1,ENABLE);
  75. }
木木guainv 发表于 2021-10-13 09:24 | 显示全部楼层
我很少用这个函数
wowu 发表于 2021-10-13 09:27 | 显示全部楼层
配置函数的形参是干嘛用的啊
xiaoqizi 发表于 2021-10-13 09:37 | 显示全部楼层
是的 串口是最常用的了
tpgf 发表于 2021-10-13 09:40 | 显示全部楼层
哪种方式最简便呢
heimaojingzhang 发表于 2021-10-13 09:42 | 显示全部楼层
为什么要选择这个选项呢
您需要登录后才可以回帖 登录 | 注册

本版积分规则

102

主题

1216

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部