1 /* Copyright (c) 2011, Google Inc.
2  * All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Neither the name of Google Inc. nor the names of its
11  * contributors may be used to endorse or promote products derived from
12  * this software without specific prior written permission.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
15  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
16  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
17  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
18  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
19  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
20  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25  */
26 
27 #ifdef _MSC_VER
28 # include <windows.h>
29 #endif
30 
31 #ifdef __cplusplus
32 # error "This file should be built as pure C to avoid name mangling"
33 #endif
34 
35 #include <stdlib.h>
36 #include <string.h>
37 
38 #include "base/third_party/dynamic_annotations/dynamic_annotations.h"
39 
40 #ifdef __GNUC__
41 /* valgrind.h uses gcc extensions so it won't build with other compilers */
42 # include "base/third_party/valgrind/valgrind.h"
43 #endif
44 
45 /* Compiler-based ThreadSanitizer defines
46    DYNAMIC_ANNOTATIONS_EXTERNAL_IMPL = 1
47    and provides its own definitions of the functions. */
48 
49 #ifndef DYNAMIC_ANNOTATIONS_EXTERNAL_IMPL
50 # define DYNAMIC_ANNOTATIONS_EXTERNAL_IMPL 0
51 #endif
52 
53 /* Each function is empty and called (via a macro) only in debug mode.
54    The arguments are captured by dynamic tools at runtime. */
55 
56 #if DYNAMIC_ANNOTATIONS_ENABLED == 1 \
57     && DYNAMIC_ANNOTATIONS_EXTERNAL_IMPL == 0
58 
59 /* Identical code folding(-Wl,--icf=all) countermeasures.
60    This makes all Annotate* functions different, which prevents the linker from
61    folding them. */
62 #ifdef __COUNTER__
63 #define DYNAMIC_ANNOTATIONS_IMPL \
64   volatile unsigned short lineno = (__LINE__ << 8) + __COUNTER__; (void)lineno;
65 #else
66 #define DYNAMIC_ANNOTATIONS_IMPL \
67   volatile unsigned short lineno = (__LINE__ << 8); (void)lineno;
68 #endif
69 
70 /* WARNING: always add new annotations to the end of the list.
71    Otherwise, lineno (see above) numbers for different Annotate* functions may
72    conflict. */
DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockCreate)73 void DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockCreate)(
74     const char *file, int line, const volatile void *lock)
75 {DYNAMIC_ANNOTATIONS_IMPL}
76 
DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockDestroy)77 void DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockDestroy)(
78     const char *file, int line, const volatile void *lock)
79 {DYNAMIC_ANNOTATIONS_IMPL}
80 
DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockAcquired)81 void DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockAcquired)(
82     const char *file, int line, const volatile void *lock, long is_w)
83 {DYNAMIC_ANNOTATIONS_IMPL}
84 
DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockReleased)85 void DYNAMIC_ANNOTATIONS_NAME(AnnotateRWLockReleased)(
86     const char *file, int line, const volatile void *lock, long is_w)
87 {DYNAMIC_ANNOTATIONS_IMPL}
88 
DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierInit)89 void DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierInit)(
90     const char *file, int line, const volatile void *barrier, long count,
91     long reinitialization_allowed)
92 {DYNAMIC_ANNOTATIONS_IMPL}
93 
DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierWaitBefore)94 void DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierWaitBefore)(
95     const char *file, int line, const volatile void *barrier)
96 {DYNAMIC_ANNOTATIONS_IMPL}
97 
DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierWaitAfter)98 void DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierWaitAfter)(
99     const char *file, int line, const volatile void *barrier)
100 {DYNAMIC_ANNOTATIONS_IMPL}
101 
DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierDestroy)102 void DYNAMIC_ANNOTATIONS_NAME(AnnotateBarrierDestroy)(
103     const char *file, int line, const volatile void *barrier)
104 {DYNAMIC_ANNOTATIONS_IMPL}
105 
DYNAMIC_ANNOTATIONS_NAME(AnnotateCondVarWait)106 void DYNAMIC_ANNOTATIONS_NAME(AnnotateCondVarWait)(
107     const char *file, int line, const volatile void *cv,
108     const volatile void *lock)
109 {DYNAMIC_ANNOTATIONS_IMPL}
110 
DYNAMIC_ANNOTATIONS_NAME(AnnotateCondVarSignal)111 void DYNAMIC_ANNOTATIONS_NAME(AnnotateCondVarSignal)(
112     const char *file, int line, const volatile void *cv)
113 {DYNAMIC_ANNOTATIONS_IMPL}
114 
DYNAMIC_ANNOTATIONS_NAME(AnnotateCondVarSignalAll)115 void DYNAMIC_ANNOTATIONS_NAME(AnnotateCondVarSignalAll)(
116     const char *file, int line, const volatile void *cv)
117 {DYNAMIC_ANNOTATIONS_IMPL}
118 
DYNAMIC_ANNOTATIONS_NAME(AnnotateHappensBefore)119 void DYNAMIC_ANNOTATIONS_NAME(AnnotateHappensBefore)(
120     const char *file, int line, const volatile void *obj)
121 {DYNAMIC_ANNOTATIONS_IMPL};
122 
DYNAMIC_ANNOTATIONS_NAME(AnnotateHappensAfter)123 void DYNAMIC_ANNOTATIONS_NAME(AnnotateHappensAfter)(
124     const char *file, int line, const volatile void *obj)
125 {DYNAMIC_ANNOTATIONS_IMPL};
126 
DYNAMIC_ANNOTATIONS_NAME(AnnotatePublishMemoryRange)127 void DYNAMIC_ANNOTATIONS_NAME(AnnotatePublishMemoryRange)(
128     const char *file, int line, const volatile void *address, long size)
129 {DYNAMIC_ANNOTATIONS_IMPL}
130 
DYNAMIC_ANNOTATIONS_NAME(AnnotateUnpublishMemoryRange)131 void DYNAMIC_ANNOTATIONS_NAME(AnnotateUnpublishMemoryRange)(
132     const char *file, int line, const volatile void *address, long size)
133 {DYNAMIC_ANNOTATIONS_IMPL}
134 
DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQCreate)135 void DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQCreate)(
136     const char *file, int line, const volatile void *pcq)
137 {DYNAMIC_ANNOTATIONS_IMPL}
138 
DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQDestroy)139 void DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQDestroy)(
140     const char *file, int line, const volatile void *pcq)
141 {DYNAMIC_ANNOTATIONS_IMPL}
142 
DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQPut)143 void DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQPut)(
144     const char *file, int line, const volatile void *pcq)
145 {DYNAMIC_ANNOTATIONS_IMPL}
146 
DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQGet)147 void DYNAMIC_ANNOTATIONS_NAME(AnnotatePCQGet)(
148     const char *file, int line, const volatile void *pcq)
149 {DYNAMIC_ANNOTATIONS_IMPL}
150 
DYNAMIC_ANNOTATIONS_NAME(AnnotateNewMemory)151 void DYNAMIC_ANNOTATIONS_NAME(AnnotateNewMemory)(
152     const char *file, int line, const volatile void *mem, long size)
153 {DYNAMIC_ANNOTATIONS_IMPL}
154 
DYNAMIC_ANNOTATIONS_NAME(AnnotateExpectRace)155 void DYNAMIC_ANNOTATIONS_NAME(AnnotateExpectRace)(
156     const char *file, int line, const volatile void *mem,
157     const char *description)
158 {DYNAMIC_ANNOTATIONS_IMPL}
159 
DYNAMIC_ANNOTATIONS_NAME(AnnotateFlushExpectedRaces)160 void DYNAMIC_ANNOTATIONS_NAME(AnnotateFlushExpectedRaces)(
161     const char *file, int line)
162 {DYNAMIC_ANNOTATIONS_IMPL}
163 
DYNAMIC_ANNOTATIONS_NAME(AnnotateBenignRace)164 void DYNAMIC_ANNOTATIONS_NAME(AnnotateBenignRace)(
165     const char *file, int line, const volatile void *mem,
166     const char *description)
167 {DYNAMIC_ANNOTATIONS_IMPL}
168 
DYNAMIC_ANNOTATIONS_NAME(AnnotateBenignRaceSized)169 void DYNAMIC_ANNOTATIONS_NAME(AnnotateBenignRaceSized)(
170     const char *file, int line, const volatile void *mem, long size,
171     const char *description)
172 {DYNAMIC_ANNOTATIONS_IMPL}
173 
DYNAMIC_ANNOTATIONS_NAME(AnnotateMutexIsUsedAsCondVar)174 void DYNAMIC_ANNOTATIONS_NAME(AnnotateMutexIsUsedAsCondVar)(
175     const char *file, int line, const volatile void *mu)
176 {DYNAMIC_ANNOTATIONS_IMPL}
177 
DYNAMIC_ANNOTATIONS_NAME(AnnotateMutexIsNotPHB)178 void DYNAMIC_ANNOTATIONS_NAME(AnnotateMutexIsNotPHB)(
179     const char *file, int line, const volatile void *mu)
180 {DYNAMIC_ANNOTATIONS_IMPL}
181 
DYNAMIC_ANNOTATIONS_NAME(AnnotateTraceMemory)182 void DYNAMIC_ANNOTATIONS_NAME(AnnotateTraceMemory)(
183     const char *file, int line, const volatile void *arg)
184 {DYNAMIC_ANNOTATIONS_IMPL}
185 
DYNAMIC_ANNOTATIONS_NAME(AnnotateThreadName)186 void DYNAMIC_ANNOTATIONS_NAME(AnnotateThreadName)(
187     const char *file, int line, const char *name)
188 {DYNAMIC_ANNOTATIONS_IMPL}
189 
DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreReadsBegin)190 void DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreReadsBegin)(
191     const char *file, int line)
192 {DYNAMIC_ANNOTATIONS_IMPL}
193 
DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreReadsEnd)194 void DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreReadsEnd)(
195     const char *file, int line)
196 {DYNAMIC_ANNOTATIONS_IMPL}
197 
DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreWritesBegin)198 void DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreWritesBegin)(
199     const char *file, int line)
200 {DYNAMIC_ANNOTATIONS_IMPL}
201 
DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreWritesEnd)202 void DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreWritesEnd)(
203     const char *file, int line)
204 {DYNAMIC_ANNOTATIONS_IMPL}
205 
DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreSyncBegin)206 void DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreSyncBegin)(
207     const char *file, int line)
208 {DYNAMIC_ANNOTATIONS_IMPL}
209 
DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreSyncEnd)210 void DYNAMIC_ANNOTATIONS_NAME(AnnotateIgnoreSyncEnd)(
211     const char *file, int line)
212 {DYNAMIC_ANNOTATIONS_IMPL}
213 
DYNAMIC_ANNOTATIONS_NAME(AnnotateEnableRaceDetection)214 void DYNAMIC_ANNOTATIONS_NAME(AnnotateEnableRaceDetection)(
215     const char *file, int line, int enable)
216 {DYNAMIC_ANNOTATIONS_IMPL}
217 
DYNAMIC_ANNOTATIONS_NAME(AnnotateNoOp)218 void DYNAMIC_ANNOTATIONS_NAME(AnnotateNoOp)(
219     const char *file, int line, const volatile void *arg)
220 {DYNAMIC_ANNOTATIONS_IMPL}
221 
DYNAMIC_ANNOTATIONS_NAME(AnnotateFlushState)222 void DYNAMIC_ANNOTATIONS_NAME(AnnotateFlushState)(
223     const char *file, int line)
224 {DYNAMIC_ANNOTATIONS_IMPL}
225 
226 #endif  /* DYNAMIC_ANNOTATIONS_ENABLED == 1
227     && DYNAMIC_ANNOTATIONS_EXTERNAL_IMPL == 0 */
228 
229 #if DYNAMIC_ANNOTATIONS_PROVIDE_RUNNING_ON_VALGRIND == 1 \
230     && DYNAMIC_ANNOTATIONS_EXTERNAL_IMPL == 0
GetRunningOnValgrind(void)231 static int GetRunningOnValgrind(void) {
232 #ifdef RUNNING_ON_VALGRIND
233   if (RUNNING_ON_VALGRIND) return 1;
234 #endif
235 
236 #ifndef _MSC_VER
237   char *running_on_valgrind_str = getenv("RUNNING_ON_VALGRIND");
238   if (running_on_valgrind_str) {
239     return strcmp(running_on_valgrind_str, "0") != 0;
240   }
241 #else
242   /* Visual Studio issues warnings if we use getenv,
243    * so we use GetEnvironmentVariableA instead.
244    */
245   char value[100] = "1";
246   int res = GetEnvironmentVariableA("RUNNING_ON_VALGRIND",
247                                     value, sizeof(value));
248   /* value will remain "1" if res == 0 or res >= sizeof(value). The latter
249    * can happen only if the given value is long, in this case it can't be "0".
250    */
251   if (res > 0 && strcmp(value, "0") != 0)
252     return 1;
253 #endif
254   return 0;
255 }
256 
257 /* See the comments in dynamic_annotations.h */
RunningOnValgrind(void)258 int RunningOnValgrind(void) {
259   static volatile int running_on_valgrind = -1;
260   /* C doesn't have thread-safe initialization of statics, and we
261      don't want to depend on pthread_once here, so hack it. */
262   int local_running_on_valgrind = running_on_valgrind;
263   if (local_running_on_valgrind == -1)
264     running_on_valgrind = local_running_on_valgrind = GetRunningOnValgrind();
265   return local_running_on_valgrind;
266 }
267 
268 #endif /* DYNAMIC_ANNOTATIONS_PROVIDE_RUNNING_ON_VALGRIND == 1
269     && DYNAMIC_ANNOTATIONS_EXTERNAL_IMPL == 0 */
270