挂起可以理解为暂时停止任务,恢复可以理解为从新启动挂起的任务
挂起API函数(可以在tasks.c中找到)
vTaskSuspend( TaskHandle_t xTaskToSuspend )
xTaskToSuspend:需要挂起的任务句柄
非中断恢复API函数(可以在tasks.c中找到)
vTaskResume( TaskHandle_t xTaskToResume )
xTaskToSuspend:需要挂起的任务句柄
中断恢复API函数(可以在tasks.c中找到)
xTaskResumeFromISR( TaskHandle_t xTaskToResume )
xTaskToSuspend:需要挂起的任务句柄
注意:中断中不可以使用
vTaskDelay( const TickType_t xTicksToDelay )
实验目的
通过按键控制LED0任务的挂起与恢复
任务挂起与恢复
#include "stm32f10x.h"
#include "stm32f10x.h"
#include
#include "FreeRTOS.h"
#include "task.h"
uint8_t main_temp = 0;
void LED_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE); //使能PE端口时钟
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_5; //端口配置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度为50MHz
GPIO_Init(GPIOC, &GPIO_InitStructure); //推挽输出 ,IO口速度为50MHz
GPIO_SetBits(GPIOC,GPIO_Pin_1|GPIO_Pin_5); //输出高
}
void KEY_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure; //定义结构体变量
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0; //选择你要设置的IO口
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;//下拉输入
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; //设置传输速率
GPIO_Init(GPIOA,&GPIO_InitStructure); /* 初始化GPIO */
}
// 外部中断初始化
void My_EXTI_Init(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
EXTI_InitTypeDef EXTI_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0);//选择GPIO管脚用作外部中断线路
//EXTI0 NVIC 配置
NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn;//EXTI0中断通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2;//抢占优先级
NVIC_InitStructure.NVIC_IRQChannelSubPriority =3; //子优先级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器
EXTI_InitStructure.EXTI_Line=EXTI_Line0;
EXTI_InitStructure.EXTI_Mode=EXTI_Mode_Interrupt;
EXTI_InitStructure.EXTI_Trigger=EXTI_Trigger_Rising;
EXTI_InitStructure.EXTI_LineCmd=ENABLE;
EXTI_Init(&EXTI_InitStructure);
}
#define START_TASK_PRIO 1 //任务优先级
#define START_STK_SIZE 128 //任务堆栈大小
TaskHandle_t StartTask_Handler; //任务句柄
void start_task(void *pvParameters); //任务函数
#define LED0_TASK_PRIO 2 //任务优先级
#define LED0_STK_SIZE 50 //任务堆栈大小
TaskHandle_t LED0Task_Handler; //任务句柄
void led0_task(void *p_arg); //任务函数
#define LED1_TASK_PRIO 2 //任务优先级
#define LED1_STK_SIZE 50 //任务堆栈大小
TaskHandle_t LED1Task_Handler; //任务句柄
void led1_task(void *p_arg); //任务函数
#define KEY0_TASK_PRIO 2 //任务优先级
#define KEY0_STK_SIZE 50 //任务堆栈大小
TaskHandle_t KEY0Task_Handler; //任务句柄
void key0_task(void *p_arg); //任务函数
int main( void )
{
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//设置系统中断优先级分组 4
LED_Init(); //初始化 LED
KEY_Init();
My_EXTI_Init();
//创建开始任务
xTaskCreate(
(TaskFunction_t )start_task, //任务函数
(const char* )"start_task", //任务名称
(uint16_t )START_STK_SIZE, //任务堆栈大小
(void* )NULL, //传递给任务函数的参数
(UBaseType_t )START_TASK_PRIO, //任务优先级
(TaskHandle_t* )&StartTask_Handler //任务句柄
);
vTaskStartScheduler(); //开启调度
}
void start_task(void *pvParameters)
{
taskENTER_CRITICAL(); //进入临界区
//创建 LED0 任务
xTaskCreate(
(TaskFunction_t )led0_task,
(const char* )"led0_task",
(uint16_t )LED0_STK_SIZE,
(void* )NULL,
(UBaseType_t )LED0_TASK_PRIO,
(TaskHandle_t* )&LED0Task_Handler
);
//创建 LED1 任务
xTaskCreate(
(TaskFunction_t )led1_task,
(const char* )"led1_task",
(uint16_t )LED1_STK_SIZE,
(void* )NULL,
(UBaseType_t )LED1_TASK_PRIO,
(TaskHandle_t* )&LED1Task_Handler
);
//创建 KEY0 任务
xTaskCreate(
(TaskFunction_t )key0_task,
(const char* )"key0_task",
(uint16_t )KEY0_STK_SIZE,
(void* )NULL,
(UBaseType_t )KEY0_TASK_PRIO,
(TaskHandle_t* )&KEY0Task_Handler
);
vTaskDelete(StartTask_Handler); //删除开始任务
taskEXIT_CRITICAL(); //退出临界区
}
//LED0 任务函数
void led0_task(void *pvParameters)
{
while(1)
{
if(GPIO_ReadInputDataBit( GPIOC, GPIO_Pin_5))
{
GPIO_ResetBits( GPIOC, GPIO_Pin_5);
}
else
{
GPIO_SetBits( GPIOC, GPIO_Pin_5);
}
vTaskDelay(400);
}
}
//LED1 任务函数
void led1_task(void *pvParameters)
{
while(1)
{
if(GPIO_ReadInputDataBit( GPIOC, GPIO_Pin_1))
{
GPIO_ResetBits( GPIOC, GPIO_Pin_1);
}
else
{
GPIO_SetBits( GPIOC, GPIO_Pin_1);
}
vTaskDelay(200);
}
}
//KEY0 任务函数
void key0_task(void *pvParameters)
{
while(1)
{
if( main_temp == 0xff ){
//任务挂起(非中断)
//vTaskSuspend(LED0Task_Handler);
}
else{
//任务恢复(非中断)
//vTaskResume(LED0Task_Handler);
}
vTaskDelay(100);
}
}
//按键中断服务函数
void EXTI0_IRQHandler(void)
{
if(EXTI_GetITStatus(EXTI_Line0)==1)
{
if( main_temp ){
main_temp = 0x00;
//中断函数中进行任务恢复(区分中断)
xTaskResumeFromISR(LED0Task_Handler);
}
else{
main_temp = 0xff;
//任务挂起(谨慎在中断使用,虽然可以达到初步效果,不保证不会出问题)
vTaskSuspend(LED0Task_Handler);
}
}
EXTI_ClearITPendingBit(EXTI_Line0);
}
--END--
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
API
+关注
关注
2文章
1528浏览量
62714 -
函数
+关注
关注
3文章
4352浏览量
63261
发布评论请先 登录
相关推荐
FreeRTOS中的任务管理
任务是 FreeRTOS 中最基本的调度单元,它是一段可执行的代码,可以独立运行。FreeRTOS 中的任务是基于优先级的抢占式调度,优先级高的任务
FreeRTOS的任务无故进入挂起状态的原因?
请教大家一个问题,我子啊使用FreeRTOS的时候创建了一个以太网的任务,任务在使用的过程中被无故挂起,请问一下出现这种现象有哪几种原因呢?
发表于 04-09 07:20
FreeRTOS里在中断中挂起任务出错的原因?怎么解决?
各位大佬,新手刚学习FreeRTOS,现在想在中断中挂起某个任务,我在教程里看到说有中断中的恢复函数xTaskResumeFromISR,但是没有在中断中的
发表于 04-16 08:26
freeRTOS操作系统的任务挂起
freeRTOS在cntTask任务中,计数到10次之后便不再进入该任务为什么?请led4Task并没有被挂起,又是为什么?本人刚入门freeRTO
发表于 10-18 21:39
UCOSII的任务挂起与恢复问题
我遇到一个问题,我在用ucosII时,我任务A一直未被挂起,但我有一个任务B(优先级比A高,1ms触发一次),每执行一次任务B都会恢复一下
发表于 07-04 04:35
FreeRTOS任务挂起和恢复的相关资料推荐
任务挂起和恢复要使用着些API则需要使能宏定义:INCLUDE_vTaskSuspend、INCLUDE_xTaskResumeFromISR任务挂
发表于 12-27 08:06
FreeRTOS笔记(四):任务创建/删除,挂起/解挂详解
FreeRTOS笔记(四):任务创建/删除,挂起/解挂详解在第二篇笔记中介绍了任务创建的API,并且简单使用了相关API,本文将详细介绍任务
发表于 12-04 19:36
•15次下载

#FreeRTOS学习笔记(二):任务创建/删除,挂起/解挂
FreeRTOS学习笔记(二):任务创建/删除,挂起/解挂上篇文章介绍了任务相关的基础知识,本篇文章对FreeRTOS
发表于 12-23 19:56
•2次下载

(一)FreeRTOS学习之FreeRTOS任务基础知识
功能,初学者必须先掌握——任务的创建、删除、挂起和恢复等操作。本章节分为如下几部分:*什么是多任务系统*FreeRTOS
发表于 12-23 19:57
•3次下载

评论