单片机上实现多任务调度机制

一起学嵌入式 2024-12-01 12:42

扫描关注一起学嵌入式,一起学习,一起成长


在嵌入式系统中,需要同时处理多个任务的需求非常普遍。

本文将介绍如何在STM32芯片上实现多任务处理,通过合理的任务调度和管理,充分发挥芯片的性能,提高系统的灵活性和效率。

下面介绍两种多任务处理的实现方法

1. 时间片轮转调度机制

时间片轮转调度机制是利用定时器中断来实现的。
设置一个定时器,当定时器中断发生时,切换到下一个任务的执行。下面是一个简单的时间片轮转调度机制的示例代码

⏩ 定义不同的任务:定义任务的优先级、堆栈大小、维护一个任务列表,通过编写调度器代码,在合适的时机选择下一个任务来执行。

#include "stm32fxxx.h"
// 定义任务的优先级
#define TASK1_PRIORITY 1
#define TASK2_PRIORITY 2
// 定义任务的堆栈大小
#define TASK_STACK_SIZE 128
// 定义任务堆栈空间
uint32_t task1_stack[TASK_STACK_SIZE];
uint32_t task2_stack[TASK_STACK_SIZE];
// 定义任务函数
void task1(void);
void task2(void);
// 定义任务控制块结构
typedef struct {
  uint32_t* stack_ptr;
} TaskControlBlock;
// 定义任务控制块实例
TaskControlBlock tcb1;
TaskControlBlock tcb2;
// 定义当前任务指针
TaskControlBlock* current_task;
// 任务1的函数
void task1(void) {
  while (1) {
    // 任务1的处理逻辑
    // 切换任务
    __asm volatile("yield");
  }
}
// 任务2的函数
void task2(void) {
  while (1) {
    // 任务2的处理逻辑

    // 切换任务
    __asm volatile("yield");
  }
}

⏩ 定时器中断:在中断处理函数中切换任务,并保存当前任务的上下文(包括寄存器、堆栈等),然后加载下一个任务的上下文,使其开始执行

// 定义定时器中断处理函数
void TIM_IRQHandler(void) {
  // 切换到下一个任务
  if (current_task == &tcb1) {
    current_task = &tcb2;
  } else {
    current_task = &tcb1;
  }
  // 加载下一个任务的堆栈指针
  __asm volatile("mov sp, %0" ::"r"(current_task->stack_ptr));
}
多个任务之间可能需要进行通信和共享资源。
可以使用全局变量或其他同步机制来实现任务间的数据传递和资源共享。
int main() {
  // 初始化任务控制块
  tcb1.stack_ptr = task1_stack + TASK_STACK_SIZE - 1;
  tcb2.stack_ptr = task2_stack + TASK_STACK_SIZE - 1;

  // 初始化定时器,设置定时器中断
  // 这里使用TIM3作为定时器,具体配置请根据实际情况进行修改
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
  TIM_TimeBaseInitTypeDef TIM_InitStruct;
  TIM_InitStruct.TIM_Prescaler = 1000;
  TIM_InitStruct.TIM_Period = 1000;
  TIM_InitStruct.TIM_CounterMode = TIM_CounterMode_Up;
  TIM_InitStruct.TIM_ClockDivision = TIM_CKD_DIV1;
  TIM_TimeBaseInit(TIM3, &TIM_InitStruct);
  TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);
  NVIC_EnableIRQ(TIM3_IRQn);
  TIM_Cmd(TIM3, ENABLE);

  // 初始化当前任务指针
  current_task = &tcb1;

  // 启动任务1
  task1();

  while (1) {
    // 主循环,任务在定时器中断中切换
  }
}

这种简单的多任务处理方式适用于较简单的应用场景,但对于复杂的多任务应用,建议使用RTOS来提供更好的任务管理和调度机制。

2. 使用RTOS(实时操作系统)

RTOS是一种常用的多任务处理解决方案,它提供了任务调度和管理机制,简化了多任务应用的开发。

对于STM32芯片,常见的RTOS有FreeRTOS、uC/OS等。以下是实现多任务处理的基本步骤

⏩ 创建任务:使用RTOS的API,在应用程序中创建多个任务。每个任务都有自己的代码和优先级

