zoukankan      html  css  js  c++  java
  • 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){} //等待发送缓冲区空才能发送下一个字符

  • 相关阅读:
    2021.1.20 学习总结
    2021.1.19 学习总结
    2021.1.18 学习总结
    学习总结13
    学习总结12
    学习总结11
    学习总结10
    ASP.NET网站开发步骤
    学习总结9
    阅读笔记4
  • 原文地址:https://www.cnblogs.com/dadidelearning/p/10046137.html
Copyright © 2011-2022 走看看