完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
假如有两个线程都需要获取某个事件bit3;
#define BIT_3 (1 << 3) //线程A执行调用 rt_event_recv(event_handle, BIT_3, RT_EVENT_FLAG_AND|RT_EVENT_FLAG_CLEAR, RT_WAItiNG_FOREVER, &recved_a); //线程B执行调用 rt_event_recv(event_handle, BIT_3, RT_EVENT_FLAG_AND|RT_EVENT_FLAG_CLEAR, RT_WAITING_FOREVER, &recved_b); 当事件BIT_3到来时,无法实现一对多的线程间同步。 分析rt_event_send()函数源码实现: if (!rt_list_isempty(&event->parent.suspend_thread)) { n = event->parent.suspend_thread.next; while (n != &(event->parent.suspend_thread)) { thread = rt_list_entry(n, struct rt_thread, tlist); status = -RT_ERROR; if (thread->event_info & RT_EVENT_FLAG_AND) { if ((thread->event_set & event->set) == thread->event_set) { status = RT_EOK; } } //如果线程接收事件OR选项 else if (thread->event_info & RT_EVENT_FLAG_OR) { if (thread->event_set & event->set) { thread->event_set = thread->event_set & event->set; status = RT_EOK; } } else { /* enable interrupt */ rt_hw_interrupt_enable(level); return -RT_EINVAL; } n = n->next; if (status == RT_EOK) { if (thread->event_info & RT_EVENT_FLAG_CLEAR) //不应该在这里clear事件吧?应该使用临时变量将要clear的事件先保存起来,在while循环外面在进行clear操作!!! event->set &= ~thread->event_set; rt_thread_resume(thread); need_schedule = RT_TRUE; } } } 麻烦各位大神协助分析下这个问题。谢谢! |
|
相关推荐
2个回答
|
|
|
|
|
|
因为你的option中设置了CLEAR,所以在recv的函数中,必须要对已经发生的事件清零。这里的逻辑没有问题。
写了一段不好看的代码,简单测试了一下,也加了必要的注释,两个线程都能获取事件。祝顺利。 当然,你可以根据AB线程的具体优先级,决定到底是哪个线程中CLEAR。这种写法更简单,我不清楚你的具体需求,所以才用了下面的代码: #include #define DBG_TAG "main" #define DBG_LVL DBG_LOG #include rt_event_t event_handle; #define EVENT_TID_A_DONE ( 1 << 2 ) #define EVENT_TID_B_DONE ( 1 << 1 ) #define BIT_0 ( 1 << 0 ) void entry_demo_a(void *parameter); void entry_demo_b(void *parameter); int main(void) { rt_thread_t tid; event_handle = rt_event_create("evt_demo", RT_IPC_FLAG_PRIO); tid = rt_thread_create("ThreadA", entry_demo_a, RT_NULL, 2048, 20, 10); rt_thread_startup(tid); tid = rt_thread_create("ThreadB", entry_demo_b, RT_NULL, 2048, 20, 10); rt_thread_startup(tid); while(1) { rt_event_send(event_handle, BIT_0); rt_thread_delay(1000); } return RT_EOK; } void entry_demo_a(void *parameter) { rt_uint32_t recved; while(1) { /* 不清除事件,OR方式等待事件 */ rt_event_recv(event_handle, BIT_0|EVENT_TID_A_DONE|EVENT_TID_B_DONE, RT_EVENT_FLAG_OR, RT_WAITING_FOREVER, &recved); /*AB均未处理事件,或者 B已处理A未处理 */ if( (BIT_0 == recved) || ( (BIT_0 | EVENT_TID_B_DONE) == recved)) { LOG_I("Thread A: Bit0 EVENT"); rt_event_send(event_handle, EVENT_TID_A_DONE); } else if ( (BIT_0|EVENT_TID_A_DONE|EVENT_TID_B_DONE) == recved ) { /* 4.0.5版本中,只提供了这一种事件的Control方式,清除所有事件 */ rt_event_control(event_handle, RT_IPC_CMD_RESET, RT_NULL); } /* 因为第一个等待是OR关系,所以每次recv都能获取到事件,防止线程永远占用CPU */ else { rt_thread_delay(10); } } } void entry_demo_b(void *parameter) { rt_uint32_t recved; while(1) { /* 不清除事件,OR方式等待事件 */ rt_event_recv(event_handle, BIT_0|EVENT_TID_A_DONE|EVENT_TID_B_DONE, RT_EVENT_FLAG_OR, RT_WAITING_FOREVER, &recved); /*AB均未处理事件,或者 A已处理B未处理 */ if( (BIT_0 == recved) || ( (BIT_0 | EVENT_TID_A_DONE) == recved)) { LOG_I("Thread B: Bit0 EVENT"); rt_event_send(event_handle, EVENT_TID_B_DONE); } else if ( (BIT_0|EVENT_TID_A_DONE|EVENT_TID_B_DONE) == recved ) { /* 4.0.5版本中,只提供了这一种事件的Control方式,清除所有事件 */ rt_event_control(event_handle, RT_IPC_CMD_RESET, RT_NULL); } /* 因为第一个等待是OR关系,所以每次recv都能获取到事件,防止线程永远占用CPU */ else { rt_thread_delay(10); } } } |
|
|
|
你正在撰写答案
如果你是对答案或其他答案精选点评或询问,请使用“评论”功能。
1599 浏览 0 评论
边缘设备的奇妙之旅:在小凌派-RK2206上部署AI模型来实现视觉巡线
894 浏览 0 评论
2555 浏览 0 评论
超强算力+实时响应!米尔发布基于STM32MP25x核心板Debian系统
7984 浏览 0 评论
AI模型部署边缘设备的奇妙之旅:边缘端设备的局域网视频流传输方案
1846 浏览 0 评论
82656 浏览 21 评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2025-4-7 01:15 , Processed in 0.591677 second(s), Total 40, Slave 35 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191