当前位置:   article > 正文

UART编程(寄存器)_uart rx 输入状态寄存器

uart rx 输入状态寄存器

1. 串口编程步骤

1.1 看原理图确定引脚

  • 有很多串口,使用哪一个?看原理图确定

1.2 配置引脚为UART功能

  • 至少用到发送、接收引脚:txd、rxd

  • 需要把这些引脚配置为UART功能,并使能UART模块

1.3 设置串口参数

  • 有哪些参数?

    • 波特率

    • 数据位

    • 校验位

    • 停止位

  • 示例: 比如15200,8n1表示波特率为115200,8个数据为,没有校验位,1个停止位

1.4 根据状态寄存器读写数据

  • 肯定有一个数据寄存器,程序把数据写入,即刻通过串口向外发送数据

  • 肯定有一个数据寄存器,程序读取这个寄存器,就可以获得先前接收到的数据

  • 很多有状态寄存器

    • 判断数据是否发送出去?是否发送成功?

    • 判断是否接收到了数据?

2. STM32F103串口框架

各类芯片的UART框图都是类似的,当设置好UART后,程序读写数据寄存器就可以接收、发送数据了。

 

3. STM32F103串口操作

3.1 看原理图确定引脚

  • 100ASM STM32F103的USART1接到一个USB串口芯片,然后就可以通过USB线连接电脑了

  • 原理图如下

  • 上图中的USART1_RX、USART1_TX,接到了PA9、PA10

3.2 配置引脚为UART功能

3.2.1 使能GPIOA/USART1模块

需要设置GPIOA的寄存器,选择引脚功能:所以要使能GPIOA模块。 GPIOA模块、USART1模块的使能都是在同一个寄存器里实现。

3.2.2 配置引脚功能

从上图可以知道,PA9、PA10有三种功能:GPIO、USART1、TIMER1。

3.3 设置串口参数

3.3.1 设置波特率

波特率算公式:

USARTDIV由整数部分、小数部分组成,计算公式如下: USARTDIV = DIV_Mantissa + (DIV_Fraction / 16) DIV_Mantissa和DIV_Fraction来自USART_BRR寄存器,如下图:

3.3.2 设置数据格式

比如数据位设置为8,无校验位,停止位设置为1。 需要设置2个寄存器。

  • USART1_CR1:用来设置数据位、校验位,使能USART

  • USART_CR2:用来设置停止位

3.4 根据状态寄存器读写数据

  • 状态寄存器

  • 数据寄存器 写、读这个寄存器,就可:发送、读取串口数据,如下图:

3.5 USART1的寄存器地址

  • 基地址

  • USART寄存器 用结构体来表示比较方便:

    typedef unsigned int uint32_t;
    typedef struct
    {
      volatile uint32_t SR;    /*!< USART Status register, Address offset: 0x00 */
      volatile uint32_t DR;    /*!< USART Data register,   Address offset: 0x04 */
      volatile uint32_t BRR;   /*!< USART Baud rate register, Address offset: 0x08 */
      volatile uint32_t CR1;   /*!< USART Control register 1, Address offset: 0x0C */
      volatile uint32_t CR2;   /*!< USART Control register 2, Address offset: 0x10 */
      volatile uint32_t CR3;   /*!< USART Control register 3, Address offset: 0x14 */
      volatile uint32_t GTPR;  /*!< USART Guard time and prescaler register, Address offset: 0x18 */
    } USART_TypeDef;
    ​
    USART_TypeDef *usart1 = (USART_TypeDef *)0x40013800;

没有指名寄存器地址,因为结构体里面的变量的相对入口的地址(变量本身的地址)就是寄存器相对串口的偏移地址,串口指向这个变量就是指向这个寄存器 

4. 写程序

