RT_thread 线程管理

线程管理

在日常生活中,我们要完成一个大任务,一般会将它分解成多个简单、容易解决的小问题,小问题逐个被解决,大问题也就随之解决了。在多线程操作系统中,也同样需要开发人员把一个复杂的应用分解成多个小的、可调度的、序列化的程序单元,当合理地划分任务并正确地执行时,这种设计能够让系统满足实时系统的性能及时间的要求,例如让嵌入式系统执行这样的任务,系统通过传感器采集数据,并通过显示屏将数据显示出来,在多线程实时系统中,可以将这个任务分解成两个子任务,如下图所示,一个子任务不间断地读取传感器数据,并将数据写到共享内存中,另外一个子任务周期性的从共享内存中读取数据,并将传感器数据输出到显示屏上。

传感器数据接收任务与显示任务的切换执行

在 RT-Thread 中,与上述子任务对应的程序实体就是线程,线程是实现任务的载体,它是 RT-Thread 中最基本的调度单位,它描述了一个任务执行的运行环境,也描述了这个任务所处的优先等级,重要的任务可设置相对较高的优先级,非重要的任务可以设置较低的优先级,不同的任务还可以设置相同的优先级,轮流运行。

当线程运行时,它会认为自己是以独占 CPU 的方式在运行,线程执行时的运行环境称为上下文,具体来说就是各个变量和数据,包括所有的寄存器变量、堆栈、内存信息等。

本章将分成 5 节内容对 RT-Thread 线程管理进行介绍,读完本章,读者会对 RT-Thread 的线程管理机制有比较深入的了解,如:线程有哪些状态、如何创建一个线程、为什么会存在空闲线程等问题,心中也会有一个明确的答案了。

线程管理的功能特点

RT-Thread 线程管理的主要功能是对线程进行管理和调度,系统中总共存在两类线程,分别是系统线程和用户线程,系统线程是由 RT-Thread 内核创建的线程,用户线程是由应用程序创建的线程,这两类线程都会从内核对象容器中分配线程对象,当线程被删除时,也会被从对象容器中删除,如图 4-2 所示,每个线程都有重要的属性,如线程控制块、线程栈、入口函数等。

对象容器与线程对象

RT-Thread 的线程调度器是抢占式的,主要的工作就是从就绪线程列表中查找最高优先级线程,保证最高优先级的线程能够被运行,最高优先级的任务一旦就绪,总能得到 CPU 的使用权。

当一个运行着的线程使一个比它优先级高的线程满足运行条件,当前线程的 CPU 使用权就被剥夺了,或者说被让出了,高优先级的线程立刻得到了 CPU 的使用权。

如果是中断服务程序使一个高优先级的线程满足运行条件,中断完成时,被中断的线程挂起,优先级高的线程开始运行。

当调度器调度线程切换时,先将当前线程上下文保存起来,当再切回到这个线程时,线程调度器将该线程的上下文信息恢复。

线程的工作机制

线程控制块

在 RT-Thread 中,线程控制块由结构体 struct rt_thread 表示,线程控制块是操作系统用于管理线程的一个数据结构,它会存放线程的一些信息,例如优先级、线程名称、线程状态等,也包含线程与线程之间连接用的链表结构,线程等待事件集合等,详细定义如下:

/* 线程控制块 */
struct rt_thread
{
    /* rt 对象 */
    char        name[RT_NAME_MAX];     /* 线程名称 */
    rt_uint8_t  type;                   /* 对象类型 */
    rt_uint8_t  flags;                  /* 标志位 */

    rt_list_t   list;                   /* 对象列表 */
    rt_list_t   tlist;                  /* 线程列表 */

    /* 栈指针与入口指针 */
    void       *sp;                      /* 栈指针 */
    void       *entry;                   /* 入口函数指针 */
    void       *parameter;              /* 参数 */
    void       *stack_addr;             /* 栈地址指针 */
    rt_uint32_t stack_size;            /* 栈大小 */

    /* 错误代码 */
    rt_err_t    error;                  /* 线程错误代码 */
    rt_uint8_t  stat;                   /* 线程状态 */

    /* 优先级 */
    rt_uint8_t  current_priority;    /* 当前优先级 */
    rt_uint8_t  init_priority;        /* 初始优先级 */
    rt_uint32_t number_mask;

    ......

    rt_ubase_t  init_tick;               /* 线程初始化计数值 */
    rt_ubase_t  remaining_tick;         /* 线程剩余计数值 */

    struct rt_timer thread_timer;      /* 内置线程定时器 */

    void (*cleanup)(struct rt_thread *tid);  /* 线程退出清除函数 */
    rt_uint32_t user_data;                      /* 用户数据 */
};

其中 init_priority 是线程创建时指定的线程优先级,在线程运行过程当中是不会被改变的(除非用户执行线程控制函数进行手动调整线程优先级)。cleanup 会在线程退出时,被空闲线程回调一次以执行用户设置的清理现场等工作。最后的一个成员 user_data 可由用户挂接一些数据信息到线程控制块中,以提供类似线程私有数据的实现。

线程重要属性

线程栈

RT-Thread 线程具有独立的栈,当进行线程切换时,会将当前线程的上下文存在栈中,当线程要恢复运行时,再从栈中读取上下文信息,进行恢复。

线程栈还用来存放函数中的局部变量:函数中的局部变量从线程栈空间中申请;函数中局部变量初始时从寄存器中分配(ARM 架构),当这个函数再调用另一个函数时,这些局部变量将放入栈中。

对于线程第一次运行,可以以手工的方式构造这个上下文来设置一些初始的环境:入口函数(PC 寄存器)、入口参数(R0 寄存器)、返回位置(LR 寄存器)、当前机器运行状态(CPSR 寄存器)。

线程栈的增长方向是芯片构架密切相关的,RT-Thread 3.1.0 以前的版本,均只支持栈由高地址向低地址增长的方式,对于 ARM Cortex-M 架构,线程栈可构造如下图所示。

线程栈 (ARM)

线程栈大小可以这样设定,对于资源相对较大的 MCU,可以适当设计较大的线程栈;也可以在初始时设置较大的栈,例如指定大小为 1K 或 2K 字节,然后在 FinSH 中用 list_thread 命令查看线程运行的过程中线程所使用的栈的大小,通过此命令,能够看到从线程启动运行时,到当前时刻点,线程使用的最大栈深度,而后加上适当的余量形成最终的线程栈大小,最后对栈空间大小加以修改。

线程状态

线程运行的过程中,同一时间内只允许一个线程在处理器中运行,从运行的过程上划分,线程有多种不同的运行状态,如初始状态、挂起状态、就绪状态等。在 RT-Thread 中,线程包含五种状态,操作系统会自动根据它运行的情况来动态调整它的状态。 RT-Thread 中线程的五种状态,如下表所示:

状态 描述
初始状态 当线程刚开始创建还没开始运行时就处于初始状态;在初始状态下,线程不参与调度。此状态在 RT-Thread 中的宏定义为 RT_THREAD_INIT
就绪状态 在就绪状态下,线程按照优先级排队,等待被执行;一旦当前线程运行完毕让出处理器,操作系统会马上寻找最高优先级的就绪态线程运行。此状态在 RT-Thread 中的宏定义为 RT_THREAD_READY
运行状态 线程当前正在运行。在单核系统中,只有 rt_thread_self() 函数返回的线程处于运行状态;在多核系统中,可能就不止这一个线程处于运行状态。此状态在 RT-Thread 中的宏定义为 RT_THREAD_RUNNING
挂起状态 也称阻塞态。它可能因为资源不可用而挂起等待,或线程主动延时一段时间而挂起。在挂起状态下,线程不参与调度。此状态在 RT-Thread 中的宏定义为 RT_THREAD_SUSPEND
关闭状态 当线程运行结束时将处于关闭状态。关闭状态的线程不参与线程的调度。此状态在 RT-Thread 中的宏定义为 RT_THREAD_CLOSE

