blob: 08a23a7fbcb0661a933352b6bc348221326b24db [file] [log] [blame]
/*
* 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.
*/
public class Main {
public class ExampleObj {
int n1;
int n2;
public ExampleObj(int n1, int n2) {
this.n1 = n1;
this.n2 = n2;
}
}
static int static_variable = 0;
public ExampleObj my_obj;
public static int number1;
public static int number2;
public static volatile int number3;
/// CHECK-START-ARM64: int Main.arrayAccess() scheduler (before)
/// CHECK: <<Const1:i\d+>> IntConstant 1
/// CHECK: <<i0:i\d+>> Phi
/// CHECK: <<res0:i\d+>> Phi
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<i0>>]
/// CHECK: <<res1:i\d+>> Add [<<res0>>,<<ArrayGet1>>]
/// CHECK: <<i1:i\d+>> Add [<<i0>>,<<Const1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<i1>>]
/// CHECK: Add [<<res1>>,<<ArrayGet2>>]
/// CHECK-START-ARM64: int Main.arrayAccess() scheduler (after)
/// CHECK: <<Const1:i\d+>> IntConstant 1
/// CHECK: <<i0:i\d+>> Phi
/// CHECK: <<res0:i\d+>> Phi
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<i0>>]
/// CHECK: <<i1:i\d+>> Add [<<i0>>,<<Const1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<i1>>]
/// CHECK: <<res1:i\d+>> Add [<<res0>>,<<ArrayGet1>>]
/// CHECK: Add [<<res1>>,<<ArrayGet2>>]
public static int arrayAccess() {
int res = 0;
int [] array = new int[10];
for (int i = 0; i < 9; i++) {
res += array[i];
res += array[i + 1];
}
return res;
}
/// CHECK-START-ARM: void Main.arrayAccessVariable(int) scheduler (before)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 2
/// CHECK-DAG: <<Const3:i\d+>> IntConstant -1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Add2:i\d+>> Add [<<Param>>,<<Const2>>]
/// CHECK: <<Add3:i\d+>> Add [<<Param>>,<<Const3>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<Add1>>]
/// CHECK: <<AddArray1:i\d+>> Add [<<ArrayGet1>>,<<Const1>>]
/// CHECK: <<ArraySet1:v\d+>> ArraySet [<<Array>>,<<Add1>>,<<AddArray1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<Add2>>]
/// CHECK: <<AddArray2:i\d+>> Add [<<ArrayGet2>>,<<Const1>>]
/// CHECK: <<ArraySet2:v\d+>> ArraySet [<<Array>>,<<Add2>>,<<AddArray2>>]
/// CHECK: <<ArrayGet3:i\d+>> ArrayGet [<<Array>>,<<Add3>>]
/// CHECK: <<AddArray3:i\d+>> Add [<<ArrayGet3>>,<<Const1>>]
/// CHECK: <<ArraySet3:v\d+>> ArraySet [<<Array>>,<<Add3>>,<<AddArray3>>]
/// CHECK-START-ARM: void Main.arrayAccessVariable(int) scheduler (after)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 2
/// CHECK-DAG: <<Const3:i\d+>> IntConstant -1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Add2:i\d+>> Add [<<Param>>,<<Const2>>]
/// CHECK: <<Add3:i\d+>> Add [<<Param>>,<<Const3>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: ArrayGet [<<Array>>,{{i\d+}}]
/// CHECK: ArrayGet [<<Array>>,{{i\d+}}]
/// CHECK: ArrayGet [<<Array>>,{{i\d+}}]
/// CHECK: Add
/// CHECK: Add
/// CHECK: Add
/// CHECK: ArraySet
/// CHECK: ArraySet
/// CHECK: ArraySet
/// CHECK-START-ARM64: void Main.arrayAccessVariable(int) scheduler (before)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 2
/// CHECK-DAG: <<Const3:i\d+>> IntConstant -1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Add2:i\d+>> Add [<<Param>>,<<Const2>>]
/// CHECK: <<Add3:i\d+>> Add [<<Param>>,<<Const3>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<Add1>>]
/// CHECK: <<AddArray1:i\d+>> Add [<<ArrayGet1>>,<<Const1>>]
/// CHECK: <<ArraySet1:v\d+>> ArraySet [<<Array>>,<<Add1>>,<<AddArray1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<Add2>>]
/// CHECK: <<AddArray2:i\d+>> Add [<<ArrayGet2>>,<<Const1>>]
/// CHECK: <<ArraySet2:v\d+>> ArraySet [<<Array>>,<<Add2>>,<<AddArray2>>]
/// CHECK: <<ArrayGet3:i\d+>> ArrayGet [<<Array>>,<<Add3>>]
/// CHECK: <<AddArray3:i\d+>> Add [<<ArrayGet3>>,<<Const1>>]
/// CHECK: <<ArraySet3:v\d+>> ArraySet [<<Array>>,<<Add3>>,<<AddArray3>>]
/// CHECK-START-ARM64: void Main.arrayAccessVariable(int) scheduler (after)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 2
/// CHECK-DAG: <<Const3:i\d+>> IntConstant -1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Add2:i\d+>> Add [<<Param>>,<<Const2>>]
/// CHECK: <<Add3:i\d+>> Add [<<Param>>,<<Const3>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: ArrayGet [<<Array>>,{{i\d+}}]
/// CHECK: ArrayGet [<<Array>>,{{i\d+}}]
/// CHECK: ArrayGet [<<Array>>,{{i\d+}}]
/// CHECK: Add
/// CHECK: Add
/// CHECK: Add
/// CHECK: ArraySet
/// CHECK: ArraySet
/// CHECK: ArraySet
public static void arrayAccessVariable(int i) {
int [] array = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
for (int j = 0; j < 100; j++) {
array[i + 1]++;
array[i + 2]++;
array[i - 1]++;
}
}
/// CHECK-START-ARM: void Main.arrayAccessSub(int) scheduler (before)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant -1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 9
/// CHECK-DAG: <<Const3:i\d+>> IntConstant 1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Sub2:i\d+>> Sub [<<Const2>>,<<Param>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<Add1>>]
/// CHECK: <<AddArray1:i\d+>> Add [<<ArrayGet1>>,<<Const3>>]
/// CHECK: <<ArraySet1:v\d+>> ArraySet [<<Array>>,<<Add1>>,<<AddArray1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<Sub2>>]
/// CHECK: <<AddArray2:i\d+>> Add [<<ArrayGet2>>,<<Const3>>]
/// CHECK: <<ArraySet2:v\d+>> ArraySet [<<Array>>,<<Sub2>>,<<AddArray2>>]
/// CHECK-START-ARM: void Main.arrayAccessSub(int) scheduler (after)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant -1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 9
/// CHECK-DAG: <<Const3:i\d+>> IntConstant 1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Sub2:i\d+>> Sub [<<Const2>>,<<Param>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<Add1>>]
/// CHECK: <<AddArray1:i\d+>> Add [<<ArrayGet1>>,<<Const3>>]
/// CHECK: <<ArraySet1:v\d+>> ArraySet [<<Array>>,<<Add1>>,<<AddArray1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<Sub2>>]
/// CHECK: <<AddArray2:i\d+>> Add [<<ArrayGet2>>,<<Const3>>]
/// CHECK: <<ArraySet2:v\d+>> ArraySet [<<Array>>,<<Sub2>>,<<AddArray2>>]
/// CHECK-START-ARM64: void Main.arrayAccessSub(int) scheduler (before)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant -1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 9
/// CHECK-DAG: <<Const3:i\d+>> IntConstant 1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Sub2:i\d+>> Sub [<<Const2>>,<<Param>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<Add1>>]
/// CHECK: <<AddArray1:i\d+>> Add [<<ArrayGet1>>,<<Const3>>]
/// CHECK: <<ArraySet1:v\d+>> ArraySet [<<Array>>,<<Add1>>,<<AddArray1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<Sub2>>]
/// CHECK: <<AddArray2:i\d+>> Add [<<ArrayGet2>>,<<Const3>>]
/// CHECK: <<ArraySet2:v\d+>> ArraySet [<<Array>>,<<Sub2>>,<<AddArray2>>]
/// CHECK-START-ARM64: void Main.arrayAccessSub(int) scheduler (after)
/// CHECK: <<Param:i\d+>> ParameterValue
/// CHECK-DAG: <<Const1:i\d+>> IntConstant -1
/// CHECK-DAG: <<Const2:i\d+>> IntConstant 9
/// CHECK-DAG: <<Const3:i\d+>> IntConstant 1
/// CHECK: <<Add1:i\d+>> Add [<<Param>>,<<Const1>>]
/// CHECK: <<Sub2:i\d+>> Sub [<<Const2>>,<<Param>>]
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet [<<Array>>,<<Add1>>]
/// CHECK: <<AddArray1:i\d+>> Add [<<ArrayGet1>>,<<Const3>>]
/// CHECK: <<ArraySet1:v\d+>> ArraySet [<<Array>>,<<Add1>>,<<AddArray1>>]
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet [<<Array>>,<<Sub2>>]
/// CHECK: <<AddArray2:i\d+>> Add [<<ArrayGet2>>,<<Const3>>]
/// CHECK: <<ArraySet2:v\d+>> ArraySet [<<Array>>,<<Sub2>>,<<AddArray2>>]
public static void arrayAccessSub(int i) {
int [] array = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
for (int j = 0; j < 100; j++) {
// These two accesses MAY ALIAS
array[i - 1]++;
array[9 - i]++;
}
}
/// CHECK-START-ARM: void Main.arrayAccessLoopVariable() scheduler (before)
/// CHECK-DAG: <<Const0:i\d+>> IntConstant 0
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK: <<Phi:i\d+>> Phi
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet
/// CHECK: <<AddArray1:i\d+>> Add
/// CHECK: <<ArraySet1:v\d+>> ArraySet
/// CHECK: <<AddVar:i\d+>> Add
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet
/// CHECK: <<AddArray2:i\d+>> Add
/// CHECK: <<ArraySet2:v\d+>> ArraySet
/// CHECK-START-ARM: void Main.arrayAccessLoopVariable() scheduler (after)
/// CHECK-DAG: <<Const0:i\d+>> IntConstant 0
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK: <<Phi:i\d+>> Phi
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<AddVar:i\d+>> Add
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet
/// CHECK: <<AddArray1:i\d+>> Add
/// CHECK: <<AddArray2:i\d+>> Add
/// CHECK: <<ArraySet1:v\d+>> ArraySet
/// CHECK: <<ArraySet2:v\d+>> ArraySet
/// CHECK-START-ARM64: void Main.arrayAccessLoopVariable() scheduler (before)
/// CHECK-DAG: <<Const0:i\d+>> IntConstant 0
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK: <<Phi:i\d+>> Phi
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet
/// CHECK: <<AddArray1:i\d+>> Add
/// CHECK: <<ArraySet1:v\d+>> ArraySet
/// CHECK: <<AddVar:i\d+>> Add
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet
/// CHECK: <<AddArray2:i\d+>> Add
/// CHECK: <<ArraySet2:v\d+>> ArraySet
/// CHECK-START-ARM64: void Main.arrayAccessLoopVariable() scheduler (after)
/// CHECK-DAG: <<Const0:i\d+>> IntConstant 0
/// CHECK-DAG: <<Const1:i\d+>> IntConstant 1
/// CHECK: <<Phi:i\d+>> Phi
/// CHECK: <<Array:i\d+>> IntermediateAddress
/// CHECK: <<AddVar:i\d+>> Add
/// CHECK: <<ArrayGet1:i\d+>> ArrayGet
/// CHECK: <<ArrayGet2:i\d+>> ArrayGet
/// CHECK: <<AddArray1:i\d+>> Add
/// CHECK: <<AddArray2:i\d+>> Add
/// CHECK: <<ArraySet1:v\d+>> ArraySet
/// CHECK: <<ArraySet2:v\d+>> ArraySet
public static void arrayAccessLoopVariable() {
int [] array = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
for (int j = 0; j < 9; j++) {
array[j]++;
array[j + 1]++;
}
}
/// CHECK-START-ARM: void Main.accessFields() scheduler (before)
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
/// CHECK-START-ARM: void Main.accessFields() scheduler (after)
/// CHECK-DAG: InstanceFieldGet
/// CHECK-DAG: InstanceFieldGet
/// CHECK-DAG: StaticFieldGet
/// CHECK-DAG: StaticFieldGet
/// CHECK: Add
/// CHECK: Add
/// CHECK: Add
/// CHECK: Add
/// CHECK-DAG: InstanceFieldSet
/// CHECK-DAG: InstanceFieldSet
/// CHECK-DAG: StaticFieldSet
/// CHECK-DAG: StaticFieldSet
/// CHECK-START-ARM64: void Main.accessFields() scheduler (before)
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
/// CHECK-START-ARM64: void Main.accessFields() scheduler (after)
/// CHECK-DAG: InstanceFieldGet
/// CHECK-DAG: InstanceFieldGet
/// CHECK-DAG: StaticFieldGet
/// CHECK-DAG: StaticFieldGet
/// CHECK: Add
/// CHECK: Add
/// CHECK: Add
/// CHECK: Add
/// CHECK-DAG: InstanceFieldSet
/// CHECK-DAG: InstanceFieldSet
/// CHECK-DAG: StaticFieldSet
/// CHECK-DAG: StaticFieldSet
public void accessFields() {
my_obj = new ExampleObj(1, 2);
for (int i = 0; i < 10; i++) {
my_obj.n1++;
my_obj.n2++;
number1++;
number2++;
}
}
/// CHECK-START-ARM: void Main.accessFieldsVolatile() scheduler (before)
/// CHECK-START-ARM64: void Main.accessFieldsVolatile() scheduler (before)
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
/// CHECK-START-ARM: void Main.accessFieldsVolatile() scheduler (after)
/// CHECK-START-ARM64: void Main.accessFieldsVolatile() scheduler (after)
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
/// CHECK: StaticFieldGet
/// CHECK: Add
/// CHECK: StaticFieldSet
public void accessFieldsVolatile() {
my_obj = new ExampleObj(1, 2);
for (int i = 0; i < 10; i++) {
my_obj.n1++;
my_obj.n2++;
number1++;
number3++;
}
}
/// CHECK-START-ARM: void Main.accessFieldsUnresolved() scheduler (before)
/// CHECK-START-ARM64: void Main.accessFieldsUnresolved() scheduler (before)
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: UnresolvedInstanceFieldGet
/// CHECK: Add
/// CHECK: UnresolvedInstanceFieldSet
/// CHECK: UnresolvedStaticFieldGet
/// CHECK: Add
/// CHECK: UnresolvedStaticFieldSet
/// CHECK-START-ARM: void Main.accessFieldsUnresolved() scheduler (after)
/// CHECK-START-ARM64: void Main.accessFieldsUnresolved() scheduler (after)
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: InstanceFieldGet
/// CHECK: Add
/// CHECK: InstanceFieldSet
/// CHECK: UnresolvedInstanceFieldGet
/// CHECK: Add
/// CHECK: UnresolvedInstanceFieldSet
/// CHECK: UnresolvedStaticFieldGet
/// CHECK: Add
/// CHECK: UnresolvedStaticFieldSet
public void accessFieldsUnresolved() {
my_obj = new ExampleObj(1, 2);
UnresolvedClass unresolved_obj = new UnresolvedClass();
for (int i = 0; i < 10; i++) {
my_obj.n1++;
my_obj.n2++;
unresolved_obj.instanceInt++;
UnresolvedClass.staticInt++;
}
}
/// CHECK-START-ARM64: int Main.intDiv(int) scheduler (before)
/// CHECK: Sub
/// CHECK: DivZeroCheck
/// CHECK: Div
/// CHECK: StaticFieldSet
/// CHECK-START-ARM64: int Main.intDiv(int) scheduler (after)
/// CHECK: Sub
/// CHECK-NOT: StaticFieldSet
/// CHECK: DivZeroCheck
/// CHECK-NOT: Sub
/// CHECK: Div
public static int intDiv(int arg) {
int res = 0;
int tmp = arg;
for (int i = 1; i < arg; i++) {
tmp -= i;
res = res / i; // div-zero check barrier.
static_variable++;
}
res += tmp;
return res;
}
public static void main(String[] args) {
if ((arrayAccess() + intDiv(10)) != -35) {
System.out.println("FAIL");
}
}
}