xref: /nrf52832-nimble/packages/NimBLE-latest/porting/npl/rtthread/src/npl_os_rtthread.c (revision 042d53a763ad75cb1465103098bb88c245d95138)
1 /*
2  * SPDX-License-Identifier: Apache-2.0
3  *
4  * Date           Author       Notes
5  * 2018-12-18     ZeroFree     first implementation
6  */
7 
8 #include <assert.h>
9 #include <stddef.h>
10 #include <string.h>
11 #include "nimble/nimble_npl.h"
12 
13 #include <rtthread.h>
14 #include <rthw.h>
15 
16 static void npl_rtthread_callout_init(struct ble_npl_callout *co, struct ble_npl_eventq *evq, ble_npl_event_fn *ev_cb, void *ev_arg);
17 static ble_npl_error_t npl_rtthread_callout_reset(struct ble_npl_callout *co, ble_npl_time_t ticks);
18 static ble_npl_time_t npl_rtthread_callout_remaining_ticks(struct ble_npl_callout *co, ble_npl_time_t now);
19 static ble_npl_error_t npl_rtthread_time_ms_to_ticks(uint32_t ms, ble_npl_time_t *out_ticks);
20 static ble_npl_error_t npl_rtthread_time_ticks_to_ms(ble_npl_time_t ticks, uint32_t *out_ms);
21 
ble_npl_get_current_task_id(void)22 void *ble_npl_get_current_task_id(void)
23 {
24     return rt_thread_self();
25 }
26 
ble_npl_task_init(struct ble_npl_task * t,const char * name,ble_npl_task_fn * func,void * arg,uint8_t prio,uint32_t sanity_itvl,uint32_t * stack_bottom,uint16_t stack_size)27 int ble_npl_task_init(struct ble_npl_task *t, const char *name,
28         ble_npl_task_fn *func,void *arg, uint8_t prio,
29         uint32_t sanity_itvl, uint32_t *stack_bottom, uint16_t stack_size)
30 {
31     rt_thread_t tid;
32 
33     tid = rt_thread_create(name, func, arg, stack_size, prio, 10);
34     if(tid)
35     {
36         t->t = tid;
37         rt_thread_startup(tid);
38         return RT_EOK;
39     }
40     return -RT_ERROR;
41 }
42 
ble_npl_eventq_init(struct ble_npl_eventq * evq)43 void ble_npl_eventq_init(struct ble_npl_eventq *evq)
44 {
45     evq->q = rt_mq_create("npl_evq", sizeof(struct ble_npl_eventq *), 32, RT_IPC_FLAG_FIFO);
46 }
47 
ble_npl_eventq_get(struct ble_npl_eventq * evq,ble_npl_time_t tmo)48 struct ble_npl_event *ble_npl_eventq_get(struct ble_npl_eventq *evq, ble_npl_time_t tmo)
49 {
50     struct ble_npl_event *ev = NULL;
51 
52     rt_mq_recv((struct rt_messagequeue *)evq->q, &ev, sizeof(struct ble_npl_eventq *), tmo);
53 
54     if (ev)
55     {
56         ev->queued = false;
57     }
58 
59     return ev;
60 }
61 
ble_npl_eventq_put(struct ble_npl_eventq * evq,struct ble_npl_event * ev)62 void ble_npl_eventq_put(struct ble_npl_eventq *evq, struct ble_npl_event *ev)
63 {
64     int ret;
65 
66     if (ev->queued)
67     {
68         return;
69     }
70 
71     ev->queued = true;
72     ret = rt_mq_send((struct rt_messagequeue *)evq->q, &ev, sizeof(struct ble_npl_eventq *));
73 
74     RT_ASSERT(ret == RT_EOK);
75 }
76 
ble_npl_eventq_remove(struct ble_npl_eventq * evq,struct ble_npl_event * ev)77 void ble_npl_eventq_remove(struct ble_npl_eventq *evq, struct ble_npl_event *ev)
78 {
79     if (!ev->queued)
80     {
81         return;
82     }
83 
84     rt_mq_control((struct rt_messagequeue *)evq->q, RT_IPC_CMD_RESET, RT_NULL);
85     ev->queued = false;
86 }
87 
ble_npl_event_run(struct ble_npl_event * ev)88 void ble_npl_event_run(struct ble_npl_event *ev)
89 {
90     ev->fn(ev);
91 }
92 
ble_npl_eventq_is_empty(struct ble_npl_eventq * evq)93 bool ble_npl_eventq_is_empty(struct ble_npl_eventq *evq)
94 {
95     int count;
96     rt_base_t level;
97 
98     level = rt_hw_interrupt_disable();
99     count = ((struct rt_messagequeue *)evq->q)->entry;
100     rt_hw_interrupt_enable(level);
101 
102     return count ? true : false;
103 }
104 
ble_npl_event_init(struct ble_npl_event * ev,ble_npl_event_fn * fn,void * arg)105 void ble_npl_event_init(struct ble_npl_event *ev, ble_npl_event_fn *fn,
106                         void *arg)
107 {
108     memset(ev, 0, sizeof(*ev));
109     ev->fn = fn;
110     ev->arg = arg;
111 }
112 
ble_npl_event_is_queued(struct ble_npl_event * ev)113 bool ble_npl_event_is_queued(struct ble_npl_event *ev)
114 {
115     return ev->queued;
116 }
117 
ble_npl_event_get_arg(struct ble_npl_event * ev)118 void *ble_npl_event_get_arg(struct ble_npl_event *ev)
119 {
120     return ev->arg;
121 }
122 
ble_npl_event_set_arg(struct ble_npl_event * ev,void * arg)123 void ble_npl_event_set_arg(struct ble_npl_event *ev, void *arg)
124 {
125     ev->arg = arg;
126 }
127 
ble_npl_mutex_init(struct ble_npl_mutex * mu)128 ble_npl_error_t ble_npl_mutex_init(struct ble_npl_mutex *mu)
129 {
130     if (!mu)
131     {
132         return BLE_NPL_INVALID_PARAM;
133     }
134 
135     mu->handle = rt_mutex_create("mutx", RT_IPC_FLAG_FIFO);
136     RT_ASSERT(mu->handle);
137 
138     return BLE_NPL_OK;
139 }
140 
ble_npl_mutex_pend(struct ble_npl_mutex * mu,ble_npl_time_t timeout)141 ble_npl_error_t ble_npl_mutex_pend(struct ble_npl_mutex *mu, ble_npl_time_t timeout)
142 {
143     int ret;
144 
145     if (!mu)
146     {
147         return BLE_NPL_INVALID_PARAM;
148     }
149 
150     RT_ASSERT(mu->handle);
151 
152     ret = rt_mutex_take((struct rt_mutex *)mu->handle, timeout);
153 
154     return ret == RT_EOK ? BLE_NPL_OK : BLE_NPL_TIMEOUT;
155 }
156 
ble_npl_mutex_release(struct ble_npl_mutex * mu)157 ble_npl_error_t ble_npl_mutex_release(struct ble_npl_mutex *mu)
158 {
159     int ret;
160 
161     if (!mu)
162     {
163         return BLE_NPL_INVALID_PARAM;
164     }
165 
166     RT_ASSERT(mu->handle);
167 
168     ret = rt_mutex_release((struct rt_mutex *)mu->handle);
169 
170     return ret == RT_EOK ? BLE_NPL_OK : BLE_NPL_ERROR;
171 }
172 
ble_npl_sem_init(struct ble_npl_sem * sem,uint16_t tokens)173 ble_npl_error_t ble_npl_sem_init(struct ble_npl_sem *sem, uint16_t tokens)
174 {
175     if (!sem)
176     {
177         return BLE_NPL_INVALID_PARAM;
178     }
179 
180     sem->handle = rt_sem_create("npl_sem", tokens, RT_IPC_FLAG_FIFO);
181     RT_ASSERT(sem->handle);
182 
183     return BLE_NPL_OK;
184 }
185 
ble_npl_sem_pend(struct ble_npl_sem * sem,ble_npl_time_t timeout)186 ble_npl_error_t ble_npl_sem_pend(struct ble_npl_sem *sem, ble_npl_time_t timeout)
187 {
188     int ret;
189 
190     if (!sem)
191     {
192         return BLE_NPL_INVALID_PARAM;
193     }
194 
195     RT_ASSERT(sem->handle);
196 
197     ret = rt_sem_take((struct rt_semaphore *)sem->handle, timeout);
198 
199     return ret == RT_EOK ? BLE_NPL_OK : BLE_NPL_TIMEOUT;
200 }
201 
ble_npl_sem_release(struct ble_npl_sem * sem)202 ble_npl_error_t ble_npl_sem_release(struct ble_npl_sem *sem)
203 {
204     int ret;
205 
206     if (!sem)
207     {
208         return BLE_NPL_INVALID_PARAM;
209     }
210 
211     RT_ASSERT(sem->handle);
212 
213     ret = rt_sem_release((struct rt_semaphore *) sem->handle);
214 
215     return ret == RT_EOK ? BLE_NPL_OK : BLE_NPL_ERROR;
216 }
217 
ble_npl_sem_get_count(struct ble_npl_sem * sem)218 uint16_t ble_npl_sem_get_count(struct ble_npl_sem *sem)
219 {
220     int count;
221     rt_base_t level;
222 
223     RT_ASSERT(sem->handle);
224 
225     level = rt_hw_interrupt_disable();
226     count = ((struct rt_semaphore *)sem->handle)->value;
227     rt_hw_interrupt_enable(level);
228 
229     return count;
230 }
231 
ble_npl_callout_init(struct ble_npl_callout * co,struct ble_npl_eventq * evq,ble_npl_event_fn * ev_cb,void * ev_arg)232 void ble_npl_callout_init(struct ble_npl_callout *co, struct ble_npl_eventq *evq,
233                           ble_npl_event_fn *ev_cb, void *ev_arg)
234 {
235     npl_rtthread_callout_init(co, evq, ev_cb, ev_arg);
236 }
237 
ble_npl_callout_reset(struct ble_npl_callout * co,ble_npl_time_t ticks)238 ble_npl_error_t ble_npl_callout_reset(struct ble_npl_callout *co, ble_npl_time_t ticks)
239 {
240     return npl_rtthread_callout_reset(co, ticks);
241 }
242 
ble_npl_callout_stop(struct ble_npl_callout * co)243 void ble_npl_callout_stop(struct ble_npl_callout *co)
244 {
245     if (co->handle)
246         rt_timer_stop((struct rt_timer *)co->handle);
247 }
248 
ble_npl_callout_is_active(struct ble_npl_callout * co)249 bool ble_npl_callout_is_active(struct ble_npl_callout *co)
250 {
251     return (((struct rt_timer *)co->handle)->parent.flag & RT_TIMER_FLAG_ACTIVATED) ? true : false;
252 }
253 
ble_npl_callout_get_ticks(struct ble_npl_callout * co)254 ble_npl_time_t ble_npl_callout_get_ticks(struct ble_npl_callout *co)
255 {
256     return ((struct rt_timer *)co->handle)->timeout_tick;
257 }
258 
ble_npl_callout_remaining_ticks(struct ble_npl_callout * co,ble_npl_time_t time)259 ble_npl_time_t ble_npl_callout_remaining_ticks(struct ble_npl_callout *co,
260         ble_npl_time_t time)
261 {
262     return npl_rtthread_callout_remaining_ticks(co, time);
263 }
264 
ble_npl_callout_set_arg(struct ble_npl_callout * co,void * arg)265 void ble_npl_callout_set_arg(struct ble_npl_callout *co, void *arg)
266 {
267     co->ev.arg = arg;
268 }
269 
ble_npl_time_get(void)270 ble_npl_time_t ble_npl_time_get(void)
271 {
272     return rt_tick_get();
273 }
274 
ble_npl_time_ms_to_ticks(uint32_t ms,ble_npl_time_t * out_ticks)275 ble_npl_error_t ble_npl_time_ms_to_ticks(uint32_t ms, ble_npl_time_t *out_ticks)
276 {
277     return npl_rtthread_time_ms_to_ticks(ms, out_ticks);
278 }
279 
ble_npl_time_ticks_to_ms(ble_npl_time_t ticks,uint32_t * out_ms)280 ble_npl_error_t ble_npl_time_ticks_to_ms(ble_npl_time_t ticks, uint32_t *out_ms)
281 {
282     return npl_rtthread_time_ticks_to_ms(ticks, out_ms);
283 }
284 
ble_npl_time_ms_to_ticks32(uint32_t ms)285 ble_npl_time_t ble_npl_time_ms_to_ticks32(uint32_t ms)
286 {
287     return ms * RT_TICK_PER_SECOND / 1000;
288 }
289 
ble_npl_time_ticks_to_ms32(ble_npl_time_t ticks)290 uint32_t ble_npl_time_ticks_to_ms32(ble_npl_time_t ticks)
291 {
292     return ticks * 1000 / RT_TICK_PER_SECOND;
293 }
294 
ble_npl_time_delay(ble_npl_time_t ticks)295 void ble_npl_time_delay(ble_npl_time_t ticks)
296 {
297     rt_thread_delay(ticks);
298 }
299 
ble_npl_hw_enter_critical(void)300 uint32_t ble_npl_hw_enter_critical(void)
301 {
302     return rt_hw_interrupt_disable();
303 }
304 
ble_npl_hw_exit_critical(uint32_t ctx)305 void ble_npl_hw_exit_critical(uint32_t ctx)
306 {
307     rt_hw_interrupt_enable(ctx);
308 }
309 
os_callout_timer_cb(void * parameter)310 static void os_callout_timer_cb(void *parameter)
311 {
312     struct ble_npl_callout *co;
313 
314     co = (struct ble_npl_callout *)parameter;
315     RT_ASSERT(co);
316 
317     if (co->evq)
318     {
319         ble_npl_eventq_put(co->evq, &co->ev);
320     }
321     else
322     {
323         co->ev.fn(&co->ev);
324     }
325 }
326 
npl_rtthread_callout_init(struct ble_npl_callout * co,struct ble_npl_eventq * evq,ble_npl_event_fn * ev_cb,void * ev_arg)327 static void npl_rtthread_callout_init(struct ble_npl_callout *co, struct ble_npl_eventq *evq,
328                                       ble_npl_event_fn *ev_cb, void *ev_arg)
329 {
330     memset(co, 0, sizeof(*co));
331     co->handle = rt_timer_create("co", os_callout_timer_cb, co, 0, RT_TIMER_FLAG_ONE_SHOT | RT_TIMER_FLAG_SOFT_TIMER);
332     co->evq = evq;
333     ble_npl_event_init(&co->ev, ev_cb, ev_arg);
334 }
335 
npl_rtthread_callout_reset(struct ble_npl_callout * co,ble_npl_time_t ticks)336 static ble_npl_error_t npl_rtthread_callout_reset(struct ble_npl_callout *co, ble_npl_time_t ticks)
337 {
338     if (ticks == 0)
339     {
340         ticks = 1;
341     }
342 
343     rt_timer_stop((struct rt_timer *)co->handle);
344     rt_timer_control((struct rt_timer *)co->handle, RT_TIMER_CTRL_SET_TIME, &ticks);
345     rt_timer_start((struct rt_timer *)co->handle);
346 
347     return BLE_NPL_OK;
348 }
349 
npl_rtthread_callout_remaining_ticks(struct ble_npl_callout * co,ble_npl_time_t now)350 static ble_npl_time_t npl_rtthread_callout_remaining_ticks(struct ble_npl_callout *co,
351         ble_npl_time_t now)
352 {
353     ble_npl_time_t rt;
354     uint32_t exp;
355     rt_base_t level;
356 
357     level = rt_hw_interrupt_disable();
358     exp = ((struct rt_timer *)co->handle)->timeout_tick;
359     rt_hw_interrupt_enable(level);
360 
361     if (exp > now)
362     {
363         rt = exp - now;
364     }
365     else
366     {
367         rt = 0;
368     }
369 
370     return rt;
371 }
372 
npl_rtthread_time_ms_to_ticks(uint32_t ms,ble_npl_time_t * out_ticks)373 static ble_npl_error_t npl_rtthread_time_ms_to_ticks(uint32_t ms, ble_npl_time_t *out_ticks)
374 {
375     uint64_t ticks;
376 
377     ticks = rt_tick_from_millisecond(ms);
378     if (ticks > UINT32_MAX)
379     {
380         return BLE_NPL_EINVAL;
381     }
382 
383     *out_ticks = ticks;
384 
385     return BLE_NPL_OK;
386 }
387 
npl_rtthread_time_ticks_to_ms(ble_npl_time_t ticks,uint32_t * out_ms)388 static ble_npl_error_t npl_rtthread_time_ticks_to_ms(ble_npl_time_t ticks, uint32_t *out_ms)
389 {
390     uint64_t ms;
391 
392     ms = ((uint64_t)ticks * 1000) / RT_TICK_PER_SECOND;
393     if (ms > UINT32_MAX)
394     {
395         return BLE_NPL_EINVAL;
396     }
397 
398     *out_ms = ms;
399 
400     return BLE_NPL_OK;
401 }
402