打印
[STM32F4]

STM32F407 采用串口+DMA方式接收传感器数据有误

[复制链接]
602|15
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
yszong|  楼主 | 2021-2-7 23:43 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
STM32F407 采用串口+DMA方式接收传感器数据有误

使用特权

评论回复
沙发
llljh| | 2021-2-7 23:46 | 只看该作者

楼主程序可以公开吗?贴程序看下吧,这么说看不出什么原因

使用特权

评论回复
板凳
yszong|  楼主 | 2021-2-7 23:48 | 只看该作者
以下是部分代码
void USART_Configuration(uint8_t no)
{                                                                                                
  GPIO_InitTypeDef GPIO_InitStructure;
  USART_InitTypeDef USART_InitStructure;
  NVIC_InitTypeDef NVIC_InitStructure;
  DMA_InitTypeDef DMA_InitStruct;

//#if        PRINT_EN
  RCC_AHB1PeriphClockCmd(Open_USART_TX_GPIO_CLK,ENABLE);
  RCC_AHB1PeriphClockCmd(Open_USART_RX_GPIO_CLK,ENABLE);


#if        (USARTx_OPEN == 1 || USARTx_OPEN == 6)
  RCC_APB2PeriphClockCmd(Open_USART_CLK,ENABLE);
  RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);
  RCC_AHB1PeriphResetCmd(RCC_AHB1Periph_DMA2, ENABLE);
  RCC_AHB1PeriphResetCmd(RCC_AHB1Periph_DMA2, DISABLE);
#elif (USARTx_OPEN == 2 || USARTx_OPEN == 3 || USARTx_OPEN == 4 || USARTx_OPEN == 5)
  RCC_APB1PeriphClockCmd(Open_USART_CLK,ENABLE);
#endif

  GPIO_PinAFConfig(Open_USART_TX_GPIO_PORT, Open_USART_TX_SOURCE, Open_USART_TX_AF);
  GPIO_PinAFConfig(Open_USART_RX_GPIO_PORT, Open_USART_RX_SOURCE, Open_USART_RX_AF);

  /*
  *  Open_USART_TX -> PA9 , Open_USART_RX -PA10
  */
  GPIO_InitStructure.GPIO_Pin = Open_USART_TX_PIN;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
  GPIO_Init(Open_USART_TX_GPIO_PORT, &GPIO_InitStructure);

  GPIO_InitStructure.GPIO_Pin = Open_USART_RX_PIN;
  GPIO_InitStructure.GPIO_OType = GPIO_OType_OD;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
  GPIO_Init(Open_USART_RX_GPIO_PORT, &GPIO_InitStructure);

