STM32中基于时间的任务调度框架

   日期:2021-02-01     浏览:105    评论:0    
核心提示:STM32中基于时间片的任务调度框架1.前言: 由于单片机只能单线程的进行工作,只是单纯在while循环中跑程序,导致效率很低,所以采用任务调度可以实现伪多线程工作,任务调度顾名思义就是在不同的时间点运行不同的程序。//我用的芯片是stm32f4时间片即CPU分配给各个程序的时间,每个线程被分配一个时间段,称作它的时间片,即该进程允许运行的时间,使各个程序从表面上看是同时进行的。如果在时间片结束时进程还在运行,则CPU将被剥夺并分配给另一个进程。如果进程在时间片结束前阻塞或结束,则CPU当即进行切换

STM32中基于时间片的任务调度框架

1.前言:

 由于单片机只能单线程的进行工作,只是单纯在while循环中跑程序,导致效率很低,所以采用任务调度可以实现伪多线程工作,任务调度顾名思义就是在不同的时间点运行不同的程序。//我用的芯片是stm32f4

时间片即CPU分配给各个程序的时间,每个线程被分配一个时间段,称作它的时间片,即该进程允许运行的时间,使各个程序从表面上看是同时进行的。如果在时间片结束时进程还在运行,则CPU将被剥夺并分配给另一个进程。如果进程在时间片结束前阻塞或结束,则CPU当即进行切换。而不会造成CPU资源浪费。在宏观上:我们可以同时打开多个应用程序,每个程序并行不悖,同时运行。但在微观上:由于只有一个CPU,一次只能处理程序要求的一部分,如何处理公平,一种方法就是引入时间片,每个程序按照时间点轮流执行

2.主要程序:

  1. 首先我们需要先配置一个定时器以及定时器中断
void TIMER_Configuration(void)
{ 
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;

	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5,ENABLE);		//使能TIM5时钟 84MHz

	TIM_TimeBaseInitStructure.TIM_Period = 999;					//自动重装载值
	TIM_TimeBaseInitStructure.TIM_Prescaler= 83;					//定时器分频
	TIM_TimeBaseInitStructure.TIM_CounterMode=TIM_CounterMode_Up;	//向上计数模式
	TIM_TimeBaseInitStructure.TIM_ClockDivision=TIM_CKD_DIV1;
	TIM_TimeBaseInit(TIM5,&TIM_TimeBaseInitStructure);				//初始化TIM5

	TIM_ITConfig(TIM5,TIM_IT_Update,ENABLE);			//允许定时器5更新中断
	TIM_Cmd(TIM5,ENABLE);								//使能定时器5

	NVIC_InitStructure.NVIC_IRQChannel=TIM5_IRQn;					//定时器5中断
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0x01;		//抢占优先级1
	NVIC_InitStructure.NVIC_IRQChannelSubPriority=0x03;				//子优先级3
	NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
	NVIC_Init(&NVIC_InitStructure);
}
unsigned char init_cnt = 0; //开关


void TIM5_IRQHandler(void)
{ 
	if(TIM_GetITStatus(TIM5,TIM_IT_Update)==SET)	//溢出中断
	{ 
		if(init_cnt == 1)
			Loop_Check();
	}
	TIM_ClearITPendingBit(TIM5,TIM_IT_Update);		//清除中断标志位
}
  1. 配置一个结构体以及定义一些函数
typedef struct
{ 
	unsigned short int Cnt_1ms;
	unsigned short int Cnt_2ms;
	unsigned short int Cnt_5ms;
	unsigned short int Cnt_10ms;
	unsigned short int Cnt_20ms;
	unsigned short int Cnt_50ms;
	
	unsigned char check_flag;
	unsigned char err_flag;
}LoopTypedef;


extern unsigned char init_cnt;


void TIMER_Configuration(void);


void Loop_Check(void);
void Duty_Loop(void);

void Duty_1ms(void);
void Duty_2ms(void);
void Duty_5ms(void);
void Duty_10ms(void);
void Duty_20ms(void);
void Duty_50ms(void);
  1. 之前定义过的函数的具体内容,通过Duty_Loop(void)这个函数来调用不同时间点的函数,不同时间点的函数里就放入我们所需要运行的程序
void Loop_Check(void)
{ 
	Loop.Cnt_2ms ++;
	Loop.Cnt_5ms ++;
	Loop.Cnt_10ms ++;
	Loop.Cnt_20ms ++;
	Loop.Cnt_50ms ++;
	
	if(Loop.check_flag == 1)
    { 
    	Loop.err_flag ++;
    }
    else
    { 
    	Loop.check_flag = 1;
    }
}



void Duty_Loop(void)
{ 
	if(Loop.check_flag == 1)
	{ 
		Duty_1ms();
		
		if(Loop.Cnt_2ms >=2)
		{ 
			Duty_2ms();
			Loop.Cnt_2ms = 0;
		}
		
		if(Loop.Cnt_5ms >=5)
		{ 
			Duty_5ms();
			Loop.Cnt_5ms = 0;
		}
		
		if(Loop.Cnt_10ms >=10)
		{ 
			Duty_10ms();
			Loop.Cnt_10ms = 0;
		}
		
		if(Loop.Cnt_20ms >=20)
		{ 
			Duty_20ms();
			Loop.Cnt_20ms = 0;
		}
		
		if(Loop.Cnt_50ms >=50)
		{ 
			Duty_50ms();
			Loop.Cnt_50ms = 0;
		}
		
		Loop.check_flag = 0;
	}
}


void Duty_1ms(void)
{ 



}

void Duty_2ms(void)
{ 
	
	
}

void Duty_5ms(void)
{ 
	

}

void Duty_10ms(void) 
{ 
	

	
}

void Duty_20ms(void)
{ 
	

}
void Duty_50ms(void)
{ 
	

}
  1. 再将任务调度整个框架放入while循环中即可
int main(void)
{ 
	init_cnt = 1;		//任务调度开关
	while(1)
	{ 
		Duty_Loop();  //任务调度 
	
	}
}
 
打赏
 本文转载自:网络 
所有权利归属于原作者,如文章来源标示错误或侵犯了您的权利请联系微信13520258486
更多>最近资讯中心
更多>最新资讯中心
0相关评论

推荐图文
推荐资讯中心
点击排行
最新信息
新手指南
采购商服务
供应商服务
交易安全
关注我们
手机网站:
新浪微博:
微信关注:

13520258486

周一至周五 9:00-18:00
(其他时间联系在线客服)

24小时在线客服