diff options
author | 2015-06-24 14:57:44 +0100 | |
---|---|---|
committer | 2015-06-24 15:02:23 +0100 | |
commit | 8df886b9214802ad689316a1dedb00a6d102555c (patch) | |
tree | 19023e2d50ae464ecac105583384fbb4425d6c8e | |
parent | 1c7f2f10230baf7a549af5bfd16fd3f0fb73ba21 (diff) |
BCE: don't assume a bounds check always gets a HArrayLength.
Deoptimizations may change it to a HPhi.
bug:22056703
Change-Id: I8995209438764dac496ed856782b147ba21f93e5
-rw-r--r-- | compiler/optimizing/bounds_check_elimination.cc | 10 | ||||
-rw-r--r-- | test/499-bce-phi-array-length/src/Main.java | 22 |
2 files changed, 30 insertions, 2 deletions
diff --git a/compiler/optimizing/bounds_check_elimination.cc b/compiler/optimizing/bounds_check_elimination.cc index 900dabea0e..7fe9fb2677 100644 --- a/compiler/optimizing/bounds_check_elimination.cc +++ b/compiler/optimizing/bounds_check_elimination.cc @@ -1388,7 +1388,7 @@ class BCEVisitor : public HGraphVisitor { if (array_length->IsPhi()) { // Input 1 of the phi contains the real array.length once the loop body is // entered. That value will be used for bound analysis. The graph is still - // strickly in SSA form. + // strictly in SSA form. array_length = array_length->AsPhi()->InputAt(1)->AsArrayLength(); } @@ -1780,7 +1780,13 @@ class BCEVisitor : public HGraphVisitor { it != first_constant_index_bounds_check_map_.end(); ++it) { HBoundsCheck* bounds_check = it->second; - HArrayLength* array_length = bounds_check->InputAt(1)->AsArrayLength(); + HInstruction* array_length = bounds_check->InputAt(1); + if (!array_length->IsArrayLength()) { + // Prior deoptimizations may have changed the array length to a phi. + // TODO(mingyao): propagate the range to the phi? + DCHECK(array_length->IsPhi()) << array_length->DebugName(); + continue; + } HIntConstant* lower_bound_const_instr = nullptr; int32_t lower_bound_const = INT_MIN; size_t counter = 0; diff --git a/test/499-bce-phi-array-length/src/Main.java b/test/499-bce-phi-array-length/src/Main.java index c8c84a1a8f..e917bc1f32 100644 --- a/test/499-bce-phi-array-length/src/Main.java +++ b/test/499-bce-phi-array-length/src/Main.java @@ -32,11 +32,33 @@ public class Main { return result; } + public static int bar(int start, int[] array) { + int result = 0; + for (int i = start; i < 3; i++) { + result += array[i]; + for (int j = 0; j < 2; ++j) { + result += array[j]; + // The following operations would lead to BCE wanting to add another + // deoptimization, but it crashed assuming the input of a `HBoundsCheck` + // must be a `HArrayLength`. + result += array[0]; + result += array[1]; + result += array[2]; + } + } + return result; + } + public static void main(String[] args) { int[] a = new int[] { 1, 2, 3, 4, 5 }; int result = foo(1, a); if (result != 11) { throw new Error("Got " + result + ", expected " + 11); } + + result = bar(1, a); + if (result != 35) { + throw new Error("Got " + result + ", expected " + 35); + } } } |