kernel: convert k_sem APIs to system calls

Signed-off-by: Andrew Boie <andrew.p.boie@intel.com>
This commit is contained in:
Andrew Boie 2017-09-23 12:51:23 -07:00 committed by Andrew Boie
commit fc273c0b23
4 changed files with 102 additions and 10 deletions

View file

@ -2477,8 +2477,13 @@ struct k_sem {
*
* @return N/A
*/
extern void k_sem_init(struct k_sem *sem, unsigned int initial_count,
unsigned int limit);
static inline void k_sem_init(struct k_sem *sem, unsigned int initial_count,
unsigned int limit);
K_SYSCALL_DECLARE3_VOID(K_SYSCALL_SEM_INIT, k_sem_init,
struct k_sem *, sem,
unsigned int, initial_count,
unsigned int, limit);
/**
* @brief Take a semaphore.
@ -2501,7 +2506,11 @@ extern void k_sem_init(struct k_sem *sem, unsigned int initial_count,
* @retval -EBUSY Returned without waiting.
* @retval -EAGAIN Waiting period timed out.
*/
extern int k_sem_take(struct k_sem *sem, s32_t timeout);
static inline int k_sem_take(struct k_sem *sem, s32_t timeout);
K_SYSCALL_DECLARE2(K_SYSCALL_SEM_TAKE, k_sem_take, int,
struct k_sem *, sem,
s32_t, timeout);
/**
* @brief Give a semaphore.
@ -2515,7 +2524,10 @@ extern int k_sem_take(struct k_sem *sem, s32_t timeout);
*
* @return N/A
*/
extern void k_sem_give(struct k_sem *sem);
static inline void k_sem_give(struct k_sem *sem);
K_SYSCALL_DECLARE1_VOID(K_SYSCALL_SEM_GIVE, k_sem_give,
struct k_sem *, sem);
/**
* @brief Reset a semaphore's count to zero.
@ -2526,11 +2538,16 @@ extern void k_sem_give(struct k_sem *sem);
*
* @return N/A
*/
static inline void k_sem_reset(struct k_sem *sem)
static inline void k_sem_reset(struct k_sem *sem);
static inline void _impl_k_sem_reset(struct k_sem *sem)
{
sem->count = 0;
}
K_SYSCALL_DECLARE1_VOID_INLINE(K_SYSCALL_SEM_RESET, k_sem_reset,
struct k_sem *, sem);
/**
* @brief Get a semaphore's count.
*
@ -2540,11 +2557,16 @@ static inline void k_sem_reset(struct k_sem *sem)
*
* @return Current semaphore count.
*/
static inline unsigned int k_sem_count_get(struct k_sem *sem)
static inline unsigned int k_sem_count_get(struct k_sem *sem);
static inline unsigned int _impl_k_sem_count_get(struct k_sem *sem)
{
return sem->count;
}
K_SYSCALL_DECLARE1_INLINE(K_SYSCALL_SEM_COUNT_GET, k_sem_count_get,
unsigned int, struct k_sem *, sem);
/**
* @brief Statically define and initialize a semaphore.
*

View file

@ -12,6 +12,11 @@
enum {
K_SYSCALL_BAD,
K_SYSCALL_SEM_INIT,
K_SYSCALL_SEM_GIVE,
K_SYSCALL_SEM_TAKE,
K_SYSCALL_SEM_RESET,
K_SYSCALL_SEM_COUNT_GET,
K_SYSCALL_LIMIT /* Always last */
};

View file