void Task1(void* pvParameters)
{
  while (1)
  {
    // Task1处理代码
  }
}

void Task2(void* pvParameters)
{
  while (1)
  {
    // Task2处理代码
  }
}

int main()
{
  // 硬件初始化和其他配置
  // 创建任务
  xTaskCreate(Task1, "Task1", configMINIMAL_STACK_SIZE, NULL1NULL);
  xTaskCreate(Task2, "Task2", configMINIMAL_STACK_SIZE, NULL2NULL);

  // 启动调度器
  vTaskStartScheduler();

  // 代码永远不会执行到这里
  while (1)
  {
  }
}
⏩ 内核参数:配置RTOS内核的一些参数,例如时钟节拍和优先级。
int main()
{
  // 硬件初始化和其他配置

  // 配置FreeRTOS内核
  // 设置时钟节拍
  TickType_t tickRate = 1000 / configTICK_RATE_HZ;
  TickTypeSet(tickRate);

  // 配置优先级分组
  NVIC_SetPriorityGrouping(0);

  // 创建任务和启动调度器
  // ...

  // 代码永远不会执行到这里
  while (1)
  {
  }
}
 任务处理代码:在任务的处理函数中,编写任务的实际处理代码。
由于FreeRTOS采用抢占式调度,每个任务的处理函数应该是一个无限循环,确保任务不会结束。
void Task1(void* pvParameters)
{
  while (1)
  {
    // Task1处理代码

    // 任务挂起一段时间,以便给其他任务执行机会
    vTaskDelay(pdMS_TO_TICKS(100));
  }
}

void Task2(void* pvParameters)
{
  while (1)
  {
    // Task2处理代码

    // 任务挂起一段时间,以便给其他任务执行机会
    vTaskDelay(pdMS_TO_TICKS(50));
  }
}
这是一个简单的示例代码,实现了两个任务(Task1和Task2),每个任务都在一个无限循环中执行自己的处理代码,并使用vTaskDelay()函数挂起一段时间,以便给其他任务执行机会。
使用RTOS可以提供较高的可靠性和灵活性,适用于复杂的多任务应用场景。
文章来源于网络,版权归原作者所有,如有侵权,请联系删除。


关注【一起学嵌入式】,回复加群进技术交流群。




觉得文章不错,点击“分享”、“”、“在看” 呗