线程优先级

RT-Thread 线程的优先级是表示线程被调度的优先程度。每个线程都具有优先级,线程越重要,赋予的优先级就应越高,- 线程被调度的可能才会越大。

RT-Thread 最大支持 256 个线程优先级 (0~255),数值越小的优先级越高,0 为最高优先级。在一些资源比较紧张的系统中,可以根据实际情况选择只支持 8 个或 32 个优先级的系统配置;对于 ARM Cortex-M 系列,普遍采用 32 个优先级。最低优先级默认分配给空闲线程使用,用户一般不使用。在系统中,当有比当前线程优先级更高的线程就绪时,当前线程将立刻被换出,高优先级线程抢占处理器运行。

时间片

每个线程都有时间片这个参数,但时间片仅对优先级相同的就绪态线程有效。系统对优先级相同的就绪态线程采用时间片轮转的调度方式进行调度时,时间片起到约束线程单次运行时长的作用,其单位是一个系统节拍(OS Tick),详见第五章。假设有 2 个优先级相同的就绪态线程 A 与 B,A 线程的时间片设置为 10,B 线程的时间片设置为 5,那么当系统中不存在比 A 优先级高的就绪态线程时,系统会在 A、B 线程间来回切换执行,并且每次对 A 线程执行 10 个节拍的时长,对 B 线程执行 5 个节拍的时长,如下图。

相同优先级时间片轮转

线程的入口函数

线程控制块中的 entry 是线程的入口函数,它是线程实现预期功能的函数。线程的入口函数由用户设计实现,一般有以下两种代码形式:

-无限循环模式:

在实时系统中,线程通常是被动式的:这个是由实时系统的特性所决定的,实时系统通常总是等待外界事件的发生,而后进行相应的服务:

void thread_entry(void* paramenter)
{
    while (1)
    {
    /* 等待事件的发生 */

    /* 对事件进行服务、进行处理 */
    }
}

线程看似没有什么限制程序执行的因素,似乎所有的操作都可以执行。但是作为一个实时系统,一个优先级明确的实时系统,如果一个线程中的程序陷入了死循环操作,那么比它优先级低的线程都将不能够得到执行。所以在实时操作系统中必须注意的一点就是:线程中不能陷入死循环操作,必须要有让出 CPU 使用权的动作,如循环中调用延时函数或者主动挂起。用户设计这种无线循环的线程的目的,就是为了让这个线程一直被系统循环调度运行,永不删除。

-顺序执行或有限次循环模式:

如简单的顺序语句、do whlie() 或 for()循环等,此类线程不会循环或不会永久循环,可谓是 “一次性” 线程,一定会被执行完毕。在执行完毕后,线程将被系统自动删除。

static void thread_entry(void* parameter)
{
    /* 处理事务 #1 */
    …
    /* 处理事务 #2 */
    …
    /* 处理事务 #3 */
}

线程错误码

一个线程就是一个执行场景,错误码是与执行环境密切相关的,所以每个线程配备了一个变量用于保存错误码,线程的错误码有以下几种:

#define RT_EOK           0 /* 无错误     */
#define RT_ERROR         1 /* 普通错误     */
#define RT_ETIMEOUT      2 /* 超时错误     */
#define RT_EFULL         3 /* 资源已满     */
#define RT_EEMPTY        4 /* 无资源     */
#define RT_ENOMEM        5 /* 无内存     */
#define RT_ENOSYS        6 /* 系统不支持     */
#define RT_EBUSY         7 /* 系统忙     */
#define RT_EIO           8 /* IO 错误       */
#define RT_EINTR         9 /* 中断系统调用   */
#define RT_EINVAL       10 /* 非法参数      */

线程状态切换

RT-Thread 提供一系列的操作系统调用接口,使得线程的状态在这五个状态之间来回切换。几种状态间的转换关系如下图所示:

线程状态转换图

线程通过调用函数 rt_thread_create/init() 进入到初始状态(RT_THREAD_INIT);初始状态的线程通过调用函数 rt_thread_startup() 进入到就绪状态(RT_THREAD_READY);就绪状态的线程被调度器调度后进入运行状态(RT_THREAD_RUNNING);当处于运行状态的线程调用 rt_thread_delay(),rt_sem_take(),rt_mutex_take(),rt_mb_recv() 等函数或者获取不到资源时,将进入到挂起状态(RT_THREAD_SUSPEND);处于挂起状态的线程,如果等待超时依然未能获得资源或由于其他线程释放了资源,那么它将返回到就绪状态。挂起状态的线程,如果调用 rt_thread_delete/detach() 函数,将更改为关闭状态(RT_THREAD_CLOSE);而运行状态的线程,如果运行结束,就会在线程的最后部分执行 rt_thread_exit() 函数,将状态更改为关闭状态。

注意事项

RT-Thread 中,实际上线程并不存在运行状态,就绪状态和运行状态是等同的。

系统线程

前文中已提到,系统线程是指由系统创建的线程,用户线程是由用户程序调用线程管理接口创建的线程,在 RT-Thread 内核中的系统线程有空闲线程和主线程。

空闲线程

空闲线程是系统创建的最低优先级的线程,线程状态永远为就绪态。当系统中无其他就绪线程存在时,调度器将调度到空闲线程,它通常是一个死循环,且永远不能被挂起。另外,空闲线程在 RT-Thread 也有着它的特殊用途:

若某线程运行完毕,系统将自动删除线程:自动执行 rt_thread_exit() 函数,先将该线程从系统就绪队列中删除,再将该线程的状态更改为关闭状态,不再参与系统调度,然后挂入 rt_thread_defunct 僵尸队列(资源未回收、处于关闭状态的线程队列)中,最后空闲线程会回收被删除线程的资源。

空闲线程也提供了接口来运行用户设置的钩子函数,在空闲线程运行时会调用该钩子函数,适合钩入功耗管理、看门狗喂狗等工作。

主线程

在系统启动时,系统会创建 main 线程,它的入口函数为 main_thread_entry(),用户的应用入口函数 main() 就是从这里真正开始的,系统调度器启动后,main 线程就开始运行,过程如下图,用户可以在 main() 函数里添加自己的应用程序初始化代码。

主线程调用过程

线程的管理方式

本章前面 2 节对线程的功能与工作机制进行了概念上的讲解,相信大家对线程已经不再陌生。本节将深入到 RT-Thread 线程的各个接口,并给出部分源码,帮助读者在代码层次上理解线程。

下图描述了线程的相关操作,包含:创建 / 初始化线程、启动线程、运行线程、删除 / 脱离线程。可以使用 rt_thread_create() 创建一个动态线程,使用 rt_thread_init() 初始化一个静态线程,动态线程与静态线程的区别是:动态线程是系统自动从动态内存堆上分配栈空间与线程句柄(初始化 heap 之后才能使用 create 创建动态线程),静态线程是由用户分配栈空间与线程句柄。

线程相关操作

创建和删除线程

创建线程

一个线程要成为可执行的对象,就必须由操作系统的内核来为它创建一个线程。可以通过如下的接口创建一个动态线程:

