1*03ce13f7SAndroid Build Coastguard Worker // Copyright 2021 The SwiftShader Authors. All Rights Reserved.
2*03ce13f7SAndroid Build Coastguard Worker //
3*03ce13f7SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*03ce13f7SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*03ce13f7SAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*03ce13f7SAndroid Build Coastguard Worker //
7*03ce13f7SAndroid Build Coastguard Worker // http://www.apache.org/licenses/LICENSE-2.0
8*03ce13f7SAndroid Build Coastguard Worker //
9*03ce13f7SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*03ce13f7SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*03ce13f7SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*03ce13f7SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*03ce13f7SAndroid Build Coastguard Worker // limitations under the License.
14*03ce13f7SAndroid Build Coastguard Worker
15*03ce13f7SAndroid Build Coastguard Worker #include "Pragma.hpp"
16*03ce13f7SAndroid Build Coastguard Worker #include "PragmaInternals.hpp"
17*03ce13f7SAndroid Build Coastguard Worker
18*03ce13f7SAndroid Build Coastguard Worker #include "Debug.hpp"
19*03ce13f7SAndroid Build Coastguard Worker
20*03ce13f7SAndroid Build Coastguard Worker // The CLANG_NO_SANITIZE_MEMORY macro suppresses MemorySanitizer checks for
21*03ce13f7SAndroid Build Coastguard Worker // use-of-uninitialized-data. It is used to decorate functions with known
22*03ce13f7SAndroid Build Coastguard Worker // false positives.
23*03ce13f7SAndroid Build Coastguard Worker #ifdef __clang__
24*03ce13f7SAndroid Build Coastguard Worker # define CLANG_NO_SANITIZE_MEMORY __attribute__((no_sanitize_memory))
25*03ce13f7SAndroid Build Coastguard Worker #else
26*03ce13f7SAndroid Build Coastguard Worker # define CLANG_NO_SANITIZE_MEMORY
27*03ce13f7SAndroid Build Coastguard Worker #endif
28*03ce13f7SAndroid Build Coastguard Worker
29*03ce13f7SAndroid Build Coastguard Worker namespace {
30*03ce13f7SAndroid Build Coastguard Worker
31*03ce13f7SAndroid Build Coastguard Worker struct PragmaState
32*03ce13f7SAndroid Build Coastguard Worker {
33*03ce13f7SAndroid Build Coastguard Worker bool memorySanitizerInstrumentation = true;
34*03ce13f7SAndroid Build Coastguard Worker bool initializeLocalVariables = false;
35*03ce13f7SAndroid Build Coastguard Worker int optimizationLevel = 2; // Default
36*03ce13f7SAndroid Build Coastguard Worker };
37*03ce13f7SAndroid Build Coastguard Worker
38*03ce13f7SAndroid Build Coastguard Worker // The initialization of static thread-local data is not observed by MemorySanitizer
39*03ce13f7SAndroid Build Coastguard Worker // when inside a shared library, leading to false-positive use-of-uninitialized-data
40*03ce13f7SAndroid Build Coastguard Worker // errors: https://github.com/google/sanitizers/issues/1409
41*03ce13f7SAndroid Build Coastguard Worker // We work around this by assigning an initial value to it ourselves on first use.
42*03ce13f7SAndroid Build Coastguard Worker // Note that since the flag to check whether this initialization has already been
43*03ce13f7SAndroid Build Coastguard Worker // done is itself a static thread-local, we must suppress the MemorySanitizer check
44*03ce13f7SAndroid Build Coastguard Worker // with a function attribute.
getPragmaState()45*03ce13f7SAndroid Build Coastguard Worker CLANG_NO_SANITIZE_MEMORY PragmaState &getPragmaState()
46*03ce13f7SAndroid Build Coastguard Worker {
47*03ce13f7SAndroid Build Coastguard Worker static thread_local bool initialized = false;
48*03ce13f7SAndroid Build Coastguard Worker static thread_local PragmaState state;
49*03ce13f7SAndroid Build Coastguard Worker
50*03ce13f7SAndroid Build Coastguard Worker if(!initialized)
51*03ce13f7SAndroid Build Coastguard Worker {
52*03ce13f7SAndroid Build Coastguard Worker state = {};
53*03ce13f7SAndroid Build Coastguard Worker
54*03ce13f7SAndroid Build Coastguard Worker initialized = true;
55*03ce13f7SAndroid Build Coastguard Worker }
56*03ce13f7SAndroid Build Coastguard Worker
57*03ce13f7SAndroid Build Coastguard Worker return state;
58*03ce13f7SAndroid Build Coastguard Worker }
59*03ce13f7SAndroid Build Coastguard Worker
60*03ce13f7SAndroid Build Coastguard Worker } // namespace
61*03ce13f7SAndroid Build Coastguard Worker
62*03ce13f7SAndroid Build Coastguard Worker namespace rr {
63*03ce13f7SAndroid Build Coastguard Worker
Pragma(BooleanPragmaOption option,bool enable)64*03ce13f7SAndroid Build Coastguard Worker void Pragma(BooleanPragmaOption option, bool enable)
65*03ce13f7SAndroid Build Coastguard Worker {
66*03ce13f7SAndroid Build Coastguard Worker PragmaState &state = ::getPragmaState();
67*03ce13f7SAndroid Build Coastguard Worker
68*03ce13f7SAndroid Build Coastguard Worker switch(option)
69*03ce13f7SAndroid Build Coastguard Worker {
70*03ce13f7SAndroid Build Coastguard Worker case MemorySanitizerInstrumentation:
71*03ce13f7SAndroid Build Coastguard Worker state.memorySanitizerInstrumentation = enable;
72*03ce13f7SAndroid Build Coastguard Worker break;
73*03ce13f7SAndroid Build Coastguard Worker case InitializeLocalVariables:
74*03ce13f7SAndroid Build Coastguard Worker state.initializeLocalVariables = enable;
75*03ce13f7SAndroid Build Coastguard Worker break;
76*03ce13f7SAndroid Build Coastguard Worker default:
77*03ce13f7SAndroid Build Coastguard Worker UNSUPPORTED("Unknown Boolean pragma option %d", int(option));
78*03ce13f7SAndroid Build Coastguard Worker }
79*03ce13f7SAndroid Build Coastguard Worker }
80*03ce13f7SAndroid Build Coastguard Worker
Pragma(IntegerPragmaOption option,int value)81*03ce13f7SAndroid Build Coastguard Worker void Pragma(IntegerPragmaOption option, int value)
82*03ce13f7SAndroid Build Coastguard Worker {
83*03ce13f7SAndroid Build Coastguard Worker PragmaState &state = ::getPragmaState();
84*03ce13f7SAndroid Build Coastguard Worker
85*03ce13f7SAndroid Build Coastguard Worker switch(option)
86*03ce13f7SAndroid Build Coastguard Worker {
87*03ce13f7SAndroid Build Coastguard Worker case OptimizationLevel:
88*03ce13f7SAndroid Build Coastguard Worker state.optimizationLevel = value;
89*03ce13f7SAndroid Build Coastguard Worker break;
90*03ce13f7SAndroid Build Coastguard Worker default:
91*03ce13f7SAndroid Build Coastguard Worker UNSUPPORTED("Unknown integer pragma option %d", int(option));
92*03ce13f7SAndroid Build Coastguard Worker }
93*03ce13f7SAndroid Build Coastguard Worker }
94*03ce13f7SAndroid Build Coastguard Worker
getPragmaState(BooleanPragmaOption option)95*03ce13f7SAndroid Build Coastguard Worker bool getPragmaState(BooleanPragmaOption option)
96*03ce13f7SAndroid Build Coastguard Worker {
97*03ce13f7SAndroid Build Coastguard Worker PragmaState &state = ::getPragmaState();
98*03ce13f7SAndroid Build Coastguard Worker
99*03ce13f7SAndroid Build Coastguard Worker switch(option)
100*03ce13f7SAndroid Build Coastguard Worker {
101*03ce13f7SAndroid Build Coastguard Worker case MemorySanitizerInstrumentation:
102*03ce13f7SAndroid Build Coastguard Worker return state.memorySanitizerInstrumentation;
103*03ce13f7SAndroid Build Coastguard Worker case InitializeLocalVariables:
104*03ce13f7SAndroid Build Coastguard Worker return state.initializeLocalVariables;
105*03ce13f7SAndroid Build Coastguard Worker default:
106*03ce13f7SAndroid Build Coastguard Worker UNSUPPORTED("Unknown Boolean pragma option %d", int(option));
107*03ce13f7SAndroid Build Coastguard Worker return false;
108*03ce13f7SAndroid Build Coastguard Worker }
109*03ce13f7SAndroid Build Coastguard Worker }
110*03ce13f7SAndroid Build Coastguard Worker
getPragmaState(IntegerPragmaOption option)111*03ce13f7SAndroid Build Coastguard Worker int getPragmaState(IntegerPragmaOption option)
112*03ce13f7SAndroid Build Coastguard Worker {
113*03ce13f7SAndroid Build Coastguard Worker PragmaState &state = ::getPragmaState();
114*03ce13f7SAndroid Build Coastguard Worker
115*03ce13f7SAndroid Build Coastguard Worker switch(option)
116*03ce13f7SAndroid Build Coastguard Worker {
117*03ce13f7SAndroid Build Coastguard Worker case OptimizationLevel:
118*03ce13f7SAndroid Build Coastguard Worker return state.optimizationLevel;
119*03ce13f7SAndroid Build Coastguard Worker default:
120*03ce13f7SAndroid Build Coastguard Worker UNSUPPORTED("Unknown integer pragma option %d", int(option));
121*03ce13f7SAndroid Build Coastguard Worker return 0;
122*03ce13f7SAndroid Build Coastguard Worker }
123*03ce13f7SAndroid Build Coastguard Worker }
124*03ce13f7SAndroid Build Coastguard Worker
ScopedPragma(BooleanPragmaOption option,bool enable)125*03ce13f7SAndroid Build Coastguard Worker ScopedPragma::ScopedPragma(BooleanPragmaOption option, bool enable)
126*03ce13f7SAndroid Build Coastguard Worker {
127*03ce13f7SAndroid Build Coastguard Worker oldState = BooleanPragma{ option, getPragmaState(option) };
128*03ce13f7SAndroid Build Coastguard Worker Pragma(option, enable);
129*03ce13f7SAndroid Build Coastguard Worker }
130*03ce13f7SAndroid Build Coastguard Worker
ScopedPragma(IntegerPragmaOption option,int value)131*03ce13f7SAndroid Build Coastguard Worker ScopedPragma::ScopedPragma(IntegerPragmaOption option, int value)
132*03ce13f7SAndroid Build Coastguard Worker {
133*03ce13f7SAndroid Build Coastguard Worker oldState = IntegerPragma{ option, getPragmaState(option) };
134*03ce13f7SAndroid Build Coastguard Worker Pragma(option, value);
135*03ce13f7SAndroid Build Coastguard Worker }
136*03ce13f7SAndroid Build Coastguard Worker
~ScopedPragma()137*03ce13f7SAndroid Build Coastguard Worker ScopedPragma::~ScopedPragma()
138*03ce13f7SAndroid Build Coastguard Worker {
139*03ce13f7SAndroid Build Coastguard Worker if(std::holds_alternative<BooleanPragma>(oldState))
140*03ce13f7SAndroid Build Coastguard Worker {
141*03ce13f7SAndroid Build Coastguard Worker auto &restore = std::get<BooleanPragma>(oldState);
142*03ce13f7SAndroid Build Coastguard Worker Pragma(restore.option, restore.enable);
143*03ce13f7SAndroid Build Coastguard Worker }
144*03ce13f7SAndroid Build Coastguard Worker else
145*03ce13f7SAndroid Build Coastguard Worker {
146*03ce13f7SAndroid Build Coastguard Worker auto &restore = std::get<IntegerPragma>(oldState);
147*03ce13f7SAndroid Build Coastguard Worker Pragma(restore.option, restore.value);
148*03ce13f7SAndroid Build Coastguard Worker }
149*03ce13f7SAndroid Build Coastguard Worker }
150*03ce13f7SAndroid Build Coastguard Worker
151*03ce13f7SAndroid Build Coastguard Worker } // namespace rr