/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ #ifndef _NIMBLE_NPL_OS_H_ #define _NIMBLE_NPL_OS_H_ #include #include #include "event/callback.h" #include "mutex.h" #include "sema.h" #include "ztimer.h" #if defined(CPU_FAM_NRF51) || defined(CPU_FAM_NRF52) #include "nrf_clock.h" #endif #ifdef __cplusplus extern "C" { #endif #define BLE_NPL_OS_ALIGNMENT 4 #define BLE_NPL_TIME_FOREVER UINT32_MAX typedef uint32_t ble_npl_time_t; typedef int32_t ble_npl_stime_t; struct ble_npl_event { event_callback_t e; void *arg; }; struct ble_npl_eventq { event_queue_t q; }; struct ble_npl_callout { ztimer_t timer; ble_npl_time_t ticks; struct ble_npl_event e; event_queue_t *q; }; struct ble_npl_mutex { mutex_t mu; }; struct ble_npl_sem { sema_t sem; }; static inline bool ble_npl_os_started(void) { return true; } static inline void * ble_npl_get_current_task_id(void) { return (void *)(uint32_t)thread_getpid(); } static inline void ble_npl_eventq_init(struct ble_npl_eventq *evq) { event_queue_init_detached(&evq->q); } static inline void ble_npl_eventq_put(struct ble_npl_eventq *evq, struct ble_npl_event *ev) { event_post(&evq->q, &ev->e.super); } static inline struct ble_npl_event * ble_npl_eventq_get(struct ble_npl_eventq *evq, ble_npl_time_t tmo) { if (evq->q.waiter == NULL) { event_queue_claim(&evq->q); } if (tmo == 0) { return (struct ble_npl_event *)event_get(&evq->q); } else if (tmo == BLE_NPL_TIME_FOREVER) { return (struct ble_npl_event *)event_wait(&evq->q); } else { return (struct ble_npl_event *)event_wait_timeout_ztimer(&evq->q, ZTIMER_MSEC, (uint32_t)tmo); } } static inline void ble_npl_eventq_remove(struct ble_npl_eventq *evq, struct ble_npl_event *ev) { event_cancel(&evq->q, &ev->e.super); } static inline void ble_npl_event_run(struct ble_npl_event *ev) { ev->e.super.handler(&ev->e.super); } static inline void ble_npl_event_init(struct ble_npl_event *ev, ble_npl_event_fn *fn, void *arg) { /* * Need to clear list_node manually since init function below does not do * this. */ ev->e.super.list_node.next = NULL; event_callback_init(&ev->e, (void(*)(void *))fn, ev); ev->arg = arg; } static inline bool ble_npl_event_is_queued(struct ble_npl_event *ev) { return (ev->e.super.list_node.next != NULL); } static inline void * ble_npl_event_get_arg(struct ble_npl_event *ev) { return ev->arg; } static inline void ble_npl_event_set_arg(struct ble_npl_event *ev, void *arg) { ev->arg = arg; } static inline ble_npl_error_t ble_npl_mutex_init(struct ble_npl_mutex *mu) { mutex_init(&mu->mu); return BLE_NPL_OK; } static inline ble_npl_error_t ble_npl_mutex_pend(struct ble_npl_mutex *mu, ble_npl_time_t timeout) { assert(timeout == BLE_NPL_TIME_FOREVER); (void)timeout; mutex_lock(&mu->mu); return BLE_NPL_OK; } static inline ble_npl_error_t ble_npl_mutex_release(struct ble_npl_mutex *mu) { mutex_unlock(&mu->mu); return BLE_NPL_OK; } static inline ble_npl_error_t ble_npl_sem_init(struct ble_npl_sem *sem, uint16_t tokens) { sema_create(&sem->sem, (unsigned)tokens); return BLE_NPL_OK; } static inline ble_npl_error_t ble_npl_sem_release(struct ble_npl_sem *sem) { int rc = sema_post(&sem->sem); return (rc == 0) ? BLE_NPL_OK : BLE_NPL_ERROR; } static inline uint16_t ble_npl_sem_get_count(struct ble_npl_sem *sem) { return (uint16_t)sema_get_value(&sem->sem); } static inline void ble_npl_callout_stop(struct ble_npl_callout *co) { ztimer_remove(ZTIMER_MSEC, &co->timer); } static inline bool ble_npl_callout_is_active(struct ble_npl_callout *c) { return ztimer_is_set(ZTIMER_MSEC, &c->timer); } static inline ble_npl_time_t ble_npl_callout_get_ticks(struct ble_npl_callout *co) { return co->ticks; } static inline void ble_npl_callout_set_arg(struct ble_npl_callout *co, void *arg) { co->e.arg = arg; } static inline ble_npl_time_t ble_npl_time_get(void) { return (ble_npl_time_t)ztimer_now(ZTIMER_MSEC); } static inline ble_npl_error_t ble_npl_time_ms_to_ticks(uint32_t ms, ble_npl_time_t *out_ticks) { *out_ticks = ms; return BLE_NPL_OK; } static inline ble_npl_error_t ble_npl_time_ticks_to_ms(ble_npl_time_t ticks, uint32_t *out_ms) { *out_ms = ticks; return BLE_NPL_OK; } static inline ble_npl_time_t ble_npl_time_ms_to_ticks32(uint32_t ms) { return ms; } static inline uint32_t ble_npl_time_ticks_to_ms32(ble_npl_time_t ticks) { return ticks; } static inline void ble_npl_time_delay(ble_npl_time_t ticks) { ztimer_sleep(ZTIMER_MSEC, (uint32_t)ticks); } static inline uint32_t ble_npl_hw_enter_critical(void) { return (uint32_t)irq_disable(); } static inline void ble_npl_hw_exit_critical(uint32_t ctx) { irq_restore((unsigned)ctx); } static inline bool ble_npl_hw_is_in_critical(void) { return (bool)!irq_is_enabled(); } /* XXX: these functions are required to build hal_timer.c, however with the * default configuration they are never used... */ #if defined(CPU_FAM_NRF51) || defined(CPU_FAM_NRF52) static inline void nrf52_clock_hfxo_request(void) { clock_hfxo_request(); } static inline void nrf52_clock_hfxo_release(void) { clock_hfxo_release(); } #endif #ifdef __cplusplus } #endif #endif /* _NPL_H_ */