rt_thread_t rt_thread_create(const char* name,
                            void (*entry)(void* parameter),
                            void* parameter,
                            rt_uint32_t stack_size,
                            rt_uint8_t priority,
                            rt_uint32_t tick);

调用这个函数时,系统会从动态堆内存中分配一个线程句柄以及按照参数中指定的栈大小从动态堆内存中分配相应的空间。分配出来的栈空间是按照 rtconfig.h 中配置的 RT_ALIGN_SIZE 方式对齐。线程创建 rt_thread_create() 的参数和返回值见下表:

参数 描述
name 线程的名称;线程名称的最大长度由 rtconfig.h 中的宏 RT_NAME_MAX 指定,多余部分会被自动截掉
entry 线程入口函数
parameter 线程入口函数参数
stack_size 线程栈大小,单位是字节
priority 线程的优先级。优先级范围根据系统配置情况(rtconfig.h 中的 RT_THREAD_PRIORITY_MAX 宏定义),如果支持的是 256 级优先级,那么范围是从 0~255,数值越小优先级越高,0 代表最高优先级
tick 线程的时间片大小。时间片(tick)的单位是操作系统的时钟节拍。当系统中存在相同优先级线程时,这个参数指定线程一次调度能够运行的最大时间长度。这个时间片运行结束时,调度器自动选择下一个就绪态的同优先级线程进行运行
返回 ——
thread 线程创建成功,返回线程句柄
RT_NULL 线程创建失败

源码分析

/**
374  * This function will create a thread object and allocate thread object memory
375  * and stack.
376  *
377  * @param name the name of thread, which shall be unique
378  * @param entry the entry function of thread
379  * @param parameter the parameter of thread enter function
380  * @param stack_size the size of thread stack
381  * @param priority the priority of thread
382  * @param tick the time slice if there are same priority thread
383  *
384  * @return the created thread object
385  */
386 rt_thread_t rt_thread_create(const char *name,
387                              void (*entry)(void *parameter),
388                              void       *parameter,
389                              rt_uint32_t stack_size,
390                              rt_uint8_t  priority,
391                              rt_uint32_t tick)
392 {
393     struct rt_thread *thread;
394     void *stack_start;
395 
        //分配一个线程对象,根据对象类型来获取对象信息,而后从内存堆中分配对象对应大小的内存空间
396     thread = (struct rt_thread *)rt_object_allocate(RT_Object_Class_Thread,
397                                                     name);
        //申请失败则返回错误码
398     if (thread == RT_NULL)
399         return RT_NULL;
400 
        //根据传入参数申请栈的大小
401     stack_start = (void *)RT_KERNEL_MALLOC(stack_size);
402     if (stack_start == RT_NULL)
403     {
404         /* allocate stack failure */
            //栈申请失败则删除上面申请的对象,并返回错误码
405         rt_object_delete((rt_object_t)thread);
406 
407         return RT_NULL;
408     }
409 
        //调用 _rt_thread_init()
410     _rt_thread_init(thread,
411                     name,
412                     entry,
413                     parameter,
414                     stack_start,
415                     stack_size,
416                     priority,
417                     tick);
418 
419     return thread;
420 }
421 RTM_EXPORT(rt_thread_create);
_rt_thread_init()
117 static rt_err_t _rt_thread_init(struct rt_thread *thread,
118                                 const char       *name,
119                                 void (*entry)(void *parameter),
120                                 void             *parameter,
121                                 void             *stack_start,
122                                 rt_uint32_t       stack_size,
123                                 rt_uint8_t        priority,
124                                 rt_uint32_t       tick)
125 {
126     /* init thread list */
        //初始化线程列表
127     rt_list_init(&(thread->tlist));
128 
        //初始化入口函数和传入入口函数的参数
129     thread->entry = (void *)entry;
130     thread->parameter = parameter;
131 
132     /* stack init */
        //初始化线程控制块栈的参数 stack_start 在rt_thread_create()函数中申请的
133     thread->stack_addr = stack_start;
134     thread->stack_size = stack_size;
135 
136     /* init thread stack */
        //对申请的栈进行初始化
137     rt_memset(thread->stack_addr, '#', thread->stack_size);
138 #ifdef ARCH_CPU_STACK_GROWS_UPWARD
139     thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter,
140                                           (void *)((char *)thread->stack_addr),
141                                           (void *)rt_thread_exit);
142 #else
143     thread->sp = (void *)rt_hw_stack_init(thread->entry, thread->parameter,
144                                           (rt_uint8_t *)((char *)thread->stack_addr + thread->stack_size - sizeof(rt_ubase_t)),
145                                           (void *)rt_thread_exit);
146 #endif
147 
148     /* priority init */
        //检查优先级的大小是否超过设置的最大优先级,初始化线程的优先级
149     RT_ASSERT(priority < RT_THREAD_PRIORITY_MAX);
150     thread->init_priority    = priority;
151     thread->current_priority = priority;
152 
153     thread->number_mask = 0;
154 #if RT_THREAD_PRIORITY_MAX > 32
155     thread->number = 0;
156     thread->high_mask = 0;
157 #endif
158 
159     /* tick init */
        //初始化线程初始化计数值和线程剩余计数值
160     thread->init_tick      = tick;
161     thread->remaining_tick = tick;
162 
163     /* error and flags */
        //线程错误代码和线程状态为初始状态
164     thread->error = RT_EOK;
165     thread->stat  = RT_THREAD_INIT;
166 
167 #ifdef RT_USING_SMP
168     /* not bind on any cpu */
169     thread->bind_cpu = RT_CPUS_NR;
170     thread->oncpu = RT_CPU_DETACHED;
171 
172     /* lock init */
173     thread->scheduler_lock_nest = 0;
174     thread->cpus_lock_nest = 0;
175     thread->critical_lock_nest = 0;
176 #endif /*RT_USING_SMP*/
177 
178     /* initialize cleanup function and user data */
        //初始化清理函数和用户数据
179     thread->cleanup   = 0;
180     thread->user_data = 0;
181 
182     /* initialize thread timer */
        //调用rt_timer_init()初始化线程的定时器,后续在定时器章节会继续分析该函数的源码
183     rt_timer_init(&(thread->thread_timer),
184                   thread->name,
185                   rt_thread_timeout,
186                   thread,
187                   0,
188                   RT_TIMER_FLAG_ONE_SHOT);
189 
190     /* initialize signal */
        //初始化信号
191 #ifdef RT_USING_SIGNALS
192     thread->sig_mask    = 0x00;
193     thread->sig_pending = 0x00;
194 
195 #ifndef RT_USING_SMP
196     thread->sig_ret     = RT_NULL;
197 #endif
198     thread->sig_vectors = RT_NULL;
199     thread->si_list     = RT_NULL;
200 #endif
201 
202 #ifdef RT_USING_LWP
203     thread->lwp = RT_NULL;
204 #endif
205 
206     RT_OBJECT_HOOK_CALL(rt_thread_inited_hook, (thread));
207 
208     return RT_EOK;
209 }

删除线程

对于一些使用 rt_thread_create() 创建出来的线程,当不需要使用,或者运行出错时,我们可以使用下面的函数接口来从系统中把线程完全删除掉:

rt_err_t rt_thread_delete(rt_thread_t thread);

调用该函数后,线程对象将会被移出线程队列并且从内核对象管理器中删除,线程占用的堆栈空间也会被释放,收回的空间将重新用于其他的内存分配。实际上,用 rt_thread_delete() 函数删除线程接口,仅仅是把相应的线程状态更改为 RT_THREAD_CLOSE 状态,然后放入到 rt_thread_defunct 队列中;而真正的删除动作(释放线程控制块和释放线程栈)需要到下一次执行空闲线程时,由空闲线程完成最后的线程删除动作。线程删除 rt_thread_delete() 接口的参数和返回值见下表:

