folly/fibers/test/FibersTest.cpp: accommodate ASAN's detect_stack_use_after_return=1
[folly.git] / folly / CPortability.h
1 /*
2  * Copyright 2017 Facebook, Inc.
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  */
16
17 #pragma once
18
19 /* These definitions are in a separate file so that they
20  * may be included from C- as well as C++-based projects. */
21
22 /**
23  * Portable version check.
24  */
25 #ifndef __GNUC_PREREQ
26 # if defined __GNUC__ && defined __GNUC_MINOR__
27 /* nolint */
28 #  define __GNUC_PREREQ(maj, min) ((__GNUC__ << 16) + __GNUC_MINOR__ >= \
29                                    ((maj) << 16) + (min))
30 # else
31 /* nolint */
32 #  define __GNUC_PREREQ(maj, min) 0
33 # endif
34 #endif
35
36 // portable version check for clang
37 #ifndef __CLANG_PREREQ
38 # if defined __clang__ && defined __clang_major__ && defined __clang_minor__
39 /* nolint */
40 #  define __CLANG_PREREQ(maj, min) \
41     ((__clang_major__ << 16) + __clang_minor__ >= ((maj) << 16) + (min))
42 # else
43 /* nolint */
44 #  define __CLANG_PREREQ(maj, min) 0
45 # endif
46 #endif
47
48 #if defined(__has_builtin)
49 #define FOLLY_HAS_BUILTIN(...) __has_builtin(__VA_ARGS__)
50 #else
51 #define FOLLY_HAS_BUILTIN(...) 0
52 #endif
53
54 #if defined(__has_feature)
55 #define FOLLY_HAS_FEATURE(...) __has_feature(__VA_ARGS__)
56 #else
57 #define FOLLY_HAS_FEATURE(...) 0
58 #endif
59
60 /* Define a convenience macro to test when address sanitizer is being used
61  * across the different compilers (e.g. clang, gcc) */
62 #if FOLLY_HAS_FEATURE(address_sanitizer) || __SANITIZE_ADDRESS__
63 # define FOLLY_SANITIZE_ADDRESS 1
64 #endif
65
66 /* Define attribute wrapper for function attribute used to disable
67  * address sanitizer instrumentation. Unfortunately, this attribute
68  * has issues when inlining is used, so disable that as well. */
69 #ifdef FOLLY_SANITIZE_ADDRESS
70 # if defined(__clang__)
71 #  if __has_attribute(__no_sanitize__)
72 #   define FOLLY_DISABLE_ADDRESS_SANITIZER \
73       __attribute__((__no_sanitize__("address"), __noinline__))
74 #  elif __has_attribute(__no_address_safety_analysis__)
75 #   define FOLLY_DISABLE_ADDRESS_SANITIZER \
76       __attribute__((__no_address_safety_analysis__, __noinline__))
77 #  elif __has_attribute(__no_sanitize_address__)
78 #   define FOLLY_DISABLE_ADDRESS_SANITIZER \
79       __attribute__((__no_sanitize_address__, __noinline__))
80 #  endif
81 # elif defined(__GNUC__)
82 #  define FOLLY_DISABLE_ADDRESS_SANITIZER \
83      __attribute__((__no_address_safety_analysis__, __noinline__))
84 # endif
85 #endif
86 #ifndef FOLLY_DISABLE_ADDRESS_SANITIZER
87 # define FOLLY_DISABLE_ADDRESS_SANITIZER
88 #endif
89
90 /* Define a convenience macro to test when thread sanitizer is being used
91  * across the different compilers (e.g. clang, gcc) */
92 #if FOLLY_HAS_FEATURE(thread_sanitizer) || __SANITIZE_THREAD__
93 # define FOLLY_SANITIZE_THREAD 1
94 #endif
95
96 /**
97  * ASAN/MSAN/TSAN define pre-processor symbols:
98  * ADDRESS_SANITIZER/MEMORY_SANITIZER/THREAD_SANITIZER.
99  *
100  * UBSAN doesn't define anything and makes it hard to
101  * conditionally compile.
102  *
103  * The build system should define UNDEFINED_SANITIZER=1 when UBSAN is
104  * used as folly whitelists some functions.
105  */
106 #if UNDEFINED_SANITIZER
107 #define FOLLY_DISABLE_UNDEFINED_BEHAVIOR_SANITIZER(...) \
108   __attribute__((no_sanitize(__VA_ARGS__)))
109 #else
110 #define FOLLY_DISABLE_UNDEFINED_BEHAVIOR_SANITIZER(...)
111 #endif // UNDEFINED_SANITIZER
112
113 /**
114  * Macro for marking functions as having public visibility.
115  */
116 #if defined(__GNUC__)
117 # if __GNUC_PREREQ(4, 9)
118 #  define FOLLY_EXPORT [[gnu::visibility("default")]]
119 # else
120 #  define FOLLY_EXPORT __attribute__((__visibility__("default")))
121 # endif
122 #else
123 # define FOLLY_EXPORT
124 #endif
125
126 // noinline
127 #ifdef _MSC_VER
128 # define FOLLY_NOINLINE __declspec(noinline)
129 #elif defined(__clang__) || defined(__GNUC__)
130 # define FOLLY_NOINLINE __attribute__((__noinline__))
131 #else
132 # define FOLLY_NOINLINE
133 #endif
134
135 // always inline
136 #ifdef _MSC_VER
137 # define FOLLY_ALWAYS_INLINE __forceinline
138 #elif defined(__clang__) || defined(__GNUC__)
139 # define FOLLY_ALWAYS_INLINE inline __attribute__((__always_inline__))
140 #else
141 # define FOLLY_ALWAYS_INLINE inline
142 #endif
143
144 // attribute hidden
145 #if _MSC_VER
146 #define FOLLY_ATTR_VISIBILITY_HIDDEN
147 #elif defined(__clang__) || defined(__GNUC__)
148 #define FOLLY_ATTR_VISIBILITY_HIDDEN __attribute__((__visibility__("hidden")))
149 #else
150 #define FOLLY_ATTR_VISIBILITY_HIDDEN
151 #endif