Elliott Hughes | 2faa5f1 | 2012-01-30 14:42:07 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2010 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
Carl Shapiro | 6c21dc1 | 2011-06-20 15:20:52 -0700 | [diff] [blame] | 16 | |
Brian Carlstrom | fc0e321 | 2013-07-17 14:40:12 -0700 | [diff] [blame] | 17 | #ifndef ART_RUNTIME_BASE_MACROS_H_ |
| 18 | #define ART_RUNTIME_BASE_MACROS_H_ |
Carl Shapiro | 6c21dc1 | 2011-06-20 15:20:52 -0700 | [diff] [blame] | 19 | |
Carl Shapiro | 12eb78e | 2011-06-24 14:51:06 -0700 | [diff] [blame] | 20 | #include <stddef.h> // for size_t |
| 21 | |
Brian Carlstrom | 7a00a3c | 2012-01-25 18:38:03 -0800 | [diff] [blame] | 22 | #define GCC_VERSION (__GNUC__ * 10000 + __GNUC_MINOR__ * 100 + __GNUC_PATCHLEVEL__) |
Shih-wei Liao | 24782c6 | 2012-01-08 12:46:11 -0800 | [diff] [blame] | 23 | |
Carl Shapiro | a5d5cfd | 2011-06-21 12:46:59 -0700 | [diff] [blame] | 24 | // The COMPILE_ASSERT macro can be used to verify that a compile time |
| 25 | // expression is true. For example, you could use it to verify the |
| 26 | // size of a static array: |
| 27 | // |
| 28 | // COMPILE_ASSERT(ARRAYSIZE(content_type_names) == CONTENT_NUM_TYPES, |
| 29 | // content_type_names_incorrect_size); |
| 30 | // |
| 31 | // or to make sure a struct is smaller than a certain size: |
| 32 | // |
| 33 | // COMPILE_ASSERT(sizeof(foo) < 128, foo_too_large); |
| 34 | // |
| 35 | // The second argument to the macro is the name of the variable. If |
| 36 | // the expression is false, most compilers will issue a warning/error |
| 37 | // containing the name of the variable. |
| 38 | |
| 39 | template <bool> |
| 40 | struct CompileAssert { |
| 41 | }; |
| 42 | |
| 43 | #define COMPILE_ASSERT(expr, msg) \ |
Elliott Hughes | 398f64b | 2012-03-26 18:05:48 -0700 | [diff] [blame] | 44 | typedef CompileAssert<(bool(expr))> msg[bool(expr) ? 1 : -1] // NOLINT |
Carl Shapiro | a5d5cfd | 2011-06-21 12:46:59 -0700 | [diff] [blame] | 45 | |
Carl Shapiro | 6c21dc1 | 2011-06-20 15:20:52 -0700 | [diff] [blame] | 46 | // DISALLOW_COPY_AND_ASSIGN disallows the copy and operator= functions. |
| 47 | // It goes in the private: declarations in a class. |
| 48 | #define DISALLOW_COPY_AND_ASSIGN(TypeName) \ |
| 49 | TypeName(const TypeName&); \ |
| 50 | void operator=(const TypeName&) |
| 51 | |
| 52 | // A macro to disallow all the implicit constructors, namely the |
| 53 | // default constructor, copy constructor and operator= functions. |
| 54 | // |
| 55 | // This should be used in the private: declarations for a class |
| 56 | // that wants to prevent anyone from instantiating it. This is |
| 57 | // especially useful for classes containing only static methods. |
| 58 | #define DISALLOW_IMPLICIT_CONSTRUCTORS(TypeName) \ |
| 59 | TypeName(); \ |
| 60 | DISALLOW_COPY_AND_ASSIGN(TypeName) |
| 61 | |
Carl Shapiro | a2e18e1 | 2011-06-21 18:57:55 -0700 | [diff] [blame] | 62 | // The arraysize(arr) macro returns the # of elements in an array arr. |
| 63 | // The expression is a compile-time constant, and therefore can be |
| 64 | // used in defining new arrays, for example. If you use arraysize on |
| 65 | // a pointer by mistake, you will get a compile-time error. |
| 66 | // |
| 67 | // One caveat is that arraysize() doesn't accept any array of an |
| 68 | // anonymous type or a type defined inside a function. In these rare |
Carl Shapiro | d2bdb57 | 2011-06-22 11:45:37 -0700 | [diff] [blame] | 69 | // cases, you have to use the unsafe ARRAYSIZE_UNSAFE() macro below. This is |
Carl Shapiro | a2e18e1 | 2011-06-21 18:57:55 -0700 | [diff] [blame] | 70 | // due to a limitation in C++'s template system. The limitation might |
| 71 | // eventually be removed, but it hasn't happened yet. |
| 72 | |
| 73 | // This template function declaration is used in defining arraysize. |
| 74 | // Note that the function doesn't need an implementation, as we only |
| 75 | // use its type. |
| 76 | template <typename T, size_t N> |
| 77 | char (&ArraySizeHelper(T (&array)[N]))[N]; |
| 78 | |
| 79 | #define arraysize(array) (sizeof(ArraySizeHelper(array))) |
| 80 | |
Carl Shapiro | d2bdb57 | 2011-06-22 11:45:37 -0700 | [diff] [blame] | 81 | // ARRAYSIZE_UNSAFE performs essentially the same calculation as arraysize, |
| 82 | // but can be used on anonymous types or types defined inside |
| 83 | // functions. It's less safe than arraysize as it accepts some |
| 84 | // (although not all) pointers. Therefore, you should use arraysize |
| 85 | // whenever possible. |
| 86 | // |
| 87 | // The expression ARRAYSIZE_UNSAFE(a) is a compile-time constant of type |
| 88 | // size_t. |
| 89 | // |
| 90 | // ARRAYSIZE_UNSAFE catches a few type errors. If you see a compiler error |
| 91 | // |
| 92 | // "warning: division by zero in ..." |
| 93 | // |
| 94 | // when using ARRAYSIZE_UNSAFE, you are (wrongfully) giving it a pointer. |
| 95 | // You should only use ARRAYSIZE_UNSAFE on statically allocated arrays. |
| 96 | // |
| 97 | // The following comments are on the implementation details, and can |
| 98 | // be ignored by the users. |
| 99 | // |
| 100 | // ARRAYSIZE_UNSAFE(arr) works by inspecting sizeof(arr) (the # of bytes in |
| 101 | // the array) and sizeof(*(arr)) (the # of bytes in one array |
| 102 | // element). If the former is divisible by the latter, perhaps arr is |
| 103 | // indeed an array, in which case the division result is the # of |
| 104 | // elements in the array. Otherwise, arr cannot possibly be an array, |
| 105 | // and we generate a compiler error to prevent the code from |
| 106 | // compiling. |
| 107 | // |
| 108 | // Since the size of bool is implementation-defined, we need to cast |
| 109 | // !(sizeof(a) & sizeof(*(a))) to size_t in order to ensure the final |
| 110 | // result has type size_t. |
| 111 | // |
| 112 | // This macro is not perfect as it wrongfully accepts certain |
| 113 | // pointers, namely where the pointer size is divisible by the pointee |
| 114 | // size. Since all our code has to go through a 32-bit compiler, |
| 115 | // where a pointer is 4 bytes, this means all pointers to a type whose |
| 116 | // size is 3 or greater than 4 will be (righteously) rejected. |
| 117 | #define ARRAYSIZE_UNSAFE(a) \ |
Elliott Hughes | 362f9bc | 2011-10-17 18:56:41 -0700 | [diff] [blame] | 118 | ((sizeof(a) / sizeof(*(a))) / static_cast<size_t>(!(sizeof(a) % sizeof(*(a))))) |
Carl Shapiro | d2bdb57 | 2011-06-22 11:45:37 -0700 | [diff] [blame] | 119 | |
Elliott Hughes | ff17f1f | 2012-01-24 18:12:29 -0800 | [diff] [blame] | 120 | #define SIZEOF_MEMBER(t, f) sizeof((reinterpret_cast<t*>(4096))->f) |
Carl Shapiro | 59e85cd | 2011-06-21 10:16:23 -0700 | [diff] [blame] | 121 | |
Elliott Hughes | 362f9bc | 2011-10-17 18:56:41 -0700 | [diff] [blame] | 122 | #define OFFSETOF_MEMBER(t, f) \ |
Ian Rogers | 8a01a3a | 2013-05-06 13:25:44 -0700 | [diff] [blame] | 123 | (reinterpret_cast<const char*>(&reinterpret_cast<t*>(16)->f) - reinterpret_cast<const char*>(16)) // NOLINT |
Carl Shapiro | 59e85cd | 2011-06-21 10:16:23 -0700 | [diff] [blame] | 124 | |
Elliott Hughes | 93e74e8 | 2011-09-13 11:07:03 -0700 | [diff] [blame] | 125 | #define OFFSETOF_VOLATILE_MEMBER(t, f) \ |
Elliott Hughes | 398f64b | 2012-03-26 18:05:48 -0700 | [diff] [blame] | 126 | (reinterpret_cast<volatile char*>(&reinterpret_cast<t*>(16)->f) - reinterpret_cast<volatile char*>(16)) // NOLINT |
Elliott Hughes | 93e74e8 | 2011-09-13 11:07:03 -0700 | [diff] [blame] | 127 | |
Brian Carlstrom | b1eba21 | 2013-07-17 18:07:19 -0700 | [diff] [blame] | 128 | #define PACKED(x) __attribute__ ((__aligned__(x), __packed__)) |
Elliott Hughes | 85d1545 | 2011-09-16 17:33:01 -0700 | [diff] [blame] | 129 | |
Elliott Hughes | ff17f1f | 2012-01-24 18:12:29 -0800 | [diff] [blame] | 130 | #define LIKELY(x) __builtin_expect((x), true) |
| 131 | #define UNLIKELY(x) __builtin_expect((x), false) |
Ian Rogers | caab8c4 | 2011-10-12 12:11:18 -0700 | [diff] [blame] | 132 | |
Ian Rogers | e8ae0dc | 2013-02-07 10:20:45 -0800 | [diff] [blame] | 133 | #ifndef NDEBUG |
Ian Rogers | 1ffa32f | 2013-02-05 18:29:08 -0800 | [diff] [blame] | 134 | #define ALWAYS_INLINE |
| 135 | #else |
Ian Rogers | e8ae0dc | 2013-02-07 10:20:45 -0800 | [diff] [blame] | 136 | #define ALWAYS_INLINE __attribute__ ((always_inline)) |
Ian Rogers | 1ffa32f | 2013-02-05 18:29:08 -0800 | [diff] [blame] | 137 | #endif |
| 138 | |
Anwar Ghuloum | 63937db | 2013-05-24 09:08:32 -0700 | [diff] [blame] | 139 | #if defined (__APPLE__) |
Anwar Ghuloum | 1d9314c | 2013-05-24 10:44:48 -0700 | [diff] [blame] | 140 | #define HOT_ATTR |
Ian Rogers | 8d31bbd | 2013-10-13 10:44:14 -0700 | [diff] [blame] | 141 | #define COLD_ATTR |
Anwar Ghuloum | 63937db | 2013-05-24 09:08:32 -0700 | [diff] [blame] | 142 | #else |
Anwar Ghuloum | 1d9314c | 2013-05-24 10:44:48 -0700 | [diff] [blame] | 143 | #define HOT_ATTR __attribute__ ((hot)) |
Ian Rogers | 8d31bbd | 2013-10-13 10:44:14 -0700 | [diff] [blame] | 144 | #define COLD_ATTR __attribute__ ((cold)) |
Anwar Ghuloum | 63937db | 2013-05-24 09:08:32 -0700 | [diff] [blame] | 145 | #endif |
| 146 | |
Ian Rogers | 96faf5b | 2013-08-09 22:05:32 -0700 | [diff] [blame] | 147 | #define PURE __attribute__ ((__pure__)) |
| 148 | |
Elliott Hughes | 74787a3 | 2011-12-16 15:39:48 -0800 | [diff] [blame] | 149 | // bionic and glibc both have TEMP_FAILURE_RETRY, but Mac OS' libc doesn't. |
| 150 | #ifndef TEMP_FAILURE_RETRY |
| 151 | #define TEMP_FAILURE_RETRY(exp) ({ \ |
Elliott Hughes | 398f64b | 2012-03-26 18:05:48 -0700 | [diff] [blame] | 152 | typeof(exp) _rc; \ |
Elliott Hughes | 74787a3 | 2011-12-16 15:39:48 -0800 | [diff] [blame] | 153 | do { \ |
| 154 | _rc = (exp); \ |
| 155 | } while (_rc == -1 && errno == EINTR); \ |
| 156 | _rc; }) |
| 157 | #endif |
| 158 | |
Elliott Hughes | c151f90 | 2012-06-21 20:33:21 -0700 | [diff] [blame] | 159 | template<typename T> void UNUSED(const T&) {} |
| 160 | |
Elliott Hughes | f834936 | 2012-06-18 15:00:06 -0700 | [diff] [blame] | 161 | #if defined(__SUPPORT_TS_ANNOTATION__) |
| 162 | |
| 163 | #define ACQUIRED_AFTER(...) __attribute__ ((acquired_after(__VA_ARGS__))) |
| 164 | #define ACQUIRED_BEFORE(...) __attribute__ ((acquired_before(__VA_ARGS__))) |
| 165 | #define EXCLUSIVE_LOCK_FUNCTION(...) __attribute__ ((exclusive_lock(__VA_ARGS__))) |
| 166 | #define EXCLUSIVE_LOCKS_REQUIRED(...) __attribute__ ((exclusive_locks_required(__VA_ARGS__))) |
| 167 | #define EXCLUSIVE_TRYLOCK_FUNCTION(...) __attribute__ ((exclusive_trylock(__VA_ARGS__))) |
| 168 | #define GUARDED_BY(x) __attribute__ ((guarded_by(x))) |
| 169 | #define GUARDED_VAR __attribute__ ((guarded)) |
| 170 | #define LOCKABLE __attribute__ ((lockable)) |
| 171 | #define LOCK_RETURNED(x) __attribute__ ((lock_returned(x))) |
| 172 | #define LOCKS_EXCLUDED(...) __attribute__ ((locks_excluded(__VA_ARGS__))) |
| 173 | #define NO_THREAD_SAFETY_ANALYSIS __attribute__ ((no_thread_safety_analysis)) |
| 174 | #define PT_GUARDED_BY(x) __attribute__ ((point_to_guarded_by(x))) |
| 175 | #define PT_GUARDED_VAR __attribute__ ((point_to_guarded)) |
| 176 | #define SCOPED_LOCKABLE __attribute__ ((scoped_lockable)) |
| 177 | #define SHARED_LOCK_FUNCTION(...) __attribute__ ((shared_lock(__VA_ARGS__))) |
| 178 | #define SHARED_LOCKS_REQUIRED(...) __attribute__ ((shared_locks_required(__VA_ARGS__))) |
| 179 | #define SHARED_TRYLOCK_FUNCTION(...) __attribute__ ((shared_trylock(__VA_ARGS__))) |
| 180 | #define UNLOCK_FUNCTION(...) __attribute__ ((unlock(__VA_ARGS__))) |
| 181 | |
| 182 | #else |
| 183 | |
| 184 | #define ACQUIRED_AFTER(...) |
| 185 | #define ACQUIRED_BEFORE(...) |
| 186 | #define EXCLUSIVE_LOCK_FUNCTION(...) |
| 187 | #define EXCLUSIVE_LOCKS_REQUIRED(...) |
| 188 | #define EXCLUSIVE_TRYLOCK_FUNCTION(...) |
| 189 | #define GUARDED_BY(x) |
| 190 | #define GUARDED_VAR |
| 191 | #define LOCKABLE |
| 192 | #define LOCK_RETURNED(x) |
| 193 | #define LOCKS_EXCLUDED(...) |
| 194 | #define NO_THREAD_SAFETY_ANALYSIS |
| 195 | #define PT_GUARDED_BY(x) |
| 196 | #define PT_GUARDED_VAR |
| 197 | #define SCOPED_LOCKABLE |
| 198 | #define SHARED_LOCK_FUNCTION(...) |
| 199 | #define SHARED_LOCKS_REQUIRED(...) |
| 200 | #define SHARED_TRYLOCK_FUNCTION(...) |
| 201 | #define UNLOCK_FUNCTION(...) |
| 202 | |
Brian Carlstrom | 7934ac2 | 2013-07-26 10:54:15 -0700 | [diff] [blame] | 203 | #endif // defined(__SUPPORT_TS_ANNOTATION__) |
Elliott Hughes | f834936 | 2012-06-18 15:00:06 -0700 | [diff] [blame] | 204 | |
Brian Carlstrom | fc0e321 | 2013-07-17 14:40:12 -0700 | [diff] [blame] | 205 | #endif // ART_RUNTIME_BASE_MACROS_H_ |