参数 描述
thread 要删除的线程句柄
返回 ——
RT_EOK 删除线程成功
-RT_ERROR 删除线程失败

这个函数仅在使能了系统动态堆时才有效(即 RT_USING_HEAP 宏定义已经定义了)。

源码分析

423 /**
424  * This function will delete a thread. The thread object will be removed from
425  * thread queue and deleted from system object management in the idle thread.
426  *
427  * @param thread the thread to be deleted
428  *
429  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
430  */
431 rt_err_t rt_thread_delete(rt_thread_t thread)
432 {
433     rt_base_t lock;
434 
435     /* thread check */
        //参数检查
436     RT_ASSERT(thread != RT_NULL);
437     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
438     RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread) == RT_FALSE);
439     
        //判断进程的状态,如果进程的状态不是初始化状态
440     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
441     {
442         /* remove from schedule */
        //将要删除的进程从就绪列表中删除
443         rt_schedule_remove_thread(thread);
444     }
445 
446     /* release thread timer */
        //脱离线程时间计数器
447     rt_timer_detach(&(thread->thread_timer));
448     
449     /* disable interrupt */
        //关闭中断进入临界区
450     lock = rt_hw_interrupt_disable();
451     
452     /* change stat */
        //改变线程的状态为关闭状态
453     thread->stat = RT_THREAD_CLOSE;
454     
455     /* insert to defunct thread list */
        //放入到 rt_thread_defunct 队列中;而真正的删除动作(释放线程控制块和释放线程栈)需要到下一次
        //执行空闲线程时,由空闲线程完成最后的线程删除动作
456     rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
457     
458     /* enable interrupt */
        //打开中断退出临界区
459     rt_hw_interrupt_enable(lock);
460     
461     return RT_EOK;
462 }
463 RTM_EXPORT(rt_thread_delete);

初始化和脱离线程

#初始化线程

线程的初始化可以使用下面的函数接口完成,来初始化静态线程对象:

rt_err_t rt_thread_init(struct rt_thread* thread,
                        const char* name,
                        void (*entry)(void* parameter), void* parameter,
                        void* stack_start, rt_uint32_t stack_size,
                        rt_uint8_t priority, rt_uint32_t tick);

静态线程的线程句柄(或者说线程控制块指针)、线程栈由用户提供。静态线程是指线程控制块、线程运行栈一般都设置为全局变量,在编译时就被确定、被分配处理,内核不负责动态分配内存空间。需要注意的是,用户提供的栈首地址需做系统对齐(例如 ARM 上需要做 4 字节对齐)。线程初始化接口 rt_thread_init() 的参数和返回值见下表:

参数 描述
thread 线程句柄。线程句柄由用户提供出来,并指向对应的线程控制块内存地址
name 线程的名称;线程名称的最大长度由 rtconfig.h 中定义的 RT_NAME_MAX 宏指定,多余部分会被自动截掉
entry 线程入口函数
parameter 线程入口函数参数
stack_start 线程栈起始地址
stack_size 线程栈大小,单位是字节。在大多数系统中需要做栈空间地址对齐(例如 ARM 体系结构中需要向 4 字节地址对齐)
priority 线程的优先级。优先级范围根据系统配置情况(rtconfig.h 中的 RT_THREAD_PRIORITY_MAX 宏定义),如果支持的是 256 级优先级,那么范围是从 0 ~ 255,数值越小优先级越高,0 代表最高优先级
tick 线程的时间片大小。时间片(tick)的单位是操作系统的时钟节拍。当系统中存在相同优先级线程时,这个参数指定线程一次调度能够运行的最大时间长度。这个时间片运行结束时,调度器自动选择下一个就绪态的同优先级线程进行运行
返回 ——
RT_EOK 线程创建成功
-RT_ERROR 线程创建失败

源码分析

217 /**
218  * This function will initialize a thread, normally it's used to initialize a
219  * static thread object.
220  *
221  * @param thread the static thread object
222  * @param name the name of thread, which shall be unique
223  * @param entry the entry function of thread
224  * @param parameter the parameter of thread enter function
225  * @param stack_start the start address of thread stack
226  * @param stack_size the size of thread stack
227  * @param priority the priority of thread
228  * @param tick the time slice if there are same priority thread
229  *
230  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
231  */
232 rt_err_t rt_thread_init(struct rt_thread *thread,
233                         const char       *name,
234                         void (*entry)(void *parameter),
235                         void             *parameter,
236                         void             *stack_start,
237                         rt_uint32_t       stack_size,
238                         rt_uint8_t        priority,
239                         rt_uint32_t       tick)
240 {
241     /* thread check */
        //参数检查
242     RT_ASSERT(thread != RT_NULL);
243     RT_ASSERT(stack_start != RT_NULL);
244 
245     /* initialize thread object */
        //初始化线程对象,rt_object_init该函数分析见对象管理
246     rt_object_init((rt_object_t)thread, RT_Object_Class_Thread, name);
247 
        // 调用_rt_thread_init()函数进行初始化,上面已经分析了
248     return _rt_thread_init(thread,
249                            name,
250                            entry,
251                            parameter,
252                            stack_start,
253                            stack_size,
254                            priority,
255                            tick);
256 }
257 RTM_EXPORT(rt_thread_init);

线程脱离

对于用 rt_thread_init() 初始化的线程,使用 rt_thread_detach() 将使线程对象在线程队列和内核对象管理器中被脱离。线程脱离函数如下:

rt_err_t rt_thread_detach (rt_thread_t thread);

线程脱离接口 rt_thread_detach() 的参数和返回值见下表:

参数 描述
thread 线程句柄,它应该是由 rt_thread_init 进行初始化的线程句柄。
返回 ——
RT_EOK 线程脱离成功
-RT_ERROR 线程脱离失败

这个函数接口是和 rt_thread_delete() 函数相对应的, rt_thread_delete() 函数操作的对象是 rt_thread_create() 创建的句柄,而 rt_thread_detach() 函数操作的对象是使用 rt_thread_init() 函数初始化的线程控制块。同样,线程本身不应调用这个接口脱离线程本身。

源码分析

324 /**
325  * This function will detach a thread. The thread object will be removed from
326  * thread queue and detached/deleted from system object management.
327  *
328  * @param thread the thread to be deleted
329  *
330  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
331  */
332 rt_err_t rt_thread_detach(rt_thread_t thread)
333 {
334     rt_base_t lock;
335 
336     /* thread check */
        //参数检查
337     RT_ASSERT(thread != RT_NULL);
338     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
339     RT_ASSERT(rt_object_is_systemobject((rt_object_t)thread));
340 
        //判断进程的状态,如果进程的状态不是初始化状态
341     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
342     {
343         /* remove from schedule */
        //将要删除的进程从就绪列表中删除
344         rt_schedule_remove_thread(thread);
345     }
346 
347     /* release thread timer */
        //脱离线程时间计数器
348     rt_timer_detach(&(thread->thread_timer));
349 
350     /* change stat */
        //改变线程的状态为关闭状态
351     thread->stat = RT_THREAD_CLOSE;
352 
353     if ((rt_object_is_systemobject((rt_object_t)thread) == RT_TRUE) &&
354         thread->cleanup == RT_NULL)
355     {
        //脱离线程对象
356         rt_object_detach((rt_object_t)thread);
357     }
358     else
358     else
359     {
360         /* disable interrupt */
        //关闭中断进入临界区
361         lock = rt_hw_interrupt_disable();
362         /* insert to defunct thread list */
        //放入到 rt_thread_defunct 队列中;而真正的删除动作(释放线程控制块和释放线程栈)需要到下一次
        //执行空闲线程时,由空闲线程完成最后的线程删除动作
363         rt_list_insert_after(&rt_thread_defunct, &(thread->tlist));
364         /* enable interrupt */
        //打开中断退出临界区
365         rt_hw_interrupt_enable(lock);
366     }
367 
368     return RT_EOK;
369 }
370 RTM_EXPORT(rt_thread_detach);

