/* * 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. */ import java.lang.reflect.Method; import java.lang.reflect.InvocationTargetException; class Test1 { int[] iarr; } class Test2 { float[] farr; } public class Main { public static Object[] getObjectArray() { return null; } public static long[] getLongArray() { return null; } public static Object getNull() { return null; } public static Test1 getNullTest1() { return null; } public static Test2 getNullTest2() { return null; } public static void $noinline$runSmaliTest(String name) throws Throwable { try { Class c = Class.forName("SmaliTests"); Method m = c.getMethod(name); m.invoke(null); } catch (InvocationTargetException ex) { throw ex.getCause(); // re-raise expected exception. } catch (Exception ex) { throw new Error(ex); } } public static void main(String[] args) { try { foo(); throw new Error("Expected NullPointerException"); } catch (NullPointerException e) { // Expected. } try { bar(); throw new Error("Expected NullPointerException"); } catch (NullPointerException e) { // Expected. } try { $noinline$runSmaliTest("bar"); throw new Error("Expected NullPointerException"); } catch (NullPointerException e) { // Expected. } catch (Throwable t) { throw new Error("Unexpected Throwable", t); } try { test1(); throw new Error("Expected NullPointerException"); } catch (NullPointerException e) { // Expected. } } /// CHECK-START: void Main.foo() load_store_elimination (after) /// CHECK-DAG: <> NullConstant /// CHECK-DAG: <> NullCheck [<>] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] public static void foo() { longField = getLongArray()[0]; objectField = getObjectArray()[0]; } /// CHECK-START: void Main.bar() load_store_elimination (after) /// CHECK-DAG: <> NullConstant /// CHECK-DAG: <> BoundType [<>] /// CHECK-DAG: <> BoundType [<>] /// CHECK-DAG: <> NullCheck [<>] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> NullCheck [<>] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] public static void bar() { // We create multiple accesses that will lead the bounds check // elimination pass to add a HDeoptimize. Not having the bounds check helped // the load store elimination think it could merge two ArrayGet with different // types. String[] array = (String[])getNull(); objectField = array[0]; objectField = array[1]; objectField = array[2]; objectField = array[3]; long[] longArray = getLongArray(); longField = longArray[0]; longField = longArray[1]; longField = longArray[2]; longField = longArray[3]; } /// CHECK-START: float Main.test1() load_store_elimination (after) /// CHECK-DAG: <> NullConstant /// CHECK-DAG: <> NullCheck [<>] /// CHECK-DAG: <> InstanceFieldGet [<>] field_name:Test1.iarr /// CHECK-DAG: <> NullCheck [<>] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: <> ArrayGet [<>,{{i\d+}}] /// CHECK-DAG: Return [<>] public static float test1() { Test1 test1 = getNullTest1(); Test2 test2 = getNullTest2(); int[] iarr = test1.iarr; float[] farr = test2.farr; iarr[0] = iarr[1]; return farr[0]; } public static long longField; public static Object objectField; }