123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- /*
- * Copyright (c) 2019 Intel Corporation
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #ifndef ZEPHYR_INCLUDE_SYS_MUTEX_H_
- #define ZEPHYR_INCLUDE_SYS_MUTEX_H_
- /*
- * sys_mutex behaves almost exactly like k_mutex, with the added advantage
- * that a sys_mutex instance can reside in user memory.
- *
- * Further enhancements will support locking/unlocking uncontended sys_mutexes
- * with simple atomic ops instead of syscalls, similar to Linux's
- * FUTEX_LOCK_PI and FUTEX_UNLOCK_PI
- */
- #ifdef __cplusplus
- extern "C" {
- #endif
- #ifdef CONFIG_USERSPACE
- #include <sys/atomic.h>
- #include <zephyr/types.h>
- #include <sys_clock.h>
- struct sys_mutex {
- /* Currently unused, but will be used to store state for fast mutexes
- * that can be locked/unlocked with atomic ops if there is no
- * contention
- */
- atomic_t val;
- };
- /**
- * @defgroup user_mutex_apis User mode mutex APIs
- * @ingroup kernel_apis
- * @{
- */
- /**
- * @brief Statically define and initialize a sys_mutex
- *
- * The mutex can be accessed outside the module where it is defined using:
- *
- * @code extern struct sys_mutex <name>; @endcode
- *
- * Route this to memory domains using K_APP_DMEM().
- *
- * @param name Name of the mutex.
- */
- #define SYS_MUTEX_DEFINE(name) \
- struct sys_mutex name
- /**
- * @brief Initialize a mutex.
- *
- * This routine initializes a mutex object, prior to its first use.
- *
- * Upon completion, the mutex is available and does not have an owner.
- *
- * This routine is only necessary to call when userspace is disabled
- * and the mutex was not created with SYS_MUTEX_DEFINE().
- *
- * @param mutex Address of the mutex.
- *
- * @return N/A
- */
- static inline void sys_mutex_init(struct sys_mutex *mutex)
- {
- ARG_UNUSED(mutex);
- /* Nothing to do, kernel-side data structures are initialized at
- * boot
- */
- }
- __syscall int z_sys_mutex_kernel_lock(struct sys_mutex *mutex,
- k_timeout_t timeout);
- __syscall int z_sys_mutex_kernel_unlock(struct sys_mutex *mutex);
- /**
- * @brief Lock a mutex.
- *
- * This routine locks @a mutex. If the mutex is locked by another thread,
- * the calling thread waits until the mutex becomes available or until
- * a timeout occurs.
- *
- * A thread is permitted to lock a mutex it has already locked. The operation
- * completes immediately and the lock count is increased by 1.
- *
- * @param mutex Address of the mutex, which may reside in user memory
- * @param timeout Waiting period to lock the mutex,
- * or one of the special values K_NO_WAIT and K_FOREVER.
- *
- * @retval 0 Mutex locked.
- * @retval -EBUSY Returned without waiting.
- * @retval -EAGAIN Waiting period timed out.
- * @retval -EACCES Caller has no access to provided mutex address
- * @retval -EINVAL Provided mutex not recognized by the kernel
- */
- static inline int sys_mutex_lock(struct sys_mutex *mutex, k_timeout_t timeout)
- {
- /* For now, make the syscall unconditionally */
- return z_sys_mutex_kernel_lock(mutex, timeout);
- }
- /**
- * @brief Unlock a mutex.
- *
- * This routine unlocks @a mutex. The mutex must already be locked by the
- * calling thread.
- *
- * The mutex cannot be claimed by another thread until it has been unlocked by
- * the calling thread as many times as it was previously locked by that
- * thread.
- *
- * @param mutex Address of the mutex, which may reside in user memory
- * @retval 0 Mutex unlocked
- * @retval -EACCES Caller has no access to provided mutex address
- * @retval -EINVAL Provided mutex not recognized by the kernel or mutex wasn't
- * locked
- * @retval -EPERM Caller does not own the mutex
- */
- static inline int sys_mutex_unlock(struct sys_mutex *mutex)
- {
- /* For now, make the syscall unconditionally */
- return z_sys_mutex_kernel_unlock(mutex);
- }
- #include <syscalls/mutex.h>
- #else
- #include <kernel.h>
- #include <kernel_structs.h>
- struct sys_mutex {
- struct k_mutex kernel_mutex;
- };
- #define SYS_MUTEX_DEFINE(name) \
- struct sys_mutex name = { \
- .kernel_mutex = Z_MUTEX_INITIALIZER(name.kernel_mutex) \
- }
- static inline void sys_mutex_init(struct sys_mutex *mutex)
- {
- k_mutex_init(&mutex->kernel_mutex);
- }
- static inline int sys_mutex_lock(struct sys_mutex *mutex, k_timeout_t timeout)
- {
- return k_mutex_lock(&mutex->kernel_mutex, timeout);
- }
- static inline int sys_mutex_unlock(struct sys_mutex *mutex)
- {
- return k_mutex_unlock(&mutex->kernel_mutex);
- }
- #endif /* CONFIG_USERSPACE */
- /**
- * @}
- */
- #ifdef __cplusplus
- }
- #endif
- #endif /* ZEPHYR_INCLUDE_SYS_MUTEX_H_ */
|