启动线程

创建(初始化)的线程状态处于初始状态,并未进入就绪线程的调度队列,我们可以在线程初始化 / 创建成功后调用下面的函数接口让该线程进入就绪态:

rt_err_t rt_thread_startup(rt_thread_t thread);

当调用这个函数时,将把线程的状态更改为就绪状态,并放到相应优先级队列中等待调度。如果新启动的线程优先级比当前线程优先级高,将立刻切换到这个线程。线程启动接口 rt_thread_startup() 的参数和返回值见下表:

参数 描述
thread 线程句柄
返回 ——
RT_EOK 线程启动成功
-RT_ERROR 线程起动失败

源码分析

282 /**
283  * This function will start a thread and put it to system ready queue
284  *
285  * @param thread the thread to be started
286  *
287  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
288  */
289 rt_err_t rt_thread_startup(rt_thread_t thread)
290 {
291     /* thread check */
        //参数检查
292     RT_ASSERT(thread != RT_NULL);
293     RT_ASSERT((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_INIT);
294     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
295 
296     /* set current priority to initialize priority */
        //将线程的初始优先级赋值给当前优先级
297     thread->current_priority = thread->init_priority;
298 
299     /* calculate priority attribute */
        //计算当前优先级
300 #if RT_THREAD_PRIORITY_MAX > 32
301     thread->number      = thread->current_priority >> 3;            /* 5bit */
302     thread->number_mask = 1L << thread->number;
303     thread->high_mask   = 1L << (thread->current_priority & 0x07);  /* 3bit */
304 #else
305     thread->number_mask = 1L << thread->current_priority;
306 #endif
307 
308     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("startup a thread:%s with priority:%d\n",
309                                    thread->name, thread->init_priority));
310     /* change thread stat */
        //改变进程的状态为挂起状态
311     thread->stat = RT_THREAD_SUSPEND;
312     /* then resume it */
        //将该进程从挂起状态添加到就绪链表中
313     rt_thread_resume(thread);
        //获得当前线程,如果有进程正在进行则进行进程切换
314     if (rt_thread_self() != RT_NULL)
315     {
316         /* do a scheduling */
317         rt_schedule();
318     }
319 
320     return RT_EOK;
321 }
322 RTM_EXPORT(rt_thread_startup);

获得当前线程

在程序的运行过程中,相同的一段代码可能会被多个线程执行,在执行的时候可以通过下面的函数接口获得当前执行的线程句柄:

rt_thread_t rt_thread_self(void);

该接口的返回值见下表:

返回 描述
thread 当前运行的线程句柄
RT_NULL 失败,调度器还未启动

源码分析

259 /**
260  * This function will return self thread object
261  *
262  * @return the self thread object
263  */
264 rt_thread_t rt_thread_self(void)
265 {
266 #ifdef RT_USING_SMP
267     rt_base_t lock;
268     rt_thread_t self;
269 
        //加锁,类似于原子操作,获取当前正在运行线程的句柄
270     lock = rt_hw_local_irq_disable();
271     self = rt_cpu_self()->current_thread;
272     rt_hw_local_irq_enable(lock);
273     return self;
274 #else
275     extern rt_thread_t rt_current_thread;
276 
277     return rt_current_thread;
278 #endif
279 }
280 RTM_EXPORT(rt_thread_self);

使线程让出处理器资源

当前线程的时间片用完或者该线程主动要求让出处理器资源时,它将不再占有处理器,调度器会选择相同优先级的下一个线程执行。线程调用这个接口后,这个线程仍然在就绪队列中。线程让出处理器使用下面的函数接口:

rt_err_t rt_thread_yield(void);

调用该函数后,当前线程首先把自己从它所在的就绪优先级线程队列中删除,然后把自己挂到这个优先级队列链表的尾部,然后激活调度器进行线程上下文切换(如果当前优先级只有这一个线程,则这个线程继续执行,不进行上下文切换动作)。

rt_thread_yield() 函数和 rt_schedule() 函数比较相像,但在有相同优先级的其他就绪态线程存在时,系统的行为却完全不一样。执行 rt_thread_yield() 函数后,当前线程被换出,相同优先级的下一个就绪线程将被执行。而执行 rt_schedule() 函数后,当前线程并不一定被换出,即使被换出,也不会被放到就绪线程链表的尾部,而是在系统中选取就绪的优先级最高的线程执行(如果系统中没有比当前线程优先级更高的线程存在,那么执行完 rt_schedule() 函数后,系统将继续执行当前线程)。

源码分析

466 /**
467  * This function will let current thread yield processor, and scheduler will
468  * choose a highest thread to run. After yield processor, the current thread
469  * is still in READY state.
470  *
471  * @return RT_EOK
472  */
473 rt_err_t rt_thread_yield(void)
474 {
        //调用rt_schedule()切换进程
475     rt_schedule();
476 
477     return RT_EOK;
478 }
479 RTM_EXPORT(rt_thread_yield);

使线程睡眠

在实际应用中,我们有时需要让运行的当前线程延迟一段时间,在指定的时间到达后重新运行,这就叫做 “线程睡眠”。线程睡眠可使用以下三个函数接口:

rt_err_t rt_thread_sleep(rt_tick_t tick);
rt_err_t rt_thread_delay(rt_tick_t tick);
rt_err_t rt_thread_mdelay(rt_int32_t ms);

这三个函数接口的作用相同,调用它们可以使当前线程挂起一段指定的时间,当这个时间过后,线程会被唤醒并再次进入就绪状态。这个函数接受一个参数,该参数指定了线程的休眠时间。线程睡眠接口 rt_thread_sleep/delay/mdelay() 的参数和返回值见下表:

参数 描述
tick/ms 线程睡眠的时间:

sleep/delay 的传入参数 tick 以 1 个 OS Tick 为单位 ;
mdelay 的传入参数 ms 以 1ms 为单位; |
| 返回 | —— |
| RT_EOK | 操作成功 |

源码分析

