STM32 FreeROTS 任务创建和删除实验(静态方法)

实验目标

学会 xTaskCreateStatic( )和 vTaskDelete( ) 的使用:

  • start_task:用来创建其他的三个任务。
  • task1:实现LED1每500ms闪烁一次。
  • task2:实现LED2每500ms闪烁一次。

task3:判断按键KEY1是否按下,按下则删掉task1。

看代码

直接看代码修改,其余和动态创建和删除一样。

STM32 FreeRTOS 任务创建和删除实验(动态方法)-CSDN博客

FreeRTOSConfig.h代码清单

#define configSUPPORT_STATIC_ALLOCATION   1
/* 软件定时器相关定义 */
#define configUSE_TIMERS 1                                         
/* 使能软件定时器, 默认: 0。为1时需要定义下面3个 */
#define configTIMER_TASK_PRIORITY (configMAX_PRIORITIES - 1)        
/* 软件定时器任务的优先级 */
#define configTIMER_QUEUE_LENGTH 5                                  
/* 软件定时器命令队列的长度 */
#define configTIMER_TASK_STACK_DEPTH (configMINIMAL_STACK_SIZE * 2) 
/* 软件定时器任务的栈空间大小 */

可以自己创建一个软件定时器任务---这里是打开的

static allocation------静态分配

开启软件定时器

设置定时器的优先级这里是最高优先级

队列长度

栈空间的大小256个字

可以尝试编译---它会提示需要两个函数。

FreeRTOS_demo.h代码清单

#ifndef __FREERTOS_DEMO_H
#define __FREERTOS_DEMO_H

#include "FreeRTOS.h"
#include "task.h"

#include "led.h"
#include "key.h"
#include <stdio.h>
#include "stm32f1xx_hal.h"

// 主入口:启动(创建启动任务,开启调度器)
void FreeRTOS_Start(void);

#endif

FreeRTOS_demo.c代码清单

引入头文件
#include "FreeROTS_demo.h"
任务设置
// 1. 启动任务函数和相关参数
#define START_TASK_NAME "Start_Task"
#define START_TASK_STACK_DEPTH 128
#define START_TASK_PRIORITY 1
TaskHandle_t start_task_handler;
StackType_t start_task_stack[START_TASK_STACK_DEPTH]; // 任务堆栈(数组)
StaticTask_t start_task_tcb;                          // 任务控制块(TCB)
void Start_Task(void *pvParameters);

// 2. Task1 任务函数和相关参数
#define TASK1_NAME "Task1"
#define TASK1_STACK_DEPTH 128
#define TASK1_PRIORITY 2
TaskHandle_t task1_handler;
StackType_t task1_stack[TASK1_STACK_DEPTH]; // 任务堆栈(数组)
StaticTask_t task1_tcb;                     // 任务控制块(TCB)
void Task1(void *pvParameters);

// 3. Task2 任务函数和相关参数
#define TASK2_NAME "Task2"
#define TASK2_STACK_DEPTH 128
#define TASK2_PRIORITY 3
TaskHandle_t task2_handler;
StackType_t task2_stack[TASK2_STACK_DEPTH]; // 任务堆栈(数组)
StaticTask_t task2_tcb;                     // 任务控制块(TCB)
void Task2(void *pvParameters);

// 4. Task3 任务函数和相关参数
#define TASK3_NAME "Task3"
#define TASK3_STACK_DEPTH 128
#define TASK3_PRIORITY 4
TaskHandle_t task3_handler;
StackType_t task3_stack[TASK3_STACK_DEPTH]; // 任务堆栈(数组)
StaticTask_t task3_tcb;                     // 任务控制块(TCB)
void Task3(void *pvParameters);
空闲任务参数和内存分配函数的实现

函数是通过函数命名---找到tasks.c文件可以找到这个函数。

// 5. 空闲任务相关参数
StackType_t idle_task_stack[configMINIMAL_STACK_SIZE]; // 任务堆栈(数组)
StaticTask_t idle_task_tcb;                            // 任务控制块(TCB)

