您的位置 首页 电路

STM32串口DMA超时接纳办法,可大大节省CPU时刻

本办法使用定时器定时查询DMA接收到的数据,如果超过设定的周期则认为本次数据包结束,将数据拷贝到缓冲区,交由其他程序处理。可以接收任…

本办法运用守时器守时查询DMA接纳到的数据,假如超越设定的周期则以为本次数据包完毕,将数据拷贝到缓冲区,交由其他程序处理。能够接纳恣意巨细的数据包,特别适用于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;
}
}
}

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/fangan/dianlu/265297.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部