一起学嵌入式 公众号【一起学嵌入式】,RTOS、Linux编程、C/C++,以及经验分享、行业资讯、物联网等技术知
评论
  • 随着市场需求不断的变化,各行各业对CPU的要求越来越高,特别是近几年流行的 AIOT,为了有更好的用户体验,CPU的算力就要求更高了。今天为大家推荐由米尔基于瑞芯微RK3576处理器推出的MYC-LR3576核心板及开发板。关于RK3576处理器国产CPU,是这些年的骄傲,华为手机全国产化,国人一片呼声,再也不用卡脖子了。RK3576处理器,就是一款由国产是厂商瑞芯微,今年第二季推出的全新通用型的高性能SOC芯片,这款CPU到底有多么的高性能,下面看看它的几个特性:8核心6 TOPS超强算力双千
    米尔电子嵌入式 2025-01-03 17:04 45浏览
  • 这篇内容主要讨论三个基本问题,硅电容是什么,为什么要使用硅电容,如何正确使用硅电容?1.  硅电容是什么首先我们需要了解电容是什么?物理学上电容的概念指的是给定电位差下自由电荷的储藏量,记为C,单位是F,指的是容纳电荷的能力,C=εS/d=ε0εrS/4πkd(真空)=Q/U。百度百科上电容器的概念指的是两个相互靠近的导体,中间夹一层不导电的绝缘介质。通过观察电容本身的定义公式中可以看到,在各个变量中比较能够改变的就是εr,S和d,也就是介质的介电常数,金属板有效相对面积以及距离。当前
    知白 2025-01-06 12:04 110浏览
  • PLC组态方式主要有三种,每种都有其独特的特点和适用场景。下面来简单说说: 1. 硬件组态   定义:硬件组态指的是选择适合的PLC型号、I/O模块、通信模块等硬件组件,并按照实际需求进行连接和配置。    灵活性:这种方式允许用户根据项目需求自由搭配硬件组件,具有较高的灵活性。    成本:可能需要额外的硬件购买成本,适用于对系统性能和扩展性有较高要求的场合。 2. 软件组态   定义:软件组态主要是通过PLC
    丙丁先生 2025-01-06 09:23 66浏览
  • 自动化已成为现代制造业的基石,而驱动隔离器作为关键组件,在提升效率、精度和可靠性方面起到了不可或缺的作用。随着工业技术不断革新,驱动隔离器正助力自动化生产设备适应新兴趋势,并推动行业未来的发展。本文将探讨自动化的核心趋势及驱动隔离器在其中的重要角色。自动化领域的新兴趋势智能工厂的崛起智能工厂已成为自动化生产的新标杆。通过结合物联网(IoT)、人工智能(AI)和机器学习(ML),智能工厂实现了实时监控和动态决策。驱动隔离器在其中至关重要,它确保了传感器、执行器和控制单元之间的信号完整性,同时提供高
    腾恩科技-彭工 2025-01-03 16:28 166浏览
  • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
    华普微HOPERF 2025-01-06 15:29 73浏览
  • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
    华普微HOPERF 2025-01-06 17:23 79浏览
  • 光耦合器,也称为光隔离器,是一种利用光在两个隔离电路之间传输电信号的组件。在医疗领域,确保患者安全和设备可靠性至关重要。在众多有助于医疗设备安全性和效率的组件中,光耦合器起着至关重要的作用。这些紧凑型设备经常被忽视,但对于隔离高压和防止敏感医疗设备中的电气危害却是必不可少的。本文深入探讨了光耦合器的功能、其在医疗应用中的重要性以及其实际使用示例。什么是光耦合器?它通常由以下部分组成:LED(发光二极管):将电信号转换为光。光电探测器(例如光电晶体管):检测光并将其转换回电信号。这种布置确保输入和
    腾恩科技-彭工 2025-01-03 16:27 171浏览
  • 根据Global Info Research项目团队最新调研,预计2030年全球封闭式电机产值达到1425百万美元,2024-2030年期间年复合增长率CAGR为3.4%。 封闭式电机是一种电动机,其外壳设计为密闭结构,通常用于要求较高的防护等级的应用场合。封闭式电机可以有效防止外部灰尘、水分和其他污染物进入内部,从而保护电机的内部组件,延长其使用寿命。 环洋市场咨询机构出版的调研分析报告【全球封闭式电机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球封闭式电机总体规
    GIRtina 2025-01-06 11:10 76浏览
  • 彼得·德鲁克被誉为“现代管理学之父”,他的管理思想影响了无数企业和管理者。然而,关于他的书籍分类,一种流行的说法令人感到困惑:德鲁克一生写了39本书,其中15本是关于管理的,而其中“专门写工商企业或为企业管理者写的”只有两本——《为成果而管理》和《创新与企业家精神》。这样的表述广为流传,但深入探讨后却发现并不完全准确。让我们一起重新审视这一说法,解析其中的矛盾与根源,进而重新认识德鲁克的管理思想及其著作的真正价值。从《创新与企业家精神》看德鲁克的视角《创新与企业家精神》通常被认为是一本专为企业管
    优思学院 2025-01-06 12:03 73浏览
  •     为控制片内设备并且查询其工作状态,MCU内部总是有一组特殊功能寄存器(SFR,Special Function Register)。    使用Eclipse环境调试MCU程序时,可以利用 Peripheral Registers Viewer来查看SFR。这个小工具是怎样知道某个型号的MCU有怎样的寄存器定义呢?它使用一种描述性的文本文件——SVD文件。这个文件存储在下面红色字体的路径下。    例:南京沁恒  &n
    电子知识打边炉 2025-01-04 20:04 66浏览
  • 本文介绍Linux系统更换开机logo方法教程,通用RK3566、RK3568、RK3588、RK3576等开发板,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。制作图片开机logo图片制作注意事项(1)图片必须为bmp格式;(2)图片大小不能大于4MB;(3)BMP位深最大是32,建议设置为8;(4)图片名称为logo.bmp和logo_kernel.bmp;开机
    Industio_触觉智能 2025-01-06 10:43 72浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