0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

FreeRTOS的任务挂起以恢复

汽车电子技术 来源:玩转单片机 作者: Julian 2023-02-10 15:06 次阅读

挂起可以理解为暂时停止任务,恢复可以理解为从新启动挂起的任务

挂起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
    API
    +关注

    关注

    2

    文章

    1437

    浏览量

    61234
  • 函数
    +关注

    关注

    3

    文章

    4118

    浏览量

    61550
收藏 人收藏

    评论

    相关推荐

    FreeRTOS任务挂起恢复与使用中断遇到的坑

    任务挂起简单点理解就是现在不需要执行这个任务,让它先暂停,就是挂起恢复就是从刚才挂起的状态下继
    的头像 发表于 09-26 09:01 5129次阅读
    <b class='flag-5'>FreeRTOS</b><b class='flag-5'>任务</b><b class='flag-5'>挂起</b><b class='flag-5'>恢复</b>与使用中断遇到的坑

    FreeRTOS中的任务管理

    任务FreeRTOS 中最基本的调度单元,它是一段可执行的代码,可以独立运行。FreeRTOS 中的任务是基于优先级的抢占式调度,优先级高的任务
    的头像 发表于 11-27 17:03 523次阅读

    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

    uCOS-III任务挂起恢复

    任务挂起恢复挂起恢复挂起恢复任务
    发表于 01-20 06:51

    FreeRTOS任务挂起与删除的区别在哪

    当一个任务暂时需要停止运行,那么就可以将任务挂起,在需要运行的时候在恢复就可以了。任务恢复运行以
    发表于 01-21 11:02

    FreeRTOS任务的使用

    FreeRTOS学习笔记(二):任务创建/删除,挂起/解挂上篇文章介绍了任务相关的基础知识,本篇文章对FreeRTOS
    发表于 02-18 07:14

    UCOS扩展例程-UCOSIII任务挂起恢复

    UCOS扩展例程-UCOSIII任务挂起恢复
    发表于 12-14 17:24 17次下载

    FreeRTOS笔记(四):任务创建/删除,挂起/解挂详解

    FreeRTOS笔记(四):任务创建/删除,挂起/解挂详解在第二篇笔记中介绍了任务创建的API,并且简单使用了相关API,本文将详细介绍任务
    发表于 12-04 19:36 15次下载
    <b class='flag-5'>FreeRTOS</b>笔记(四):<b class='flag-5'>任务</b>创建/删除,<b class='flag-5'>挂起</b>/解挂详解

    #FreeRTOS学习笔记(二):任务创建/删除,挂起/解挂

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

    (一)FreeRTOS学习之FreeRTOS任务基础知识

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

    FreeRTOS系列第11篇---FreeRTOS任务控制

    FreeRTOS任务控制API函数主要实现任务延时、任务挂起、解除任务
    发表于 01-26 17:54 12次下载
    <b class='flag-5'>FreeRTOS</b>系列第11篇---<b class='flag-5'>FreeRTOS</b><b class='flag-5'>任务</b>控制