码迷,mamicode.com
首页 > 其他好文 > 详细

STM32调试大法 之 串口通讯

时间:2014-09-04 01:22:47      阅读:277      评论:0      收藏:0      [点我收藏+]

标签:style   blog   http   color   os   io   使用   strong   ar   

开发过程经常需要查看某些特定参数。通常的方法可以使用paintf进行打印输出,观察具体的变量值。STM32内部集成有USART的串口功能,可以通过串口直接输出到电脑(上位机)。使用非常方便,基本不需要不需要写代码,只要配置一下就可以使用。

 

bubuko.com,布布扣

简单设置就可以看到上面的效果

 

配置方法:

  1、重定向printf的输出函数 int fputc(int ch, FILE *f)

  2、配置STM32F10x的USART串口

 

重定向方法 stdio.h 的输出内容

int fputc(int ch, FILE *f)
{
  // 等待USART1 数据发送完成(发送区域空)
  while (!(USART1->SR & USART_SR_TXE));
  // 填充发送寄存器,数据 + 校验位 最多9位
  USART1->DR = (ch & 0x1FF);
  
  return (ch);
}

启动STM32的 USART1 串口

    1、使能 复用功能

    2、使能 A引脚(USART1 在 PA9,PA10)

    3、使能 USART1

    4、设置 PA9(TX)复用推挽输出

    5、设置 PA10(RX)输入浮空

    6、 设置串口参数:波特率、数据位、校验、停止位、流控制

    7、使能串口、使能输入、使能输出

    OK

void serial_init(void)
{
  //
  //  设置串口调试
  //   
  //     输  出: USART1
  //     引  脚: PA9(TX), PA10(RX)
  //     波特率: 9600
  //     数据位: 8 bit (default)  (CR1)
  //     校  验: none  (default)  (CR1)
  //     停止位: 1 bit (default)  (CR2)
  //     流控制: none  (default)  (CR3)
  //
  // 清除设置后上面配置为系统默认状态
  
  int i;
  /// 使能复用功能,使能GPIOA,使能USART1
  RCC->APB2ENR |= RCC_APB2ENR_AFIOEN | RCC_APB2ENR_IOPAEN | RCC_APB2ENR_USART1EN;
  // 关闭映射,确保USART使用 PA9,PA10
  AFIO->MAPR &= ~AFIO_MAPR_USART1_REMAP;
  
  // 清除PA9,PA10状态
  GPIOA->CRH &= ~( GPIO_CRH_CNF9 | GPIO_CRH_MODE9 | 
                   GPIO_CRH_CNF10 | GPIO_CRH_MODE10 );
  // 设置PA9 发送 为复用推挽输出 2MHz
  GPIOA->CRH |= GPIO_CR_AFOUT_PP2MHz & ( GPIO_CRH_CNF9 | GPIO_CRH_MODE9 );
  // 设置PA10接收 为复用上拉下拉模式
  GPIOA->CRH |= GPIO_CR_AFIN_PULLDOWN & ( GPIO_CRH_CNF10 | GPIO_CRH_MODE10 );
  

  // 设置波特率为 9600
  // 计算方法
  //          系统时钟 / (16分频 * 波特率)
  //   Baud = 72,000,000 / (16 * 9600) = 468.75
  //   整数部分 << 4 + 取整(小数部分 * 16)
  //   468 << 4 + 0.75 * 16
  USART1->BRR = __USART_BRR(SystemCoreClock, 9600);
  
  // 清除寄存器状态
  USART1->CR1 = USART_CR1_REST;
  USART1->CR2 = USART_CR2_REST;           // 停止位 1
  USART1->CR3 = USART_CR3_REST;           // 没用控制流
  
  // 防止产生不必要的信息
  for (i = 0; i < 0x1000; i++) __NOP();
  
  // USART1 使能, 使能输出,使能输入
  USART1->CR1 =  USART_CR1_UE | USART_CR1_TE | USART_CR1_RE;
  
}

 

整个过程中关键的输出定向主要用到了2个寄存器:

   一个状态寄存器,检查发送是否为空(USART_ST_TXE)

  一个数据寄存器,用于发送数据

int main(void)
{
  uint8_t ud = a;
  Delay_init();
  // 初始化串口调试
  serial_init();
  
  // 使用寄存器直接 输出 a~z
  while (ud <= z) {
    while (!(USART1->SR & USART_SR_TXE));
    USART1->DR = ud;
    ud++;
  }
    
  // 使用打印重定向输出字符串
  printf("\ntest!\n");
  printf("USART1 使能, 使能输出,使能输入\n");
  
  ud = a;
  while (1) {  
    Delay(20); // 太快降低速度方便看结果(200ms延迟)
    // 使用寄存器直接输出
    while (!(USART1->SR & USART_SR_TXE));
    USART1->DR = ud;
    ud++;    
    
    // 使用打印输出换行
    if (ud > z) {
      ud = a;
      printf("\n");  
    }
  };       
  
}

 

具体的配置寄存器可以查看《参考手册》

 

完整代码

bubuko.com,布布扣
#define GPIO_CR_RESET                 (uint32_t)0x44444444 
#define GPIO_CR_MODE_INPUT            (uint32_t)0x00000000
#define GPIO_CR_MODE_2MHz             (uint32_t)0x22222222
#define GPIO_CR_MODE_10MHz            (uint32_t)0x11111111
#define GPIO_CR_MODE_50MHz            (uint32_t)0x33333333

