1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 03:53:04 +01:00
RIOT/pkg/lwip/contrib/sys_arch.c

226 lines
4.8 KiB
C
Raw Normal View History

2015-11-15 20:58:39 +01:00
/*
* Copyright (C) 2015 Martine Lenders <mlenders@inf.fu-berlin.de>
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @{
*
* @file
*/
#include <errno.h>
#include <stdbool.h>
#include <string.h>
#include "arch/cc.h"
#include "arch/sys_arch.h"
#include "lwip/err.h"
#include "lwip/mem.h"
#include "lwip/opt.h"
#include "lwip/sys.h"
#include "msg.h"
#include "sema.h"
#include "thread.h"
#include "xtimer.h"
2016-09-02 14:56:19 +02:00
#define _MSG_SUCCESS (0x5cac)
#define _MSG_TIMEOUT (0x5cad)
2015-11-15 20:58:39 +01:00
void sys_init(void)
{
return;
}
2017-05-10 20:50:28 +02:00
u32_t sys_now(void)
{
return (uint32_t)(xtimer_now_usec64() / US_PER_MS);
}
2015-11-15 20:58:39 +01:00
err_t sys_mutex_new(sys_mutex_t *mutex)
{
mutex_init((mutex_t *)mutex);
return ERR_OK;
}
void sys_mutex_lock(sys_mutex_t *mutex)
{
mutex_lock((mutex_t *)mutex);
}
void sys_mutex_unlock(sys_mutex_t *mutex)
{
mutex_unlock((mutex_t *)mutex);
}
void sys_mutex_free(sys_mutex_t *mutex)
{
mem_free(mutex);
}
err_t sys_sem_new(sys_sem_t *sem, u8_t count)
{
2017-01-18 15:07:54 +01:00
sema_create((sema_t *)sem, (unsigned int)count);
2015-11-15 20:58:39 +01:00
return ERR_OK;
}
void sys_sem_free(sys_sem_t *sem)
{
sema_destroy((sema_t *)sem);
}
void sys_sem_signal(sys_sem_t *sem)
{
LWIP_ASSERT("invalid semaphore", sys_sem_valid(sem));
2015-11-15 20:58:39 +01:00
sema_post((sema_t *)sem);
}
u32_t sys_arch_sem_wait(sys_sem_t *sem, u32_t count)
{
LWIP_ASSERT("invalid semaphore", sys_sem_valid(sem));
2015-11-15 20:58:39 +01:00
if (count != 0) {
uint64_t stop, start;
start = xtimer_now_usec64();
int res = sema_wait_timed((sema_t *)sem, count * US_PER_MS);
stop = xtimer_now_usec64() - start;
2015-11-15 20:58:39 +01:00
if (res == -ETIMEDOUT) {
return SYS_ARCH_TIMEOUT;
}
return (u32_t)(stop / US_PER_MS);
2015-11-15 20:58:39 +01:00
}
else {
2017-03-06 16:55:02 +01:00
sema_wait((sema_t *)sem);
2015-11-15 20:58:39 +01:00
return 0;
}
}
err_t sys_mbox_new(sys_mbox_t *mbox, int size)
{
(void)size;
2016-09-02 14:56:19 +02:00
mbox_init(&mbox->mbox, mbox->msgs, SYS_MBOX_SIZE);
2015-11-15 20:58:39 +01:00
return ERR_OK;
}
void sys_mbox_free(sys_mbox_t *mbox)
{
2016-09-02 14:56:19 +02:00
(void)mbox;
return;
2015-11-15 20:58:39 +01:00
}
void sys_mbox_post(sys_mbox_t *mbox, void *msg)
{
2016-09-02 14:56:19 +02:00
msg_t m = { .content = { .ptr = msg }, .type = _MSG_SUCCESS };
2015-11-15 20:58:39 +01:00
LWIP_ASSERT("invalid mbox", sys_mbox_valid(mbox));
2016-09-02 14:56:19 +02:00
mbox_put(&mbox->mbox, &m);
2015-11-15 20:58:39 +01:00
}
err_t sys_mbox_trypost(sys_mbox_t *mbox, void *msg)
{
2016-09-02 14:56:19 +02:00
msg_t m = { .content = { .ptr = msg }, .type = _MSG_SUCCESS };
2015-11-15 20:58:39 +01:00
2016-09-02 14:56:19 +02:00
if (mbox_try_put(&mbox->mbox, &m)) {
return ERR_OK;
2015-11-15 20:58:39 +01:00
}
2016-09-02 14:56:19 +02:00
else {
return ERR_MEM;
2015-11-15 20:58:39 +01:00
}
2016-09-02 14:56:19 +02:00
}
static void _mbox_timeout(void *arg)
{
msg_t m = { .type = _MSG_TIMEOUT };
mbox_put(arg, &m);
2015-11-15 20:58:39 +01:00
}
u32_t sys_arch_mbox_fetch(sys_mbox_t *mbox, void **msg, u32_t timeout)
{
2016-09-02 14:56:19 +02:00
msg_t m;
xtimer_t timer = { .callback = _mbox_timeout, .arg = &mbox->mbox };
uint64_t start, stop;
2015-11-15 20:58:39 +01:00
start = xtimer_now_usec64();
2016-09-02 14:56:19 +02:00
if (timeout > 0) {
uint64_t u_timeout = (timeout * US_PER_MS);
xtimer_set64(&timer, u_timeout);
2015-11-15 20:58:39 +01:00
}
2016-09-02 14:56:19 +02:00
mbox_get(&mbox->mbox, &m);
stop = xtimer_now_usec64();
xtimer_remove(&timer); /* in case timer did not time out */
2016-09-02 14:56:19 +02:00
switch (m.type) {
case _MSG_SUCCESS:
*msg = m.content.ptr;
return (u32_t)((stop - start) / US_PER_MS);
2016-09-02 14:56:19 +02:00
case _MSG_TIMEOUT:
break;
default: /* should not happen */
LWIP_ASSERT("invalid message received", false);
break;
2015-11-15 20:58:39 +01:00
}
2016-09-02 14:56:19 +02:00
return SYS_ARCH_TIMEOUT;
2015-11-15 20:58:39 +01:00
}
u32_t sys_arch_mbox_tryfetch(sys_mbox_t *mbox, void **msg)
{
2016-09-02 14:56:19 +02:00
msg_t m;
2015-11-15 20:58:39 +01:00
2016-09-02 14:56:19 +02:00
if (mbox_try_get(&mbox->mbox, &m)) {
LWIP_ASSERT("invalid message received", (m.type == _MSG_SUCCESS));
*msg = m.content.ptr;
return ERR_OK;
2015-11-15 20:58:39 +01:00
}
2016-09-02 14:56:19 +02:00
else {
return SYS_MBOX_EMPTY;
2015-11-15 20:58:39 +01:00
}
}
2018-07-17 14:13:01 +02:00
/**
* @brief parameters to _lwip_thread_wrapper
*/
typedef struct {
mutex_t sync;
lwip_thread_fn thread;
void *arg;
} _lwip_thread_params_t;
static void *_lwip_thread_wrapper(void *params_ptr)
{
_lwip_thread_params_t *params = params_ptr;
lwip_thread_fn thread = params->thread;
void *arg = params->arg;
mutex_unlock(&params->sync);
thread(arg);
/* TODO: free stack? */
return NULL;
}
2015-11-15 20:58:39 +01:00
sys_thread_t sys_thread_new(const char *name, lwip_thread_fn thread, void *arg,
int stacksize, int prio)
{
kernel_pid_t res;
char *stack = mem_malloc((size_t)stacksize);
2018-07-17 14:13:01 +02:00
_lwip_thread_params_t params = {
.sync = MUTEX_INIT_LOCKED,
.thread = thread,
.arg = arg,
};
2015-11-15 20:58:39 +01:00
if (stack == NULL) {
return ERR_MEM;
}
if ((res = thread_create(stack, stacksize, prio, THREAD_CREATE_STACKTEST,
2018-07-17 14:13:01 +02:00
_lwip_thread_wrapper, &params,
name)) <= KERNEL_PID_UNDEF) {
2015-11-15 20:58:39 +01:00
abort();
}
2018-07-17 14:13:01 +02:00
mutex_lock(&params.sync);
thread_yield_higher();
2015-11-15 20:58:39 +01:00
return res;
}
/** @} */