1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2009 Sun Microsystems, Inc.
5
* Copyright 2005-2008 Intel Corporation. All Rights Reserved.
7
* This program is free software; you can redistribute it and/or modify
8
* it under the terms of the GNU General Public License as published by
9
* the Free Software Foundation; version 2 of the License.
11
* This program is distributed in the hope that it will be useful,
12
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14
* GNU General Public License for more details.
16
* You should have received a copy of the GNU General Public License
17
* along with this program; if not, write to the Free Software
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
21
#ifndef DRIZZLED_ATOMICS_H
22
#define DRIZZLED_ATOMICS_H
24
# if defined(__SUNPRO_CC)
25
# include <drizzled/atomic/sun_studio.h>
27
# if !defined(__ICC) && !defined(__clang__) && (defined(HAVE_GCC_ATOMIC_BUILTINS) || defined(__SUNPRO_CC))
28
# include <drizzled/atomic/gcc_traits.h>
29
# define ATOMIC_TRAITS internal::gcc_traits
30
# else /* use pthread impl */
31
# define ATOMIC_TRAITS internal::pthread_traits
34
# if (SIZEOF_SIZE_T >= SIZEOF_LONG_LONG) || (!defined(HAVE_GCC_ATOMIC_BUILTINS) || !defined(__SUNPRO_CC)) || defined(__ppc__)
36
# include <drizzled/atomic/pthread_traits.h>
45
template<typename I> // Primary template
48
atomic_base() : my_value(0) {}
49
virtual ~atomic_base() {}
52
template<typename I, typename D, typename T >
53
class atomic_impl: private atomic_base<I>
59
atomic_impl() : atomic_base<I>(), traits() {}
61
value_type add_and_fetch( D addend )
63
return traits.add_and_fetch(&this->my_value, addend);
66
value_type fetch_and_add( D addend )
68
return traits.fetch_and_add(&this->my_value, addend);
71
value_type fetch_and_increment()
73
return traits.fetch_and_increment(&this->my_value);
76
value_type fetch_and_decrement()
78
return traits.fetch_and_decrement(&this->my_value);
81
value_type fetch_and_store( value_type value )
83
return traits.fetch_and_store(&this->my_value, value);
86
bool compare_and_swap( value_type value, value_type comparand )
88
return traits.compare_and_swap(&this->my_value, value, comparand);
91
operator value_type() const volatile
93
return traits.fetch(&this->my_value);
96
value_type& _internal_reference() const
98
return this->my_value;
102
value_type store_with_release( value_type rhs )
104
return traits.store_with_release(&this->my_value, rhs);
108
atomic_impl<I,D,T>& operator+=( D addend )
114
atomic_impl<I,D,T>& operator-=( D addend )
116
// Additive inverse of addend computed using binary minus,
117
// instead of unary minus, for sake of avoiding compiler warnings.
118
return operator+=(D(0)-addend);
121
value_type increment()
123
return add_and_fetch(1);
126
value_type decrement()
128
return add_and_fetch(D(-1));
132
} /* namespace internal */
134
//! Primary template for atomic.
135
/** See the Reference for details.
136
@ingroup synchronization */
142
#define __DRIZZLE_DECL_ATOMIC(T) \
143
template<> struct atomic<T> \
144
: internal::atomic_impl<T,T,ATOMIC_TRAITS<T,T> > { \
145
atomic<T>() : internal::atomic_impl<T,T,ATOMIC_TRAITS<T,T> >() {} \
146
atomic<T>& operator=( T rhs ) { store_with_release(rhs); return *this; } \
151
__DRIZZLE_DECL_ATOMIC(long)
152
__DRIZZLE_DECL_ATOMIC(unsigned long)
153
__DRIZZLE_DECL_ATOMIC(unsigned int)
154
__DRIZZLE_DECL_ATOMIC(int)
155
__DRIZZLE_DECL_ATOMIC(unsigned short)
156
__DRIZZLE_DECL_ATOMIC(short)
157
__DRIZZLE_DECL_ATOMIC(char)
158
__DRIZZLE_DECL_ATOMIC(signed char)
159
__DRIZZLE_DECL_ATOMIC(unsigned char)
160
__DRIZZLE_DECL_ATOMIC(bool)
162
/* 32-bit platforms don't have a GCC atomic operation for 64-bit types,
163
* so we'll use pthread locks to handler 64-bit types on that platforms
166
# if !defined(__ppc__) && (defined(_INT64_TYPE) || defined(_LP64))
167
__DRIZZLE_DECL_ATOMIC(long long)
168
__DRIZZLE_DECL_ATOMIC(unsigned long long)
170
# define __DRIZZLE_DECL_ATOMIC64(T) \
171
template<> struct atomic<T> \
172
: internal::atomic_impl<T,T,internal::pthread_traits<T,T> > { \
174
: internal::atomic_impl<T,T,internal::pthread_traits<T,T> >() {} \
175
T operator=( T rhs ) { return store_with_release(rhs); } \
177
__DRIZZLE_DECL_ATOMIC64(long long)
178
__DRIZZLE_DECL_ATOMIC64(unsigned long long)
184
#endif /* DRIZZLED_ATOMICS_H */