/*
                    USARTx configured as follow:
         - BaudRate = 115200 baud  
                 - Word Length = 8 Bits
         - One Stop Bit
         - No parity
         - Hardware flow control disabled (RTS and CTS signals)
         - Receive and transmit   
*/

  USART_InitStructure.USART_BaudRate = 115200;
  USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  USART_InitStructure.USART_StopBits = USART_StopBits_1;
  USART_InitStructure.USART_Parity = USART_Parity_No;
  USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  USART_Init(Open_USART, &USART_InitStructure);
  /* Enable the Open_USART Transmit interrupt: this interrupt is generated when the
     Open_USART transmit data register is empty */
  (USART_DMAy_Streamx[no-1], DISABLE);
                          //Check if the DMA Stream has been effectively disabled.
                          while (DMA_GetCmdStatus(USART_DMAy_Streamx[no-1]) != DISABLE);

                          //init DMA channel
                          DMA_StructInit(&DMA_InitStruct);
                          DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&USART_PORT[no-1]->DR;
                          DMA_InitStruct.DMA_Memory0BaseAddr = (uint32_t)&USART_RxBuffer[no-1][0];
                          DMA_InitStruct.DMA_BufferSize = USART_MAX_BUF_LEN;
                          DMA_InitStruct.DMA_Channel = USART_DMA_CHAN[no-1];
                          DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralToMemory;
                          DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
                          DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
                          DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
                          DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
                          DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
                          DMA_InitStruct.DMA_Priority = DMA_Priority_High;
                          DMA_InitStruct.DMA_FIFOMode = DMA_FIFOMode_Disable;
                          DMA_InitStruct.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
                          DMA_InitStruct.DMA_MemoryBurst = DMA_MemoryBurst_Single;
                          DMA_InitStruct.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
                          DMA_Init(USART_DMAy_Streamx[no-1], &DMA_InitStruct);
                          // Enable DMA Stream Transfer Complete interrupt
                          DMA_ITConfig(USART_DMAy_Streamx[no-1], DMA_IT_TC, ENABLE);
                          // DMA Stream enable
                          DMA_Cmd(USART_DMAy_Streamx[no-1], ENABLE);
                          //Check if the DMA Stream has been effectively enabled.
                          while (DMA_GetCmdStatus(USART_DMAy_Streamx[no-1]) != ENABLE);
                          /* Enable the DMA Stream IRQ Channel */
                          NVIC_InitStructure.NVIC_IRQChannel = USART_DMA_IRQn[no-1];
                          NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
                          NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
                          NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
                          NVIC_Init(&NVIC_InitStructure);
                        USART_ITConfig(USART_PORT[no-1], USART_IT_IDLE, ENABLE);
                        USART_ITConfig(Open_USART,USART_IT_RXNE,DISABLE);
                          USART_DMACmd(USART_PORT[no-1], USART_DMAReq_Rx, ENABLE);


                  USART_Cmd(USART_PORT[no-1], ENABLE);
                  USART_NVIC_Config(no);
//                  UartData[no-1].len = 0;
//                  UartData[no-1].callback = callback;



//USART_Cmd(Open_USART, ENABLE);
//  USART_NVIC_Config(USARTx_OPEN);
}

void USART_IRQHandler(uint8_t no)
{
        volatile char ch;
        if(no > 6)
                return;

        CoEnterISR();
        if(USART_GetITStatus(USART_PORT[no-1], USART_IT_RXNE) != RESET)
        {
                ch = USART_ReceiveData(USART_PORT[no-1]);       // get received character

                if(UartData[no-1].len >= USART_MAX_BUF_LEN)
                {
                        mem_cpy(&UartData[no-1].data[0], &UartData[no-1].data[1], USART_MAX_BUF_LEN-1);
                        UartData[no-1].data[USART_MAX_BUF_LEN-1] = ch;
                }
                else
                {
                        UartData[no-1].data[UartData[no-1].len] = ch;
                        UartData[no-1].len += 1;
                }
//
        }
        else if(USART_GetITStatus(USART_PORT[no-1], USART_IT_IDLE) != RESET)
        {
                DMA_InitTypeDef DMA_InitStruct;
                ch = USART_ReceiveData(USART_PORT[no-1]);        //clear IDLE flag by read USART_SR and USART_DR

                //get received data length
                UartData[no-1].len = USART_MAX_BUF_LEN - USART_DMAy_Streamx[no-1]->NDTR;
                mem_cpy(&UartData[no-1].data[0], &USART_RxBuffer[no-1][0], UartData[no-1].len);
                //analyze the received data and send out the response
                if(UartData[no-1].callback != NULL)
                        ((void (*)(uint8_t))UartData[no-1].callback)(no);
                //----------------
                switch(no)
                {
                        case 1:
                        case 6:
                                RCC_AHB1PeriphResetCmd(RCC_AHB1Periph_DMA2, ENABLE);
                                RCC_AHB1PeriphResetCmd(RCC_AHB1Periph_DMA2, DISABLE);
                                break;
                        case 2:
                        case 3:
                        case 4:
                        case 5:
                                RCC_AHB1PeriphResetCmd(RCC_AHB1Periph_DMA1, ENABLE);
                                RCC_AHB1PeriphResetCmd(RCC_AHB1Periph_DMA1, DISABLE);
                                break;
                }
                //----------------
                //disable DMA
                DMA_Cmd(USART_DMAy_Streamx[no-1], DISABLE);
                //Check if the DMA Stream has been effectively disabled.
                while (DMA_GetCmdStatus(USART_DMAy_Streamx[no-1]) != DISABLE);
                //init DMA channel
                DMA_StructInit(&DMA_InitStruct);
                DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&USART_PORT[no-1]->DR;
                DMA_InitStruct.DMA_Memory0BaseAddr = (uint32_t)&USART_RxBuffer[no-1][0];
                DMA_InitStruct.DMA_BufferSize = USART_MAX_BUF_LEN;
                DMA_InitStruct.DMA_Channel = USART_DMA_CHAN[no-1];
                DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralToMemory;
                DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
                DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
                DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
                DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
                DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
                DMA_InitStruct.DMA_Priority = DMA_Priority_High;
                DMA_InitStruct.DMA_FIFOMode = DMA_FIFOMode_Disable;
                DMA_InitStruct.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
                DMA_InitStruct.DMA_MemoryBurst = DMA_MemoryBurst_Single;
                DMA_InitStruct.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
                DMA_Init(USART_DMAy_Streamx[no-1], &DMA_InitStruct);
                // Enable DMA Stream Transfer Complete interrupt
                DMA_ITConfig(USART_DMAy_Streamx[no-1], DMA_IT_TC, ENABLE);
                // DMA Stream enable
                DMA_Cmd(USART_DMAy_Streamx[no-1], ENABLE);
                //Check if the DMA Stream has been effectively enabled.
                while (DMA_GetCmdStatus(USART_DMAy_Streamx[no-1]) != ENABLE);
        }
        CoExitISR();
}