// 空闲任务内存分配
void vApplicationGetIdleTaskMemory(StaticTask_t **ppxIdleTaskTCBBuffer,
                                   StackType_t **ppxIdleTaskStackBuffer,
                                   uint32_t *pulIdleTaskStackSize)
{
    *ppxIdleTaskTCBBuffer = &idle_task_tcb;
    *ppxIdleTaskStackBuffer = idle_task_stack;
    *pulIdleTaskStackSize = configMINIMAL_STACK_SIZE;
}
定时器任务参数及函数实现

通过函数命名---可以明白这个函数是在timers.c这个文件中

// 5. 定时器任务相关参数
StackType_t timer_task_stack[configTIMER_TASK_STACK_DEPTH]; // 任务堆栈(数组)
StaticTask_t timer_task_tcb;                                // 任务控制块(TCB)
void vApplicationGetTimerTaskMemory(StaticTask_t **ppxTimerTaskTCBBuffer,
                                    StackType_t **ppxTimerTaskStackBuffer,
                                    uint32_t *pulTimerTaskStackSize)
{
    *ppxTimerTaskTCBBuffer = &timer_task_tcb;
    *ppxTimerTaskStackBuffer = timer_task_stack;
    *pulTimerTaskStackSize = configTIMER_TASK_STACK_DEPTH;
}
启动函数
void FreeRTOS_Start(void)
{
    printf("FreeRTOS启动!\n");

    // 1. 创建启动任务
    start_task_handler = xTaskCreateStatic(
        (TaskFunction_t)Start_Task,
        (char *)START_TASK_NAME,
        (uint32_t)START_TASK_STACK_DEPTH,
        (void *)NULL,
        (UBaseType_t)START_TASK_PRIORITY,
        start_task_stack,
        &start_task_tcb);

    // 2. 开启任务调度器
    vTaskStartScheduler();
}
启动任务的函数体
// 启动任务的函数体
void Start_Task(void *pvParameters)
{
    // 进入临界区
    taskENTER_CRITICAL();

    // 1. 依次创建任务
    task1_handler = xTaskCreateStatic(
        (TaskFunction_t)Task1,
        (char *)TASK1_NAME,
        (uint32_t)TASK1_STACK_DEPTH,
        (void *)NULL,
        (UBaseType_t)TASK1_PRIORITY,
        task1_stack,
        &task1_tcb);
    task2_handler = xTaskCreateStatic(
        (TaskFunction_t)Task2,
        (char *)TASK2_NAME,
        (uint32_t)TASK2_STACK_DEPTH,
        (void *)NULL,
        (UBaseType_t)TASK2_PRIORITY,
        task2_stack,
        &task2_tcb);
    task3_handler = xTaskCreateStatic(
        (TaskFunction_t)Task3,
        (char *)TASK3_NAME,
        (uint32_t)TASK3_STACK_DEPTH,
        (void *)NULL,
        (UBaseType_t)TASK3_PRIORITY,
        task3_stack,
        &task3_tcb);

    // 2. 删除任务(自身)
    vTaskDelete(NULL);

    // 退出临界区
    taskEXIT_CRITICAL();
}
启动任务函数体中的各个任务函数实现
// Task1函数体:每500ms翻转一次LED1
void Task1(void *pvParameters)
{
    LED_Off(LED1_Pin);
    while (1)
    {
        printf("Task1 运行...\n");
        LED_Toggle(LED1_Pin);
        vTaskDelay(500);
    }
}

// Task2函数体:每500ms翻转一次LED2
void Task2(void *pvParameters)
{
    LED_Off(LED2_Pin);
    while (1)
    {
        printf("Task2 运行...\n");
        LED_Toggle(LED2_Pin);
        vTaskDelay(500);
    }
}