uart.h

  1. #ifndef _UART_H_
  2. #define _UART_H_
  3. typedef unsigned int uint32_t;
  4. typedef struct
  5. {
  6. volatile uint32_t SR; /*!< USART Status register, Address offset: 0x00 */
  7. volatile uint32_t DR; /*!< USART Data register, Address offset: 0x04 */
  8. volatile uint32_t BRR; /*!< USART Baud rate register, Address offset: 0x08 */
  9. volatile uint32_t CR1; /*!< USART Control register 1, Address offset: 0x0C */
  10. volatile uint32_t CR2; /*!< USART Control register 2, Address offset: 0x10 */
  11. volatile uint32_t CR3; /*!< USART Control register 3, Address offset: 0x14 */
  12. volatile uint32_t GTPR; /*!< USART Guard time and prescaler register, Address offset: 0x18 */
  13. } USART_TypeDef;
  14. void uart_init(void);
  15. char getchar(void);
  16. void putchar(char c);
  17. #endif

uart.c

  1. #include "uart.h"
  2. void uart_init(void)
  3. {
  4. USART_TypeDef *usart1 = (USART_TypeDef *)0x40013800;
  5. //使能GPIOA/USART1模块
  6. unsigned int *pRcc = (unsigned int *)(0x40021000 + 0x18);
  7. *pRcc |= (1<<2);
  8. pRcc = (unsigned int *)(0x40021000 + 0x18);
  9. *pRcc |= (1<<14);
  10. //配置引脚功能
  11. unsigned int *pMode = (unsigned int *)(0x40010800 + 0x04);
  12. *pMode &= ~((3<<4) | (3<<6));//PA9 -TX
  13. *pMode |= (1<<4) | (2<<6);
  14. *pMode &= ~((3<<8) | (3<<10));//PA10 -RX
  15. *pMode |= (0<<8) | (1<<10);
  16. /*设置波特率
  17. * 115200 = 8000000/16/USARTDIV
  18. * USARTDIV = 4.34
  19. * DIV_Mantissa = 4
  20. * DIV_Fraction / 16 = 0.34
  21. * DIV_Fraction = 16 * 0.34 = 5
  22. * 真实波特率:
  23. * DIV_Fraction / 16 = 5 / 16 = 0.3125
  24. * USARTDIV = DIV_Mantissa + DIV_Fraction / 16 = 4.3125
  25. * baudrate = 8000000/16/4.3125 = 115942
  26. */
  27. #define DIV_Mantissa 4
  28. #define DIV_Fraction 5
  29. usart1 -> BRR = (DIV_Mantissa << 4) | (5);
  30. //设置数据格式
  31. usart1 -> CR1 = (1 << 13) | (0 << 12) | (0 << 10) | (1 << 3) | (1 << 2);
  32. usart1 -> CR2 &= ~(0x11 << 12); //1位停止位
  33. //使能USART1
  34. //在配置CR1时已经使能tx,rx
  35. }
  36. char getchar(void)
  37. {
  38. USART_TypeDef *usart1 = (USART_TypeDef *)0x40013800;
  39. while((usart1 -> SR & (1 << 5)) == 0);//出来循环就是有数据
  40. return usart1 -> DR;
  41. }
  42. void putchar(char c)
  43. {
  44. USART_TypeDef *usart1 = (USART_TypeDef *)0x40013800;
  45. while((usart1 -> SR & (1 << 7)) == 0);//出来循环就是发送成功
  46. usart1 -> DR = c;
  47. }

main.c

  1. #include "uart.h"
  2. void delay(int time)
  3. {
  4. while(time --);
  5. }
  6. int main()
  7. {
  8. uart_init();
  9. char c;
  10. putchar('1');
  11. putchar('0');
  12. putchar('0');
  13. putchar('a');
  14. putchar('s');
  15. putchar('k');
  16. putchar('\n');
  17. putchar('\r');
  18. while(1)
  19. {
  20. c = getchar();
  21. putchar(c);
  22. putchar(c+1);
  23. }
  24. }

 测试结果:

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/855176
推荐阅读
相关标签
  

闽ICP备14008679号