问题:将串口1收到的传感器数据在经过串口1打印输出到屏幕上时,与传感器发送数据对不上;若将触感器直接通过串口工具传到串口调试助手上,数据正常,请问大神问题出在哪里?

使用特权

评论回复
地板
yinxiangh| | 2021-2-7 23:50 | 只看该作者
串口1收再串口1发,你串口1什么总线?挂了多少东西

使用特权

评论回复
5
zyf部长| | 2021-2-7 23:51 | 只看该作者
半双工的吗?

使用特权

评论回复
6
jiahy| | 2021-2-7 23:53 | 只看该作者

用示波器看看发送数据有没有问题

使用特权

评论回复
7
dengdc| | 2021-2-7 23:55 | 只看该作者

普通模式会有问题吗

使用特权

评论回复
8
zhaoxqi| | 2021-2-7 23:57 | 只看该作者
首先要找到是发送方的问题还是接收方的问题

使用特权

评论回复
9
chuxh| | 2021-2-9 22:25 | 只看该作者
时序安排不合理

使用特权

评论回复
10
llljh| | 2021-2-9 22:29 | 只看该作者
先发 别收 试试看

使用特权

评论回复
11
yszong|  楼主 | 2021-2-9 22:35 | 只看该作者

好的,我明天去单位试一下,多谢各位大侠了哈        

使用特权

评论回复
12
labasi| | 2021-3-2 21:06 | 只看该作者
发生的是什么错误啊

使用特权

评论回复
13
paotangsan| | 2021-3-2 21:08 | 只看该作者
改用普通模式会有问题吗

使用特权

评论回复
14
renzheshengui| | 2021-3-2 21:08 | 只看该作者
如何判定是时序的问题呢

使用特权

评论回复
15
wakayi| | 2021-3-2 21:09 | 只看该作者
串口只能是半双工啊

使用特权

评论回复
16
wowu| | 2021-3-2 21:10 | 只看该作者
中间加一个小小的延时

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

830

主题

11379

帖子

4

粉丝