// Task3函数体:检测按键按下,如果Key1按下,就删除Task1
void Task3(void *pvParameters)
{
    while (1)
    {
        printf("Task3 正在运行...\n");
        // 检测如果key1按下,就删除Task1
        if (Key_Detect() == KEY1_PRESS)
        {
            if (task1_handler != NULL)
            {
                printf("删除 Task1 任务...\n");
                vTaskDelete(task1_handler);

                // 令句柄 = NULL,表示已经删除完毕
                task1_handler = NULL;
            }
        }
        vTaskDelay(10);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/956108.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

家政服务小程序,打造智慧家政新体验

春节即将来临&#xff0c;家政市场呈现出了火热的场景&#xff0c;大众对家政服务的需求持续增加。 近年来&#xff0c;家政市场开始倾向数字化、智能化&#xff0c;借助科学技术打造家政数字化平台&#xff0c;让大众在手机上就可以预约家政服务&#xff0c;减少传统家政市场…

《贪心算法:原理剖析与典型例题精解》

必刷的贪心算法典型例题&#xff01; 算法竞赛&#xff08;蓝桥杯&#xff09;贪心算法1——数塔问题-CSDN博客 算法竞赛&#xff08;蓝桥杯&#xff09;贪心算法2——需要安排几位师傅加工零件-CSDN博客 算法&#xff08;蓝桥杯&#xff09;贪心算法3——二维数组排序与贪心算…

Vue3 nginx 打包后遇到的问题

前端vite文件配置 export default defineConfig({plugins: [vue(),DefineOptions()],base:./,resolve:{alias:{:/src, //配置指向src目录components:/src/components,views:/src/views}},server:{// host:0.0.0.0,// port:7000,proxy:{/api:{target:xxx, // 目标服务器地址 &am…

linux下配置python环境及库配置

概述 使用Linux作为开发环境拥有完整的开源工具链且易于通过系统包管理器安装&#xff0c;与系统集成良好。对于开源项目开发&#xff0c;能方便地从源代码编译安装软件&#xff0c;使用构建工具&#xff0c;提供更原生的开发环境。 可高度定制系统&#xff0c;满足特殊开发需求…

【机器学习实战入门】使用OpenCV进行性别和年龄检测

Gender and Age Detection Python 项目 首先,向您介绍用于此高级 Python 项目的性别和年龄检测中的术语: 什么是计算机视觉? 计算机视觉是一门让计算机能够像人类一样观察和识别数字图像和视频的学科。它面临的挑战大多源于对生物视觉有限的了解。计算机视觉涉及获取、处…

左神算法基础提升--4

文章目录 树形dp问题Morris遍历 树形dp问题 求解这个问题需要用到我们在基础班上学到的从节点的左子树和右子树上拿信息的方法。 求最大距离主要分为两种情况&#xff1a;1.当前节点参与最大距离的求解&#xff1b;2.当前节点不参与最大距离的求解&#xff1b; 1.当前节点参与最…

spark任务优化参数整理

以下参数中有sql字眼的一般只有spark-sql模块生效&#xff0c;如果你看过spark的源码&#xff0c;你会发现sql模块是在core模块上硬生生干了一层&#xff0c;所以反过来spark-sql可以复用core模块的配置&#xff0c;例外的时候会另行说明&#xff0c;此外由于总结这些参数是在不…

华为数据中心CE系列交换机级联M-LAG配置示例

M-LAG组网简介 M-LAG&#xff08;Multi-chassis Link Aggregation&#xff09;技术是一种跨设备的链路聚合技术&#xff0c;它通过将两台交换机组成一个逻辑设备&#xff0c;实现链路的负载分担和故障切换&#xff0c;从而提高网络的可靠性和稳定性。下面给大家详细介绍如何在…

游戏引擎学习第80天

Blackboard&#xff1a;增强碰撞循环&#xff0c;循环遍历两种类型的 t 值 计划对现有的碰撞检测循环进行修改&#xff0c;以便实现一些新的功能。具体来说&#xff0c;是希望处理在游戏中定义可行走区域和地面的一些实体。尽管这是一个2D游戏&#xff0c;目标是构建一些更丰富…

EMS专题 | 守护数据安全:数据中心和服务器机房环境温湿度监测

您需要服务器机房温度监测解决方案吗&#xff1f; 服务器机房是企业中用于存储、管理和维护服务器及其相关组件的设施。服务器机房通常位于数据中心内&#xff0c;是一个专门设计的物理环境&#xff0c;旨在确保服务器的稳定运行和数据的安全性。服务器机房主要起到存储和管理数…

4 AXI USER IP

前言 使用AXI Interface封装IP&#xff0c;并使用AXI Interface实现对IP内部寄存器进行读写实现控制LED的demo&#xff0c;这个demo是非常必要的&#xff0c;因为在前面的笔记中基本都需哟PS端与PL端就行通信互相交互&#xff0c;在PL端可以通过中断的形式来告知PS端一些事情&…

网络编程 | UDP套接字通信及编程实现经验教程

1、UDP基础 传输层主要应用的协议模型有两种&#xff0c;一种是TCP协议&#xff0c;另外一种则是UDP协议。在上一篇博客文章中&#xff0c;已经对TCP协议及如何编程实现进行了详细的梳理讲解&#xff0c;在本文中&#xff0c;主要讲解与TCP一样广泛使用了另一种协议&#xff1a…

A5.Springboot-LLama3.2服务自动化构建(二)——Jenkins流水线构建配置初始化设置

下面我们接着上一篇文章《A4.Springboot-LLama3.2服务自动化构建(一)——构建docker镜像配置》继续往下分析,在自动化流水线构建过程当中的相关初始化设置和脚本编写。 一、首先需要先安装Jenkins 主部分请参考我前面写的一篇文章《Jenkins持续集成与交付安装配置》 二、…

开发神器之cursor

文章目录 cursor简介主要特点 下载cursor页面的简单介绍切换大模型指定ai学习的文件指定特定的代码喂给ai创建项目框架文件 cursor简介 Cursor 是一款专为开发者设计的智能代码编辑器&#xff0c;集成了先进的 AI 技术&#xff0c;旨在提升编程效率。以下是其主要特点和功能&a…

基于机器学习随机森林算法的个人职业预测研究

1.背景调研 随着信息技术的飞速发展&#xff0c;特别是大数据和云计算技术的广泛应用&#xff0c;各行各业都积累了大量的数据。这些数据中蕴含着丰富的信息和模式&#xff0c;为利用机器学习进行职业预测提供了可能。机器学习算法的不断进步&#xff0c;如深度学习、强化学习等…

【王树森搜索引擎技术】概要01:搜索引擎的基本概念

1. 基本名词 query&#xff1a;查询词SUG&#xff1a;搜索建议文档&#xff1a;搜索结果标签/筛选项 文档单列曝光 文档双列曝光 2. 曝光与点击 曝光&#xff1a;用户在搜索结果页上看到文档&#xff0c;就算曝光文档点击&#xff1a;在曝光后&#xff0c;用户点击文档&…

图论DFS:黑红树

我的个人主页 {\large \mathsf{{\color{Red} 我的个人主页} } } 我的个人主页 往 {\color{Red} {\Huge 往} } 往 期 {\color{Green} {\Huge 期} } 期 文 {\color{Blue} {\Huge 文} } 文 章 {\color{Orange} {\Huge 章}} 章 DFS 算法&#xff1a;记忆化搜索DFS 算法&#xf…

ros2-7.5 做一个自动巡检机器人

7.5.1 需求及设计 又到了小鱼老师带着做最佳实践项目了。需求&#xff1a;做一个在各个房间不断巡逻并记录图像的机器人。 到达目标点后首先通过语音播放到达目标点信息&#xff0c; 再通过摄像头拍摄一张图片保存到本地。 7.5.2 编写巡检控制节点 在chapt7_ws/src下新建功…

告别繁琐编译!make和makefile的便捷之道

Linux系列 文章目录 Linux系列前言一、make/makefile是什么&#xff1f;二、make/makefile的使用2.1、语法规则2.2、依赖关系和依赖方法2.3、清理可执行文件2.4、执行依据 三、循环依赖问题总结 前言 上一篇博客给大家分享了在Linux下编译源代码的两个工具&#xff0c;gcc和g…

【鸿蒙】0x02-LiteOS-M基于Qemu RISC-V运行

OpenHarmony LiteOS-M基于Qemu RISC-V运行 系列文章目录更新日志OpenHarmony技术架构OH技术架构OH支持系统类型轻量系统&#xff08;mini system&#xff09;小型系统&#xff08;small system&#xff09;标准系统&#xff08;standard system&#xff09; 简介环境准备安装QE…