@ -25,6 +25,7 @@
#include <misc/dlist.h>
#include <ksched.h>
#include <init.h>
#include <syscall_handler.h>
extern struct k_sem _k_sem_list_start[];
extern struct k_sem _k_sem_list_end[];
@ -52,8 +53,8 @@ SYS_INIT(init_sem_module, PRE_KERNEL_1, CONFIG_KERNEL_INIT_PRIORITY_OBJECTS);
#endif /* CONFIG_OBJECT_TRACING */
void k_sem_init(struct k_sem *sem, unsigned int initial_count,
unsigned int limit)
void _impl_k_sem_init(struct k_sem *sem, unsigned int initial_count,
unsigned int limit)
{
__ASSERT(limit != 0, "limit cannot be zero");
@ -69,6 +70,18 @@ void k_sem_init(struct k_sem *sem, unsigned int initial_count,
_k_object_init(sem);
}
#ifdef CONFIG_USERSPACE
u32_t _handler_k_sem_init(u32_t sem_ptr, u32_t initial_count, u32_t limit,
u32_t arg4, u32_t arg5, u32_t arg6, void *ssf)
{
_SYSCALL_ARG3;
_SYSCALL_IS_OBJ(sem_ptr, K_OBJ_SEM, 1, ssf);
_SYSCALL_VERIFY(limit != 0, ssf);
_impl_k_sem_init((struct k_sem *)sem_ptr, initial_count, limit);
return 0;
}
#endif
/* returns 1 if a reschedule must take place, 0 otherwise */
static inline int handle_poll_events(struct k_sem *sem)
@ -129,7 +142,7 @@ void _sem_give_non_preemptible(struct k_sem *sem)
_set_thread_return_value(thread, 0);
}
void k_sem_give(struct k_sem *sem)
void _impl_k_sem_give(struct k_sem *sem)
{
unsigned int key;
@ -142,7 +155,20 @@ void k_sem_give(struct k_sem *sem)
}
}
int k_sem_take(struct k_sem *sem, s32_t timeout)
#ifdef CONFIG_USERSPACE
u32_t _handler_k_sem_give(u32_t sem_ptr, u32_t arg2, u32_t arg3,
u32_t arg4, u32_t arg5, u32_t arg6, void *ssf)
{
_SYSCALL_ARG1;
_SYSCALL_IS_OBJ(sem_ptr, K_OBJ_SEM, 0, ssf);
_impl_k_sem_give((struct k_sem *)sem_ptr);
return 0;
}
#endif /* CONFIG_USERSPACE */
int _impl_k_sem_take(struct k_sem *sem, s32_t timeout)
{
__ASSERT(!_is_in_isr() || timeout == K_NO_WAIT, "");
@ -163,3 +189,36 @@ int k_sem_take(struct k_sem *sem, s32_t timeout)
return _Swap(key);
}
#ifdef CONFIG_USERSPACE
u32_t _handler_k_sem_take(u32_t sem_ptr, u32_t timeout, u32_t arg3,
u32_t arg4, u32_t arg5, u32_t arg6, void *ssf)
{
_SYSCALL_ARG2;
_SYSCALL_IS_OBJ(sem_ptr, K_OBJ_SEM, 0, ssf);
_impl_k_sem_take((struct k_sem *)sem_ptr, timeout);
return 0;
}
u32_t _handler_k_sem_reset(u32_t sem_ptr, u32_t arg2, u32_t arg3,
u32_t arg4, u32_t arg5, u32_t arg6, void *ssf)
{
_SYSCALL_ARG1;
_SYSCALL_IS_OBJ(sem_ptr, K_OBJ_SEM, 0, ssf);
_impl_k_sem_reset((struct k_sem *)sem_ptr);
return 0;
}
u32_t _handler_k_sem_count_get(u32_t sem_ptr, u32_t arg2, u32_t arg3,
u32_t arg4, u32_t arg5, u32_t arg6, void *ssf)
{
_SYSCALL_ARG1;
_SYSCALL_IS_OBJ(sem_ptr, K_OBJ_SEM, 0, ssf);
return _impl_k_sem_count_get((struct k_sem *)sem_ptr);
}
#endif /* CONFIG_USERSPACE */

View file

@ -192,4 +192,10 @@ static u32_t _handler_bad_syscall(u32_t bad_id, u32_t arg2, u32_t arg3,
const _k_syscall_handler_t _k_syscall_table[K_SYSCALL_LIMIT] = {
[K_SYSCALL_BAD] = _handler_bad_syscall,
[K_SYSCALL_SEM_INIT] = _handler_k_sem_init,
[K_SYSCALL_SEM_GIVE] = _handler_k_sem_give,
[K_SYSCALL_SEM_TAKE] = _handler_k_sem_take,
[K_SYSCALL_SEM_RESET] = _handler_k_sem_reset,
[K_SYSCALL_SEM_COUNT_GET] = _handler_k_sem_count_get,
};