FreeRTOS信号量使用教程

11.1 信号量说明信号量是操作系统中重要的一部分,信号量一般用来进行资源管理和任务同步, freertos中信号量又分为二值信号量、 计数型信号量、互斥信号量和递归互斥信号量。信号量在实际应用中最广泛的两个用途是:
临界资源的锁机制:用于控制共享资源访问的场景相当于一个上锁机制, 代码只有获得了这个锁的钥匙才能够执行。多个任务同步机制:用于任务与任务或中断与任务之间的同步。在编写中断服务函数的时候我们都知道一定要快进快出,中断服务函数里面不能放太多的代码,否则的话会影响的中断的实时性。 裸机编写中断服务函数的时候一般都只是在中断服务函数中打个标记,然后在其他的地方根据标记来做具体的处理过程。在使用 rtos 系统的时候我们就可以借助信号量完成此功能, 当中断发生的时候就释放信号量,中断服务函数不做具体的处理。具体的处理过程做成一个任务,这个任务会获取信号量,如果获取到信号量就说明中断发生了,那么就开始完成相应的处理,这样做的好处就是中断执行时间非常短。 这个例子就是中断与任务之间使用信号量来完成同步,当然了, 任务与任务之间也可以使用信号量来完成同步。
在实际4g/wifi等网络应用中,一般最简单的方法就是使用一个任务去查询 4g/wifi 模块是否有数据到来,当有数据的时候就处理这个网络数据。但这样使用轮询的方式是很浪费cpu 资源的,而且也阻止了其他任务的运行。一种理想的解决方法应该是当没有网络数据的时候网络任务就进入阻塞态,把 cpu 让给其他的任务,当有数据的时候网络任务才去执行。现在使用二值信号量就可以实现这样的功能,任务通过获取信号量来判断是否有网络数据,没有的话就进入阻塞态,而网络中断服务函数通过释放信号量来通知任务以太网外设接收到了网络数据,网络任务可以去提取处理了。网络任务只是在一直的获取二值信号量,它不会释放信号量,而中断服务函数是一直在释放信号量,它不会获取信号量。
接下来,我们以二值信号量为例,讲解信号量实现任务同步的大致流程。其实,二值信号量其实就是一个只有一个队列项的队列,这个特殊的队列要么是满的,要么是空的。二值信号量通常用于互斥访问或任务同步,它与互斥信号量非常类似,但是还是有一些细微的差别,互斥信号量拥有优先级继承机制,二值信号量没有优先级继承。因此二值信号另更适合用于同步(任务与任务或任务与中断的同步),而互斥信号量适合用于简单的互斥访问。
任务获取信号量有效
如下图所示,任务task通过调用 xsemaphoretake() 函数获取信号量,但此时二值信号量无效,所以任务task进入到阻塞态。
中断释放信号量
当有数据到来产生中断后,在中断服务处理程序中通过调用 xsemaphoregivefromisr() 函数释放信号量,此后二值信号量有效。
任务获取信号量有效
由于此时信号量已经有效了,所以任务 task 获取信号量成功,任务从阻塞态解除,开始执行相关的处理过程。
任务再次进入阻塞
由于任务一般是一个大的死循环,所以在任务做完相关处理以后就会再次调用 xsemaphoretake() 函数获取信号量,但此时信号量已经失效,因此任务将进入到状态1下阻塞等待信号量有效。11.2 信号量api说明11.2.1 获取信号量不管是二值信号量、计数型信号量还是互斥信号量,它们都使用以下两个函数获取信号量。
/* 参数: xsemaphore: 要获取的信号量句柄。   xblocktime: 阻塞时间。   返回值: pdtrue: 获取信号量成功。 pdfalse: 超时,获取信号量失败。 */basetype_t xsemaphoretake(semaphorehandle_t xsemaphore,ticktype_t xblocktime)/*说明: 此函数用于在中断服务函数中获取信号量, 此函数用于获取二值信号量和计数型信号量,绝对不能使用此函数来获取互斥信号量。参数: xsemaphore: 要获取的信号量句柄。 pxhigherprioritytaskwoken: 标记退出此函数以后是否进行任务切换,这个变量的值由这三个函数来设置的,用户不用进行设置, 用户只需要提供一个变量来保存这个值就行了。当此值为 pdtrue 的时候在退出中断服务函数之前一定要进行一次任务切换。回值: pdpass: 获取信号量成功。 pdfalse: 获取信号量失败。 */ basetype_t xsemaphoretakefromisr(semaphorehandle_t xsemaphore,basetype_t * pxhigherprioritytaskwoken)11.2.2 释放信号量释放信号量分为任务级和中断级。不管是二值信号量、计数型信号量还是互斥信号量,它们都使用以下两个函数释放信号量。
/* 参数:xsemaphore: 要释放的信号量句柄。返回值: pdpass: 释放信号量成功; errqueue_full: 释放信号量失败。 */ basetype_t xsemaphoregive( xsemaphore )/* 说明: 此函数用于在中断中释放信号量, 此函数只能用来释放二值信号量和计数型信号量,绝对不能用来在中断服务函数中释放互斥信号量。参数: xsemaphore: 要释放的信号量句柄。 pxhigherprioritytaskwoken: 标记退出此函数以后是否进行任务切换,这个变量的值由这三个函数来设置的,用户不用进行设置, 用户只需要提供一个变量来保存这个值就行了。当此值为 pdtrue 的时候在退出中断服务函数之前一定要进行一次任务切换。返回值: pdpass: 释放信号量成功。 errqueue_full: 释放信号量失败。*/ basetype_t xsemaphoretakefromisr(semaphorehandle_t xsemaphore,basetype_t * pxhigherprioritytaskwoken)11.2.3 创建二值信号量在freertos中,有两个函数可以创建二值信号量:
/*说明: 使用此函数创建二值信号量的话信号量所需要的 ram 是由 freertos 的内存管理部分来动态分配的。返回值: null: 二值信号量创建失败。 其他值: 创建成功的二值信号量的句柄。 */semaphorehandle_t xsemaphorecreatebinary( void )/*说明: 此函数也是创建二值信号量的,只不过使用此函数创建二值信号量的话信号量所需要的ram需要由用户来分配参数: pxsemaphorebuffer: 此参数指向一个 staticsemaphore_t 类型的变量,用来保存信号量结构体。返回值: null: 二值信号量创建失败。 其他值: 创建成功的二值信号量句柄。 */semaphorehandle_t xsemaphorecreatebinarystatic( staticsemaphore_t *pxsemaphorebuffer )11.2.4 创建互斥信号量在freertos中,有两个函数可以创建互斥信号量:
/*说明:此函数用于创建一个互斥信号量,所需要的内存通过动态内存管理方法分配。参数:无返回值: null: 互斥信号量创建失败。 其他值: 创建成功的互斥信号量的句柄。 */semaphorehandle_t xsemaphorecreatemutex( void )/*说明: 此函数也是创建互斥信号量的,只不过使用此函数创建互斥信号量的话信号量所需要的ram 需要由用户来分配参数: pxmutexbuffer: 此参数指向一个 staticsemaphore_t 类型的变量,用来保存信号量结构体。返回值: null: 互斥信号量创建失败。 其他值: 创建成功的互斥信号量的句柄。 */semaphorehandle_t xsemaphorecreatemutexstatic( staticsemaphore_t *pxmutexbuffer )11.2.5 创建计数信号量在freertos中,有两个函数可以创建计数信号量:
/*说明: 此函数用于创建一个计数型信号量,所需要的内存通过动态内存管理方法分配。参数: uxmaxcount: 计数信号量最大计数值,当信号量值等于此值的时候释放信号量就会失败。 uxinitialcount: 计数信号量初始值。返回值: null: 计数型信号量创建失败。 其他值: 计数型信号量创建成功,返回计数型信号量句柄。 */semaphorehandle_t xsemaphorecreatecounting(ubasetype_t uxmaxcount,ubasetype_t uxinitialcount )/*说明: 此函数也是用来创建计数型信号量的,使用此函数创建计数型信号量的时候所需要的内存需要由用户分配。参数: uxmaxcount: 计数信号量最大计数值,当信号量值等于此值的时候释放信号量就会失败。 uxinitialcount: 计数信号量初始值。 pxsemaphorebuffer: 指向一个 staticsemaphore_t 类型的变量,用来保存信号量结构体。返回值: null: 计数型信号量创建失败。 其他值: 计数型号量创建成功,返回计数型信号量句柄。 */semaphorehandle_t xsemaphorecreatecountingstatic( ubasetype_t uxmaxcount,ubasetype_t uxinitialcount,staticsemaphore_t * pxsemaphorebuffer )11.2.6 递归互斥信号量递归互斥信号量可以看是一种特殊的互斥信号量,已经获取了信号量的任务就不能再次获取这个互斥量,但是递归互斥信号量不同,已经获取了递归互斥信号量的任务可以再次获取这个递归互斥信号量,次数不限。也就是在同一个任务中可以无限次获取递归互斥信号量,中途不需要释放,而互斥信号量获取一次后就失效,需要再次释放才有效。注意:任务获取多少次递归互斥信号量,就要释放多少次递归信号量。
/*说明: 动态创建递归互斥信号量参数: 无返回值:创建失败null,创建成功返回信号量句柄。*/semaphorehandle_t xsemaphorecreaterecursivemutex( void )#define xsemaphorecreaterecursivemutex() xqueuecreatemutex( queuequeue_type_recursive_mutex )/*说明: 静态创建递归互斥信号量参数: pxmutexbuffer:指向staticsemaphore_t类型的变量,用来保存信号量结构体。返回值: 创建失败null,创建成功返回信号量句柄。*/ semaphorehandle_t xsemaphorecreaterecursivemutexstatic( staticsemaphore_t *pxmutexbuffer )#define xsemaphorecreaterecursivemutexstatic( pxstaticsemaphore ) xqueuecreatemutexstatic( queuequeue_type_recursive_mutex, pxstaticsemaphore )/*说明: 释放递归互斥信号量参数: xmutex: 信号量句柄返回值: pdpass: 释放信号量成功 pdfail: 释放信号量失败*/xsemaphoregiverecursive( semaphorehandle_t xmutex )#define xsemaphoregiverecursive( xmutex ) xqueuegivemutexrecursive( ( xmutex ) )/*说明:获取递归互斥信号量参数: xmutex: 信号量句柄 xblocktime:阻塞时间返回值: pdpass: 获取信号量成功 pdfail: 获取信号量失败*/xsemaphoretakerecursive( semaphorehandle_t xmutex, ticktype_t xblocktime );#define xsemaphoretakerecursive( xmutex, xblocktime ) xqueuetakemutexrecursive( ( xmutex ), ( xblocktime ) )11.3 信号量多任务同步实例在这里我们将创建两个led线程,其中蓝色led线程每隔200ms闪烁一次,而红色led线程则每隔1200ms闪烁一次。然后我们使用二值信号量的机制,让两个led线程同步交替闪烁一次。
11.3.1 创建两个线程任务如下图所示,单击窗格顶部的 “new thread” 按钮,添加两个线程分别命名为 thread_led1 和 thread_led2 ,其它的保持默认配置即可,并重新生成代码。
如下图所示,单击窗口的 “new object” 按钮,选择 “binary semaphore” 添加一个二值信号量,然后修改该信号量的名称为 g_led_semaphore ,并重新生成代码。
11.3.2 修改信号量实例代码修改 thread_led1_entry.c 源码如下:
#include thread_led1.h/* led thread entry function *//* pvparameters contains taskhandle_t */void thread_led1_entry(void *pvparameters){ fsp_parameter_not_used (pvparameters); r_bsp_pinaccessenable(); /* enable access to the pfs registers. */ /* todo: add your own code here */ while (1) { xsemaphoretake( g_led_semaphore, portmax_delay ); r_bsp_pinwrite(ledred, bsp_io_level_high); vtaskdelay (600); r_bsp_pinwrite(ledred, bsp_io_level_low); vtaskdelay (600); }}如果不加信号量操作部分代码,红色led将每隔1200ms闪烁一次;这里调用了xsemaphoretake() 函数来获取信号量 g_led_semaphore,如果 thread_led2 线程没有运行并释放信号量的话,它将会阻塞;xsemaphoretake() 函数在获取到信号之后,将会让红色led闪烁一次;之后该线程将会再次调用了xsemaphoretake() 函数,等待 thread_led2 线程释放信号;修改 thread_led2_entry.c 源码如下:
#include thread_led2.h/* led thread entry function *//* pvparameters contains taskhandle_t */void thread_led2_entry(void *pvparameters){ fsp_parameter_not_used (pvparameters); r_bsp_pinaccessenable(); /* enable access to the pfs registers. */ /* todo: add your own code here */ while (1) { if( pdtrue == xsemaphoregive( g_led_semaphore) ) { r_bsp_pinwrite(ledblue, bsp_io_level_high); vtaskdelay (100); r_bsp_pinwrite(ledblue, bsp_io_level_low); vtaskdelay (100); } }}如果不加信号量操作部分代码,蓝色led将每隔200ms闪烁一次;这里调用了xsemaphoregive() 函数释放信号量 g_led_semaphore,用来同步通知 thread_led1 线程运行;xsemaphoregive() 函数在释放信号之后,如果 thread_led1 没有来得及获取信号量的话,则会返回pdfalse;这样,如果 thread_led1 没有运行的话,蓝色led也不会闪烁了,从而实现了两个线程中的led同步交替闪烁一次效果了;

艾迈斯半导体推出16通道LED背光灯控制器改善电视画质并降低能耗
一图读懂IP防护等级,防尘防水一目了然
汽车ECU诊断Debounce算法介绍
Adobe宣布斥资47.5亿美元收购营销软件公司Marketo
西门子万人裁员计划已开始实施!
FreeRTOS信号量使用教程
VR交通安全体验馆—VR在交通安全中的应用
如何才能在物联网的万亿级市场中寻找到绝佳的机会
液晶显示技术追踪
基于微流控技术的表面增强拉曼在医疗领域的应用
物联网新讯:工业机器人智能系统完成融资 中国电信实现飞机上网技术
刘海屏被认为是更进步更贵的屏幕?
硅谷巨头到底出了什么问题?肢解他们的可能性有多大?
睿思芯科瞄准数据中心及边缘计算市场
优快签约10家机器人服务商,为用户提供专业、高效、实惠的技术服务
发电机的保护常识
继电器的主要测试参数与方法
为促进远程医疗落地,云计算能够做些什么
如何做一个车载吸尘器PCBA方案
智能手机屏幕形态进化史 “真·全面屏”设计成未来趋势