打印
[应用相关]

【转】STM32的串口DMA

[复制链接]
733|3
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
Mattheww|  楼主 | 2018-7-27 20:12 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
1.我们知道DMA可以自动的不在CPU干预下,自动把数据重外设存储到内存(我们这节讲的),内存到外设,内存到内存等。但是DMA接收的是指定长度的,在接收不定长数据的时候DMA就傻眼了。网上有许多方法讲解运用定时器超时检测来接收不定长数据,而我们现在要讲的是运用串口空闲中断+DMA的方式接收不定长数据。

2.我们调试用的是串口1、DMA_Channel_4。具体的配置见下面程序:

DMA接收配置:
void USART1_DMA_Config(void)
{
    DMA_InitTypeDef DMA_InitStructure;

        RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);//开启DMA时钟
        DMA_InitStructure.DMA_Channel = DMA_Channel_4;//通道4  
        DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)USART1_DR_Address; //外设地址为[#define USART1_DR_Address    (0x40011000+0x04)]
        DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)Data_Buffer; //内存地址
        DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;//外设到内存
        DMA_InitStructure.DMA_BufferSize = 65535; //缓冲大小
        DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址不增
        DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;//内存地址增
        DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;//外设数据大小1byte
        DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;//内存数据大小1byte
        DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;//循环模式
        DMA_InitStructure.DMA_Priority = DMA_Priority_High;//优先级高
        DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;// 不开fifo
        DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull; //
        DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;//
        DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;//
        DMA_Init(DMA2_Stream5, &DMA_InitStructure);//初始化dma2流5
        DMA_Cmd(DMA2_Stream5, ENABLE);//开启dma2流5
}  
复制代码


串口1相关配置及空闲中断配置:

void STM_EVAL_COMInit(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    NVIC_InitTypeDef   NVIC_InitStructure;
    USART_InitTypeDef USART_InitStructure;  
    /* Enable GPIO clock */
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
    /* Enable UART clock */
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
    /* Connect PXx to USARTx_Tx*/
    GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_USART1);
    /* Connect PXx to USARTx_Rx*/
    GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_USART1);
    /* Configure USART Tx as alternate function  */
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    /* Configure USART Rx as alternate function  */
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    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 configuration */
    USART_Init(USART1, USART_InitStructure);
    NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 11;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);
    USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);
    /* Enable USART */
    USART_Cmd(USART1, ENABLE);
}  
复制代码



串口相关配置总函数:
void USART_CONFIG(void)
{
  STM_EVAL_COMInit();
  USART1_DMA_Config();//主要是配置外设地址和内存地址以及缓冲区大小,配置好后DMA就会自动的把串口数据存到相应的内存地址。
  USART_DMACmd(USART1, USART_DMAReq_Rx, ENABLE);配置串口1DMA接收
}  
复制代码


串口1空闲中断服务函数:

void USART1_IRQHandler(void)
{
    u16 i;
    if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET)//如果为空闲总线中断
    {
        i = USART1->SR;
        i = USART1->DR;
        USART_ClearITPendingBit(USART1, USART_IT_IDLE);
        DMA_Cmd(DMA2_Stream5, DISABLE);//关闭DMA,防止处理其间有数据
        Data_Len=65535-DMA_GetCurrDataCounter(DMA2_Stream5);
        data_recv_flag_set(); //标志位
        printf("Data_Len=%d\n\r",Data_Len);
        //DMA_Cmd(DMA2_Stream5, ENABLE);//开启DMA
    }
}  
复制代码


这里需要说明下,我们在配置DMA的时候,数据大小是一个字节,内存缓冲区为65535(特别强调下:DMA最大的缓冲区为65535,再大就溢出了。YY下,如果DMA接收缓冲区上M那就更加好了)。
函数DMA_GetCurrDataCounter(DMA2_Stream5);是获取当前指针计数值,用内存缓冲区大小 - 此计数值 = 接收到的数据长度(这里单位为字节)。需要说明下在读取数据长度的时候需要先把接收DMA关闭,读取完了或者是数据处理完了在打开接收DMA,防止在处理的过程中有数据到来而出错。

大家可以用串口调试助手测试下,记住把旁边的send as hex打上勾,然后发送不定长数据,每次发送完就会进空闲中断,并把接收的数据长度打印出来。如果需要读取数据,可以从Data_Buffer(前面DMA配置的内存地址)中读取出来。
沙发
wahahaheihei| | 2018-7-28 22:31 | 只看该作者
我还以为配置好有数据就自动传,没数据就等候呢

使用特权

评论回复
板凳
mmuuss586| | 2018-7-29 10:38 | 只看该作者
感谢分享;

使用特权

评论回复
地板
東南博士| | 2018-7-29 19:44 | 只看该作者
串口空闲中断

也可以接受 不定长度的 任意串口啊 而且 更简单 配合乒乓操作!

如果是 DMA 例如 串口 5 可能就不具备 DMA

使用特权

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

本版积分规则

85

主题

500

帖子

0

粉丝