libevl/lib/flags.c

295 lines
5.7 KiB
C

/*
* SPDX-License-Identifier: MIT
*
* Copyright (C) 2019 Philippe Gerum <rpm@xenomai.org>
*/
#include <stdbool.h>
#include <sys/types.h>
#include <sys/ioctl.h>
#include <time.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <pthread.h>
#include <evl/compiler.h>
#include <evl/atomic.h>
#include <evl/evl.h>
#include <evl/flags.h>
#include <evl/thread.h>
#include <evl/syscall.h>
#include <linux/types.h>
#include "internal.h"
#define __FLAGS_ACTIVE_MAGIC 0xb42bb42b
#define __FLAGS_DEAD_MAGIC 0
int evl_create_flags(struct evl_flags *flg, int clockfd,
int initval, int flags,
const char *fmt, ...)
{
struct evl_monitor_attrs attrs;
struct evl_element_ids eids;
char *name = NULL;
int efd, ret;
va_list ap;
if (evl_shared_memory == NULL)
return -ENXIO;
if (fmt) {
va_start(ap, fmt);
ret = vasprintf(&name, fmt, ap);
va_end(ap);
if (ret < 0)
return -ENOMEM;
}
attrs.type = EVL_MONITOR_EVENT;
attrs.protocol = EVL_EVENT_MASK;
attrs.clockfd = clockfd;
attrs.initval = initval;
efd = create_evl_element(EVL_MONITOR_DEV, name, &attrs, flags, &eids);
if (name)
free(name);
if (efd < 0)
return efd;
flg->u.active.state = evl_shared_memory + eids.state_offset;
/* Force sync the PTE. */
atomic_set(&flg->u.active.state->u.event.value, initval);
flg->u.active.fundle = eids.fundle;
flg->u.active.efd = efd;
flg->magic = __FLAGS_ACTIVE_MAGIC;
return efd;
}
int evl_open_flags(struct evl_flags *flg, const char *fmt, ...)
{
struct evl_monitor_binding bind;
int ret, efd;
va_list ap;
va_start(ap, fmt);
efd = open_evl_element_vargs(EVL_MONITOR_DEV, fmt, ap);
va_end(ap);
if (efd < 0)
return efd;
ret = ioctl(efd, EVL_MONIOC_BIND, &bind);
if (ret) {
ret = -errno;
goto fail;
}
if (bind.type != EVL_MONITOR_EVENT ||
bind.protocol != EVL_EVENT_MASK) {
ret = -EINVAL;
goto fail;
}
flg->u.active.state = evl_shared_memory + bind.eids.state_offset;
__force_read_access(flg->u.active.state->u.event.value);
flg->u.active.fundle = bind.eids.fundle;
flg->u.active.efd = efd;
flg->magic = __FLAGS_ACTIVE_MAGIC;
return efd;
fail:
close(efd);
return ret;
}
int evl_close_flags(struct evl_flags *flg)
{
int ret;
if (flg->magic == __FLAGS_UNINIT_MAGIC)
return 0;
if (flg->magic != __FLAGS_ACTIVE_MAGIC)
return -EINVAL;
ret = close(flg->u.active.efd);
if (ret)
return -errno;
flg->u.active.fundle = EVL_NO_HANDLE;
flg->u.active.state = NULL;
flg->magic = __FLAGS_DEAD_MAGIC;
return 0;
}
static int check_sanity(struct evl_flags *flg)
{
int efd;
if (flg->magic == __FLAGS_UNINIT_MAGIC) {
efd = evl_create_flags(flg,
flg->u.uninit.clockfd,
flg->u.uninit.initval,
flg->u.uninit.flags,
flg->u.uninit.name);
return efd < 0 ? efd : 0;
}
return flg->magic != __FLAGS_ACTIVE_MAGIC ? -EINVAL : 0;
}
static int try_wait(struct evl_monitor_state *state)
{
int val, prev;
val = atomic_read(&state->u.event.value);
if (!val)
return 0;
do {
prev = val;
val = atomic_cmpxchg(&state->u.event.value, prev, 0);
if (!val)
return 0;
} while (val != prev);
return val;
}
static inline bool is_polled(struct evl_monitor_state *state)
{
return !!atomic_read(&state->u.event.pollrefs);
}
int evl_timedwait_flags(struct evl_flags *flg,
const struct timespec *timeout, int *r_bits)
{
struct evl_monitor_state *state;
struct evl_monitor_waitreq req;
struct __evl_timespec kts;
fundle_t current;
int ret;
current = evl_get_current();
if (current == EVL_NO_HANDLE)
return -EPERM;
ret = check_sanity(flg);
if (ret)
return ret;
state = flg->u.active.state;
if (!is_polled(state)) {
ret = try_wait(state);
if (ret) {
*r_bits = ret;
return 0;
}
}
req.gatefd = -1;
req.timeout_ptr = __evl_ktimespec_ptr64(timeout, kts);
req.status = -EINVAL;
req.value = 0;
ret = oob_ioctl(flg->u.active.efd, EVL_MONIOC_WAIT, &req);
if (ret)
return -errno;
if (req.status)
return req.status;
*r_bits = req.value;
return 0;
}
int evl_wait_flags(struct evl_flags *flg, int *r_bits)
{
struct timespec timeout = { .tv_sec = 0, .tv_nsec = 0 };
return evl_timedwait_flags(flg, &timeout, r_bits);
}
int evl_trywait_flags(struct evl_flags *flg, int *r_bits)
{
int ret;
ret = check_sanity(flg);
if (ret)
return ret;
ret = try_wait(flg->u.active.state);
if (!ret)
return -EAGAIN;
*r_bits = ret;
return 0;
}
int evl_post_flags(struct evl_flags *flg, int bits)
{
struct evl_monitor_state *state;
int val, prev, next, ret;
__s32 mask = bits;
ret = check_sanity(flg);
if (ret)
return ret;
if (!bits)
return -EINVAL;
/*
* Unlike with gated event, we have no raceless way to detect
* that somebody is waiting on the flag group from userland,
* so we do a kernel entry each time a zero->non-zero
* transition is observed for the value. Fortunately, having
* some thread(s) already waiting for a flag to be posted is
* the most likely situation, so such entry will be required
* in most cases anyway.
*/
state = flg->u.active.state;
val = atomic_read(&state->u.event.value);
if (!val || is_polled(state)) {
slow_path:
if (evl_get_current())
ret = oob_ioctl(flg->u.active.efd,
EVL_MONIOC_SIGNAL, &mask);
else
ret = ioctl(flg->u.active.efd,
EVL_MONIOC_SIGNAL, &mask);
return ret ? -errno : 0;
}
do {
prev = val;
next = prev | bits;
val = atomic_cmpxchg(&state->u.event.value, prev, next);
if (!val)
goto slow_path;
if (is_polled(state)) {
/* If swap happened, just trigger a wakeup. */
if (val == prev)
mask = 0;
goto slow_path;
}
} while (val != prev);
return 0;
}
int evl_peek_flags(struct evl_flags *flg, int *r_bits)
{
if (flg->magic != __FLAGS_ACTIVE_MAGIC)
return -EINVAL;
*r_bits = atomic_read(&flg->u.active.state->u.event.value);
return 0;
}