STM32 串口+DMA的使用

STM32串口+DMA使用

        STM32有5个串口资源(USART1,USART2,USART3及UART4,UART5)。其中3个USART(通用同步/异步收/发器universalsynchronous asynchronous receiver and transmitter);2个UART(通用异步收/发器universalasynchronous receiver and transmitter);至于USART与UART的区别,如果只是拿来做串口用,没什么区别,在车载项目里,我们拿来做串口用,USART与UART在编程上并没有区别。

        其中USART1,USART2,USART3,UART4支持DMA方式,UART5不支持DMA。(详见数据手册stm32f105&107_datesheet_English的P18/2.3.17)。

        DMA(Derect MemoryAcess直接存储器存取),STM32有2个DMA,DMA1有7个通道,DMA2有5和通道,每个通道对应不同的外设(详见数据手册P272/13.3.7)。


1.时钟RCC配置:

串口时钟 + DMA时钟 + IO时钟

staticvoid RCC_Configuration(void)

{

       RCC_APB1PeriphClockCmd(RCC_APB1Periph_UART4,ENABLE);  //串口时钟       

       RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA2,ENABLE);  //DMA2时钟 

       RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC| RCC_APB2Periph_AFIO, ENABLE);//IO时钟

}

 

2.GPIO配置

UART4TXPC10脚,发送端配置为复用推挽输出模式(GPIO_Mode_AF_PP

UART4RXPC11脚,接收端配置为浮空输入模式(GPIO_Mode_IN_FLOATING

staticvoid GPIO_Configuration(void)

{

       GPIO_InitTypeDef GPIO_InitStructure;

                           

       GPIO_InitStructure.GPIO_Pin =GPIO_Pin_10;              

       GPIO_InitStructure.GPIO_Mode =GPIO_Mode_AF_PP;  //TX复用推挽输出模式

       GPIO_InitStructure.GPIO_Speed =GPIO_Speed_50MHz;

       GPIO_Init(GPIOC,&GPIO_InitStructure);               

 

       GPIO_InitStructure.GPIO_Pin =GPIO_Pin_11;             

       GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //RX浮空输入模式

       GPIO_InitStructure.GPIO_Speed =GPIO_Speed_50MHz;

       GPIO_Init(GPIOC,&GPIO_InitStructure);

}

 

3.中断NVIC配置:

配置两个DMA通道中断:

UART4RXDMA通道为DMA2的通道3

UART4TXDMA通道为DMA2的通道5

staticvoid NVIC_Configuration(void)

{

    NVIC_InitTypeDef NVIC_InitStructure;

 

    NVIC_InitStructure.NVIC_IRQChannel =UART4_IRQn;//串口中断

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

    NVIC_InitStructure.NVIC_IRQChannelSubPriority= 0;

    NVIC_InitStructure.NVIC_IRQChannelCmd =ENABLE;

    NVIC_Init(&NVIC_InitStructure);

      

    NVIC_InitStructure.NVIC_IRQChannel = DMA2_Channel3_IRQn;

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;

    NVIC_InitStructure.NVIC_IRQChannelCmd =ENABLE;

    NVIC_Init(&NVIC_InitStructure);

      

    NVIC_InitStructure.NVIC_IRQChannel = DMA2_Channel5_IRQn;

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; 

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; 

    NVIC_InitStructure.NVIC_IRQChannelCmd= ENABLE; 

 

    NVIC_Init(&NVIC_InitStructure);

}

 

4.串口配置:

即填充串口配置结构体

staticvoid UART4_Configuration(void)

{

       USART_InitTypeDef USART_InitStructure;

      

       USART_InitStructure.USART_BaudRate =115200;

       USART_InitStructure.USART_WordLength =USART_WordLength_8b;//数据位8位

       USART_InitStructure.USART_StopBits =USART_StopBits_1;//停止位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;//TX、RX都开启

 

       USART_Init(UART4,&USART_InitStructure);

       USART_Cmd(UART4, ENABLE); //使能UART4外设

}

 

5.DMA配置:

DMA可以把数据从外设转移到内存(如串口接收的时候),也可以从内存转移到外设(如串口发送的时候);不同方向的数据转移要各做相应的配置

 

串口接收:

voidUART4_Start_DMA_Recv(void * recvBuf, uint32_t bufLen)

{

    DMA_InitTypeDef DMA_InitStructure;

   

    UART4_Configuration();

   

    /* DMA1 Channel5 (triggered by USART1 Rxevent) Config */

    DMA_InitStructure.DMA_PeripheralBaseAddr =(u32)&(UART4->DR);//外设基地址,串口4数据寄存器

    DMA_InitStructure.DMA_MemoryBaseAddr =(u32)recvBuf;//内存基地址,数组UART4_DMA_HeadBuf

    DMA_InitStructure.DMA_DIR =DMA_DIR_PeripheralSRC;//SRC外设到内存

    DMA_InitStructure.DMA_BufferSize =bufLen;//DMA数据传输长度

    DMA_InitStructure.DMA_PeripheralInc =DMA_PeripheralInc_Disable;//外设地址不自增

    DMA_InitStructure.DMA_MemoryInc =DMA_MemoryInc_Enable;//内存地址自增

    DMA_InitStructure.DMA_PeripheralDataSize =DMA_PeripheralDataSize_Byte;//外设数据单位为1字节

    DMA_InitStructure.DMA_MemoryDataSize =DMA_MemoryDataSize_Byte;//内存数据单位为1字节

    DMA_InitStructure.DMA_Mode =DMA_Mode_Normal;//DMA传输数据模式,正常模式,传一轮

    DMA_InitStructure.DMA_Priority =DMA_Priority_High;//DMA通道优先级

    DMA_InitStructure.DMA_M2M =DMA_M2M_Disable;//禁止DMA内存到内存传输

      

    DMA_DeInit(DMA2_Channel3);//UART4RXDMA2通道3

    DMA_Init(DMA2_Channel3,&DMA_InitStructure);

    DMA_ITConfig(DMA2_Channel3, DMA_IT_TC,ENABLE);//配置DMA2发送完成后产生中断

    USART_DMACmd(UART4, USART_DMAReq_Rx,ENABLE);//配置串口向DMA发出Tx请求,请求传输数据

    DMA_Cmd(DMA2_Channel3, ENABLE);//正式开启DMA

}

 

串口发送:

voidUART4_Start_DMA_Send(void * sendBuf, uint32_t bufLen)

{

     DMA_InitTypeDefDMA_InitStructure;

 

    if (bufLen == 0)

        return ;

        

        memcpy(UART4_DMA_SendBuf, sendBuf, bufLen);

             

    DMA_InitStructure.DMA_PeripheralBaseAddr =(u32)(&UART4->DR);//外设基地址,串口4数据寄存器

    DMA_InitStructure.DMA_MemoryBaseAddr =(uint32_t)UART4_DMA_SendBuf; 

    DMA_InitStructure.DMA_DIR =DMA_DIR_PeripheralDST;//DST内存到外设 

    DMA_InitStructure.DMA_BufferSize =bufLen; 

    DMA_InitStructure.DMA_PeripheralInc =DMA_PeripheralInc_Disable; 

    DMA_InitStructure.DMA_MemoryInc =DMA_MemoryInc_Enable; 

    DMA_InitStructure.DMA_PeripheralDataSize =DMA_PeripheralDataSize_Byte; 

    DMA_InitStructure.DMA_MemoryDataSize =DMA_PeripheralDataSize_Byte; 

    DMA_InitStructure.DMA_Mode =DMA_Mode_Normal; 

    DMA_InitStructure.DMA_Priority =DMA_Priority_High; 

    DMA_InitStructure.DMA_M2M =DMA_M2M_Disable;

             

    DMA_DeInit(DMA2_Channel5); //UART4TXDMA2通道5

    DMA_Init(DMA2_Channel5,&DMA_InitStructure);          

    DMA_ITConfig(DMA2_Channel5, DMA_IT_TC,ENABLE);//配置DMA2发送完成后产生中断

    USART_DMACmd(UART4,USART_DMAReq_Tx,ENABLE);//配置串口向DMA发出Tx请求,请求传输数据

    DMA_Cmd(DMA2_Channel5, ENABLE);//正式开启DMA

   

    gDMA2Channel5Running = true;

 

}

 

6.DMA中断函数:

串口接收或发送的时候,DMA数据传输完成后会产生中断,在相应中断函数编写代码(注意中断函数名一定要与启动文件中断向量表一致)

 

DMA串口接收完成中断:

voidDMA2_Channel3_IRQHandler(void)//接收完成中断

{    

   

    OSIntEnter();//ucos进入中断服务函数

 

    if(DMA_GetITStatus(DMA2_IT_TC3))

    {

        //获取剩余长度,一般都为0,调试用

        DMA_ClearITPendingBit(DMA2_IT_GL3);   //清除全部中断标志  

        DMA_Cmd(DMA2_Channel3, DISABLE);

 

        if(0 == DMAReciveState)

        {

            if(HOST_MSG_START_CODE_FIRST_BYTE== UART4_DMA_HeadBuf[0])//比较读取的第1个字节FF

            {

                DMAReciveState = 1;

                UART4_Start_DMA_Recv((void*)(UART4_DMA_HeadBuf + 1), 3);//再读取后3个字节FFFFFF

            }

            else

            {

               UART4_Start_DMA_Recv((void*)UART4_DMA_HeadBuf, 1);

            }

        }

        else if(1 == DMAReciveState)

        {

            if(HOST_MSG_START_CODE ==*(uint32_t *)UART4_DMA_HeadBuf)//比较整个起始码FFFFFFFF

            {

                 DMAReciveState = 2;

                 UART4_Start_DMA_Recv((void*)(UART4_DMA_HeadBuf + 4), 8);//再读取后8个字节(cmdtype+bodylen)

            }

            else

            {

                DMAReciveState = 0;

                memset(UART4_DMA_HeadBuf, 0, sizeof(UART4_DMA_HeadBuf));

                UART4_Start_DMA_Recv((void*)UART4_DMA_HeadBuf, 1);

            }

        }

        else if(2 == DMAReciveState)

        {

            HOST_MSG_HEADER_T *pMCUMsgHeader;

            DMAReciveState = 3;

            pMsgBuffer = (uint8_t*)GetPhoneRecvBuf();//申请一个PhoneRecvBuf接收内存块

            if(pMsgBuffer == NULL)

            {

                DMAReciveState = 0;

                memset(UART4_DMA_HeadBuf, 0,sizeof(UART4_DMA_HeadBuf));

                UART4_Start_DMA_Recv((void*)UART4_DMA_HeadBuf, 1);

                OSIntExit();    //means get out of the inturrept!

                return;

            }

            memcpy(pMsgBuffer,UART4_DMA_HeadBuf, sizeof(HOST_MSG_HEADER_T));//把数组数据(startcode+cmdtype+bodylen)拷贝到内存块

            pMCUMsgHeader = (HOST_MSG_HEADER_T*)pMsgBuffer;//指针类型转换

           UART4_Start_DMA_Recv((void*)(pMsgBuffer +sizeof(HOST_MSG_HEADER_T)), pMCUMsgHeader->bodyLen + CRC_LEN);//再读取后面数据(data+crc)到内存块

        }

        else if(3 == DMAReciveState)

        {

            DMAReciveState = 0;

            memset(UART4_DMA_HeadBuf, 0,sizeof(UART4_DMA_HeadBuf));

            UART4_Start_DMA_Recv((void*)UART4_DMA_HeadBuf, 1);//此处再读1个字节(起始码第1个字节FF),开始下一轮接收数据

            if (pMsgBuffer != NULL)

            {

                PutMsg2PhoneRecvQueue(pMsgBuffer);//把PhoneRecvBuf接收内存块指针发送到PhoneRecvQ接收消息队列

                pMsgBuffer = NULL;

            }

        }

    }

 

    OSIntExit();//ucos退出中断服务函数

}

 

DMA串口发送完成中断:

voidDMA2_Channel5_IRQHandler(void)//发送完成中断

{

    OSIntEnter();

 

    if(DMA_GetITStatus(DMA2_IT_TC5)==SET)

    {

         DMA_ClearFlag(DMA2_IT_GL5);

         DMA_Cmd(DMA2_Channel5,DISABLE);

         gDMA2Channel5Running = false;

    }

   

    OSIntExit();

}

 

----------------------------------------------------------------------------------------------------------------------------------

附:

DMA常用库函数:

STM32 串口+DMA的使用_第1张图片

 

你可能感兴趣的:(STM32)