numam-dpdk/lib/eal/windows/eal_alarm.c
William Tu f1f6ebc0ea eal: remove sys/queue.h from public headers
Currently there are some public headers that include 'sys/queue.h', which
is not POSIX, but usually provided by the Linux/BSD system library.
(Not in POSIX.1, POSIX.1-2001, or POSIX.1-2008. Present on the BSDs.)
The file is missing on Windows. During the Windows build, DPDK uses a
bundled copy, so building a DPDK library works fine.  But when OVS or other
applications use DPDK as a library, because some DPDK public headers
include 'sys/queue.h', on Windows, it triggers an error due to no such
file.

One solution is to install the 'lib/eal/windows/include/sys/queue.h' into
Windows environment, such as [1]. However, this means DPDK exports the
functionalities of 'sys/queue.h' into the environment, which might cause
symbols, macros, headers clashing with other applications.

The patch fixes it by removing the "#include <sys/queue.h>" from
DPDK public headers, so programs including DPDK headers don't depend
on the system to provide 'sys/queue.h'. When these public headers use
macros such as TAILQ_xxx, we replace it by the ones with RTE_ prefix.
For Windows, we copy the definitions from <sys/queue.h> to rte_os.h
in Windows EAL. Note that these RTE_ macros are compatible with
<sys/queue.h>, both at the level of API (to use with <sys/queue.h>
macros in C files) and ABI (to avoid breaking it).

Additionally, the TAILQ_FOREACH_SAFE is not part of <sys/queue.h>,
the patch replaces it with RTE_TAILQ_FOREACH_SAFE.

[1] http://mails.dpdk.org/archives/dev/2021-August/216304.html

Suggested-by: Nick Connolly <nick.connolly@mayadata.io>
Suggested-by: Dmitry Kozlyuk <dmitry.kozliuk@gmail.com>
Signed-off-by: William Tu <u9012063@gmail.com>
Acked-by: Dmitry Kozlyuk <dmitry.kozliuk@gmail.com>
Acked-by: Narcisa Vasile <navasile@linux.microsoft.com>
2021-10-01 13:09:43 +02:00

