mirror of
https://github.com/mpv-player/mpv
synced 2025-01-06 15:10:20 +00:00
0025f0042f
As usual, we use C11 semantics, and emulate it if <stdatomic.h> is not available. It's a bit messy with __sync_val_compare_and_swap(). We assume it has "strong" semantics (it can't fail sporadically), but I'm not sure if this is really the case. On the other hand, weak semantics don't seem to be possible, since the builtin can't distinguish between the two failure cases that could occur. Also, to match the C11 interface, use of gcc builtins is unavoidable. Add a check to the build system to make sure the compiler supports them (although I don't think there's any compiler which supports __sync_*, but not these extensions). Needed for the following commit.
89 lines
2.8 KiB
C
89 lines
2.8 KiB
C
/*
|
|
* This file is part of mpv.
|
|
* Copyright (c) 2013 Stefano Pigozzi <stefano.pigozzi@gmail.com>
|
|
*
|
|
* mpv is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* mpv is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with mpv. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifndef MP_ATOMICS_H
|
|
#define MP_ATOMICS_H
|
|
|
|
#include <inttypes.h>
|
|
#include "config.h"
|
|
|
|
#define HAVE_ATOMICS 1
|
|
|
|
#if HAVE_STDATOMIC
|
|
#include <stdatomic.h>
|
|
#else
|
|
|
|
// Emulate the parts of C11 stdatomic.h needed by mpv.
|
|
// Still relies on gcc/clang atomic builtins.
|
|
|
|
typedef struct { volatile unsigned long v; } atomic_ulong;
|
|
typedef struct { volatile int v; } atomic_int;
|
|
typedef struct { volatile _Bool v; } atomic_bool;
|
|
typedef struct { volatile long long v; } atomic_llong;
|
|
typedef struct { volatile uint_least32_t v; } atomic_uint_least32_t;
|
|
typedef struct { volatile unsigned long long v; } atomic_ullong;
|
|
|
|
#define ATOMIC_VAR_INIT(x) \
|
|
{.v = (x)}
|
|
|
|
#if HAVE_ATOMIC_BUILTINS
|
|
|
|
#define atomic_load(p) \
|
|
__atomic_load_n(&(p)->v, __ATOMIC_SEQ_CST)
|
|
#define atomic_store(p, val) \
|
|
__atomic_store_n(&(p)->v, val, __ATOMIC_SEQ_CST)
|
|
#define atomic_fetch_add(a, b) \
|
|
__atomic_fetch_add(&(a)->v, b, __ATOMIC_SEQ_CST)
|
|
#define atomic_compare_exchange_strong(a, b, c) \
|
|
__atomic_compare_exchange_n(&(a)->v, b, c, 0, __ATOMIC_SEQ_CST, \
|
|
__ATOMIC_SEQ_CST)
|
|
|
|
#elif HAVE_SYNC_BUILTINS
|
|
|
|
#define atomic_load(p) \
|
|
__sync_fetch_and_add(&(p)->v, 0)
|
|
#define atomic_store(p, val) \
|
|
(__sync_synchronize(), (p)->v = (val), __sync_synchronize())
|
|
#define atomic_fetch_add(a, b) \
|
|
__sync_fetch_and_add(&(a)->v, b)
|
|
// Assumes __sync_val_compare_and_swap is "strong" (using the C11 meaning).
|
|
#define atomic_compare_exchange_strong(p, old, new) \
|
|
({ __typeof__((p)->v) val_ = __sync_val_compare_and_swap(&(p)->v, *(old), new); \
|
|
bool ok_ = val_ == *(old); \
|
|
if (!ok_) *(old) = val_; \
|
|
ok_; })
|
|
|
|
#else
|
|
|
|
// This is extremely wrong. The build system actually disables code that has
|
|
// a serious dependency on working atomics, so this is barely ok.
|
|
#define atomic_load(p) ((p)->v)
|
|
#define atomic_store(p, val) ((p)->v = (val))
|
|
#define atomic_fetch_add(a, b) (((a)->v += (b)) - (b))
|
|
#define atomic_compare_exchange_strong(p, old, new) \
|
|
((p)->v == *(old) ? ((p)->v = (new), 1) : (*(old) = (p)->v, 0))
|
|
|
|
#undef HAVE_ATOMICS
|
|
#define HAVE_ATOMICS 0
|
|
|
|
#endif /* no atomics */
|
|
|
|
#endif /* else HAVE_STDATOMIC */
|
|
|
|
#endif
|