新聞中心

        EEPW首頁 > 嵌入式系統 > 設計應用 > STM32串口DMA超時接收方法,可大大節約CPU時間

        STM32串口DMA超時接收方法,可大大節約CPU時間

        作者: 時間:2016-11-19 來源:網絡 收藏
        本辦法使用定時器定時查詢DMA接收到的數據,如果超過設定的周期則認為本次數據包結束,將數據拷貝到緩沖區,交由其他程序處理??梢越邮杖我獯笮〉臄祿绕溥m用于MODBUS等協議,曾經用于GPS、GPRS等接收,很實用。本方法占用CPU時間極少,尤其是波特率很高時,效果更加明顯。
        當某一個串口的數據接收超時以后,定時器中斷中將數據拷貝到緩沖區,在主程序中可以判斷數據標志UART1_Flag,大于0的時候即代表有數據接收到,可以處理,處理完后將此變量清零即可。
        兩個數據包間隔較小時,可以將定時器的周期調短些。

        //超時時間定義
        #define UART1_TimeoutComp 2//20ms
        #define UART2_TimeoutComp 10//100ms
        #define UART3_TimeoutComp 10//100ms

        #define SRC_USART1_DR (&(USART1->DR)) //串口接收寄存器作為源頭
        #define SRC_USART2_DR (&(USART2->DR)) //串口接收寄存器作為源頭
        #define SRC_USART3_DR (&(USART3->DR)) //串口接收寄存器作為源頭


        extern u16 UART1_Flag,UART2_Flag,UART3_Flag;
        extern u8 uart1_data[200],uart3_data[500],uart2_data[500];

        u8 UART1_Timeout,UART2_Timeout,UART3_Timeout;
        u16 UART1_FlagTemp,UART2_FlagTemp,UART3_FlagTemp;
        u8 uart1_data_temp[200],uart2_data_temp[500],uart3_data_temp[500];

        u16 uart1_Flag_last=0,uart2_Flag_last=0,uart3_Flag_last=0;

        //定時器初始化
        void TimerInit(void)
        {
        //定時器初始化數據結構定義
        TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
        //初始化定時器,用于超時接收,20ms

        //復位計數器
        TIM_DeInit(TIM2);

        TIM_TimeBaseStructure.TIM_Period = 100; //計數上限,100*100us = 10000us = 10ms
        TIM_TimeBaseStructure.TIM_Prescaler = 4799; //預分頻4800,48MHz主頻,分頻后時鐘周期100us
        TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;//不分頻
        TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//向上計數
        TIM_TimeBaseStructure.TIM_RepetitionCounter=0;
        //初始化
        TIM_TimeBaseInit(TIM2,&TIM_TimeBaseStructure);

        //清中斷
        TIM_ClearFlag(TIM2, TIM_FLAG_Update);


        //使能定時器中斷
        TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);
        TIM_UpdateDisableConfig(TIM2,DISABLE);
        //定時器清零
        TIM_SetCounter(TIM2,0);
        //定時器啟動
        TIM_Cmd(TIM2,ENABLE);
        }


        //DMA初始化,只列出一個通道,其他兩個通道相同
        void DMA5_Init(void)
        {
        DMA_InitTypeDef DMA_InitStructure;

        DMA_DeInit(DMA1_Channel5); //將DMA的通道1寄存器重設為缺省值
        DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)SRC_USART1_DR; //源頭BUF既是 (&(USART1->DR))
        DMA_InitStructure.DMA_MemoryBaseAddr = (u32)uart1_data_temp; //目標BUF 既是要寫在哪個個數組之中
        DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //外設作源頭//外設是作為數據傳輸的目的地還是來源
        DMA_InitStructure.DMA_BufferSize = 200; //DMA緩存的大小 單位在下邊設定
        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_Circular; //工作在循環緩存模式
        DMA_InitStructure.DMA_Priority = DMA_Priority_High; //4優先級之一的(高優先)VeryHigh/High/Medium/Low
        DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //非內存到內存
        DMA_Init(DMA1_Channel5, &DMA_InitStructure); //根據DMA_InitStruct中指定的參數初始化DMA的通道1寄存器
        DMA_ITConfig(DMA1_Channel5, DMA_IT_TC, ENABLE); //DMA5傳輸完成中斷
        USART_DMACmd(USART1,USART_DMAReq_Rx,ENABLE); //使能USART1的接收DMA請求

        DMA_Cmd(DMA1_Channel5, ENABLE); //正式允許DMA
        }

        //串口初始化,只列出一個通道,其他兩個通道相同
        void USART1_Configuration(void)
        {
        //串口初始化數據結構定義
        USART_InitTypeDef USART_InitStructure;

        //初始化串口為38400,n,8,1
        USART_InitStructure.USART_BaudRate = 38400;
        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(USART1, &USART_InitStructure);

        //啟動串口,不需要接收中斷
        USART_Cmd(USART1, ENABLE);

        //默認設置為輸入狀態
        DMA5_Init();
        }

        //定時器中斷服務程序
        void TIM2_IRQHandler(void)
        {
        u16 i;
        //清定時器中斷
        TIM_ClearITPendingBit(TIM2, TIM_FLAG_Update);

        UART1_Timeout++;
        UART2_Timeout++;
        UART3_Timeout++;
        //------------------------------------------------------------------
        i=DMA_GetCurrDataCounter(DMA1_Channel5);
        DMA_ClearITPendingBit(DMA1_IT_GL5); //清除全部中斷標志

        if(i!=uart1_Flag_last)//未完成傳輸
        {
        UART1_Timeout=0;
        uart1_Flag_last=i;
        }
        else
        {
        if(UART1_Timeout>UART1_TimeoutComp)//產生超時
        {
        if(i<200) //有數據接收到
        {
        UART1_FlagTemp=200-i; //得到接收到的字節數

        for(i=0;i uart1_data[i]=uart1_data_temp[i];
        UART1_Flag=UART1_FlagTemp;

        DMA_ClearFlag(DMA1_FLAG_TC5);
        DMA_Cmd(DMA1_Channel5, DISABLE); //正式允許DMA
        DMA5_Init();
        }
        UART1_Timeout=0;
        }
        }
        //------------------------------------------------------------------
        i=DMA_GetCurrDataCounter(DMA1_Channel6);
        DMA_ClearITPendingBit(DMA1_IT_GL6); //清除全部中斷標志

        if(i!=uart2_Flag_last)//未完成傳輸
        {
        UART2_Timeout=0;
        uart2_Flag_last=i;
        }
        else
        {
        if(UART2_Timeout>UART2_TimeoutComp)//產生超時
        {
        if(i<500) //有數據接收到
        {
        UART2_FlagTemp=500-i;//得到接收到的字節數

        for(i=0;i uart2_data[i]=uart2_data_temp[i];
        UART2_Flag=UART2_FlagTemp;

        DMA_ClearFlag(DMA1_FLAG_TC6);
        DMA_Cmd(DMA1_Channel6, DISABLE); //正式允許DMA
        DMA6_Init();

        }
        UART2_Timeout=0;
        }
        }
        //------------------------------------------------------------------
        i=DMA_GetCurrDataCounter(DMA1_Channel3);
        DMA_ClearITPendingBit(DMA1_IT_GL3); //清除全部中斷標志

        if(i!=uart3_Flag_last)//未完成傳輸
        {
        UART3_Timeout=0;
        uart3_Flag_last=i;
        }
        else
        {
        if(UART3_Timeout>UART3_TimeoutComp)//產生超時
        {
        if(i<500) //有數據接收到
        {
        UART3_FlagTemp=500-i;//得到接收到的字節數

        for(i=0;i uart3_data[i]=uart3_data_temp[i];
        UART3_Flag=UART3_FlagTemp;

        DMA_ClearFlag(DMA1_FLAG_TC3);
        DMA_Cmd(DMA1_Channel3, DISABLE); //正式允許DMA
        DMA3_Init();

        }
        UART3_Timeout=0;
        }
        }
        }


        評論


        技術專區

        關閉
        主站蜘蛛池模板: 桦川县| 萨迦县| 壶关县| 旺苍县| 嘉义县| 通许县| 观塘区| 麻城市| 抚顺市| 九江市| 揭东县| 郴州市| 陆川县| 永兴县| 潜山县| 兰西县| 海伦市| 鄂伦春自治旗| 宁陵县| 惠安县| 通化市| 松潘县| 五河县| 区。| 罗田县| 盐城市| 雷波县| 化德县| 汉寿县| 卓资县| 永川市| 威宁| 钟祥市| 合水县| 东乌珠穆沁旗| 高陵县| 黄大仙区| 遵化市| 惠安县| 云安县| 宕昌县|