1 /*
2 * Copyright (C) 2012 The Android Open Source Project
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 #include "reg_type-inl.h"
18
19 #include "android-base/stringprintf.h"
20
21 #include "base/arena_bit_vector.h"
22 #include "base/bit_vector-inl.h"
23 #include "base/casts.h"
24 #include "class_linker-inl.h"
25 #include "dex/descriptors_names.h"
26 #include "dex/dex_file-inl.h"
27 #include "method_verifier.h"
28 #include "mirror/class-inl.h"
29 #include "mirror/class.h"
30 #include "mirror/object-inl.h"
31 #include "mirror/object_array-inl.h"
32 #include "reg_type_cache-inl.h"
33 #include "scoped_thread_state_change-inl.h"
34
35 #include <limits>
36 #include <sstream>
37
38 namespace art {
39 namespace verifier {
40
41 using android::base::StringPrintf;
42
43 const UndefinedType* UndefinedType::instance_ = nullptr;
44 const ConflictType* ConflictType::instance_ = nullptr;
45 const BooleanType* BooleanType::instance_ = nullptr;
46 const ByteType* ByteType::instance_ = nullptr;
47 const ShortType* ShortType::instance_ = nullptr;
48 const CharType* CharType::instance_ = nullptr;
49 const FloatType* FloatType::instance_ = nullptr;
50 const LongLoType* LongLoType::instance_ = nullptr;
51 const LongHiType* LongHiType::instance_ = nullptr;
52 const DoubleLoType* DoubleLoType::instance_ = nullptr;
53 const DoubleHiType* DoubleHiType::instance_ = nullptr;
54 const IntegerType* IntegerType::instance_ = nullptr;
55 const NullType* NullType::instance_ = nullptr;
56
PrimitiveType(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)57 PrimitiveType::PrimitiveType(ObjPtr<mirror::Class> klass,
58 const std::string_view& descriptor,
59 uint16_t cache_id)
60 : RegType(klass, descriptor, cache_id) {
61 CHECK(klass != nullptr);
62 CHECK(!descriptor.empty());
63 }
64
Cat1Type(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)65 Cat1Type::Cat1Type(ObjPtr<mirror::Class> klass,
66 const std::string_view& descriptor,
67 uint16_t cache_id)
68 : PrimitiveType(klass, descriptor, cache_id) {
69 }
70
Cat2Type(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)71 Cat2Type::Cat2Type(ObjPtr<mirror::Class> klass,
72 const std::string_view& descriptor,
73 uint16_t cache_id)
74 : PrimitiveType(klass, descriptor, cache_id) {
75 }
76
Dump() const77 std::string PreciseConstType::Dump() const {
78 std::stringstream result;
79 uint32_t val = ConstantValue();
80 if (val == 0) {
81 CHECK(IsPreciseConstant());
82 result << "Zero/null";
83 } else {
84 result << "Precise ";
85 if (IsConstantShort()) {
86 result << StringPrintf("Constant: %d", val);
87 } else {
88 result << StringPrintf("Constant: 0x%x", val);
89 }
90 }
91 return result.str();
92 }
93
Dump() const94 std::string BooleanType::Dump() const {
95 return "Boolean";
96 }
97
Dump() const98 std::string ConflictType::Dump() const {
99 return "Conflict";
100 }
101
Dump() const102 std::string ByteType::Dump() const {
103 return "Byte";
104 }
105
Dump() const106 std::string ShortType::Dump() const {
107 return "Short";
108 }
109
Dump() const110 std::string CharType::Dump() const {
111 return "Char";
112 }
113
Dump() const114 std::string FloatType::Dump() const {
115 return "Float";
116 }
117
Dump() const118 std::string LongLoType::Dump() const {
119 return "Long (Low Half)";
120 }
121
Dump() const122 std::string LongHiType::Dump() const {
123 return "Long (High Half)";
124 }
125
Dump() const126 std::string DoubleLoType::Dump() const {
127 return "Double (Low Half)";
128 }
129
Dump() const130 std::string DoubleHiType::Dump() const {
131 return "Double (High Half)";
132 }
133
Dump() const134 std::string IntegerType::Dump() const {
135 return "Integer";
136 }
137
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)138 const DoubleHiType* DoubleHiType::CreateInstance(ObjPtr<mirror::Class> klass,
139 const std::string_view& descriptor,
140 uint16_t cache_id) {
141 CHECK(instance_ == nullptr);
142 instance_ = new DoubleHiType(klass, descriptor, cache_id);
143 return instance_;
144 }
145
Destroy()146 void DoubleHiType::Destroy() {
147 if (instance_ != nullptr) {
148 delete instance_;
149 instance_ = nullptr;
150 }
151 }
152
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)153 const DoubleLoType* DoubleLoType::CreateInstance(ObjPtr<mirror::Class> klass,
154 const std::string_view& descriptor,
155 uint16_t cache_id) {
156 CHECK(instance_ == nullptr);
157 instance_ = new DoubleLoType(klass, descriptor, cache_id);
158 return instance_;
159 }
160
Destroy()161 void DoubleLoType::Destroy() {
162 if (instance_ != nullptr) {
163 delete instance_;
164 instance_ = nullptr;
165 }
166 }
167
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)168 const LongLoType* LongLoType::CreateInstance(ObjPtr<mirror::Class> klass,
169 const std::string_view& descriptor,
170 uint16_t cache_id) {
171 CHECK(instance_ == nullptr);
172 instance_ = new LongLoType(klass, descriptor, cache_id);
173 return instance_;
174 }
175
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)176 const LongHiType* LongHiType::CreateInstance(ObjPtr<mirror::Class> klass,
177 const std::string_view& descriptor,
178 uint16_t cache_id) {
179 CHECK(instance_ == nullptr);
180 instance_ = new LongHiType(klass, descriptor, cache_id);
181 return instance_;
182 }
183
Destroy()184 void LongHiType::Destroy() {
185 if (instance_ != nullptr) {
186 delete instance_;
187 instance_ = nullptr;
188 }
189 }
190
Destroy()191 void LongLoType::Destroy() {
192 if (instance_ != nullptr) {
193 delete instance_;
194 instance_ = nullptr;
195 }
196 }
197
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)198 const FloatType* FloatType::CreateInstance(ObjPtr<mirror::Class> klass,
199 const std::string_view& descriptor,
200 uint16_t cache_id) {
201 CHECK(instance_ == nullptr);
202 instance_ = new FloatType(klass, descriptor, cache_id);
203 return instance_;
204 }
205
Destroy()206 void FloatType::Destroy() {
207 if (instance_ != nullptr) {
208 delete instance_;
209 instance_ = nullptr;
210 }
211 }
212
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)213 const CharType* CharType::CreateInstance(ObjPtr<mirror::Class> klass,
214 const std::string_view& descriptor,
215 uint16_t cache_id) {
216 CHECK(instance_ == nullptr);
217 instance_ = new CharType(klass, descriptor, cache_id);
218 return instance_;
219 }
220
Destroy()221 void CharType::Destroy() {
222 if (instance_ != nullptr) {
223 delete instance_;
224 instance_ = nullptr;
225 }
226 }
227
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)228 const ShortType* ShortType::CreateInstance(ObjPtr<mirror::Class> klass,
229 const std::string_view& descriptor,
230 uint16_t cache_id) {
231 CHECK(instance_ == nullptr);
232 instance_ = new ShortType(klass, descriptor, cache_id);
233 return instance_;
234 }
235
Destroy()236 void ShortType::Destroy() {
237 if (instance_ != nullptr) {
238 delete instance_;
239 instance_ = nullptr;
240 }
241 }
242
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)243 const ByteType* ByteType::CreateInstance(ObjPtr<mirror::Class> klass,
244 const std::string_view& descriptor,
245 uint16_t cache_id) {
246 CHECK(instance_ == nullptr);
247 instance_ = new ByteType(klass, descriptor, cache_id);
248 return instance_;
249 }
250
Destroy()251 void ByteType::Destroy() {
252 if (instance_ != nullptr) {
253 delete instance_;
254 instance_ = nullptr;
255 }
256 }
257
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)258 const IntegerType* IntegerType::CreateInstance(ObjPtr<mirror::Class> klass,
259 const std::string_view& descriptor,
260 uint16_t cache_id) {
261 CHECK(instance_ == nullptr);
262 instance_ = new IntegerType(klass, descriptor, cache_id);
263 return instance_;
264 }
265
Destroy()266 void IntegerType::Destroy() {
267 if (instance_ != nullptr) {
268 delete instance_;
269 instance_ = nullptr;
270 }
271 }
272
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)273 const ConflictType* ConflictType::CreateInstance(ObjPtr<mirror::Class> klass,
274 const std::string_view& descriptor,
275 uint16_t cache_id) {
276 CHECK(instance_ == nullptr);
277 instance_ = new ConflictType(klass, descriptor, cache_id);
278 return instance_;
279 }
280
Destroy()281 void ConflictType::Destroy() {
282 if (instance_ != nullptr) {
283 delete instance_;
284 instance_ = nullptr;
285 }
286 }
287
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)288 const BooleanType* BooleanType::CreateInstance(ObjPtr<mirror::Class> klass,
289 const std::string_view& descriptor,
290 uint16_t cache_id) {
291 CHECK(BooleanType::instance_ == nullptr);
292 instance_ = new BooleanType(klass, descriptor, cache_id);
293 return BooleanType::instance_;
294 }
295
Destroy()296 void BooleanType::Destroy() {
297 if (BooleanType::instance_ != nullptr) {
298 delete instance_;
299 instance_ = nullptr;
300 }
301 }
302
Dump() const303 std::string UndefinedType::Dump() const REQUIRES_SHARED(Locks::mutator_lock_) {
304 return "Undefined";
305 }
306
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)307 const UndefinedType* UndefinedType::CreateInstance(ObjPtr<mirror::Class> klass,
308 const std::string_view& descriptor,
309 uint16_t cache_id) {
310 CHECK(instance_ == nullptr);
311 instance_ = new UndefinedType(klass, descriptor, cache_id);
312 return instance_;
313 }
314
Destroy()315 void UndefinedType::Destroy() {
316 if (instance_ != nullptr) {
317 delete instance_;
318 instance_ = nullptr;
319 }
320 }
321
PreciseReferenceType(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)322 PreciseReferenceType::PreciseReferenceType(ObjPtr<mirror::Class> klass,
323 const std::string_view& descriptor,
324 uint16_t cache_id)
325 : RegType(klass, descriptor, cache_id) {
326 // Note: no check for IsInstantiable() here. We may produce this in case an InstantiationError
327 // would be thrown at runtime, but we need to continue verification and *not* create a
328 // hard failure or abort.
329 CheckConstructorInvariants(this);
330 }
331
Dump() const332 std::string UnresolvedMergedType::Dump() const {
333 std::stringstream result;
334 result << "UnresolvedMergedReferences(" << GetResolvedPart().Dump() << " | ";
335 const BitVector& types = GetUnresolvedTypes();
336
337 bool first = true;
338 for (uint32_t idx : types.Indexes()) {
339 if (!first) {
340 result << ", ";
341 } else {
342 first = false;
343 }
344 result << reg_type_cache_->GetFromId(idx).Dump();
345 }
346 result << ")";
347 return result.str();
348 }
349
Dump() const350 std::string UnresolvedSuperClass::Dump() const {
351 std::stringstream result;
352 uint16_t super_type_id = GetUnresolvedSuperClassChildId();
353 result << "UnresolvedSuperClass(" << reg_type_cache_->GetFromId(super_type_id).Dump() << ")";
354 return result.str();
355 }
356
Dump() const357 std::string UnresolvedReferenceType::Dump() const {
358 std::stringstream result;
359 result << "Unresolved Reference: " << PrettyDescriptor(std::string(GetDescriptor()).c_str());
360 return result.str();
361 }
362
Dump() const363 std::string UnresolvedUninitializedRefType::Dump() const {
364 std::stringstream result;
365 result << "Unresolved And Uninitialized Reference: "
366 << PrettyDescriptor(std::string(GetDescriptor()).c_str())
367 << " Allocation PC: " << GetAllocationPc();
368 return result.str();
369 }
370
Dump() const371 std::string UnresolvedUninitializedThisRefType::Dump() const {
372 std::stringstream result;
373 result << "Unresolved And Uninitialized This Reference: "
374 << PrettyDescriptor(std::string(GetDescriptor()).c_str());
375 return result.str();
376 }
377
Dump() const378 std::string ReferenceType::Dump() const {
379 std::stringstream result;
380 result << "Reference: " << mirror::Class::PrettyDescriptor(GetClass());
381 return result.str();
382 }
383
Dump() const384 std::string PreciseReferenceType::Dump() const {
385 std::stringstream result;
386 result << "Precise Reference: " << mirror::Class::PrettyDescriptor(GetClass());
387 return result.str();
388 }
389
Dump() const390 std::string UninitializedReferenceType::Dump() const {
391 std::stringstream result;
392 result << "Uninitialized Reference: " << mirror::Class::PrettyDescriptor(GetClass());
393 result << " Allocation PC: " << GetAllocationPc();
394 return result.str();
395 }
396
Dump() const397 std::string UninitializedThisReferenceType::Dump() const {
398 std::stringstream result;
399 result << "Uninitialized This Reference: " << mirror::Class::PrettyDescriptor(GetClass());
400 result << "Allocation PC: " << GetAllocationPc();
401 return result.str();
402 }
403
Dump() const404 std::string ImpreciseConstType::Dump() const {
405 std::stringstream result;
406 uint32_t val = ConstantValue();
407 if (val == 0) {
408 result << "Zero/null";
409 } else {
410 result << "Imprecise ";
411 if (IsConstantShort()) {
412 result << StringPrintf("Constant: %d", val);
413 } else {
414 result << StringPrintf("Constant: 0x%x", val);
415 }
416 }
417 return result.str();
418 }
Dump() const419 std::string PreciseConstLoType::Dump() const {
420 std::stringstream result;
421
422 int32_t val = ConstantValueLo();
423 result << "Precise ";
424 if (val >= std::numeric_limits<jshort>::min() &&
425 val <= std::numeric_limits<jshort>::max()) {
426 result << StringPrintf("Low-half Constant: %d", val);
427 } else {
428 result << StringPrintf("Low-half Constant: 0x%x", val);
429 }
430 return result.str();
431 }
432
Dump() const433 std::string ImpreciseConstLoType::Dump() const {
434 std::stringstream result;
435
436 int32_t val = ConstantValueLo();
437 result << "Imprecise ";
438 if (val >= std::numeric_limits<jshort>::min() &&
439 val <= std::numeric_limits<jshort>::max()) {
440 result << StringPrintf("Low-half Constant: %d", val);
441 } else {
442 result << StringPrintf("Low-half Constant: 0x%x", val);
443 }
444 return result.str();
445 }
446
Dump() const447 std::string PreciseConstHiType::Dump() const {
448 std::stringstream result;
449 int32_t val = ConstantValueHi();
450 result << "Precise ";
451 if (val >= std::numeric_limits<jshort>::min() &&
452 val <= std::numeric_limits<jshort>::max()) {
453 result << StringPrintf("High-half Constant: %d", val);
454 } else {
455 result << StringPrintf("High-half Constant: 0x%x", val);
456 }
457 return result.str();
458 }
459
Dump() const460 std::string ImpreciseConstHiType::Dump() const {
461 std::stringstream result;
462 int32_t val = ConstantValueHi();
463 result << "Imprecise ";
464 if (val >= std::numeric_limits<jshort>::min() &&
465 val <= std::numeric_limits<jshort>::max()) {
466 result << StringPrintf("High-half Constant: %d", val);
467 } else {
468 result << StringPrintf("High-half Constant: 0x%x", val);
469 }
470 return result.str();
471 }
472
HighHalf(RegTypeCache * cache) const473 const RegType& RegType::HighHalf(RegTypeCache* cache) const {
474 DCHECK(IsLowHalf());
475 if (IsLongLo()) {
476 return cache->LongHi();
477 } else if (IsDoubleLo()) {
478 return cache->DoubleHi();
479 } else {
480 DCHECK(IsImpreciseConstantLo());
481 const ConstantType* const_val = down_cast<const ConstantType*>(this);
482 return cache->FromCat2ConstHi(const_val->ConstantValue(), false);
483 }
484 }
485
GetPrimitiveType() const486 Primitive::Type RegType::GetPrimitiveType() const {
487 if (IsNonZeroReferenceTypes()) {
488 return Primitive::kPrimNot;
489 } else if (IsBooleanTypes()) {
490 return Primitive::kPrimBoolean;
491 } else if (IsByteTypes()) {
492 return Primitive::kPrimByte;
493 } else if (IsShortTypes()) {
494 return Primitive::kPrimShort;
495 } else if (IsCharTypes()) {
496 return Primitive::kPrimChar;
497 } else if (IsFloat()) {
498 return Primitive::kPrimFloat;
499 } else if (IsIntegralTypes()) {
500 return Primitive::kPrimInt;
501 } else if (IsDoubleLo()) {
502 return Primitive::kPrimDouble;
503 } else {
504 DCHECK(IsLongTypes());
505 return Primitive::kPrimLong;
506 }
507 }
508
IsUninitializedTypes() const509 bool UninitializedType::IsUninitializedTypes() const {
510 return true;
511 }
512
IsNonZeroReferenceTypes() const513 bool UninitializedType::IsNonZeroReferenceTypes() const {
514 return true;
515 }
516
IsNonZeroReferenceTypes() const517 bool UnresolvedType::IsNonZeroReferenceTypes() const {
518 return true;
519 }
520
GetSuperClass(RegTypeCache * cache) const521 const RegType& RegType::GetSuperClass(RegTypeCache* cache) const {
522 if (!IsUnresolvedTypes()) {
523 ObjPtr<mirror::Class> super_klass = GetClass()->GetSuperClass();
524 if (super_klass != nullptr) {
525 // A super class of a precise type isn't precise as a precise type indicates the register
526 // holds exactly that type.
527 std::string temp;
528 return cache->FromClass(super_klass->GetDescriptor(&temp), super_klass, false);
529 } else {
530 return cache->Zero();
531 }
532 } else {
533 if (!IsUnresolvedMergedReference() && !IsUnresolvedSuperClass() &&
534 GetDescriptor()[0] == '[') {
535 // Super class of all arrays is Object.
536 return cache->JavaLangObject(true);
537 } else {
538 return cache->FromUnresolvedSuperClass(*this);
539 }
540 }
541 }
542
IsJavaLangObject() const543 bool RegType::IsJavaLangObject() const REQUIRES_SHARED(Locks::mutator_lock_) {
544 return IsReference() && GetClass()->IsObjectClass();
545 }
546
IsObjectArrayTypes() const547 bool RegType::IsObjectArrayTypes() const REQUIRES_SHARED(Locks::mutator_lock_) {
548 if (IsUnresolvedTypes()) {
549 DCHECK(!IsUnresolvedMergedReference());
550
551 if (IsUnresolvedSuperClass()) {
552 // Cannot be an array, as the superclass of arrays is java.lang.Object (which cannot be
553 // unresolved).
554 return false;
555 }
556
557 // Primitive arrays will always resolve.
558 DCHECK(descriptor_[1] == 'L' || descriptor_[1] == '[');
559 return descriptor_[0] == '[';
560 } else if (HasClass()) {
561 ObjPtr<mirror::Class> type = GetClass();
562 return type->IsArrayClass() && !type->GetComponentType()->IsPrimitive();
563 } else {
564 return false;
565 }
566 }
567
IsArrayTypes() const568 bool RegType::IsArrayTypes() const REQUIRES_SHARED(Locks::mutator_lock_) {
569 if (IsUnresolvedTypes()) {
570 DCHECK(!IsUnresolvedMergedReference());
571
572 if (IsUnresolvedSuperClass()) {
573 // Cannot be an array, as the superclass of arrays is java.lang.Object (which cannot be
574 // unresolved).
575 return false;
576 }
577 return descriptor_[0] == '[';
578 } else if (HasClass()) {
579 return GetClass()->IsArrayClass();
580 } else {
581 return false;
582 }
583 }
584
IsJavaLangObjectArray() const585 bool RegType::IsJavaLangObjectArray() const {
586 if (HasClass()) {
587 ObjPtr<mirror::Class> type = GetClass();
588 return type->IsArrayClass() && type->GetComponentType()->IsObjectClass();
589 }
590 return false;
591 }
592
IsInstantiableTypes() const593 bool RegType::IsInstantiableTypes() const {
594 return IsUnresolvedTypes() || (IsNonZeroReferenceTypes() && GetClass()->IsInstantiable());
595 }
596
SelectNonConstant(const RegType & a,const RegType & b)597 static const RegType& SelectNonConstant(const RegType& a, const RegType& b) {
598 return a.IsConstantTypes() ? b : a;
599 }
600
SelectNonConstant2(const RegType & a,const RegType & b)601 static const RegType& SelectNonConstant2(const RegType& a, const RegType& b) {
602 return a.IsConstantTypes() ? (b.IsZero() ? a : b) : a;
603 }
604
605
606 namespace {
607
608 ObjPtr<mirror::Class> ArrayClassJoin(ObjPtr<mirror::Class> s,
609 ObjPtr<mirror::Class> t,
610 ClassLinker* class_linker)
611 REQUIRES_SHARED(Locks::mutator_lock_);
612
613 ObjPtr<mirror::Class> InterfaceClassJoin(ObjPtr<mirror::Class> s, ObjPtr<mirror::Class> t)
614 REQUIRES_SHARED(Locks::mutator_lock_);
615
616 /*
617 * A basic Join operation on classes. For a pair of types S and T the Join, written S v T = J, is
618 * S <: J, T <: J and for-all U such that S <: U, T <: U then J <: U. That is J is the parent of
619 * S and T such that there isn't a parent of both S and T that isn't also the parent of J (ie J
620 * is the deepest (lowest upper bound) parent of S and T).
621 *
622 * This operation applies for regular classes and arrays, however, for interface types there
623 * needn't be a partial ordering on the types. We could solve the problem of a lack of a partial
624 * order by introducing sets of types, however, the only operation permissible on an interface is
625 * invoke-interface. In the tradition of Java verifiers [1] we defer the verification of interface
626 * types until an invoke-interface call on the interface typed reference at runtime and allow
627 * the perversion of Object being assignable to an interface type (note, however, that we don't
628 * allow assignment of Object or Interface to any concrete class and are therefore type safe).
629 *
630 * Note: This may return null in case of internal errors, e.g., OOME when a new class would have
631 * to be created but there is no heap space. The exception will stay pending, and it is
632 * the job of the caller to handle it.
633 *
634 * [1] Java bytecode verification: algorithms and formalizations, Xavier Leroy
635 */
ClassJoin(ObjPtr<mirror::Class> s,ObjPtr<mirror::Class> t,ClassLinker * class_linker)636 ObjPtr<mirror::Class> ClassJoin(ObjPtr<mirror::Class> s,
637 ObjPtr<mirror::Class> t,
638 ClassLinker* class_linker)
639 REQUIRES_SHARED(Locks::mutator_lock_) {
640 DCHECK(!s->IsPrimitive()) << s->PrettyClass();
641 DCHECK(!t->IsPrimitive()) << t->PrettyClass();
642 if (s == t) {
643 return s;
644 } else if (s->IsAssignableFrom(t)) {
645 return s;
646 } else if (t->IsAssignableFrom(s)) {
647 return t;
648 } else if (s->IsArrayClass() && t->IsArrayClass()) {
649 return ArrayClassJoin(s, t, class_linker);
650 } else if (s->IsInterface() || t->IsInterface()) {
651 return InterfaceClassJoin(s, t);
652 } else {
653 size_t s_depth = s->Depth();
654 size_t t_depth = t->Depth();
655 // Get s and t to the same depth in the hierarchy
656 if (s_depth > t_depth) {
657 while (s_depth > t_depth) {
658 s = s->GetSuperClass();
659 s_depth--;
660 }
661 } else {
662 while (t_depth > s_depth) {
663 t = t->GetSuperClass();
664 t_depth--;
665 }
666 }
667 // Go up the hierarchy until we get to the common parent
668 while (s != t) {
669 s = s->GetSuperClass();
670 t = t->GetSuperClass();
671 }
672 return s;
673 }
674 }
675
ArrayClassJoin(ObjPtr<mirror::Class> s,ObjPtr<mirror::Class> t,ClassLinker * class_linker)676 ObjPtr<mirror::Class> ArrayClassJoin(ObjPtr<mirror::Class> s,
677 ObjPtr<mirror::Class> t,
678 ClassLinker* class_linker) {
679 ObjPtr<mirror::Class> s_ct = s->GetComponentType();
680 ObjPtr<mirror::Class> t_ct = t->GetComponentType();
681 if (s_ct->IsPrimitive() || t_ct->IsPrimitive()) {
682 // Given the types aren't the same, if either array is of primitive types then the only
683 // common parent is java.lang.Object
684 ObjPtr<mirror::Class> result = s->GetSuperClass(); // short-cut to java.lang.Object
685 DCHECK(result->IsObjectClass());
686 return result;
687 }
688 Thread* self = Thread::Current();
689 ObjPtr<mirror::Class> common_elem = ClassJoin(s_ct, t_ct, class_linker);
690 if (UNLIKELY(common_elem == nullptr)) {
691 self->AssertPendingException();
692 return nullptr;
693 }
694 // Note: The following lookup invalidates existing ObjPtr<>s.
695 ObjPtr<mirror::Class> array_class = class_linker->FindArrayClass(self, common_elem);
696 if (UNLIKELY(array_class == nullptr)) {
697 self->AssertPendingException();
698 return nullptr;
699 }
700 return array_class;
701 }
702
InterfaceClassJoin(ObjPtr<mirror::Class> s,ObjPtr<mirror::Class> t)703 ObjPtr<mirror::Class> InterfaceClassJoin(ObjPtr<mirror::Class> s, ObjPtr<mirror::Class> t) {
704 // This is expensive, as we do not have good data structures to do this even halfway
705 // efficiently.
706 //
707 // We're not following JVMS for interface verification (not everything is assignable to an
708 // interface, we trade this for IMT dispatch). We also don't have set types to make up for
709 // it. So we choose one arbitrary common ancestor interface by walking the interface tables
710 // backwards.
711 //
712 // For comparison, runtimes following the JVMS will punt all interface type checking to
713 // runtime.
714 ObjPtr<mirror::IfTable> s_if = s->GetIfTable();
715 int32_t s_if_count = s->GetIfTableCount();
716 ObjPtr<mirror::IfTable> t_if = t->GetIfTable();
717 int32_t t_if_count = t->GetIfTableCount();
718
719 // Note: we'll be using index == count to stand for the argument itself.
720 for (int32_t s_it = s_if_count; s_it >= 0; --s_it) {
721 ObjPtr<mirror::Class> s_cl = s_it == s_if_count ? s : s_if->GetInterface(s_it);
722 if (!s_cl->IsInterface()) {
723 continue;
724 }
725
726 for (int32_t t_it = t_if_count; t_it >= 0; --t_it) {
727 ObjPtr<mirror::Class> t_cl = t_it == t_if_count ? t : t_if->GetInterface(t_it);
728 if (!t_cl->IsInterface()) {
729 continue;
730 }
731
732 if (s_cl == t_cl) {
733 // Found something arbitrary in common.
734 return s_cl;
735 }
736 }
737 }
738
739 // Return java.lang.Object.
740 ObjPtr<mirror::Class> obj_class = s->IsInterface() ? s->GetSuperClass() : t->GetSuperClass();
741 DCHECK(obj_class->IsObjectClass());
742 return obj_class;
743 }
744
745 } // namespace
746
Merge(const RegType & incoming_type,RegTypeCache * reg_types,MethodVerifier * verifier) const747 const RegType& RegType::Merge(const RegType& incoming_type,
748 RegTypeCache* reg_types,
749 MethodVerifier* verifier) const {
750 DCHECK(!Equals(incoming_type)); // Trivial equality handled by caller
751 // Perform pointer equality tests for undefined and conflict to avoid virtual method dispatch.
752 const UndefinedType& undefined = reg_types->Undefined();
753 const ConflictType& conflict = reg_types->Conflict();
754 DCHECK_EQ(this == &undefined, IsUndefined());
755 DCHECK_EQ(&incoming_type == &undefined, incoming_type.IsUndefined());
756 DCHECK_EQ(this == &conflict, IsConflict());
757 DCHECK_EQ(&incoming_type == &conflict, incoming_type.IsConflict());
758 if (this == &undefined || &incoming_type == &undefined) {
759 // There is a difference between undefined and conflict. Conflicts may be copied around, but
760 // not used. Undefined registers must not be copied. So any merge with undefined should return
761 // undefined.
762 return undefined;
763 } else if (this == &conflict || &incoming_type == &conflict) {
764 return conflict; // (Conflict MERGE *) or (* MERGE Conflict) => Conflict
765 } else if (IsConstant() && incoming_type.IsConstant()) {
766 const ConstantType& type1 = *down_cast<const ConstantType*>(this);
767 const ConstantType& type2 = *down_cast<const ConstantType*>(&incoming_type);
768 int32_t val1 = type1.ConstantValue();
769 int32_t val2 = type2.ConstantValue();
770 if (val1 >= 0 && val2 >= 0) {
771 // +ve1 MERGE +ve2 => MAX(+ve1, +ve2)
772 if (val1 >= val2) {
773 if (!type1.IsPreciseConstant()) {
774 return *this;
775 } else {
776 return reg_types->FromCat1Const(val1, false);
777 }
778 } else {
779 if (!type2.IsPreciseConstant()) {
780 return type2;
781 } else {
782 return reg_types->FromCat1Const(val2, false);
783 }
784 }
785 } else if (val1 < 0 && val2 < 0) {
786 // -ve1 MERGE -ve2 => MIN(-ve1, -ve2)
787 if (val1 <= val2) {
788 if (!type1.IsPreciseConstant()) {
789 return *this;
790 } else {
791 return reg_types->FromCat1Const(val1, false);
792 }
793 } else {
794 if (!type2.IsPreciseConstant()) {
795 return type2;
796 } else {
797 return reg_types->FromCat1Const(val2, false);
798 }
799 }
800 } else {
801 // Values are +ve and -ve, choose smallest signed type in which they both fit
802 if (type1.IsConstantByte()) {
803 if (type2.IsConstantByte()) {
804 return reg_types->ByteConstant();
805 } else if (type2.IsConstantShort()) {
806 return reg_types->ShortConstant();
807 } else {
808 return reg_types->IntConstant();
809 }
810 } else if (type1.IsConstantShort()) {
811 if (type2.IsConstantShort()) {
812 return reg_types->ShortConstant();
813 } else {
814 return reg_types->IntConstant();
815 }
816 } else {
817 return reg_types->IntConstant();
818 }
819 }
820 } else if (IsConstantLo() && incoming_type.IsConstantLo()) {
821 const ConstantType& type1 = *down_cast<const ConstantType*>(this);
822 const ConstantType& type2 = *down_cast<const ConstantType*>(&incoming_type);
823 int32_t val1 = type1.ConstantValueLo();
824 int32_t val2 = type2.ConstantValueLo();
825 return reg_types->FromCat2ConstLo(val1 | val2, false);
826 } else if (IsConstantHi() && incoming_type.IsConstantHi()) {
827 const ConstantType& type1 = *down_cast<const ConstantType*>(this);
828 const ConstantType& type2 = *down_cast<const ConstantType*>(&incoming_type);
829 int32_t val1 = type1.ConstantValueHi();
830 int32_t val2 = type2.ConstantValueHi();
831 return reg_types->FromCat2ConstHi(val1 | val2, false);
832 } else if (IsIntegralTypes() && incoming_type.IsIntegralTypes()) {
833 if (IsBooleanTypes() && incoming_type.IsBooleanTypes()) {
834 return reg_types->Boolean(); // boolean MERGE boolean => boolean
835 }
836 if (IsByteTypes() && incoming_type.IsByteTypes()) {
837 return reg_types->Byte(); // byte MERGE byte => byte
838 }
839 if (IsShortTypes() && incoming_type.IsShortTypes()) {
840 return reg_types->Short(); // short MERGE short => short
841 }
842 if (IsCharTypes() && incoming_type.IsCharTypes()) {
843 return reg_types->Char(); // char MERGE char => char
844 }
845 return reg_types->Integer(); // int MERGE * => int
846 } else if ((IsFloatTypes() && incoming_type.IsFloatTypes()) ||
847 (IsLongTypes() && incoming_type.IsLongTypes()) ||
848 (IsLongHighTypes() && incoming_type.IsLongHighTypes()) ||
849 (IsDoubleTypes() && incoming_type.IsDoubleTypes()) ||
850 (IsDoubleHighTypes() && incoming_type.IsDoubleHighTypes())) {
851 // check constant case was handled prior to entry
852 DCHECK_IMPLIES(IsConstant(), !incoming_type.IsConstant());
853 // float/long/double MERGE float/long/double_constant => float/long/double
854 return SelectNonConstant(*this, incoming_type);
855 } else if (IsReferenceTypes() && incoming_type.IsReferenceTypes()) {
856 if (IsUninitializedTypes() || incoming_type.IsUninitializedTypes()) {
857 // Something that is uninitialized hasn't had its constructor called. Unitialized types are
858 // special. They may only ever be merged with themselves (must be taken care of by the
859 // caller of Merge(), see the DCHECK on entry). So mark any other merge as conflicting here.
860 return conflict;
861 } else if (IsZeroOrNull() || incoming_type.IsZeroOrNull()) {
862 return SelectNonConstant2(*this, incoming_type); // 0 MERGE ref => ref
863 } else if (IsJavaLangObject() || incoming_type.IsJavaLangObject()) {
864 return reg_types->JavaLangObject(false); // Object MERGE ref => Object
865 } else if (IsUnresolvedTypes() || incoming_type.IsUnresolvedTypes()) {
866 // We know how to merge an unresolved type with itself, 0 or Object. In this case we
867 // have two sub-classes and don't know how to merge. Create a new string-based unresolved
868 // type that reflects our lack of knowledge and that allows the rest of the unresolved
869 // mechanics to continue.
870 return reg_types->FromUnresolvedMerge(*this, incoming_type, verifier);
871 } else { // Two reference types, compute Join
872 // Do not cache the classes as ClassJoin() can suspend and invalidate ObjPtr<>s.
873 DCHECK(GetClass() != nullptr && !GetClass()->IsPrimitive());
874 DCHECK(incoming_type.GetClass() != nullptr && !incoming_type.GetClass()->IsPrimitive());
875 ObjPtr<mirror::Class> join_class = ClassJoin(GetClass(),
876 incoming_type.GetClass(),
877 reg_types->GetClassLinker());
878 if (UNLIKELY(join_class == nullptr)) {
879 // Internal error joining the classes (e.g., OOME). Report an unresolved reference type.
880 // We cannot report an unresolved merge type, as that will attempt to merge the resolved
881 // components, leaving us in an infinite loop.
882 // We do not want to report the originating exception, as that would require a fast path
883 // out all the way to VerifyClass. Instead attempt to continue on without a detailed type.
884 Thread* self = Thread::Current();
885 self->AssertPendingException();
886 self->ClearException();
887
888 // When compiling on the host, we rather want to abort to ensure determinism for preopting.
889 // (In that case, it is likely a misconfiguration of dex2oat.)
890 if (!kIsTargetBuild && (verifier != nullptr && verifier->IsAotMode())) {
891 LOG(FATAL) << "Could not create class join of "
892 << GetClass()->PrettyClass()
893 << " & "
894 << incoming_type.GetClass()->PrettyClass();
895 UNREACHABLE();
896 }
897
898 return reg_types->MakeUnresolvedReference();
899 }
900
901 // Record the dependency that both `GetClass()` and `incoming_type.GetClass()`
902 // are assignable to `join_class`. The `verifier` is null during unit tests.
903 if (verifier != nullptr) {
904 VerifierDeps::MaybeRecordAssignability(verifier->GetVerifierDeps(),
905 verifier->GetDexFile(),
906 verifier->GetClassDef(),
907 join_class,
908 GetClass());
909 VerifierDeps::MaybeRecordAssignability(verifier->GetVerifierDeps(),
910 verifier->GetDexFile(),
911 verifier->GetClassDef(),
912 join_class,
913 incoming_type.GetClass());
914 }
915 if (GetClass() == join_class && !IsPreciseReference()) {
916 return *this;
917 } else if (incoming_type.GetClass() == join_class && !incoming_type.IsPreciseReference()) {
918 return incoming_type;
919 } else {
920 std::string temp;
921 const char* descriptor = join_class->GetDescriptor(&temp);
922 return reg_types->FromClass(descriptor, join_class, /* precise= */ false);
923 }
924 }
925 } else {
926 return conflict; // Unexpected types => Conflict
927 }
928 }
929
CheckInvariants() const930 void RegType::CheckInvariants() const {
931 if (IsConstant() || IsConstantLo() || IsConstantHi()) {
932 CHECK(descriptor_.empty()) << *this;
933 CHECK(klass_.IsNull()) << *this;
934 }
935 if (!klass_.IsNull()) {
936 CHECK(!descriptor_.empty()) << *this;
937 std::string temp;
938 CHECK_EQ(descriptor_, klass_.Read()->GetDescriptor(&temp)) << *this;
939 }
940 }
941
VisitRoots(RootVisitor * visitor,const RootInfo & root_info) const942 void RegType::VisitRoots(RootVisitor* visitor, const RootInfo& root_info) const {
943 klass_.VisitRootIfNonNull(visitor, root_info);
944 }
945
CheckInvariants() const946 void UninitializedThisReferenceType::CheckInvariants() const {
947 CHECK_EQ(GetAllocationPc(), 0U) << *this;
948 }
949
CheckInvariants() const950 void UnresolvedUninitializedThisRefType::CheckInvariants() const {
951 CHECK_EQ(GetAllocationPc(), 0U) << *this;
952 CHECK(!descriptor_.empty()) << *this;
953 CHECK(klass_.IsNull()) << *this;
954 }
955
CheckInvariants() const956 void UnresolvedUninitializedRefType::CheckInvariants() const {
957 CHECK(!descriptor_.empty()) << *this;
958 CHECK(klass_.IsNull()) << *this;
959 }
960
UnresolvedMergedType(const RegType & resolved,const BitVector & unresolved,const RegTypeCache * reg_type_cache,uint16_t cache_id)961 UnresolvedMergedType::UnresolvedMergedType(const RegType& resolved,
962 const BitVector& unresolved,
963 const RegTypeCache* reg_type_cache,
964 uint16_t cache_id)
965 : UnresolvedType("", cache_id),
966 reg_type_cache_(reg_type_cache),
967 resolved_part_(resolved),
968 unresolved_types_(unresolved, false, unresolved.GetAllocator()) {
969 CheckConstructorInvariants(this);
970 }
CheckInvariants() const971 void UnresolvedMergedType::CheckInvariants() const {
972 CHECK(reg_type_cache_ != nullptr);
973
974 // Unresolved merged types: merged types should be defined.
975 CHECK(descriptor_.empty()) << *this;
976 CHECK(klass_.IsNull()) << *this;
977
978 CHECK(!resolved_part_.IsConflict());
979 CHECK(resolved_part_.IsReferenceTypes());
980 CHECK(!resolved_part_.IsUnresolvedTypes());
981
982 CHECK(resolved_part_.IsZero() ||
983 !(resolved_part_.IsArrayTypes() && !resolved_part_.IsObjectArrayTypes()));
984
985 CHECK_GT(unresolved_types_.NumSetBits(), 0U);
986 bool unresolved_is_array =
987 reg_type_cache_->GetFromId(unresolved_types_.GetHighestBitSet()).IsArrayTypes();
988 for (uint32_t idx : unresolved_types_.Indexes()) {
989 const RegType& t = reg_type_cache_->GetFromId(idx);
990 CHECK_EQ(unresolved_is_array, t.IsArrayTypes());
991 }
992
993 if (!resolved_part_.IsZero()) {
994 CHECK_EQ(resolved_part_.IsArrayTypes(), unresolved_is_array);
995 }
996 }
997
IsArrayTypes() const998 bool UnresolvedMergedType::IsArrayTypes() const {
999 // For a merge to be an array, both the resolved and the unresolved part need to be object
1000 // arrays.
1001 // (Note: we encode a missing resolved part [which doesn't need to be an array] as zero.)
1002
1003 if (!resolved_part_.IsZero() && !resolved_part_.IsArrayTypes()) {
1004 return false;
1005 }
1006
1007 // It is enough to check just one of the merged types. Otherwise the merge should have been
1008 // collapsed (checked in CheckInvariants on construction).
1009 uint32_t idx = unresolved_types_.GetHighestBitSet();
1010 const RegType& unresolved = reg_type_cache_->GetFromId(idx);
1011 return unresolved.IsArrayTypes();
1012 }
IsObjectArrayTypes() const1013 bool UnresolvedMergedType::IsObjectArrayTypes() const {
1014 // Same as IsArrayTypes, as primitive arrays are always resolved.
1015 return IsArrayTypes();
1016 }
1017
CheckInvariants() const1018 void UnresolvedReferenceType::CheckInvariants() const {
1019 CHECK(!descriptor_.empty()) << *this;
1020 CHECK(klass_.IsNull()) << *this;
1021 }
1022
CheckInvariants() const1023 void UnresolvedSuperClass::CheckInvariants() const {
1024 // Unresolved merged types: merged types should be defined.
1025 CHECK(descriptor_.empty()) << *this;
1026 CHECK(klass_.IsNull()) << *this;
1027 CHECK_NE(unresolved_child_id_, 0U) << *this;
1028 }
1029
operator <<(std::ostream & os,const RegType & rhs)1030 std::ostream& operator<<(std::ostream& os, const RegType& rhs) {
1031 os << rhs.Dump();
1032 return os;
1033 }
1034
CreateInstance(ObjPtr<mirror::Class> klass,const std::string_view & descriptor,uint16_t cache_id)1035 const NullType* NullType::CreateInstance(ObjPtr<mirror::Class> klass,
1036 const std::string_view& descriptor,
1037 uint16_t cache_id) {
1038 CHECK(instance_ == nullptr);
1039 instance_ = new NullType(klass, descriptor, cache_id);
1040 return instance_;
1041 }
1042
Destroy()1043 void NullType::Destroy() {
1044 if (NullType::instance_ != nullptr) {
1045 delete instance_;
1046 instance_ = nullptr;
1047 }
1048 }
1049
1050
1051 } // namespace verifier
1052 } // namespace art
1053