Files
checker_m4/source/OpenAMP/libmetal/lib/system/linux/condition.h
2023-06-21 18:00:56 +08:00

75 lines
1.8 KiB
C

/*
* Copyright (c) 2016, Xilinx Inc. and Contributors. All rights reserved.
*
* SPDX-License-Identifier: BSD-3-Clause
*/
/*
* @file linux/condition.h
* @brief Linux condition variable primitives for libmetal.
*/
#ifndef __METAL_CONDITION__H__
#error "Include metal/condition.h instead of metal/linux/condition.h"
#endif
#ifndef __METAL_LINUX_CONDITION__H__
#define __METAL_LINUX_CONDITION__H__
#include <unistd.h>
#include <sys/syscall.h>
#include <linux/futex.h>
#include <metal/atomic.h>
#include <stdint.h>
#include <limits.h>
#include <metal/errno.h>
#ifdef __cplusplus
extern "C" {
#endif
struct metal_condition {
metal_mutex_t *m; /**< mutex.
The condition variable is attached to
this mutex when it is waiting.
It is also used to check correctness
in case there are multiple waiters. */
atomic_int waiters; /**< number of waiters. */
atomic_int wakeups; /**< number of wakeups. */
};
/** Static metal condition variable initialization. */
#define METAL_CONDITION_INIT { NULL, ATOMIC_VAR_INIT(0), ATOMIC_VAR_INIT(0) }
static inline void metal_condition_init(struct metal_condition *cv)
{
cv->m = NULL;
atomic_init(&cv->waiters, 0);
atomic_init(&cv->wakeups, 0);
}
static inline int metal_condition_signal(struct metal_condition *cv)
{
if (!cv)
return -EINVAL;
atomic_fetch_add(&cv->wakeups, 1);
if (atomic_load(&cv->waiters) > 0)
syscall(SYS_futex, &cv->wakeups, FUTEX_WAKE, 1, NULL, NULL, 0);
return 0;
}
static inline int metal_condition_broadcast(struct metal_condition *cv)
{
if (!cv)
return -EINVAL;
atomic_fetch_add(&cv->wakeups, 1);
if (atomic_load(&cv->waiters) > 0)
syscall(SYS_futex, &cv->wakeups, FUTEX_WAKE, INT_MAX, NULL, NULL, 0);
return 0;
}
#endif /* __METAL_LINUX_CONDITION__H__ */