481 /**
482  * This function will let current thread sleep for some ticks.
483  *
484  * @param tick the sleep ticks
485  *
486  * @return RT_EOK
487  */
488 rt_err_t rt_thread_sleep(rt_tick_t tick)
489 {
490     register rt_base_t temp;
491     struct rt_thread *thread;
492 
493     /* set to current thread */
        //获取当前正在运行的线程句柄
494     thread = rt_thread_self();
495     RT_ASSERT(thread != RT_NULL);
        //判断当前线程的类型
496     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
497     
498     /* disable interrupt */
        //关闭中断进入临界区
499     temp = rt_hw_interrupt_disable();
500 
501     /* suspend thread */
        //将当前进程状态设置为挂起状态
502     rt_thread_suspend(thread);
503 
504     /* reset the timeout of thread timer and start it */
        //重新计算下一个需要被唤醒的进程的等待时间
505     rt_timer_control(&(thread->thread_timer), RT_TIMER_CTRL_SET_TIME, &tick);
506     rt_timer_start(&(thread->thread_timer));
507 
508     /* enable interrupt */
        //打开中断,进入临界区
509     rt_hw_interrupt_enable(temp);
510 
        //调用rt_schedule()切换进程
511     rt_schedule();
512 
513     /* clear error number of this thread to RT_EOK */
        //清除此线程的错误号
514     if (thread->error == -RT_ETIMEOUT)
515         thread->error = RT_EOK;
516 
517     return RT_EOK;
518 }
520 /** 
521  * This function will let current thread delay for some ticks.
522  *
523  * @param tick the delay ticks
524  *
525  * @return RT_EOK
526  */
527 rt_err_t rt_thread_delay(rt_tick_t tick)
528 {
        //调用rt_thread_sleep()函数
529     return rt_thread_sleep(tick);
530 }
531 RTM_EXPORT(rt_thread_delay);
590 /**
591  * This function will let current thread delay for some milliseconds.
592  *
593  * @param tick the delay time
594  *
595  * @return RT_EOK
596  */
597 rt_err_t rt_thread_mdelay(rt_int32_t ms)
598 {
599     rt_tick_t tick;
600 
        //时间转换函数
601     tick = rt_tick_from_millisecond(ms);
602 
        //调用rt_thread_sleep()函数
603     return rt_thread_sleep(tick);
604 }
605 RTM_EXPORT(rt_thread_mdelay);

挂起和恢复线程

挂起线程

当线程调用 rt_thread_delay() 时,线程将主动挂起;当调用 rt_sem_take(),rt_mb_recv() 等函数时,资源不可使用也将导致线程挂起。处于挂起状态的线程,如果其等待的资源超时(超过其设定的等待时间),那么该线程将不再等待这些资源,并返回到就绪状态;或者,当其他线程释放掉该线程所等待的资源时,该线程也会返回到就绪状态。

线程挂起使用下面的函数接口:

rt_err_t rt_thread_suspend (rt_thread_t thread);

线程挂起接口 rt_thread_suspend() 的参数和返回值见下表:

参数 描述
thread 线程句柄
返回 ——
RT_EOK 线程挂起成功
-RT_ERROR 线程挂起失败,因为该线程的状态并不是就绪状态

注意事项

通常不应该使用这个函数来挂起线程本身,如果确实需要采用 rt_thread_suspend() 函数挂起当前任务,需要在调用 rt_thread_suspend() 函数后立刻调用 rt_schedule() 函数进行手动的线程上下文切换。用户只需要了解该接口的作用,不推荐使用该接口。

源码分析

705 /**
706  * This function will suspend the specified thread.
707  *
708  * @param thread the thread to be suspended
709  *
710  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
711  *
712  * @note if suspend self thread, after this function call, the
713  * rt_schedule() must be invoked.
714  */
715 rt_err_t rt_thread_suspend(rt_thread_t thread)
716 {
717     register rt_base_t stat;
718     register rt_base_t temp;
719 
720     /* thread check */
721     RT_ASSERT(thread != RT_NULL);
        //判断当前线程的类型是否正确
722     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
723 
724     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend:  %s\n", thread->name));
725 
        //获取当前进程的状态
726     stat = thread->stat & RT_THREAD_STAT_MASK;
        //如果当前进程的状态不是运行态和就绪态则返回错误码
727     if ((stat != RT_THREAD_READY) && (stat != RT_THREAD_RUNNING))
728     {
729         RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread suspend: thread disorder, 0x%2x\n",
730                                        thread->stat));
731         return -RT_ERROR;
732     }
733 
734     /* disable interrupt */
        //关闭中断,进入临界区
735     temp = rt_hw_interrupt_disable();
        //如果需要挂起的线程为正在运行的状态
736     if (stat == RT_THREAD_RUNNING)
737     {
738         /* not suspend running status thread on other core */
739         RT_ASSERT(thread == rt_thread_self());
740     }
741 
742     /* change thread stat */
        //将线程从相应的状态链表中移除,并改变线程的状态为挂起态
743     rt_schedule_remove_thread(thread);
744     thread->stat = RT_THREAD_SUSPEND | (thread->stat & ~RT_THREAD_STAT_MASK);
745 
746     /* stop thread timer anyway */
        //停止线程计时器
747     rt_timer_stop(&(thread->thread_timer));
748 
749     /* enable interrupt */
        //打开中断,退出临界区
750     rt_hw_interrupt_enable(temp);
751 
752     RT_OBJECT_HOOK_CALL(rt_thread_suspend_hook, (thread));
753     return RT_EOK;
754 }
755 RTM_EXPORT(rt_thread_suspend);

恢复线程

恢复线程就是让挂起的线程重新进入就绪状态,并将线程放入系统的就绪队列中;如果被恢复线程在所有就绪态线程中,位于最高优先级链表的第一位,那么系统将进行线程上下文的切换。线程恢复使用下面的函数接口:

rt_err_t rt_thread_resume (rt_thread_t thread);

线程恢复接口 rt_thread_resume() 的参数和返回值见下表:

参数 描述
thread 线程句柄
返回 ——
RT_EOK 线程恢复成功
-RT_ERROR 线程恢复失败,因为该个线程的状态并不是 RT_THREAD_SUSPEND 状态

源码分析

757 /**
758  * This function will resume a thread and put it to system ready queue.
759  *
760  * @param thread the thread to be resumed
761  *
762  * @return the operation status, RT_EOK on OK, -RT_ERROR on error
763  */
764 rt_err_t rt_thread_resume(rt_thread_t thread)
765 {
766     register rt_base_t temp;
767 
768     /* thread check */
        //判断当前线程的类型是否正确
769     RT_ASSERT(thread != RT_NULL);
770     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
771 
772     RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume:  %s\n", thread->name));
773 
        //如果当前进程的状态不是挂起态则返回错误码
774     if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_SUSPEND)
775     {
776         RT_DEBUG_LOG(RT_DEBUG_THREAD, ("thread resume: thread disorder, %d\n",
777                                        thread->stat));
778 
779         return -RT_ERROR;
780     }
781 
782     /* disable interrupt */
        //关闭中断,进入临界区
783     temp = rt_hw_interrupt_disable();
784 
785     /* remove from suspend list */
        //将线程从挂起链表中删除
786     rt_list_remove(&(thread->tlist));
787 
        //停止线程计时器
788     rt_timer_stop(&thread->thread_timer);
789 
790     /* enable interrupt */
        //打开中断,退出临界区
791     rt_hw_interrupt_enable(temp);
792 
793     /* insert to schedule ready list */
        //将线程添加到就绪列表中
794     rt_schedule_insert_thread(thread);
795 
796     RT_OBJECT_HOOK_CALL(rt_thread_resume_hook, (thread));
797     return RT_EOK;

控制线程

当需要对线程进行一些其他控制时,例如动态更改线程的优先级,可以调用如下函数接口:

rt_err_t rt_thread_control(rt_thread_t thread, rt_uint8_t cmd, void* arg);

线程控制接口 rt_thread_control() 的参数和返回值见下表:

函数参数 描述
thread 线程句柄
cmd 指示控制命令
arg 控制参数
返回 ——
RT_EOK 控制执行正确
-RT_ERROR 失败

指示控制命令 cmd 当前支持的命令包括:

•RT_THREAD_CTRL_CHANGE_PRIORITY:动态更改线程的优先级;

•RT_THREAD_CTRL_STARTUP:开始运行一个线程,等同于 rt_thread_startup() 函数调用;

