xref: /aosp_15_r20/art/runtime/verifier/class_verifier.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
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                                        &reg_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