/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* Regression tests for LICM.
*/
public class Main {
static int sA;
//
// We cannot hoist the null check (can throw) above the field
// assignment (has write side effects) because that would result
// in throwing an exception before the assignment is done.
//
/// CHECK-START: void Main.foo(int[]) licm (before)
/// CHECK-DAG: LoadClass loop:<<Loop:B\d+>> outer_loop:none
/// CHECK-DAG: StaticFieldSet loop:<<Loop>> outer_loop:none
/// CHECK-DAG: NullCheck loop:<<Loop>> outer_loop:none
/// CHECK-DAG: ArrayLength loop:<<Loop>> outer_loop:none
//
/// CHECK-START: void Main.foo(int[]) licm (after)
/// CHECK-DAG: LoadClass loop:none
/// CHECK-DAG: StaticFieldSet loop:<<Loop:B\d+>> outer_loop:none
/// CHECK-DAG: NullCheck loop:<<Loop>> outer_loop:none
/// CHECK-DAG: ArrayLength loop:<<Loop>> outer_loop:none
//
/// CHECK-START: void Main.foo(int[]) licm (after)
/// CHECK-NOT: LoadClass loop:{{B\d+}} outer_loop:none
static void foo(int[] arr) {
int j = 0;
do {
sA = 1;
} while (j < arr.length);
}
//
// Similar situation as in foo(), but now a proper induction value
// is assigned to the field inside the do-while loop.
//
/// CHECK-START: void Main.bar(int[]) licm (before)
/// CHECK-DAG: LoadClass loop:<<Loop:B\d+>> outer_loop:none
/// CHECK-DAG: StaticFieldSet loop:<<Loop>> outer_loop:none
/// CHECK-DAG: NullCheck loop:<<Loop>> outer_loop:none
/// CHECK-DAG: ArrayLength loop:<<Loop>> outer_loop:none
//
/// CHECK-START: void Main.bar(int[]) licm (after)
/// CHECK-DAG: LoadClass loop:none
/// CHECK-DAG: StaticFieldSet loop:<<Loop:B\d+>> outer_loop:none
/// CHECK-DAG: NullCheck loop:<<Loop>> outer_loop:none
/// CHECK-DAG: ArrayLength loop:<<Loop>> outer_loop:none
//
/// CHECK-START: void Main.bar(int[]) licm (after)
/// CHECK-NOT: LoadClass loop:{{B\d+}} outer_loop:none
static void bar(int[] arr) {
int j = 0;
do {
j++;
sA = j;
} while (j < arr.length);
}
//
// Similar situation as in bar(), but now an explicit catch
// statement may need the latest value of local j.
//
/// CHECK-START: int Main.catcher(int[]) licm (before)
/// CHECK-DAG: NullCheck loop:<<Loop:B\d+>> outer_loop:none
/// CHECK-DAG: ArrayLength loop:<<Loop>> outer_loop:none
//
/// CHECK-START: int Main.catcher(int[]) licm (after)
/// CHECK-DAG: NullCheck loop:<<Loop:B\d+>> outer_loop:none
/// CHECK-DAG: ArrayLength loop:<<Loop>> outer_loop:none
static int catcher(int[] arr) {
int j = 0;
try {
do {
j++;
} while (j < arr.length);
} catch (NullPointerException e) {
return -j; // flag exception with negative value
}
return j;
}
public static void main(String[] args) {
sA = 0;
try {
foo(null);
throw new Error("Expected NPE");
} catch (NullPointerException e) {
}
expectEquals(1, sA);
sA = 0;
try {
bar(null);
throw new Error("Expected NPE");
} catch (NullPointerException e) {
}
expectEquals(1, sA);
for (int i = 0; i < 5; i++) {
sA = 0;
bar(new int[i]);
expectEquals(i == 0 ? 1 : i, sA);
}
expectEquals(-1, catcher(null));
for (int i = 0; i < 5; i++) {
expectEquals(i == 0 ? 1 : i, catcher(new int[i]));
}
System.out.println("passed");
}
private static void expectEquals(int expected, int result) {
if (expected != result) {
throw new Error("Expected: " + expected + ", found: " + result);
}
}
}