AD+DMA+USART实验中的收获和总结

  • 由于实验室用的是USART3接口,但是在基地实验时,由于没有RS232,只能换到USART1,进行实验。(在交作业的时候,记得要再换回去)

在这个过程中,遇到困难,用串口软件发送数据时无响应,应该意味着我的数据没有发成功。

仔细排查,发现,由串口1到串口3,需要改动以下部分。

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);//USART1

RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART1, ENABLE);//USART3

      //---- USART1_TX   PA9     ----------------
  USART_GPIO_Init.GPIO_Pin = GPIO_Pin_9; 
  USART_GPIO_Init.GPIO_Speed = GPIO_Speed_50MHz;
  USART_GPIO_Init.GPIO_Mode = GPIO_Mode_AF_PP;      //复用推挽输出
  GPIO_Init(GPIOA, &USART_GPIO_Init);               //初始化PA9
   
     //-----USART1_RX   PA10     -------------
  USART_GPIO_Init.GPIO_Pin = GPIO_Pin_10;
  USART_GPIO_Init.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入
  GPIO_Init(GPIOA, &USART_GPIO_Init);                //初始化PA10

//----------------------------------------------------------------------------------
      //---- USART3_TX   PB10     ----------------
  USART_GPIO_Init.GPIO_Pin = GPIO_Pin_10; 
  USART_GPIO_Init.GPIO_Speed = GPIO_Speed_50MHz;
  USART_GPIO_Init.GPIO_Mode = GPIO_Mode_AF_PP;      //复用推挽输出
  GPIO_Init(GPIOB, &USART_GPIO_Init);               //初始化PA9
   
     //-----USART3_RX   PB11     -------------
  USART_GPIO_Init.GPIO_Pin = GPIO_Pin_11;
  USART_GPIO_Init.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入
  GPIO_Init(GPIOB, &USART_GPIO_Init);                //初始化PA10

USART1需要配置PA9及PA10
由于USART属于可选功能项
所以TX要设置成GPIO_Mode_AF_PP;,即复用推挽输出
RX用于数据输入,被设置成浮空输入

USART2同理。

最后不要忘了:

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);    //使能USART1,GPIOA时钟

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);    //使能USART1,GPIOB时钟
  • 在使用TIM的时候要在RCCset函数中打开使能
            RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); 
  •  发完一个数据之后,TXE自动置为1,那就会立刻开始下面的中断吗?中断是硬件帮我实现的事情,CPU应该会继续往下执行。{???}
  • TXE和TC标志位详细说明----仔细阅读

    在USART的发送端有2个寄存器,一个是程序可以看到的USART_DR寄存器(下图中阴影部分的TDR),另一个是程序看不到的移位寄存器(下图中阴影部分Transmit Shift Register)。

    对应USART数据发送有两个标志,一个是TXE=发送数据寄存器空,另一个是TC=发送结束;对照下图,当TDR中的数据传送到移位寄存器后,TXE被设置,此时移位寄存器开始向TX信号线按位传输数据,但因为TDR已经变空,程序可以把下一个要发送的字节(操作USART_DR)写入TDR中,而不必等到移位寄存器中所有位发送结束,所有位发送结束时(送出停止位后)硬件会设置TC标志。

    另一方面,在刚刚初始化好USART还没有发送任何数据时,也会有TXE标志,因为这时发送数据寄存器是空的。

    TXEIE和TCIE的意义很简单,TXEIE允许在TXE标志为'1'时产生中断,而TCIE允许在TC标志为'1'时产生中断。

    至于什么时候使用哪个标志,需要根据你的需要自己决定。但我认为TXE允许程序有更充裕的时间填写TDR寄存器,保证发送的数据流不间断。TC可以让程序知道发送结束的确切时间,有利于程序控制外部数据流的时序。

  • 1. 问题及现象

    使用USART_SendData()函数非连续发送单个字符是没有问题的;当连续发送字符时(两个字符间没有延时),就会发现发送缓冲区有溢出现象。若发送的数据量很小时,此时串口发送的只是最后一个字符,当发送数据量大时,就会导致发送的数据莫名其妙的丢失。

    2. 原因

    此API函数不完善,函数体内部没有一个判断一个字符是否发送完毕的语句,而是把数据直接放入发送缓冲区,当连续发送数据时,由于发送移位寄存器的速度限制(与通信波特率有关),导致发送缓冲区的数据溢出,老的数据还未及时发送出去,新的数据又把发送缓冲区的老数据覆盖了。

    3. 解决方法

    发送后等待一段时间延迟的方法就不说了,等待时间不确定,此为下下策。提供下面2种方案:

    方案1. 在每一个字符发送后检测状态位

    USART_SendData(USART1, RxBuffer[TxCounter]);

    while(USART_GetFlagStatus(USARTx, USART_FLAG_TXE) == RESET){} //等待发送缓冲区空才能发送下一个字符

原文地址:https://www.cnblogs.com/dadidelearning/p/10046137.html