blob: 4dd0d26b8999dca4bdac69b9ecb6797a392485d0 [file] [log] [blame]
Nicolas Geoffray818f2102014-02-18 16:43:35 +00001/*
Nicolas Geoffray818f2102014-02-18 16:43:35 +00002 * Copyright (C) 2014 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
Nicolas Geoffraye5038322014-07-04 09:41:32 +010017#include "builder.h"
18
Mathieu Chartierc7853442015-03-27 14:35:38 -070019#include "art_field-inl.h"
David Srbecky0cf44932015-12-09 14:09:59 +000020#include "base/arena_bit_vector.h"
21#include "base/bit_vector-inl.h"
Andreas Gamped881df52014-11-24 23:28:39 -080022#include "base/logging.h"
Nicolas Geoffraye5038322014-07-04 09:41:32 +010023#include "class_linker.h"
Jeff Hao848f70a2014-01-15 13:49:50 -080024#include "dex/verified_method.h"
Nicolas Geoffray8ccc3f52014-03-19 10:34:11 +000025#include "dex_file-inl.h"
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +000026#include "dex_instruction-inl.h"
Roland Levillain4c0eb422015-04-24 16:43:49 +010027#include "dex/verified_method.h"
Nicolas Geoffraye5038322014-07-04 09:41:32 +010028#include "driver/compiler_driver-inl.h"
Vladimir Marko20f85592015-03-19 10:07:02 +000029#include "driver/compiler_options.h"
Nicolas Geoffraye5038322014-07-04 09:41:32 +010030#include "mirror/class_loader.h"
31#include "mirror/dex_cache.h"
Nicolas Geoffray818f2102014-02-18 16:43:35 +000032#include "nodes.h"
Nicolas Geoffray8ccc3f52014-03-19 10:34:11 +000033#include "primitive.h"
Nicolas Geoffraye5038322014-07-04 09:41:32 +010034#include "scoped_thread_state_change.h"
35#include "thread.h"
Vladimir Marko58155012015-08-19 12:49:41 +000036#include "utils/dex_cache_arrays_layout-inl.h"
Nicolas Geoffray818f2102014-02-18 16:43:35 +000037
38namespace art {
39
Nicolas Geoffrayf12feb82014-07-17 18:32:41 +010040/**
41 * Helper class to add HTemporary instructions. This class is used when
42 * converting a DEX instruction to multiple HInstruction, and where those
43 * instructions do not die at the following instruction, but instead spans
44 * multiple instructions.
45 */
46class Temporaries : public ValueObject {
47 public:
Calin Juravlef97f9fb2014-11-11 15:38:19 +000048 explicit Temporaries(HGraph* graph) : graph_(graph), index_(0) {}
Nicolas Geoffrayf12feb82014-07-17 18:32:41 +010049
50 void Add(HInstruction* instruction) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +060051 HInstruction* temp = new (graph_->GetArena()) HTemporary(index_, instruction->GetDexPc());
Nicolas Geoffrayf12feb82014-07-17 18:32:41 +010052 instruction->GetBlock()->AddInstruction(temp);
Calin Juravlef97f9fb2014-11-11 15:38:19 +000053
Nicolas Geoffrayf12feb82014-07-17 18:32:41 +010054 DCHECK(temp->GetPrevious() == instruction);
Calin Juravlef97f9fb2014-11-11 15:38:19 +000055
56 size_t offset;
57 if (instruction->GetType() == Primitive::kPrimLong
58 || instruction->GetType() == Primitive::kPrimDouble) {
59 offset = 2;
60 } else {
61 offset = 1;
62 }
63 index_ += offset;
64
65 graph_->UpdateTemporariesVRegSlots(index_);
Nicolas Geoffrayf12feb82014-07-17 18:32:41 +010066 }
67
68 private:
69 HGraph* const graph_;
70
Nicolas Geoffrayf12feb82014-07-17 18:32:41 +010071 // Current index in the temporary stack, updated by `Add`.
72 size_t index_;
73};
74
Andreas Gamped881df52014-11-24 23:28:39 -080075class SwitchTable : public ValueObject {
76 public:
77 SwitchTable(const Instruction& instruction, uint32_t dex_pc, bool sparse)
78 : instruction_(instruction), dex_pc_(dex_pc), sparse_(sparse) {
79 int32_t table_offset = instruction.VRegB_31t();
80 const uint16_t* table = reinterpret_cast<const uint16_t*>(&instruction) + table_offset;
81 if (sparse) {
82 CHECK_EQ(table[0], static_cast<uint16_t>(Instruction::kSparseSwitchSignature));
83 } else {
84 CHECK_EQ(table[0], static_cast<uint16_t>(Instruction::kPackedSwitchSignature));
85 }
86 num_entries_ = table[1];
87 values_ = reinterpret_cast<const int32_t*>(&table[2]);
88 }
89
90 uint16_t GetNumEntries() const {
91 return num_entries_;
92 }
93
Andreas Gampee4d4d322014-12-04 09:09:57 -080094 void CheckIndex(size_t index) const {
95 if (sparse_) {
96 // In a sparse table, we have num_entries_ keys and num_entries_ values, in that order.
97 DCHECK_LT(index, 2 * static_cast<size_t>(num_entries_));
98 } else {
99 // In a packed table, we have the starting key and num_entries_ values.
100 DCHECK_LT(index, 1 + static_cast<size_t>(num_entries_));
101 }
102 }
103
Andreas Gamped881df52014-11-24 23:28:39 -0800104 int32_t GetEntryAt(size_t index) const {
Andreas Gampee4d4d322014-12-04 09:09:57 -0800105 CheckIndex(index);
Andreas Gamped881df52014-11-24 23:28:39 -0800106 return values_[index];
107 }
108
109 uint32_t GetDexPcForIndex(size_t index) const {
Andreas Gampee4d4d322014-12-04 09:09:57 -0800110 CheckIndex(index);
Andreas Gamped881df52014-11-24 23:28:39 -0800111 return dex_pc_ +
112 (reinterpret_cast<const int16_t*>(values_ + index) -
113 reinterpret_cast<const int16_t*>(&instruction_));
114 }
115
Andreas Gampee4d4d322014-12-04 09:09:57 -0800116 // Index of the first value in the table.
117 size_t GetFirstValueIndex() const {
118 if (sparse_) {
119 // In a sparse table, we have num_entries_ keys and num_entries_ values, in that order.
120 return num_entries_;
121 } else {
122 // In a packed table, we have the starting key and num_entries_ values.
123 return 1;
124 }
125 }
126
Andreas Gamped881df52014-11-24 23:28:39 -0800127 private:
128 const Instruction& instruction_;
129 const uint32_t dex_pc_;
130
131 // Whether this is a sparse-switch table (or a packed-switch one).
132 const bool sparse_;
133
134 // This can't be const as it needs to be computed off of the given instruction, and complicated
135 // expressions in the initializer list seemed very ugly.
136 uint16_t num_entries_;
137
138 const int32_t* values_;
139
140 DISALLOW_COPY_AND_ASSIGN(SwitchTable);
141};
142
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100143void HGraphBuilder::InitializeLocals(uint16_t count) {
144 graph_->SetNumberOfVRegs(count);
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100145 locals_.resize(count);
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +0000146 for (int i = 0; i < count; i++) {
147 HLocal* local = new (arena_) HLocal(i);
148 entry_block_->AddInstruction(local);
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100149 locals_[i] = local;
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +0000150 }
151}
152
Nicolas Geoffray52e832b2014-11-06 15:15:31 +0000153void HGraphBuilder::InitializeParameters(uint16_t number_of_parameters) {
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100154 // dex_compilation_unit_ is null only when unit testing.
155 if (dex_compilation_unit_ == nullptr) {
Nicolas Geoffray52e832b2014-11-06 15:15:31 +0000156 return;
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100157 }
158
159 graph_->SetNumberOfInVRegs(number_of_parameters);
160 const char* shorty = dex_compilation_unit_->GetShorty();
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100161 int locals_index = locals_.size() - number_of_parameters;
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100162 int parameter_index = 0;
163
Calin Juravlee6e3bea2015-10-14 13:53:10 +0000164 const DexFile::MethodId& referrer_method_id =
165 dex_file_->GetMethodId(dex_compilation_unit_->GetDexMethodIndex());
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100166 if (!dex_compilation_unit_->IsStatic()) {
167 // Add the implicit 'this' argument, not expressed in the signature.
Calin Juravlee6e3bea2015-10-14 13:53:10 +0000168 HParameterValue* parameter = new (arena_) HParameterValue(*dex_file_,
169 referrer_method_id.class_idx_,
170 parameter_index++,
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600171 Primitive::kPrimNot,
172 true);
Nicolas Geoffrayb55f8352014-04-07 15:26:35 +0100173 entry_block_->AddInstruction(parameter);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100174 HLocal* local = GetLocalAt(locals_index++);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600175 entry_block_->AddInstruction(new (arena_) HStoreLocal(local, parameter, local->GetDexPc()));
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100176 number_of_parameters--;
177 }
178
Calin Juravlee6e3bea2015-10-14 13:53:10 +0000179 const DexFile::ProtoId& proto = dex_file_->GetMethodPrototype(referrer_method_id);
180 const DexFile::TypeList* arg_types = dex_file_->GetProtoParameters(proto);
181 for (int i = 0, shorty_pos = 1; i < number_of_parameters; i++) {
182 HParameterValue* parameter = new (arena_) HParameterValue(
183 *dex_file_,
184 arg_types->GetTypeItem(shorty_pos - 1).type_idx_,
185 parameter_index++,
186 Primitive::GetType(shorty[shorty_pos]),
187 false);
188 ++shorty_pos;
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +0100189 entry_block_->AddInstruction(parameter);
190 HLocal* local = GetLocalAt(locals_index++);
191 // Store the parameter value in the local that the dex code will use
192 // to reference that parameter.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600193 entry_block_->AddInstruction(new (arena_) HStoreLocal(local, parameter, local->GetDexPc()));
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +0100194 bool is_wide = (parameter->GetType() == Primitive::kPrimLong)
195 || (parameter->GetType() == Primitive::kPrimDouble);
196 if (is_wide) {
197 i++;
198 locals_index++;
199 parameter_index++;
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100200 }
201 }
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100202}
203
Nicolas Geoffrayb55f8352014-04-07 15:26:35 +0100204template<typename T>
Calin Juravle225ff812014-11-13 16:46:39 +0000205void HGraphBuilder::If_22t(const Instruction& instruction, uint32_t dex_pc) {
Nicolas Geoffrayfbc695f2014-09-15 15:33:30 +0000206 int32_t target_offset = instruction.GetTargetOffset();
David Brazdil852eaff2015-02-02 15:23:05 +0000207 HBasicBlock* branch_target = FindBlockStartingAt(dex_pc + target_offset);
208 HBasicBlock* fallthrough_target = FindBlockStartingAt(dex_pc + instruction.SizeInCodeUnits());
209 DCHECK(branch_target != nullptr);
210 DCHECK(fallthrough_target != nullptr);
211 PotentiallyAddSuspendCheck(branch_target, dex_pc);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600212 HInstruction* first = LoadLocal(instruction.VRegA(), Primitive::kPrimInt, dex_pc);
213 HInstruction* second = LoadLocal(instruction.VRegB(), Primitive::kPrimInt, dex_pc);
214 T* comparison = new (arena_) T(first, second, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -0700215 current_block_->AddInstruction(comparison);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600216 HInstruction* ifinst = new (arena_) HIf(comparison, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -0700217 current_block_->AddInstruction(ifinst);
David Brazdil852eaff2015-02-02 15:23:05 +0000218 current_block_->AddSuccessor(branch_target);
219 current_block_->AddSuccessor(fallthrough_target);
Dave Allison20dfc792014-06-16 20:44:29 -0700220 current_block_ = nullptr;
221}
222
223template<typename T>
Calin Juravle225ff812014-11-13 16:46:39 +0000224void HGraphBuilder::If_21t(const Instruction& instruction, uint32_t dex_pc) {
Nicolas Geoffrayfbc695f2014-09-15 15:33:30 +0000225 int32_t target_offset = instruction.GetTargetOffset();
David Brazdil852eaff2015-02-02 15:23:05 +0000226 HBasicBlock* branch_target = FindBlockStartingAt(dex_pc + target_offset);
227 HBasicBlock* fallthrough_target = FindBlockStartingAt(dex_pc + instruction.SizeInCodeUnits());
228 DCHECK(branch_target != nullptr);
229 DCHECK(fallthrough_target != nullptr);
230 PotentiallyAddSuspendCheck(branch_target, dex_pc);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600231 HInstruction* value = LoadLocal(instruction.VRegA(), Primitive::kPrimInt, dex_pc);
232 T* comparison = new (arena_) T(value, graph_->GetIntConstant(0, dex_pc), dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -0700233 current_block_->AddInstruction(comparison);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600234 HInstruction* ifinst = new (arena_) HIf(comparison, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -0700235 current_block_->AddInstruction(ifinst);
David Brazdil852eaff2015-02-02 15:23:05 +0000236 current_block_->AddSuccessor(branch_target);
237 current_block_->AddSuccessor(fallthrough_target);
Nicolas Geoffrayb55f8352014-04-07 15:26:35 +0100238 current_block_ = nullptr;
239}
240
Calin Juravle48c2b032014-12-09 18:11:36 +0000241void HGraphBuilder::MaybeRecordStat(MethodCompilationStat compilation_stat) {
242 if (compilation_stats_ != nullptr) {
243 compilation_stats_->RecordStat(compilation_stat);
244 }
245}
246
David Brazdil1b498722015-03-31 11:37:18 +0100247bool HGraphBuilder::SkipCompilation(const DexFile::CodeItem& code_item,
Calin Juravle48c2b032014-12-09 18:11:36 +0000248 size_t number_of_branches) {
249 const CompilerOptions& compiler_options = compiler_driver_->GetCompilerOptions();
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000250 CompilerOptions::CompilerFilter compiler_filter = compiler_options.GetCompilerFilter();
251 if (compiler_filter == CompilerOptions::kEverything) {
252 return false;
253 }
254
David Brazdil1b498722015-03-31 11:37:18 +0100255 if (compiler_options.IsHugeMethod(code_item.insns_size_in_code_units_)) {
Calin Juravle48c2b032014-12-09 18:11:36 +0000256 VLOG(compiler) << "Skip compilation of huge method "
257 << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_)
David Brazdil1b498722015-03-31 11:37:18 +0100258 << ": " << code_item.insns_size_in_code_units_ << " code units";
Calin Juravle48c2b032014-12-09 18:11:36 +0000259 MaybeRecordStat(MethodCompilationStat::kNotCompiledHugeMethod);
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000260 return true;
261 }
262
263 // If it's large and contains no branches, it's likely to be machine generated initialization.
David Brazdil1b498722015-03-31 11:37:18 +0100264 if (compiler_options.IsLargeMethod(code_item.insns_size_in_code_units_)
265 && (number_of_branches == 0)) {
Calin Juravle48c2b032014-12-09 18:11:36 +0000266 VLOG(compiler) << "Skip compilation of large method with no branch "
267 << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_)
David Brazdil1b498722015-03-31 11:37:18 +0100268 << ": " << code_item.insns_size_in_code_units_ << " code units";
Calin Juravle48c2b032014-12-09 18:11:36 +0000269 MaybeRecordStat(MethodCompilationStat::kNotCompiledLargeMethodNoBranches);
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000270 return true;
271 }
272
273 return false;
274}
275
David Brazdilfc6a86a2015-06-26 10:33:45 +0000276void HGraphBuilder::CreateBlocksForTryCatch(const DexFile::CodeItem& code_item) {
277 if (code_item.tries_size_ == 0) {
278 return;
279 }
280
281 // Create branch targets at the start/end of the TryItem range. These are
282 // places where the program might fall through into/out of the a block and
283 // where TryBoundary instructions will be inserted later. Other edges which
284 // enter/exit the try blocks are a result of branches/switches.
285 for (size_t idx = 0; idx < code_item.tries_size_; ++idx) {
286 const DexFile::TryItem* try_item = DexFile::GetTryItems(code_item, idx);
287 uint32_t dex_pc_start = try_item->start_addr_;
288 uint32_t dex_pc_end = dex_pc_start + try_item->insn_count_;
289 FindOrCreateBlockStartingAt(dex_pc_start);
290 if (dex_pc_end < code_item.insns_size_in_code_units_) {
291 // TODO: Do not create block if the last instruction cannot fall through.
292 FindOrCreateBlockStartingAt(dex_pc_end);
293 } else {
294 // The TryItem spans until the very end of the CodeItem (or beyond if
295 // invalid) and therefore cannot have any code afterwards.
296 }
297 }
298
299 // Create branch targets for exception handlers.
300 const uint8_t* handlers_ptr = DexFile::GetCatchHandlerData(code_item, 0);
301 uint32_t handlers_size = DecodeUnsignedLeb128(&handlers_ptr);
302 for (uint32_t idx = 0; idx < handlers_size; ++idx) {
303 CatchHandlerIterator iterator(handlers_ptr);
304 for (; iterator.HasNext(); iterator.Next()) {
305 uint32_t address = iterator.GetHandlerAddress();
306 HBasicBlock* block = FindOrCreateBlockStartingAt(address);
David Brazdilec16f792015-08-19 15:04:01 +0100307 block->SetTryCatchInformation(
308 new (arena_) TryCatchInformation(iterator.GetHandlerTypeIndex(), *dex_file_));
David Brazdilfc6a86a2015-06-26 10:33:45 +0000309 }
310 handlers_ptr = iterator.EndDataPointer();
311 }
312}
313
David Brazdild7558da2015-09-22 13:04:14 +0100314// Returns the TryItem stored for `block` or nullptr if there is no info for it.
315static const DexFile::TryItem* GetTryItem(
316 HBasicBlock* block,
317 const ArenaSafeMap<uint32_t, const DexFile::TryItem*>& try_block_info) {
318 auto iterator = try_block_info.find(block->GetBlockId());
319 return (iterator == try_block_info.end()) ? nullptr : iterator->second;
320}
David Brazdil56e1acc2015-06-30 15:41:36 +0100321
David Brazdild7558da2015-09-22 13:04:14 +0100322void HGraphBuilder::LinkToCatchBlocks(HTryBoundary* try_boundary,
323 const DexFile::CodeItem& code_item,
324 const DexFile::TryItem* try_item) {
325 for (CatchHandlerIterator it(code_item, *try_item); it.HasNext(); it.Next()) {
David Brazdil56e1acc2015-06-30 15:41:36 +0100326 try_boundary->AddExceptionHandler(FindBlockStartingAt(it.GetHandlerAddress()));
327 }
328}
329
David Brazdilfc6a86a2015-06-26 10:33:45 +0000330void HGraphBuilder::InsertTryBoundaryBlocks(const DexFile::CodeItem& code_item) {
331 if (code_item.tries_size_ == 0) {
332 return;
333 }
334
David Brazdild7558da2015-09-22 13:04:14 +0100335 // Keep a map of all try blocks and their respective TryItems. We do not use
336 // the block's pointer but rather its id to ensure deterministic iteration.
337 ArenaSafeMap<uint32_t, const DexFile::TryItem*> try_block_info(
Vladimir Marko5233f932015-09-29 19:01:15 +0100338 std::less<uint32_t>(), arena_->Adapter(kArenaAllocGraphBuilder));
David Brazdilbff75032015-07-08 17:26:51 +0000339
David Brazdild7558da2015-09-22 13:04:14 +0100340 // Obtain TryItem information for blocks with throwing instructions, and split
341 // blocks which are both try & catch to simplify the graph.
342 // NOTE: We are appending new blocks inside the loop, so we need to use index
343 // because iterators can be invalidated. We remember the initial size to avoid
344 // iterating over the new blocks which cannot throw.
345 for (size_t i = 0, e = graph_->GetBlocks().size(); i < e; ++i) {
346 HBasicBlock* block = graph_->GetBlocks()[i];
David Brazdil72783ff2015-07-09 14:36:05 +0100347
David Brazdild7558da2015-09-22 13:04:14 +0100348 // Do not bother creating exceptional edges for try blocks which have no
349 // throwing instructions. In that case we simply assume that the block is
350 // not covered by a TryItem. This prevents us from creating a throw-catch
351 // loop for synchronized blocks.
352 if (block->HasThrowingInstructions()) {
353 // Try to find a TryItem covering the block.
354 DCHECK_NE(block->GetDexPc(), kNoDexPc) << "Block must have a dec_pc to find its TryItem.";
355 const int32_t try_item_idx = DexFile::FindTryItem(code_item, block->GetDexPc());
356 if (try_item_idx != -1) {
357 // Block throwing and in a TryItem. Store the try block information.
358 HBasicBlock* throwing_block = block;
359 if (block->IsCatchBlock()) {
360 // Simplify blocks which are both try and catch, otherwise we would
361 // need a strategy for splitting exceptional edges. We split the block
362 // after the move-exception (if present) and mark the first part not
363 // throwing. The normal-flow edge between them will be split later.
David Brazdil9bc43612015-11-05 21:25:24 +0000364 throwing_block = block->SplitCatchBlockAfterMoveException();
365 // Move-exception does not throw and the block has throwing insructions
366 // so it must have been possible to split it.
367 DCHECK(throwing_block != nullptr);
David Brazdil72783ff2015-07-09 14:36:05 +0100368 }
David Brazdild7558da2015-09-22 13:04:14 +0100369
370 try_block_info.Put(throwing_block->GetBlockId(),
371 DexFile::GetTryItems(code_item, try_item_idx));
David Brazdil72783ff2015-07-09 14:36:05 +0100372 }
David Brazdil72783ff2015-07-09 14:36:05 +0100373 }
David Brazdilbff75032015-07-08 17:26:51 +0000374 }
375
David Brazdild7558da2015-09-22 13:04:14 +0100376 // Do a pass over the try blocks and insert entering TryBoundaries where at
377 // least one predecessor is not covered by the same TryItem as the try block.
378 // We do not split each edge separately, but rather create one boundary block
379 // that all predecessors are relinked to. This preserves loop headers (b/23895756).
380 for (auto entry : try_block_info) {
Vladimir Markoec7802a2015-10-01 20:57:57 +0100381 HBasicBlock* try_block = graph_->GetBlocks()[entry.first];
David Brazdild7558da2015-09-22 13:04:14 +0100382 for (HBasicBlock* predecessor : try_block->GetPredecessors()) {
383 if (GetTryItem(predecessor, try_block_info) != entry.second) {
384 // Found a predecessor not covered by the same TryItem. Insert entering
385 // boundary block.
386 HTryBoundary* try_entry =
387 new (arena_) HTryBoundary(HTryBoundary::kEntry, try_block->GetDexPc());
388 try_block->CreateImmediateDominator()->AddInstruction(try_entry);
389 LinkToCatchBlocks(try_entry, code_item, entry.second);
390 break;
391 }
David Brazdil281a6322015-07-03 10:34:57 +0100392 }
David Brazdild7558da2015-09-22 13:04:14 +0100393 }
David Brazdil281a6322015-07-03 10:34:57 +0100394
David Brazdild7558da2015-09-22 13:04:14 +0100395 // Do a second pass over the try blocks and insert exit TryBoundaries where
396 // the successor is not in the same TryItem.
397 for (auto entry : try_block_info) {
Vladimir Markoec7802a2015-10-01 20:57:57 +0100398 HBasicBlock* try_block = graph_->GetBlocks()[entry.first];
David Brazdild7558da2015-09-22 13:04:14 +0100399 // NOTE: Do not use iterators because SplitEdge would invalidate them.
400 for (size_t i = 0, e = try_block->GetSuccessors().size(); i < e; ++i) {
Vladimir Markoec7802a2015-10-01 20:57:57 +0100401 HBasicBlock* successor = try_block->GetSuccessors()[i];
David Brazdil72783ff2015-07-09 14:36:05 +0100402
David Brazdild7558da2015-09-22 13:04:14 +0100403 // If the successor is a try block, all of its predecessors must be
404 // covered by the same TryItem. Otherwise the previous pass would have
405 // created a non-throwing boundary block.
406 if (GetTryItem(successor, try_block_info) != nullptr) {
407 DCHECK_EQ(entry.second, GetTryItem(successor, try_block_info));
David Brazdil72783ff2015-07-09 14:36:05 +0100408 continue;
David Brazdilfc6a86a2015-06-26 10:33:45 +0000409 }
David Brazdil281a6322015-07-03 10:34:57 +0100410
David Brazdild7558da2015-09-22 13:04:14 +0100411 // Preserve the invariant that Return(Void) always jumps to Exit by moving
412 // it outside the try block if necessary.
413 HInstruction* last_instruction = try_block->GetLastInstruction();
414 if (last_instruction->IsReturn() || last_instruction->IsReturnVoid()) {
415 DCHECK_EQ(successor, exit_block_);
416 successor = try_block->SplitBefore(last_instruction);
David Brazdil281a6322015-07-03 10:34:57 +0100417 }
David Brazdild7558da2015-09-22 13:04:14 +0100418
419 // Insert TryBoundary and link to catch blocks.
420 HTryBoundary* try_exit =
421 new (arena_) HTryBoundary(HTryBoundary::kExit, successor->GetDexPc());
422 graph_->SplitEdge(try_block, successor)->AddInstruction(try_exit);
423 LinkToCatchBlocks(try_exit, code_item, entry.second);
David Brazdil281a6322015-07-03 10:34:57 +0100424 }
David Brazdilfc6a86a2015-06-26 10:33:45 +0000425 }
426}
427
David Brazdil5e8b1372015-01-23 14:39:08 +0000428bool HGraphBuilder::BuildGraph(const DexFile::CodeItem& code_item) {
Vladimir Markofa6b93c2015-09-15 10:15:55 +0100429 DCHECK(graph_->GetBlocks().empty());
David Brazdil5e8b1372015-01-23 14:39:08 +0000430
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +0000431 const uint16_t* code_ptr = code_item.insns_;
432 const uint16_t* code_end = code_item.insns_ + code_item.insns_size_in_code_units_;
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +0100433 code_start_ = code_ptr;
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +0000434
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000435 // Setup the graph with the entry block and exit block.
Nicolas Geoffray3c049742014-09-24 18:10:46 +0100436 entry_block_ = new (arena_) HBasicBlock(graph_, 0);
Nicolas Geoffray818f2102014-02-18 16:43:35 +0000437 graph_->AddBlock(entry_block_);
Nicolas Geoffray3c049742014-09-24 18:10:46 +0100438 exit_block_ = new (arena_) HBasicBlock(graph_, kNoDexPc);
Nicolas Geoffray787c3072014-03-17 10:20:19 +0000439 graph_->SetEntryBlock(entry_block_);
440 graph_->SetExitBlock(exit_block_);
Nicolas Geoffray818f2102014-02-18 16:43:35 +0000441
David Brazdil77a48ae2015-09-15 12:34:04 +0000442 graph_->SetHasTryCatch(code_item.tries_size_ != 0);
443
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +0000444 InitializeLocals(code_item.registers_size_);
Nicolas Geoffraye53798a2014-12-01 10:31:54 +0000445 graph_->SetMaximumNumberOfOutVRegs(code_item.outs_size_);
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +0000446
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000447 // Compute the number of dex instructions, blocks, and branches. We will
448 // check these values against limits given to the compiler.
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000449 size_t number_of_branches = 0;
450
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000451 // To avoid splitting blocks, we compute ahead of time the instructions that
452 // start a new block, and create these blocks.
Calin Juravle702d2602015-04-30 19:28:21 +0100453 if (!ComputeBranchTargets(code_ptr, code_end, &number_of_branches)) {
454 MaybeRecordStat(MethodCompilationStat::kNotCompiledBranchOutsideMethodCode);
455 return false;
456 }
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000457
458 // Note that the compiler driver is null when unit testing.
David Brazdil1b498722015-03-31 11:37:18 +0100459 if ((compiler_driver_ != nullptr) && SkipCompilation(code_item, number_of_branches)) {
David Brazdil5e8b1372015-01-23 14:39:08 +0000460 return false;
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000461 }
Nicolas Geoffray818f2102014-02-18 16:43:35 +0000462
David Srbecky0cf44932015-12-09 14:09:59 +0000463 // Find locations where we want to generate extra stackmaps for native debugging.
464 // This allows us to generate the info only at interesting points (for example,
465 // at start of java statement) rather than before every dex instruction.
466 const bool native_debuggable = compiler_driver_ != nullptr &&
467 compiler_driver_->GetCompilerOptions().GetNativeDebuggable();
468 ArenaBitVector* native_debug_info_locations;
469 if (native_debuggable) {
470 const uint32_t num_instructions = code_item.insns_size_in_code_units_;
471 native_debug_info_locations = new (arena_) ArenaBitVector (arena_, num_instructions, false);
472 native_debug_info_locations->ClearAllBits();
473 FindNativeDebugInfoLocations(code_item, native_debug_info_locations);
474 }
475
David Brazdilfc6a86a2015-06-26 10:33:45 +0000476 CreateBlocksForTryCatch(code_item);
Nicolas Geoffrayde58ab22014-11-05 12:46:03 +0000477
Nicolas Geoffray52e832b2014-11-06 15:15:31 +0000478 InitializeParameters(code_item.ins_size_);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100479
Calin Juravle225ff812014-11-13 16:46:39 +0000480 size_t dex_pc = 0;
Nicolas Geoffray818f2102014-02-18 16:43:35 +0000481 while (code_ptr < code_end) {
Calin Juravle225ff812014-11-13 16:46:39 +0000482 // Update the current block if dex_pc starts a new block.
483 MaybeUpdateCurrentBlock(dex_pc);
Nicolas Geoffray818f2102014-02-18 16:43:35 +0000484 const Instruction& instruction = *Instruction::At(code_ptr);
David Srbecky0cf44932015-12-09 14:09:59 +0000485 if (native_debuggable && native_debug_info_locations->IsBitSet(dex_pc)) {
486 if (current_block_ != nullptr) {
487 current_block_->AddInstruction(new (arena_) HNativeDebugInfo(dex_pc));
488 }
489 }
Calin Juravle48c2b032014-12-09 18:11:36 +0000490 if (!AnalyzeDexInstruction(instruction, dex_pc)) {
David Brazdil5e8b1372015-01-23 14:39:08 +0000491 return false;
Calin Juravle48c2b032014-12-09 18:11:36 +0000492 }
Calin Juravle225ff812014-11-13 16:46:39 +0000493 dex_pc += instruction.SizeInCodeUnits();
Nicolas Geoffray818f2102014-02-18 16:43:35 +0000494 code_ptr += instruction.SizeInCodeUnits();
495 }
496
David Brazdilfc6a86a2015-06-26 10:33:45 +0000497 // Add Exit to the exit block.
David Brazdil3e187382015-06-26 09:59:52 +0000498 exit_block_->AddInstruction(new (arena_) HExit());
Nicolas Geoffrayfbc695f2014-09-15 15:33:30 +0000499 // Add the suspend check to the entry block.
500 entry_block_->AddInstruction(new (arena_) HSuspendCheck(0));
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +0000501 entry_block_->AddInstruction(new (arena_) HGoto());
David Brazdilbff75032015-07-08 17:26:51 +0000502 // Add the exit block at the end.
503 graph_->AddBlock(exit_block_);
David Brazdil5e8b1372015-01-23 14:39:08 +0000504
David Brazdilfc6a86a2015-06-26 10:33:45 +0000505 // Iterate over blocks covered by TryItems and insert TryBoundaries at entry
506 // and exit points. This requires all control-flow instructions and
507 // non-exceptional edges to have been created.
508 InsertTryBoundaryBlocks(code_item);
509
David Brazdil5e8b1372015-01-23 14:39:08 +0000510 return true;
Nicolas Geoffray818f2102014-02-18 16:43:35 +0000511}
512
David Brazdilfc6a86a2015-06-26 10:33:45 +0000513void HGraphBuilder::MaybeUpdateCurrentBlock(size_t dex_pc) {
514 HBasicBlock* block = FindBlockStartingAt(dex_pc);
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +0000515 if (block == nullptr) {
516 return;
517 }
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000518
519 if (current_block_ != nullptr) {
520 // Branching instructions clear current_block, so we know
521 // the last instruction of the current block is not a branching
522 // instruction. We add an unconditional goto to the found block.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600523 current_block_->AddInstruction(new (arena_) HGoto(dex_pc));
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000524 current_block_->AddSuccessor(block);
525 }
526 graph_->AddBlock(block);
527 current_block_ = block;
528}
529
David Srbecky0cf44932015-12-09 14:09:59 +0000530void HGraphBuilder::FindNativeDebugInfoLocations(const DexFile::CodeItem& code_item,
531 ArenaBitVector* locations) {
532 // The callback gets called when the line number changes.
533 // In other words, it marks the start of new java statement.
534 struct Callback {
535 static bool Position(void* ctx, const DexFile::PositionInfo& entry) {
536 static_cast<ArenaBitVector*>(ctx)->SetBit(entry.address_);
537 return false;
538 }
539 };
540 dex_file_->DecodeDebugPositionInfo(&code_item, Callback::Position, locations);
541 // Add native debug info at the start of every basic block.
542 for (uint32_t pc = 0; pc < code_item.insns_size_in_code_units_; pc++) {
543 if (FindBlockStartingAt(pc) != nullptr) {
544 locations->SetBit(pc);
545 }
546 }
547 // Instruction-specific tweaks.
548 const Instruction* const begin = Instruction::At(code_item.insns_);
549 const Instruction* const end = begin->RelativeAt(code_item.insns_size_in_code_units_);
550 for (const Instruction* inst = begin; inst < end; inst = inst->Next()) {
551 switch (inst->Opcode()) {
552 case Instruction::MOVE_EXCEPTION:
553 case Instruction::MOVE_RESULT:
554 case Instruction::MOVE_RESULT_WIDE:
555 case Instruction::MOVE_RESULT_OBJECT: {
556 // The compiler checks that there are no instructions before those.
557 // So generate HNativeDebugInfo after them instead.
558 locations->ClearBit(inst->GetDexPc(code_item.insns_));
559 const Instruction* next = inst->Next();
560 if (next < end) {
561 locations->SetBit(next->GetDexPc(code_item.insns_));
562 }
563 break;
564 }
565 default:
566 break;
567 }
568 }
569}
570
Calin Juravle702d2602015-04-30 19:28:21 +0100571bool HGraphBuilder::ComputeBranchTargets(const uint16_t* code_ptr,
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000572 const uint16_t* code_end,
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000573 size_t* number_of_branches) {
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100574 branch_targets_.resize(code_end - code_ptr, nullptr);
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000575
576 // Create the first block for the dex instructions, single successor of the entry block.
Nicolas Geoffray3c049742014-09-24 18:10:46 +0100577 HBasicBlock* block = new (arena_) HBasicBlock(graph_, 0);
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100578 branch_targets_[0] = block;
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000579 entry_block_->AddSuccessor(block);
580
581 // Iterate over all instructions and find branching instructions. Create blocks for
582 // the locations these instructions branch to.
Andreas Gamped881df52014-11-24 23:28:39 -0800583 uint32_t dex_pc = 0;
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000584 while (code_ptr < code_end) {
585 const Instruction& instruction = *Instruction::At(code_ptr);
586 if (instruction.IsBranch()) {
Nicolas Geoffray43a539f2014-12-02 10:19:51 +0000587 (*number_of_branches)++;
Calin Juravle225ff812014-11-13 16:46:39 +0000588 int32_t target = instruction.GetTargetOffset() + dex_pc;
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000589 // Create a block for the target instruction.
David Brazdilfc6a86a2015-06-26 10:33:45 +0000590 FindOrCreateBlockStartingAt(target);
591
Calin Juravle225ff812014-11-13 16:46:39 +0000592 dex_pc += instruction.SizeInCodeUnits();
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000593 code_ptr += instruction.SizeInCodeUnits();
Calin Juravle702d2602015-04-30 19:28:21 +0100594
David Brazdilfe659462015-06-24 14:23:56 +0100595 if (instruction.CanFlowThrough()) {
596 if (code_ptr >= code_end) {
Calin Juravle702d2602015-04-30 19:28:21 +0100597 // In the normal case we should never hit this but someone can artificially forge a dex
598 // file to fall-through out the method code. In this case we bail out compilation.
599 return false;
David Brazdilfc6a86a2015-06-26 10:33:45 +0000600 } else {
601 FindOrCreateBlockStartingAt(dex_pc);
Calin Juravle702d2602015-04-30 19:28:21 +0100602 }
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000603 }
Andreas Gampee4d4d322014-12-04 09:09:57 -0800604 } else if (instruction.IsSwitch()) {
605 SwitchTable table(instruction, dex_pc, instruction.Opcode() == Instruction::SPARSE_SWITCH);
Andreas Gamped881df52014-11-24 23:28:39 -0800606
607 uint16_t num_entries = table.GetNumEntries();
608
Andreas Gampee4d4d322014-12-04 09:09:57 -0800609 // In a packed-switch, the entry at index 0 is the starting key. In a sparse-switch, the
610 // entry at index 0 is the first key, and values are after *all* keys.
611 size_t offset = table.GetFirstValueIndex();
612
613 // Use a larger loop counter type to avoid overflow issues.
614 for (size_t i = 0; i < num_entries; ++i) {
Andreas Gamped881df52014-11-24 23:28:39 -0800615 // The target of the case.
Andreas Gampee4d4d322014-12-04 09:09:57 -0800616 uint32_t target = dex_pc + table.GetEntryAt(i + offset);
David Brazdilfc6a86a2015-06-26 10:33:45 +0000617 FindOrCreateBlockStartingAt(target);
Andreas Gamped881df52014-11-24 23:28:39 -0800618
David Brazdil281a6322015-07-03 10:34:57 +0100619 // Create a block for the switch-case logic. The block gets the dex_pc
620 // of the SWITCH instruction because it is part of its semantics.
621 block = new (arena_) HBasicBlock(graph_, dex_pc);
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100622 branch_targets_[table.GetDexPcForIndex(i)] = block;
Andreas Gamped881df52014-11-24 23:28:39 -0800623 }
624
625 // Fall-through. Add a block if there is more code afterwards.
626 dex_pc += instruction.SizeInCodeUnits();
627 code_ptr += instruction.SizeInCodeUnits();
Calin Juravle702d2602015-04-30 19:28:21 +0100628 if (code_ptr >= code_end) {
629 // In the normal case we should never hit this but someone can artificially forge a dex
630 // file to fall-through out the method code. In this case we bail out compilation.
631 // (A switch can fall-through so we don't need to check CanFlowThrough().)
632 return false;
David Brazdilfc6a86a2015-06-26 10:33:45 +0000633 } else {
634 FindOrCreateBlockStartingAt(dex_pc);
Andreas Gamped881df52014-11-24 23:28:39 -0800635 }
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000636 } else {
637 code_ptr += instruction.SizeInCodeUnits();
Calin Juravle225ff812014-11-13 16:46:39 +0000638 dex_pc += instruction.SizeInCodeUnits();
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000639 }
640 }
Calin Juravle702d2602015-04-30 19:28:21 +0100641 return true;
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000642}
643
David Brazdilfc6a86a2015-06-26 10:33:45 +0000644HBasicBlock* HGraphBuilder::FindBlockStartingAt(int32_t dex_pc) const {
645 DCHECK_GE(dex_pc, 0);
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100646 return branch_targets_[dex_pc];
David Brazdilfc6a86a2015-06-26 10:33:45 +0000647}
648
649HBasicBlock* HGraphBuilder::FindOrCreateBlockStartingAt(int32_t dex_pc) {
650 HBasicBlock* block = FindBlockStartingAt(dex_pc);
651 if (block == nullptr) {
652 block = new (arena_) HBasicBlock(graph_, dex_pc);
Vladimir Marko2aaa4b52015-09-17 17:03:26 +0100653 branch_targets_[dex_pc] = block;
David Brazdilfc6a86a2015-06-26 10:33:45 +0000654 }
655 return block;
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +0000656}
657
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100658template<typename T>
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600659void HGraphBuilder::Unop_12x(const Instruction& instruction,
660 Primitive::Type type,
661 uint32_t dex_pc) {
662 HInstruction* first = LoadLocal(instruction.VRegB(), type, dex_pc);
663 current_block_->AddInstruction(new (arena_) T(type, first, dex_pc));
664 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Roland Levillain88cb1752014-10-20 16:36:47 +0100665}
666
Roland Levillaindff1f282014-11-05 14:15:05 +0000667void HGraphBuilder::Conversion_12x(const Instruction& instruction,
668 Primitive::Type input_type,
Roland Levillain624279f2014-12-04 11:54:28 +0000669 Primitive::Type result_type,
670 uint32_t dex_pc) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600671 HInstruction* first = LoadLocal(instruction.VRegB(), input_type, dex_pc);
Roland Levillain624279f2014-12-04 11:54:28 +0000672 current_block_->AddInstruction(new (arena_) HTypeConversion(result_type, first, dex_pc));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600673 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100674}
675
676template<typename T>
Calin Juravled6fb6cf2014-11-11 19:07:44 +0000677void HGraphBuilder::Binop_23x(const Instruction& instruction,
678 Primitive::Type type,
679 uint32_t dex_pc) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600680 HInstruction* first = LoadLocal(instruction.VRegB(), type, dex_pc);
681 HInstruction* second = LoadLocal(instruction.VRegC(), type, dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +0000682 current_block_->AddInstruction(new (arena_) T(type, first, second, dex_pc));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600683 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +0000684}
685
686template<typename T>
Calin Juravle9aec02f2014-11-18 23:06:35 +0000687void HGraphBuilder::Binop_23x_shift(const Instruction& instruction,
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600688 Primitive::Type type,
689 uint32_t dex_pc) {
690 HInstruction* first = LoadLocal(instruction.VRegB(), type, dex_pc);
691 HInstruction* second = LoadLocal(instruction.VRegC(), Primitive::kPrimInt, dex_pc);
692 current_block_->AddInstruction(new (arena_) T(type, first, second, dex_pc));
693 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +0000694}
695
Calin Juravleddb7df22014-11-25 20:56:51 +0000696void HGraphBuilder::Binop_23x_cmp(const Instruction& instruction,
697 Primitive::Type type,
Mark Mendellc4701932015-04-10 13:18:51 -0400698 ComparisonBias bias,
Alexey Frunze4dda3372015-06-01 18:31:49 -0700699 uint32_t dex_pc) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600700 HInstruction* first = LoadLocal(instruction.VRegB(), type, dex_pc);
701 HInstruction* second = LoadLocal(instruction.VRegC(), type, dex_pc);
Alexey Frunze4dda3372015-06-01 18:31:49 -0700702 current_block_->AddInstruction(new (arena_) HCompare(type, first, second, bias, dex_pc));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600703 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Calin Juravleddb7df22014-11-25 20:56:51 +0000704}
705
Calin Juravle9aec02f2014-11-18 23:06:35 +0000706template<typename T>
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600707void HGraphBuilder::Binop_12x_shift(const Instruction& instruction, Primitive::Type type,
708 uint32_t dex_pc) {
709 HInstruction* first = LoadLocal(instruction.VRegA(), type, dex_pc);
710 HInstruction* second = LoadLocal(instruction.VRegB(), Primitive::kPrimInt, dex_pc);
711 current_block_->AddInstruction(new (arena_) T(type, first, second, dex_pc));
712 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +0000713}
714
715template<typename T>
Calin Juravled6fb6cf2014-11-11 19:07:44 +0000716void HGraphBuilder::Binop_12x(const Instruction& instruction,
717 Primitive::Type type,
718 uint32_t dex_pc) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600719 HInstruction* first = LoadLocal(instruction.VRegA(), type, dex_pc);
720 HInstruction* second = LoadLocal(instruction.VRegB(), type, dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +0000721 current_block_->AddInstruction(new (arena_) T(type, first, second, dex_pc));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600722 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +0000723}
724
725template<typename T>
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600726void HGraphBuilder::Binop_22s(const Instruction& instruction, bool reverse, uint32_t dex_pc) {
727 HInstruction* first = LoadLocal(instruction.VRegB(), Primitive::kPrimInt, dex_pc);
728 HInstruction* second = graph_->GetIntConstant(instruction.VRegC_22s(), dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100729 if (reverse) {
730 std::swap(first, second);
731 }
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600732 current_block_->AddInstruction(new (arena_) T(Primitive::kPrimInt, first, second, dex_pc));
733 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100734}
735
736template<typename T>
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600737void HGraphBuilder::Binop_22b(const Instruction& instruction, bool reverse, uint32_t dex_pc) {
738 HInstruction* first = LoadLocal(instruction.VRegB(), Primitive::kPrimInt, dex_pc);
739 HInstruction* second = graph_->GetIntConstant(instruction.VRegC_22b(), dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100740 if (reverse) {
741 std::swap(first, second);
742 }
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600743 current_block_->AddInstruction(new (arena_) T(Primitive::kPrimInt, first, second, dex_pc));
744 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +0100745}
746
Calin Juravle0c25d102015-04-20 14:49:09 +0100747static bool RequiresConstructorBarrier(const DexCompilationUnit* cu, const CompilerDriver& driver) {
Calin Juravle27df7582015-04-17 19:12:31 +0100748 Thread* self = Thread::Current();
Calin Juravle0c25d102015-04-20 14:49:09 +0100749 return cu->IsConstructor()
750 && driver.RequiresConstructorBarrier(self, cu->GetDexFile(), cu->GetClassDefIndex());
Calin Juravle27df7582015-04-17 19:12:31 +0100751}
752
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600753void HGraphBuilder::BuildReturn(const Instruction& instruction,
754 Primitive::Type type,
755 uint32_t dex_pc) {
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +0100756 if (type == Primitive::kPrimVoid) {
Calin Juravle3cd4fc82015-05-14 15:15:42 +0100757 if (graph_->ShouldGenerateConstructorBarrier()) {
758 // The compilation unit is null during testing.
759 if (dex_compilation_unit_ != nullptr) {
760 DCHECK(RequiresConstructorBarrier(dex_compilation_unit_, *compiler_driver_))
761 << "Inconsistent use of ShouldGenerateConstructorBarrier. Should not generate a barrier.";
762 }
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600763 current_block_->AddInstruction(new (arena_) HMemoryBarrier(kStoreStore, dex_pc));
Calin Juravle27df7582015-04-17 19:12:31 +0100764 }
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600765 current_block_->AddInstruction(new (arena_) HReturnVoid(dex_pc));
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +0100766 } else {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +0600767 HInstruction* value = LoadLocal(instruction.VRegA(), type, dex_pc);
768 current_block_->AddInstruction(new (arena_) HReturn(value, dex_pc));
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +0100769 }
770 current_block_->AddSuccessor(exit_block_);
771 current_block_ = nullptr;
772}
773
Calin Juravle68ad6492015-08-18 17:08:12 +0100774static InvokeType GetInvokeTypeFromOpCode(Instruction::Code opcode) {
775 switch (opcode) {
776 case Instruction::INVOKE_STATIC:
777 case Instruction::INVOKE_STATIC_RANGE:
778 return kStatic;
779 case Instruction::INVOKE_DIRECT:
780 case Instruction::INVOKE_DIRECT_RANGE:
781 return kDirect;
782 case Instruction::INVOKE_VIRTUAL:
783 case Instruction::INVOKE_VIRTUAL_QUICK:
784 case Instruction::INVOKE_VIRTUAL_RANGE:
785 case Instruction::INVOKE_VIRTUAL_RANGE_QUICK:
786 return kVirtual;
787 case Instruction::INVOKE_INTERFACE:
788 case Instruction::INVOKE_INTERFACE_RANGE:
789 return kInterface;
790 case Instruction::INVOKE_SUPER_RANGE:
791 case Instruction::INVOKE_SUPER:
792 return kSuper;
793 default:
794 LOG(FATAL) << "Unexpected invoke opcode: " << opcode;
795 UNREACHABLE();
Nicolas Geoffray2e7cd752015-07-10 11:38:52 +0100796 }
Calin Juravle68ad6492015-08-18 17:08:12 +0100797}
798
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000799ArtMethod* HGraphBuilder::ResolveMethod(uint16_t method_idx, InvokeType invoke_type) {
800 ScopedObjectAccess soa(Thread::Current());
801 StackHandleScope<2> hs(soa.Self());
802
803 ClassLinker* class_linker = dex_compilation_unit_->GetClassLinker();
804 Handle<mirror::ClassLoader> class_loader(hs.NewHandle(
805 soa.Decode<mirror::ClassLoader*>(dex_compilation_unit_->GetClassLoader())));
806 Handle<mirror::Class> compiling_class(hs.NewHandle(GetCompilingClass()));
807
Andreas Gampedae24142015-12-03 17:27:32 -0800808 ArtMethod* resolved_method = class_linker->ResolveMethod<ClassLinker::kForceICCECheck>(
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000809 *dex_compilation_unit_->GetDexFile(),
810 method_idx,
811 dex_compilation_unit_->GetDexCache(),
812 class_loader,
813 /* referrer */ nullptr,
814 invoke_type);
815
816 if (UNLIKELY(resolved_method == nullptr)) {
817 // Clean up any exception left by type resolution.
818 soa.Self()->ClearException();
819 return nullptr;
820 }
821
822 // Check access. The class linker has a fast path for looking into the dex cache
823 // and does not check the access if it hits it.
824 if (compiling_class.Get() == nullptr) {
825 if (!resolved_method->IsPublic()) {
826 return nullptr;
827 }
828 } else if (!compiling_class->CanAccessResolvedMethod(resolved_method->GetDeclaringClass(),
829 resolved_method,
830 dex_compilation_unit_->GetDexCache().Get(),
831 method_idx)) {
832 return nullptr;
833 }
834
835 // We have to special case the invoke-super case, as ClassLinker::ResolveMethod does not.
836 // We need to look at the referrer's super class vtable.
837 if (invoke_type == kSuper) {
838 if (compiling_class.Get() == nullptr) {
839 // Invoking a super method requires knowing the actual super class. If we did not resolve
840 // the compiling method's declaring class (which only happens for ahead of time compilation),
841 // bail out.
842 DCHECK(Runtime::Current()->IsAotCompiler());
843 return nullptr;
844 }
845 uint16_t vtable_index = resolved_method->GetMethodIndex();
846 ArtMethod* actual_method = compiling_class->GetSuperClass()->GetVTableEntry(
847 vtable_index, class_linker->GetImagePointerSize());
848 if (actual_method != resolved_method &&
Nicolas Geoffray41844e52015-12-10 15:06:15 +0000849 !IsSameDexFile(*actual_method->GetDexFile(), *dex_compilation_unit_->GetDexFile())) {
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000850 // TODO: The actual method could still be referenced in the current dex file, so we
851 // could try locating it.
852 // TODO: Remove the dex_file restriction.
853 return nullptr;
854 }
855 if (!actual_method->IsInvokable()) {
856 // Fail if the actual method cannot be invoked. Otherwise, the runtime resolution stub
857 // could resolve the callee to the wrong method.
858 return nullptr;
859 }
860 resolved_method = actual_method;
861 }
862
863 // Check for incompatible class changes. The class linker has a fast path for
864 // looking into the dex cache and does not check incompatible class changes if it hits it.
865 if (resolved_method->CheckIncompatibleClassChange(invoke_type)) {
866 return nullptr;
867 }
868
869 return resolved_method;
870}
871
Calin Juravle68ad6492015-08-18 17:08:12 +0100872bool HGraphBuilder::BuildInvoke(const Instruction& instruction,
873 uint32_t dex_pc,
874 uint32_t method_idx,
875 uint32_t number_of_vreg_arguments,
876 bool is_range,
877 uint32_t* args,
878 uint32_t register_index) {
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000879 InvokeType invoke_type = GetInvokeTypeFromOpCode(instruction.Opcode());
Calin Juravle68ad6492015-08-18 17:08:12 +0100880 const char* descriptor = dex_file_->GetMethodShorty(method_idx);
881 Primitive::Type return_type = Primitive::GetType(descriptor[0]);
882
883 // Remove the return type from the 'proto'.
884 size_t number_of_arguments = strlen(descriptor) - 1;
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000885 if (invoke_type != kStatic) { // instance call
Calin Juravle68ad6492015-08-18 17:08:12 +0100886 // One extra argument for 'this'.
887 number_of_arguments++;
888 }
889
890 MethodReference target_method(dex_file_, method_idx);
Calin Juravle68ad6492015-08-18 17:08:12 +0100891
Calin Juravle5d01db12015-08-25 15:02:42 +0100892 // Special handling for string init.
893 int32_t string_init_offset = 0;
894 bool is_string_init = compiler_driver_->IsStringInit(method_idx,
895 dex_file_,
896 &string_init_offset);
897 // Replace calls to String.<init> with StringFactory.
898 if (is_string_init) {
Vladimir Markodc151b22015-10-15 18:02:30 +0100899 HInvokeStaticOrDirect::DispatchInfo dispatch_info = {
900 HInvokeStaticOrDirect::MethodLoadKind::kStringInit,
901 HInvokeStaticOrDirect::CodePtrLocation::kCallArtMethod,
902 dchecked_integral_cast<uint64_t>(string_init_offset),
903 0U
904 };
Calin Juravle5d01db12015-08-25 15:02:42 +0100905 HInvoke* invoke = new (arena_) HInvokeStaticOrDirect(
906 arena_,
907 number_of_arguments - 1,
908 Primitive::kPrimNot /*return_type */,
909 dex_pc,
910 method_idx,
911 target_method,
912 dispatch_info,
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000913 invoke_type,
Calin Juravle5d01db12015-08-25 15:02:42 +0100914 kStatic /* optimized_invoke_type */,
915 HInvokeStaticOrDirect::ClinitCheckRequirement::kImplicit);
916 return HandleStringInit(invoke,
917 number_of_vreg_arguments,
918 args,
919 register_index,
920 is_range,
921 descriptor);
922 }
923
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000924 ArtMethod* resolved_method = ResolveMethod(method_idx, invoke_type);
925
926 if (resolved_method == nullptr) {
Calin Juravle175dc732015-08-25 15:42:32 +0100927 MaybeRecordStat(MethodCompilationStat::kUnresolvedMethod);
928 HInvoke* invoke = new (arena_) HInvokeUnresolved(arena_,
929 number_of_arguments,
930 return_type,
931 dex_pc,
932 method_idx,
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000933 invoke_type);
Calin Juravle175dc732015-08-25 15:42:32 +0100934 return HandleInvoke(invoke,
935 number_of_vreg_arguments,
936 args,
937 register_index,
938 is_range,
939 descriptor,
940 nullptr /* clinit_check */);
Calin Juravle68ad6492015-08-18 17:08:12 +0100941 }
942
Calin Juravle68ad6492015-08-18 17:08:12 +0100943 // Potential class initialization check, in the case of a static method call.
944 HClinitCheck* clinit_check = nullptr;
945 HInvoke* invoke = nullptr;
946
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000947 if (invoke_type == kDirect || invoke_type == kStatic || invoke_type == kSuper) {
Calin Juravle68ad6492015-08-18 17:08:12 +0100948 // By default, consider that the called method implicitly requires
949 // an initialization check of its declaring method.
950 HInvokeStaticOrDirect::ClinitCheckRequirement clinit_check_requirement
951 = HInvokeStaticOrDirect::ClinitCheckRequirement::kImplicit;
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000952 ScopedObjectAccess soa(Thread::Current());
953 if (invoke_type == kStatic) {
954 clinit_check = ProcessClinitCheckForInvoke(
955 dex_pc, resolved_method, method_idx, &clinit_check_requirement);
956 } else if (invoke_type == kSuper) {
957 if (IsSameDexFile(*resolved_method->GetDexFile(), *dex_compilation_unit_->GetDexFile())) {
958 // Update the target method to the one resolved. Note that this may be a no-op if
959 // we resolved to the method referenced by the instruction.
960 method_idx = resolved_method->GetDexMethodIndex();
961 target_method = MethodReference(dex_file_, method_idx);
962 }
Nicolas Geoffray2e7cd752015-07-10 11:38:52 +0100963 }
Calin Juravle68ad6492015-08-18 17:08:12 +0100964
Vladimir Markodc151b22015-10-15 18:02:30 +0100965 HInvokeStaticOrDirect::DispatchInfo dispatch_info = {
966 HInvokeStaticOrDirect::MethodLoadKind::kDexCacheViaMethod,
967 HInvokeStaticOrDirect::CodePtrLocation::kCallArtMethod,
968 0u,
969 0U
970 };
Calin Juravle68ad6492015-08-18 17:08:12 +0100971 invoke = new (arena_) HInvokeStaticOrDirect(arena_,
972 number_of_arguments,
973 return_type,
974 dex_pc,
975 method_idx,
976 target_method,
977 dispatch_info,
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000978 invoke_type,
979 invoke_type,
Calin Juravle68ad6492015-08-18 17:08:12 +0100980 clinit_check_requirement);
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000981 } else if (invoke_type == kVirtual) {
982 ScopedObjectAccess soa(Thread::Current()); // Needed for the method index
Calin Juravle68ad6492015-08-18 17:08:12 +0100983 invoke = new (arena_) HInvokeVirtual(arena_,
984 number_of_arguments,
985 return_type,
986 dex_pc,
987 method_idx,
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000988 resolved_method->GetMethodIndex());
Nicolas Geoffray2e7cd752015-07-10 11:38:52 +0100989 } else {
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000990 DCHECK_EQ(invoke_type, kInterface);
991 ScopedObjectAccess soa(Thread::Current()); // Needed for the method index
Calin Juravle68ad6492015-08-18 17:08:12 +0100992 invoke = new (arena_) HInvokeInterface(arena_,
993 number_of_arguments,
994 return_type,
995 dex_pc,
996 method_idx,
Nicolas Geoffraye5234232015-12-02 09:06:11 +0000997 resolved_method->GetDexMethodIndex());
Nicolas Geoffray2e7cd752015-07-10 11:38:52 +0100998 }
Calin Juravle68ad6492015-08-18 17:08:12 +0100999
Calin Juravle5d01db12015-08-25 15:02:42 +01001000 return HandleInvoke(invoke,
1001 number_of_vreg_arguments,
1002 args,
1003 register_index,
1004 is_range,
1005 descriptor,
1006 clinit_check);
Calin Juravle68ad6492015-08-18 17:08:12 +01001007}
1008
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001009bool HGraphBuilder::BuildNewInstance(uint16_t type_index, uint32_t dex_pc) {
1010 bool finalizable;
1011 bool can_throw = NeedsAccessCheck(type_index, &finalizable);
1012
1013 // Only the non-resolved entrypoint handles the finalizable class case. If we
1014 // need access checks, then we haven't resolved the method and the class may
1015 // again be finalizable.
1016 QuickEntrypointEnum entrypoint = (finalizable || can_throw)
1017 ? kQuickAllocObject
1018 : kQuickAllocObjectInitialized;
1019
1020 ScopedObjectAccess soa(Thread::Current());
1021 StackHandleScope<3> hs(soa.Self());
1022 Handle<mirror::DexCache> dex_cache(hs.NewHandle(
1023 dex_compilation_unit_->GetClassLinker()->FindDexCache(
1024 soa.Self(), *dex_compilation_unit_->GetDexFile())));
1025 Handle<mirror::Class> resolved_class(hs.NewHandle(dex_cache->GetResolvedType(type_index)));
1026 const DexFile& outer_dex_file = *outer_compilation_unit_->GetDexFile();
1027 Handle<mirror::DexCache> outer_dex_cache(hs.NewHandle(
1028 outer_compilation_unit_->GetClassLinker()->FindDexCache(soa.Self(), outer_dex_file)));
1029
1030 if (outer_dex_cache.Get() != dex_cache.Get()) {
1031 // We currently do not support inlining allocations across dex files.
1032 return false;
1033 }
1034
1035 HLoadClass* load_class = new (arena_) HLoadClass(
1036 graph_->GetCurrentMethod(),
1037 type_index,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001038 outer_dex_file,
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001039 IsOutermostCompilingClass(type_index),
1040 dex_pc,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001041 /*needs_access_check*/ can_throw,
1042 compiler_driver_->CanAssumeTypeIsPresentInDexCache(outer_dex_file, type_index));
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001043
1044 current_block_->AddInstruction(load_class);
1045 HInstruction* cls = load_class;
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001046 if (!IsInitialized(resolved_class)) {
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001047 cls = new (arena_) HClinitCheck(load_class, dex_pc);
1048 current_block_->AddInstruction(cls);
1049 }
1050
1051 current_block_->AddInstruction(new (arena_) HNewInstance(
1052 cls,
1053 graph_->GetCurrentMethod(),
1054 dex_pc,
1055 type_index,
1056 *dex_compilation_unit_->GetDexFile(),
1057 can_throw,
1058 finalizable,
1059 entrypoint));
1060 return true;
1061}
1062
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001063static bool IsSubClass(mirror::Class* to_test, mirror::Class* super_class)
1064 SHARED_REQUIRES(Locks::mutator_lock_) {
1065 return to_test != nullptr && !to_test->IsInterface() && to_test->IsSubClass(super_class);
1066}
1067
1068bool HGraphBuilder::IsInitialized(Handle<mirror::Class> cls) const {
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001069 if (cls.Get() == nullptr) {
1070 return false;
1071 }
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001072
1073 // `CanAssumeClassIsLoaded` will return true if we're JITting, or will
1074 // check whether the class is in an image for the AOT compilation.
1075 if (cls->IsInitialized() &&
1076 compiler_driver_->CanAssumeClassIsLoaded(cls.Get())) {
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001077 return true;
1078 }
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001079
1080 if (IsSubClass(GetOutermostCompilingClass(), cls.Get())) {
1081 return true;
1082 }
1083
1084 // TODO: We should walk over the inlined methods, but we don't pass
1085 // that information to the builder.
1086 if (IsSubClass(GetCompilingClass(), cls.Get())) {
1087 return true;
1088 }
1089
1090 return false;
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001091}
1092
Calin Juravle68ad6492015-08-18 17:08:12 +01001093HClinitCheck* HGraphBuilder::ProcessClinitCheckForInvoke(
1094 uint32_t dex_pc,
Nicolas Geoffraye5234232015-12-02 09:06:11 +00001095 ArtMethod* resolved_method,
Calin Juravle68ad6492015-08-18 17:08:12 +01001096 uint32_t method_idx,
1097 HInvokeStaticOrDirect::ClinitCheckRequirement* clinit_check_requirement) {
Nicolas Geoffraye5234232015-12-02 09:06:11 +00001098 const DexFile& outer_dex_file = *outer_compilation_unit_->GetDexFile();
1099 Thread* self = Thread::Current();
1100 StackHandleScope<4> hs(self);
Calin Juravle68ad6492015-08-18 17:08:12 +01001101 Handle<mirror::DexCache> dex_cache(hs.NewHandle(
1102 dex_compilation_unit_->GetClassLinker()->FindDexCache(
Nicolas Geoffraye5234232015-12-02 09:06:11 +00001103 self, *dex_compilation_unit_->GetDexFile())));
Calin Juravle68ad6492015-08-18 17:08:12 +01001104 Handle<mirror::DexCache> outer_dex_cache(hs.NewHandle(
Nicolas Geoffraye5234232015-12-02 09:06:11 +00001105 outer_compilation_unit_->GetClassLinker()->FindDexCache(
1106 self, outer_dex_file)));
Calin Juravle68ad6492015-08-18 17:08:12 +01001107 Handle<mirror::Class> outer_class(hs.NewHandle(GetOutermostCompilingClass()));
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001108 Handle<mirror::Class> resolved_method_class(hs.NewHandle(resolved_method->GetDeclaringClass()));
Calin Juravle68ad6492015-08-18 17:08:12 +01001109
1110 // The index at which the method's class is stored in the DexCache's type array.
1111 uint32_t storage_index = DexFile::kDexNoIndex;
1112 bool is_outer_class = (resolved_method->GetDeclaringClass() == outer_class.Get());
1113 if (is_outer_class) {
1114 storage_index = outer_class->GetDexTypeIndex();
1115 } else if (outer_dex_cache.Get() == dex_cache.Get()) {
1116 // Get `storage_index` from IsClassOfStaticMethodAvailableToReferrer.
1117 compiler_driver_->IsClassOfStaticMethodAvailableToReferrer(outer_dex_cache.Get(),
1118 GetCompilingClass(),
1119 resolved_method,
1120 method_idx,
1121 &storage_index);
1122 }
1123
1124 HClinitCheck* clinit_check = nullptr;
1125
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001126 if (IsInitialized(resolved_method_class)) {
Calin Juravle68ad6492015-08-18 17:08:12 +01001127 *clinit_check_requirement = HInvokeStaticOrDirect::ClinitCheckRequirement::kNone;
1128 } else if (storage_index != DexFile::kDexNoIndex) {
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001129 *clinit_check_requirement = HInvokeStaticOrDirect::ClinitCheckRequirement::kExplicit;
1130 HLoadClass* load_class = new (arena_) HLoadClass(
1131 graph_->GetCurrentMethod(),
1132 storage_index,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001133 outer_dex_file,
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001134 is_outer_class,
1135 dex_pc,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001136 /*needs_access_check*/ false,
1137 compiler_driver_->CanAssumeTypeIsPresentInDexCache(outer_dex_file, storage_index));
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001138 current_block_->AddInstruction(load_class);
1139 clinit_check = new (arena_) HClinitCheck(load_class, dex_pc);
1140 current_block_->AddInstruction(clinit_check);
Calin Juravle68ad6492015-08-18 17:08:12 +01001141 }
1142 return clinit_check;
Nicolas Geoffray2e7cd752015-07-10 11:38:52 +01001143}
1144
Calin Juravle5d01db12015-08-25 15:02:42 +01001145bool HGraphBuilder::SetupInvokeArguments(HInvoke* invoke,
1146 uint32_t number_of_vreg_arguments,
1147 uint32_t* args,
1148 uint32_t register_index,
1149 bool is_range,
1150 const char* descriptor,
1151 size_t start_index,
1152 size_t* argument_index) {
Calin Juravle68ad6492015-08-18 17:08:12 +01001153 uint32_t descriptor_index = 1; // Skip the return type.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001154 uint32_t dex_pc = invoke->GetDexPc();
Calin Juravle68ad6492015-08-18 17:08:12 +01001155
Nicolas Geoffray2e335252015-06-18 11:11:27 +01001156 for (size_t i = start_index;
1157 // Make sure we don't go over the expected arguments or over the number of
1158 // dex registers given. If the instruction was seen as dead by the verifier,
1159 // it hasn't been properly checked.
Calin Juravle5d01db12015-08-25 15:02:42 +01001160 (i < number_of_vreg_arguments) && (*argument_index < invoke->GetNumberOfArguments());
1161 i++, (*argument_index)++) {
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01001162 Primitive::Type type = Primitive::GetType(descriptor[descriptor_index++]);
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01001163 bool is_wide = (type == Primitive::kPrimLong) || (type == Primitive::kPrimDouble);
Nicolas Geoffray2e335252015-06-18 11:11:27 +01001164 if (!is_range
1165 && is_wide
1166 && ((i + 1 == number_of_vreg_arguments) || (args[i] + 1 != args[i + 1]))) {
1167 // Longs and doubles should be in pairs, that is, sequential registers. The verifier should
1168 // reject any class where this is violated. However, the verifier only does these checks
1169 // on non trivially dead instructions, so we just bailout the compilation.
1170 VLOG(compiler) << "Did not compile "
1171 << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_)
1172 << " because of non-sequential dex register pair in wide argument";
1173 MaybeRecordStat(MethodCompilationStat::kNotCompiledMalformedOpcode);
1174 return false;
1175 }
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001176 HInstruction* arg = LoadLocal(is_range ? register_index + i : args[i], type, dex_pc);
Calin Juravle5d01db12015-08-25 15:02:42 +01001177 invoke->SetArgumentAt(*argument_index, arg);
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01001178 if (is_wide) {
Nicolas Geoffrayabed4d02014-07-14 15:24:11 +01001179 i++;
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01001180 }
1181 }
Nicolas Geoffray2e335252015-06-18 11:11:27 +01001182
Calin Juravle5d01db12015-08-25 15:02:42 +01001183 if (*argument_index != invoke->GetNumberOfArguments()) {
Nicolas Geoffray2e335252015-06-18 11:11:27 +01001184 VLOG(compiler) << "Did not compile "
1185 << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_)
1186 << " because of wrong number of arguments in invoke instruction";
1187 MaybeRecordStat(MethodCompilationStat::kNotCompiledMalformedOpcode);
1188 return false;
1189 }
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01001190
Vladimir Markob554b5a2015-11-06 12:57:55 +00001191 if (invoke->IsInvokeStaticOrDirect() &&
1192 HInvokeStaticOrDirect::NeedsCurrentMethodInput(
1193 invoke->AsInvokeStaticOrDirect()->GetMethodLoadKind())) {
Calin Juravle5d01db12015-08-25 15:02:42 +01001194 invoke->SetArgumentAt(*argument_index, graph_->GetCurrentMethod());
1195 (*argument_index)++;
1196 }
1197
1198 return true;
1199}
1200
1201bool HGraphBuilder::HandleInvoke(HInvoke* invoke,
1202 uint32_t number_of_vreg_arguments,
1203 uint32_t* args,
1204 uint32_t register_index,
1205 bool is_range,
1206 const char* descriptor,
1207 HClinitCheck* clinit_check) {
1208 DCHECK(!invoke->IsInvokeStaticOrDirect() || !invoke->AsInvokeStaticOrDirect()->IsStringInit());
1209
1210 size_t start_index = 0;
1211 size_t argument_index = 0;
1212 if (invoke->GetOriginalInvokeType() != InvokeType::kStatic) { // Instance call.
1213 Temporaries temps(graph_);
1214 HInstruction* arg = LoadLocal(
1215 is_range ? register_index : args[0], Primitive::kPrimNot, invoke->GetDexPc());
1216 HNullCheck* null_check = new (arena_) HNullCheck(arg, invoke->GetDexPc());
1217 current_block_->AddInstruction(null_check);
1218 temps.Add(null_check);
1219 invoke->SetArgumentAt(0, null_check);
1220 start_index = 1;
1221 argument_index = 1;
1222 }
1223
1224 if (!SetupInvokeArguments(invoke,
1225 number_of_vreg_arguments,
1226 args,
1227 register_index,
1228 is_range,
1229 descriptor,
1230 start_index,
1231 &argument_index)) {
1232 return false;
Nicolas Geoffray38207af2015-06-01 15:46:22 +01001233 }
1234
Calin Juravle68ad6492015-08-18 17:08:12 +01001235 if (clinit_check != nullptr) {
Roland Levillain4c0eb422015-04-24 16:43:49 +01001236 // Add the class initialization check as last input of `invoke`.
Calin Juravle68ad6492015-08-18 17:08:12 +01001237 DCHECK(invoke->IsInvokeStaticOrDirect());
1238 DCHECK(invoke->AsInvokeStaticOrDirect()->GetClinitCheckRequirement()
1239 == HInvokeStaticOrDirect::ClinitCheckRequirement::kExplicit);
Roland Levillain3e3d7332015-04-28 11:00:54 +01001240 invoke->SetArgumentAt(argument_index, clinit_check);
Nicolas Geoffray2e7cd752015-07-10 11:38:52 +01001241 argument_index++;
Roland Levillain4c0eb422015-04-24 16:43:49 +01001242 }
1243
Calin Juravle5d01db12015-08-25 15:02:42 +01001244 current_block_->AddInstruction(invoke);
1245 latest_result_ = invoke;
1246
1247 return true;
1248}
1249
1250bool HGraphBuilder::HandleStringInit(HInvoke* invoke,
1251 uint32_t number_of_vreg_arguments,
1252 uint32_t* args,
1253 uint32_t register_index,
1254 bool is_range,
1255 const char* descriptor) {
1256 DCHECK(invoke->IsInvokeStaticOrDirect());
1257 DCHECK(invoke->AsInvokeStaticOrDirect()->IsStringInit());
1258
1259 size_t start_index = 1;
1260 size_t argument_index = 0;
1261 if (!SetupInvokeArguments(invoke,
1262 number_of_vreg_arguments,
1263 args,
1264 register_index,
1265 is_range,
1266 descriptor,
1267 start_index,
1268 &argument_index)) {
1269 return false;
Jeff Hao848f70a2014-01-15 13:49:50 -08001270 }
Calin Juravle0eedd7e2015-08-20 14:48:00 +01001271
Calin Juravle5d01db12015-08-25 15:02:42 +01001272 // Add move-result for StringFactory method.
1273 uint32_t orig_this_reg = is_range ? register_index : args[0];
1274 HInstruction* fake_string = LoadLocal(orig_this_reg, Primitive::kPrimNot, invoke->GetDexPc());
1275 invoke->SetArgumentAt(argument_index, fake_string);
1276 current_block_->AddInstruction(invoke);
1277 PotentiallySimplifyFakeString(orig_this_reg, invoke->GetDexPc(), invoke);
1278
Calin Juravle0eedd7e2015-08-20 14:48:00 +01001279 latest_result_ = invoke;
1280
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01001281 return true;
1282}
1283
Calin Juravle68ad6492015-08-18 17:08:12 +01001284void HGraphBuilder::PotentiallySimplifyFakeString(uint16_t original_dex_register,
1285 uint32_t dex_pc,
1286 HInvoke* actual_string) {
1287 if (!graph_->IsDebuggable()) {
1288 // Notify that we cannot compile with baseline. The dex registers aliasing
1289 // with `original_dex_register` will be handled when we optimize
1290 // (see HInstructionSimplifer::VisitFakeString).
1291 can_use_baseline_for_string_init_ = false;
1292 return;
1293 }
1294 const VerifiedMethod* verified_method =
1295 compiler_driver_->GetVerifiedMethod(dex_file_, dex_compilation_unit_->GetDexMethodIndex());
1296 if (verified_method != nullptr) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001297 UpdateLocal(original_dex_register, actual_string, dex_pc);
Calin Juravle68ad6492015-08-18 17:08:12 +01001298 const SafeMap<uint32_t, std::set<uint32_t>>& string_init_map =
1299 verified_method->GetStringInitPcRegMap();
1300 auto map_it = string_init_map.find(dex_pc);
1301 if (map_it != string_init_map.end()) {
Vladimir Markodbc23372015-10-12 12:45:52 +01001302 for (uint32_t reg : map_it->second) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001303 HInstruction* load_local = LoadLocal(original_dex_register, Primitive::kPrimNot, dex_pc);
Vladimir Markodbc23372015-10-12 12:45:52 +01001304 UpdateLocal(reg, load_local, dex_pc);
Calin Juravle68ad6492015-08-18 17:08:12 +01001305 }
1306 }
1307 } else {
1308 can_use_baseline_for_string_init_ = false;
1309 }
1310}
1311
Calin Juravlee460d1d2015-09-29 04:52:17 +01001312static Primitive::Type GetFieldAccessType(const DexFile& dex_file, uint16_t field_index) {
1313 const DexFile::FieldId& field_id = dex_file.GetFieldId(field_index);
1314 const char* type = dex_file.GetFieldTypeDescriptor(field_id);
1315 return Primitive::GetType(type[0]);
1316}
1317
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001318bool HGraphBuilder::BuildInstanceFieldAccess(const Instruction& instruction,
Calin Juravle225ff812014-11-13 16:46:39 +00001319 uint32_t dex_pc,
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001320 bool is_put) {
Nicolas Geoffraye5038322014-07-04 09:41:32 +01001321 uint32_t source_or_dest_reg = instruction.VRegA_22c();
1322 uint32_t obj_reg = instruction.VRegB_22c();
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00001323 uint16_t field_index;
1324 if (instruction.IsQuickened()) {
1325 if (!CanDecodeQuickenedInfo()) {
1326 return false;
1327 }
1328 field_index = LookupQuickenedInfo(dex_pc);
1329 } else {
1330 field_index = instruction.VRegC_22c();
1331 }
Nicolas Geoffraye5038322014-07-04 09:41:32 +01001332
1333 ScopedObjectAccess soa(Thread::Current());
Mathieu Chartierc7853442015-03-27 14:35:38 -07001334 ArtField* resolved_field =
1335 compiler_driver_->ComputeInstanceFieldInfo(field_index, dex_compilation_unit_, is_put, soa);
Nicolas Geoffraye5038322014-07-04 09:41:32 +01001336
Nicolas Geoffrayabed4d02014-07-14 15:24:11 +01001337
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001338 HInstruction* object = LoadLocal(obj_reg, Primitive::kPrimNot, dex_pc);
Calin Juravlee460d1d2015-09-29 04:52:17 +01001339 HInstruction* null_check = new (arena_) HNullCheck(object, dex_pc);
1340 current_block_->AddInstruction(null_check);
1341
1342 Primitive::Type field_type = (resolved_field == nullptr)
1343 ? GetFieldAccessType(*dex_file_, field_index)
1344 : resolved_field->GetTypeAsPrimitiveType();
Nicolas Geoffraye5038322014-07-04 09:41:32 +01001345 if (is_put) {
Calin Juravlef97f9fb2014-11-11 15:38:19 +00001346 Temporaries temps(graph_);
Nicolas Geoffraye5038322014-07-04 09:41:32 +01001347 // We need one temporary for the null check.
1348 temps.Add(null_check);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001349 HInstruction* value = LoadLocal(source_or_dest_reg, field_type, dex_pc);
Calin Juravlee460d1d2015-09-29 04:52:17 +01001350 HInstruction* field_set = nullptr;
1351 if (resolved_field == nullptr) {
1352 MaybeRecordStat(MethodCompilationStat::kUnresolvedField);
1353 field_set = new (arena_) HUnresolvedInstanceFieldSet(null_check,
1354 value,
1355 field_type,
1356 field_index,
1357 dex_pc);
1358 } else {
Mingyao Yang8df69d42015-10-22 15:40:58 -07001359 uint16_t class_def_index = resolved_field->GetDeclaringClass()->GetDexClassDefIndex();
Calin Juravlee460d1d2015-09-29 04:52:17 +01001360 field_set = new (arena_) HInstanceFieldSet(null_check,
1361 value,
1362 field_type,
1363 resolved_field->GetOffset(),
1364 resolved_field->IsVolatile(),
1365 field_index,
Mingyao Yang8df69d42015-10-22 15:40:58 -07001366 class_def_index,
Calin Juravlee460d1d2015-09-29 04:52:17 +01001367 *dex_file_,
1368 dex_compilation_unit_->GetDexCache(),
1369 dex_pc);
1370 }
1371 current_block_->AddInstruction(field_set);
Nicolas Geoffraye5038322014-07-04 09:41:32 +01001372 } else {
Calin Juravlee460d1d2015-09-29 04:52:17 +01001373 HInstruction* field_get = nullptr;
1374 if (resolved_field == nullptr) {
1375 MaybeRecordStat(MethodCompilationStat::kUnresolvedField);
1376 field_get = new (arena_) HUnresolvedInstanceFieldGet(null_check,
1377 field_type,
1378 field_index,
1379 dex_pc);
1380 } else {
Mingyao Yang8df69d42015-10-22 15:40:58 -07001381 uint16_t class_def_index = resolved_field->GetDeclaringClass()->GetDexClassDefIndex();
Calin Juravlee460d1d2015-09-29 04:52:17 +01001382 field_get = new (arena_) HInstanceFieldGet(null_check,
1383 field_type,
1384 resolved_field->GetOffset(),
1385 resolved_field->IsVolatile(),
1386 field_index,
Mingyao Yang8df69d42015-10-22 15:40:58 -07001387 class_def_index,
Calin Juravlee460d1d2015-09-29 04:52:17 +01001388 *dex_file_,
1389 dex_compilation_unit_->GetDexCache(),
1390 dex_pc);
1391 }
1392 current_block_->AddInstruction(field_get);
1393 UpdateLocal(source_or_dest_reg, field_get, dex_pc);
Calin Juravlee6f49b42015-09-17 14:04:33 +00001394 }
Calin Juravlee460d1d2015-09-29 04:52:17 +01001395
Nicolas Geoffraye5038322014-07-04 09:41:32 +01001396 return true;
1397}
1398
Nicolas Geoffray30451742015-06-19 13:32:41 +01001399static mirror::Class* GetClassFrom(CompilerDriver* driver,
1400 const DexCompilationUnit& compilation_unit) {
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001401 ScopedObjectAccess soa(Thread::Current());
1402 StackHandleScope<2> hs(soa.Self());
Nicolas Geoffray30451742015-06-19 13:32:41 +01001403 const DexFile& dex_file = *compilation_unit.GetDexFile();
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001404 Handle<mirror::ClassLoader> class_loader(hs.NewHandle(
Nicolas Geoffray30451742015-06-19 13:32:41 +01001405 soa.Decode<mirror::ClassLoader*>(compilation_unit.GetClassLoader())));
1406 Handle<mirror::DexCache> dex_cache(hs.NewHandle(
Mathieu Chartier673ed3d2015-08-28 14:56:43 -07001407 compilation_unit.GetClassLinker()->FindDexCache(soa.Self(), dex_file)));
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001408
Nicolas Geoffray30451742015-06-19 13:32:41 +01001409 return driver->ResolveCompilingMethodsClass(soa, dex_cache, class_loader, &compilation_unit);
1410}
1411
1412mirror::Class* HGraphBuilder::GetOutermostCompilingClass() const {
1413 return GetClassFrom(compiler_driver_, *outer_compilation_unit_);
1414}
1415
1416mirror::Class* HGraphBuilder::GetCompilingClass() const {
1417 return GetClassFrom(compiler_driver_, *dex_compilation_unit_);
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001418}
1419
1420bool HGraphBuilder::IsOutermostCompilingClass(uint16_t type_index) const {
1421 ScopedObjectAccess soa(Thread::Current());
1422 StackHandleScope<4> hs(soa.Self());
1423 Handle<mirror::DexCache> dex_cache(hs.NewHandle(
Mathieu Chartier673ed3d2015-08-28 14:56:43 -07001424 dex_compilation_unit_->GetClassLinker()->FindDexCache(
1425 soa.Self(), *dex_compilation_unit_->GetDexFile())));
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001426 Handle<mirror::ClassLoader> class_loader(hs.NewHandle(
1427 soa.Decode<mirror::ClassLoader*>(dex_compilation_unit_->GetClassLoader())));
1428 Handle<mirror::Class> cls(hs.NewHandle(compiler_driver_->ResolveClass(
1429 soa, dex_cache, class_loader, type_index, dex_compilation_unit_)));
Nicolas Geoffrayafd06412015-06-20 22:44:47 +01001430 Handle<mirror::Class> outer_class(hs.NewHandle(GetOutermostCompilingClass()));
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001431
Calin Juravle4e2a5572015-10-07 18:55:43 +01001432 // GetOutermostCompilingClass returns null when the class is unresolved
1433 // (e.g. if it derives from an unresolved class). This is bogus knowing that
1434 // we are compiling it.
1435 // When this happens we cannot establish a direct relation between the current
1436 // class and the outer class, so we return false.
1437 // (Note that this is only used for optimizing invokes and field accesses)
1438 return (cls.Get() != nullptr) && (outer_class.Get() == cls.Get());
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001439}
1440
Calin Juravle07380a22015-09-17 14:15:12 +01001441void HGraphBuilder::BuildUnresolvedStaticFieldAccess(const Instruction& instruction,
1442 uint32_t dex_pc,
1443 bool is_put,
1444 Primitive::Type field_type) {
1445 uint32_t source_or_dest_reg = instruction.VRegA_21c();
1446 uint16_t field_index = instruction.VRegB_21c();
1447
1448 if (is_put) {
1449 HInstruction* value = LoadLocal(source_or_dest_reg, field_type, dex_pc);
1450 current_block_->AddInstruction(
1451 new (arena_) HUnresolvedStaticFieldSet(value, field_type, field_index, dex_pc));
1452 } else {
1453 current_block_->AddInstruction(
1454 new (arena_) HUnresolvedStaticFieldGet(field_type, field_index, dex_pc));
1455 UpdateLocal(source_or_dest_reg, current_block_->GetLastInstruction(), dex_pc);
1456 }
1457}
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001458bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,
Calin Juravle225ff812014-11-13 16:46:39 +00001459 uint32_t dex_pc,
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001460 bool is_put) {
1461 uint32_t source_or_dest_reg = instruction.VRegA_21c();
1462 uint16_t field_index = instruction.VRegB_21c();
1463
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00001464 ScopedObjectAccess soa(Thread::Current());
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001465 StackHandleScope<5> hs(soa.Self());
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00001466 Handle<mirror::DexCache> dex_cache(hs.NewHandle(
Mathieu Chartier673ed3d2015-08-28 14:56:43 -07001467 dex_compilation_unit_->GetClassLinker()->FindDexCache(
1468 soa.Self(), *dex_compilation_unit_->GetDexFile())));
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00001469 Handle<mirror::ClassLoader> class_loader(hs.NewHandle(
1470 soa.Decode<mirror::ClassLoader*>(dex_compilation_unit_->GetClassLoader())));
Mathieu Chartierc7853442015-03-27 14:35:38 -07001471 ArtField* resolved_field = compiler_driver_->ResolveField(
1472 soa, dex_cache, class_loader, dex_compilation_unit_, field_index, true);
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001473
Mathieu Chartierc7853442015-03-27 14:35:38 -07001474 if (resolved_field == nullptr) {
Calin Juravlee460d1d2015-09-29 04:52:17 +01001475 MaybeRecordStat(MethodCompilationStat::kUnresolvedField);
1476 Primitive::Type field_type = GetFieldAccessType(*dex_file_, field_index);
Calin Juravle07380a22015-09-17 14:15:12 +01001477 BuildUnresolvedStaticFieldAccess(instruction, dex_pc, is_put, field_type);
Calin Juravlee460d1d2015-09-29 04:52:17 +01001478 return true;
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001479 }
1480
Calin Juravle07380a22015-09-17 14:15:12 +01001481 Primitive::Type field_type = resolved_field->GetTypeAsPrimitiveType();
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001482 const DexFile& outer_dex_file = *outer_compilation_unit_->GetDexFile();
1483 Handle<mirror::DexCache> outer_dex_cache(hs.NewHandle(
Mathieu Chartier673ed3d2015-08-28 14:56:43 -07001484 outer_compilation_unit_->GetClassLinker()->FindDexCache(soa.Self(), outer_dex_file)));
Nicolas Geoffray30451742015-06-19 13:32:41 +01001485 Handle<mirror::Class> outer_class(hs.NewHandle(GetOutermostCompilingClass()));
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00001486
1487 // The index at which the field's class is stored in the DexCache's type array.
1488 uint32_t storage_index;
Nicolas Geoffray30451742015-06-19 13:32:41 +01001489 bool is_outer_class = (outer_class.Get() == resolved_field->GetDeclaringClass());
1490 if (is_outer_class) {
1491 storage_index = outer_class->GetDexTypeIndex();
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001492 } else if (outer_dex_cache.Get() != dex_cache.Get()) {
Roland Levillain4c0eb422015-04-24 16:43:49 +01001493 // The compiler driver cannot currently understand multiple dex caches involved. Just bailout.
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00001494 return false;
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001495 } else {
Calin Juravle07380a22015-09-17 14:15:12 +01001496 // TODO: This is rather expensive. Perf it and cache the results if needed.
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001497 std::pair<bool, bool> pair = compiler_driver_->IsFastStaticField(
1498 outer_dex_cache.Get(),
Nicolas Geoffray30451742015-06-19 13:32:41 +01001499 GetCompilingClass(),
Mathieu Chartierc7853442015-03-27 14:35:38 -07001500 resolved_field,
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001501 field_index,
1502 &storage_index);
1503 bool can_easily_access = is_put ? pair.second : pair.first;
1504 if (!can_easily_access) {
Calin Juravle07380a22015-09-17 14:15:12 +01001505 MaybeRecordStat(MethodCompilationStat::kUnresolvedFieldNotAFastAccess);
1506 BuildUnresolvedStaticFieldAccess(instruction, dex_pc, is_put, field_type);
1507 return true;
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001508 }
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00001509 }
1510
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001511 bool is_in_cache =
1512 compiler_driver_->CanAssumeTypeIsPresentInDexCache(outer_dex_file, storage_index);
Nicolas Geoffray76b1e172015-05-27 17:18:33 +01001513 HLoadClass* constant = new (arena_) HLoadClass(graph_->GetCurrentMethod(),
1514 storage_index,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001515 outer_dex_file,
Nicolas Geoffray30451742015-06-19 13:32:41 +01001516 is_outer_class,
Calin Juravle98893e12015-10-02 21:05:03 +01001517 dex_pc,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001518 /*needs_access_check*/ false,
1519 is_in_cache);
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001520 current_block_->AddInstruction(constant);
1521
1522 HInstruction* cls = constant;
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001523
1524 Handle<mirror::Class> klass(hs.NewHandle(resolved_field->GetDeclaringClass()));
Nicolas Geoffrayd9dc6f42015-11-24 14:06:57 +00001525 if (!IsInitialized(klass)) {
Calin Juravle225ff812014-11-13 16:46:39 +00001526 cls = new (arena_) HClinitCheck(constant, dex_pc);
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001527 current_block_->AddInstruction(cls);
1528 }
Mingyao Yang8df69d42015-10-22 15:40:58 -07001529
Nicolas Geoffray729645a2015-11-19 13:29:02 +00001530 uint16_t class_def_index = klass->GetDexClassDefIndex();
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001531 if (is_put) {
1532 // We need to keep the class alive before loading the value.
Calin Juravlef97f9fb2014-11-11 15:38:19 +00001533 Temporaries temps(graph_);
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001534 temps.Add(cls);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001535 HInstruction* value = LoadLocal(source_or_dest_reg, field_type, dex_pc);
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001536 DCHECK_EQ(value->GetType(), field_type);
Guillaume "Vermeille" Sanchez104fd8a2015-05-20 17:52:13 +01001537 current_block_->AddInstruction(new (arena_) HStaticFieldSet(cls,
1538 value,
1539 field_type,
1540 resolved_field->GetOffset(),
1541 resolved_field->IsVolatile(),
1542 field_index,
Mingyao Yang8df69d42015-10-22 15:40:58 -07001543 class_def_index,
Mathieu Chartier736b5602015-09-02 14:54:11 -07001544 *dex_file_,
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001545 dex_cache_,
1546 dex_pc));
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001547 } else {
Guillaume "Vermeille" Sanchez104fd8a2015-05-20 17:52:13 +01001548 current_block_->AddInstruction(new (arena_) HStaticFieldGet(cls,
1549 field_type,
1550 resolved_field->GetOffset(),
1551 resolved_field->IsVolatile(),
1552 field_index,
Mingyao Yang8df69d42015-10-22 15:40:58 -07001553 class_def_index,
Mathieu Chartier736b5602015-09-02 14:54:11 -07001554 *dex_file_,
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001555 dex_cache_,
1556 dex_pc));
1557 UpdateLocal(source_or_dest_reg, current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01001558 }
1559 return true;
1560}
1561
Calin Juravlebacfec32014-11-14 15:54:36 +00001562void HGraphBuilder::BuildCheckedDivRem(uint16_t out_vreg,
1563 uint16_t first_vreg,
1564 int64_t second_vreg_or_constant,
1565 uint32_t dex_pc,
1566 Primitive::Type type,
1567 bool second_is_constant,
1568 bool isDiv) {
Calin Juravled6fb6cf2014-11-11 19:07:44 +00001569 DCHECK(type == Primitive::kPrimInt || type == Primitive::kPrimLong);
Calin Juravled0d48522014-11-04 16:40:20 +00001570
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001571 HInstruction* first = LoadLocal(first_vreg, type, dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +00001572 HInstruction* second = nullptr;
1573 if (second_is_constant) {
1574 if (type == Primitive::kPrimInt) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001575 second = graph_->GetIntConstant(second_vreg_or_constant, dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +00001576 } else {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001577 second = graph_->GetLongConstant(second_vreg_or_constant, dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +00001578 }
1579 } else {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001580 second = LoadLocal(second_vreg_or_constant, type, dex_pc);
Calin Juravled6fb6cf2014-11-11 19:07:44 +00001581 }
1582
1583 if (!second_is_constant
1584 || (type == Primitive::kPrimInt && second->AsIntConstant()->GetValue() == 0)
1585 || (type == Primitive::kPrimLong && second->AsLongConstant()->GetValue() == 0)) {
1586 second = new (arena_) HDivZeroCheck(second, dex_pc);
Calin Juravlef97f9fb2014-11-11 15:38:19 +00001587 Temporaries temps(graph_);
Calin Juravled0d48522014-11-04 16:40:20 +00001588 current_block_->AddInstruction(second);
1589 temps.Add(current_block_->GetLastInstruction());
1590 }
1591
Calin Juravlebacfec32014-11-14 15:54:36 +00001592 if (isDiv) {
1593 current_block_->AddInstruction(new (arena_) HDiv(type, first, second, dex_pc));
1594 } else {
1595 current_block_->AddInstruction(new (arena_) HRem(type, first, second, dex_pc));
1596 }
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001597 UpdateLocal(out_vreg, current_block_->GetLastInstruction(), dex_pc);
Calin Juravled0d48522014-11-04 16:40:20 +00001598}
1599
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001600void HGraphBuilder::BuildArrayAccess(const Instruction& instruction,
Calin Juravle225ff812014-11-13 16:46:39 +00001601 uint32_t dex_pc,
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001602 bool is_put,
1603 Primitive::Type anticipated_type) {
1604 uint8_t source_or_dest_reg = instruction.VRegA_23x();
1605 uint8_t array_reg = instruction.VRegB_23x();
1606 uint8_t index_reg = instruction.VRegC_23x();
1607
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001608 // We need one temporary for the null check, one for the index, and one for the length.
Calin Juravlef97f9fb2014-11-11 15:38:19 +00001609 Temporaries temps(graph_);
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001610
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001611 HInstruction* object = LoadLocal(array_reg, Primitive::kPrimNot, dex_pc);
Calin Juravle225ff812014-11-13 16:46:39 +00001612 object = new (arena_) HNullCheck(object, dex_pc);
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001613 current_block_->AddInstruction(object);
1614 temps.Add(object);
1615
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001616 HInstruction* length = new (arena_) HArrayLength(object, dex_pc);
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001617 current_block_->AddInstruction(length);
1618 temps.Add(length);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001619 HInstruction* index = LoadLocal(index_reg, Primitive::kPrimInt, dex_pc);
Calin Juravle225ff812014-11-13 16:46:39 +00001620 index = new (arena_) HBoundsCheck(index, length, dex_pc);
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001621 current_block_->AddInstruction(index);
1622 temps.Add(index);
1623 if (is_put) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001624 HInstruction* value = LoadLocal(source_or_dest_reg, anticipated_type, dex_pc);
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001625 // TODO: Insert a type check node if the type is Object.
Nicolas Geoffray39468442014-09-02 15:17:15 +01001626 current_block_->AddInstruction(new (arena_) HArraySet(
Calin Juravle225ff812014-11-13 16:46:39 +00001627 object, index, value, anticipated_type, dex_pc));
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001628 } else {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001629 current_block_->AddInstruction(new (arena_) HArrayGet(object, index, anticipated_type, dex_pc));
1630 UpdateLocal(source_or_dest_reg, current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001631 }
Mark Mendell1152c922015-04-24 17:06:35 -04001632 graph_->SetHasBoundsChecks(true);
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01001633}
1634
Calin Juravle225ff812014-11-13 16:46:39 +00001635void HGraphBuilder::BuildFilledNewArray(uint32_t dex_pc,
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001636 uint32_t type_index,
1637 uint32_t number_of_vreg_arguments,
1638 bool is_range,
1639 uint32_t* args,
1640 uint32_t register_index) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001641 HInstruction* length = graph_->GetIntConstant(number_of_vreg_arguments, dex_pc);
Mingyao Yangfb8464a2015-11-02 10:56:59 -08001642 bool finalizable;
1643 QuickEntrypointEnum entrypoint = NeedsAccessCheck(type_index, &finalizable)
Nicolas Geoffraycb1b00a2015-01-28 14:50:01 +00001644 ? kQuickAllocArrayWithAccessCheck
1645 : kQuickAllocArray;
Guillaume "Vermeille" Sanchez81d804a2015-05-20 12:42:25 +01001646 HInstruction* object = new (arena_) HNewArray(length,
Nicolas Geoffray69aa6012015-06-09 10:34:25 +01001647 graph_->GetCurrentMethod(),
Guillaume "Vermeille" Sanchez81d804a2015-05-20 12:42:25 +01001648 dex_pc,
1649 type_index,
1650 *dex_compilation_unit_->GetDexFile(),
1651 entrypoint);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001652 current_block_->AddInstruction(object);
1653
1654 const char* descriptor = dex_file_->StringByTypeIdx(type_index);
1655 DCHECK_EQ(descriptor[0], '[') << descriptor;
1656 char primitive = descriptor[1];
1657 DCHECK(primitive == 'I'
1658 || primitive == 'L'
1659 || primitive == '[') << descriptor;
1660 bool is_reference_array = (primitive == 'L') || (primitive == '[');
1661 Primitive::Type type = is_reference_array ? Primitive::kPrimNot : Primitive::kPrimInt;
1662
Calin Juravlef97f9fb2014-11-11 15:38:19 +00001663 Temporaries temps(graph_);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001664 temps.Add(object);
1665 for (size_t i = 0; i < number_of_vreg_arguments; ++i) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001666 HInstruction* value = LoadLocal(is_range ? register_index + i : args[i], type, dex_pc);
1667 HInstruction* index = graph_->GetIntConstant(i, dex_pc);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001668 current_block_->AddInstruction(
Calin Juravle225ff812014-11-13 16:46:39 +00001669 new (arena_) HArraySet(object, index, value, type, dex_pc));
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001670 }
1671 latest_result_ = object;
1672}
1673
1674template <typename T>
1675void HGraphBuilder::BuildFillArrayData(HInstruction* object,
1676 const T* data,
1677 uint32_t element_count,
1678 Primitive::Type anticipated_type,
Calin Juravle225ff812014-11-13 16:46:39 +00001679 uint32_t dex_pc) {
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001680 for (uint32_t i = 0; i < element_count; ++i) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001681 HInstruction* index = graph_->GetIntConstant(i, dex_pc);
1682 HInstruction* value = graph_->GetIntConstant(data[i], dex_pc);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001683 current_block_->AddInstruction(new (arena_) HArraySet(
Calin Juravle225ff812014-11-13 16:46:39 +00001684 object, index, value, anticipated_type, dex_pc));
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001685 }
1686}
1687
Calin Juravle225ff812014-11-13 16:46:39 +00001688void HGraphBuilder::BuildFillArrayData(const Instruction& instruction, uint32_t dex_pc) {
Calin Juravlef97f9fb2014-11-11 15:38:19 +00001689 Temporaries temps(graph_);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001690 HInstruction* array = LoadLocal(instruction.VRegA_31t(), Primitive::kPrimNot, dex_pc);
Calin Juravle225ff812014-11-13 16:46:39 +00001691 HNullCheck* null_check = new (arena_) HNullCheck(array, dex_pc);
Calin Juravled0d48522014-11-04 16:40:20 +00001692 current_block_->AddInstruction(null_check);
1693 temps.Add(null_check);
1694
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001695 HInstruction* length = new (arena_) HArrayLength(null_check, dex_pc);
Calin Juravled0d48522014-11-04 16:40:20 +00001696 current_block_->AddInstruction(length);
1697
Calin Juravle225ff812014-11-13 16:46:39 +00001698 int32_t payload_offset = instruction.VRegB_31t() + dex_pc;
Calin Juravled0d48522014-11-04 16:40:20 +00001699 const Instruction::ArrayDataPayload* payload =
1700 reinterpret_cast<const Instruction::ArrayDataPayload*>(code_start_ + payload_offset);
1701 const uint8_t* data = payload->data;
1702 uint32_t element_count = payload->element_count;
1703
1704 // Implementation of this DEX instruction seems to be that the bounds check is
1705 // done before doing any stores.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001706 HInstruction* last_index = graph_->GetIntConstant(payload->element_count - 1, dex_pc);
Calin Juravle225ff812014-11-13 16:46:39 +00001707 current_block_->AddInstruction(new (arena_) HBoundsCheck(last_index, length, dex_pc));
Calin Juravled0d48522014-11-04 16:40:20 +00001708
1709 switch (payload->element_width) {
1710 case 1:
1711 BuildFillArrayData(null_check,
1712 reinterpret_cast<const int8_t*>(data),
1713 element_count,
1714 Primitive::kPrimByte,
Calin Juravle225ff812014-11-13 16:46:39 +00001715 dex_pc);
Calin Juravled0d48522014-11-04 16:40:20 +00001716 break;
1717 case 2:
1718 BuildFillArrayData(null_check,
1719 reinterpret_cast<const int16_t*>(data),
1720 element_count,
1721 Primitive::kPrimShort,
Calin Juravle225ff812014-11-13 16:46:39 +00001722 dex_pc);
Calin Juravled0d48522014-11-04 16:40:20 +00001723 break;
1724 case 4:
1725 BuildFillArrayData(null_check,
1726 reinterpret_cast<const int32_t*>(data),
1727 element_count,
1728 Primitive::kPrimInt,
Calin Juravle225ff812014-11-13 16:46:39 +00001729 dex_pc);
Calin Juravled0d48522014-11-04 16:40:20 +00001730 break;
1731 case 8:
1732 BuildFillWideArrayData(null_check,
1733 reinterpret_cast<const int64_t*>(data),
1734 element_count,
Calin Juravle225ff812014-11-13 16:46:39 +00001735 dex_pc);
Calin Juravled0d48522014-11-04 16:40:20 +00001736 break;
1737 default:
1738 LOG(FATAL) << "Unknown element width for " << payload->element_width;
1739 }
Mark Mendell1152c922015-04-24 17:06:35 -04001740 graph_->SetHasBoundsChecks(true);
Calin Juravled0d48522014-11-04 16:40:20 +00001741}
1742
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001743void HGraphBuilder::BuildFillWideArrayData(HInstruction* object,
Nicolas Geoffray8d6ae522014-10-23 18:32:13 +01001744 const int64_t* data,
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001745 uint32_t element_count,
Calin Juravle225ff812014-11-13 16:46:39 +00001746 uint32_t dex_pc) {
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001747 for (uint32_t i = 0; i < element_count; ++i) {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001748 HInstruction* index = graph_->GetIntConstant(i, dex_pc);
1749 HInstruction* value = graph_->GetLongConstant(data[i], dex_pc);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001750 current_block_->AddInstruction(new (arena_) HArraySet(
Calin Juravle225ff812014-11-13 16:46:39 +00001751 object, index, value, Primitive::kPrimLong, dex_pc));
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01001752 }
1753}
1754
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001755static TypeCheckKind ComputeTypeCheckKind(Handle<mirror::Class> cls)
1756 SHARED_REQUIRES(Locks::mutator_lock_) {
Calin Juravle98893e12015-10-02 21:05:03 +01001757 if (cls.Get() == nullptr) {
1758 return TypeCheckKind::kUnresolvedCheck;
1759 } else if (cls->IsInterface()) {
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001760 return TypeCheckKind::kInterfaceCheck;
1761 } else if (cls->IsArrayClass()) {
1762 if (cls->GetComponentType()->IsObjectClass()) {
1763 return TypeCheckKind::kArrayObjectCheck;
1764 } else if (cls->CannotBeAssignedFromOtherTypes()) {
1765 return TypeCheckKind::kExactCheck;
1766 } else {
1767 return TypeCheckKind::kArrayCheck;
1768 }
1769 } else if (cls->IsFinal()) {
1770 return TypeCheckKind::kExactCheck;
1771 } else if (cls->IsAbstract()) {
1772 return TypeCheckKind::kAbstractClassCheck;
1773 } else {
1774 return TypeCheckKind::kClassHierarchyCheck;
1775 }
1776}
1777
Calin Juravle98893e12015-10-02 21:05:03 +01001778void HGraphBuilder::BuildTypeCheck(const Instruction& instruction,
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001779 uint8_t destination,
1780 uint8_t reference,
1781 uint16_t type_index,
Calin Juravle225ff812014-11-13 16:46:39 +00001782 uint32_t dex_pc) {
Calin Juravle98893e12015-10-02 21:05:03 +01001783 bool type_known_final, type_known_abstract, use_declaring_class;
1784 bool can_access = compiler_driver_->CanAccessTypeWithoutChecks(
1785 dex_compilation_unit_->GetDexMethodIndex(),
1786 *dex_compilation_unit_->GetDexFile(),
1787 type_index,
1788 &type_known_final,
1789 &type_known_abstract,
1790 &use_declaring_class);
1791
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001792 ScopedObjectAccess soa(Thread::Current());
1793 StackHandleScope<2> hs(soa.Self());
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001794 const DexFile& dex_file = *dex_compilation_unit_->GetDexFile();
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001795 Handle<mirror::DexCache> dex_cache(hs.NewHandle(
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001796 dex_compilation_unit_->GetClassLinker()->FindDexCache(soa.Self(), dex_file)));
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001797 Handle<mirror::Class> resolved_class(hs.NewHandle(dex_cache->GetResolvedType(type_index)));
1798
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001799 HInstruction* object = LoadLocal(reference, Primitive::kPrimNot, dex_pc);
Nicolas Geoffray9437b782015-03-25 10:08:51 +00001800 HLoadClass* cls = new (arena_) HLoadClass(
Nicolas Geoffray76b1e172015-05-27 17:18:33 +01001801 graph_->GetCurrentMethod(),
Nicolas Geoffrayd5111bf2015-05-22 15:37:09 +01001802 type_index,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001803 dex_file,
Nicolas Geoffrayd5111bf2015-05-22 15:37:09 +01001804 IsOutermostCompilingClass(type_index),
Calin Juravle98893e12015-10-02 21:05:03 +01001805 dex_pc,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00001806 !can_access,
1807 compiler_driver_->CanAssumeTypeIsPresentInDexCache(dex_file, type_index));
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001808 current_block_->AddInstruction(cls);
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001809
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001810 // The class needs a temporary before being used by the type check.
Calin Juravlef97f9fb2014-11-11 15:38:19 +00001811 Temporaries temps(graph_);
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001812 temps.Add(cls);
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001813
1814 TypeCheckKind check_kind = ComputeTypeCheckKind(resolved_class);
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001815 if (instruction.Opcode() == Instruction::INSTANCE_OF) {
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001816 current_block_->AddInstruction(new (arena_) HInstanceOf(object, cls, check_kind, dex_pc));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001817 UpdateLocal(destination, current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001818 } else {
1819 DCHECK_EQ(instruction.Opcode(), Instruction::CHECK_CAST);
Nicolas Geoffray85c7bab2015-09-18 13:40:46 +00001820 current_block_->AddInstruction(new (arena_) HCheckCast(object, cls, check_kind, dex_pc));
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001821 }
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00001822}
1823
Mingyao Yangfb8464a2015-11-02 10:56:59 -08001824bool HGraphBuilder::NeedsAccessCheck(uint32_t type_index, bool* finalizable) const {
Nicolas Geoffraycb1b00a2015-01-28 14:50:01 +00001825 return !compiler_driver_->CanAccessInstantiableTypeWithoutChecks(
Mingyao Yangfb8464a2015-11-02 10:56:59 -08001826 dex_compilation_unit_->GetDexMethodIndex(), *dex_file_, type_index, finalizable);
Nicolas Geoffraycb1b00a2015-01-28 14:50:01 +00001827}
1828
Mark Mendellfe57faa2015-09-18 09:26:15 -04001829void HGraphBuilder::BuildSwitchJumpTable(const SwitchTable& table,
1830 const Instruction& instruction,
1831 HInstruction* value,
1832 uint32_t dex_pc) {
1833 // Add the successor blocks to the current block.
1834 uint16_t num_entries = table.GetNumEntries();
1835 for (size_t i = 1; i <= num_entries; i++) {
1836 int32_t target_offset = table.GetEntryAt(i);
1837 HBasicBlock* case_target = FindBlockStartingAt(dex_pc + target_offset);
1838 DCHECK(case_target != nullptr);
1839
1840 // Add the target block as a successor.
1841 current_block_->AddSuccessor(case_target);
1842 }
1843
1844 // Add the default target block as the last successor.
1845 HBasicBlock* default_target = FindBlockStartingAt(dex_pc + instruction.SizeInCodeUnits());
1846 DCHECK(default_target != nullptr);
1847 current_block_->AddSuccessor(default_target);
1848
1849 // Now add the Switch instruction.
1850 int32_t starting_key = table.GetEntryAt(0);
1851 current_block_->AddInstruction(
1852 new (arena_) HPackedSwitch(starting_key, num_entries, value, dex_pc));
1853 // This block ends with control flow.
1854 current_block_ = nullptr;
1855}
1856
Calin Juravle48c2b032014-12-09 18:11:36 +00001857void HGraphBuilder::BuildPackedSwitch(const Instruction& instruction, uint32_t dex_pc) {
David Brazdil2ef645b2015-06-17 18:20:52 +01001858 // Verifier guarantees that the payload for PackedSwitch contains:
1859 // (a) number of entries (may be zero)
1860 // (b) first and lowest switch case value (entry 0, always present)
1861 // (c) list of target pcs (entries 1 <= i <= N)
Andreas Gamped881df52014-11-24 23:28:39 -08001862 SwitchTable table(instruction, dex_pc, false);
1863
1864 // Value to test against.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001865 HInstruction* value = LoadLocal(instruction.VRegA(), Primitive::kPrimInt, dex_pc);
Andreas Gamped881df52014-11-24 23:28:39 -08001866
Mark Mendellfe57faa2015-09-18 09:26:15 -04001867 // Starting key value.
1868 int32_t starting_key = table.GetEntryAt(0);
1869
David Brazdil2ef645b2015-06-17 18:20:52 +01001870 // Retrieve number of entries.
Andreas Gampee4d4d322014-12-04 09:09:57 -08001871 uint16_t num_entries = table.GetNumEntries();
David Brazdil2ef645b2015-06-17 18:20:52 +01001872 if (num_entries == 0) {
1873 return;
1874 }
Andreas Gampee4d4d322014-12-04 09:09:57 -08001875
Mark Mendellfe57faa2015-09-18 09:26:15 -04001876 // Don't use a packed switch if there are very few entries.
1877 if (num_entries > kSmallSwitchThreshold) {
1878 BuildSwitchJumpTable(table, instruction, value, dex_pc);
1879 } else {
1880 // Chained cmp-and-branch, starting from starting_key.
1881 for (size_t i = 1; i <= num_entries; i++) {
Mark Mendell3b9f3042015-09-24 08:43:40 -04001882 BuildSwitchCaseHelper(instruction,
1883 i,
1884 i == num_entries,
1885 table,
1886 value,
1887 starting_key + i - 1,
1888 table.GetEntryAt(i),
1889 dex_pc);
Mark Mendellfe57faa2015-09-18 09:26:15 -04001890 }
Andreas Gamped881df52014-11-24 23:28:39 -08001891 }
Andreas Gamped881df52014-11-24 23:28:39 -08001892}
1893
Calin Juravle48c2b032014-12-09 18:11:36 +00001894void HGraphBuilder::BuildSparseSwitch(const Instruction& instruction, uint32_t dex_pc) {
David Brazdil2ef645b2015-06-17 18:20:52 +01001895 // Verifier guarantees that the payload for SparseSwitch contains:
1896 // (a) number of entries (may be zero)
1897 // (b) sorted key values (entries 0 <= i < N)
1898 // (c) target pcs corresponding to the switch values (entries N <= i < 2*N)
Andreas Gampee4d4d322014-12-04 09:09:57 -08001899 SwitchTable table(instruction, dex_pc, true);
1900
1901 // Value to test against.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001902 HInstruction* value = LoadLocal(instruction.VRegA(), Primitive::kPrimInt, dex_pc);
Andreas Gampee4d4d322014-12-04 09:09:57 -08001903
1904 uint16_t num_entries = table.GetNumEntries();
Andreas Gampee4d4d322014-12-04 09:09:57 -08001905
1906 for (size_t i = 0; i < num_entries; i++) {
1907 BuildSwitchCaseHelper(instruction, i, i == static_cast<size_t>(num_entries) - 1, table, value,
1908 table.GetEntryAt(i), table.GetEntryAt(i + num_entries), dex_pc);
1909 }
Andreas Gampee4d4d322014-12-04 09:09:57 -08001910}
1911
1912void HGraphBuilder::BuildSwitchCaseHelper(const Instruction& instruction, size_t index,
1913 bool is_last_case, const SwitchTable& table,
1914 HInstruction* value, int32_t case_value_int,
1915 int32_t target_offset, uint32_t dex_pc) {
David Brazdil852eaff2015-02-02 15:23:05 +00001916 HBasicBlock* case_target = FindBlockStartingAt(dex_pc + target_offset);
1917 DCHECK(case_target != nullptr);
1918 PotentiallyAddSuspendCheck(case_target, dex_pc);
Andreas Gampee4d4d322014-12-04 09:09:57 -08001919
1920 // The current case's value.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001921 HInstruction* this_case_value = graph_->GetIntConstant(case_value_int, dex_pc);
Andreas Gampee4d4d322014-12-04 09:09:57 -08001922
1923 // Compare value and this_case_value.
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001924 HEqual* comparison = new (arena_) HEqual(value, this_case_value, dex_pc);
Andreas Gampee4d4d322014-12-04 09:09:57 -08001925 current_block_->AddInstruction(comparison);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001926 HInstruction* ifinst = new (arena_) HIf(comparison, dex_pc);
Andreas Gampee4d4d322014-12-04 09:09:57 -08001927 current_block_->AddInstruction(ifinst);
1928
1929 // Case hit: use the target offset to determine where to go.
Andreas Gampee4d4d322014-12-04 09:09:57 -08001930 current_block_->AddSuccessor(case_target);
1931
1932 // Case miss: go to the next case (or default fall-through).
1933 // When there is a next case, we use the block stored with the table offset representing this
1934 // case (that is where we registered them in ComputeBranchTargets).
1935 // When there is no next case, we use the following instruction.
1936 // TODO: Find a good way to peel the last iteration to avoid conditional, but still have re-use.
1937 if (!is_last_case) {
1938 HBasicBlock* next_case_target = FindBlockStartingAt(table.GetDexPcForIndex(index));
1939 DCHECK(next_case_target != nullptr);
1940 current_block_->AddSuccessor(next_case_target);
1941
1942 // Need to manually add the block, as there is no dex-pc transition for the cases.
1943 graph_->AddBlock(next_case_target);
1944
1945 current_block_ = next_case_target;
1946 } else {
1947 HBasicBlock* default_target = FindBlockStartingAt(dex_pc + instruction.SizeInCodeUnits());
1948 DCHECK(default_target != nullptr);
1949 current_block_->AddSuccessor(default_target);
1950 current_block_ = nullptr;
1951 }
1952}
1953
David Brazdil852eaff2015-02-02 15:23:05 +00001954void HGraphBuilder::PotentiallyAddSuspendCheck(HBasicBlock* target, uint32_t dex_pc) {
1955 int32_t target_offset = target->GetDexPc() - dex_pc;
Nicolas Geoffrayfbc695f2014-09-15 15:33:30 +00001956 if (target_offset <= 0) {
David Brazdil852eaff2015-02-02 15:23:05 +00001957 // DX generates back edges to the first encountered return. We can save
1958 // time of later passes by not adding redundant suspend checks.
David Brazdil2fd6aa52015-02-02 18:58:27 +00001959 HInstruction* last_in_target = target->GetLastInstruction();
1960 if (last_in_target != nullptr &&
1961 (last_in_target->IsReturn() || last_in_target->IsReturnVoid())) {
1962 return;
David Brazdil852eaff2015-02-02 15:23:05 +00001963 }
1964
1965 // Add a suspend check to backward branches which may potentially loop. We
1966 // can remove them after we recognize loops in the graph.
Calin Juravle225ff812014-11-13 16:46:39 +00001967 current_block_->AddInstruction(new (arena_) HSuspendCheck(dex_pc));
Nicolas Geoffrayfbc695f2014-09-15 15:33:30 +00001968 }
1969}
1970
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00001971bool HGraphBuilder::CanDecodeQuickenedInfo() const {
1972 return interpreter_metadata_ != nullptr;
1973}
1974
1975uint16_t HGraphBuilder::LookupQuickenedInfo(uint32_t dex_pc) {
1976 DCHECK(interpreter_metadata_ != nullptr);
1977 uint32_t dex_pc_in_map = DecodeUnsignedLeb128(&interpreter_metadata_);
1978 DCHECK_EQ(dex_pc, dex_pc_in_map);
1979 return DecodeUnsignedLeb128(&interpreter_metadata_);
1980}
1981
Calin Juravle225ff812014-11-13 16:46:39 +00001982bool HGraphBuilder::AnalyzeDexInstruction(const Instruction& instruction, uint32_t dex_pc) {
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00001983 if (current_block_ == nullptr) {
1984 return true; // Dead code
1985 }
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +00001986
Nicolas Geoffray818f2102014-02-18 16:43:35 +00001987 switch (instruction.Opcode()) {
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +00001988 case Instruction::CONST_4: {
1989 int32_t register_index = instruction.VRegA();
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001990 HIntConstant* constant = graph_->GetIntConstant(instruction.VRegB_11n(), dex_pc);
1991 UpdateLocal(register_index, constant, dex_pc);
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +00001992 break;
1993 }
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00001994
Nicolas Geoffrayb55f8352014-04-07 15:26:35 +01001995 case Instruction::CONST_16: {
1996 int32_t register_index = instruction.VRegA();
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06001997 HIntConstant* constant = graph_->GetIntConstant(instruction.VRegB_21s(), dex_pc);
1998 UpdateLocal(register_index, constant, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01001999 break;
2000 }
2001
Dave Allison20dfc792014-06-16 20:44:29 -07002002 case Instruction::CONST: {
2003 int32_t register_index = instruction.VRegA();
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002004 HIntConstant* constant = graph_->GetIntConstant(instruction.VRegB_31i(), dex_pc);
2005 UpdateLocal(register_index, constant, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -07002006 break;
2007 }
2008
2009 case Instruction::CONST_HIGH16: {
2010 int32_t register_index = instruction.VRegA();
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002011 HIntConstant* constant = graph_->GetIntConstant(instruction.VRegB_21h() << 16, dex_pc);
2012 UpdateLocal(register_index, constant, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -07002013 break;
2014 }
2015
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002016 case Instruction::CONST_WIDE_16: {
2017 int32_t register_index = instruction.VRegA();
Dave Allison20dfc792014-06-16 20:44:29 -07002018 // Get 16 bits of constant value, sign extended to 64 bits.
2019 int64_t value = instruction.VRegB_21s();
2020 value <<= 48;
2021 value >>= 48;
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002022 HLongConstant* constant = graph_->GetLongConstant(value, dex_pc);
2023 UpdateLocal(register_index, constant, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002024 break;
2025 }
2026
2027 case Instruction::CONST_WIDE_32: {
2028 int32_t register_index = instruction.VRegA();
Dave Allison20dfc792014-06-16 20:44:29 -07002029 // Get 32 bits of constant value, sign extended to 64 bits.
2030 int64_t value = instruction.VRegB_31i();
2031 value <<= 32;
2032 value >>= 32;
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002033 HLongConstant* constant = graph_->GetLongConstant(value, dex_pc);
2034 UpdateLocal(register_index, constant, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002035 break;
2036 }
2037
2038 case Instruction::CONST_WIDE: {
2039 int32_t register_index = instruction.VRegA();
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002040 HLongConstant* constant = graph_->GetLongConstant(instruction.VRegB_51l(), dex_pc);
2041 UpdateLocal(register_index, constant, dex_pc);
Nicolas Geoffrayb55f8352014-04-07 15:26:35 +01002042 break;
2043 }
2044
Dave Allison20dfc792014-06-16 20:44:29 -07002045 case Instruction::CONST_WIDE_HIGH16: {
2046 int32_t register_index = instruction.VRegA();
2047 int64_t value = static_cast<int64_t>(instruction.VRegB_21h()) << 48;
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002048 HLongConstant* constant = graph_->GetLongConstant(value, dex_pc);
2049 UpdateLocal(register_index, constant, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -07002050 break;
2051 }
2052
Nicolas Geoffraydadf3172014-11-07 16:36:02 +00002053 // Note that the SSA building will refine the types.
Dave Allison20dfc792014-06-16 20:44:29 -07002054 case Instruction::MOVE:
2055 case Instruction::MOVE_FROM16:
2056 case Instruction::MOVE_16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002057 HInstruction* value = LoadLocal(instruction.VRegB(), Primitive::kPrimInt, dex_pc);
2058 UpdateLocal(instruction.VRegA(), value, dex_pc);
Nicolas Geoffrayb55f8352014-04-07 15:26:35 +01002059 break;
2060 }
2061
Nicolas Geoffraydadf3172014-11-07 16:36:02 +00002062 // Note that the SSA building will refine the types.
Dave Allison20dfc792014-06-16 20:44:29 -07002063 case Instruction::MOVE_WIDE:
2064 case Instruction::MOVE_WIDE_FROM16:
2065 case Instruction::MOVE_WIDE_16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002066 HInstruction* value = LoadLocal(instruction.VRegB(), Primitive::kPrimLong, dex_pc);
2067 UpdateLocal(instruction.VRegA(), value, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -07002068 break;
2069 }
2070
2071 case Instruction::MOVE_OBJECT:
2072 case Instruction::MOVE_OBJECT_16:
2073 case Instruction::MOVE_OBJECT_FROM16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002074 HInstruction* value = LoadLocal(instruction.VRegB(), Primitive::kPrimNot, dex_pc);
2075 UpdateLocal(instruction.VRegA(), value, dex_pc);
Dave Allison20dfc792014-06-16 20:44:29 -07002076 break;
2077 }
2078
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002079 case Instruction::RETURN_VOID_NO_BARRIER:
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00002080 case Instruction::RETURN_VOID: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002081 BuildReturn(instruction, Primitive::kPrimVoid, dex_pc);
Nicolas Geoffray818f2102014-02-18 16:43:35 +00002082 break;
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00002083 }
2084
Dave Allison20dfc792014-06-16 20:44:29 -07002085#define IF_XX(comparison, cond) \
Calin Juravle225ff812014-11-13 16:46:39 +00002086 case Instruction::IF_##cond: If_22t<comparison>(instruction, dex_pc); break; \
2087 case Instruction::IF_##cond##Z: If_21t<comparison>(instruction, dex_pc); break
Nicolas Geoffrayb55f8352014-04-07 15:26:35 +01002088
Dave Allison20dfc792014-06-16 20:44:29 -07002089 IF_XX(HEqual, EQ);
2090 IF_XX(HNotEqual, NE);
2091 IF_XX(HLessThan, LT);
2092 IF_XX(HLessThanOrEqual, LE);
2093 IF_XX(HGreaterThan, GT);
2094 IF_XX(HGreaterThanOrEqual, GE);
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00002095
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +00002096 case Instruction::GOTO:
2097 case Instruction::GOTO_16:
2098 case Instruction::GOTO_32: {
Nicolas Geoffrayfbc695f2014-09-15 15:33:30 +00002099 int32_t offset = instruction.GetTargetOffset();
Calin Juravle225ff812014-11-13 16:46:39 +00002100 HBasicBlock* target = FindBlockStartingAt(offset + dex_pc);
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +00002101 DCHECK(target != nullptr);
David Brazdil852eaff2015-02-02 15:23:05 +00002102 PotentiallyAddSuspendCheck(target, dex_pc);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002103 current_block_->AddInstruction(new (arena_) HGoto(dex_pc));
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +00002104 current_block_->AddSuccessor(target);
2105 current_block_ = nullptr;
2106 break;
2107 }
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00002108
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002109 case Instruction::RETURN: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002110 BuildReturn(instruction, return_type_, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002111 break;
2112 }
2113
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002114 case Instruction::RETURN_OBJECT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002115 BuildReturn(instruction, return_type_, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002116 break;
2117 }
2118
2119 case Instruction::RETURN_WIDE: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002120 BuildReturn(instruction, return_type_, dex_pc);
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00002121 break;
2122 }
2123
Nicolas Geoffraye982f0b2014-08-13 02:11:24 +01002124 case Instruction::INVOKE_DIRECT:
Nicolas Geoffray0d8db992014-11-11 14:40:10 +00002125 case Instruction::INVOKE_INTERFACE:
2126 case Instruction::INVOKE_STATIC:
2127 case Instruction::INVOKE_SUPER:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002128 case Instruction::INVOKE_VIRTUAL:
2129 case Instruction::INVOKE_VIRTUAL_QUICK: {
2130 uint16_t method_idx;
2131 if (instruction.Opcode() == Instruction::INVOKE_VIRTUAL_QUICK) {
2132 if (!CanDecodeQuickenedInfo()) {
2133 return false;
2134 }
2135 method_idx = LookupQuickenedInfo(dex_pc);
2136 } else {
2137 method_idx = instruction.VRegB_35c();
2138 }
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002139 uint32_t number_of_vreg_arguments = instruction.VRegA_35c();
Nicolas Geoffray4a34a422014-04-03 10:38:37 +01002140 uint32_t args[5];
Ian Rogers29a26482014-05-02 15:27:29 -07002141 instruction.GetVarArgs(args);
Calin Juravle225ff812014-11-13 16:46:39 +00002142 if (!BuildInvoke(instruction, dex_pc, method_idx,
Nicolas Geoffraydadf3172014-11-07 16:36:02 +00002143 number_of_vreg_arguments, false, args, -1)) {
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002144 return false;
Nicolas Geoffray4a34a422014-04-03 10:38:37 +01002145 }
Nicolas Geoffray4a34a422014-04-03 10:38:37 +01002146 break;
2147 }
2148
Nicolas Geoffraye982f0b2014-08-13 02:11:24 +01002149 case Instruction::INVOKE_DIRECT_RANGE:
Nicolas Geoffray0d8db992014-11-11 14:40:10 +00002150 case Instruction::INVOKE_INTERFACE_RANGE:
2151 case Instruction::INVOKE_STATIC_RANGE:
2152 case Instruction::INVOKE_SUPER_RANGE:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002153 case Instruction::INVOKE_VIRTUAL_RANGE:
2154 case Instruction::INVOKE_VIRTUAL_RANGE_QUICK: {
2155 uint16_t method_idx;
2156 if (instruction.Opcode() == Instruction::INVOKE_VIRTUAL_RANGE_QUICK) {
2157 if (!CanDecodeQuickenedInfo()) {
2158 return false;
2159 }
2160 method_idx = LookupQuickenedInfo(dex_pc);
2161 } else {
2162 method_idx = instruction.VRegB_3rc();
2163 }
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002164 uint32_t number_of_vreg_arguments = instruction.VRegA_3rc();
2165 uint32_t register_index = instruction.VRegC();
Calin Juravle225ff812014-11-13 16:46:39 +00002166 if (!BuildInvoke(instruction, dex_pc, method_idx,
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002167 number_of_vreg_arguments, true, nullptr, register_index)) {
Nicolas Geoffray4a34a422014-04-03 10:38:37 +01002168 return false;
2169 }
Nicolas Geoffray8ccc3f52014-03-19 10:34:11 +00002170 break;
2171 }
2172
Roland Levillain88cb1752014-10-20 16:36:47 +01002173 case Instruction::NEG_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002174 Unop_12x<HNeg>(instruction, Primitive::kPrimInt, dex_pc);
Roland Levillain88cb1752014-10-20 16:36:47 +01002175 break;
2176 }
2177
Roland Levillain2e07b4f2014-10-23 18:12:09 +01002178 case Instruction::NEG_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002179 Unop_12x<HNeg>(instruction, Primitive::kPrimLong, dex_pc);
Roland Levillain2e07b4f2014-10-23 18:12:09 +01002180 break;
2181 }
2182
Roland Levillain3dbcb382014-10-28 17:30:07 +00002183 case Instruction::NEG_FLOAT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002184 Unop_12x<HNeg>(instruction, Primitive::kPrimFloat, dex_pc);
Roland Levillain3dbcb382014-10-28 17:30:07 +00002185 break;
2186 }
2187
2188 case Instruction::NEG_DOUBLE: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002189 Unop_12x<HNeg>(instruction, Primitive::kPrimDouble, dex_pc);
Roland Levillain3dbcb382014-10-28 17:30:07 +00002190 break;
2191 }
2192
Roland Levillain1cc5f2512014-10-22 18:06:21 +01002193 case Instruction::NOT_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002194 Unop_12x<HNot>(instruction, Primitive::kPrimInt, dex_pc);
Roland Levillain1cc5f2512014-10-22 18:06:21 +01002195 break;
2196 }
2197
Roland Levillain70566432014-10-24 16:20:17 +01002198 case Instruction::NOT_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002199 Unop_12x<HNot>(instruction, Primitive::kPrimLong, dex_pc);
Roland Levillain70566432014-10-24 16:20:17 +01002200 break;
2201 }
2202
Roland Levillaindff1f282014-11-05 14:15:05 +00002203 case Instruction::INT_TO_LONG: {
Roland Levillain624279f2014-12-04 11:54:28 +00002204 Conversion_12x(instruction, Primitive::kPrimInt, Primitive::kPrimLong, dex_pc);
Roland Levillaindff1f282014-11-05 14:15:05 +00002205 break;
2206 }
2207
Roland Levillaincff13742014-11-17 14:32:17 +00002208 case Instruction::INT_TO_FLOAT: {
Roland Levillain624279f2014-12-04 11:54:28 +00002209 Conversion_12x(instruction, Primitive::kPrimInt, Primitive::kPrimFloat, dex_pc);
Roland Levillaincff13742014-11-17 14:32:17 +00002210 break;
2211 }
2212
2213 case Instruction::INT_TO_DOUBLE: {
Roland Levillain624279f2014-12-04 11:54:28 +00002214 Conversion_12x(instruction, Primitive::kPrimInt, Primitive::kPrimDouble, dex_pc);
Roland Levillaincff13742014-11-17 14:32:17 +00002215 break;
2216 }
2217
Roland Levillain946e1432014-11-11 17:35:19 +00002218 case Instruction::LONG_TO_INT: {
Roland Levillain624279f2014-12-04 11:54:28 +00002219 Conversion_12x(instruction, Primitive::kPrimLong, Primitive::kPrimInt, dex_pc);
Roland Levillain946e1432014-11-11 17:35:19 +00002220 break;
2221 }
2222
Roland Levillain6d0e4832014-11-27 18:31:21 +00002223 case Instruction::LONG_TO_FLOAT: {
Roland Levillain624279f2014-12-04 11:54:28 +00002224 Conversion_12x(instruction, Primitive::kPrimLong, Primitive::kPrimFloat, dex_pc);
Roland Levillain6d0e4832014-11-27 18:31:21 +00002225 break;
2226 }
2227
Roland Levillain647b9ed2014-11-27 12:06:00 +00002228 case Instruction::LONG_TO_DOUBLE: {
Roland Levillain624279f2014-12-04 11:54:28 +00002229 Conversion_12x(instruction, Primitive::kPrimLong, Primitive::kPrimDouble, dex_pc);
Roland Levillain647b9ed2014-11-27 12:06:00 +00002230 break;
2231 }
2232
Roland Levillain3f8f9362014-12-02 17:45:01 +00002233 case Instruction::FLOAT_TO_INT: {
Roland Levillain624279f2014-12-04 11:54:28 +00002234 Conversion_12x(instruction, Primitive::kPrimFloat, Primitive::kPrimInt, dex_pc);
2235 break;
2236 }
2237
2238 case Instruction::FLOAT_TO_LONG: {
2239 Conversion_12x(instruction, Primitive::kPrimFloat, Primitive::kPrimLong, dex_pc);
Roland Levillain3f8f9362014-12-02 17:45:01 +00002240 break;
2241 }
2242
Roland Levillain8964e2b2014-12-04 12:10:50 +00002243 case Instruction::FLOAT_TO_DOUBLE: {
2244 Conversion_12x(instruction, Primitive::kPrimFloat, Primitive::kPrimDouble, dex_pc);
2245 break;
2246 }
2247
Roland Levillain4c0b61f2014-12-05 12:06:01 +00002248 case Instruction::DOUBLE_TO_INT: {
2249 Conversion_12x(instruction, Primitive::kPrimDouble, Primitive::kPrimInt, dex_pc);
2250 break;
2251 }
2252
2253 case Instruction::DOUBLE_TO_LONG: {
2254 Conversion_12x(instruction, Primitive::kPrimDouble, Primitive::kPrimLong, dex_pc);
2255 break;
2256 }
2257
Roland Levillain8964e2b2014-12-04 12:10:50 +00002258 case Instruction::DOUBLE_TO_FLOAT: {
2259 Conversion_12x(instruction, Primitive::kPrimDouble, Primitive::kPrimFloat, dex_pc);
2260 break;
2261 }
2262
Roland Levillain51d3fc42014-11-13 14:11:42 +00002263 case Instruction::INT_TO_BYTE: {
Roland Levillain624279f2014-12-04 11:54:28 +00002264 Conversion_12x(instruction, Primitive::kPrimInt, Primitive::kPrimByte, dex_pc);
Roland Levillain51d3fc42014-11-13 14:11:42 +00002265 break;
2266 }
2267
Roland Levillain01a8d712014-11-14 16:27:39 +00002268 case Instruction::INT_TO_SHORT: {
Roland Levillain624279f2014-12-04 11:54:28 +00002269 Conversion_12x(instruction, Primitive::kPrimInt, Primitive::kPrimShort, dex_pc);
Roland Levillain01a8d712014-11-14 16:27:39 +00002270 break;
2271 }
2272
Roland Levillain981e4542014-11-14 11:47:14 +00002273 case Instruction::INT_TO_CHAR: {
Roland Levillain624279f2014-12-04 11:54:28 +00002274 Conversion_12x(instruction, Primitive::kPrimInt, Primitive::kPrimChar, dex_pc);
Roland Levillain981e4542014-11-14 11:47:14 +00002275 break;
2276 }
2277
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002278 case Instruction::ADD_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002279 Binop_23x<HAdd>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002280 break;
2281 }
2282
2283 case Instruction::ADD_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002284 Binop_23x<HAdd>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002285 break;
2286 }
2287
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01002288 case Instruction::ADD_DOUBLE: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002289 Binop_23x<HAdd>(instruction, Primitive::kPrimDouble, dex_pc);
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01002290 break;
2291 }
2292
2293 case Instruction::ADD_FLOAT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002294 Binop_23x<HAdd>(instruction, Primitive::kPrimFloat, dex_pc);
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01002295 break;
2296 }
2297
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002298 case Instruction::SUB_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002299 Binop_23x<HSub>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002300 break;
2301 }
2302
2303 case Instruction::SUB_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002304 Binop_23x<HSub>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002305 break;
2306 }
2307
Calin Juravle096cc022014-10-23 17:01:13 +01002308 case Instruction::SUB_FLOAT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002309 Binop_23x<HSub>(instruction, Primitive::kPrimFloat, dex_pc);
Calin Juravle096cc022014-10-23 17:01:13 +01002310 break;
2311 }
2312
2313 case Instruction::SUB_DOUBLE: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002314 Binop_23x<HSub>(instruction, Primitive::kPrimDouble, dex_pc);
Calin Juravle096cc022014-10-23 17:01:13 +01002315 break;
2316 }
2317
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002318 case Instruction::ADD_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002319 Binop_12x<HAdd>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002320 break;
2321 }
2322
Calin Juravle34bacdf2014-10-07 20:23:36 +01002323 case Instruction::MUL_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002324 Binop_23x<HMul>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle34bacdf2014-10-07 20:23:36 +01002325 break;
2326 }
2327
2328 case Instruction::MUL_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002329 Binop_23x<HMul>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle34bacdf2014-10-07 20:23:36 +01002330 break;
2331 }
2332
Calin Juravleb5bfa962014-10-21 18:02:24 +01002333 case Instruction::MUL_FLOAT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002334 Binop_23x<HMul>(instruction, Primitive::kPrimFloat, dex_pc);
Calin Juravleb5bfa962014-10-21 18:02:24 +01002335 break;
2336 }
2337
2338 case Instruction::MUL_DOUBLE: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002339 Binop_23x<HMul>(instruction, Primitive::kPrimDouble, dex_pc);
Calin Juravleb5bfa962014-10-21 18:02:24 +01002340 break;
2341 }
2342
Calin Juravled0d48522014-11-04 16:40:20 +00002343 case Instruction::DIV_INT: {
Calin Juravlebacfec32014-11-14 15:54:36 +00002344 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegB(), instruction.VRegC(),
2345 dex_pc, Primitive::kPrimInt, false, true);
Calin Juravled0d48522014-11-04 16:40:20 +00002346 break;
2347 }
2348
Calin Juravled6fb6cf2014-11-11 19:07:44 +00002349 case Instruction::DIV_LONG: {
Calin Juravlebacfec32014-11-14 15:54:36 +00002350 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegB(), instruction.VRegC(),
2351 dex_pc, Primitive::kPrimLong, false, true);
Calin Juravled6fb6cf2014-11-11 19:07:44 +00002352 break;
2353 }
2354
Calin Juravle7c4954d2014-10-28 16:57:40 +00002355 case Instruction::DIV_FLOAT: {
Calin Juravle225ff812014-11-13 16:46:39 +00002356 Binop_23x<HDiv>(instruction, Primitive::kPrimFloat, dex_pc);
Calin Juravle7c4954d2014-10-28 16:57:40 +00002357 break;
2358 }
2359
2360 case Instruction::DIV_DOUBLE: {
Calin Juravle225ff812014-11-13 16:46:39 +00002361 Binop_23x<HDiv>(instruction, Primitive::kPrimDouble, dex_pc);
Calin Juravle7c4954d2014-10-28 16:57:40 +00002362 break;
2363 }
2364
Calin Juravlebacfec32014-11-14 15:54:36 +00002365 case Instruction::REM_INT: {
2366 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegB(), instruction.VRegC(),
2367 dex_pc, Primitive::kPrimInt, false, false);
2368 break;
2369 }
2370
2371 case Instruction::REM_LONG: {
2372 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegB(), instruction.VRegC(),
2373 dex_pc, Primitive::kPrimLong, false, false);
2374 break;
2375 }
2376
Calin Juravled2ec87d2014-12-08 14:24:46 +00002377 case Instruction::REM_FLOAT: {
2378 Binop_23x<HRem>(instruction, Primitive::kPrimFloat, dex_pc);
2379 break;
2380 }
2381
2382 case Instruction::REM_DOUBLE: {
2383 Binop_23x<HRem>(instruction, Primitive::kPrimDouble, dex_pc);
2384 break;
2385 }
2386
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002387 case Instruction::AND_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002388 Binop_23x<HAnd>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002389 break;
2390 }
2391
2392 case Instruction::AND_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002393 Binop_23x<HAnd>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002394 break;
2395 }
2396
Calin Juravle9aec02f2014-11-18 23:06:35 +00002397 case Instruction::SHL_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002398 Binop_23x_shift<HShl>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002399 break;
2400 }
2401
2402 case Instruction::SHL_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002403 Binop_23x_shift<HShl>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002404 break;
2405 }
2406
2407 case Instruction::SHR_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002408 Binop_23x_shift<HShr>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002409 break;
2410 }
2411
2412 case Instruction::SHR_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002413 Binop_23x_shift<HShr>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002414 break;
2415 }
2416
2417 case Instruction::USHR_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002418 Binop_23x_shift<HUShr>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002419 break;
2420 }
2421
2422 case Instruction::USHR_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002423 Binop_23x_shift<HUShr>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002424 break;
2425 }
2426
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002427 case Instruction::OR_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002428 Binop_23x<HOr>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002429 break;
2430 }
2431
2432 case Instruction::OR_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002433 Binop_23x<HOr>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002434 break;
2435 }
2436
2437 case Instruction::XOR_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002438 Binop_23x<HXor>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002439 break;
2440 }
2441
2442 case Instruction::XOR_LONG: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002443 Binop_23x<HXor>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002444 break;
2445 }
2446
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002447 case Instruction::ADD_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002448 Binop_12x<HAdd>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002449 break;
2450 }
2451
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01002452 case Instruction::ADD_DOUBLE_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002453 Binop_12x<HAdd>(instruction, Primitive::kPrimDouble, dex_pc);
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01002454 break;
2455 }
2456
2457 case Instruction::ADD_FLOAT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002458 Binop_12x<HAdd>(instruction, Primitive::kPrimFloat, dex_pc);
Nicolas Geoffray7fb49da2014-10-06 09:12:41 +01002459 break;
2460 }
2461
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002462 case Instruction::SUB_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002463 Binop_12x<HSub>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002464 break;
2465 }
2466
2467 case Instruction::SUB_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002468 Binop_12x<HSub>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002469 break;
2470 }
2471
Calin Juravle096cc022014-10-23 17:01:13 +01002472 case Instruction::SUB_FLOAT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002473 Binop_12x<HSub>(instruction, Primitive::kPrimFloat, dex_pc);
Calin Juravle096cc022014-10-23 17:01:13 +01002474 break;
2475 }
2476
2477 case Instruction::SUB_DOUBLE_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002478 Binop_12x<HSub>(instruction, Primitive::kPrimDouble, dex_pc);
Calin Juravle096cc022014-10-23 17:01:13 +01002479 break;
2480 }
2481
Calin Juravle34bacdf2014-10-07 20:23:36 +01002482 case Instruction::MUL_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002483 Binop_12x<HMul>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle34bacdf2014-10-07 20:23:36 +01002484 break;
2485 }
2486
2487 case Instruction::MUL_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002488 Binop_12x<HMul>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle34bacdf2014-10-07 20:23:36 +01002489 break;
2490 }
2491
Calin Juravleb5bfa962014-10-21 18:02:24 +01002492 case Instruction::MUL_FLOAT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002493 Binop_12x<HMul>(instruction, Primitive::kPrimFloat, dex_pc);
Calin Juravleb5bfa962014-10-21 18:02:24 +01002494 break;
2495 }
2496
2497 case Instruction::MUL_DOUBLE_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002498 Binop_12x<HMul>(instruction, Primitive::kPrimDouble, dex_pc);
Calin Juravleb5bfa962014-10-21 18:02:24 +01002499 break;
2500 }
2501
Calin Juravle865fc882014-11-06 17:09:03 +00002502 case Instruction::DIV_INT_2ADDR: {
Calin Juravlebacfec32014-11-14 15:54:36 +00002503 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegA(), instruction.VRegB(),
2504 dex_pc, Primitive::kPrimInt, false, true);
Calin Juravle865fc882014-11-06 17:09:03 +00002505 break;
2506 }
2507
Calin Juravled6fb6cf2014-11-11 19:07:44 +00002508 case Instruction::DIV_LONG_2ADDR: {
Calin Juravlebacfec32014-11-14 15:54:36 +00002509 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegA(), instruction.VRegB(),
2510 dex_pc, Primitive::kPrimLong, false, true);
2511 break;
2512 }
2513
2514 case Instruction::REM_INT_2ADDR: {
2515 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegA(), instruction.VRegB(),
2516 dex_pc, Primitive::kPrimInt, false, false);
2517 break;
2518 }
2519
2520 case Instruction::REM_LONG_2ADDR: {
2521 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegA(), instruction.VRegB(),
2522 dex_pc, Primitive::kPrimLong, false, false);
Calin Juravled6fb6cf2014-11-11 19:07:44 +00002523 break;
2524 }
2525
Calin Juravled2ec87d2014-12-08 14:24:46 +00002526 case Instruction::REM_FLOAT_2ADDR: {
2527 Binop_12x<HRem>(instruction, Primitive::kPrimFloat, dex_pc);
2528 break;
2529 }
2530
2531 case Instruction::REM_DOUBLE_2ADDR: {
2532 Binop_12x<HRem>(instruction, Primitive::kPrimDouble, dex_pc);
2533 break;
2534 }
2535
Calin Juravle9aec02f2014-11-18 23:06:35 +00002536 case Instruction::SHL_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002537 Binop_12x_shift<HShl>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002538 break;
2539 }
2540
2541 case Instruction::SHL_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002542 Binop_12x_shift<HShl>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002543 break;
2544 }
2545
2546 case Instruction::SHR_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002547 Binop_12x_shift<HShr>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002548 break;
2549 }
2550
2551 case Instruction::SHR_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002552 Binop_12x_shift<HShr>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002553 break;
2554 }
2555
2556 case Instruction::USHR_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002557 Binop_12x_shift<HUShr>(instruction, Primitive::kPrimInt, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002558 break;
2559 }
2560
2561 case Instruction::USHR_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002562 Binop_12x_shift<HUShr>(instruction, Primitive::kPrimLong, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002563 break;
2564 }
2565
Calin Juravle7c4954d2014-10-28 16:57:40 +00002566 case Instruction::DIV_FLOAT_2ADDR: {
Calin Juravle225ff812014-11-13 16:46:39 +00002567 Binop_12x<HDiv>(instruction, Primitive::kPrimFloat, dex_pc);
Calin Juravle7c4954d2014-10-28 16:57:40 +00002568 break;
2569 }
2570
2571 case Instruction::DIV_DOUBLE_2ADDR: {
Calin Juravle225ff812014-11-13 16:46:39 +00002572 Binop_12x<HDiv>(instruction, Primitive::kPrimDouble, dex_pc);
Calin Juravle7c4954d2014-10-28 16:57:40 +00002573 break;
2574 }
2575
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002576 case Instruction::AND_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002577 Binop_12x<HAnd>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002578 break;
2579 }
2580
2581 case Instruction::AND_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002582 Binop_12x<HAnd>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002583 break;
2584 }
2585
2586 case Instruction::OR_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002587 Binop_12x<HOr>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002588 break;
2589 }
2590
2591 case Instruction::OR_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002592 Binop_12x<HOr>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002593 break;
2594 }
2595
2596 case Instruction::XOR_INT_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002597 Binop_12x<HXor>(instruction, Primitive::kPrimInt, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002598 break;
2599 }
2600
2601 case Instruction::XOR_LONG_2ADDR: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002602 Binop_12x<HXor>(instruction, Primitive::kPrimLong, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002603 break;
2604 }
2605
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002606 case Instruction::ADD_INT_LIT16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002607 Binop_22s<HAdd>(instruction, false, dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002608 break;
2609 }
2610
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002611 case Instruction::AND_INT_LIT16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002612 Binop_22s<HAnd>(instruction, false, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002613 break;
2614 }
2615
2616 case Instruction::OR_INT_LIT16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002617 Binop_22s<HOr>(instruction, false, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002618 break;
2619 }
2620
2621 case Instruction::XOR_INT_LIT16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002622 Binop_22s<HXor>(instruction, false, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002623 break;
2624 }
2625
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002626 case Instruction::RSUB_INT: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002627 Binop_22s<HSub>(instruction, true, dex_pc);
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002628 break;
2629 }
2630
Calin Juravle34bacdf2014-10-07 20:23:36 +01002631 case Instruction::MUL_INT_LIT16: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002632 Binop_22s<HMul>(instruction, false, dex_pc);
Calin Juravle34bacdf2014-10-07 20:23:36 +01002633 break;
2634 }
2635
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002636 case Instruction::ADD_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002637 Binop_22b<HAdd>(instruction, false, dex_pc);
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002638 break;
2639 }
2640
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002641 case Instruction::AND_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002642 Binop_22b<HAnd>(instruction, false, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002643 break;
2644 }
2645
2646 case Instruction::OR_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002647 Binop_22b<HOr>(instruction, false, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002648 break;
2649 }
2650
2651 case Instruction::XOR_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002652 Binop_22b<HXor>(instruction, false, dex_pc);
Nicolas Geoffray9574c4b2014-11-12 13:19:37 +00002653 break;
2654 }
2655
Nicolas Geoffrayf583e592014-04-07 13:20:42 +01002656 case Instruction::RSUB_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002657 Binop_22b<HSub>(instruction, true, dex_pc);
Nicolas Geoffrayd8ee7372014-03-28 15:43:40 +00002658 break;
2659 }
2660
Calin Juravle34bacdf2014-10-07 20:23:36 +01002661 case Instruction::MUL_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002662 Binop_22b<HMul>(instruction, false, dex_pc);
Calin Juravle34bacdf2014-10-07 20:23:36 +01002663 break;
2664 }
2665
Calin Juravled0d48522014-11-04 16:40:20 +00002666 case Instruction::DIV_INT_LIT16:
2667 case Instruction::DIV_INT_LIT8: {
Calin Juravlebacfec32014-11-14 15:54:36 +00002668 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegB(), instruction.VRegC(),
2669 dex_pc, Primitive::kPrimInt, true, true);
2670 break;
2671 }
2672
2673 case Instruction::REM_INT_LIT16:
2674 case Instruction::REM_INT_LIT8: {
2675 BuildCheckedDivRem(instruction.VRegA(), instruction.VRegB(), instruction.VRegC(),
2676 dex_pc, Primitive::kPrimInt, true, false);
Calin Juravled0d48522014-11-04 16:40:20 +00002677 break;
2678 }
2679
Calin Juravle9aec02f2014-11-18 23:06:35 +00002680 case Instruction::SHL_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002681 Binop_22b<HShl>(instruction, false, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002682 break;
2683 }
2684
2685 case Instruction::SHR_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002686 Binop_22b<HShr>(instruction, false, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002687 break;
2688 }
2689
2690 case Instruction::USHR_INT_LIT8: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002691 Binop_22b<HUShr>(instruction, false, dex_pc);
Calin Juravle9aec02f2014-11-18 23:06:35 +00002692 break;
2693 }
2694
Nicolas Geoffray2e7038a2014-04-03 18:49:58 +01002695 case Instruction::NEW_INSTANCE: {
Nicolas Geoffraycb1b00a2015-01-28 14:50:01 +00002696 uint16_t type_index = instruction.VRegB_21c();
Jeff Hao848f70a2014-01-15 13:49:50 -08002697 if (compiler_driver_->IsStringTypeIndex(type_index, dex_file_)) {
Jeff Hao848f70a2014-01-15 13:49:50 -08002698 int32_t register_index = instruction.VRegA();
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002699 HFakeString* fake_string = new (arena_) HFakeString(dex_pc);
Nicolas Geoffray2e7cd752015-07-10 11:38:52 +01002700 current_block_->AddInstruction(fake_string);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002701 UpdateLocal(register_index, fake_string, dex_pc);
Jeff Hao848f70a2014-01-15 13:49:50 -08002702 } else {
Nicolas Geoffray729645a2015-11-19 13:29:02 +00002703 if (!BuildNewInstance(type_index, dex_pc)) {
2704 return false;
2705 }
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002706 UpdateLocal(instruction.VRegA(), current_block_->GetLastInstruction(), dex_pc);
Jeff Hao848f70a2014-01-15 13:49:50 -08002707 }
Nicolas Geoffray2e7038a2014-04-03 18:49:58 +01002708 break;
2709 }
2710
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01002711 case Instruction::NEW_ARRAY: {
Nicolas Geoffraycb1b00a2015-01-28 14:50:01 +00002712 uint16_t type_index = instruction.VRegC_22c();
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002713 HInstruction* length = LoadLocal(instruction.VRegB_22c(), Primitive::kPrimInt, dex_pc);
Mingyao Yangfb8464a2015-11-02 10:56:59 -08002714 bool finalizable;
2715 QuickEntrypointEnum entrypoint = NeedsAccessCheck(type_index, &finalizable)
Nicolas Geoffraycb1b00a2015-01-28 14:50:01 +00002716 ? kQuickAllocArrayWithAccessCheck
2717 : kQuickAllocArray;
Nicolas Geoffray69aa6012015-06-09 10:34:25 +01002718 current_block_->AddInstruction(new (arena_) HNewArray(length,
2719 graph_->GetCurrentMethod(),
2720 dex_pc,
2721 type_index,
2722 *dex_compilation_unit_->GetDexFile(),
2723 entrypoint));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002724 UpdateLocal(instruction.VRegA_22c(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01002725 break;
2726 }
2727
2728 case Instruction::FILLED_NEW_ARRAY: {
2729 uint32_t number_of_vreg_arguments = instruction.VRegA_35c();
2730 uint32_t type_index = instruction.VRegB_35c();
2731 uint32_t args[5];
2732 instruction.GetVarArgs(args);
Calin Juravle225ff812014-11-13 16:46:39 +00002733 BuildFilledNewArray(dex_pc, type_index, number_of_vreg_arguments, false, args, 0);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01002734 break;
2735 }
2736
2737 case Instruction::FILLED_NEW_ARRAY_RANGE: {
2738 uint32_t number_of_vreg_arguments = instruction.VRegA_3rc();
2739 uint32_t type_index = instruction.VRegB_3rc();
2740 uint32_t register_index = instruction.VRegC_3rc();
2741 BuildFilledNewArray(
Calin Juravle225ff812014-11-13 16:46:39 +00002742 dex_pc, type_index, number_of_vreg_arguments, true, nullptr, register_index);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01002743 break;
2744 }
2745
2746 case Instruction::FILL_ARRAY_DATA: {
Calin Juravle225ff812014-11-13 16:46:39 +00002747 BuildFillArrayData(instruction, dex_pc);
Nicolas Geoffraya3d05a42014-10-20 17:41:32 +01002748 break;
2749 }
2750
Nicolas Geoffrayddb311f2014-05-16 09:28:54 +01002751 case Instruction::MOVE_RESULT:
Dave Allison20dfc792014-06-16 20:44:29 -07002752 case Instruction::MOVE_RESULT_WIDE:
David Brazdilfc6a86a2015-06-26 10:33:45 +00002753 case Instruction::MOVE_RESULT_OBJECT: {
Nicolas Geoffray1efcc222015-06-24 12:41:20 +01002754 if (latest_result_ == nullptr) {
2755 // Only dead code can lead to this situation, where the verifier
2756 // does not reject the method.
2757 } else {
David Brazdilfc6a86a2015-06-26 10:33:45 +00002758 // An Invoke/FilledNewArray and its MoveResult could have landed in
2759 // different blocks if there was a try/catch block boundary between
2760 // them. For Invoke, we insert a StoreLocal after the instruction. For
2761 // FilledNewArray, the local needs to be updated after the array was
2762 // filled, otherwise we might overwrite an input vreg.
2763 HStoreLocal* update_local =
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002764 new (arena_) HStoreLocal(GetLocalAt(instruction.VRegA()), latest_result_, dex_pc);
David Brazdilfc6a86a2015-06-26 10:33:45 +00002765 HBasicBlock* block = latest_result_->GetBlock();
2766 if (block == current_block_) {
2767 // MoveResult and the previous instruction are in the same block.
2768 current_block_->AddInstruction(update_local);
2769 } else {
2770 // The two instructions are in different blocks. Insert the MoveResult
2771 // before the final control-flow instruction of the previous block.
2772 DCHECK(block->EndsWithControlFlowInstruction());
2773 DCHECK(current_block_->GetInstructions().IsEmpty());
2774 block->InsertInstructionBefore(update_local, block->GetLastInstruction());
2775 }
Nicolas Geoffray1efcc222015-06-24 12:41:20 +01002776 latest_result_ = nullptr;
2777 }
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002778 break;
David Brazdilfc6a86a2015-06-26 10:33:45 +00002779 }
Nicolas Geoffray01bc96d2014-04-11 17:43:50 +01002780
Nicolas Geoffray412f10c2014-06-19 10:00:34 +01002781 case Instruction::CMP_LONG: {
Roland Levillain4fa13f62015-07-06 18:11:54 +01002782 Binop_23x_cmp(instruction, Primitive::kPrimLong, ComparisonBias::kNoBias, dex_pc);
Calin Juravleddb7df22014-11-25 20:56:51 +00002783 break;
2784 }
2785
2786 case Instruction::CMPG_FLOAT: {
Roland Levillain4fa13f62015-07-06 18:11:54 +01002787 Binop_23x_cmp(instruction, Primitive::kPrimFloat, ComparisonBias::kGtBias, dex_pc);
Calin Juravleddb7df22014-11-25 20:56:51 +00002788 break;
2789 }
2790
2791 case Instruction::CMPG_DOUBLE: {
Roland Levillain4fa13f62015-07-06 18:11:54 +01002792 Binop_23x_cmp(instruction, Primitive::kPrimDouble, ComparisonBias::kGtBias, dex_pc);
Calin Juravleddb7df22014-11-25 20:56:51 +00002793 break;
2794 }
2795
2796 case Instruction::CMPL_FLOAT: {
Roland Levillain4fa13f62015-07-06 18:11:54 +01002797 Binop_23x_cmp(instruction, Primitive::kPrimFloat, ComparisonBias::kLtBias, dex_pc);
Calin Juravleddb7df22014-11-25 20:56:51 +00002798 break;
2799 }
2800
2801 case Instruction::CMPL_DOUBLE: {
Roland Levillain4fa13f62015-07-06 18:11:54 +01002802 Binop_23x_cmp(instruction, Primitive::kPrimDouble, ComparisonBias::kLtBias, dex_pc);
Nicolas Geoffray412f10c2014-06-19 10:00:34 +01002803 break;
2804 }
2805
Nicolas Geoffraybe9a92a2014-02-25 14:22:56 +00002806 case Instruction::NOP:
2807 break;
Nicolas Geoffraybab4ed72014-03-11 17:53:17 +00002808
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002809 case Instruction::IGET:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002810 case Instruction::IGET_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002811 case Instruction::IGET_WIDE:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002812 case Instruction::IGET_WIDE_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002813 case Instruction::IGET_OBJECT:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002814 case Instruction::IGET_OBJECT_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002815 case Instruction::IGET_BOOLEAN:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002816 case Instruction::IGET_BOOLEAN_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002817 case Instruction::IGET_BYTE:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002818 case Instruction::IGET_BYTE_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002819 case Instruction::IGET_CHAR:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002820 case Instruction::IGET_CHAR_QUICK:
2821 case Instruction::IGET_SHORT:
2822 case Instruction::IGET_SHORT_QUICK: {
Calin Juravle225ff812014-11-13 16:46:39 +00002823 if (!BuildInstanceFieldAccess(instruction, dex_pc, false)) {
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002824 return false;
2825 }
2826 break;
2827 }
2828
2829 case Instruction::IPUT:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002830 case Instruction::IPUT_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002831 case Instruction::IPUT_WIDE:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002832 case Instruction::IPUT_WIDE_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002833 case Instruction::IPUT_OBJECT:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002834 case Instruction::IPUT_OBJECT_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002835 case Instruction::IPUT_BOOLEAN:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002836 case Instruction::IPUT_BOOLEAN_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002837 case Instruction::IPUT_BYTE:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002838 case Instruction::IPUT_BYTE_QUICK:
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002839 case Instruction::IPUT_CHAR:
Nicolas Geoffray9523a3e2015-07-17 11:51:28 +00002840 case Instruction::IPUT_CHAR_QUICK:
2841 case Instruction::IPUT_SHORT:
2842 case Instruction::IPUT_SHORT_QUICK: {
Calin Juravle225ff812014-11-13 16:46:39 +00002843 if (!BuildInstanceFieldAccess(instruction, dex_pc, true)) {
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01002844 return false;
2845 }
2846 break;
2847 }
2848
2849 case Instruction::SGET:
2850 case Instruction::SGET_WIDE:
2851 case Instruction::SGET_OBJECT:
2852 case Instruction::SGET_BOOLEAN:
2853 case Instruction::SGET_BYTE:
2854 case Instruction::SGET_CHAR:
2855 case Instruction::SGET_SHORT: {
Calin Juravle225ff812014-11-13 16:46:39 +00002856 if (!BuildStaticFieldAccess(instruction, dex_pc, false)) {
Nicolas Geoffray19a19cf2014-10-22 16:07:05 +01002857 return false;
2858 }
2859 break;
2860 }
2861
2862 case Instruction::SPUT:
2863 case Instruction::SPUT_WIDE:
2864 case Instruction::SPUT_OBJECT:
2865 case Instruction::SPUT_BOOLEAN:
2866 case Instruction::SPUT_BYTE:
2867 case Instruction::SPUT_CHAR:
2868 case Instruction::SPUT_SHORT: {
Calin Juravle225ff812014-11-13 16:46:39 +00002869 if (!BuildStaticFieldAccess(instruction, dex_pc, true)) {
Nicolas Geoffraye5038322014-07-04 09:41:32 +01002870 return false;
2871 }
2872 break;
2873 }
2874
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01002875#define ARRAY_XX(kind, anticipated_type) \
2876 case Instruction::AGET##kind: { \
Calin Juravle225ff812014-11-13 16:46:39 +00002877 BuildArrayAccess(instruction, dex_pc, false, anticipated_type); \
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01002878 break; \
2879 } \
2880 case Instruction::APUT##kind: { \
Calin Juravle225ff812014-11-13 16:46:39 +00002881 BuildArrayAccess(instruction, dex_pc, true, anticipated_type); \
Nicolas Geoffray3c7bb982014-07-23 16:04:16 +01002882 break; \
2883 }
2884
2885 ARRAY_XX(, Primitive::kPrimInt);
2886 ARRAY_XX(_WIDE, Primitive::kPrimLong);
2887 ARRAY_XX(_OBJECT, Primitive::kPrimNot);
2888 ARRAY_XX(_BOOLEAN, Primitive::kPrimBoolean);
2889 ARRAY_XX(_BYTE, Primitive::kPrimByte);
2890 ARRAY_XX(_CHAR, Primitive::kPrimChar);
2891 ARRAY_XX(_SHORT, Primitive::kPrimShort);
2892
Nicolas Geoffray39468442014-09-02 15:17:15 +01002893 case Instruction::ARRAY_LENGTH: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002894 HInstruction* object = LoadLocal(instruction.VRegB_12x(), Primitive::kPrimNot, dex_pc);
Nicolas Geoffrayde58ab22014-11-05 12:46:03 +00002895 // No need for a temporary for the null check, it is the only input of the following
2896 // instruction.
Calin Juravle225ff812014-11-13 16:46:39 +00002897 object = new (arena_) HNullCheck(object, dex_pc);
Nicolas Geoffrayde58ab22014-11-05 12:46:03 +00002898 current_block_->AddInstruction(object);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002899 current_block_->AddInstruction(new (arena_) HArrayLength(object, dex_pc));
2900 UpdateLocal(instruction.VRegA_12x(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffray39468442014-09-02 15:17:15 +01002901 break;
2902 }
2903
Nicolas Geoffrayb5f62b32014-10-30 10:58:41 +00002904 case Instruction::CONST_STRING: {
Nicolas Geoffray917d0162015-11-24 18:25:35 +00002905 uint32_t string_index = instruction.VRegB_21c();
2906 bool in_dex_cache = compiler_driver_->CanAssumeStringIsPresentInDexCache(
2907 *dex_file_, string_index);
Nicolas Geoffrayfbdaa302015-05-29 12:06:56 +01002908 current_block_->AddInstruction(
Nicolas Geoffray917d0162015-11-24 18:25:35 +00002909 new (arena_) HLoadString(graph_->GetCurrentMethod(), string_index, dex_pc, in_dex_cache));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002910 UpdateLocal(instruction.VRegA_21c(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffrayb5f62b32014-10-30 10:58:41 +00002911 break;
2912 }
2913
2914 case Instruction::CONST_STRING_JUMBO: {
Nicolas Geoffray917d0162015-11-24 18:25:35 +00002915 uint32_t string_index = instruction.VRegB_31c();
2916 bool in_dex_cache = compiler_driver_->CanAssumeStringIsPresentInDexCache(
2917 *dex_file_, string_index);
Nicolas Geoffrayfbdaa302015-05-29 12:06:56 +01002918 current_block_->AddInstruction(
Nicolas Geoffray917d0162015-11-24 18:25:35 +00002919 new (arena_) HLoadString(graph_->GetCurrentMethod(), string_index, dex_pc, in_dex_cache));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002920 UpdateLocal(instruction.VRegA_31c(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffrayb5f62b32014-10-30 10:58:41 +00002921 break;
2922 }
2923
Nicolas Geoffray424f6762014-11-03 14:51:25 +00002924 case Instruction::CONST_CLASS: {
2925 uint16_t type_index = instruction.VRegB_21c();
2926 bool type_known_final;
2927 bool type_known_abstract;
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00002928 bool dont_use_is_referrers_class;
2929 // `CanAccessTypeWithoutChecks` will tell whether the method being
2930 // built is trying to access its own class, so that the generated
2931 // code can optimize for this case. However, the optimization does not
Nicolas Geoffray9437b782015-03-25 10:08:51 +00002932 // work for inlining, so we use `IsOutermostCompilingClass` instead.
Nicolas Geoffray424f6762014-11-03 14:51:25 +00002933 bool can_access = compiler_driver_->CanAccessTypeWithoutChecks(
2934 dex_compilation_unit_->GetDexMethodIndex(), *dex_file_, type_index,
Nicolas Geoffraye53798a2014-12-01 10:31:54 +00002935 &type_known_final, &type_known_abstract, &dont_use_is_referrers_class);
Nicolas Geoffrayd5111bf2015-05-22 15:37:09 +01002936 current_block_->AddInstruction(new (arena_) HLoadClass(
Nicolas Geoffray76b1e172015-05-27 17:18:33 +01002937 graph_->GetCurrentMethod(),
Nicolas Geoffrayd5111bf2015-05-22 15:37:09 +01002938 type_index,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00002939 *dex_file_,
Nicolas Geoffrayd5111bf2015-05-22 15:37:09 +01002940 IsOutermostCompilingClass(type_index),
Calin Juravle98893e12015-10-02 21:05:03 +01002941 dex_pc,
Nicolas Geoffray42e372e2015-11-24 15:48:56 +00002942 !can_access,
2943 compiler_driver_->CanAssumeTypeIsPresentInDexCache(*dex_file_, type_index)));
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002944 UpdateLocal(instruction.VRegA_21c(), current_block_->GetLastInstruction(), dex_pc);
Nicolas Geoffray424f6762014-11-03 14:51:25 +00002945 break;
2946 }
2947
Nicolas Geoffrayde58ab22014-11-05 12:46:03 +00002948 case Instruction::MOVE_EXCEPTION: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002949 current_block_->AddInstruction(new (arena_) HLoadException(dex_pc));
2950 UpdateLocal(instruction.VRegA_11x(), current_block_->GetLastInstruction(), dex_pc);
2951 current_block_->AddInstruction(new (arena_) HClearException(dex_pc));
Nicolas Geoffrayde58ab22014-11-05 12:46:03 +00002952 break;
2953 }
2954
2955 case Instruction::THROW: {
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002956 HInstruction* exception = LoadLocal(instruction.VRegA_11x(), Primitive::kPrimNot, dex_pc);
Calin Juravle225ff812014-11-13 16:46:39 +00002957 current_block_->AddInstruction(new (arena_) HThrow(exception, dex_pc));
Nicolas Geoffrayde58ab22014-11-05 12:46:03 +00002958 // A throw instruction must branch to the exit block.
2959 current_block_->AddSuccessor(exit_block_);
2960 // We finished building this block. Set the current block to null to avoid
2961 // adding dead instructions to it.
2962 current_block_ = nullptr;
2963 break;
2964 }
2965
Nicolas Geoffray6f5c41f2014-11-06 08:59:20 +00002966 case Instruction::INSTANCE_OF: {
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00002967 uint8_t destination = instruction.VRegA_22c();
2968 uint8_t reference = instruction.VRegB_22c();
Nicolas Geoffray6f5c41f2014-11-06 08:59:20 +00002969 uint16_t type_index = instruction.VRegC_22c();
Calin Juravle98893e12015-10-02 21:05:03 +01002970 BuildTypeCheck(instruction, destination, reference, type_index, dex_pc);
Nicolas Geoffray57a88d42014-11-10 15:09:21 +00002971 break;
2972 }
2973
2974 case Instruction::CHECK_CAST: {
2975 uint8_t reference = instruction.VRegA_21c();
2976 uint16_t type_index = instruction.VRegB_21c();
Calin Juravle98893e12015-10-02 21:05:03 +01002977 BuildTypeCheck(instruction, -1, reference, type_index, dex_pc);
Nicolas Geoffray6f5c41f2014-11-06 08:59:20 +00002978 break;
2979 }
2980
Nicolas Geoffrayb7baf5c2014-11-11 16:29:44 +00002981 case Instruction::MONITOR_ENTER: {
2982 current_block_->AddInstruction(new (arena_) HMonitorOperation(
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002983 LoadLocal(instruction.VRegA_11x(), Primitive::kPrimNot, dex_pc),
Nicolas Geoffrayb7baf5c2014-11-11 16:29:44 +00002984 HMonitorOperation::kEnter,
Calin Juravle225ff812014-11-13 16:46:39 +00002985 dex_pc));
Nicolas Geoffrayb7baf5c2014-11-11 16:29:44 +00002986 break;
2987 }
2988
2989 case Instruction::MONITOR_EXIT: {
2990 current_block_->AddInstruction(new (arena_) HMonitorOperation(
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06002991 LoadLocal(instruction.VRegA_11x(), Primitive::kPrimNot, dex_pc),
Nicolas Geoffrayb7baf5c2014-11-11 16:29:44 +00002992 HMonitorOperation::kExit,
Calin Juravle225ff812014-11-13 16:46:39 +00002993 dex_pc));
Nicolas Geoffrayb7baf5c2014-11-11 16:29:44 +00002994 break;
2995 }
2996
Andreas Gamped881df52014-11-24 23:28:39 -08002997 case Instruction::PACKED_SWITCH: {
Calin Juravle48c2b032014-12-09 18:11:36 +00002998 BuildPackedSwitch(instruction, dex_pc);
Andreas Gamped881df52014-11-24 23:28:39 -08002999 break;
3000 }
3001
Andreas Gampee4d4d322014-12-04 09:09:57 -08003002 case Instruction::SPARSE_SWITCH: {
Calin Juravle48c2b032014-12-09 18:11:36 +00003003 BuildSparseSwitch(instruction, dex_pc);
Andreas Gampee4d4d322014-12-04 09:09:57 -08003004 break;
3005 }
3006
Nicolas Geoffray818f2102014-02-18 16:43:35 +00003007 default:
Calin Juravle48c2b032014-12-09 18:11:36 +00003008 VLOG(compiler) << "Did not compile "
3009 << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_)
3010 << " because of unhandled instruction "
3011 << instruction.Name();
3012 MaybeRecordStat(MethodCompilationStat::kNotCompiledUnhandledInstruction);
Nicolas Geoffray818f2102014-02-18 16:43:35 +00003013 return false;
3014 }
3015 return true;
Nicolas Geoffraydadf3172014-11-07 16:36:02 +00003016} // NOLINT(readability/fn_size)
Nicolas Geoffray818f2102014-02-18 16:43:35 +00003017
Vladimir Marko2aaa4b52015-09-17 17:03:26 +01003018HLocal* HGraphBuilder::GetLocalAt(uint32_t register_index) const {
Vladimir Marko2aaa4b52015-09-17 17:03:26 +01003019 return locals_[register_index];
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +00003020}
3021
Vladimir Marko2aaa4b52015-09-17 17:03:26 +01003022void HGraphBuilder::UpdateLocal(uint32_t register_index,
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06003023 HInstruction* instruction,
3024 uint32_t dex_pc) const {
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +00003025 HLocal* local = GetLocalAt(register_index);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06003026 current_block_->AddInstruction(new (arena_) HStoreLocal(local, instruction, dex_pc));
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +00003027}
3028
Vladimir Marko2aaa4b52015-09-17 17:03:26 +01003029HInstruction* HGraphBuilder::LoadLocal(uint32_t register_index,
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06003030 Primitive::Type type,
3031 uint32_t dex_pc) const {
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +00003032 HLocal* local = GetLocalAt(register_index);
Yevgeny Rouban3ecfd652015-09-07 17:57:00 +06003033 current_block_->AddInstruction(new (arena_) HLoadLocal(local, type, dex_pc));
Nicolas Geoffray787c3072014-03-17 10:20:19 +00003034 return current_block_->GetLastInstruction();
Nicolas Geoffray3ff386a2014-03-04 14:46:47 +00003035}
3036
Nicolas Geoffray818f2102014-02-18 16:43:35 +00003037} // namespace art