1*d6eebaa4SHervé Poussineau /*
2*d6eebaa4SHervé Poussineau * Copyright (c) 2017 Simon Goldschmidt
3*d6eebaa4SHervé Poussineau * All rights reserved.
4*d6eebaa4SHervé Poussineau *
5*d6eebaa4SHervé Poussineau * Redistribution and use in source and binary forms, with or without modification,
6*d6eebaa4SHervé Poussineau * are permitted provided that the following conditions are met:
7*d6eebaa4SHervé Poussineau *
8*d6eebaa4SHervé Poussineau * 1. Redistributions of source code must retain the above copyright notice,
9*d6eebaa4SHervé Poussineau * this list of conditions and the following disclaimer.
10*d6eebaa4SHervé Poussineau * 2. Redistributions in binary form must reproduce the above copyright notice,
11*d6eebaa4SHervé Poussineau * this list of conditions and the following disclaimer in the documentation
12*d6eebaa4SHervé Poussineau * and/or other materials provided with the distribution.
13*d6eebaa4SHervé Poussineau * 3. The name of the author may not be used to endorse or promote products
14*d6eebaa4SHervé Poussineau * derived from this software without specific prior written permission.
15*d6eebaa4SHervé Poussineau *
16*d6eebaa4SHervé Poussineau * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17*d6eebaa4SHervé Poussineau * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18*d6eebaa4SHervé Poussineau * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
19*d6eebaa4SHervé Poussineau * SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20*d6eebaa4SHervé Poussineau * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
21*d6eebaa4SHervé Poussineau * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
22*d6eebaa4SHervé Poussineau * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
23*d6eebaa4SHervé Poussineau * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
24*d6eebaa4SHervé Poussineau * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
25*d6eebaa4SHervé Poussineau * OF SUCH DAMAGE.
26*d6eebaa4SHervé Poussineau *
27*d6eebaa4SHervé Poussineau * This file is part of the lwIP TCP/IP stack.
28*d6eebaa4SHervé Poussineau *
29*d6eebaa4SHervé Poussineau * Author: Simon Goldschmidt
30*d6eebaa4SHervé Poussineau *
31*d6eebaa4SHervé Poussineau */
32*d6eebaa4SHervé Poussineau
33*d6eebaa4SHervé Poussineau
34*d6eebaa4SHervé Poussineau #include <lwip/opt.h>
35*d6eebaa4SHervé Poussineau #include <lwip/arch.h>
36*d6eebaa4SHervé Poussineau #if !NO_SYS
37*d6eebaa4SHervé Poussineau #include "sys_arch.h"
38*d6eebaa4SHervé Poussineau #endif
39*d6eebaa4SHervé Poussineau #include <lwip/stats.h>
40*d6eebaa4SHervé Poussineau #include <lwip/debug.h>
41*d6eebaa4SHervé Poussineau #include <lwip/sys.h>
42*d6eebaa4SHervé Poussineau
43*d6eebaa4SHervé Poussineau #include <string.h>
44*d6eebaa4SHervé Poussineau
45*d6eebaa4SHervé Poussineau u32_t lwip_sys_now;
46*d6eebaa4SHervé Poussineau
47*d6eebaa4SHervé Poussineau u32_t
sys_jiffies(void)48*d6eebaa4SHervé Poussineau sys_jiffies(void)
49*d6eebaa4SHervé Poussineau {
50*d6eebaa4SHervé Poussineau return lwip_sys_now;
51*d6eebaa4SHervé Poussineau }
52*d6eebaa4SHervé Poussineau
53*d6eebaa4SHervé Poussineau u32_t
sys_now(void)54*d6eebaa4SHervé Poussineau sys_now(void)
55*d6eebaa4SHervé Poussineau {
56*d6eebaa4SHervé Poussineau return lwip_sys_now;
57*d6eebaa4SHervé Poussineau }
58*d6eebaa4SHervé Poussineau
59*d6eebaa4SHervé Poussineau void
sys_init(void)60*d6eebaa4SHervé Poussineau sys_init(void)
61*d6eebaa4SHervé Poussineau {
62*d6eebaa4SHervé Poussineau }
63*d6eebaa4SHervé Poussineau
64*d6eebaa4SHervé Poussineau #if !NO_SYS
65*d6eebaa4SHervé Poussineau
66*d6eebaa4SHervé Poussineau test_sys_arch_waiting_fn the_waiting_fn;
67*d6eebaa4SHervé Poussineau
68*d6eebaa4SHervé Poussineau void
test_sys_arch_wait_callback(test_sys_arch_waiting_fn waiting_fn)69*d6eebaa4SHervé Poussineau test_sys_arch_wait_callback(test_sys_arch_waiting_fn waiting_fn)
70*d6eebaa4SHervé Poussineau {
71*d6eebaa4SHervé Poussineau the_waiting_fn = waiting_fn;
72*d6eebaa4SHervé Poussineau }
73*d6eebaa4SHervé Poussineau
74*d6eebaa4SHervé Poussineau err_t
sys_sem_new(sys_sem_t * sem,u8_t count)75*d6eebaa4SHervé Poussineau sys_sem_new(sys_sem_t *sem, u8_t count)
76*d6eebaa4SHervé Poussineau {
77*d6eebaa4SHervé Poussineau LWIP_ASSERT("sem != NULL", sem != NULL);
78*d6eebaa4SHervé Poussineau *sem = count + 1;
79*d6eebaa4SHervé Poussineau return ERR_OK;
80*d6eebaa4SHervé Poussineau }
81*d6eebaa4SHervé Poussineau
82*d6eebaa4SHervé Poussineau void
sys_sem_free(sys_sem_t * sem)83*d6eebaa4SHervé Poussineau sys_sem_free(sys_sem_t *sem)
84*d6eebaa4SHervé Poussineau {
85*d6eebaa4SHervé Poussineau LWIP_ASSERT("sem != NULL", sem != NULL);
86*d6eebaa4SHervé Poussineau *sem = 0;
87*d6eebaa4SHervé Poussineau }
88*d6eebaa4SHervé Poussineau
89*d6eebaa4SHervé Poussineau void
sys_sem_set_invalid(sys_sem_t * sem)90*d6eebaa4SHervé Poussineau sys_sem_set_invalid(sys_sem_t *sem)
91*d6eebaa4SHervé Poussineau {
92*d6eebaa4SHervé Poussineau LWIP_ASSERT("sem != NULL", sem != NULL);
93*d6eebaa4SHervé Poussineau *sem = 0;
94*d6eebaa4SHervé Poussineau }
95*d6eebaa4SHervé Poussineau
96*d6eebaa4SHervé Poussineau /* semaphores are 1-based because RAM is initialized as 0, which would be valid */
97*d6eebaa4SHervé Poussineau u32_t
sys_arch_sem_wait(sys_sem_t * sem,u32_t timeout)98*d6eebaa4SHervé Poussineau sys_arch_sem_wait(sys_sem_t *sem, u32_t timeout)
99*d6eebaa4SHervé Poussineau {
100*d6eebaa4SHervé Poussineau u32_t ret = 0;
101*d6eebaa4SHervé Poussineau LWIP_ASSERT("sem != NULL", sem != NULL);
102*d6eebaa4SHervé Poussineau LWIP_ASSERT("*sem > 0", *sem > 0);
103*d6eebaa4SHervé Poussineau if (*sem == 1) {
104*d6eebaa4SHervé Poussineau /* need to wait */
105*d6eebaa4SHervé Poussineau if(!timeout)
106*d6eebaa4SHervé Poussineau {
107*d6eebaa4SHervé Poussineau /* wait infinite */
108*d6eebaa4SHervé Poussineau LWIP_ASSERT("cannot wait without waiting callback", the_waiting_fn != NULL);
109*d6eebaa4SHervé Poussineau do {
110*d6eebaa4SHervé Poussineau int expectSomething = the_waiting_fn(sem, NULL);
111*d6eebaa4SHervé Poussineau LWIP_ASSERT("*sem > 0", *sem > 0);
112*d6eebaa4SHervé Poussineau LWIP_ASSERT("expecting a semaphore count but it's 0", !expectSomething || (*sem > 1));
113*d6eebaa4SHervé Poussineau ret++;
114*d6eebaa4SHervé Poussineau if (ret == SYS_ARCH_TIMEOUT) {
115*d6eebaa4SHervé Poussineau ret--;
116*d6eebaa4SHervé Poussineau }
117*d6eebaa4SHervé Poussineau } while(*sem == 1);
118*d6eebaa4SHervé Poussineau }
119*d6eebaa4SHervé Poussineau else
120*d6eebaa4SHervé Poussineau {
121*d6eebaa4SHervé Poussineau if (the_waiting_fn) {
122*d6eebaa4SHervé Poussineau int expectSomething = the_waiting_fn(sem, NULL);
123*d6eebaa4SHervé Poussineau LWIP_ASSERT("expecting a semaphore count but it's 0", !expectSomething || (*sem > 1));
124*d6eebaa4SHervé Poussineau }
125*d6eebaa4SHervé Poussineau LWIP_ASSERT("*sem > 0", *sem > 0);
126*d6eebaa4SHervé Poussineau if (*sem == 1) {
127*d6eebaa4SHervé Poussineau return SYS_ARCH_TIMEOUT;
128*d6eebaa4SHervé Poussineau }
129*d6eebaa4SHervé Poussineau ret = 1;
130*d6eebaa4SHervé Poussineau }
131*d6eebaa4SHervé Poussineau }
132*d6eebaa4SHervé Poussineau LWIP_ASSERT("*sem > 0", *sem > 0);
133*d6eebaa4SHervé Poussineau (*sem)--;
134*d6eebaa4SHervé Poussineau LWIP_ASSERT("*sem > 0", *sem > 0);
135*d6eebaa4SHervé Poussineau /* return the time we waited for the sem */
136*d6eebaa4SHervé Poussineau return ret;
137*d6eebaa4SHervé Poussineau }
138*d6eebaa4SHervé Poussineau
139*d6eebaa4SHervé Poussineau void
sys_sem_signal(sys_sem_t * sem)140*d6eebaa4SHervé Poussineau sys_sem_signal(sys_sem_t *sem)
141*d6eebaa4SHervé Poussineau {
142*d6eebaa4SHervé Poussineau LWIP_ASSERT("sem != NULL", sem != NULL);
143*d6eebaa4SHervé Poussineau LWIP_ASSERT("*sem > 0", *sem > 0);
144*d6eebaa4SHervé Poussineau (*sem)++;
145*d6eebaa4SHervé Poussineau LWIP_ASSERT("*sem > 0", *sem > 0);
146*d6eebaa4SHervé Poussineau }
147*d6eebaa4SHervé Poussineau
148*d6eebaa4SHervé Poussineau err_t
sys_mutex_new(sys_mutex_t * mutex)149*d6eebaa4SHervé Poussineau sys_mutex_new(sys_mutex_t *mutex)
150*d6eebaa4SHervé Poussineau {
151*d6eebaa4SHervé Poussineau LWIP_ASSERT("mutex != NULL", mutex != NULL);
152*d6eebaa4SHervé Poussineau *mutex = 1; /* 1 allocated */
153*d6eebaa4SHervé Poussineau return ERR_OK;
154*d6eebaa4SHervé Poussineau }
155*d6eebaa4SHervé Poussineau
156*d6eebaa4SHervé Poussineau void
sys_mutex_free(sys_mutex_t * mutex)157*d6eebaa4SHervé Poussineau sys_mutex_free(sys_mutex_t *mutex)
158*d6eebaa4SHervé Poussineau {
159*d6eebaa4SHervé Poussineau /* parameter check */
160*d6eebaa4SHervé Poussineau LWIP_ASSERT("mutex != NULL", mutex != NULL);
161*d6eebaa4SHervé Poussineau LWIP_ASSERT("*mutex >= 1", *mutex >= 1);
162*d6eebaa4SHervé Poussineau *mutex = 0;
163*d6eebaa4SHervé Poussineau }
164*d6eebaa4SHervé Poussineau
165*d6eebaa4SHervé Poussineau void
sys_mutex_set_invalid(sys_mutex_t * mutex)166*d6eebaa4SHervé Poussineau sys_mutex_set_invalid(sys_mutex_t *mutex)
167*d6eebaa4SHervé Poussineau {
168*d6eebaa4SHervé Poussineau LWIP_ASSERT("mutex != NULL", mutex != NULL);
169*d6eebaa4SHervé Poussineau *mutex = 0;
170*d6eebaa4SHervé Poussineau }
171*d6eebaa4SHervé Poussineau
172*d6eebaa4SHervé Poussineau void
sys_mutex_lock(sys_mutex_t * mutex)173*d6eebaa4SHervé Poussineau sys_mutex_lock(sys_mutex_t *mutex)
174*d6eebaa4SHervé Poussineau {
175*d6eebaa4SHervé Poussineau /* nothing to do, no multithreading supported */
176*d6eebaa4SHervé Poussineau LWIP_ASSERT("mutex != NULL", mutex != NULL);
177*d6eebaa4SHervé Poussineau /* check that the mutext is valid and unlocked (no nested locking) */
178*d6eebaa4SHervé Poussineau LWIP_ASSERT("*mutex >= 1", *mutex == 1);
179*d6eebaa4SHervé Poussineau /* we count up just to check the correct pairing of lock/unlock */
180*d6eebaa4SHervé Poussineau (*mutex)++;
181*d6eebaa4SHervé Poussineau LWIP_ASSERT("*mutex >= 1", *mutex >= 1);
182*d6eebaa4SHervé Poussineau }
183*d6eebaa4SHervé Poussineau
184*d6eebaa4SHervé Poussineau void
sys_mutex_unlock(sys_mutex_t * mutex)185*d6eebaa4SHervé Poussineau sys_mutex_unlock(sys_mutex_t *mutex)
186*d6eebaa4SHervé Poussineau {
187*d6eebaa4SHervé Poussineau /* nothing to do, no multithreading supported */
188*d6eebaa4SHervé Poussineau LWIP_ASSERT("mutex != NULL", mutex != NULL);
189*d6eebaa4SHervé Poussineau LWIP_ASSERT("*mutex >= 1", *mutex >= 1);
190*d6eebaa4SHervé Poussineau /* we count down just to check the correct pairing of lock/unlock */
191*d6eebaa4SHervé Poussineau (*mutex)--;
192*d6eebaa4SHervé Poussineau LWIP_ASSERT("*mutex >= 1", *mutex >= 1);
193*d6eebaa4SHervé Poussineau }
194*d6eebaa4SHervé Poussineau
195*d6eebaa4SHervé Poussineau
196*d6eebaa4SHervé Poussineau sys_thread_t
sys_thread_new(const char * name,lwip_thread_fn function,void * arg,int stacksize,int prio)197*d6eebaa4SHervé Poussineau sys_thread_new(const char *name, lwip_thread_fn function, void *arg, int stacksize, int prio)
198*d6eebaa4SHervé Poussineau {
199*d6eebaa4SHervé Poussineau LWIP_UNUSED_ARG(name);
200*d6eebaa4SHervé Poussineau LWIP_UNUSED_ARG(function);
201*d6eebaa4SHervé Poussineau LWIP_UNUSED_ARG(arg);
202*d6eebaa4SHervé Poussineau LWIP_UNUSED_ARG(stacksize);
203*d6eebaa4SHervé Poussineau LWIP_UNUSED_ARG(prio);
204*d6eebaa4SHervé Poussineau /* threads not supported */
205*d6eebaa4SHervé Poussineau return 0;
206*d6eebaa4SHervé Poussineau }
207*d6eebaa4SHervé Poussineau
208*d6eebaa4SHervé Poussineau err_t
sys_mbox_new(sys_mbox_t * mbox,int size)209*d6eebaa4SHervé Poussineau sys_mbox_new(sys_mbox_t *mbox, int size)
210*d6eebaa4SHervé Poussineau {
211*d6eebaa4SHervé Poussineau int mboxsize = size;
212*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox != NULL", mbox != NULL);
213*d6eebaa4SHervé Poussineau LWIP_ASSERT("size >= 0", size >= 0);
214*d6eebaa4SHervé Poussineau if (size == 0) {
215*d6eebaa4SHervé Poussineau mboxsize = 1024;
216*d6eebaa4SHervé Poussineau }
217*d6eebaa4SHervé Poussineau mbox->head = mbox->tail = 0;
218*d6eebaa4SHervé Poussineau mbox->sem = mbox; /* just point to something for sys_mbox_valid() */
219*d6eebaa4SHervé Poussineau mbox->q_mem = (void**)malloc(sizeof(void*)*mboxsize);
220*d6eebaa4SHervé Poussineau mbox->size = mboxsize;
221*d6eebaa4SHervé Poussineau mbox->used = 0;
222*d6eebaa4SHervé Poussineau
223*d6eebaa4SHervé Poussineau memset(mbox->q_mem, 0, sizeof(void*)*mboxsize);
224*d6eebaa4SHervé Poussineau return ERR_OK;
225*d6eebaa4SHervé Poussineau }
226*d6eebaa4SHervé Poussineau
227*d6eebaa4SHervé Poussineau void
sys_mbox_free(sys_mbox_t * mbox)228*d6eebaa4SHervé Poussineau sys_mbox_free(sys_mbox_t *mbox)
229*d6eebaa4SHervé Poussineau {
230*d6eebaa4SHervé Poussineau /* parameter check */
231*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox != NULL", mbox != NULL);
232*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox->sem != NULL", mbox->sem != NULL);
233*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox->sem == mbox", mbox->sem == mbox);
234*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox->q_mem != NULL", mbox->q_mem != NULL);
235*d6eebaa4SHervé Poussineau mbox->sem = NULL;
236*d6eebaa4SHervé Poussineau free(mbox->q_mem);
237*d6eebaa4SHervé Poussineau mbox->q_mem = NULL;
238*d6eebaa4SHervé Poussineau }
239*d6eebaa4SHervé Poussineau
240*d6eebaa4SHervé Poussineau void
sys_mbox_set_invalid(sys_mbox_t * mbox)241*d6eebaa4SHervé Poussineau sys_mbox_set_invalid(sys_mbox_t *mbox)
242*d6eebaa4SHervé Poussineau {
243*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox != NULL", mbox != NULL);
244*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox->q_mem == NULL", mbox->q_mem == NULL);
245*d6eebaa4SHervé Poussineau mbox->sem = NULL;
246*d6eebaa4SHervé Poussineau mbox->q_mem = NULL;
247*d6eebaa4SHervé Poussineau }
248*d6eebaa4SHervé Poussineau
249*d6eebaa4SHervé Poussineau void
sys_mbox_post(sys_mbox_t * q,void * msg)250*d6eebaa4SHervé Poussineau sys_mbox_post(sys_mbox_t *q, void *msg)
251*d6eebaa4SHervé Poussineau {
252*d6eebaa4SHervé Poussineau LWIP_ASSERT("q != SYS_MBOX_NULL", q != SYS_MBOX_NULL);
253*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->sem == q", q->sem == q);
254*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->q_mem != NULL", q->q_mem != NULL);
255*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used >= 0", q->used >= 0);
256*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->size > 0", q->size > 0);
257*d6eebaa4SHervé Poussineau
258*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox already full", q->used < q->size);
259*d6eebaa4SHervé Poussineau
260*d6eebaa4SHervé Poussineau q->q_mem[q->head] = msg;
261*d6eebaa4SHervé Poussineau q->head++;
262*d6eebaa4SHervé Poussineau if (q->head >= (unsigned int)q->size) {
263*d6eebaa4SHervé Poussineau q->head = 0;
264*d6eebaa4SHervé Poussineau }
265*d6eebaa4SHervé Poussineau LWIP_ASSERT("mbox is full!", q->head != q->tail);
266*d6eebaa4SHervé Poussineau q->used++;
267*d6eebaa4SHervé Poussineau }
268*d6eebaa4SHervé Poussineau
269*d6eebaa4SHervé Poussineau err_t
sys_mbox_trypost(sys_mbox_t * q,void * msg)270*d6eebaa4SHervé Poussineau sys_mbox_trypost(sys_mbox_t *q, void *msg)
271*d6eebaa4SHervé Poussineau {
272*d6eebaa4SHervé Poussineau LWIP_ASSERT("q != SYS_MBOX_NULL", q != SYS_MBOX_NULL);
273*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->sem == q", q->sem == q);
274*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->q_mem != NULL", q->q_mem != NULL);
275*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used >= 0", q->used >= 0);
276*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->size > 0", q->size > 0);
277*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used <= q->size", q->used <= q->size);
278*d6eebaa4SHervé Poussineau
279*d6eebaa4SHervé Poussineau if (q->used == q->size) {
280*d6eebaa4SHervé Poussineau return ERR_MEM;
281*d6eebaa4SHervé Poussineau }
282*d6eebaa4SHervé Poussineau sys_mbox_post(q, msg);
283*d6eebaa4SHervé Poussineau return ERR_OK;
284*d6eebaa4SHervé Poussineau }
285*d6eebaa4SHervé Poussineau
286*d6eebaa4SHervé Poussineau err_t
sys_mbox_trypost_fromisr(sys_mbox_t * q,void * msg)287*d6eebaa4SHervé Poussineau sys_mbox_trypost_fromisr(sys_mbox_t *q, void *msg)
288*d6eebaa4SHervé Poussineau {
289*d6eebaa4SHervé Poussineau return sys_mbox_trypost(q, msg);
290*d6eebaa4SHervé Poussineau }
291*d6eebaa4SHervé Poussineau
292*d6eebaa4SHervé Poussineau u32_t
sys_arch_mbox_fetch(sys_mbox_t * q,void ** msg,u32_t timeout)293*d6eebaa4SHervé Poussineau sys_arch_mbox_fetch(sys_mbox_t *q, void **msg, u32_t timeout)
294*d6eebaa4SHervé Poussineau {
295*d6eebaa4SHervé Poussineau u32_t ret = 0;
296*d6eebaa4SHervé Poussineau u32_t ret2;
297*d6eebaa4SHervé Poussineau LWIP_ASSERT("q != SYS_MBOX_NULL", q != SYS_MBOX_NULL);
298*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->sem == q", q->sem == q);
299*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->q_mem != NULL", q->q_mem != NULL);
300*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used >= 0", q->used >= 0);
301*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->size > 0", q->size > 0);
302*d6eebaa4SHervé Poussineau
303*d6eebaa4SHervé Poussineau if (q->used == 0) {
304*d6eebaa4SHervé Poussineau /* need to wait */
305*d6eebaa4SHervé Poussineau /* need to wait */
306*d6eebaa4SHervé Poussineau if(!timeout)
307*d6eebaa4SHervé Poussineau {
308*d6eebaa4SHervé Poussineau /* wait infinite */
309*d6eebaa4SHervé Poussineau LWIP_ASSERT("cannot wait without waiting callback", the_waiting_fn != NULL);
310*d6eebaa4SHervé Poussineau do {
311*d6eebaa4SHervé Poussineau int expectSomething = the_waiting_fn(NULL, q);
312*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used >= 0", q->used >= 0);
313*d6eebaa4SHervé Poussineau LWIP_ASSERT("expecting item available but it's 0", !expectSomething || (q->used > 0));
314*d6eebaa4SHervé Poussineau ret++;
315*d6eebaa4SHervé Poussineau if (ret == SYS_ARCH_TIMEOUT) {
316*d6eebaa4SHervé Poussineau ret--;
317*d6eebaa4SHervé Poussineau }
318*d6eebaa4SHervé Poussineau } while(q->used == 0);
319*d6eebaa4SHervé Poussineau }
320*d6eebaa4SHervé Poussineau else
321*d6eebaa4SHervé Poussineau {
322*d6eebaa4SHervé Poussineau if (the_waiting_fn) {
323*d6eebaa4SHervé Poussineau int expectSomething = the_waiting_fn(NULL, q);
324*d6eebaa4SHervé Poussineau LWIP_ASSERT("expecting item available count but it's 0", !expectSomething || (q->used > 0));
325*d6eebaa4SHervé Poussineau }
326*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used >= 0", q->used >= 0);
327*d6eebaa4SHervé Poussineau if (q->used == 0) {
328*d6eebaa4SHervé Poussineau if(msg) {
329*d6eebaa4SHervé Poussineau *msg = NULL;
330*d6eebaa4SHervé Poussineau }
331*d6eebaa4SHervé Poussineau return SYS_ARCH_TIMEOUT;
332*d6eebaa4SHervé Poussineau }
333*d6eebaa4SHervé Poussineau ret = 1;
334*d6eebaa4SHervé Poussineau }
335*d6eebaa4SHervé Poussineau }
336*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used > 0", q->used > 0);
337*d6eebaa4SHervé Poussineau ret2 = sys_arch_mbox_tryfetch(q, msg);
338*d6eebaa4SHervé Poussineau LWIP_ASSERT("got no message", ret2 == 0);
339*d6eebaa4SHervé Poussineau return ret;
340*d6eebaa4SHervé Poussineau }
341*d6eebaa4SHervé Poussineau
342*d6eebaa4SHervé Poussineau u32_t
sys_arch_mbox_tryfetch(sys_mbox_t * q,void ** msg)343*d6eebaa4SHervé Poussineau sys_arch_mbox_tryfetch(sys_mbox_t *q, void **msg)
344*d6eebaa4SHervé Poussineau {
345*d6eebaa4SHervé Poussineau LWIP_ASSERT("q != SYS_MBOX_NULL", q != SYS_MBOX_NULL);
346*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->sem == q", q->sem == q);
347*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->q_mem != NULL", q->q_mem != NULL);
348*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used >= 0", q->used >= 0);
349*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->size > 0", q->size > 0);
350*d6eebaa4SHervé Poussineau
351*d6eebaa4SHervé Poussineau if (!q->used) {
352*d6eebaa4SHervé Poussineau return SYS_ARCH_TIMEOUT;
353*d6eebaa4SHervé Poussineau }
354*d6eebaa4SHervé Poussineau if(msg) {
355*d6eebaa4SHervé Poussineau *msg = q->q_mem[q->tail];
356*d6eebaa4SHervé Poussineau }
357*d6eebaa4SHervé Poussineau
358*d6eebaa4SHervé Poussineau q->tail++;
359*d6eebaa4SHervé Poussineau if (q->tail >= (unsigned int)q->size) {
360*d6eebaa4SHervé Poussineau q->tail = 0;
361*d6eebaa4SHervé Poussineau }
362*d6eebaa4SHervé Poussineau q->used--;
363*d6eebaa4SHervé Poussineau LWIP_ASSERT("q->used >= 0", q->used >= 0);
364*d6eebaa4SHervé Poussineau return 0;
365*d6eebaa4SHervé Poussineau }
366*d6eebaa4SHervé Poussineau
367*d6eebaa4SHervé Poussineau #if LWIP_NETCONN_SEM_PER_THREAD
368*d6eebaa4SHervé Poussineau /* Simple implementation of this: unit tests only support one thread */
369*d6eebaa4SHervé Poussineau static sys_sem_t global_netconn_sem;
370*d6eebaa4SHervé Poussineau
sys_arch_netconn_sem_get(void)371*d6eebaa4SHervé Poussineau sys_sem_t* sys_arch_netconn_sem_get(void)
372*d6eebaa4SHervé Poussineau {
373*d6eebaa4SHervé Poussineau return &global_netconn_sem;
374*d6eebaa4SHervé Poussineau }
375*d6eebaa4SHervé Poussineau
sys_arch_netconn_sem_alloc(void)376*d6eebaa4SHervé Poussineau void sys_arch_netconn_sem_alloc(void)
377*d6eebaa4SHervé Poussineau {
378*d6eebaa4SHervé Poussineau sys_sem_new(&global_netconn_sem, 0);
379*d6eebaa4SHervé Poussineau }
380*d6eebaa4SHervé Poussineau
sys_arch_netconn_sem_free(void)381*d6eebaa4SHervé Poussineau void sys_arch_netconn_sem_free(void)
382*d6eebaa4SHervé Poussineau {
383*d6eebaa4SHervé Poussineau sys_sem_free(&global_netconn_sem);
384*d6eebaa4SHervé Poussineau }
385*d6eebaa4SHervé Poussineau #endif /* LWIP_NETCONN_SEM_PER_THREAD */
386*d6eebaa4SHervé Poussineau
387*d6eebaa4SHervé Poussineau #endif /* !NO_SYS */
388