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