1# 任务管理 2 3 4## 基本概念 5 6从系统角度看,任务是竞争系统资源的最小运行单元。任务可以使用或等待CPU、使用内存空间等系统资源,各任务的运行相互独立。 7 8OpenHarmony LiteOS-M的任务模块可以给用户提供多个任务,实现任务间的切换,帮助用户管理业务程序流程。任务模块具有如下特性: 9 10- 支持多任务。 11 12- 一个任务表示一个线程。 13 14- 抢占式调度机制,高优先级的任务可打断低优先级任务,低优先级任务必须在高优先级任务阻塞或结束后才能得到调度。 15 16- 相同优先级任务支持时间片轮转调度方式。 17 18- 共有32个优先级[0-31],最高优先级为0,最低优先级为31。 19 20 21### 任务相关概念 22 23**任务状态** 24 25任务有多种运行状态。系统初始化完成后,创建的任务就可以在系统中竞争一定的资源,由内核进行调度。 26 27任务状态通常分为以下四种: 28 29- 就绪(Ready):该任务在就绪队列中,只等待CPU。 30 31- 运行(Running):该任务正在执行。 32 33- 阻塞(Blocked):该任务不在就绪队列中。包含任务被挂起(suspend状态)、任务被延时(delay状态)、任务正在等待信号量、读写队列或者等待事件等。 34 35- 退出态(Dead):该任务运行结束,等待系统回收资源。 36 37**任务状态迁移** 38 39 **图1** 任务状态示意图 40 41 ![zh-cn_image_0000001200612002](figures/zh-cn_image_0000001200612002.png) 42 43系统会同时存在多个任务,因此就绪态和阻塞态的任务分别会加入就绪队列和阻塞队列。队列只是相同状态任务的合集,加入队列的先后与任务状态迁移的顺序无关。运行态任务仅存在一个,不存在运行态队列。 44 45**任务状态迁移说明** 46 47- 就绪态→运行态 48 任务创建后进入就绪态,发生任务切换时,就绪队列中最高优先级的任务被执行,从而进入运行态,同时该任务从就绪队列中移出。 49 50- 运行态→阻塞态 51 正在运行的任务发生阻塞(挂起、延时、读信号量等)时,将该任务插入到对应的阻塞队列中,任务状态由运行态变成阻塞态,然后发生任务切换,运行就绪队列中最高优先级任务。 52 53- 阻塞态→就绪态(阻塞态→运行态的前置条件) 54 阻塞的任务被恢复后(任务恢复、延时时间超时、读信号量超时或读到信号量等),此时被恢复的任务会被加入就绪队列,从而由阻塞态变成就绪态;此时如果被恢复任务的优先级高于正在运行任务的优先级,则会发生任务切换,该任务由就绪态变成运行态。 55 56- 就绪态→阻塞态 57 任务也有可能在就绪态时被阻塞(挂起),此时任务状态由就绪态变为阻塞态,该任务从就绪队列中移出,不会参与任务调度,直到该任务被恢复。 58 59- 运行态→就绪态 60 有更高优先级任务创建或者恢复后,会发生任务调度,此刻就绪队列中最高优先级任务变为运行态,那么原先运行的任务由运行态变为就绪态,依然在就绪队列中。 61 62- 运行态→退出态 63 运行中的任务运行结束,任务状态由运行态变为退出态。退出态包含任务运行结束的正常退出状态以及Invalid状态。例如,任务运行结束但是没有自删除,对外呈现的就是Invalid状态,即退出态。 64 65- 阻塞态→退出态 66 阻塞的任务调用删除接口,任务状态由阻塞态变为退出态。 67 68**任务ID** 69 70在任务创建时通过参数返回给用户。系统中任务ID号是唯一的,是任务的重要标识。用户可以通过任务ID对指定任务进行任务挂起、任务恢复、查询任务名等操作。 71 72**任务优先级** 73 74优先级表示任务执行的优先顺序。任务的优先级决定了在发生任务切换时即将要执行的任务,就绪队列中最高优先级的任务将得到执行。 75 76**任务入口函数** 77 78新任务得到调度后将执行的函数。该函数由用户实现,在任务创建时,通过任务创建结构体设置。 79 80**任务栈** 81 82每个任务都拥有一个独立的栈空间,我们称为任务栈。栈空间里保存的信息包含局部变量、寄存器、函数参数、函数返回地址等。 83 84**任务上下文** 85 86任务在运行过程中使用的一些资源,如寄存器等,称为任务上下文。当这个任务挂起时,其他任务继续执行,可能会修改寄存器等资源中的值。如果任务切换时没有保存任务上下文,可能会导致任务恢复后出现未知错误。因此在任务切换时会将切出任务的任务上下文信息,保存在自身的任务栈中,以便任务恢复后,从栈空间中恢复挂起时的上下文信息,从而继续执行挂起时被打断的代码。 87 88**任务控制块(TCB)** 89 90每个任务都含有一个任务控制块(TCB)。TCB包含了任务上下文栈指针(stack pointer)、任务状态、任务优先级、任务ID、任务名、任务栈大小等信息。TCB可以反映出每个任务运行情况。 91 92**任务切换** 93 94任务切换包含获取就绪队列中最高优先级任务、切出任务上下文保存、切入任务上下文恢复等动作。 95 96 97### 任务运行机制 98 99用户创建任务时,系统会初始化任务栈,预置上下文。此外,系统还会将“任务入口函数”地址放在相应位置。这样在任务第一次启动进入运行态时,将会执行“任务入口函数”。 100 101 102## 接口说明 103 104OpenHarmony LiteOS-M内核的任务管理模块提供下面几种功能,接口详细信息可以查看API参考。 105 106 **表1** 任务管理模块接口 107 108| 功能分类 | 接口描述 | 109| -------- | -------- | 110| 创建和删除任务 | LOS_TaskCreateOnly:创建任务,并使该任务进入suspend状态。<br/>LOS_TaskCreate:创建任务,并使该任务进入ready状态,如果就绪队列中没有更高优先级的任务,则运行该任务。<br/>LOS_TaskDelete:删除指定的任务。 | 111| 控制任务状态 | LOS_TaskResume:恢复挂起的任务,使该任务进入ready状态。<br/>LOS_TaskSuspend:挂起指定的任务,然后切换任务。<br/>LOS_TaskJoin:挂起当前任务,等待指定任务运行结束并回收其任务控制块资源<br/>LOS_TaskDelay:任务延时等待,释放CPU,等待时间到期后该任务会重新进入ready状态。传入参数为Tick数目。<br/>LOS_Msleep:任务延时等待,释放CPU,等待时间到期后该任务会重新进入ready状态。传入参数为毫秒数。<br/>LOS_TaskYield:当前任务时间片设置为0,释放CPU,触发调度运行就绪任务队列中优先级最高的任务。 | 112| 控制任务调度 | LOS_TaskLock:锁任务调度,但任务仍可被中断打断。<br/>LOS_TaskUnlock:解锁任务调度。<br/>LOS_Schedule:触发任务调度。 | 113| 控制任务优先级 | LOS_CurTaskPriSet:设置当前任务的优先级。<br/>LOS_TaskPriSet:设置指定任务的优先级。<br/>LOS_TaskPriGet:获取指定任务的优先级。 | 114| 获取任务信息 | LOS_CurTaskIDGet:获取当前任务的ID。<br/>LOS_NextTaskIDGet:获取任务就绪队列中优先级最高的任务的ID。<br/>LOS_NewTaskIDGet:等同LOS_NextTaskIDGet。<br/>LOS_CurTaskNameGet:获取当前任务的名称。<br/>LOS_TaskNameGet:获取指定任务的名称。<br/>LOS_TaskStatusGet:获取指定任务的状态。<br/>LOS_TaskInfoGet:获取指定任务的信息,包括任务状态、优先级、任务栈大小、栈顶指针SP、任务入口函数、已使用的任务栈大小等。<br/>LOS_TaskIsRunning:获取任务模块是否已经开始调度运行。 | 115| 任务信息维测 | LOS_TaskSwitchInfoGet:获取任务切换信息,需要开启编译控制宏:LOSCFG_BASE_CORE_EXC_TSK_SWITCH。 | 116 117## 开发流程 118 119本节介绍任务模块的典型场景开发流程: 120 1211. 锁任务调度LOS_TaskLock,防止高优先级任务调度。 122 1232. 创建任务LOS_TaskCreate。 124 1253. 解锁任务LOS_TaskUnlock,让任务按照优先级进行调度。 126 1274. 延时任务LOS_TaskDelay,任务延时等待。 128 1295. 挂起指定的任务LOS_TaskSuspend,任务挂起等待恢复操作。 130 1316. 恢复挂起的任务LOS_TaskResume。 132 133> ![icon-note.gif](public_sys-resources/icon-note.gif) **说明:** 134> - 执行Idle任务时,会对待回收链表中的任务控制块和任务栈进行回收。 135> 136> - 任务名是指针,并没有分配空间,在设置任务名时,禁止将局部变量的地址赋值给任务名指针。 137> 138> - 任务栈的大小按8字节大小对齐。确定任务栈大小的原则是,够用就行,多了浪费,少了任务栈溢出。 139> 140> - 挂起当前任务时,如果已经锁任务调度,则无法挂起。 141> 142> - Idle任务及软件定时器任务不能被挂起或者删除。 143> 144> - 在中断处理函数中或者在锁任务的情况下,执行LOS_TaskDelay会失败。 145> 146> - 锁任务调度,并不关中断,因此任务仍可被中断打断。 147> 148> - 锁任务调度必须和解锁任务调度配合使用。 149> 150> - 设置任务优先级时可能会发生任务调度。 151> 152> - 可配置的系统最大任务数是指:整个系统的任务总个数,而非用户能使用的任务个数。例如:系统软件定时器多占用一个任务资源,那么用户能使用的任务资源就会减少一个。 153> 154> - LOS_CurTaskPriSet和LOS_TaskPriSet接口不能在中断中使用,也不能用于修改软件定时器任务的优先级。 155> 156> - LOS_TaskPriGet接口传入的task ID对应的任务未创建或者超过最大任务数,统一返回-1。 157> 158> - 在删除任务时要保证任务申请的资源(如互斥锁、信号量等)已被释放。 159 160 161## 编程实例 162 163本实例介绍基本的任务操作方法,包含2个不同优先级任务的创建、任务延时、任务锁与解锁调度、挂起和恢复等操作,阐述任务优先级调度的机制以及各接口的应用。示例代码如下: 164 165本演示代码在 ./kernel/liteos_m/testsuites/src/osTest.c 中编译验证,在TestTaskEntry中调用验证入口函数ExampleTask。 166 167 168``` 169#include "los_task.h" 170 171UINT32 g_taskHiId; 172UINT32 g_taskLoId; 173#define TSK_PRIOR_HI 3 /* 高优先级任务的优先级 */ 174#define TSK_PRIOR_LO 4 /* 低优先级任务的优先级 */ 175 176UINT32 ExampleTaskHi(VOID) 177{ 178 UINT32 ret; 179 180 printf("Enter TaskHi Handler.\n"); 181 182 /* 延时100个Ticks,延时后该任务会挂起,执行剩余任务中最高优先级的任务(即TaskLo任务) */ 183 ret = LOS_TaskDelay(100); 184 if (ret != LOS_OK) { 185 printf("Delay TaskHi Failed.\n"); 186 return LOS_NOK; 187 } 188 189 /* 100个Ticks时间到了后,该任务恢复,继续执行 */ 190 printf("TaskHi LOS_TaskDelay Done.\n"); 191 192 /* 挂起自身任务 */ 193 ret = LOS_TaskSuspend(g_taskHiId); 194 if (ret != LOS_OK) { 195 printf("Suspend TaskHi Failed.\n"); 196 return LOS_NOK; 197 } 198 printf("TaskHi LOS_TaskResume Success.\n"); 199 return ret; 200} 201 202/* 低优先级任务入口函数 */ 203UINT32 ExampleTaskLo(VOID) 204{ 205 UINT32 ret; 206 207 printf("Enter TaskLo Handler.\n"); 208 209 /* 延时100个Ticks,延时后该任务会挂起,执行剩余任务中最高优先级的任务 */ 210 ret = LOS_TaskDelay(100); 211 if (ret != LOS_OK) { 212 printf("Delay TaskLo Failed.\n"); 213 return LOS_NOK; 214 } 215 216 printf("TaskHi LOS_TaskSuspend Success.\n"); 217 218 /* 恢复被挂起的任务g_taskHiId */ 219 ret = LOS_TaskResume(g_taskHiId); 220 if (ret != LOS_OK) { 221 printf("Resume TaskHi Failed.\n"); 222 return LOS_NOK; 223 } 224 return ret; 225} 226 227/* 任务测试入口函数,创建两个不同优先级的任务 */ 228UINT32 ExampleTask(VOID) 229{ 230 UINT32 ret; 231 TSK_INIT_PARAM_S taskParam1 = { 0 }; 232 TSK_INIT_PARAM_S taskParam2 = { 0 }; 233 234 /* 锁任务调度,防止新创建的任务比本任务高而发生调度 */ 235 LOS_TaskLock(); 236 237 printf("LOS_TaskLock() Success!\n"); 238 239 taskParam1.pfnTaskEntry = (TSK_ENTRY_FUNC)ExampleTaskHi; 240 taskParam1.usTaskPrio = TSK_PRIOR_HI; 241 taskParam1.pcName = "TaskHi"; 242 taskParam1.uwStackSize = LOSCFG_BASE_CORE_TSK_DEFAULT_STACK_SIZE; 243 taskParam1.uwResved = LOS_TASK_ATTR_JOINABLE; /* detach 属性 */ 244 245 /* 创建高优先级任务,由于锁任务调度,任务创建成功后不会马上执行 */ 246 ret = LOS_TaskCreate(&g_taskHiId, &taskParam1); 247 if (ret != LOS_OK) { 248 LOS_TaskUnlock(); 249 250 printf("Example_TaskHi create Failed!\n"); 251 return LOS_NOK; 252 } 253 254 printf("Example_TaskHi create Success!\n"); 255 256 taskParam2.pfnTaskEntry = (TSK_ENTRY_FUNC)ExampleTaskLo; 257 taskParam2.usTaskPrio = TSK_PRIOR_LO; 258 taskParam2.pcName = "TaskLo"; 259 taskParam2.uwStackSize = LOSCFG_BASE_CORE_TSK_DEFAULT_STACK_SIZE; 260 261 /* 创建低优先级任务,由于锁任务调度,任务创建成功后不会马上执行 */ 262 ret = LOS_TaskCreate(&g_taskLoId, &taskParam2); 263 if (ret != LOS_OK) { 264 LOS_TaskUnlock(); 265 printf("Example_TaskLo create Failed!\n"); 266 return LOS_NOK; 267 } 268 269 printf("Example_TaskLo create Success!\n"); 270 271 /* 解锁任务调度,此时会发生任务调度,执行就绪队列中最高优先级任务 */ 272 LOS_TaskUnlock(); 273 ret = LOS_TaskJoin(g_taskHiId, NULL); 274 if (ret != LOS_OK) { 275 printf("Join Example_TaskHi Failed!, 0x%x\n", ret); 276 } else { 277 printf("Join Example_TaskHi Success!\n"); 278 } 279 return LOS_OK; 280} 281``` 282 283 284 **结果验证** 285 286编译运行得到的结果为: 287 288 289``` 290LOS_TaskLock() Success! 291Example_TaskHi create Success! 292Example_TaskLo create Success! 293Enter TaskHi Handler. 294Enter TaskLo Handler. 295TaskHi LOS_TaskDelay Done. 296TaskHi LOS_TaskSuspend Success. 297TaskHi LOS_TaskResume Success. 298Join Example_TaskHi Success! 299``` 300