•RT_THREAD_CTRL_CLOSE:关闭一个线程,等同于 rt_thread_delete() 函数调用。

源码分析

607 /**
608  * This function will control thread behaviors according to control command.
609  *
610  * @param thread the specified thread to be controlled
611  * @param cmd the control command, which includes
612  *  RT_THREAD_CTRL_CHANGE_PRIORITY for changing priority level of thread;
613  *  RT_THREAD_CTRL_STARTUP for starting a thread;
614  *  RT_THREAD_CTRL_CLOSE for delete a thread;
615  *  RT_THREAD_CTRL_BIND_CPU for bind the thread to a CPU.
616  * @param arg the argument of control command
617  *
618  * @return RT_EOK
619  */
620 rt_err_t rt_thread_control(rt_thread_t thread, int cmd, void *arg)
621 {
622     register rt_base_t temp;
623 
624     /* thread check */
        //判断当前线程的类型是否正确
625     RT_ASSERT(thread != RT_NULL);
626     RT_ASSERT(rt_object_get_type((rt_object_t)thread) == RT_Object_Class_Thread);
627 
628     switch (cmd)
629     {
630     case RT_THREAD_CTRL_CHANGE_PRIORITY:
631         /* disable interrupt */
            //关闭中断,进入临界区
632         temp = rt_hw_interrupt_disable();
633 
634         /* for ready thread, change queue */
            //如果当前状态为就绪状态,将线程从就绪列表中删除
635         if ((thread->stat & RT_THREAD_STAT_MASK) == RT_THREAD_READY)
636         {
637             /* remove thread from schedule queue first */
638             rt_schedule_remove_thread(thread);
639 
640             /* change thread priority */
               //改变任务的优先级
641             thread->current_priority = *(rt_uint8_t *)arg;
642 
643             /* recalculate priority attribute */
               //计算任务的优先级
644 #if RT_THREAD_PRIORITY_MAX > 32
645             thread->number      = thread->current_priority >> 3;            /* 5bit */
646             thread->number_mask = 1 << thread->number;
647             thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
648 #else
649             thread->number_mask = 1 << thread->current_priority;
650 #endif
651 
652             /* insert thread to schedule queue again */
               //改完优先级后再次将任务插入到就绪列表
653             rt_schedule_insert_thread(thread);
654         }
               //如果当前状态不是就绪状态
655         else
656         {
               //改变任务的优先级
657             thread->current_priority = *(rt_uint8_t *)arg;
658 
659             /* recalculate priority attribute */
               //计算任务的优先级
660 #if RT_THREAD_PRIORITY_MAX > 32
661             thread->number      = thread->current_priority >> 3;            /* 5bit */
662             thread->number_mask = 1 << thread->number;
663             thread->high_mask   = 1 << (thread->current_priority & 0x07);   /* 3bit */
664 #else
665             thread->number_mask = 1 << thread->current_priority;
666 #endif
667         }
668 
669         /* enable interrupt */
            //打开中断,退出临界区
670         rt_hw_interrupt_enable(temp);
671         break;
672 
673     case RT_THREAD_CTRL_STARTUP:
674         return rt_thread_startup(thread);
675 
676 #ifdef RT_USING_HEAP
677     case RT_THREAD_CTRL_CLOSE:
            //开始运行一个线程
678         return rt_thread_delete(thread);
679 #endif
680 
681 #ifdef RT_USING_SMP
682     case RT_THREAD_CTRL_BIND_CPU:
683     {
684         rt_uint8_t cpu;
685 
686         if ((thread->stat & RT_THREAD_STAT_MASK) != RT_THREAD_INIT)
687         {
688             /* we only support bind cpu before started phase. */
689             return RT_ERROR;
690         }
691 
692         cpu = (rt_uint8_t)(size_t)arg;
693         thread->bind_cpu = cpu > RT_CPUS_NR? RT_CPUS_NR : cpu;
694         break;
695     }
696 #endif /*RT_USING_SMP*/
697     default:
698         break;
699     }
700 
701     return RT_EOK;
702 }
703 RTM_EXPORT(rt_thread_control);

设置和删除空闲钩子

空闲钩子函数是空闲线程的钩子函数,如果设置了空闲钩子函数,就可以在系统执行空闲线程时,自动执行空闲钩子函数来做一些其他事情,比如系统指示灯。设置 / 删除空闲钩子的接口如下:

rt_err_t rt_thread_idle_sethook(void (*hook)(void));
rt_err_t rt_thread_idle_delhook(void (*hook)(void));

设置空闲钩子函数 rt_thread_idle_sethook() 的输入参数和返回值如下表所示:

函数参数 描述
hook 设置的钩子函数
返回 ——
RT_EOK 设置成功
-RT_EFULL 设置失败

删除空闲钩子函数 rt_thread_idle_delhook() 的输入参数和返回值如下表所示:

函数参数 描述
hook 删除的钩子函数
返回 ——
RT_EOK 删除成功
-RT_ENOSYS 删除失败

注意事项

空闲线程是一个线程状态永远为就绪态的线程,因此设置的钩子函数必须保证空闲线程在任何时刻都不会处于挂起状态,例如 rt_thread_delay(),rt_sem_take() 等可能会导致线程挂起的函数都不能使用。

设置调度器钩子

在整个系统的运行时,系统都处于线程运行、中断触发 - 响应中断、切换到其他线程,甚至是线程间的切换过程中,或者说系统的上下文切换是系统中最普遍的事件。有时用户可能会想知道在一个时刻发生了什么样的线程切换,可以通过调用下面的函数接口设置一个相应的钩子函数。在系统线程切换时,这个钩子函数将被调用:

void rt_scheduler_sethook(void (*hook)(struct rt_thread* from, struct rt_thread* to));

设置调度器钩子函数的输入参数如下表所示:

函数参数 描述
hook 表示用户定义的钩子函数指针

钩子函数 hook() 的声明如下:

void hook(struct rt_thread* from, struct rt_thread* to);

调度器钩子函数 hook() 的输入参数如下表所示:

函数参数 描述
from 表示系统所要切换出的线程控制块指针
to 表示系统所要切换到的线程控制块指针

注意事项

请仔细编写你的钩子函数,稍有不慎将很可能导致整个系统运行不正常(在这个钩子函数中,基本上不允许调用系统 API,更不应该导致当前运行的上下文挂起)。

线程应用示例

下面给出在 Keil 模拟器环境下的应用示例。

创建线程示例

这个例子创建一个动态线程初始化一个静态线程,一个线程在运行完毕后自动被系统删除,另一个线程一直打印计数,如下代码:

#include <rtthread.h>

#define THREAD_PRIORITY         25
#define THREAD_STACK_SIZE       512
#define THREAD_TIMESLICE        5

static rt_thread_t tid1 = RT_NULL;

/* 线程 1 的入口函数 */
static void thread1_entry(void *parameter)
{
    rt_uint32_t count = 0;

    while (1)
    {
        /* 线程 1 采用低优先级运行,一直打印计数值 */
        rt_kprintf("thread1 count: %d\n", count ++);
        rt_thread_mdelay(500);
    }
}

ALIGN(RT_ALIGN_SIZE)
static char thread2_stack[1024];
static struct rt_thread thread2;
/* 线程 2 入口 */
static void thread2_entry(void *param)
{
    rt_uint32_t count = 0;

    /* 线程 2 拥有较高的优先级,以抢占线程 1 而获得执行 */
    for (count = 0; count < 10 ; count++)
    {
        /* 线程 2 打印计数值 */
        rt_kprintf("thread2 count: %d\n", count);
    }
    rt_kprintf("thread2 exit\n");
    /* 线程 2 运行结束后也将自动被系统脱离 */
}