257 lines
5.1 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright (c) 2020 Dmitry Kozlyuk
*/
#include <stdatomic.h>
#include <stdbool.h>
#include <sys/queue.h>
#include <rte_alarm.h>
#include <rte_spinlock.h>
#include <rte_eal_trace.h>
#include "eal_windows.h"
enum alarm_state {
ALARM_ARMED,
ALARM_TRIGGERED,
ALARM_CANCELLED
};
struct alarm_entry {
LIST_ENTRY(alarm_entry) next;
rte_eal_alarm_callback cb_fn;
void *cb_arg;
HANDLE timer;
atomic_uint state;
};
static LIST_HEAD(alarm_list, alarm_entry) alarm_list = LIST_HEAD_INITIALIZER();
static rte_spinlock_t alarm_lock = RTE_SPINLOCK_INITIALIZER;
static int intr_thread_exec_sync(void (*func)(void *arg), void *arg);
static void
alarm_remove_unsafe(struct alarm_entry *ap)
{
LIST_REMOVE(ap, next);
CloseHandle(ap->timer);
free(ap);
}
static void
alarm_callback(void *arg, DWORD low __rte_unused, DWORD high __rte_unused)
{
struct alarm_entry *ap = arg;
unsigned int state = ALARM_ARMED;
if (!atomic_compare_exchange_strong(
&ap->state, &state, ALARM_TRIGGERED))
return;
ap->cb_fn(ap->cb_arg);
rte_spinlock_lock(&alarm_lock);
alarm_remove_unsafe(ap);
rte_spinlock_unlock(&alarm_lock);
}
static int
alarm_set(struct alarm_entry *entry, LARGE_INTEGER deadline)
{
BOOL ret = SetWaitableTimer(
entry->timer, &deadline, 0, alarm_callback, entry, FALSE);
if (!ret) {
RTE_LOG_WIN32_ERR("SetWaitableTimer");
return -1;
}
return 0;
}
struct alarm_task {
struct alarm_entry *entry;
LARGE_INTEGER deadline;
int ret;
};
static void
alarm_task_exec(void *arg)
{
struct alarm_task *task = arg;
task->ret = alarm_set(task->entry, task->deadline);
}
int
rte_eal_alarm_set(uint64_t us, rte_eal_alarm_callback cb_fn, void *cb_arg)
{
struct alarm_entry *ap;
HANDLE timer;
FILETIME ft;
LARGE_INTEGER deadline;
int ret;
if (cb_fn == NULL) {
RTE_LOG(ERR, EAL, "NULL callback\n");
ret = -EINVAL;
goto exit;
}
/* Calculate deadline ASAP, unit of measure = 100ns. */
GetSystemTimePreciseAsFileTime(&ft);
deadline.LowPart = ft.dwLowDateTime;
deadline.HighPart = ft.dwHighDateTime;
deadline.QuadPart += 10 * us;
ap = calloc(1, sizeof(*ap));
if (ap == NULL) {
RTE_LOG(ERR, EAL, "Cannot allocate alarm entry\n");
ret = -ENOMEM;
goto exit;
}
timer = CreateWaitableTimer(NULL, FALSE, NULL);
if (timer == NULL) {
RTE_LOG_WIN32_ERR("CreateWaitableTimer()");
ret = -EINVAL;
goto fail;
}
ap->timer = timer;
ap->cb_fn = cb_fn;
ap->cb_arg = cb_arg;
/* Waitable timer must be set in the same thread that will
* do an alertable wait for the alarm to trigger, that is,
* in the interrupt thread.
*/
if (rte_thread_is_intr()) {
/* Directly schedule callback execution. */
ret = alarm_set(ap, deadline);
if (ret < 0) {
RTE_LOG(ERR, EAL, "Cannot setup alarm\n");
goto fail;
}
} else {
/* Dispatch a task to set alarm into the interrupt thread.
* Execute it synchronously, because it can fail.
*/
struct alarm_task task = {
.entry = ap,
.deadline = deadline,
};
ret = intr_thread_exec_sync(alarm_task_exec, &task);
if (ret < 0) {
RTE_LOG(ERR, EAL, "Cannot setup alarm in interrupt thread\n");
goto fail;
}
ret = task.ret;
if (ret < 0)
goto fail;
}
rte_spinlock_lock(&alarm_lock);
LIST_INSERT_HEAD(&alarm_list, ap, next);
rte_spinlock_unlock(&alarm_lock);
goto exit;
fail:
if (timer != NULL)
CloseHandle(timer);
if (ap != NULL)
free(ap);
exit:
rte_eal_trace_alarm_set(us, cb_fn, cb_arg, ret);
return ret;
}
static bool
alarm_matches(const struct alarm_entry *ap,
rte_eal_alarm_callback cb_fn, void *cb_arg)
{
bool any_arg = cb_arg == (void *)(-1);
return (ap->cb_fn == cb_fn) && (any_arg || ap->cb_arg == cb_arg);
}
int
rte_eal_alarm_cancel(rte_eal_alarm_callback cb_fn, void *cb_arg)
{
struct alarm_entry *ap;
unsigned int state;
int removed;
bool executing;
removed = 0;
if (cb_fn == NULL) {
RTE_LOG(ERR, EAL, "NULL callback\n");
return -EINVAL;
}
do {
executing = false;
rte_spinlock_lock(&alarm_lock);
LIST_FOREACH(ap, &alarm_list, next) {
if (!alarm_matches(ap, cb_fn, cb_arg))
continue;
state = ALARM_ARMED;
if (atomic_compare_exchange_strong(
&ap->state, &state, ALARM_CANCELLED)) {
alarm_remove_unsafe(ap);
removed++;
} else if (state == ALARM_TRIGGERED)
executing = true;
}
rte_spinlock_unlock(&alarm_lock);
} while (executing);
rte_eal_trace_alarm_cancel(cb_fn, cb_arg, removed);
return removed;
}
struct intr_task {
void (*func)(void *arg);
void *arg;
rte_spinlock_t lock; /* unlocked at task completion */
};
static void
intr_thread_entry(void *arg)
{
struct intr_task *task = arg;
task->func(task->arg);
rte_spinlock_unlock(&task->lock);
}
static int
intr_thread_exec_sync(void (*func)(void *arg), void *arg)
{
struct intr_task task;
int ret;
task.func = func;
task.arg = arg;
rte_spinlock_init(&task.lock);
/* Make timers more precise by synchronizing in userspace. */
rte_spinlock_lock(&task.lock);
ret = eal_intr_thread_schedule(intr_thread_entry, &task);
if (ret < 0) {
RTE_LOG(ERR, EAL, "Cannot schedule task to interrupt thread\n");
return -EINVAL;
}
/* Wait for the task to complete. */
rte_spinlock_lock(&task.lock);
return 0;
}