文章目录
- 1、基本概念
- 2、使用场景:用于任务间、任务和中断间的同步(非数据传输)
- 3、事件组结构体(事件控制块)内容解析
- 4、为什么采用的是关闭调度器而不是关闭中断?
- 5、创建事件——xEventGroupCreate
- 6、删除事件——vEventGroupDelete
- 7、设置事件
- (1)原理解析
- (2)xEventGroupSetBits()
- (3)xEventGroupSetBitsFromISR
- 8、等待事件
- (1)原理解析
- (2)xEventGroupWaitBits
- 9、任务同步—— xEventGroupSync
1、基本概念
事件组存储在一个EventBits_t 类 型的变量中 , 该 变量在事件组结构体中定义 。
事件组可以简单地认为就是一个整数。
那么这个整数是几位的呢?
![在这里插入图片描述](https://img-blog.csdnimg.cn/c78cd12e935a4c278d0e2cf140f790a8.png)
其的二进制每一位表示一个事件,值为 1 表示事件发生了,值为 0 表示事件没发生。
一个或多个任务、ISR 都可以去写这些位;一个或多个任务、ISR 都可以去读这些位。
可以等待某一位、某些位中的任意一个,也可以等待多位。
![在这里插入图片描述](https://img-blog.csdnimg.cn/3bb0529c31ca4f7682cee1989ddd32d0.png)
2、使用场景:用于任务间、任务和中断间的同步(非数据传输)
事件组的核心是关中断,且事件组不像前面的信号量和互斥量一样去复用队列的那套API。
即可以代替裸机开发中的标志位。
多次向任务设置同一事件(如果任务还未来得及读走),等效于只设置一次。允许多个任务对同一事件进行读写操作。
事件发生时,会唤醒所有符合条件的任务,简单地说它有"广播"的作用。
3、事件组结构体(事件控制块)内容解析
![在这里插入图片描述](https://img-blog.csdnimg.cn/fd7ec4fab4784d51a8e35abce1dec646.png)
等待事件的任务,所有在等待此事件的任务均会被挂载在等待事件列表 xTasksWaitingForBits这个链表里面。
4、为什么采用的是关闭调度器而不是关闭中断?
因为队列可以在中断中使用(即可以在中断中读写队列),关闭中断是为了防止其他任务和中断打断队列的读写。
而事件组不会在中断中使用,所以仅仅关闭调度器就可以防止其他任务来打扰事件组的设置等。
5、创建事件——xEventGroupCreate
xEventGroupCreate()用于创建一个事件组,并返回对应的句柄。
EventGroupHandle_t xEventGroupCreate( void );
EventGroupHandle_t xEventGroupCreateStatic( StaticEventGroup_t * pxEventGroupBuffer );
6、删除事件——vEventGroupDelete
void vEventGroupDelete( EventGroupHandle_t xEventGroup )
7、设置事件
(1)原理解析
![在这里插入图片描述](https://img-blog.csdnimg.cn/5282c3a1159b4b1d9abf3f38b858fb5b.png)
即在设置事件之后会自动唤醒符合条件的所有任务。
(2)xEventGroupSetBits()
调用这个函数的时候,第二个参数二进制表示中1对应的位会被置1。
EventBits_t xEventGroupSetBits( EventGroupHandle_t xEventGroup,const EventBits_t uxBitsToSet );
(3)xEventGroupSetBitsFromISR
BaseType_t xEventGroupSetBitsFromISR( EventGroupHandle_t xEventGroup,
const EventBits_t uxBitsToSet,
BaseType_t * pxHigherPriorityTaskWoken );
xEventGroupSetBitsFromISR函数不是直接去设置事件组的,而是给一个FreeRTOS后台任务(daemon task)发送队列数,由这个任务来设置事件组。如下图
![在这里插入图片描述](https://img-blog.csdnimg.cn/fb8a2b30f7da42b3a9f0e353aa8df95b.png)
8、等待事件
(1)原理解析
![在这里插入图片描述](https://img-blog.csdnimg.cn/0d2662b8fe834526a0aa61ddaf23fd12.png)
关键还是链表,要是阻塞时选择了等待,则会进入休眠状态。
(2)xEventGroupWaitBits
EventBits_t xEventGroupWaitBits( EventGroupHandle_t xEventGroup,
const EventBits_t uxBitsToWaitFor,
const BaseType_t xClearOnExit,
const BaseType_t xWaitForAllBits,
TickType_t xTicksToWait );
![在这里插入图片描述](https://img-blog.csdnimg.cn/1fa35f6e771642da80b4c62a1a16fc63.png)
9、任务同步—— xEventGroupSync
EventBits_t xEventGroupSync( EventGroupHandle_t xEventGroup,
const EventBits_t uxBitsToSet,
const EventBits_t uxBitsToWaitFor,
TickType_t xTicksToWait );
![在这里插入图片描述](https://img-blog.csdnimg.cn/89b1a231585a495eacad8d112a48b07c.png)
例如:我已经做好某件事了,但我还需要等待其他事件做好,这时候会进入阻塞状态(阻塞时间可以自己设置)
xEventGroupSync(xEventGroup, COOKING, ALL, portMAX_DELAY);
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)