blob: 33d6c14ba706412b225c67fbbea056623e4c14be [file] [log] [blame]
Vladimir Markof59f18b2014-02-17 15:53:57 +00001/*
2 * 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
Vladimir Markof59f18b2014-02-17 15:53:57 +000017#include "compiler_internals.h"
Vladimir Marko95a05972014-05-30 10:01:32 +010018#include "global_value_numbering.h"
19#include "local_value_numbering.h"
Vladimir Markof59f18b2014-02-17 15:53:57 +000020#include "gtest/gtest.h"
21
22namespace art {
23
24class LocalValueNumberingTest : public testing::Test {
25 protected:
26 struct IFieldDef {
27 uint16_t field_idx;
28 uintptr_t declaring_dex_file;
29 uint16_t declaring_field_idx;
30 bool is_volatile;
31 };
32
33 struct SFieldDef {
34 uint16_t field_idx;
35 uintptr_t declaring_dex_file;
36 uint16_t declaring_field_idx;
37 bool is_volatile;
38 };
39
40 struct MIRDef {
41 static constexpr size_t kMaxSsaDefs = 2;
Vladimir Marko2ac01fc2014-05-22 12:09:08 +010042 static constexpr size_t kMaxSsaUses = 4;
Vladimir Markof59f18b2014-02-17 15:53:57 +000043
44 Instruction::Code opcode;
45 int64_t value;
Vladimir Markobe0e5462014-02-26 11:24:15 +000046 uint32_t field_info;
Vladimir Markof59f18b2014-02-17 15:53:57 +000047 size_t num_uses;
48 int32_t uses[kMaxSsaUses];
49 size_t num_defs;
50 int32_t defs[kMaxSsaDefs];
51 };
52
53#define DEF_CONST(opcode, reg, value) \
54 { opcode, value, 0u, 0, { }, 1, { reg } }
55#define DEF_CONST_WIDE(opcode, reg, value) \
56 { opcode, value, 0u, 0, { }, 2, { reg, reg + 1 } }
Vladimir Marko2ac01fc2014-05-22 12:09:08 +010057#define DEF_CONST_STRING(opcode, reg, index) \
58 { opcode, index, 0u, 0, { }, 1, { reg } }
Vladimir Markobe0e5462014-02-26 11:24:15 +000059#define DEF_IGET(opcode, reg, obj, field_info) \
60 { opcode, 0u, field_info, 1, { obj }, 1, { reg } }
61#define DEF_IGET_WIDE(opcode, reg, obj, field_info) \
62 { opcode, 0u, field_info, 1, { obj }, 2, { reg, reg + 1 } }
63#define DEF_IPUT(opcode, reg, obj, field_info) \
64 { opcode, 0u, field_info, 2, { reg, obj }, 0, { } }
65#define DEF_IPUT_WIDE(opcode, reg, obj, field_info) \
66 { opcode, 0u, field_info, 3, { reg, reg + 1, obj }, 0, { } }
67#define DEF_SGET(opcode, reg, field_info) \
68 { opcode, 0u, field_info, 0, { }, 1, { reg } }
69#define DEF_SGET_WIDE(opcode, reg, field_info) \
70 { opcode, 0u, field_info, 0, { }, 2, { reg, reg + 1 } }
71#define DEF_SPUT(opcode, reg, field_info) \
72 { opcode, 0u, field_info, 1, { reg }, 0, { } }
73#define DEF_SPUT_WIDE(opcode, reg, field_info) \
74 { opcode, 0u, field_info, 2, { reg, reg + 1 }, 0, { } }
Vladimir Marko2ac01fc2014-05-22 12:09:08 +010075#define DEF_AGET(opcode, reg, obj, idx) \
76 { opcode, 0u, 0u, 2, { obj, idx }, 1, { reg } }
77#define DEF_AGET_WIDE(opcode, reg, obj, idx) \
78 { opcode, 0u, 0u, 2, { obj, idx }, 2, { reg, reg + 1 } }
79#define DEF_APUT(opcode, reg, obj, idx) \
80 { opcode, 0u, 0u, 3, { reg, obj, idx }, 0, { } }
81#define DEF_APUT_WIDE(opcode, reg, obj, idx) \
82 { opcode, 0u, 0u, 4, { reg, reg + 1, obj, idx }, 0, { } }
Vladimir Markof59f18b2014-02-17 15:53:57 +000083#define DEF_INVOKE1(opcode, reg) \
84 { opcode, 0u, 0u, 1, { reg }, 0, { } }
85#define DEF_UNIQUE_REF(opcode, reg) \
86 { opcode, 0u, 0u, 0, { }, 1, { reg } } // CONST_CLASS, CONST_STRING, NEW_ARRAY, ...
87
88 void DoPrepareIFields(const IFieldDef* defs, size_t count) {
Vladimir Markoe39c54e2014-09-22 14:50:02 +010089 cu_.mir_graph->ifield_lowering_infos_.clear();
90 cu_.mir_graph->ifield_lowering_infos_.reserve(count);
Vladimir Markobe0e5462014-02-26 11:24:15 +000091 for (size_t i = 0u; i != count; ++i) {
92 const IFieldDef* def = &defs[i];
93 MirIFieldLoweringInfo field_info(def->field_idx);
94 if (def->declaring_dex_file != 0u) {
95 field_info.declaring_dex_file_ = reinterpret_cast<const DexFile*>(def->declaring_dex_file);
96 field_info.declaring_field_idx_ = def->declaring_field_idx;
97 field_info.flags_ = 0u | // Without kFlagIsStatic.
98 (def->is_volatile ? MirIFieldLoweringInfo::kFlagIsVolatile : 0u);
99 }
Vladimir Markoe39c54e2014-09-22 14:50:02 +0100100 cu_.mir_graph->ifield_lowering_infos_.push_back(field_info);
Vladimir Markobe0e5462014-02-26 11:24:15 +0000101 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000102 }
103
104 template <size_t count>
105 void PrepareIFields(const IFieldDef (&defs)[count]) {
106 DoPrepareIFields(defs, count);
107 }
108
109 void DoPrepareSFields(const SFieldDef* defs, size_t count) {
Vladimir Markoe39c54e2014-09-22 14:50:02 +0100110 cu_.mir_graph->sfield_lowering_infos_.clear();
111 cu_.mir_graph->sfield_lowering_infos_.reserve(count);
Vladimir Markobe0e5462014-02-26 11:24:15 +0000112 for (size_t i = 0u; i != count; ++i) {
113 const SFieldDef* def = &defs[i];
114 MirSFieldLoweringInfo field_info(def->field_idx);
Vladimir Markof418f322014-07-09 14:45:36 +0100115 // Mark even unresolved fields as initialized.
116 field_info.flags_ = MirSFieldLoweringInfo::kFlagIsStatic |
117 MirSFieldLoweringInfo::kFlagIsInitialized;
Vladimir Markobe0e5462014-02-26 11:24:15 +0000118 if (def->declaring_dex_file != 0u) {
119 field_info.declaring_dex_file_ = reinterpret_cast<const DexFile*>(def->declaring_dex_file);
120 field_info.declaring_field_idx_ = def->declaring_field_idx;
Vladimir Markof418f322014-07-09 14:45:36 +0100121 field_info.flags_ |= (def->is_volatile ? MirSFieldLoweringInfo::kFlagIsVolatile : 0u);
Vladimir Markobe0e5462014-02-26 11:24:15 +0000122 }
Vladimir Markoe39c54e2014-09-22 14:50:02 +0100123 cu_.mir_graph->sfield_lowering_infos_.push_back(field_info);
Vladimir Markobe0e5462014-02-26 11:24:15 +0000124 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000125 }
126
127 template <size_t count>
128 void PrepareSFields(const SFieldDef (&defs)[count]) {
129 DoPrepareSFields(defs, count);
130 }
131
132 void DoPrepareMIRs(const MIRDef* defs, size_t count) {
133 mir_count_ = count;
Vladimir Marko83cc7ae2014-02-12 18:02:05 +0000134 mirs_ = reinterpret_cast<MIR*>(cu_.arena.Alloc(sizeof(MIR) * count, kArenaAllocMIR));
Vladimir Markof59f18b2014-02-17 15:53:57 +0000135 ssa_reps_.resize(count);
136 for (size_t i = 0u; i != count; ++i) {
137 const MIRDef* def = &defs[i];
138 MIR* mir = &mirs_[i];
139 mir->dalvikInsn.opcode = def->opcode;
140 mir->dalvikInsn.vB = static_cast<int32_t>(def->value);
141 mir->dalvikInsn.vB_wide = def->value;
Vladimir Markobe0e5462014-02-26 11:24:15 +0000142 if (def->opcode >= Instruction::IGET && def->opcode <= Instruction::IPUT_SHORT) {
Vladimir Markoe39c54e2014-09-22 14:50:02 +0100143 ASSERT_LT(def->field_info, cu_.mir_graph->ifield_lowering_infos_.size());
Vladimir Markobe0e5462014-02-26 11:24:15 +0000144 mir->meta.ifield_lowering_info = def->field_info;
145 } else if (def->opcode >= Instruction::SGET && def->opcode <= Instruction::SPUT_SHORT) {
Vladimir Markoe39c54e2014-09-22 14:50:02 +0100146 ASSERT_LT(def->field_info, cu_.mir_graph->sfield_lowering_infos_.size());
Vladimir Markobe0e5462014-02-26 11:24:15 +0000147 mir->meta.sfield_lowering_info = def->field_info;
148 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000149 mir->ssa_rep = &ssa_reps_[i];
150 mir->ssa_rep->num_uses = def->num_uses;
151 mir->ssa_rep->uses = const_cast<int32_t*>(def->uses); // Not modified by LVN.
152 mir->ssa_rep->fp_use = nullptr; // Not used by LVN.
153 mir->ssa_rep->num_defs = def->num_defs;
154 mir->ssa_rep->defs = const_cast<int32_t*>(def->defs); // Not modified by LVN.
155 mir->ssa_rep->fp_def = nullptr; // Not used by LVN.
156 mir->dalvikInsn.opcode = def->opcode;
157 mir->offset = i; // LVN uses offset only for debug output
Vladimir Markof59f18b2014-02-17 15:53:57 +0000158 mir->optimization_flags = 0u;
159
160 if (i != 0u) {
161 mirs_[i - 1u].next = mir;
162 }
163 }
164 mirs_[count - 1u].next = nullptr;
165 }
166
167 template <size_t count>
168 void PrepareMIRs(const MIRDef (&defs)[count]) {
169 DoPrepareMIRs(defs, count);
170 }
171
Vladimir Markof418f322014-07-09 14:45:36 +0100172 void MakeSFieldUninitialized(uint32_t sfield_index) {
Vladimir Markoe39c54e2014-09-22 14:50:02 +0100173 CHECK_LT(sfield_index, cu_.mir_graph->sfield_lowering_infos_.size());
174 cu_.mir_graph->sfield_lowering_infos_[sfield_index].flags_ &=
Vladimir Markof418f322014-07-09 14:45:36 +0100175 ~MirSFieldLoweringInfo::kFlagIsInitialized;
176 }
177
Vladimir Markof59f18b2014-02-17 15:53:57 +0000178 void PerformLVN() {
179 value_names_.resize(mir_count_);
180 for (size_t i = 0; i != mir_count_; ++i) {
Vladimir Marko83cc7ae2014-02-12 18:02:05 +0000181 value_names_[i] = lvn_->GetValueNumber(&mirs_[i]);
Vladimir Markof59f18b2014-02-17 15:53:57 +0000182 }
Vladimir Marko95a05972014-05-30 10:01:32 +0100183 EXPECT_TRUE(gvn_->Good());
Vladimir Markof59f18b2014-02-17 15:53:57 +0000184 }
185
Vladimir Marko83cc7ae2014-02-12 18:02:05 +0000186 LocalValueNumberingTest()
187 : pool_(),
188 cu_(&pool_),
189 mir_count_(0u),
190 mirs_(nullptr),
Vladimir Marko95a05972014-05-30 10:01:32 +0100191 ssa_reps_(),
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100192 allocator_(),
Vladimir Marko95a05972014-05-30 10:01:32 +0100193 gvn_(),
194 lvn_(),
195 value_names_() {
Vladimir Markof59f18b2014-02-17 15:53:57 +0000196 cu_.mir_graph.reset(new MIRGraph(&cu_, &cu_.arena));
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100197 allocator_.reset(ScopedArenaAllocator::Create(&cu_.arena_stack));
Vladimir Marko415ac882014-09-30 18:09:14 +0100198 gvn_.reset(new (allocator_.get()) GlobalValueNumbering(&cu_, allocator_.get(),
199 GlobalValueNumbering::kModeLvn));
Vladimir Markob19955d2014-07-29 12:04:10 +0100200 lvn_.reset(new (allocator_.get()) LocalValueNumbering(gvn_.get(), 0u, allocator_.get()));
Vladimir Markof59f18b2014-02-17 15:53:57 +0000201 }
202
203 ArenaPool pool_;
204 CompilationUnit cu_;
205 size_t mir_count_;
206 MIR* mirs_;
207 std::vector<SSARepresentation> ssa_reps_;
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100208 std::unique_ptr<ScopedArenaAllocator> allocator_;
Vladimir Marko95a05972014-05-30 10:01:32 +0100209 std::unique_ptr<GlobalValueNumbering> gvn_;
Ian Rogers700a4022014-05-19 16:49:03 -0700210 std::unique_ptr<LocalValueNumbering> lvn_;
Vladimir Marko95a05972014-05-30 10:01:32 +0100211 std::vector<uint16_t> value_names_;
Vladimir Markof59f18b2014-02-17 15:53:57 +0000212};
213
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100214TEST_F(LocalValueNumberingTest, IGetIGetInvokeIGet) {
Vladimir Markof59f18b2014-02-17 15:53:57 +0000215 static const IFieldDef ifields[] = {
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100216 { 1u, 1u, 1u, false },
Vladimir Markof59f18b2014-02-17 15:53:57 +0000217 };
218 static const MIRDef mirs[] = {
219 DEF_IGET(Instruction::IGET, 0u, 10u, 0u),
220 DEF_IGET(Instruction::IGET, 1u, 10u, 0u),
221 DEF_INVOKE1(Instruction::INVOKE_VIRTUAL, 11u),
222 DEF_IGET(Instruction::IGET, 2u, 10u, 0u),
223 };
224
225 PrepareIFields(ifields);
226 PrepareMIRs(mirs);
227 PerformLVN();
228 ASSERT_EQ(value_names_.size(), 4u);
229 EXPECT_EQ(value_names_[0], value_names_[1]);
230 EXPECT_NE(value_names_[0], value_names_[3]);
231 EXPECT_EQ(mirs_[0].optimization_flags, 0u);
232 EXPECT_EQ(mirs_[1].optimization_flags, MIR_IGNORE_NULL_CHECK);
233 EXPECT_EQ(mirs_[2].optimization_flags, 0u);
234 EXPECT_EQ(mirs_[3].optimization_flags, MIR_IGNORE_NULL_CHECK);
235}
236
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100237TEST_F(LocalValueNumberingTest, IGetIPutIGetIGetIGet) {
Vladimir Markof59f18b2014-02-17 15:53:57 +0000238 static const IFieldDef ifields[] = {
239 { 1u, 1u, 1u, false },
240 { 2u, 1u, 2u, false },
241 };
242 static const MIRDef mirs[] = {
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100243 DEF_IGET(Instruction::IGET_OBJECT, 0u, 10u, 0u),
244 DEF_IPUT(Instruction::IPUT_OBJECT, 1u, 11u, 0u), // May alias.
245 DEF_IGET(Instruction::IGET_OBJECT, 2u, 10u, 0u),
Vladimir Markof59f18b2014-02-17 15:53:57 +0000246 DEF_IGET(Instruction::IGET, 3u, 0u, 1u),
247 DEF_IGET(Instruction::IGET, 4u, 2u, 1u),
248 };
249
250 PrepareIFields(ifields);
251 PrepareMIRs(mirs);
252 PerformLVN();
253 ASSERT_EQ(value_names_.size(), 5u);
254 EXPECT_NE(value_names_[0], value_names_[2]);
255 EXPECT_NE(value_names_[3], value_names_[4]);
Vladimir Marko95a05972014-05-30 10:01:32 +0100256 for (size_t i = 0; i != arraysize(mirs); ++i) {
257 EXPECT_EQ((i == 2u) ? MIR_IGNORE_NULL_CHECK : 0,
258 mirs_[i].optimization_flags) << i;
259 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000260}
261
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100262TEST_F(LocalValueNumberingTest, UniquePreserve1) {
Vladimir Markof59f18b2014-02-17 15:53:57 +0000263 static const IFieldDef ifields[] = {
264 { 1u, 1u, 1u, false },
265 };
266 static const MIRDef mirs[] = {
267 DEF_UNIQUE_REF(Instruction::NEW_INSTANCE, 10u),
268 DEF_IGET(Instruction::IGET, 0u, 10u, 0u),
269 DEF_IPUT(Instruction::IPUT, 1u, 11u, 0u), // No aliasing since 10u is unique.
270 DEF_IGET(Instruction::IGET, 2u, 10u, 0u),
271 };
272
273 PrepareIFields(ifields);
274 PrepareMIRs(mirs);
275 PerformLVN();
276 ASSERT_EQ(value_names_.size(), 4u);
277 EXPECT_EQ(value_names_[1], value_names_[3]);
Vladimir Marko95a05972014-05-30 10:01:32 +0100278 for (size_t i = 0; i != arraysize(mirs); ++i) {
279 EXPECT_EQ((i == 1u || i == 3u) ? MIR_IGNORE_NULL_CHECK : 0,
280 mirs_[i].optimization_flags) << i;
281 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000282}
283
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100284TEST_F(LocalValueNumberingTest, UniquePreserve2) {
Vladimir Markof59f18b2014-02-17 15:53:57 +0000285 static const IFieldDef ifields[] = {
286 { 1u, 1u, 1u, false },
287 };
288 static const MIRDef mirs[] = {
289 DEF_UNIQUE_REF(Instruction::NEW_INSTANCE, 11u),
290 DEF_IGET(Instruction::IGET, 0u, 10u, 0u),
291 DEF_IPUT(Instruction::IPUT, 1u, 11u, 0u), // No aliasing since 11u is unique.
292 DEF_IGET(Instruction::IGET, 2u, 10u, 0u),
293 };
294
295 PrepareIFields(ifields);
296 PrepareMIRs(mirs);
297 PerformLVN();
298 ASSERT_EQ(value_names_.size(), 4u);
299 EXPECT_EQ(value_names_[1], value_names_[3]);
Vladimir Marko95a05972014-05-30 10:01:32 +0100300 for (size_t i = 0; i != arraysize(mirs); ++i) {
301 EXPECT_EQ((i == 2u || i == 3u) ? MIR_IGNORE_NULL_CHECK : 0,
302 mirs_[i].optimization_flags) << i;
303 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000304}
305
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100306TEST_F(LocalValueNumberingTest, UniquePreserveAndEscape) {
Vladimir Markof59f18b2014-02-17 15:53:57 +0000307 static const IFieldDef ifields[] = {
308 { 1u, 1u, 1u, false },
309 };
310 static const MIRDef mirs[] = {
311 DEF_UNIQUE_REF(Instruction::NEW_INSTANCE, 10u),
312 DEF_IGET(Instruction::IGET, 0u, 10u, 0u),
313 DEF_INVOKE1(Instruction::INVOKE_VIRTUAL, 11u), // 10u still unique.
314 DEF_IGET(Instruction::IGET, 2u, 10u, 0u),
315 DEF_INVOKE1(Instruction::INVOKE_VIRTUAL, 10u), // 10u not unique anymore.
316 DEF_IGET(Instruction::IGET, 3u, 10u, 0u),
317 };
318
319 PrepareIFields(ifields);
320 PrepareMIRs(mirs);
321 PerformLVN();
322 ASSERT_EQ(value_names_.size(), 6u);
323 EXPECT_EQ(value_names_[1], value_names_[3]);
324 EXPECT_NE(value_names_[1], value_names_[5]);
Vladimir Marko95a05972014-05-30 10:01:32 +0100325 for (size_t i = 0; i != arraysize(mirs); ++i) {
326 EXPECT_EQ((i == 1u || i == 3u || i == 4u || i == 5u) ? MIR_IGNORE_NULL_CHECK : 0,
327 mirs_[i].optimization_flags) << i;
328 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000329}
Vladimir Markof59f18b2014-02-17 15:53:57 +0000330
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100331TEST_F(LocalValueNumberingTest, Volatile) {
Vladimir Markof59f18b2014-02-17 15:53:57 +0000332 static const IFieldDef ifields[] = {
333 { 1u, 1u, 1u, false },
334 { 2u, 1u, 2u, true },
335 };
336 static const MIRDef mirs[] = {
337 DEF_IGET(Instruction::IGET, 0u, 10u, 1u), // Volatile.
338 DEF_IGET(Instruction::IGET, 1u, 0u, 0u), // Non-volatile.
339 DEF_IGET(Instruction::IGET, 2u, 10u, 1u), // Volatile.
340 DEF_IGET(Instruction::IGET, 3u, 2u, 1u), // Non-volatile.
Vladimir Markofa236452014-09-29 17:58:10 +0100341 DEF_IGET(Instruction::IGET, 4u, 0u, 0u), // Non-volatile.
Vladimir Markof59f18b2014-02-17 15:53:57 +0000342 };
343
344 PrepareIFields(ifields);
345 PrepareMIRs(mirs);
346 PerformLVN();
Vladimir Markofa236452014-09-29 17:58:10 +0100347 ASSERT_EQ(value_names_.size(), 5u);
Vladimir Markof59f18b2014-02-17 15:53:57 +0000348 EXPECT_NE(value_names_[0], value_names_[2]); // Volatile has always different value name.
349 EXPECT_NE(value_names_[1], value_names_[3]); // Used different base because of volatile.
Vladimir Markofa236452014-09-29 17:58:10 +0100350 EXPECT_NE(value_names_[1], value_names_[4]); // Not guaranteed to be the same after "acquire".
351
Vladimir Marko95a05972014-05-30 10:01:32 +0100352 for (size_t i = 0; i != arraysize(mirs); ++i) {
Vladimir Markofa236452014-09-29 17:58:10 +0100353 EXPECT_EQ((i == 2u || i == 4u) ? MIR_IGNORE_NULL_CHECK : 0,
Vladimir Marko95a05972014-05-30 10:01:32 +0100354 mirs_[i].optimization_flags) << i;
355 }
Vladimir Markof59f18b2014-02-17 15:53:57 +0000356}
357
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100358TEST_F(LocalValueNumberingTest, UnresolvedIField) {
359 static const IFieldDef ifields[] = {
360 { 1u, 1u, 1u, false }, // Resolved field #1.
361 { 2u, 1u, 2u, false }, // Resolved field #2.
362 { 3u, 0u, 0u, false }, // Unresolved field.
363 };
364 static const MIRDef mirs[] = {
365 DEF_UNIQUE_REF(Instruction::NEW_INSTANCE, 20u),
366 DEF_IGET(Instruction::IGET, 1u, 20u, 0u), // Resolved field #1, unique object.
367 DEF_IGET(Instruction::IGET, 2u, 21u, 0u), // Resolved field #1.
368 DEF_IGET_WIDE(Instruction::IGET_WIDE, 3u, 21u, 1u), // Resolved field #2.
Vladimir Markofa236452014-09-29 17:58:10 +0100369 DEF_IGET(Instruction::IGET, 4u, 22u, 2u), // Unresolved IGET can be "acquire".
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100370 DEF_IGET(Instruction::IGET, 5u, 20u, 0u), // Resolved field #1, unique object.
371 DEF_IGET(Instruction::IGET, 6u, 21u, 0u), // Resolved field #1.
372 DEF_IGET_WIDE(Instruction::IGET_WIDE, 7u, 21u, 1u), // Resolved field #2.
Vladimir Marko95a05972014-05-30 10:01:32 +0100373 DEF_IPUT(Instruction::IPUT, 8u, 22u, 2u), // IPUT clobbers field #1 (#2 is wide).
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100374 DEF_IGET(Instruction::IGET, 9u, 20u, 0u), // Resolved field #1, unique object.
375 DEF_IGET(Instruction::IGET, 10u, 21u, 0u), // Resolved field #1, new value name.
376 DEF_IGET_WIDE(Instruction::IGET_WIDE, 11u, 21u, 1u), // Resolved field #2.
Vladimir Marko95a05972014-05-30 10:01:32 +0100377 DEF_IGET_WIDE(Instruction::IGET_WIDE, 12u, 20u, 1u), // Resolved field #2, unique object.
378 DEF_IPUT(Instruction::IPUT, 13u, 20u, 2u), // IPUT clobbers field #1 (#2 is wide).
379 DEF_IGET(Instruction::IGET, 14u, 20u, 0u), // Resolved field #1, unique object.
380 DEF_IGET_WIDE(Instruction::IGET_WIDE, 15u, 20u, 1u), // Resolved field #2, unique object.
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100381 };
382
383 PrepareIFields(ifields);
384 PrepareMIRs(mirs);
385 PerformLVN();
Vladimir Marko95a05972014-05-30 10:01:32 +0100386 ASSERT_EQ(value_names_.size(), 16u);
Vladimir Markofa236452014-09-29 17:58:10 +0100387 // Unresolved field is potentially volatile, so we need to adhere to the volatile semantics.
388 EXPECT_EQ(value_names_[1], value_names_[5]); // Unique object.
389 EXPECT_NE(value_names_[2], value_names_[6]); // Not guaranteed to be the same after "acquire".
390 EXPECT_NE(value_names_[3], value_names_[7]); // Not guaranteed to be the same after "acquire".
391 EXPECT_EQ(value_names_[1], value_names_[9]); // Unique object.
392 EXPECT_NE(value_names_[6], value_names_[10]); // This aliased with unresolved IPUT.
393 EXPECT_EQ(value_names_[7], value_names_[11]); // Still the same after "release".
394 EXPECT_EQ(value_names_[12], value_names_[15]); // Still the same after "release".
395 EXPECT_NE(value_names_[1], value_names_[14]); // This aliased with unresolved IPUT.
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100396 EXPECT_EQ(mirs_[0].optimization_flags, 0u);
397 EXPECT_EQ(mirs_[1].optimization_flags, MIR_IGNORE_NULL_CHECK);
398 EXPECT_EQ(mirs_[2].optimization_flags, 0u);
399 EXPECT_EQ(mirs_[3].optimization_flags, MIR_IGNORE_NULL_CHECK);
400 EXPECT_EQ(mirs_[4].optimization_flags, 0u);
401 for (size_t i = 5u; i != mir_count_; ++i) {
Vladimir Marko95a05972014-05-30 10:01:32 +0100402 EXPECT_EQ((i == 1u || i == 3u || i >=5u) ? MIR_IGNORE_NULL_CHECK : 0,
403 mirs_[i].optimization_flags) << i;
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100404 }
405}
406
407TEST_F(LocalValueNumberingTest, UnresolvedSField) {
408 static const SFieldDef sfields[] = {
409 { 1u, 1u, 1u, false }, // Resolved field #1.
410 { 2u, 1u, 2u, false }, // Resolved field #2.
411 { 3u, 0u, 0u, false }, // Unresolved field.
412 };
413 static const MIRDef mirs[] = {
414 DEF_SGET(Instruction::SGET, 0u, 0u), // Resolved field #1.
415 DEF_SGET_WIDE(Instruction::SGET_WIDE, 1u, 1u), // Resolved field #2.
Vladimir Markofa236452014-09-29 17:58:10 +0100416 DEF_SGET(Instruction::SGET, 2u, 2u), // Unresolved SGET can be "acquire".
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100417 DEF_SGET(Instruction::SGET, 3u, 0u), // Resolved field #1.
418 DEF_SGET_WIDE(Instruction::SGET_WIDE, 4u, 1u), // Resolved field #2.
419 DEF_SPUT(Instruction::SPUT, 5u, 2u), // SPUT clobbers field #1 (#2 is wide).
420 DEF_SGET(Instruction::SGET, 6u, 0u), // Resolved field #1.
421 DEF_SGET_WIDE(Instruction::SGET_WIDE, 7u, 1u), // Resolved field #2.
422 };
423
424 PrepareSFields(sfields);
425 PrepareMIRs(mirs);
426 PerformLVN();
427 ASSERT_EQ(value_names_.size(), 8u);
Vladimir Markofa236452014-09-29 17:58:10 +0100428 // Unresolved field is potentially volatile, so we need to adhere to the volatile semantics.
429 EXPECT_NE(value_names_[0], value_names_[3]); // Not guaranteed to be the same after "acquire".
430 EXPECT_NE(value_names_[1], value_names_[4]); // Not guaranteed to be the same after "acquire".
431 EXPECT_NE(value_names_[3], value_names_[6]); // This aliased with unresolved IPUT.
432 EXPECT_EQ(value_names_[4], value_names_[7]); // Still the same after "release".
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100433 for (size_t i = 0u; i != mir_count_; ++i) {
Vladimir Marko95a05972014-05-30 10:01:32 +0100434 EXPECT_EQ(0, mirs_[i].optimization_flags) << i;
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100435 }
436}
437
Vladimir Marko95a05972014-05-30 10:01:32 +0100438TEST_F(LocalValueNumberingTest, UninitializedSField) {
439 static const IFieldDef ifields[] = {
440 { 1u, 1u, 1u, false }, // Resolved field #1.
441 };
442 static const SFieldDef sfields[] = {
443 { 1u, 1u, 1u, false }, // Resolved field #1.
444 { 2u, 1u, 2u, false }, // Resolved field #2; uninitialized.
445 };
446 static const MIRDef mirs[] = {
447 DEF_UNIQUE_REF(Instruction::NEW_INSTANCE, 200u),
448 DEF_IGET(Instruction::IGET, 1u, 100u, 0u),
449 DEF_IGET(Instruction::IGET, 2u, 200u, 0u),
450 DEF_SGET(Instruction::SGET, 3u, 0u),
451 DEF_SGET(Instruction::SGET, 4u, 1u), // Can call <clinit>().
452 DEF_IGET(Instruction::IGET, 5u, 100u, 0u), // Differs from 1u.
453 DEF_IGET(Instruction::IGET, 6u, 200u, 0u), // Same as 2u.
454 DEF_SGET(Instruction::SGET, 7u, 0u), // Differs from 3u.
455 };
456
457 PrepareIFields(ifields);
458 PrepareSFields(sfields);
459 MakeSFieldUninitialized(1u);
460 PrepareMIRs(mirs);
461 PerformLVN();
462 ASSERT_EQ(value_names_.size(), 8u);
463 EXPECT_NE(value_names_[1], value_names_[5]);
464 EXPECT_EQ(value_names_[2], value_names_[6]);
465 EXPECT_NE(value_names_[3], value_names_[7]);
466}
467
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100468TEST_F(LocalValueNumberingTest, ConstString) {
469 static const MIRDef mirs[] = {
470 DEF_CONST_STRING(Instruction::CONST_STRING, 0u, 0u),
471 DEF_CONST_STRING(Instruction::CONST_STRING, 1u, 0u),
472 DEF_CONST_STRING(Instruction::CONST_STRING, 2u, 2u),
473 DEF_CONST_STRING(Instruction::CONST_STRING, 3u, 0u),
474 DEF_INVOKE1(Instruction::INVOKE_DIRECT, 2u),
475 DEF_CONST_STRING(Instruction::CONST_STRING, 4u, 2u),
476 };
477
478 PrepareMIRs(mirs);
479 PerformLVN();
480 ASSERT_EQ(value_names_.size(), 6u);
481 EXPECT_EQ(value_names_[1], value_names_[0]);
482 EXPECT_NE(value_names_[2], value_names_[0]);
483 EXPECT_EQ(value_names_[3], value_names_[0]);
484 EXPECT_EQ(value_names_[5], value_names_[2]);
485}
486
487TEST_F(LocalValueNumberingTest, SameValueInDifferentMemoryLocations) {
488 static const IFieldDef ifields[] = {
489 { 1u, 1u, 1u, false },
490 { 2u, 1u, 2u, false },
491 };
492 static const SFieldDef sfields[] = {
493 { 3u, 1u, 3u, false },
494 };
495 static const MIRDef mirs[] = {
Vladimir Marko95a05972014-05-30 10:01:32 +0100496 DEF_UNIQUE_REF(Instruction::NEW_ARRAY, 201u),
497 DEF_IGET(Instruction::IGET, 0u, 100u, 0u),
498 DEF_IPUT(Instruction::IPUT, 0u, 100u, 1u),
499 DEF_IPUT(Instruction::IPUT, 0u, 101u, 1u),
500 DEF_APUT(Instruction::APUT, 0u, 200u, 300u),
501 DEF_APUT(Instruction::APUT, 0u, 200u, 301u),
502 DEF_APUT(Instruction::APUT, 0u, 201u, 300u),
503 DEF_APUT(Instruction::APUT, 0u, 201u, 301u),
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100504 DEF_SPUT(Instruction::SPUT, 0u, 0u),
Vladimir Marko95a05972014-05-30 10:01:32 +0100505 DEF_IGET(Instruction::IGET, 9u, 100u, 0u),
506 DEF_IGET(Instruction::IGET, 10u, 100u, 1u),
507 DEF_IGET(Instruction::IGET, 11u, 101u, 1u),
508 DEF_AGET(Instruction::AGET, 12u, 200u, 300u),
509 DEF_AGET(Instruction::AGET, 13u, 200u, 301u),
510 DEF_AGET(Instruction::AGET, 14u, 201u, 300u),
511 DEF_AGET(Instruction::AGET, 15u, 201u, 301u),
512 DEF_SGET(Instruction::SGET, 16u, 0u),
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100513 };
514
515 PrepareIFields(ifields);
516 PrepareSFields(sfields);
517 PrepareMIRs(mirs);
518 PerformLVN();
Vladimir Marko95a05972014-05-30 10:01:32 +0100519 ASSERT_EQ(value_names_.size(), 17u);
520 for (size_t i = 9; i != arraysize(mirs); ++i) {
521 EXPECT_EQ(value_names_[1], value_names_[i]) << i;
522 }
523 for (size_t i = 0; i != arraysize(mirs); ++i) {
524 int expected_flags =
525 ((i == 2u || (i >= 5u && i <= 7u) || (i >= 9u && i <= 15u)) ? MIR_IGNORE_NULL_CHECK : 0) |
526 ((i >= 12u && i <= 15u) ? MIR_IGNORE_RANGE_CHECK : 0);
527 EXPECT_EQ(expected_flags, mirs_[i].optimization_flags) << i;
528 }
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100529}
530
531TEST_F(LocalValueNumberingTest, UniqueArrayAliasing) {
532 static const MIRDef mirs[] = {
533 DEF_UNIQUE_REF(Instruction::NEW_ARRAY, 20u),
534 DEF_AGET(Instruction::AGET, 1u, 20u, 40u),
535 DEF_APUT(Instruction::APUT, 2u, 20u, 41u), // May alias with index for sreg 40u.
536 DEF_AGET(Instruction::AGET, 3u, 20u, 40u),
537 };
538
539 PrepareMIRs(mirs);
540 PerformLVN();
541 ASSERT_EQ(value_names_.size(), 4u);
542 EXPECT_NE(value_names_[1], value_names_[3]);
Vladimir Marko95a05972014-05-30 10:01:32 +0100543 for (size_t i = 0; i != arraysize(mirs); ++i) {
544 int expected_flags =
545 ((i >= 1u) ? MIR_IGNORE_NULL_CHECK : 0) |
546 ((i == 3u) ? MIR_IGNORE_RANGE_CHECK : 0);
547 EXPECT_EQ(expected_flags, mirs_[i].optimization_flags) << i;
548 }
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100549}
550
551TEST_F(LocalValueNumberingTest, EscapingRefs) {
552 static const IFieldDef ifields[] = {
553 { 1u, 1u, 1u, false }, // Field #1.
554 { 2u, 1u, 2u, false }, // Field #2.
555 { 3u, 1u, 3u, false }, // Reference field for storing escaping refs.
556 { 4u, 1u, 4u, false }, // Wide.
557 { 5u, 0u, 0u, false }, // Unresolved field, int.
558 { 6u, 0u, 0u, false }, // Unresolved field, wide.
559 };
560 static const MIRDef mirs[] = {
561 DEF_UNIQUE_REF(Instruction::NEW_INSTANCE, 20u),
562 DEF_IGET(Instruction::IGET, 1u, 20u, 0u),
563 DEF_IGET(Instruction::IGET, 2u, 20u, 1u),
564 DEF_IPUT(Instruction::IPUT_OBJECT, 20u, 30u, 2u), // Ref escapes.
565 DEF_IGET(Instruction::IGET, 4u, 20u, 0u),
566 DEF_IGET(Instruction::IGET, 5u, 20u, 1u),
567 DEF_IPUT(Instruction::IPUT, 6u, 31u, 0u), // May alias with field #1.
568 DEF_IGET(Instruction::IGET, 7u, 20u, 0u), // New value.
569 DEF_IGET(Instruction::IGET, 8u, 20u, 1u), // Still the same.
570 DEF_IPUT_WIDE(Instruction::IPUT_WIDE, 9u, 31u, 3u), // No aliasing, different type.
571 DEF_IGET(Instruction::IGET, 10u, 20u, 0u),
572 DEF_IGET(Instruction::IGET, 11u, 20u, 1u),
573 DEF_IPUT_WIDE(Instruction::IPUT_WIDE, 12u, 31u, 5u), // No aliasing, different type.
574 DEF_IGET(Instruction::IGET, 13u, 20u, 0u),
575 DEF_IGET(Instruction::IGET, 14u, 20u, 1u),
576 DEF_IPUT(Instruction::IPUT, 15u, 31u, 4u), // Aliasing, same type.
577 DEF_IGET(Instruction::IGET, 16u, 20u, 0u),
578 DEF_IGET(Instruction::IGET, 17u, 20u, 1u),
579 };
580
581 PrepareIFields(ifields);
582 PrepareMIRs(mirs);
583 PerformLVN();
584 ASSERT_EQ(value_names_.size(), 18u);
585 EXPECT_EQ(value_names_[1], value_names_[4]);
586 EXPECT_EQ(value_names_[2], value_names_[5]);
587 EXPECT_NE(value_names_[4], value_names_[7]); // New value.
588 EXPECT_EQ(value_names_[5], value_names_[8]);
589 EXPECT_EQ(value_names_[7], value_names_[10]);
590 EXPECT_EQ(value_names_[8], value_names_[11]);
591 EXPECT_EQ(value_names_[10], value_names_[13]);
592 EXPECT_EQ(value_names_[11], value_names_[14]);
593 EXPECT_NE(value_names_[13], value_names_[16]); // New value.
594 EXPECT_NE(value_names_[14], value_names_[17]); // New value.
595 for (size_t i = 0u; i != mir_count_; ++i) {
Vladimir Marko95a05972014-05-30 10:01:32 +0100596 int expected = (i != 0u && i != 3u && i != 6u) ? MIR_IGNORE_NULL_CHECK : 0;
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100597 EXPECT_EQ(expected, mirs_[i].optimization_flags) << i;
598 }
599}
600
601TEST_F(LocalValueNumberingTest, EscapingArrayRefs) {
602 static const MIRDef mirs[] = {
603 DEF_UNIQUE_REF(Instruction::NEW_ARRAY, 20u),
604 DEF_AGET(Instruction::AGET, 1u, 20u, 40u),
605 DEF_AGET(Instruction::AGET, 2u, 20u, 41u),
606 DEF_APUT(Instruction::APUT_OBJECT, 20u, 30u, 42u), // Array ref escapes.
607 DEF_AGET(Instruction::AGET, 4u, 20u, 40u),
608 DEF_AGET(Instruction::AGET, 5u, 20u, 41u),
609 DEF_APUT_WIDE(Instruction::APUT_WIDE, 6u, 31u, 43u), // No aliasing, different type.
610 DEF_AGET(Instruction::AGET, 7u, 20u, 40u),
611 DEF_AGET(Instruction::AGET, 8u, 20u, 41u),
612 DEF_APUT(Instruction::APUT, 9u, 32u, 40u), // May alias with all elements.
613 DEF_AGET(Instruction::AGET, 10u, 20u, 40u), // New value (same index name).
614 DEF_AGET(Instruction::AGET, 11u, 20u, 41u), // New value (different index name).
615 };
616
617 PrepareMIRs(mirs);
618 PerformLVN();
619 ASSERT_EQ(value_names_.size(), 12u);
620 EXPECT_EQ(value_names_[1], value_names_[4]);
621 EXPECT_EQ(value_names_[2], value_names_[5]);
622 EXPECT_EQ(value_names_[4], value_names_[7]);
623 EXPECT_EQ(value_names_[5], value_names_[8]);
624 EXPECT_NE(value_names_[7], value_names_[10]); // New value.
625 EXPECT_NE(value_names_[8], value_names_[11]); // New value.
626 for (size_t i = 0u; i != mir_count_; ++i) {
627 int expected =
628 ((i != 0u && i != 3u && i != 6u && i != 9u) ? MIR_IGNORE_NULL_CHECK : 0u) |
629 ((i >= 4 && i != 6u && i != 9u) ? MIR_IGNORE_RANGE_CHECK : 0u);
630 EXPECT_EQ(expected, mirs_[i].optimization_flags) << i;
631 }
632}
633
634TEST_F(LocalValueNumberingTest, StoringSameValueKeepsMemoryVersion) {
635 static const IFieldDef ifields[] = {
636 { 1u, 1u, 1u, false },
Vladimir Marko95a05972014-05-30 10:01:32 +0100637 { 2u, 1u, 2u, false },
638 };
639 static const SFieldDef sfields[] = {
640 { 2u, 1u, 2u, false },
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100641 };
642 static const MIRDef mirs[] = {
Vladimir Marko95a05972014-05-30 10:01:32 +0100643 DEF_IGET(Instruction::IGET, 0u, 30u, 0u),
644 DEF_IGET(Instruction::IGET, 1u, 31u, 0u),
645 DEF_IPUT(Instruction::IPUT, 1u, 31u, 0u), // Store the same value.
646 DEF_IGET(Instruction::IGET, 3u, 30u, 0u),
647 DEF_AGET(Instruction::AGET, 4u, 32u, 40u),
648 DEF_AGET(Instruction::AGET, 5u, 33u, 40u),
649 DEF_APUT(Instruction::APUT, 5u, 33u, 40u), // Store the same value.
650 DEF_AGET(Instruction::AGET, 7u, 32u, 40u),
651 DEF_SGET(Instruction::SGET, 8u, 0u),
652 DEF_SPUT(Instruction::SPUT, 8u, 0u), // Store the same value.
653 DEF_SGET(Instruction::SGET, 10u, 0u),
654 DEF_UNIQUE_REF(Instruction::NEW_INSTANCE, 50u), // Test with unique references.
655 { Instruction::FILLED_NEW_ARRAY, 0, 0u, 2, { 12u, 13u }, 0, { } },
656 DEF_UNIQUE_REF(Instruction::MOVE_RESULT_OBJECT, 51u),
657 DEF_IGET(Instruction::IGET, 14u, 50u, 0u),
658 DEF_IGET(Instruction::IGET, 15u, 50u, 1u),
659 DEF_IPUT(Instruction::IPUT, 15u, 50u, 1u), // Store the same value.
660 DEF_IGET(Instruction::IGET, 17u, 50u, 0u),
661 DEF_AGET(Instruction::AGET, 18u, 51u, 40u),
662 DEF_AGET(Instruction::AGET, 19u, 51u, 41u),
663 DEF_APUT(Instruction::APUT, 19u, 51u, 41u), // Store the same value.
664 DEF_AGET(Instruction::AGET, 21u, 51u, 40u),
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100665 };
666
667 PrepareIFields(ifields);
Vladimir Marko95a05972014-05-30 10:01:32 +0100668 PrepareSFields(sfields);
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100669 PrepareMIRs(mirs);
670 PerformLVN();
Vladimir Marko95a05972014-05-30 10:01:32 +0100671 ASSERT_EQ(value_names_.size(), 22u);
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100672 EXPECT_NE(value_names_[0], value_names_[1]);
673 EXPECT_EQ(value_names_[0], value_names_[3]);
674 EXPECT_NE(value_names_[4], value_names_[5]);
675 EXPECT_EQ(value_names_[4], value_names_[7]);
Vladimir Marko95a05972014-05-30 10:01:32 +0100676 EXPECT_EQ(value_names_[8], value_names_[10]);
677 EXPECT_NE(value_names_[14], value_names_[15]);
678 EXPECT_EQ(value_names_[14], value_names_[17]);
679 EXPECT_NE(value_names_[18], value_names_[19]);
680 EXPECT_EQ(value_names_[18], value_names_[21]);
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100681 for (size_t i = 0u; i != mir_count_; ++i) {
682 int expected =
Vladimir Marko95a05972014-05-30 10:01:32 +0100683 ((i == 2u || i == 3u || i == 6u || i == 7u || (i >= 14u)) ? MIR_IGNORE_NULL_CHECK : 0u) |
684 ((i == 6u || i == 7u || i >= 20u) ? MIR_IGNORE_RANGE_CHECK : 0u);
685 EXPECT_EQ(expected, mirs_[i].optimization_flags) << i;
686 }
687}
688
689TEST_F(LocalValueNumberingTest, FilledNewArrayTracking) {
690 if (!kLocalValueNumberingEnableFilledNewArrayTracking) {
691 // Feature disabled.
692 return;
693 }
694 static const MIRDef mirs[] = {
695 DEF_CONST(Instruction::CONST, 0u, 100),
696 DEF_CONST(Instruction::CONST, 1u, 200),
697 { Instruction::FILLED_NEW_ARRAY, 0, 0u, 2, { 0u, 1u }, 0, { } },
698 DEF_UNIQUE_REF(Instruction::MOVE_RESULT_OBJECT, 10u),
699 DEF_CONST(Instruction::CONST, 20u, 0),
700 DEF_CONST(Instruction::CONST, 21u, 1),
701 DEF_AGET(Instruction::AGET, 6u, 10u, 20u),
702 DEF_AGET(Instruction::AGET, 7u, 10u, 21u),
703 };
704
705 PrepareMIRs(mirs);
706 PerformLVN();
707 ASSERT_EQ(value_names_.size(), 8u);
708 EXPECT_EQ(value_names_[0], value_names_[6]);
709 EXPECT_EQ(value_names_[1], value_names_[7]);
710 for (size_t i = 0u; i != mir_count_; ++i) {
711 int expected = (i == 6u || i == 7u) ? (MIR_IGNORE_NULL_CHECK | MIR_IGNORE_RANGE_CHECK) : 0u;
Vladimir Marko2ac01fc2014-05-22 12:09:08 +0100712 EXPECT_EQ(expected, mirs_[i].optimization_flags) << i;
713 }
714}
715
Vladimir Markof418f322014-07-09 14:45:36 +0100716TEST_F(LocalValueNumberingTest, ClInitOnSget) {
717 static const SFieldDef sfields[] = {
718 { 0u, 1u, 0u, false },
719 { 1u, 2u, 1u, false },
720 };
721 static const MIRDef mirs[] = {
722 DEF_SGET(Instruction::SGET_OBJECT, 0u, 0u),
723 DEF_AGET(Instruction::AGET, 1u, 0u, 100u),
724 DEF_SGET(Instruction::SGET_OBJECT, 2u, 1u),
725 DEF_SGET(Instruction::SGET_OBJECT, 3u, 0u),
726 DEF_AGET(Instruction::AGET, 4u, 3u, 100u),
727 };
728
729 PrepareSFields(sfields);
730 MakeSFieldUninitialized(1u);
731 PrepareMIRs(mirs);
732 PerformLVN();
733 ASSERT_EQ(value_names_.size(), 5u);
734 EXPECT_NE(value_names_[0], value_names_[3]);
735}
736
Vladimir Markof59f18b2014-02-17 15:53:57 +0000737} // namespace art