1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2011 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker *
4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker *
8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker *
10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker */
16*795d594fSAndroid Build Coastguard Worker
17*795d594fSAndroid Build Coastguard Worker #include "class_verifier.h"
18*795d594fSAndroid Build Coastguard Worker
19*795d594fSAndroid Build Coastguard Worker #include <android-base/logging.h>
20*795d594fSAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
21*795d594fSAndroid Build Coastguard Worker
22*795d594fSAndroid Build Coastguard Worker #include "art_method-inl.h"
23*795d594fSAndroid Build Coastguard Worker #include "base/locks.h"
24*795d594fSAndroid Build Coastguard Worker #include "base/logging.h"
25*795d594fSAndroid Build Coastguard Worker #include "base/pointer_size.h"
26*795d594fSAndroid Build Coastguard Worker #include "base/systrace.h"
27*795d594fSAndroid Build Coastguard Worker #include "base/utils.h"
28*795d594fSAndroid Build Coastguard Worker #include "class_linker.h"
29*795d594fSAndroid Build Coastguard Worker #include "compiler_callbacks.h"
30*795d594fSAndroid Build Coastguard Worker #include "dex/class_accessor-inl.h"
31*795d594fSAndroid Build Coastguard Worker #include "dex/class_reference.h"
32*795d594fSAndroid Build Coastguard Worker #include "dex/descriptors_names.h"
33*795d594fSAndroid Build Coastguard Worker #include "dex/dex_file-inl.h"
34*795d594fSAndroid Build Coastguard Worker #include "handle.h"
35*795d594fSAndroid Build Coastguard Worker #include "handle_scope-inl.h"
36*795d594fSAndroid Build Coastguard Worker #include "method_verifier-inl.h"
37*795d594fSAndroid Build Coastguard Worker #include "mirror/class-inl.h"
38*795d594fSAndroid Build Coastguard Worker #include "mirror/dex_cache.h"
39*795d594fSAndroid Build Coastguard Worker #include "runtime.h"
40*795d594fSAndroid Build Coastguard Worker #include "thread.h"
41*795d594fSAndroid Build Coastguard Worker #include "verifier_compiler_binding.h"
42*795d594fSAndroid Build Coastguard Worker #include "verifier/method_verifier.h"
43*795d594fSAndroid Build Coastguard Worker #include "verifier/reg_type_cache.h"
44*795d594fSAndroid Build Coastguard Worker
45*795d594fSAndroid Build Coastguard Worker namespace art HIDDEN {
46*795d594fSAndroid Build Coastguard Worker namespace verifier {
47*795d594fSAndroid Build Coastguard Worker
48*795d594fSAndroid Build Coastguard Worker using android::base::StringPrintf;
49*795d594fSAndroid Build Coastguard Worker
50*795d594fSAndroid Build Coastguard Worker // We print a warning blurb about "dx --no-optimize" when we find monitor-locking issues. Make
51*795d594fSAndroid Build Coastguard Worker // sure we only print this once.
52*795d594fSAndroid Build Coastguard Worker static bool gPrintedDxMonitorText = false;
53*795d594fSAndroid Build Coastguard Worker
UpdateMethodFlags(uint32_t method_index,Handle<mirror::Class> klass,Handle<mirror::DexCache> dex_cache,CompilerCallbacks * callbacks,int error_types)54*795d594fSAndroid Build Coastguard Worker static void UpdateMethodFlags(uint32_t method_index,
55*795d594fSAndroid Build Coastguard Worker Handle<mirror::Class> klass,
56*795d594fSAndroid Build Coastguard Worker Handle<mirror::DexCache> dex_cache,
57*795d594fSAndroid Build Coastguard Worker CompilerCallbacks* callbacks,
58*795d594fSAndroid Build Coastguard Worker int error_types)
59*795d594fSAndroid Build Coastguard Worker REQUIRES_SHARED(Locks::mutator_lock_) {
60*795d594fSAndroid Build Coastguard Worker if (callbacks != nullptr && !CanCompilerHandleVerificationFailure(error_types)) {
61*795d594fSAndroid Build Coastguard Worker MethodReference ref(dex_cache->GetDexFile(), method_index);
62*795d594fSAndroid Build Coastguard Worker callbacks->AddUncompilableMethod(ref);
63*795d594fSAndroid Build Coastguard Worker }
64*795d594fSAndroid Build Coastguard Worker if (klass == nullptr) {
65*795d594fSAndroid Build Coastguard Worker DCHECK(Runtime::Current()->IsAotCompiler());
66*795d594fSAndroid Build Coastguard Worker // Flags will be set at runtime.
67*795d594fSAndroid Build Coastguard Worker return;
68*795d594fSAndroid Build Coastguard Worker }
69*795d594fSAndroid Build Coastguard Worker
70*795d594fSAndroid Build Coastguard Worker // Mark methods with DontCompile/MustCountLocks flags.
71*795d594fSAndroid Build Coastguard Worker ClassLinker* const linker = Runtime::Current()->GetClassLinker();
72*795d594fSAndroid Build Coastguard Worker ArtMethod* method =
73*795d594fSAndroid Build Coastguard Worker klass->FindClassMethod(dex_cache.Get(), method_index, linker->GetImagePointerSize());
74*795d594fSAndroid Build Coastguard Worker DCHECK(method != nullptr);
75*795d594fSAndroid Build Coastguard Worker DCHECK(method->GetDeclaringClass() == klass.Get());
76*795d594fSAndroid Build Coastguard Worker if (!CanCompilerHandleVerificationFailure(error_types)) {
77*795d594fSAndroid Build Coastguard Worker method->SetDontCompile();
78*795d594fSAndroid Build Coastguard Worker }
79*795d594fSAndroid Build Coastguard Worker if ((error_types & VerifyError::VERIFY_ERROR_LOCKING) != 0) {
80*795d594fSAndroid Build Coastguard Worker method->SetMustCountLocks();
81*795d594fSAndroid Build Coastguard Worker }
82*795d594fSAndroid Build Coastguard Worker }
83*795d594fSAndroid Build Coastguard Worker
VerifyClass(Thread * self,VerifierDeps * verifier_deps,const DexFile * dex_file,Handle<mirror::Class> klass,Handle<mirror::DexCache> dex_cache,Handle<mirror::ClassLoader> class_loader,const dex::ClassDef & class_def,CompilerCallbacks * callbacks,HardFailLogMode log_level,uint32_t api_level,std::string * error)84*795d594fSAndroid Build Coastguard Worker FailureKind ClassVerifier::VerifyClass(Thread* self,
85*795d594fSAndroid Build Coastguard Worker VerifierDeps* verifier_deps,
86*795d594fSAndroid Build Coastguard Worker const DexFile* dex_file,
87*795d594fSAndroid Build Coastguard Worker Handle<mirror::Class> klass,
88*795d594fSAndroid Build Coastguard Worker Handle<mirror::DexCache> dex_cache,
89*795d594fSAndroid Build Coastguard Worker Handle<mirror::ClassLoader> class_loader,
90*795d594fSAndroid Build Coastguard Worker const dex::ClassDef& class_def,
91*795d594fSAndroid Build Coastguard Worker CompilerCallbacks* callbacks,
92*795d594fSAndroid Build Coastguard Worker HardFailLogMode log_level,
93*795d594fSAndroid Build Coastguard Worker uint32_t api_level,
94*795d594fSAndroid Build Coastguard Worker std::string* error) {
95*795d594fSAndroid Build Coastguard Worker // A class must not be abstract and final.
96*795d594fSAndroid Build Coastguard Worker if ((class_def.access_flags_ & (kAccAbstract | kAccFinal)) == (kAccAbstract | kAccFinal)) {
97*795d594fSAndroid Build Coastguard Worker *error = "Verifier rejected class ";
98*795d594fSAndroid Build Coastguard Worker *error += PrettyDescriptor(dex_file->GetClassDescriptor(class_def));
99*795d594fSAndroid Build Coastguard Worker *error += ": class is abstract and final.";
100*795d594fSAndroid Build Coastguard Worker return FailureKind::kHardFailure;
101*795d594fSAndroid Build Coastguard Worker }
102*795d594fSAndroid Build Coastguard Worker
103*795d594fSAndroid Build Coastguard Worker // Note that `klass` can be a redefined class, not in the loader's table yet.
104*795d594fSAndroid Build Coastguard Worker // Therefore, we do not use it for class resolution, but only when needing to
105*795d594fSAndroid Build Coastguard Worker // update its methods' flags.
106*795d594fSAndroid Build Coastguard Worker ClassAccessor accessor(*dex_file, class_def);
107*795d594fSAndroid Build Coastguard Worker SCOPED_TRACE << "VerifyClass " << PrettyDescriptor(accessor.GetDescriptor());
108*795d594fSAndroid Build Coastguard Worker metrics::AutoTimer timer{GetMetrics()->ClassVerificationTotalTime()};
109*795d594fSAndroid Build Coastguard Worker
110*795d594fSAndroid Build Coastguard Worker int64_t previous_method_idx[2] = { -1, -1 };
111*795d594fSAndroid Build Coastguard Worker MethodVerifier::FailureData failure_data;
112*795d594fSAndroid Build Coastguard Worker ClassLinker* const linker = Runtime::Current()->GetClassLinker();
113*795d594fSAndroid Build Coastguard Worker
114*795d594fSAndroid Build Coastguard Worker if (accessor.NumMethods() != 0u) {
115*795d594fSAndroid Build Coastguard Worker ArenaPool* arena_pool = Runtime::Current()->GetArenaPool();
116*795d594fSAndroid Build Coastguard Worker RegTypeCache reg_types(self, linker, arena_pool, class_loader, dex_file);
117*795d594fSAndroid Build Coastguard Worker for (const ClassAccessor::Method& method : accessor.GetMethods()) {
118*795d594fSAndroid Build Coastguard Worker int64_t* previous_idx = &previous_method_idx[method.IsStaticOrDirect() ? 0u : 1u];
119*795d594fSAndroid Build Coastguard Worker self->AllowThreadSuspension();
120*795d594fSAndroid Build Coastguard Worker const uint32_t method_idx = method.GetIndex();
121*795d594fSAndroid Build Coastguard Worker if (method_idx == *previous_idx) {
122*795d594fSAndroid Build Coastguard Worker // smali can create dex files with two encoded_methods sharing the same method_idx
123*795d594fSAndroid Build Coastguard Worker // http://code.google.com/p/smali/issues/detail?id=119
124*795d594fSAndroid Build Coastguard Worker continue;
125*795d594fSAndroid Build Coastguard Worker }
126*795d594fSAndroid Build Coastguard Worker *previous_idx = method_idx;
127*795d594fSAndroid Build Coastguard Worker std::string hard_failure_msg;
128*795d594fSAndroid Build Coastguard Worker MethodVerifier::FailureData result =
129*795d594fSAndroid Build Coastguard Worker MethodVerifier::VerifyMethod(self,
130*795d594fSAndroid Build Coastguard Worker arena_pool,
131*795d594fSAndroid Build Coastguard Worker ®_types,
132*795d594fSAndroid Build Coastguard Worker verifier_deps,
133*795d594fSAndroid Build Coastguard Worker method_idx,
134*795d594fSAndroid Build Coastguard Worker dex_cache,
135*795d594fSAndroid Build Coastguard Worker class_def,
136*795d594fSAndroid Build Coastguard Worker method.GetCodeItem(),
137*795d594fSAndroid Build Coastguard Worker method.GetAccessFlags(),
138*795d594fSAndroid Build Coastguard Worker log_level,
139*795d594fSAndroid Build Coastguard Worker api_level,
140*795d594fSAndroid Build Coastguard Worker Runtime::Current()->IsAotCompiler(),
141*795d594fSAndroid Build Coastguard Worker &hard_failure_msg);
142*795d594fSAndroid Build Coastguard Worker if (result.kind == FailureKind::kHardFailure) {
143*795d594fSAndroid Build Coastguard Worker if (failure_data.kind == FailureKind::kHardFailure) {
144*795d594fSAndroid Build Coastguard Worker // If we logged an error before, we need a newline.
145*795d594fSAndroid Build Coastguard Worker *error += "\n";
146*795d594fSAndroid Build Coastguard Worker } else {
147*795d594fSAndroid Build Coastguard Worker // If we didn't log a hard failure before, print the header of the message.
148*795d594fSAndroid Build Coastguard Worker *error += "Verifier rejected class ";
149*795d594fSAndroid Build Coastguard Worker *error += PrettyDescriptor(dex_file->GetClassDescriptor(class_def));
150*795d594fSAndroid Build Coastguard Worker *error += ":";
151*795d594fSAndroid Build Coastguard Worker }
152*795d594fSAndroid Build Coastguard Worker *error += " ";
153*795d594fSAndroid Build Coastguard Worker *error += hard_failure_msg;
154*795d594fSAndroid Build Coastguard Worker } else if (result.kind != FailureKind::kNoFailure) {
155*795d594fSAndroid Build Coastguard Worker UpdateMethodFlags(method.GetIndex(), klass, dex_cache, callbacks, result.types);
156*795d594fSAndroid Build Coastguard Worker if ((result.types & VerifyError::VERIFY_ERROR_LOCKING) != 0) {
157*795d594fSAndroid Build Coastguard Worker // Print a warning about expected slow-down.
158*795d594fSAndroid Build Coastguard Worker // Use a string temporary to print one contiguous warning.
159*795d594fSAndroid Build Coastguard Worker std::string tmp =
160*795d594fSAndroid Build Coastguard Worker StringPrintf("Method %s failed lock verification and will run slower.",
161*795d594fSAndroid Build Coastguard Worker dex_file->PrettyMethod(method.GetIndex()).c_str());
162*795d594fSAndroid Build Coastguard Worker if (!gPrintedDxMonitorText) {
163*795d594fSAndroid Build Coastguard Worker tmp +=
164*795d594fSAndroid Build Coastguard Worker "\nCommon causes for lock verification issues are non-optimized dex code\n"
165*795d594fSAndroid Build Coastguard Worker "and incorrect proguard optimizations.";
166*795d594fSAndroid Build Coastguard Worker gPrintedDxMonitorText = true;
167*795d594fSAndroid Build Coastguard Worker }
168*795d594fSAndroid Build Coastguard Worker LOG(WARNING) << tmp;
169*795d594fSAndroid Build Coastguard Worker }
170*795d594fSAndroid Build Coastguard Worker }
171*795d594fSAndroid Build Coastguard Worker
172*795d594fSAndroid Build Coastguard Worker // Merge the result for the method into the global state for the class.
173*795d594fSAndroid Build Coastguard Worker failure_data.Merge(result);
174*795d594fSAndroid Build Coastguard Worker }
175*795d594fSAndroid Build Coastguard Worker }
176*795d594fSAndroid Build Coastguard Worker uint64_t elapsed_time_microseconds = timer.Stop();
177*795d594fSAndroid Build Coastguard Worker VLOG(verifier) << "VerifyClass took " << PrettyDuration(UsToNs(elapsed_time_microseconds))
178*795d594fSAndroid Build Coastguard Worker << ", class: " << PrettyDescriptor(dex_file->GetClassDescriptor(class_def));
179*795d594fSAndroid Build Coastguard Worker
180*795d594fSAndroid Build Coastguard Worker GetMetrics()->ClassVerificationCount()->AddOne();
181*795d594fSAndroid Build Coastguard Worker
182*795d594fSAndroid Build Coastguard Worker GetMetrics()->ClassVerificationTotalTimeDelta()->Add(elapsed_time_microseconds);
183*795d594fSAndroid Build Coastguard Worker GetMetrics()->ClassVerificationCountDelta()->AddOne();
184*795d594fSAndroid Build Coastguard Worker
185*795d594fSAndroid Build Coastguard Worker if (failure_data.kind == verifier::FailureKind::kHardFailure && callbacks != nullptr) {
186*795d594fSAndroid Build Coastguard Worker ClassReference ref(dex_file, dex_file->GetIndexForClassDef(class_def));
187*795d594fSAndroid Build Coastguard Worker callbacks->ClassRejected(ref);
188*795d594fSAndroid Build Coastguard Worker }
189*795d594fSAndroid Build Coastguard Worker
190*795d594fSAndroid Build Coastguard Worker return failure_data.kind;
191*795d594fSAndroid Build Coastguard Worker }
192*795d594fSAndroid Build Coastguard Worker
193*795d594fSAndroid Build Coastguard Worker } // namespace verifier
194*795d594fSAndroid Build Coastguard Worker } // namespace art
195