You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
476 lines
13 KiB
476 lines
13 KiB
# 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
|