blob: 4721ecaa09cec4debee9ac80b27c2121b6c6c4b7 [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.
.class public LTestCase;
.super Ljava/lang/Object;
# Test that all vregs holding the new-instance are updated after the
# StringFactory call.
## CHECK-START: java.lang.String TestCase.vregAliasing(byte[]) register (after)
## CHECK-DAG: Return [<<String:l\d+>>]
## CHECK-DAG: <<String>> InvokeStaticOrDirect method_name:java.lang.String.<init>
.method public static vregAliasing([B)Ljava/lang/String;
.registers 5
# Create new instance of String and store it to v0, v1, v2.
new-instance v0, Ljava/lang/String;
move-object v1, v0
move-object v2, v0
# Call String.<init> on v1.
const-string v3, "UTF8"
invoke-direct {v1, p0, v3}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
# Return the object from v2.
return-object v2
.end method
# Test usage of String new-instance before it is initialized.
## CHECK-START: void TestCase.compareNewInstance() register (after)
## CHECK-DAG: <<Null:l\d+>> InvokeStaticOrDirect method_name:Main.$noinline$HiddenNull
## CHECK-DAG: <<String:l\d+>> NewInstance
## CHECK-DAG: <<Cond:z\d+>> NotEqual [<<Null>>,<<String>>]
## CHECK-DAG: If [<<Cond>>]
.method public static compareNewInstance()V
.registers 3
invoke-static {}, LMain;->$noinline$HiddenNull()Ljava/lang/Object;
move-result-object v1
new-instance v0, Ljava/lang/String;
if-ne v0, v1, :return
# Will throw NullPointerException if this branch is taken.
const v1, 0x0
const-string v2, "UTF8"
invoke-direct {v0, v1, v2}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-void
:return
return-void
.end method
# Test deoptimization between String's allocation and initialization. When not
# compiling --debuggable, the NewInstance will be optimized out.
## CHECK-START: int TestCase.deoptimizeNewInstance(int[], byte[]) register (after)
## CHECK: <<Null:l\d+>> NullConstant
## CHECK: Deoptimize env:[[<<Null>>,{{.*]]}}
## CHECK: InvokeStaticOrDirect method_name:java.lang.String.<init>
## CHECK-START-DEBUGGABLE: int TestCase.deoptimizeNewInstance(int[], byte[]) register (after)
## CHECK: <<String:l\d+>> NewInstance
## CHECK: Deoptimize env:[[<<String>>,{{.*]]}}
## CHECK: InvokeStaticOrDirect method_name:java.lang.String.<init>
.method public static deoptimizeNewInstance([I[B)I
.registers 6
const v2, 0x0
const v1, 0x1
new-instance v0, Ljava/lang/String; # HNewInstance(String)
# Deoptimize here if the array is too short.
aget v1, p0, v1 # v1 = int_array[0x1]
add-int/2addr v2, v1 # v2 = 0x0 + v1
# Check that we're being executed by the interpreter.
invoke-static {}, LMain;->assertIsInterpreted()V
# String allocation should succeed.
const-string v3, "UTF8"
invoke-direct {v0, p1, v3}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
# Transformed into invoke StringFactory(p1,v3).
# The use of v0 is dropped (so HNewInstance(String) ends up having 0 uses and is removed).
# This ArrayGet will throw ArrayIndexOutOfBoundsException.
const v1, 0x4
aget v1, p0, v1
add-int/2addr v2, v1
return v2
.end method
# Test that a redundant NewInstance is removed if not used and not compiling
# --debuggable.
## CHECK-START: java.lang.String TestCase.removeNewInstance(byte[]) register (after)
## CHECK-NOT: NewInstance
## CHECK-NOT: LoadClass
## CHECK-START-DEBUGGABLE: java.lang.String TestCase.removeNewInstance(byte[]) register (after)
## CHECK: NewInstance
.method public static removeNewInstance([B)Ljava/lang/String;
.registers 5
new-instance v0, Ljava/lang/String;
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v0
# Although it looks like we "use" the new-instance v0 here, the optimizing compiler
# transforms all uses of the new-instance into uses of the StringFactory invoke.
# therefore the HNewInstance for v0 becomes dead and is removed.
.end method
# Test #1 for irreducible loops and String.<init>.
.method public static irreducibleLoopAndStringInit1([BZ)Ljava/lang/String;
.registers 5
new-instance v0, Ljava/lang/String;
# Irreducible loop
if-eqz p1, :loop_entry
:loop_header
xor-int/lit8 p1, p1, 0x1
:loop_entry
if-eqz p1, :string_init
goto :loop_header
:string_init
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v0
.end method
# Test #2 for irreducible loops and String.<init>.
.method public static irreducibleLoopAndStringInit2([BZ)Ljava/lang/String;
.registers 5
new-instance v0, Ljava/lang/String;
# Irreducible loop
if-eqz p1, :loop_entry
:loop_header
if-eqz p1, :string_init
:loop_entry
xor-int/lit8 p1, p1, 0x1
goto :loop_header
:string_init
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v0
.end method
# Test #3 for irreducible loops and String.<init> alias.
.method public static irreducibleLoopAndStringInit3([BZ)Ljava/lang/String;
.registers 5
new-instance v0, Ljava/lang/String;
move-object v2, v0
# Irreducible loop
if-eqz p1, :loop_entry
:loop_header
xor-int/lit8 p1, p1, 0x1
:loop_entry
if-eqz p1, :string_init
goto :loop_header
:string_init
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v2
.end method
# Test with a loop between allocation and String.<init>.
.method public static loopAndStringInit([BZ)Ljava/lang/String;
.registers 5
new-instance v0, Ljava/lang/String;
# Loop
:loop_header
if-eqz p1, :loop_exit
xor-int/lit8 p1, p1, 0x1
goto :loop_header
:loop_exit
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v0
.end method
# Test with a loop and aliases between allocation and String.<init>.
.method public static loopAndStringInitAlias([BZ)Ljava/lang/String;
.registers 5
new-instance v0, Ljava/lang/String;
move-object v2, v0
# Loop
:loop_header
if-eqz p1, :loop_exit
xor-int/lit8 p1, p1, 0x1
goto :loop_header
:loop_exit
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v2
.end method
# Test deoptimization after String initialization of a phi.
## CHECK-START: int TestCase.deoptimizeNewInstanceAfterLoop(int[], byte[], int) register (after)
## CHECK: <<Invoke:l\d+>> InvokeStaticOrDirect method_name:java.lang.String.<init>
## CHECK: Deoptimize env:[[<<Invoke>>,{{.*]]}}
.method public static deoptimizeNewInstanceAfterLoop([I[BI)I
.registers 8
const v2, 0x0
const v1, 0x1
new-instance v0, Ljava/lang/String; # HNewInstance(String)
move-object v4, v0
# Loop
:loop_header
if-eqz p2, :loop_exit
xor-int/lit8 p2, p2, 0x1
goto :loop_header
:loop_exit
const-string v3, "UTF8"
invoke-direct {v0, p1, v3}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
# Deoptimize here if the array is too short.
aget v1, p0, v1 # v1 = int_array[0x1]
add-int/2addr v2, v1 # v2 = 0x0 + v1
# Check that we're being executed by the interpreter.
invoke-static {}, LMain;->assertIsInterpreted()V
# Check that the environments contain the right string.
invoke-static {p1, v0}, LMain;->assertEqual([BLjava/lang/String;)V
invoke-static {p1, v4}, LMain;->assertEqual([BLjava/lang/String;)V
# This ArrayGet will throw ArrayIndexOutOfBoundsException.
const v1, 0x4
aget v1, p0, v1
add-int/2addr v2, v1
return v2
.end method
# Test with a loop between allocation and String.<init> and a null check.
## CHECK-START: java.lang.String TestCase.loopAndStringInitAndTest(byte[], boolean) builder (after)
## CHECK-DAG: <<Null:l\d+>> NullConstant
## CHECK-DAG: <<String:l\d+>> NewInstance
## CHECK-DAG: <<Cond:z\d+>> NotEqual [<<String>>,<<Null>>]
## CHECK-START: java.lang.String TestCase.loopAndStringInitAndTest(byte[], boolean) register (after)
## CHECK-DAG: <<String:l\d+>> NewInstance
.method public static loopAndStringInitAndTest([BZ)Ljava/lang/String;
.registers 5
new-instance v0, Ljava/lang/String;
# Loop
:loop_header
# Use the new-instance in the only way it can be used.
if-nez v0, :loop_exit
xor-int/lit8 p1, p1, 0x1
goto :loop_header
:loop_exit
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v0
.end method
## CHECK-START: java.lang.String TestCase.loopAndStringInitAndPhi(byte[], boolean) register (after)
## CHECK-NOT: NewInstance
## CHECK-DAG: <<Invoke1:l\d+>> InvokeStaticOrDirect method_name:java.lang.String.<init>
## CHECK-DAG: <<Invoke2:l\d+>> InvokeStaticOrDirect method_name:java.lang.String.<init>
## CHECK-DAG: <<Phi:l\d+>> Phi [<<Invoke2>>,<<Invoke1>>]
## CHECK-DAG: Return [<<Phi>>]
.method public static loopAndStringInitAndPhi([BZ)Ljava/lang/String;
.registers 4
if-nez p1, :allocate_other
new-instance v0, Ljava/lang/String;
# Loop
:loop_header
if-eqz p1, :loop_exit
goto :loop_header
:loop_exit
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
goto : exit
:allocate_other
const-string v1, "UTF8"
new-instance v0, Ljava/lang/String;
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
:exit
return-object v0
.end method
.method public static loopAndTwoStringInitAndPhi([BZZ)Ljava/lang/String;
.registers 6
new-instance v0, Ljava/lang/String;
new-instance v2, Ljava/lang/String;
if-nez p2, :allocate_other
# Loop
:loop_header
if-eqz p1, :loop_exit
goto :loop_header
:loop_exit
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
goto :exit
:allocate_other
# Loop
:loop_header2
if-eqz p1, :loop_exit2
goto :loop_header2
:loop_exit2
const-string v1, "UTF8"
invoke-direct {v2, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
move-object v0, v2
:exit
return-object v0
.end method
# Regression test for a new string flowing into a catch phi.
.method public static stringAndCatch([BZ)Ljava/lang/Object;
.registers 4
const v0, 0x0
:try_start_a
new-instance v0, Ljava/lang/String;
# Loop
:loop_header
if-eqz p1, :loop_exit
goto :loop_header
:loop_exit
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
goto :exit
:try_end_a
.catch Ljava/lang/Exception; {:try_start_a .. :try_end_a} :catch_a
:catch_a
# Initially, we create a catch phi with the potential uninitalized string, which used to
# trip the compiler. However, using that catch phi is an error caught by the verifier, so
# having the phi is benign.
const v0, 0x0
:exit
return-object v0
.end method
# Same test as above, but with a catch phi being used by the string constructor.
.method public static stringAndCatch2([BZ)Ljava/lang/Object;
.registers 4
const v0, 0x0
new-instance v0, Ljava/lang/String;
:try_start_a
const-string v1, "UTF8"
:try_end_a
.catch Ljava/lang/Exception; {:try_start_a .. :try_end_a} :catch_a
:catch_a
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
return-object v0
.end method
# Same test as above, but with a catch phi being used by the string constructor and
# a null test.
.method public static stringAndCatch3([BZ)Ljava/lang/Object;
.registers 4
const v0, 0x0
new-instance v0, Ljava/lang/String;
:try_start_a
const-string v1, "UTF8"
:try_end_a
.catch Ljava/lang/Exception; {:try_start_a .. :try_end_a} :catch_a
:catch_a
if-eqz v0, :unexpected
const-string v1, "UTF8"
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
goto :exit
:unexpected
const-string v0, "UTF8"
:exit
return-object v0
.end method
# Regression test that tripped the compiler.
.method public static stringAndPhi([BZ)Ljava/lang/Object;
.registers 4
new-instance v0, Ljava/lang/String;
const-string v1, "UTF8"
:loop_header
if-nez p1, :unused
if-eqz p1, :invoke
goto :loop_header
:invoke
invoke-direct {v0, p0, v1}, Ljava/lang/String;-><init>([BLjava/lang/String;)V
goto :exit
:unused
const-string v0, "UTF8"
if-nez p1, :exit
goto :unused
:exit
return-object v0
.end method