#define GPIO_CR_GP_PUSHPULL           (uint32_t)0x00000000
#define GPIO_CR_GP_OPENDRAIN          (uint32_t)0x44444444 
#define GPIO_CR_OUT_PP2MHz            (GPIO_CR_MODE_2MHz | GPIO_CR_GP_PUSHPULL)
#define GPIO_CR_OUT_PP50MHz           (GPIO_CR_MODE_50MHz | GPIO_CR_GP_PUSHPULL)

   
#define GPIO_CR_AFO_PUSHPULL           (uint32_t)0x88888888
#define GPIO_CR_AFO_OPENDRAIN          (uint32_t)0xcccccccc 

#define GPIO_CR_AFOUT_PP2MHz          (GPIO_CR_MODE_2MHz | GPIO_CR_AFO_PUSHPULL)  // 复用推挽输出,2MHz
#define GPIO_CR_AFIN_FLOAT            (uint32_t)0x44444444  // 复用开漏输入
#define GPIO_CR_AFIN_PULLDOWN         (uint32_t)0x88888888  // 复用上拉下拉输入


#define USART_CR1_REST                (uint32_t)0x00000000
#define USART_CR2_REST                (uint32_t)0x00000000
#define USART_CR3_REST                (uint32_t)0x00000000
常量定义
bubuko.com,布布扣
/**
  ********************************************************************
  *
  * @file     serial.c
  * @author   fpack
  * @version  v1.0
  * @date     2014-9-1
  * @brief    小穆妹纸串口调试
  *
  ********************************************************************
  **/

#include <stdio.h>
#include "armsis.h"

/*----------------------------------------------------------------------------
 Define  Baudrate setting (BRR) for USART
 *----------------------------------------------------------------------------*/
#define __DIV(__PCLK, __BAUD)       ((__PCLK*25)/(4*__BAUD))
#define __DIVMANT(__PCLK, __BAUD)   (__DIV(__PCLK, __BAUD)/100)
#define __DIVFRAQ(__PCLK, __BAUD)   (((__DIV(__PCLK, __BAUD) - (__DIVMANT(__PCLK, __BAUD) * 100)) * 16 + 50) / 100)
#define __USART_BRR(__PCLK, __BAUD) ((__DIVMANT(__PCLK, __BAUD) << 4)|(__DIVFRAQ(__PCLK, __BAUD) & 0x0F))


//struct __FILE { int handle; /* Add whatever you need here */ };
//FILE __stdout;
//FILE __stdin;

int fputc(int ch, FILE *f)
{
  // 等待USART1 数据发送完成(发送区域空)
  while (!(USART1->SR & USART_SR_TXE));
  USART1->DR = (ch & 0x1FF);
  
  return (ch);
}


void serial_init(void)
{
  //
  //  设置串口调试
  //   
  //     输  出: USART1
  //     引  脚: PA9(TX), PA10(RX)
  //     波特率: 9600
  //     数据位: 8 bit (default)  (CR1)
  //     校  验: none  (default)  (CR1)
  //     停止位: 1 bit (default)  (CR2)
  //     流控制: none  (default)  (CR3)
  //
  // 清除设置后上面配置为系统默认状态
  
  int i;
  /// 使能复用功能,使能GPIOA,使能USART1
  RCC->APB2ENR |= RCC_APB2ENR_AFIOEN | RCC_APB2ENR_IOPAEN | RCC_APB2ENR_USART1EN;
  // 关闭映射,确保USART使用 PA9,PA10
  AFIO->MAPR &= ~AFIO_MAPR_USART1_REMAP;
  
  // 清除PA9,PA10状态
  GPIOA->CRH &= ~( GPIO_CRH_CNF9 | GPIO_CRH_MODE9 | 
                   GPIO_CRH_CNF10 | GPIO_CRH_MODE10 );
  // 设置PA9 发送 为复用推挽输出 2MHz
  GPIOA->CRH |= GPIO_CR_AFOUT_PP2MHz & ( GPIO_CRH_CNF9 | GPIO_CRH_MODE9 );
  // 设置PA10接收 为复用上拉下拉模式
  GPIOA->CRH |= GPIO_CR_AFIN_PULLDOWN & ( GPIO_CRH_CNF10 | GPIO_CRH_MODE10 );
  

  // 设置波特率为 9600
  // 计算方法
  //          系统时钟 / (16分频 * 波特率)
  //   Baud = 72,000,000 / (16 * 9600) = 468.75
  //   整数部分 << 4 + 取整(小数部分 * 16)
  //   468 << 4 + 0.75 * 16
  USART1->BRR = __USART_BRR(SystemCoreClock, 9600);
  
  // 清除寄存器状态
  USART1->CR1 = USART_CR1_REST;
  USART1->CR2 = USART_CR2_REST;           // 停止位 1
  USART1->CR3 = USART_CR3_REST;           // 没用控制流
  
  // 防止产生不必要的信息
  for (i = 0; i < 0x1000; i++) __NOP();
  
  // USART1 使能, 使能输出,使能输入
  USART1->CR1 =  USART_CR1_UE | USART_CR1_TE | USART_CR1_RE;
  
}
整理后的串口配置代码 Serial.c
bubuko.com,布布扣
/**
  ********************************************************************
  *
  * @file     serial.h
  * @author   fpack
  * @version  v1.0
  * @date     2014-9-1
  * @brief    小穆妹纸串口调试
  *
  ********************************************************************
  **/


#ifndef __SERIAL_H__
#define __SERIAL_H__

#ifdef __cplusplus
  extern "C" {
#endif

void serial_init(void);


#ifdef __cplusplus    
  }
#endif

#endif   
串口调试头 Serial.h

 

STM32调试大法 之 串口通讯

标签:style   blog   http   color   os   io   使用   strong   ar   

原文地址:http://www.cnblogs.com/gleam/p/3954220.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!