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