/* 线程示例 */
int thread_sample(void)
{
    /* 创建线程 1,名称是 thread1,入口是 thread1_entry*/
    tid1 = rt_thread_create("thread1",
                            thread1_entry, RT_NULL,
                            THREAD_STACK_SIZE,
                            THREAD_PRIORITY, THREAD_TIMESLICE);

    /* 如果获得线程控制块,启动这个线程 */
    if (tid1 != RT_NULL)
        rt_thread_startup(tid1);

    /* 初始化线程 2,名称是 thread2,入口是 thread2_entry */
    rt_thread_init(&thread2,
                   "thread2",
                   thread2_entry,
                   RT_NULL,
                   &thread2_stack[0],
                   sizeof(thread2_stack),
                   THREAD_PRIORITY - 1, THREAD_TIMESLICE);
    rt_thread_startup(&thread2);

    return 0;
}

/* 导出到 msh 命令列表中 */
MSH_CMD_EXPORT(thread_sample, thread sample);

仿真运行结果如下:

\ | /
- RT -     Thread Operating System
 / | \     3.1.0 build Aug 24 2018
 2006 - 2018 Copyright by rt-thread team
msh >thread_sample
msh >thread2 count: 0
thread2 count: 1
thread2 count: 2
thread2 count: 3
thread2 count: 4
thread2 count: 5
thread2 count: 6
thread2 count: 7
thread2 count: 8
thread2 count: 9
thread2 exit
thread1 count: 0
thread1 count: 1
thread1 count: 2
thread1 count: 3
…

线程 2 计数到一定值会执行完毕,线程 2 被系统自动删除,计数停止。线程 1 一直打印计数。

注意事项

关于删除线程:大多数线程是循环执行的,无需删除;而能运行完毕的线程,RT-Thread 在线程运行完毕后,自动删除线程,在 rt_thread_exit() 里完成删除动作。用户只需要了解该接口的作用,不推荐使用该接口(可以由其他线程调用此接口或在定时器超时函数中调用此接口删除一个线程,但是这种使用非常少)。

线程时间片轮转调度示例

这个例子创建两个线程,在执行时会一直打印计数,如下代码:

#include <rtthread.h>

#define THREAD_STACK_SIZE   1024
#define THREAD_PRIORITY     20
#define THREAD_TIMESLICE    10

/* 线程入口 */
static void thread_entry(void* parameter)
{
    rt_uint32_t value;
    rt_uint32_t count = 0;

    value = (rt_uint32_t)parameter;
    while (1)
    {
        if(0 == (count % 5))
        {
            rt_kprintf("thread %d is running ,thread %d count = %d\n", value , value , count);

            if(count> 200)
                return;
        }
         count++;
     }
}

int timeslice_sample(void)
{
    rt_thread_t tid = RT_NULL;
    /* 创建线程 1 */
    tid = rt_thread_create("thread1",
                            thread_entry, (void*)1,
                            THREAD_STACK_SIZE,
                            THREAD_PRIORITY, THREAD_TIMESLICE);
    if (tid != RT_NULL)
        rt_thread_startup(tid);

    /* 创建线程 2 */
    tid = rt_thread_create("thread2",
                            thread_entry, (void*)2,
                            THREAD_STACK_SIZE,
                            THREAD_PRIORITY, THREAD_TIMESLICE-5);
    if (tid != RT_NULL)
        rt_thread_startup(tid);
    return 0;
}

/* 导出到 msh 命令列表中 */
MSH_CMD_EXPORT(timeslice_sample, timeslice sample);

仿真运行结果如下:

 \ | /
- RT -     Thread Operating System
 / | \     3.1.0 build Aug 27 2018
 2006 - 2018 Copyright by rt-thread team
msh >timeslice_sample
msh >thread 1 is running ,thread 1 count = 0
thread 1 is running ,thread 1 count = 5
thread 1 is running ,thread 1 count = 10
thread 1 is running ,thread 1 count = 15
…
thread 1 is running ,thread 1 count = 125
thread 1 is rthread 2 is running ,thread 2 count = 0
thread 2 is running ,thread 2 count = 5
thread 2 is running ,thread 2 count = 10
thread 2 is running ,thread 2 count = 15
thread 2 is running ,thread 2 count = 20
thread 2 is running ,thread 2 count = 25
thread 2 is running ,thread 2 count = 30
thread 2 is running ,thread 2 count = 35
thread 2 is running ,thread 2 count = 40
thread 2 is running ,thread 2 count = 45
thread 2 is running ,thread 2 count = 50
thread 2 is running ,thread 2 count = 55
thread 2 is running ,thread 2 count = 60
thread 2 is running ,thread 2 cunning ,thread 2 count = 65
thread 1 is running ,thread 1 count = 135
…
thread 2 is running ,thread 2 count = 205

由运行的计数结果可以看出,线程 2 的运行时间是线程 1 的一半。

线程调度器钩子示例

在线程进行调度切换时,会执行调度,我们可以设置一个调度器钩子,这样可以在线程切换时,做一些额外的事情,这个例子是在调度器钩子函数中打印线程间的切换信息,如下代码:

#include <rtthread.h>

#define THREAD_STACK_SIZE   1024
#define THREAD_PRIORITY     20
#define THREAD_TIMESLICE    10

/* 针对每个线程的计数器 */
volatile rt_uint32_t count[2];

/* 线程 1、2 共用一个入口,但入口参数不同 */
static void thread_entry(void* parameter)
{
    rt_uint32_t value;

    value = (rt_uint32_t)parameter;
    while (1)
    {
        rt_kprintf("thread %d is running\n", value);
        rt_thread_mdelay(1000); // 延时一段时间
    }
}

static rt_thread_t tid1 = RT_NULL;
static rt_thread_t tid2 = RT_NULL;

static void hook_of_scheduler(struct rt_thread* from, struct rt_thread* to)
{
    rt_kprintf("from: %s -->  to: %s \n", from->name , to->name);
}

int scheduler_hook(void)
{
    /* 设置调度器钩子 */
    rt_scheduler_sethook(hook_of_scheduler);

    /* 创建线程 1 */
    tid1 = rt_thread_create("thread1",
                            thread_entry, (void*)1,
                            THREAD_STACK_SIZE,
                            THREAD_PRIORITY, THREAD_TIMESLICE);
    if (tid1 != RT_NULL)
        rt_thread_startup(tid1);

    /* 创建线程 2 */
    tid2 = rt_thread_create("thread2",
                            thread_entry, (void*)2,
                            THREAD_STACK_SIZE,
                            THREAD_PRIORITY,THREAD_TIMESLICE - 5);
    if (tid2 != RT_NULL)
        rt_thread_startup(tid2);
    return 0;
}

/* 导出到 msh 命令列表中 */
MSH_CMD_EXPORT(scheduler_hook, scheduler_hook sample);

仿真运行结果如下:

 \ | /
- RT -     Thread Operating System
 / | \     3.1.0 build Aug 27 2018
 2006 - 2018 Copyright by rt-thread team
msh > scheduler_hook
msh >from: tshell -->  to: thread1
thread 1 is running
from: thread1 -->  to: thread2
thread 2 is running
from: thread2 -->  to: tidle
from: tidle -->  to: thread1
thread 1 is running
from: thread1 -->  to: tidle
from: tidle -->  to: thread2
thread 2 is running
from: thread2 -->  to: tidle
…
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容