xref: /nrf52832-nimble/rt-thread/components/drivers/src/waitqueue.c (revision 104654410c56c573564690304ae786df310c91fc)
1*10465441SEvalZero /*
2*10465441SEvalZero  * Copyright (c) 2006-2018, RT-Thread Development Team
3*10465441SEvalZero  *
4*10465441SEvalZero  * SPDX-License-Identifier: Apache-2.0
5*10465441SEvalZero  *
6*10465441SEvalZero  * Change Logs:
7*10465441SEvalZero  * Date           Author       Notes
8*10465441SEvalZero  * 2018/06/26     Bernard      Fix the wait queue issue when wakeup a soon
9*10465441SEvalZero  *                             to blocked thread.
10*10465441SEvalZero  */
11*10465441SEvalZero 
12*10465441SEvalZero #include <stdint.h>
13*10465441SEvalZero 
14*10465441SEvalZero #include <rthw.h>
15*10465441SEvalZero #include <rtdevice.h>
16*10465441SEvalZero #include <rtservice.h>
17*10465441SEvalZero 
rt_wqueue_add(rt_wqueue_t * queue,struct rt_wqueue_node * node)18*10465441SEvalZero void rt_wqueue_add(rt_wqueue_t *queue, struct rt_wqueue_node *node)
19*10465441SEvalZero {
20*10465441SEvalZero     rt_base_t level;
21*10465441SEvalZero 
22*10465441SEvalZero     level = rt_hw_interrupt_disable();
23*10465441SEvalZero     rt_list_insert_before(&(queue->waiting_list), &(node->list));
24*10465441SEvalZero     rt_hw_interrupt_enable(level);
25*10465441SEvalZero }
26*10465441SEvalZero 
rt_wqueue_remove(struct rt_wqueue_node * node)27*10465441SEvalZero void rt_wqueue_remove(struct rt_wqueue_node *node)
28*10465441SEvalZero {
29*10465441SEvalZero     rt_base_t level;
30*10465441SEvalZero 
31*10465441SEvalZero     level = rt_hw_interrupt_disable();
32*10465441SEvalZero     rt_list_remove(&(node->list));
33*10465441SEvalZero     rt_hw_interrupt_enable(level);
34*10465441SEvalZero }
35*10465441SEvalZero 
__wqueue_default_wake(struct rt_wqueue_node * wait,void * key)36*10465441SEvalZero int __wqueue_default_wake(struct rt_wqueue_node *wait, void *key)
37*10465441SEvalZero {
38*10465441SEvalZero     return 0;
39*10465441SEvalZero }
40*10465441SEvalZero 
rt_wqueue_wakeup(rt_wqueue_t * queue,void * key)41*10465441SEvalZero void rt_wqueue_wakeup(rt_wqueue_t *queue, void *key)
42*10465441SEvalZero {
43*10465441SEvalZero     rt_base_t level;
44*10465441SEvalZero     register int need_schedule = 0;
45*10465441SEvalZero 
46*10465441SEvalZero     rt_list_t *queue_list;
47*10465441SEvalZero     struct rt_list_node *node;
48*10465441SEvalZero     struct rt_wqueue_node *entry;
49*10465441SEvalZero 
50*10465441SEvalZero     queue_list = &(queue->waiting_list);
51*10465441SEvalZero 
52*10465441SEvalZero     level = rt_hw_interrupt_disable();
53*10465441SEvalZero     /* set wakeup flag in the queue */
54*10465441SEvalZero     queue->flag = RT_WQ_FLAG_WAKEUP;
55*10465441SEvalZero 
56*10465441SEvalZero     if (!(rt_list_isempty(queue_list)))
57*10465441SEvalZero     {
58*10465441SEvalZero         for (node = queue_list->next; node != queue_list; node = node->next)
59*10465441SEvalZero         {
60*10465441SEvalZero             entry = rt_list_entry(node, struct rt_wqueue_node, list);
61*10465441SEvalZero             if (entry->wakeup(entry, key) == 0)
62*10465441SEvalZero             {
63*10465441SEvalZero                 rt_thread_resume(entry->polling_thread);
64*10465441SEvalZero                 need_schedule = 1;
65*10465441SEvalZero 
66*10465441SEvalZero                 rt_wqueue_remove(entry);
67*10465441SEvalZero                 break;
68*10465441SEvalZero             }
69*10465441SEvalZero         }
70*10465441SEvalZero     }
71*10465441SEvalZero     rt_hw_interrupt_enable(level);
72*10465441SEvalZero 
73*10465441SEvalZero     if (need_schedule)
74*10465441SEvalZero         rt_schedule();
75*10465441SEvalZero }
76*10465441SEvalZero 
rt_wqueue_wait(rt_wqueue_t * queue,int condition,int msec)77*10465441SEvalZero int rt_wqueue_wait(rt_wqueue_t *queue, int condition, int msec)
78*10465441SEvalZero {
79*10465441SEvalZero     int tick;
80*10465441SEvalZero     rt_thread_t tid = rt_thread_self();
81*10465441SEvalZero     rt_timer_t  tmr = &(tid->thread_timer);
82*10465441SEvalZero     struct rt_wqueue_node __wait;
83*10465441SEvalZero     rt_base_t level;
84*10465441SEvalZero 
85*10465441SEvalZero     /* current context checking */
86*10465441SEvalZero     RT_DEBUG_NOT_IN_INTERRUPT;
87*10465441SEvalZero 
88*10465441SEvalZero     tick = rt_tick_from_millisecond(msec);
89*10465441SEvalZero 
90*10465441SEvalZero     if ((condition) || (tick == 0))
91*10465441SEvalZero         return 0;
92*10465441SEvalZero 
93*10465441SEvalZero     __wait.polling_thread = rt_thread_self();
94*10465441SEvalZero     __wait.key = 0;
95*10465441SEvalZero     __wait.wakeup = __wqueue_default_wake;
96*10465441SEvalZero     rt_list_init(&__wait.list);
97*10465441SEvalZero 
98*10465441SEvalZero     level = rt_hw_interrupt_disable();
99*10465441SEvalZero     if (queue->flag == RT_WQ_FLAG_WAKEUP)
100*10465441SEvalZero     {
101*10465441SEvalZero         /* already wakeup */
102*10465441SEvalZero         goto __exit_wakeup;
103*10465441SEvalZero     }
104*10465441SEvalZero 
105*10465441SEvalZero     rt_wqueue_add(queue, &__wait);
106*10465441SEvalZero     rt_thread_suspend(tid);
107*10465441SEvalZero 
108*10465441SEvalZero     /* start timer */
109*10465441SEvalZero     if (tick != RT_WAITING_FOREVER)
110*10465441SEvalZero     {
111*10465441SEvalZero         rt_timer_control(tmr,
112*10465441SEvalZero                          RT_TIMER_CTRL_SET_TIME,
113*10465441SEvalZero                          &tick);
114*10465441SEvalZero 
115*10465441SEvalZero         rt_timer_start(tmr);
116*10465441SEvalZero     }
117*10465441SEvalZero     rt_hw_interrupt_enable(level);
118*10465441SEvalZero 
119*10465441SEvalZero     rt_schedule();
120*10465441SEvalZero 
121*10465441SEvalZero     level = rt_hw_interrupt_disable();
122*10465441SEvalZero 
123*10465441SEvalZero __exit_wakeup:
124*10465441SEvalZero     queue->flag = 0;
125*10465441SEvalZero     rt_hw_interrupt_enable(level);
126*10465441SEvalZero 
127*10465441SEvalZero     rt_wqueue_remove(&__wait);
128*10465441SEvalZero 
129*10465441SEvalZero     return 0;
130*10465441SEvalZero }
131