blob: 9c3a719a0189faf555f0ad093233ed5a088738ea [file] [log] [blame]
Calin Juravle10e244f2015-01-26 18:54:32 +00001/*
2 * Copyright (C) 2015 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 "reference_type_propagation.h"
18
Mathieu Chartiere401d142015-04-22 13:56:20 -070019#include "class_linker-inl.h"
Calin Juravleacf735c2015-02-12 15:25:22 +000020#include "mirror/class-inl.h"
21#include "mirror/dex_cache.h"
22#include "scoped_thread_state_change.h"
23
Calin Juravle10e244f2015-01-26 18:54:32 +000024namespace art {
25
Vladimir Marko456307a2016-04-19 14:12:13 +000026static inline mirror::DexCache* FindDexCacheWithHint(Thread* self,
27 const DexFile& dex_file,
28 Handle<mirror::DexCache> hint_dex_cache)
29 SHARED_REQUIRES(Locks::mutator_lock_) {
30 if (LIKELY(hint_dex_cache->GetDexFile() == &dex_file)) {
31 return hint_dex_cache.Get();
32 } else {
33 return Runtime::Current()->GetClassLinker()->FindDexCache(self, dex_file);
34 }
35}
36
Vladimir Marko7d1fbf32016-01-26 15:01:12 +000037static inline ReferenceTypeInfo::TypeHandle GetRootHandle(StackHandleScopeCollection* handles,
38 ClassLinker::ClassRoot class_root,
39 ReferenceTypeInfo::TypeHandle* cache) {
40 if (!ReferenceTypeInfo::IsValidHandle(*cache)) {
41 // Mutator lock is required for NewHandle.
42 ClassLinker* linker = Runtime::Current()->GetClassLinker();
43 ScopedObjectAccess soa(Thread::Current());
44 *cache = handles->NewHandle(linker->GetClassRoot(class_root));
45 }
46 return *cache;
47}
48
Aart Bikf417ff42016-04-25 12:51:37 -070049// Returns true if klass is admissible to the propagation: non-null and non-erroneous.
50// For an array type, we also check if the component type is admissible.
51static bool IsAdmissible(mirror::Class* klass) SHARED_REQUIRES(Locks::mutator_lock_) {
52 return klass != nullptr && !klass->IsErroneous() &&
53 (!klass->IsArrayClass() || IsAdmissible(klass->GetComponentType()));
54}
55
Vladimir Marko7d1fbf32016-01-26 15:01:12 +000056ReferenceTypeInfo::TypeHandle ReferenceTypePropagation::HandleCache::GetObjectClassHandle() {
57 return GetRootHandle(handles_, ClassLinker::kJavaLangObject, &object_class_handle_);
58}
59
60ReferenceTypeInfo::TypeHandle ReferenceTypePropagation::HandleCache::GetClassClassHandle() {
61 return GetRootHandle(handles_, ClassLinker::kJavaLangClass, &class_class_handle_);
62}
63
64ReferenceTypeInfo::TypeHandle ReferenceTypePropagation::HandleCache::GetStringClassHandle() {
65 return GetRootHandle(handles_, ClassLinker::kJavaLangString, &string_class_handle_);
66}
67
68ReferenceTypeInfo::TypeHandle ReferenceTypePropagation::HandleCache::GetThrowableClassHandle() {
69 return GetRootHandle(handles_, ClassLinker::kJavaLangThrowable, &throwable_class_handle_);
70}
71
72class ReferenceTypePropagation::RTPVisitor : public HGraphDelegateVisitor {
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +010073 public:
Calin Juravle2e768302015-07-28 14:41:11 +000074 RTPVisitor(HGraph* graph,
Vladimir Marko456307a2016-04-19 14:12:13 +000075 Handle<mirror::DexCache> hint_dex_cache,
Vladimir Marko7d1fbf32016-01-26 15:01:12 +000076 HandleCache* handle_cache,
Nicolas Geoffrayd9994f02016-02-11 17:35:55 +000077 ArenaVector<HInstruction*>* worklist,
78 bool is_first_run)
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +010079 : HGraphDelegateVisitor(graph),
Vladimir Marko456307a2016-04-19 14:12:13 +000080 hint_dex_cache_(hint_dex_cache),
Vladimir Marko7d1fbf32016-01-26 15:01:12 +000081 handle_cache_(handle_cache),
Nicolas Geoffrayd9994f02016-02-11 17:35:55 +000082 worklist_(worklist),
83 is_first_run_(is_first_run) {}
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +010084
85 void VisitNewInstance(HNewInstance* new_instance) OVERRIDE;
86 void VisitLoadClass(HLoadClass* load_class) OVERRIDE;
Calin Juravle2e768302015-07-28 14:41:11 +000087 void VisitClinitCheck(HClinitCheck* clinit_check) OVERRIDE;
88 void VisitLoadString(HLoadString* instr) OVERRIDE;
David Brazdilbbd733e2015-08-18 17:48:17 +010089 void VisitLoadException(HLoadException* instr) OVERRIDE;
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +010090 void VisitNewArray(HNewArray* instr) OVERRIDE;
Calin Juravle2e768302015-07-28 14:41:11 +000091 void VisitParameterValue(HParameterValue* instr) OVERRIDE;
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +010092 void UpdateFieldAccessTypeInfo(HInstruction* instr, const FieldInfo& info);
Vladimir Marko62977ff2016-04-20 15:06:31 +010093 void SetClassAsTypeInfo(HInstruction* instr, mirror::Class* klass, bool is_exact)
94 SHARED_REQUIRES(Locks::mutator_lock_);
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +010095 void VisitInstanceFieldGet(HInstanceFieldGet* instr) OVERRIDE;
96 void VisitStaticFieldGet(HStaticFieldGet* instr) OVERRIDE;
Calin Juravlee460d1d2015-09-29 04:52:17 +010097 void VisitUnresolvedInstanceFieldGet(HUnresolvedInstanceFieldGet* instr) OVERRIDE;
98 void VisitUnresolvedStaticFieldGet(HUnresolvedStaticFieldGet* instr) OVERRIDE;
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +010099 void VisitInvoke(HInvoke* instr) OVERRIDE;
Guillaume "Vermeille" Sanchez72a5eb52015-06-02 17:39:45 +0100100 void VisitArrayGet(HArrayGet* instr) OVERRIDE;
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000101 void VisitCheckCast(HCheckCast* instr) OVERRIDE;
David Brazdilf5552582015-12-27 13:36:12 +0000102 void VisitBoundType(HBoundType* instr) OVERRIDE;
Calin Juravle2e768302015-07-28 14:41:11 +0000103 void VisitNullCheck(HNullCheck* instr) OVERRIDE;
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +0100104 void UpdateReferenceTypeInfo(HInstruction* instr,
105 uint16_t type_idx,
106 const DexFile& dex_file,
107 bool is_exact);
108
109 private:
Vladimir Marko456307a2016-04-19 14:12:13 +0000110 Handle<mirror::DexCache> hint_dex_cache_;
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000111 HandleCache* handle_cache_;
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100112 ArenaVector<HInstruction*>* worklist_;
Nicolas Geoffrayd9994f02016-02-11 17:35:55 +0000113 const bool is_first_run_;
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +0100114};
115
Calin Juravle2e768302015-07-28 14:41:11 +0000116ReferenceTypePropagation::ReferenceTypePropagation(HGraph* graph,
Vladimir Marko456307a2016-04-19 14:12:13 +0000117 Handle<mirror::DexCache> hint_dex_cache,
Calin Juravle2e768302015-07-28 14:41:11 +0000118 StackHandleScopeCollection* handles,
Nicolas Geoffrayd9994f02016-02-11 17:35:55 +0000119 bool is_first_run,
Calin Juravle2e768302015-07-28 14:41:11 +0000120 const char* name)
121 : HOptimization(graph, name),
Vladimir Marko456307a2016-04-19 14:12:13 +0000122 hint_dex_cache_(hint_dex_cache),
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000123 handle_cache_(handles),
Nicolas Geoffrayd9994f02016-02-11 17:35:55 +0000124 worklist_(graph->GetArena()->Adapter(kArenaAllocReferenceTypePropagation)),
125 is_first_run_(is_first_run) {
Calin Juravle2e768302015-07-28 14:41:11 +0000126}
127
Calin Juravlecdfed3d2015-10-26 14:05:01 +0000128void ReferenceTypePropagation::ValidateTypes() {
129 // TODO: move this to the graph checker.
Calin Juravle2e768302015-07-28 14:41:11 +0000130 if (kIsDebugBuild) {
Calin Juravle2e768302015-07-28 14:41:11 +0000131 ScopedObjectAccess soa(Thread::Current());
132 for (HReversePostOrderIterator it(*graph_); !it.Done(); it.Advance()) {
133 HBasicBlock* block = it.Current();
134 for (HInstructionIterator iti(block->GetInstructions()); !iti.Done(); iti.Advance()) {
135 HInstruction* instr = iti.Current();
136 if (instr->GetType() == Primitive::kPrimNot) {
137 DCHECK(instr->GetReferenceTypeInfo().IsValid())
138 << "Invalid RTI for instruction: " << instr->DebugName();
139 if (instr->IsBoundType()) {
140 DCHECK(instr->AsBoundType()->GetUpperBound().IsValid());
141 } else if (instr->IsLoadClass()) {
Calin Juravle98893e12015-10-02 21:05:03 +0100142 HLoadClass* cls = instr->AsLoadClass();
143 DCHECK(cls->GetReferenceTypeInfo().IsExact());
144 DCHECK(!cls->GetLoadedClassRTI().IsValid() || cls->GetLoadedClassRTI().IsExact());
Calin Juravle2e768302015-07-28 14:41:11 +0000145 } else if (instr->IsNullCheck()) {
146 DCHECK(instr->GetReferenceTypeInfo().IsEqual(instr->InputAt(0)->GetReferenceTypeInfo()))
147 << "NullCheck " << instr->GetReferenceTypeInfo()
148 << "Input(0) " << instr->InputAt(0)->GetReferenceTypeInfo();
149 }
150 }
151 }
152 }
153 }
Calin Juravle10e244f2015-01-26 18:54:32 +0000154}
155
Vladimir Markobe10e8e2016-01-22 12:09:44 +0000156void ReferenceTypePropagation::Visit(HInstruction* instruction) {
Vladimir Marko456307a2016-04-19 14:12:13 +0000157 RTPVisitor visitor(graph_, hint_dex_cache_, &handle_cache_, &worklist_, is_first_run_);
Vladimir Markobe10e8e2016-01-22 12:09:44 +0000158 instruction->Accept(&visitor);
159}
160
Calin Juravlecdfed3d2015-10-26 14:05:01 +0000161void ReferenceTypePropagation::Run() {
Vladimir Markobe10e8e2016-01-22 12:09:44 +0000162 worklist_.reserve(kDefaultWorklistSize);
163
Calin Juravlecdfed3d2015-10-26 14:05:01 +0000164 // To properly propagate type info we need to visit in the dominator-based order.
165 // Reverse post order guarantees a node's dominators are visited first.
166 // We take advantage of this order in `VisitBasicBlock`.
167 for (HReversePostOrderIterator it(*graph_); !it.Done(); it.Advance()) {
168 VisitBasicBlock(it.Current());
169 }
170
171 ProcessWorklist();
172 ValidateTypes();
173}
174
Calin Juravleb1498f62015-02-16 13:13:29 +0000175void ReferenceTypePropagation::VisitBasicBlock(HBasicBlock* block) {
Vladimir Marko456307a2016-04-19 14:12:13 +0000176 RTPVisitor visitor(graph_, hint_dex_cache_, &handle_cache_, &worklist_, is_first_run_);
Calin Juravle2e768302015-07-28 14:41:11 +0000177 // Handle Phis first as there might be instructions in the same block who depend on them.
178 for (HInstructionIterator it(block->GetPhis()); !it.Done(); it.Advance()) {
179 VisitPhi(it.Current()->AsPhi());
180 }
Calin Juravlebeba9302015-07-08 15:57:18 +0000181
Calin Juravle2e768302015-07-28 14:41:11 +0000182 // Handle instructions.
Calin Juravleb1498f62015-02-16 13:13:29 +0000183 for (HInstructionIterator it(block->GetInstructions()); !it.Done(); it.Advance()) {
184 HInstruction* instr = it.Current();
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +0100185 instr->Accept(&visitor);
Calin Juravleb1498f62015-02-16 13:13:29 +0000186 }
187
Calin Juravleb1498f62015-02-16 13:13:29 +0000188 // Add extra nodes to bound types.
Calin Juravle61d544b2015-02-23 16:46:57 +0000189 BoundTypeForIfNotNull(block);
Calin Juravleb1498f62015-02-16 13:13:29 +0000190 BoundTypeForIfInstanceOf(block);
Calin Juravle10e244f2015-01-26 18:54:32 +0000191}
192
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000193// Check if we should create a bound type for the given object at the specified
194// position. Because of inlining and the fact we run RTP more than once and we
195// might have a HBoundType already. If we do, we should not create a new one.
196// In this case we also assert that there are no other uses of the object (except
197// the bound type) dominated by the specified dominator_instr or dominator_block.
198static bool ShouldCreateBoundType(HInstruction* position,
199 HInstruction* obj,
200 ReferenceTypeInfo upper_bound,
201 HInstruction* dominator_instr,
202 HBasicBlock* dominator_block)
203 SHARED_REQUIRES(Locks::mutator_lock_) {
204 // If the position where we should insert the bound type is not already a
205 // a bound type then we need to create one.
206 if (position == nullptr || !position->IsBoundType()) {
207 return true;
208 }
209
210 HBoundType* existing_bound_type = position->AsBoundType();
211 if (existing_bound_type->GetUpperBound().IsSupertypeOf(upper_bound)) {
212 if (kIsDebugBuild) {
213 // Check that the existing HBoundType dominates all the uses.
Vladimir Marko46817b82016-03-29 12:21:58 +0100214 for (const HUseListNode<HInstruction*>& use : obj->GetUses()) {
215 HInstruction* user = use.GetUser();
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000216 if (dominator_instr != nullptr) {
217 DCHECK(!dominator_instr->StrictlyDominates(user)
218 || user == existing_bound_type
219 || existing_bound_type->StrictlyDominates(user));
220 } else if (dominator_block != nullptr) {
221 DCHECK(!dominator_block->Dominates(user->GetBlock())
222 || user == existing_bound_type
223 || existing_bound_type->StrictlyDominates(user));
224 }
225 }
226 }
227 } else {
228 // TODO: if the current bound type is a refinement we could update the
229 // existing_bound_type with the a new upper limit. However, we also need to
230 // update its users and have access to the work list.
231 }
232 return false;
233}
234
Calin Juravle61d544b2015-02-23 16:46:57 +0000235void ReferenceTypePropagation::BoundTypeForIfNotNull(HBasicBlock* block) {
Calin Juravleb3306642015-04-20 18:30:42 +0100236 HIf* ifInstruction = block->GetLastInstruction()->AsIf();
237 if (ifInstruction == nullptr) {
Calin Juravle61d544b2015-02-23 16:46:57 +0000238 return;
239 }
Calin Juravleb3306642015-04-20 18:30:42 +0100240 HInstruction* ifInput = ifInstruction->InputAt(0);
Calin Juravle61d544b2015-02-23 16:46:57 +0000241 if (!ifInput->IsNotEqual() && !ifInput->IsEqual()) {
242 return;
243 }
244 HInstruction* input0 = ifInput->InputAt(0);
245 HInstruction* input1 = ifInput->InputAt(1);
Calin Juravleedad8ad2015-04-23 14:34:33 +0100246 HInstruction* obj = nullptr;
Calin Juravle61d544b2015-02-23 16:46:57 +0000247
Calin Juravleedad8ad2015-04-23 14:34:33 +0100248 if (input1->IsNullConstant()) {
Calin Juravle61d544b2015-02-23 16:46:57 +0000249 obj = input0;
Calin Juravleedad8ad2015-04-23 14:34:33 +0100250 } else if (input0->IsNullConstant()) {
Calin Juravle61d544b2015-02-23 16:46:57 +0000251 obj = input1;
252 } else {
253 return;
254 }
255
Nicolas Geoffray7d5ea032015-07-02 15:48:27 +0100256 if (!obj->CanBeNull() || obj->IsNullConstant()) {
257 // Null check is dead code and will be removed by DCE.
258 return;
259 }
260 DCHECK(!obj->IsLoadClass()) << "We should not replace HLoadClass instructions";
261
Calin Juravleb3306642015-04-20 18:30:42 +0100262 // We only need to bound the type if we have uses in the relevant block.
263 // So start with null and create the HBoundType lazily, only if it's needed.
264 HBoundType* bound_type = nullptr;
Calin Juravle61d544b2015-02-23 16:46:57 +0000265 HBasicBlock* notNullBlock = ifInput->IsNotEqual()
Calin Juravleb3306642015-04-20 18:30:42 +0100266 ? ifInstruction->IfTrueSuccessor()
267 : ifInstruction->IfFalseSuccessor();
268
Vladimir Marko46817b82016-03-29 12:21:58 +0100269 const HUseList<HInstruction*>& uses = obj->GetUses();
270 for (auto it = uses.begin(), end = uses.end(); it != end; /* ++it below */) {
271 HInstruction* user = it->GetUser();
272 size_t index = it->GetIndex();
273 // Increment `it` now because `*it` may disappear thanks to user->ReplaceInput().
274 ++it;
Calin Juravle61d544b2015-02-23 16:46:57 +0000275 if (notNullBlock->Dominates(user->GetBlock())) {
Calin Juravleb3306642015-04-20 18:30:42 +0100276 if (bound_type == nullptr) {
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000277 ScopedObjectAccess soa(Thread::Current());
278 HInstruction* insert_point = notNullBlock->GetFirstInstruction();
Calin Juravle2e768302015-07-28 14:41:11 +0000279 ReferenceTypeInfo object_rti = ReferenceTypeInfo::Create(
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000280 handle_cache_.GetObjectClassHandle(), /* is_exact */ true);
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000281 if (ShouldCreateBoundType(insert_point, obj, object_rti, nullptr, notNullBlock)) {
David Brazdilf5552582015-12-27 13:36:12 +0000282 bound_type = new (graph_->GetArena()) HBoundType(obj);
283 bound_type->SetUpperBound(object_rti, /* bound_can_be_null */ false);
Calin Juravle2e768302015-07-28 14:41:11 +0000284 if (obj->GetReferenceTypeInfo().IsValid()) {
285 bound_type->SetReferenceTypeInfo(obj->GetReferenceTypeInfo());
286 }
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000287 notNullBlock->InsertInstructionBefore(bound_type, insert_point);
288 } else {
289 // We already have a bound type on the position we would need to insert
290 // the new one. The existing bound type should dominate all the users
291 // (dchecked) so there's no need to continue.
292 break;
293 }
Calin Juravleb3306642015-04-20 18:30:42 +0100294 }
Vladimir Marko46817b82016-03-29 12:21:58 +0100295 user->ReplaceInput(bound_type, index);
Calin Juravle61d544b2015-02-23 16:46:57 +0000296 }
297 }
298}
299
David Brazdild9510df2015-11-04 23:30:22 +0000300// Returns true if one of the patterns below has been recognized. If so, the
301// InstanceOf instruction together with the true branch of `ifInstruction` will
302// be returned using the out parameters.
303// Recognized patterns:
304// (1) patterns equivalent to `if (obj instanceof X)`
305// (a) InstanceOf -> Equal to 1 -> If
306// (b) InstanceOf -> NotEqual to 0 -> If
307// (c) InstanceOf -> If
308// (2) patterns equivalent to `if (!(obj instanceof X))`
309// (a) InstanceOf -> Equal to 0 -> If
310// (b) InstanceOf -> NotEqual to 1 -> If
311// (c) InstanceOf -> BooleanNot -> If
312static bool MatchIfInstanceOf(HIf* ifInstruction,
313 /* out */ HInstanceOf** instanceOf,
314 /* out */ HBasicBlock** trueBranch) {
315 HInstruction* input = ifInstruction->InputAt(0);
316
317 if (input->IsEqual()) {
318 HInstruction* rhs = input->AsEqual()->GetConstantRight();
319 if (rhs != nullptr) {
320 HInstruction* lhs = input->AsEqual()->GetLeastConstantLeft();
321 if (lhs->IsInstanceOf() && rhs->IsIntConstant()) {
Roland Levillain1a653882016-03-18 18:05:57 +0000322 if (rhs->AsIntConstant()->IsTrue()) {
David Brazdild9510df2015-11-04 23:30:22 +0000323 // Case (1a)
324 *trueBranch = ifInstruction->IfTrueSuccessor();
325 } else {
326 // Case (2a)
Roland Levillain1a653882016-03-18 18:05:57 +0000327 DCHECK(rhs->AsIntConstant()->IsFalse()) << rhs->AsIntConstant()->GetValue();
David Brazdild9510df2015-11-04 23:30:22 +0000328 *trueBranch = ifInstruction->IfFalseSuccessor();
329 }
330 *instanceOf = lhs->AsInstanceOf();
331 return true;
332 }
333 }
334 } else if (input->IsNotEqual()) {
335 HInstruction* rhs = input->AsNotEqual()->GetConstantRight();
336 if (rhs != nullptr) {
337 HInstruction* lhs = input->AsNotEqual()->GetLeastConstantLeft();
338 if (lhs->IsInstanceOf() && rhs->IsIntConstant()) {
Roland Levillain1a653882016-03-18 18:05:57 +0000339 if (rhs->AsIntConstant()->IsFalse()) {
David Brazdild9510df2015-11-04 23:30:22 +0000340 // Case (1b)
341 *trueBranch = ifInstruction->IfTrueSuccessor();
342 } else {
343 // Case (2b)
Roland Levillain1a653882016-03-18 18:05:57 +0000344 DCHECK(rhs->AsIntConstant()->IsTrue()) << rhs->AsIntConstant()->GetValue();
David Brazdild9510df2015-11-04 23:30:22 +0000345 *trueBranch = ifInstruction->IfFalseSuccessor();
346 }
347 *instanceOf = lhs->AsInstanceOf();
348 return true;
349 }
350 }
351 } else if (input->IsInstanceOf()) {
352 // Case (1c)
353 *instanceOf = input->AsInstanceOf();
354 *trueBranch = ifInstruction->IfTrueSuccessor();
355 return true;
356 } else if (input->IsBooleanNot()) {
357 HInstruction* not_input = input->InputAt(0);
358 if (not_input->IsInstanceOf()) {
359 // Case (2c)
360 *instanceOf = not_input->AsInstanceOf();
361 *trueBranch = ifInstruction->IfFalseSuccessor();
362 return true;
363 }
364 }
365
366 return false;
367}
368
Calin Juravleb1498f62015-02-16 13:13:29 +0000369// Detects if `block` is the True block for the pattern
370// `if (x instanceof ClassX) { }`
371// If that's the case insert an HBoundType instruction to bound the type of `x`
372// to `ClassX` in the scope of the dominated blocks.
373void ReferenceTypePropagation::BoundTypeForIfInstanceOf(HBasicBlock* block) {
Calin Juravleb3306642015-04-20 18:30:42 +0100374 HIf* ifInstruction = block->GetLastInstruction()->AsIf();
375 if (ifInstruction == nullptr) {
Calin Juravleb1498f62015-02-16 13:13:29 +0000376 return;
377 }
David Brazdil0d13fee2015-04-17 14:52:19 +0100378
David Brazdild9510df2015-11-04 23:30:22 +0000379 // Try to recognize common `if (instanceof)` and `if (!instanceof)` patterns.
380 HInstanceOf* instanceOf = nullptr;
381 HBasicBlock* instanceOfTrueBlock = nullptr;
382 if (!MatchIfInstanceOf(ifInstruction, &instanceOf, &instanceOfTrueBlock)) {
Calin Juravleb1498f62015-02-16 13:13:29 +0000383 return;
Calin Juravleacf735c2015-02-12 15:25:22 +0000384 }
385
Calin Juravle98893e12015-10-02 21:05:03 +0100386 HLoadClass* load_class = instanceOf->InputAt(1)->AsLoadClass();
387 ReferenceTypeInfo class_rti = load_class->GetLoadedClassRTI();
388 {
Calin Juravle98893e12015-10-02 21:05:03 +0100389 if (!class_rti.IsValid()) {
390 // He have loaded an unresolved class. Don't bother bounding the type.
391 return;
392 }
393 }
Calin Juravleb3306642015-04-20 18:30:42 +0100394 // We only need to bound the type if we have uses in the relevant block.
395 // So start with null and create the HBoundType lazily, only if it's needed.
396 HBoundType* bound_type = nullptr;
397
Calin Juravleb1498f62015-02-16 13:13:29 +0000398 HInstruction* obj = instanceOf->InputAt(0);
Nicolas Geoffrayf9a19952015-06-29 13:43:54 +0100399 if (obj->GetReferenceTypeInfo().IsExact() && !obj->IsPhi()) {
400 // This method is being called while doing a fixed-point calculation
401 // over phis. Non-phis instruction whose type is already known do
402 // not need to be bound to another type.
403 // Not that this also prevents replacing `HLoadClass` with a `HBoundType`.
404 // `HCheckCast` and `HInstanceOf` expect a `HLoadClass` as a second
405 // input.
406 return;
407 }
Nicolas Geoffray7d5ea032015-07-02 15:48:27 +0100408 DCHECK(!obj->IsLoadClass()) << "We should not replace HLoadClass instructions";
Vladimir Marko46817b82016-03-29 12:21:58 +0100409 const HUseList<HInstruction*>& uses = obj->GetUses();
410 for (auto it = uses.begin(), end = uses.end(); it != end; /* ++it below */) {
411 HInstruction* user = it->GetUser();
412 size_t index = it->GetIndex();
413 // Increment `it` now because `*it` may disappear thanks to user->ReplaceInput().
414 ++it;
Calin Juravleb1498f62015-02-16 13:13:29 +0000415 if (instanceOfTrueBlock->Dominates(user->GetBlock())) {
Calin Juravleb3306642015-04-20 18:30:42 +0100416 if (bound_type == nullptr) {
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000417 ScopedObjectAccess soa(Thread::Current());
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000418 HInstruction* insert_point = instanceOfTrueBlock->GetFirstInstruction();
419 if (ShouldCreateBoundType(insert_point, obj, class_rti, nullptr, instanceOfTrueBlock)) {
David Brazdilf5552582015-12-27 13:36:12 +0000420 bound_type = new (graph_->GetArena()) HBoundType(obj);
421 bound_type->SetUpperBound(class_rti, /* InstanceOf fails for null. */ false);
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000422 instanceOfTrueBlock->InsertInstructionBefore(bound_type, insert_point);
423 } else {
424 // We already have a bound type on the position we would need to insert
425 // the new one. The existing bound type should dominate all the users
426 // (dchecked) so there's no need to continue.
427 break;
Calin Juravleb3306642015-04-20 18:30:42 +0100428 }
Calin Juravleb3306642015-04-20 18:30:42 +0100429 }
Vladimir Marko46817b82016-03-29 12:21:58 +0100430 user->ReplaceInput(bound_type, index);
Calin Juravleb1498f62015-02-16 13:13:29 +0000431 }
432 }
Calin Juravleacf735c2015-02-12 15:25:22 +0000433}
434
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000435void ReferenceTypePropagation::RTPVisitor::SetClassAsTypeInfo(HInstruction* instr,
436 mirror::Class* klass,
437 bool is_exact) {
Calin Juravle2e768302015-07-28 14:41:11 +0000438 if (instr->IsInvokeStaticOrDirect() && instr->AsInvokeStaticOrDirect()->IsStringInit()) {
439 // Calls to String.<init> are replaced with a StringFactory.
440 if (kIsDebugBuild) {
Nicolas Geoffrayc89715c2015-10-28 12:06:25 +0000441 HInvoke* invoke = instr->AsInvoke();
Calin Juravle2e768302015-07-28 14:41:11 +0000442 ClassLinker* cl = Runtime::Current()->GetClassLinker();
Vladimir Marko62977ff2016-04-20 15:06:31 +0100443 Thread* self = Thread::Current();
444 StackHandleScope<2> hs(self);
Nicolas Geoffrayc89715c2015-10-28 12:06:25 +0000445 Handle<mirror::DexCache> dex_cache(
Vladimir Marko62977ff2016-04-20 15:06:31 +0100446 hs.NewHandle(FindDexCacheWithHint(self, invoke->GetDexFile(), hint_dex_cache_)));
Nicolas Geoffrayc89715c2015-10-28 12:06:25 +0000447 // Use a null loader. We should probably use the compiling method's class loader,
448 // but then we would need to pass it to RTPVisitor just for this debug check. Since
449 // the method is from the String class, the null loader is good enough.
450 Handle<mirror::ClassLoader> loader;
Andreas Gampe42ef8ab2015-12-03 17:27:32 -0800451 ArtMethod* method = cl->ResolveMethod<ClassLinker::kNoICCECheckForCache>(
Nicolas Geoffrayc89715c2015-10-28 12:06:25 +0000452 invoke->GetDexFile(), invoke->GetDexMethodIndex(), dex_cache, loader, nullptr, kDirect);
Calin Juravle2e768302015-07-28 14:41:11 +0000453 DCHECK(method != nullptr);
454 mirror::Class* declaring_class = method->GetDeclaringClass();
455 DCHECK(declaring_class != nullptr);
456 DCHECK(declaring_class->IsStringClass())
457 << "Expected String class: " << PrettyDescriptor(declaring_class);
458 DCHECK(method->IsConstructor())
459 << "Expected String.<init>: " << PrettyMethod(method);
460 }
461 instr->SetReferenceTypeInfo(
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000462 ReferenceTypeInfo::Create(handle_cache_->GetStringClassHandle(), /* is_exact */ true));
Aart Bikf417ff42016-04-25 12:51:37 -0700463 } else if (IsAdmissible(klass)) {
464 ReferenceTypeInfo::TypeHandle handle = handle_cache_->NewHandle(klass);
465 is_exact = is_exact || handle->CannotBeAssignedFromOtherTypes();
466 instr->SetReferenceTypeInfo(ReferenceTypeInfo::Create(handle, is_exact));
Calin Juravle2e768302015-07-28 14:41:11 +0000467 } else {
Nicolas Geoffray18401b72016-03-11 13:35:51 +0000468 instr->SetReferenceTypeInfo(instr->GetBlock()->GetGraph()->GetInexactObjectRti());
Guillaume "Vermeille" Sanchez104fd8a2015-05-20 17:52:13 +0100469 }
470}
471
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000472void ReferenceTypePropagation::RTPVisitor::UpdateReferenceTypeInfo(HInstruction* instr,
473 uint16_t type_idx,
474 const DexFile& dex_file,
475 bool is_exact) {
Guillaume "Vermeille" Sanchez81d804a2015-05-20 12:42:25 +0100476 DCHECK_EQ(instr->GetType(), Primitive::kPrimNot);
477
Calin Juravleacf735c2015-02-12 15:25:22 +0000478 ScopedObjectAccess soa(Thread::Current());
Vladimir Marko456307a2016-04-19 14:12:13 +0000479 mirror::DexCache* dex_cache = FindDexCacheWithHint(soa.Self(), dex_file, hint_dex_cache_);
Calin Juravleacf735c2015-02-12 15:25:22 +0000480 // Get type from dex cache assuming it was populated by the verifier.
Guillaume Sanchez222862c2015-06-09 18:33:02 +0100481 SetClassAsTypeInfo(instr, dex_cache->GetResolvedType(type_idx), is_exact);
Calin Juravleacf735c2015-02-12 15:25:22 +0000482}
483
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000484void ReferenceTypePropagation::RTPVisitor::VisitNewInstance(HNewInstance* instr) {
Guillaume Sanchez222862c2015-06-09 18:33:02 +0100485 UpdateReferenceTypeInfo(instr, instr->GetTypeIndex(), instr->GetDexFile(), /* is_exact */ true);
Guillaume "Vermeille" Sanchez81d804a2015-05-20 12:42:25 +0100486}
487
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000488void ReferenceTypePropagation::RTPVisitor::VisitNewArray(HNewArray* instr) {
Guillaume Sanchez222862c2015-06-09 18:33:02 +0100489 UpdateReferenceTypeInfo(instr, instr->GetTypeIndex(), instr->GetDexFile(), /* is_exact */ true);
Guillaume "Vermeille" Sanchez81d804a2015-05-20 12:42:25 +0100490}
491
Vladimir Marko456307a2016-04-19 14:12:13 +0000492static mirror::Class* GetClassFromDexCache(Thread* self,
493 const DexFile& dex_file,
494 uint16_t type_idx,
495 Handle<mirror::DexCache> hint_dex_cache)
Calin Juravlee6e3bea2015-10-14 13:53:10 +0000496 SHARED_REQUIRES(Locks::mutator_lock_) {
Vladimir Marko456307a2016-04-19 14:12:13 +0000497 mirror::DexCache* dex_cache = FindDexCacheWithHint(self, dex_file, hint_dex_cache);
Calin Juravlee6e3bea2015-10-14 13:53:10 +0000498 // Get type from dex cache assuming it was populated by the verifier.
499 return dex_cache->GetResolvedType(type_idx);
500}
501
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000502void ReferenceTypePropagation::RTPVisitor::VisitParameterValue(HParameterValue* instr) {
Nicolas Geoffraye418dda2015-08-11 20:03:09 -0700503 // We check if the existing type is valid: the inliner may have set it.
504 if (instr->GetType() == Primitive::kPrimNot && !instr->GetReferenceTypeInfo().IsValid()) {
Vladimir Marko456307a2016-04-19 14:12:13 +0000505 ScopedObjectAccess soa(Thread::Current());
506 mirror::Class* resolved_class = GetClassFromDexCache(soa.Self(),
507 instr->GetDexFile(),
508 instr->GetTypeIndex(),
509 hint_dex_cache_);
Calin Juravlee6e3bea2015-10-14 13:53:10 +0000510 SetClassAsTypeInfo(instr, resolved_class, /* is_exact */ false);
Calin Juravle2e768302015-07-28 14:41:11 +0000511 }
512}
513
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000514void ReferenceTypePropagation::RTPVisitor::UpdateFieldAccessTypeInfo(HInstruction* instr,
515 const FieldInfo& info) {
David Brazdild9510df2015-11-04 23:30:22 +0000516 if (instr->GetType() != Primitive::kPrimNot) {
Guillaume "Vermeille" Sanchez104fd8a2015-05-20 17:52:13 +0100517 return;
518 }
519
Vladimir Marko62977ff2016-04-20 15:06:31 +0100520 ScopedObjectAccess soa(Thread::Current());
David Brazdild9510df2015-11-04 23:30:22 +0000521 mirror::Class* klass = nullptr;
522
523 // The field index is unknown only during tests.
524 if (info.GetFieldIndex() != kUnknownFieldIndex) {
525 ClassLinker* cl = Runtime::Current()->GetClassLinker();
526 ArtField* field = cl->GetResolvedField(info.GetFieldIndex(), info.GetDexCache().Get());
527 // TODO: There are certain cases where we can't resolve the field.
528 // b/21914925 is open to keep track of a repro case for this issue.
529 if (field != nullptr) {
530 klass = field->GetType<false>();
531 }
532 }
533
Calin Juravle2e768302015-07-28 14:41:11 +0000534 SetClassAsTypeInfo(instr, klass, /* is_exact */ false);
Guillaume "Vermeille" Sanchez104fd8a2015-05-20 17:52:13 +0100535}
536
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000537void ReferenceTypePropagation::RTPVisitor::VisitInstanceFieldGet(HInstanceFieldGet* instr) {
Guillaume "Vermeille" Sanchez104fd8a2015-05-20 17:52:13 +0100538 UpdateFieldAccessTypeInfo(instr, instr->GetFieldInfo());
539}
540
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000541void ReferenceTypePropagation::RTPVisitor::VisitStaticFieldGet(HStaticFieldGet* instr) {
Guillaume "Vermeille" Sanchez104fd8a2015-05-20 17:52:13 +0100542 UpdateFieldAccessTypeInfo(instr, instr->GetFieldInfo());
543}
544
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000545void ReferenceTypePropagation::RTPVisitor::VisitUnresolvedInstanceFieldGet(
546 HUnresolvedInstanceFieldGet* instr) {
Calin Juravlee460d1d2015-09-29 04:52:17 +0100547 // TODO: Use descriptor to get the actual type.
548 if (instr->GetFieldType() == Primitive::kPrimNot) {
Nicolas Geoffray18401b72016-03-11 13:35:51 +0000549 instr->SetReferenceTypeInfo(instr->GetBlock()->GetGraph()->GetInexactObjectRti());
Calin Juravlee460d1d2015-09-29 04:52:17 +0100550 }
551}
552
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000553void ReferenceTypePropagation::RTPVisitor::VisitUnresolvedStaticFieldGet(
554 HUnresolvedStaticFieldGet* instr) {
Calin Juravlee460d1d2015-09-29 04:52:17 +0100555 // TODO: Use descriptor to get the actual type.
556 if (instr->GetFieldType() == Primitive::kPrimNot) {
Nicolas Geoffray18401b72016-03-11 13:35:51 +0000557 instr->SetReferenceTypeInfo(instr->GetBlock()->GetGraph()->GetInexactObjectRti());
Calin Juravlee460d1d2015-09-29 04:52:17 +0100558 }
559}
560
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000561void ReferenceTypePropagation::RTPVisitor::VisitLoadClass(HLoadClass* instr) {
Calin Juravleacf735c2015-02-12 15:25:22 +0000562 ScopedObjectAccess soa(Thread::Current());
Calin Juravleacf735c2015-02-12 15:25:22 +0000563 // Get type from dex cache assuming it was populated by the verifier.
Vladimir Marko456307a2016-04-19 14:12:13 +0000564 mirror::Class* resolved_class = GetClassFromDexCache(soa.Self(),
565 instr->GetDexFile(),
566 instr->GetTypeIndex(),
567 hint_dex_cache_);
Aart Bikf417ff42016-04-25 12:51:37 -0700568 if (IsAdmissible(resolved_class)) {
Calin Juravle98893e12015-10-02 21:05:03 +0100569 instr->SetLoadedClassRTI(ReferenceTypeInfo::Create(
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000570 handle_cache_->NewHandle(resolved_class), /* is_exact */ true));
Calin Juravle98893e12015-10-02 21:05:03 +0100571 }
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000572 instr->SetReferenceTypeInfo(
573 ReferenceTypeInfo::Create(handle_cache_->GetClassClassHandle(), /* is_exact */ true));
Calin Juravle2e768302015-07-28 14:41:11 +0000574}
575
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000576void ReferenceTypePropagation::RTPVisitor::VisitClinitCheck(HClinitCheck* instr) {
Calin Juravle2e768302015-07-28 14:41:11 +0000577 instr->SetReferenceTypeInfo(instr->InputAt(0)->GetReferenceTypeInfo());
578}
579
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000580void ReferenceTypePropagation::RTPVisitor::VisitLoadString(HLoadString* instr) {
581 instr->SetReferenceTypeInfo(
582 ReferenceTypeInfo::Create(handle_cache_->GetStringClassHandle(), /* is_exact */ true));
Calin Juravle2e768302015-07-28 14:41:11 +0000583}
584
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000585void ReferenceTypePropagation::RTPVisitor::VisitLoadException(HLoadException* instr) {
David Brazdilbbd733e2015-08-18 17:48:17 +0100586 DCHECK(instr->GetBlock()->IsCatchBlock());
587 TryCatchInformation* catch_info = instr->GetBlock()->GetTryCatchInformation();
588
589 if (catch_info->IsCatchAllTypeIndex()) {
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000590 instr->SetReferenceTypeInfo(
591 ReferenceTypeInfo::Create(handle_cache_->GetThrowableClassHandle(), /* is_exact */ false));
David Brazdilbbd733e2015-08-18 17:48:17 +0100592 } else {
593 UpdateReferenceTypeInfo(instr,
594 catch_info->GetCatchTypeIndex(),
595 catch_info->GetCatchDexFile(),
596 /* is_exact */ false);
597 }
598}
599
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000600void ReferenceTypePropagation::RTPVisitor::VisitNullCheck(HNullCheck* instr) {
Calin Juravle2e768302015-07-28 14:41:11 +0000601 ReferenceTypeInfo parent_rti = instr->InputAt(0)->GetReferenceTypeInfo();
David Brazdilfe860702015-12-02 09:06:57 +0000602 if (parent_rti.IsValid()) {
603 instr->SetReferenceTypeInfo(parent_rti);
604 }
Calin Juravle2e768302015-07-28 14:41:11 +0000605}
606
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000607void ReferenceTypePropagation::RTPVisitor::VisitBoundType(HBoundType* instr) {
David Brazdilf5552582015-12-27 13:36:12 +0000608 ReferenceTypeInfo class_rti = instr->GetUpperBound();
609 if (class_rti.IsValid()) {
Vladimir Marko456307a2016-04-19 14:12:13 +0000610 ScopedObjectAccess soa(Thread::Current());
David Brazdilf5552582015-12-27 13:36:12 +0000611 // Narrow the type as much as possible.
612 HInstruction* obj = instr->InputAt(0);
613 ReferenceTypeInfo obj_rti = obj->GetReferenceTypeInfo();
614 if (class_rti.GetTypeHandle()->CannotBeAssignedFromOtherTypes()) {
615 instr->SetReferenceTypeInfo(
616 ReferenceTypeInfo::Create(class_rti.GetTypeHandle(), /* is_exact */ true));
David Brazdil744a1c62015-12-28 10:53:34 +0000617 } else if (obj_rti.IsValid()) {
618 if (class_rti.IsSupertypeOf(obj_rti)) {
619 // Object type is more specific.
620 instr->SetReferenceTypeInfo(obj_rti);
621 } else {
622 // Upper bound is more specific.
623 instr->SetReferenceTypeInfo(
624 ReferenceTypeInfo::Create(class_rti.GetTypeHandle(), /* is_exact */ false));
625 }
David Brazdilf5552582015-12-27 13:36:12 +0000626 } else {
David Brazdil744a1c62015-12-28 10:53:34 +0000627 // Object not typed yet. Leave BoundType untyped for now rather than
628 // assign the type conservatively.
David Brazdilf5552582015-12-27 13:36:12 +0000629 }
630 instr->SetCanBeNull(obj->CanBeNull() && instr->GetUpperCanBeNull());
631 } else {
632 // The owner of the BoundType was already visited. If the class is unresolved,
633 // the BoundType should have been removed from the data flow and this method
634 // should remove it from the graph.
635 DCHECK(!instr->HasUses());
636 instr->GetBlock()->RemoveInstruction(instr);
637 }
638}
639
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000640void ReferenceTypePropagation::RTPVisitor::VisitCheckCast(HCheckCast* check_cast) {
Calin Juravle98893e12015-10-02 21:05:03 +0100641 HLoadClass* load_class = check_cast->InputAt(1)->AsLoadClass();
642 ReferenceTypeInfo class_rti = load_class->GetLoadedClassRTI();
David Brazdilf5552582015-12-27 13:36:12 +0000643 HBoundType* bound_type = check_cast->GetNext()->AsBoundType();
644 if (bound_type == nullptr || bound_type->GetUpperBound().IsValid()) {
645 // The next instruction is not an uninitialized BoundType. This must be
646 // an RTP pass after SsaBuilder and we do not need to do anything.
647 return;
Calin Juravle98893e12015-10-02 21:05:03 +0100648 }
David Brazdilf5552582015-12-27 13:36:12 +0000649 DCHECK_EQ(bound_type->InputAt(0), check_cast->InputAt(0));
650
651 if (class_rti.IsValid()) {
Nicolas Geoffrayd9994f02016-02-11 17:35:55 +0000652 DCHECK(is_first_run_);
David Brazdilf5552582015-12-27 13:36:12 +0000653 // This is the first run of RTP and class is resolved.
654 bound_type->SetUpperBound(class_rti, /* CheckCast succeeds for nulls. */ true);
655 } else {
656 // This is the first run of RTP and class is unresolved. Remove the binding.
657 // The instruction itself is removed in VisitBoundType so as to not
658 // invalidate HInstructionIterator.
659 bound_type->ReplaceWith(bound_type->InputAt(0));
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000660 }
661}
662
Calin Juravleb1498f62015-02-16 13:13:29 +0000663void ReferenceTypePropagation::VisitPhi(HPhi* phi) {
David Brazdild9510df2015-11-04 23:30:22 +0000664 if (phi->IsDead() || phi->GetType() != Primitive::kPrimNot) {
Calin Juravleb1498f62015-02-16 13:13:29 +0000665 return;
Calin Juravleacf735c2015-02-12 15:25:22 +0000666 }
Calin Juravleb1498f62015-02-16 13:13:29 +0000667
668 if (phi->GetBlock()->IsLoopHeader()) {
669 // Set the initial type for the phi. Use the non back edge input for reaching
670 // a fixed point faster.
David Brazdilfe860702015-12-02 09:06:57 +0000671 HInstruction* first_input = phi->InputAt(0);
672 ReferenceTypeInfo first_input_rti = first_input->GetReferenceTypeInfo();
673 if (first_input_rti.IsValid() && !first_input->IsNullConstant()) {
674 phi->SetCanBeNull(first_input->CanBeNull());
675 phi->SetReferenceTypeInfo(first_input_rti);
676 }
Calin Juravleb1498f62015-02-16 13:13:29 +0000677 AddToWorklist(phi);
Calin Juravle10e244f2015-01-26 18:54:32 +0000678 } else {
Calin Juravleb1498f62015-02-16 13:13:29 +0000679 // Eagerly compute the type of the phi, for quicker convergence. Note
680 // that we don't need to add users to the worklist because we are
681 // doing a reverse post-order visit, therefore either the phi users are
682 // non-loop phi and will be visited later in the visit, or are loop-phis,
683 // and they are already in the work list.
684 UpdateNullability(phi);
685 UpdateReferenceTypeInfo(phi);
686 }
687}
688
689ReferenceTypeInfo ReferenceTypePropagation::MergeTypes(const ReferenceTypeInfo& a,
690 const ReferenceTypeInfo& b) {
Calin Juravle2e768302015-07-28 14:41:11 +0000691 if (!b.IsValid()) {
692 return a;
693 }
694 if (!a.IsValid()) {
695 return b;
Calin Juravle20e60712015-07-01 18:41:04 +0100696 }
697
Calin Juravle2e768302015-07-28 14:41:11 +0000698 bool is_exact = a.IsExact() && b.IsExact();
Calin Juravle52503d82015-11-11 16:58:31 +0000699 ReferenceTypeInfo::TypeHandle result_type_handle;
700 ReferenceTypeInfo::TypeHandle a_type_handle = a.GetTypeHandle();
701 ReferenceTypeInfo::TypeHandle b_type_handle = b.GetTypeHandle();
702 bool a_is_interface = a_type_handle->IsInterface();
703 bool b_is_interface = b_type_handle->IsInterface();
Calin Juravle2e768302015-07-28 14:41:11 +0000704
705 if (a.GetTypeHandle().Get() == b.GetTypeHandle().Get()) {
Calin Juravle52503d82015-11-11 16:58:31 +0000706 result_type_handle = a_type_handle;
Calin Juravle2e768302015-07-28 14:41:11 +0000707 } else if (a.IsSupertypeOf(b)) {
Calin Juravle52503d82015-11-11 16:58:31 +0000708 result_type_handle = a_type_handle;
Calin Juravle2e768302015-07-28 14:41:11 +0000709 is_exact = false;
710 } else if (b.IsSupertypeOf(a)) {
Calin Juravle52503d82015-11-11 16:58:31 +0000711 result_type_handle = b_type_handle;
712 is_exact = false;
713 } else if (!a_is_interface && !b_is_interface) {
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000714 result_type_handle =
715 handle_cache_.NewHandle(a_type_handle->GetCommonSuperClass(b_type_handle));
Calin Juravle2e768302015-07-28 14:41:11 +0000716 is_exact = false;
717 } else {
Calin Juravle52503d82015-11-11 16:58:31 +0000718 // This can happen if:
719 // - both types are interfaces. TODO(calin): implement
720 // - one is an interface, the other a class, and the type does not implement the interface
721 // e.g:
722 // void foo(Interface i, boolean cond) {
723 // Object o = cond ? i : new Object();
724 // }
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000725 result_type_handle = handle_cache_.GetObjectClassHandle();
Calin Juravle2e768302015-07-28 14:41:11 +0000726 is_exact = false;
727 }
728
Calin Juravle52503d82015-11-11 16:58:31 +0000729 return ReferenceTypeInfo::Create(result_type_handle, is_exact);
Calin Juravle2e768302015-07-28 14:41:11 +0000730}
731
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000732void ReferenceTypePropagation::UpdateArrayGet(HArrayGet* instr, HandleCache* handle_cache) {
Calin Juravle2e768302015-07-28 14:41:11 +0000733 DCHECK_EQ(Primitive::kPrimNot, instr->GetType());
734
735 ReferenceTypeInfo parent_rti = instr->InputAt(0)->GetReferenceTypeInfo();
David Brazdilfe860702015-12-02 09:06:57 +0000736 if (!parent_rti.IsValid()) {
737 return;
738 }
Calin Juravle2e768302015-07-28 14:41:11 +0000739
740 Handle<mirror::Class> handle = parent_rti.GetTypeHandle();
Aart Bikf417ff42016-04-25 12:51:37 -0700741 if (handle->IsObjectArrayClass() && IsAdmissible(handle->GetComponentType())) {
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000742 ReferenceTypeInfo::TypeHandle component_handle =
743 handle_cache->NewHandle(handle->GetComponentType());
Nicolas Geoffray18401b72016-03-11 13:35:51 +0000744 bool is_exact = component_handle->CannotBeAssignedFromOtherTypes();
745 instr->SetReferenceTypeInfo(ReferenceTypeInfo::Create(component_handle, is_exact));
Calin Juravle2e768302015-07-28 14:41:11 +0000746 } else {
747 // We don't know what the parent actually is, so we fallback to object.
Nicolas Geoffray18401b72016-03-11 13:35:51 +0000748 instr->SetReferenceTypeInfo(instr->GetBlock()->GetGraph()->GetInexactObjectRti());
Calin Juravle2e768302015-07-28 14:41:11 +0000749 }
Calin Juravleb1498f62015-02-16 13:13:29 +0000750}
751
752bool ReferenceTypePropagation::UpdateReferenceTypeInfo(HInstruction* instr) {
753 ScopedObjectAccess soa(Thread::Current());
754
755 ReferenceTypeInfo previous_rti = instr->GetReferenceTypeInfo();
756 if (instr->IsBoundType()) {
757 UpdateBoundType(instr->AsBoundType());
758 } else if (instr->IsPhi()) {
759 UpdatePhi(instr->AsPhi());
Calin Juravle2e768302015-07-28 14:41:11 +0000760 } else if (instr->IsNullCheck()) {
761 ReferenceTypeInfo parent_rti = instr->InputAt(0)->GetReferenceTypeInfo();
762 if (parent_rti.IsValid()) {
763 instr->SetReferenceTypeInfo(parent_rti);
764 }
765 } else if (instr->IsArrayGet()) {
David Brazdilfe860702015-12-02 09:06:57 +0000766 // TODO: consider if it's worth "looking back" and binding the input object
Calin Juravle2e768302015-07-28 14:41:11 +0000767 // to an array type.
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000768 UpdateArrayGet(instr->AsArrayGet(), &handle_cache_);
Calin Juravleb1498f62015-02-16 13:13:29 +0000769 } else {
770 LOG(FATAL) << "Invalid instruction (should not get here)";
771 }
772
773 return !previous_rti.IsEqual(instr->GetReferenceTypeInfo());
774}
775
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000776void ReferenceTypePropagation::RTPVisitor::VisitInvoke(HInvoke* instr) {
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +0100777 if (instr->GetType() != Primitive::kPrimNot) {
778 return;
779 }
780
781 ScopedObjectAccess soa(Thread::Current());
782 ClassLinker* cl = Runtime::Current()->GetClassLinker();
Vladimir Marko456307a2016-04-19 14:12:13 +0000783 mirror::DexCache* dex_cache =
784 FindDexCacheWithHint(soa.Self(), instr->GetDexFile(), hint_dex_cache_);
Vladimir Marko05792b92015-08-03 11:56:49 +0100785 size_t pointer_size = cl->GetImagePointerSize();
786 ArtMethod* method = dex_cache->GetResolvedMethod(instr->GetDexMethodIndex(), pointer_size);
787 mirror::Class* klass = (method == nullptr) ? nullptr : method->GetReturnType(false, pointer_size);
Calin Juravle2e768302015-07-28 14:41:11 +0000788 SetClassAsTypeInfo(instr, klass, /* is_exact */ false);
Guillaume "Vermeille" Sanchezae09d2d2015-05-29 10:52:55 +0100789}
790
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000791void ReferenceTypePropagation::RTPVisitor::VisitArrayGet(HArrayGet* instr) {
Guillaume "Vermeille" Sanchez72a5eb52015-06-02 17:39:45 +0100792 if (instr->GetType() != Primitive::kPrimNot) {
793 return;
794 }
David Brazdilfe860702015-12-02 09:06:57 +0000795
Guillaume "Vermeille" Sanchez72a5eb52015-06-02 17:39:45 +0100796 ScopedObjectAccess soa(Thread::Current());
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000797 UpdateArrayGet(instr, handle_cache_);
Calin Juravle2e768302015-07-28 14:41:11 +0000798 if (!instr->GetReferenceTypeInfo().IsValid()) {
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100799 worklist_->push_back(instr);
Guillaume "Vermeille" Sanchez72a5eb52015-06-02 17:39:45 +0100800 }
801}
802
Calin Juravleb1498f62015-02-16 13:13:29 +0000803void ReferenceTypePropagation::UpdateBoundType(HBoundType* instr) {
804 ReferenceTypeInfo new_rti = instr->InputAt(0)->GetReferenceTypeInfo();
Calin Juravle2e768302015-07-28 14:41:11 +0000805 if (!new_rti.IsValid()) {
806 return; // No new info yet.
807 }
808
809 // Make sure that we don't go over the bounded type.
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000810 ReferenceTypeInfo upper_bound_rti = instr->GetUpperBound();
811 if (!upper_bound_rti.IsSupertypeOf(new_rti)) {
812 new_rti = upper_bound_rti;
Calin Juravleb1498f62015-02-16 13:13:29 +0000813 }
814 instr->SetReferenceTypeInfo(new_rti);
815}
816
Calin Juravle617bd922015-11-11 14:59:46 +0000817// NullConstant inputs are ignored during merging as they do not provide any useful information.
818// If all the inputs are NullConstants then the type of the phi will be set to Object.
Calin Juravleb1498f62015-02-16 13:13:29 +0000819void ReferenceTypePropagation::UpdatePhi(HPhi* instr) {
David Brazdild9510df2015-11-04 23:30:22 +0000820 DCHECK(instr->IsLive());
821
Vladimir Marko372f10e2016-05-17 16:30:10 +0100822 auto&& inputs = instr->GetInputs();
Calin Juravle617bd922015-11-11 14:59:46 +0000823 size_t first_input_index_not_null = 0;
Vladimir Marko372f10e2016-05-17 16:30:10 +0100824 while (first_input_index_not_null < inputs.size() &&
825 inputs[first_input_index_not_null]->IsNullConstant()) {
Calin Juravle617bd922015-11-11 14:59:46 +0000826 first_input_index_not_null++;
827 }
Vladimir Marko372f10e2016-05-17 16:30:10 +0100828 if (first_input_index_not_null == inputs.size()) {
Calin Juravle617bd922015-11-11 14:59:46 +0000829 // All inputs are NullConstants, set the type to object.
830 // This may happen in the presence of inlining.
Nicolas Geoffray18401b72016-03-11 13:35:51 +0000831 instr->SetReferenceTypeInfo(instr->GetBlock()->GetGraph()->GetInexactObjectRti());
Calin Juravle617bd922015-11-11 14:59:46 +0000832 return;
833 }
834
835 ReferenceTypeInfo new_rti = instr->InputAt(first_input_index_not_null)->GetReferenceTypeInfo();
836
Calin Juravle2e768302015-07-28 14:41:11 +0000837 if (new_rti.IsValid() && new_rti.IsObjectClass() && !new_rti.IsExact()) {
838 // Early return if we are Object and inexact.
Calin Juravleb1498f62015-02-16 13:13:29 +0000839 instr->SetReferenceTypeInfo(new_rti);
840 return;
841 }
Calin Juravle617bd922015-11-11 14:59:46 +0000842
Vladimir Marko372f10e2016-05-17 16:30:10 +0100843 for (size_t i = first_input_index_not_null + 1; i < inputs.size(); i++) {
844 if (inputs[i]->IsNullConstant()) {
Calin Juravle617bd922015-11-11 14:59:46 +0000845 continue;
846 }
Vladimir Marko372f10e2016-05-17 16:30:10 +0100847 new_rti = MergeTypes(new_rti, inputs[i]->GetReferenceTypeInfo());
Calin Juravle2e768302015-07-28 14:41:11 +0000848 if (new_rti.IsValid() && new_rti.IsObjectClass()) {
Calin Juravleb1498f62015-02-16 13:13:29 +0000849 if (!new_rti.IsExact()) {
850 break;
851 } else {
852 continue;
Nicolas Geoffrayd6138ef2015-02-18 14:48:53 +0000853 }
Calin Juravle10e244f2015-01-26 18:54:32 +0000854 }
855 }
David Brazdilfe860702015-12-02 09:06:57 +0000856
857 if (new_rti.IsValid()) {
858 instr->SetReferenceTypeInfo(new_rti);
859 }
Calin Juravleb1498f62015-02-16 13:13:29 +0000860}
861
862// Re-computes and updates the nullability of the instruction. Returns whether or
863// not the nullability was changed.
864bool ReferenceTypePropagation::UpdateNullability(HInstruction* instr) {
David Brazdild9510df2015-11-04 23:30:22 +0000865 DCHECK((instr->IsPhi() && instr->AsPhi()->IsLive())
Calin Juravle2e768302015-07-28 14:41:11 +0000866 || instr->IsBoundType()
867 || instr->IsNullCheck()
868 || instr->IsArrayGet());
869
870 if (!instr->IsPhi() && !instr->IsBoundType()) {
871 return false;
872 }
Calin Juravleb1498f62015-02-16 13:13:29 +0000873
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000874 bool existing_can_be_null = instr->CanBeNull();
875 if (instr->IsPhi()) {
876 HPhi* phi = instr->AsPhi();
877 bool new_can_be_null = false;
Vladimir Marko372f10e2016-05-17 16:30:10 +0100878 for (HInstruction* input : phi->GetInputs()) {
879 if (input->CanBeNull()) {
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000880 new_can_be_null = true;
881 break;
882 }
883 }
884 phi->SetCanBeNull(new_can_be_null);
885 } else if (instr->IsBoundType()) {
886 HBoundType* bound_type = instr->AsBoundType();
887 bound_type->SetCanBeNull(instr->InputAt(0)->CanBeNull() && bound_type->GetUpperCanBeNull());
Calin Juravleb1498f62015-02-16 13:13:29 +0000888 }
Calin Juravlea5ae3c32015-07-28 14:40:50 +0000889 return existing_can_be_null != instr->CanBeNull();
Calin Juravle10e244f2015-01-26 18:54:32 +0000890}
891
892void ReferenceTypePropagation::ProcessWorklist() {
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100893 while (!worklist_.empty()) {
894 HInstruction* instruction = worklist_.back();
895 worklist_.pop_back();
Calin Juravle12617592015-10-16 16:28:46 +0100896 bool updated_nullability = UpdateNullability(instruction);
897 bool updated_reference_type = UpdateReferenceTypeInfo(instruction);
898 if (updated_nullability || updated_reference_type) {
Calin Juravle10e244f2015-01-26 18:54:32 +0000899 AddDependentInstructionsToWorklist(instruction);
900 }
901 }
902}
903
Calin Juravleb1498f62015-02-16 13:13:29 +0000904void ReferenceTypePropagation::AddToWorklist(HInstruction* instruction) {
Calin Juravle2e768302015-07-28 14:41:11 +0000905 DCHECK_EQ(instruction->GetType(), Primitive::kPrimNot)
906 << instruction->DebugName() << ":" << instruction->GetType();
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100907 worklist_.push_back(instruction);
Calin Juravle10e244f2015-01-26 18:54:32 +0000908}
909
Calin Juravleb1498f62015-02-16 13:13:29 +0000910void ReferenceTypePropagation::AddDependentInstructionsToWorklist(HInstruction* instruction) {
Vladimir Marko46817b82016-03-29 12:21:58 +0100911 for (const HUseListNode<HInstruction*>& use : instruction->GetUses()) {
912 HInstruction* user = use.GetUser();
David Brazdild9510df2015-11-04 23:30:22 +0000913 if ((user->IsPhi() && user->AsPhi()->IsLive())
Calin Juravle2e768302015-07-28 14:41:11 +0000914 || user->IsBoundType()
915 || user->IsNullCheck()
916 || (user->IsArrayGet() && (user->GetType() == Primitive::kPrimNot))) {
Calin Juravlebeba9302015-07-08 15:57:18 +0000917 AddToWorklist(user);
Calin Juravle10e244f2015-01-26 18:54:32 +0000918 }
919 }
920}
Vladimir Marko7d1fbf32016-01-26 15:01:12 +0000921
Calin Juravle10e244f2015-01-26 18:54:32 +0000922} // namespace art