mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
255 lines
5.4 KiB
C
255 lines
5.4 KiB
C
/*
|
|
* 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 "lwip/tcpip.h"
|
|
|
|
#include "irq.h"
|
|
#include "msg.h"
|
|
#include "sema.h"
|
|
#include "thread.h"
|
|
#include "ztimer.h"
|
|
|
|
#define _MSG_SUCCESS (0x5cac)
|
|
#define _MSG_TIMEOUT (0x5cad)
|
|
|
|
void sys_init(void)
|
|
{
|
|
return;
|
|
}
|
|
|
|
u32_t sys_now(void)
|
|
{
|
|
return ztimer_now(ZTIMER_MSEC);
|
|
}
|
|
|
|
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)
|
|
{
|
|
sema_create((sema_t *)sem, (unsigned int)count);
|
|
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));
|
|
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));
|
|
if (count != 0) {
|
|
uint32_t stop, start;
|
|
start = ztimer_now(ZTIMER_MSEC);
|
|
int res = sema_wait_timed_ztimer((sema_t *)sem, ZTIMER_MSEC, count);
|
|
stop = ztimer_now(ZTIMER_MSEC);
|
|
if (res == -ETIMEDOUT) {
|
|
return SYS_ARCH_TIMEOUT;
|
|
}
|
|
return stop - start;
|
|
}
|
|
else {
|
|
sema_wait((sema_t *)sem);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
err_t sys_mbox_new(sys_mbox_t *mbox, int size)
|
|
{
|
|
(void)size;
|
|
mbox_init(&mbox->mbox, mbox->msgs, SYS_MBOX_SIZE);
|
|
return ERR_OK;
|
|
}
|
|
|
|
void sys_mbox_free(sys_mbox_t *mbox)
|
|
{
|
|
(void)mbox;
|
|
return;
|
|
}
|
|
|
|
void sys_mbox_post(sys_mbox_t *mbox, void *msg)
|
|
{
|
|
msg_t m = { .content = { .ptr = msg }, .type = _MSG_SUCCESS };
|
|
|
|
LWIP_ASSERT("invalid mbox", sys_mbox_valid(mbox));
|
|
mbox_put(&mbox->mbox, &m);
|
|
}
|
|
|
|
err_t sys_mbox_trypost(sys_mbox_t *mbox, void *msg)
|
|
{
|
|
msg_t m = { .content = { .ptr = msg }, .type = _MSG_SUCCESS };
|
|
|
|
if (mbox_try_put(&mbox->mbox, &m)) {
|
|
return ERR_OK;
|
|
}
|
|
else {
|
|
return ERR_MEM;
|
|
}
|
|
}
|
|
|
|
static void _mbox_timeout(void *arg)
|
|
{
|
|
msg_t m = { .type = _MSG_TIMEOUT };
|
|
mbox_put(arg, &m);
|
|
}
|
|
|
|
u32_t sys_arch_mbox_fetch(sys_mbox_t *mbox, void **msg, u32_t timeout)
|
|
{
|
|
msg_t m;
|
|
ztimer_t timer = { .callback = _mbox_timeout, .arg = &mbox->mbox };
|
|
uint32_t start, stop;
|
|
|
|
start = ztimer_now(ZTIMER_MSEC);
|
|
if (timeout > 0) {
|
|
ztimer_set(ZTIMER_MSEC, &timer, timeout);
|
|
}
|
|
mbox_get(&mbox->mbox, &m);
|
|
stop = ztimer_now(ZTIMER_MSEC);
|
|
ztimer_remove(ZTIMER_MSEC, &timer); /* in case timer did not time out */
|
|
switch (m.type) {
|
|
case _MSG_SUCCESS:
|
|
*msg = m.content.ptr;
|
|
return stop - start;
|
|
case _MSG_TIMEOUT:
|
|
break;
|
|
default: /* should not happen */
|
|
LWIP_ASSERT("invalid message received", false);
|
|
break;
|
|
}
|
|
return SYS_ARCH_TIMEOUT;
|
|
}
|
|
|
|
u32_t sys_arch_mbox_tryfetch(sys_mbox_t *mbox, void **msg)
|
|
{
|
|
msg_t m;
|
|
|
|
if (mbox_try_get(&mbox->mbox, &m)) {
|
|
LWIP_ASSERT("invalid message received", (m.type == _MSG_SUCCESS));
|
|
*msg = m.content.ptr;
|
|
return ERR_OK;
|
|
}
|
|
else {
|
|
return SYS_MBOX_EMPTY;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @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(¶ms->sync);
|
|
thread(arg);
|
|
/* TODO: free stack? */
|
|
return NULL;
|
|
}
|
|
|
|
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);
|
|
_lwip_thread_params_t params = {
|
|
.sync = MUTEX_INIT_LOCKED,
|
|
.thread = thread,
|
|
.arg = arg,
|
|
};
|
|
|
|
if (stack == NULL) {
|
|
return ERR_MEM;
|
|
}
|
|
if ((res = thread_create(stack, stacksize, prio, THREAD_CREATE_STACKTEST,
|
|
_lwip_thread_wrapper, ¶ms,
|
|
name)) <= KERNEL_PID_UNDEF) {
|
|
abort();
|
|
}
|
|
mutex_lock(¶ms.sync);
|
|
thread_yield_higher();
|
|
return res;
|
|
}
|
|
|
|
static kernel_pid_t lwip_tcpip_thread = KERNEL_PID_UNDEF;
|
|
static kernel_pid_t lwip_lock_thread;
|
|
|
|
void sys_mark_tcpip_thread(void) {
|
|
lwip_tcpip_thread = thread_getpid();
|
|
}
|
|
|
|
void sys_lock_tcpip_core(void) {
|
|
sys_mutex_lock(&lock_tcpip_core);
|
|
lwip_lock_thread = thread_getpid();
|
|
}
|
|
void sys_unlock_tcpip_core(void) {
|
|
lwip_lock_thread = KERNEL_PID_UNDEF;
|
|
sys_mutex_unlock(&lock_tcpip_core);
|
|
}
|
|
|
|
bool sys_check_core_locked(void) {
|
|
/* Don't call from inside isr */
|
|
if (irq_is_in()) {
|
|
return false;
|
|
}
|
|
if (lwip_tcpip_thread != KERNEL_PID_UNDEF) {
|
|
/* only call from thread with lock */
|
|
return lwip_lock_thread == thread_getpid();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/** @} */
|