xrootd
Loading...
Searching...
No Matches
XrdSysAtomics.hh
Go to the documentation of this file.
1#ifndef _XRDSYSATOMICS_
2#define _XRDSYSATOMICS_
3/******************************************************************************/
4/* */
5/* X r d S y s A t o m i c s . h h */
6/* */
7/* (c) 2011 by the Board of Trustees of the Leland Stanford, Jr., University */
8/* All Rights Reserved */
9/* Produced by Andrew Hanushevsky for Stanford University under contract */
10/* DE-AC02-76-SFO0515 with the Department of Energy */
11/* */
12/* This file is part of the XRootD software suite. */
13/* */
14/* XRootD is free software: you can redistribute it and/or modify it under */
15/* the terms of the GNU Lesser General Public License as published by the */
16/* Free Software Foundation, either version 3 of the License, or (at your */
17/* option) any later version. */
18/* */
19/* XRootD is distributed in the hope that it will be useful, but WITHOUT */
20/* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or */
21/* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public */
22/* License for more details. */
23/* */
24/* You should have received a copy of the GNU Lesser General Public License */
25/* along with XRootD in a file called COPYING.LESSER (LGPL license) and file */
26/* COPYING (GPL license). If not, see <http://www.gnu.org/licenses/>. */
27/* */
28/* The copyright holder's institutional names and contributor's names may not */
29/* be used to endorse or promote products derived from this software without */
30/* specific prior written permission of the institution or contributor. */
31/******************************************************************************/
32
33/* The following instruction acronyms are used:
34 AtomicCAS() -> Compare And [if equal] Set
35 AtomicFAZ() -> Fetch And Zero
36*/
37
38/* Portability note: When using fetch-add or fetch-sub in the context of an
39 assignment statement, you must use the the AtomicFAdd
40 and AtomicFSub macros to ensure portability and correct
41 results. Additionally, whenever you use AtomicFAZ as the
42 only consequent of an if-else statement you *must* places
43 braces around it otherwise the code will not be portable!
44 Alternatively, always use the AtomicFZAP macro.
45*/
46
47#ifdef HAVE_ATOMICS
48#define AtomicBeg(Mtx)
49#define AtomicEnd(Mtx)
50#define AtomicAdd(x, y) __sync_fetch_and_add(&x, y)
51#define AtomicFAdd(w,x,y) w = __sync_fetch_and_add(&x, y)
52#define AtomicCAS(x, y, z) __sync_bool_compare_and_swap(&x, y, z)
53#define AtomicDec(x) __sync_fetch_and_sub(&x, 1)
54#define AtomicFAZ(x) __sync_fetch_and_and(&x, 0)
55#define AtomicFZAP(w,x) w = __sync_fetch_and_and(&x, 0)
56#define AtomicGet(x) __sync_fetch_and_or(&x, 0)
57#define AtomicInc(x) __sync_fetch_and_add(&x, 1)
58#define AtomicSub(x, y) __sync_fetch_and_sub(&x, y)
59#define AtomicFSub(w,x,y) w = __sync_fetch_and_sub(&x, y)
60#define AtomicZAP(x) __sync_fetch_and_and(&x, 0)
61#define AtomicRet(mtx, x) return AtomicGet(x)
62#else
63#define AtomicBeg(Mtx) Mtx.Lock()
64#define AtomicEnd(Mtx) Mtx.UnLock()
65#define AtomicAdd(x, y) x += y // When assigning use AtomicFAdd!
66#define AtomicFAdd(w,x,y) {w = x; x += y;}
67#define AtomicCAS(x, y, z) if (x == y) x = z
68#define AtomicDec(x) x--
69#define AtomicFAZ(x) x; x = 0 // Braces when used with if-else!
70#define AtomicFZAP(w,x) {w = x; x = 0;}
71#define AtomicGet(x) x
72#define AtomicInc(x) x++
73#define AtomicSub(x, y) x -= y // When assigning use AtomicFSub!
74#define AtomicFSub(w,x,y) {w = x; x -= y;}
75#define AtomicZAP(x) x = 0
76#define AtomicRet(mtx, x) {mtx.Lock(); int _ ## x = x; \
77 mtx.UnLock(); return _ ## x;}
78#endif
79#endif
80
81/*
82 * The following definitions give a mechanism for using C++ atomics
83 * (when using at least C++11). *Note* that these can't be relied
84 * on for correct behavior as they are non-atomic for C++03 compilers.
85 *
86 * Only use them for standards correctness (eliminating C++11 undefined
87 * behavior).
88 */
89#if __cplusplus >= 201103L
90#include <atomic>
91#define CPP_ATOMIC_LOAD(x, order) x.load(order)
92#define CPP_ATOMIC_STORE(x, val, order) x.store(val, order)
93#define CPP_ATOMIC_TYPE(kind) std::atomic<kind>
94#else
95#define CPP_ATOMIC_LOAD(x, order) x
96#define CPP_ATOMIC_STORE(x, val, order) x = val
97#define CPP_ATOMIC_TYPE(kind) kind
98#endif
99
100