845 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			845 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Java
		
	
	
	
| /*
 | |
|  * Copyright (C) 2017 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.Array;
 | |
| import java.lang.reflect.Field;
 | |
| import java.lang.reflect.Method;
 | |
| import java.lang.reflect.Modifier;
 | |
| 
 | |
| // Baseline class. This has no final fields, so there are no additional freezes
 | |
| // in its constructor.
 | |
| //
 | |
| // The new-instance itself always has 1 freeze for the happens-before on the object header
 | |
| // write (i.e. [obj.class = X] happens-before any access to obj).
 | |
| //
 | |
| // Total freezes for "new Base()": 1.
 | |
| class Base {
 | |
|   int w0;
 | |
|   int w1;
 | |
|   int w2;
 | |
|   int w3;
 | |
| 
 | |
|   @Override
 | |
|   public String toString() {
 | |
|     return getClass().getName() + "(" + baseString() + ")";
 | |
|   }
 | |
| 
 | |
|   protected String baseString() {
 | |
|     return String.format("w0: %d, w1: %d, w2: %d, w3: %d", w0, w1, w2, w3);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // This has a final field in its constructor, so there must be a field freeze
 | |
| // at the end of <init>.
 | |
| //
 | |
| // Total freezes for "new OneFinal()": 2.
 | |
| class OneFinal extends Base {
 | |
|   final int x;
 | |
|   OneFinal(int x) {
 | |
|     this.x = x;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   protected String baseString() {
 | |
|     return String.format("%s, x: %d", super.baseString(), x);
 | |
|   }
 | |
| }
 | |
| 
 | |
| class Assert {
 | |
|   public static void stringEquals(String expected, Object actual) {
 | |
|     stringEquals$noinline$(expected, actual);
 | |
|   }
 | |
| 
 | |
|   // Forbid compiler from inlining this to avoid overly clever optimizations.
 | |
|   private static void stringEquals$noinline$(String expected, Object actual) {
 | |
|     String actualStr = Main.valueToString(actual);
 | |
|     if (!expected.equals(actualStr)) {
 | |
|       throw new AssertionError("Expected: " + expected + ", actual: " + actualStr);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| interface Test {
 | |
|   public void exercise();
 | |
|   public void check();
 | |
| }
 | |
| 
 | |
| class TestOneFinal implements Test {
 | |
|   // Initialize at least once before actual test.
 | |
|   public static Object external;
 | |
| 
 | |
|   /// CHECK-START: void TestOneFinal.exercise() constructor_fence_redundancy_elimination (before)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
| 
 | |
|   /// CHECK-START: void TestOneFinal.exercise() constructor_fence_redundancy_elimination (after)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   @Override
 | |
|   public void exercise() {
 | |
|       Base b = new OneFinal(1);
 | |
|       // 1 store, 2 freezes.
 | |
| 
 | |
|       // Stores to 'b' do not escape b.
 | |
|       b.w0 = 1;
 | |
|       b.w1 = 2;
 | |
|       b.w2 = 3;
 | |
| 
 | |
|       // Publish the result to a global so that it is not LSE-eliminated.
 | |
|       external = b;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   public void check() {
 | |
|     Assert.stringEquals("OneFinal(w0: 1, w1: 2, w2: 3, w3: 0, x: 1)", external);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // This has a final field in its constructor, so there must be a field freeze
 | |
| // at the end of <init>. The previous base class's freezes accumulate on top
 | |
| // of this one.
 | |
| //
 | |
| // Total freezes for "new TwoFinal()": 3.
 | |
| class TwoFinal extends OneFinal {
 | |
|   final int y;
 | |
|   TwoFinal(int x, int y) {
 | |
|     super(x);
 | |
|     this.y = y;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   protected String baseString() {
 | |
|     return String.format("%s, y: %d", super.baseString(), y);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // This has a final field in its constructor, so there must be a field freeze
 | |
| // at the end of <init>. The previous base class's freezes accumulate on top
 | |
| // of this one.
 | |
| //
 | |
| // Total freezes for "new ThreeFinal()": 4.
 | |
| class ThreeFinal extends TwoFinal {
 | |
|   final int z;
 | |
|   ThreeFinal(int x, int y, int z) {
 | |
|     super(x, y);
 | |
|     this.z = z;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   protected String baseString() {
 | |
|     return String.format("%s, z: %d", super.baseString(), z);
 | |
|   }
 | |
| }
 | |
| 
 | |
| class TestThreeFinal implements Test {
 | |
|   // Initialize at least once before actual test.
 | |
|   public static Object external;
 | |
| 
 | |
|   /// CHECK-START: void TestThreeFinal.exercise() constructor_fence_redundancy_elimination (before)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
| 
 | |
|   /// CHECK-START: void TestThreeFinal.exercise() constructor_fence_redundancy_elimination (after)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   @Override
 | |
|   public void exercise() {
 | |
|     Base b = new ThreeFinal(1, 1, 2);
 | |
|     // 3 store, 4 freezes.
 | |
| 
 | |
|     // Stores to 'b' do not escape b.
 | |
|     b.w0 = 3;
 | |
| 
 | |
|     // Publish the result to a global so that it is not LSE-eliminated.
 | |
|     external = b;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   public void check() {
 | |
|     Assert.stringEquals("ThreeFinal(w0: 3, w1: 0, w2: 0, w3: 0, x: 1, y: 1, z: 2)", external);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Ensure "freezes" between multiple new-instances are optimized out.
 | |
| class TestMultiAlloc implements Test {
 | |
|   public static Object external;
 | |
|   public static Object external2;
 | |
| 
 | |
|   /// CHECK-START: void TestMultiAlloc.exercise() constructor_fence_redundancy_elimination (before)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External2:l\d+>>,<<NewInstance2>>]
 | |
| 
 | |
|   /// CHECK-START: void TestMultiAlloc.exercise() constructor_fence_redundancy_elimination (after)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>,<<NewInstance>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External2:l\d+>>,<<NewInstance2>>]
 | |
|   @Override
 | |
|   public void exercise() {
 | |
|     // 1 freeze
 | |
|     Base b = new Base();
 | |
|     // 1 freeze
 | |
|     Base b2 = new Base();
 | |
| 
 | |
|     // Merge 2 freezes above into 1 constructor fence.
 | |
|     external = b;
 | |
|     external2 = b2;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   public void check() {
 | |
|     Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external);
 | |
|     Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external2);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Ensure "freezes" between multiple new-instances are optimized out.
 | |
| class TestThreeFinalTwice implements Test {
 | |
|   // Initialize at least once before actual test.
 | |
|   public static Object external;
 | |
|   public static Object external2;
 | |
| 
 | |
|   /// CHECK-START: void TestThreeFinalTwice.exercise() constructor_fence_redundancy_elimination (before)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External2:l\d+>>,<<NewInstance2>>]
 | |
| 
 | |
|   /// CHECK-START: void TestThreeFinalTwice.exercise() constructor_fence_redundancy_elimination (after)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>,<<NewInstance>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External2:l\d+>>,<<NewInstance2>>]
 | |
|   @Override
 | |
|   public void exercise() {
 | |
|     Base b = new ThreeFinal(1, 1, 2);
 | |
|     // 3 store, 4 freezes.
 | |
| 
 | |
|     // Stores to 'b' do not escape b.
 | |
|     b.w0 = 3;
 | |
| 
 | |
|     Base b2 = new ThreeFinal(4, 5, 6);
 | |
|     // 3 store, 4 freezes.
 | |
| 
 | |
|     // Stores to 'b2' do not escape b2.
 | |
|     b2.w0 = 7;
 | |
| 
 | |
|     // Publish the result to a global so that it is not LSE-eliminated.
 | |
|     // Publishing is done at the end to give freezes above a chance to merge.
 | |
|     external = b;
 | |
|     external2 = b2;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   public void check() {
 | |
|     Assert.stringEquals("ThreeFinal(w0: 3, w1: 0, w2: 0, w3: 0, x: 1, y: 1, z: 2)", external);
 | |
|     Assert.stringEquals("ThreeFinal(w0: 7, w1: 0, w2: 0, w3: 0, x: 4, y: 5, z: 6)", external2);
 | |
|   }
 | |
| }
 | |
| 
 | |
| class TestNonEscaping {
 | |
|   // Prevent constant folding.
 | |
|   static boolean test;
 | |
| 
 | |
|   static Object external;
 | |
|   static Object external2;
 | |
|   static Object external3;
 | |
|   static Object external4;
 | |
| 
 | |
|   static class Invoke implements Test {
 | |
|     /// CHECK-START: void TestNonEscaping$Invoke.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK:                          InvokeStaticOrDirect
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestNonEscaping$Invoke.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          InvokeStaticOrDirect
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>,<<NewInstance>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
| 
 | |
|       // b cannot possibly escape into this invoke because it hasn't escaped onto the heap earlier,
 | |
|       // and the invoke doesn't take it as a parameter.
 | |
|       noEscape$noinline$();
 | |
| 
 | |
|       // Remove the Constructor Fence for b, merging into the fence for b2.
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external = b;
 | |
|       external2 = b2;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external2);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   public static int[] array = new int[1];
 | |
|   static Base base = new Base();
 | |
| 
 | |
|   static class Store implements Test {
 | |
|     /// CHECK-START: void TestNonEscaping$Store.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                      ArraySet
 | |
|     /// CHECK-DAG:                      StaticFieldSet
 | |
|     /// CHECK-DAG:                      InstanceFieldSet
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestNonEscaping$Store.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>   NewInstance
 | |
|     /// CHECK-DAG: <<NewInstance2:l\d+>>  NewInstance
 | |
|     /// CHECK-DAG:                        ConstructorFence [<<NewInstance2>>,<<NewInstance>>]
 | |
|     /// CHECK-NOT:                        ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
| 
 | |
|       // Stores of inputs other than the fence target do not publish 'b'.
 | |
|       array[0] = b.w0;  // aput
 | |
|       external = array; // sput
 | |
|       base.w0 = b.w0;   // iput
 | |
| 
 | |
|       // Remove the Constructor Fence for b, merging into the fence for b2.
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external3 = b;
 | |
|       external4 = b2;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals("[0]", array);
 | |
|       Assert.stringEquals("[0]", external);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", base);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external3);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external4);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   private static void noEscape$noinline$() {
 | |
|   }
 | |
| }
 | |
| 
 | |
| class TestDontOptimizeAcrossBlocks implements Test {
 | |
|   // Prevent constant folding.
 | |
|   static boolean test;
 | |
| 
 | |
|   static Object external;
 | |
|   static Object external3;
 | |
| 
 | |
|   /// CHECK-START: void TestDontOptimizeAcrossBlocks.exercise() constructor_fence_redundancy_elimination (before)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External2:l\d+>>,<<NewInstance2>>]
 | |
| 
 | |
|   /// CHECK-START: void TestDontOptimizeAcrossBlocks.exercise() constructor_fence_redundancy_elimination (after)
 | |
|   /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|   /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|   /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|   /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|   /// CHECK-NOT:                      ConstructorFence
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External:l\d+>>,<<NewInstance>>]
 | |
|   /// CHECK-DAG:                      StaticFieldSet [<<External2:l\d+>>,<<NewInstance2>>]
 | |
|   @Override
 | |
|   public void exercise() {
 | |
|     Base b = new Base();
 | |
| 
 | |
|     // Do not move constructor fence across this block, even though 'b' is not published yet.
 | |
|     if (test) {
 | |
|       external = null;
 | |
|     }
 | |
| 
 | |
|     Base b2 = new Base();
 | |
|     external = b2;
 | |
|     external3 = b;
 | |
|   }
 | |
| 
 | |
|   @Override
 | |
|   public void check() {
 | |
|     Assert.stringEquals("false", test);
 | |
|     Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external);
 | |
|     Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external3);
 | |
|   }
 | |
| }
 | |
| 
 | |
| class TestDontOptimizeAcrossEscape {
 | |
|   // Prevent constant folding.
 | |
|   static boolean test;
 | |
| 
 | |
|   static Object external;
 | |
|   static Object external2;
 | |
|   static Object external3;
 | |
|   static Object external4;
 | |
| 
 | |
|   static class Invoke implements Test {
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$Invoke.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK:                          InvokeStaticOrDirect
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$Invoke.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK:                          InvokeStaticOrDirect
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
|       // Do not optimize across invokes into which the fence target escapes.
 | |
|       invoke$noinline$(b);
 | |
| 
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external = b;
 | |
|       external2 = b2;
 | |
|     }
 | |
| 
 | |
|     private static void invoke$noinline$(Object b) {
 | |
|       // Even though 'b' does not escape this method, we conservatively assume all parameters
 | |
|       // of an invoke escape.
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external2);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   public static Object[] array = new Object[3];
 | |
|   static Base base = new Base();
 | |
| 
 | |
|   static class InstanceEscaper {
 | |
|     public Object holder;
 | |
| 
 | |
|     @Override
 | |
|     public String toString() {
 | |
|       return getClass().getName() + "(" + baseString() + ")";
 | |
|     }
 | |
| 
 | |
|     protected String baseString() {
 | |
|       return String.format("holder: %s", Main.valueToString(holder));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static InstanceEscaper instanceEscaper = new InstanceEscaper();
 | |
| 
 | |
|   static class StoreIput implements Test {
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$StoreIput.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                      InstanceFieldSet
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$StoreIput.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>   NewInstance
 | |
|     /// CHECK:                            ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG: <<NewInstance2:l\d+>>  NewInstance
 | |
|     /// CHECK-DAG:                        ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                        ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
| 
 | |
|       // A store of 'b' into another instance will publish 'b'.
 | |
|       instanceEscaper.holder = b;
 | |
| 
 | |
|       // Do not remove any constructor fences above.
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external3 = b;
 | |
|       external4 = b2;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals(
 | |
|           "TestDontOptimizeAcrossEscape$InstanceEscaper(holder: Base(w0: 0, w1: 0, w2: 0, w3: 0))",
 | |
|           instanceEscaper);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external3);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external4);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static class StoreAput implements Test {
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$StoreAput.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                      ArraySet
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$StoreAput.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>   NewInstance
 | |
|     /// CHECK:                            ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG: <<NewInstance2:l\d+>>  NewInstance
 | |
|     /// CHECK-DAG:                        ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                        ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
| 
 | |
|       // A store of 'b' into another array will publish 'b'.
 | |
|       array[0] = b;  // aput
 | |
| 
 | |
|       // Do not remove any constructor fences above.
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external3 = b;
 | |
|       external4 = b2;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals("[Base(w0: 0, w1: 0, w2: 0, w3: 0),<null>,<null>]", array);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external3);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external4);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static class StoreSput implements Test {
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$StoreSput.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                      StaticFieldSet
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$StoreSput.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>   NewInstance
 | |
|     /// CHECK:                            ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG: <<NewInstance2:l\d+>>  NewInstance
 | |
|     /// CHECK-DAG:                        ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                        ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
| 
 | |
|       // A store of 'b' into a static will publish 'b'.
 | |
|       external = b;
 | |
| 
 | |
|       // Do not remove any constructor fences above.
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external3 = b;
 | |
|       external4 = b2;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external3);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external4);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static class Deopt implements Test {
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$Deopt.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                      Deoptimize
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$Deopt.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>   NewInstance
 | |
|     /// CHECK:                            ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG: <<NewInstance2:l\d+>>  NewInstance
 | |
|     /// CHECK-DAG:                        ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                        ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
| 
 | |
|       // An array access generates a Deopt to avoid doing bounds check.
 | |
|       array[0] = external;  // aput
 | |
|       array[1] = external;  // aput
 | |
|       array[2] = external;  // aput
 | |
| 
 | |
|       // Do not remove any constructor fences above.
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external3 = b;
 | |
|       external4 = b2;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals("[Base(w0: 0, w1: 0, w2: 0, w3: 0),"
 | |
|               + "Base(w0: 0, w1: 0, w2: 0, w3: 0),"
 | |
|               + "Base(w0: 0, w1: 0, w2: 0, w3: 0)]",
 | |
|           array);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external3);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external4);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static class Select implements Test {
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$Select.exercise() constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK: <<NewInstance:l\d+>>     NewInstance
 | |
|     /// CHECK:                          ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                      Select
 | |
|     /// CHECK: <<NewInstance2:l\d+>>    NewInstance
 | |
|     /// CHECK-DAG:                      ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                      ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$Select.exercise() constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>   NewInstance
 | |
|     /// CHECK:                            ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG: <<NewInstance2:l\d+>>  NewInstance
 | |
|     /// CHECK-DAG:                        ConstructorFence [<<NewInstance2>>]
 | |
|     /// CHECK-NOT:                        ConstructorFence
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       Base b = new Base();
 | |
| 
 | |
|       boolean localTest = test;
 | |
|       Object localExternal = external3;
 | |
| 
 | |
|       // Selecting 'b' creates an alias, which we conservatively assume escapes immediately.
 | |
|       external = localTest ? b : localExternal;
 | |
| 
 | |
|       // Do not remove any constructor fences above.
 | |
|       Base b2 = new Base();
 | |
| 
 | |
|       // Do not LSE-eliminate b,b2
 | |
|       external3 = b;
 | |
|       external4 = b2;
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external3);
 | |
|       Assert.stringEquals("Base(w0: 0, w1: 0, w2: 0, w3: 0)", external4);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static class MakeBoundTypeTest implements Test {
 | |
|     public static Object makeBoundType;
 | |
|     public static Object makeBoundTypeSub;
 | |
| 
 | |
|     @Override
 | |
|     public void exercise() {
 | |
|       // Note: MakeBoundType is special and we have to call the constructor directly
 | |
|       // to prevent inlining it.
 | |
|       try {
 | |
|         makeBoundType = exerciseNewInstance(MakeBoundType.class, 123);
 | |
|         makeBoundTypeSub = exerciseNewInstance(MakeBoundTypeSub.class, 123);
 | |
|       } catch (Exception e) {
 | |
|         throw new RuntimeException(e);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     @Override
 | |
|     public void check() {
 | |
|       Assert.stringEquals(
 | |
|           "TestDontOptimizeAcrossEscape$MakeBoundTypeTest$MakeBoundType(abcdefgh: 123, x: 2)",
 | |
|           makeBoundType);
 | |
|       Assert.stringEquals(
 | |
|           "TestDontOptimizeAcrossEscape$MakeBoundTypeTest$MakeBoundTypeSub(abcdefgh: 123, x: 1)",
 | |
|           makeBoundTypeSub);
 | |
|     }
 | |
| 
 | |
|     // Make a new instance of 'klass'.
 | |
|     private static <T> T exerciseNewInstance(Class<T> klass, int params) throws Exception {
 | |
|       return klass.cast(klass.getDeclaredConstructor(int.class).newInstance(params));
 | |
|     }
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$MakeBoundTypeTest$MakeBoundType.<init>(int) constructor_fence_redundancy_elimination (before)
 | |
|     /// CHECK-DAG: <<This:l\d+>>         ParameterValue
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>  NewInstance
 | |
|     /// CHECK:                           ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                       BoundType
 | |
|     /// CHECK-DAG:                       ConstructorFence [<<This>>]
 | |
|     /// CHECK-NOT:                       ConstructorFence
 | |
| 
 | |
|     /// CHECK-START: void TestDontOptimizeAcrossEscape$MakeBoundTypeTest$MakeBoundType.<init>(int) constructor_fence_redundancy_elimination (after)
 | |
|     /// CHECK-DAG: <<This:l\d+>>         ParameterValue
 | |
|     /// CHECK-DAG: <<NewInstance:l\d+>>  NewInstance
 | |
|     /// CHECK:                           ConstructorFence [<<NewInstance>>]
 | |
|     /// CHECK-DAG:                       BoundType
 | |
|     /// CHECK-DAG:                       ConstructorFence [<<This>>]
 | |
|     /// CHECK-NOT:                       ConstructorFence
 | |
|     static class MakeBoundType {
 | |
|       final int abcdefgh;
 | |
|       int x;
 | |
| 
 | |
|       MakeBoundType(int param) {
 | |
|         abcdefgh = param;
 | |
| 
 | |
|         Base b = new Base();
 | |
|         // constructor-fence(b)
 | |
| 
 | |
|         if (this instanceof MakeBoundTypeSub) {
 | |
|           // Create a "BoundType(this)" which prevents
 | |
|           // a merged constructor-fence(this, b)
 | |
|           x = 1;
 | |
|         } else {
 | |
|           x = 2;
 | |
|         }
 | |
| 
 | |
|         // publish(b).
 | |
|         external = b;
 | |
| 
 | |
|         // constructor-fence(this)
 | |
|       }
 | |
| 
 | |
|       @Override
 | |
|       public String toString() {
 | |
|         return getClass().getName() + "(" + baseString() + ")";
 | |
|       }
 | |
| 
 | |
|       protected String baseString() {
 | |
|         return String.format("abcdefgh: %d, x: %d", abcdefgh, x);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     static class MakeBoundTypeSub extends MakeBoundType {
 | |
|       MakeBoundTypeSub(int xyz) {
 | |
|         super(xyz);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| public class Main {
 | |
|   public static void main(String[] args) throws Exception {
 | |
|     // Ensure that all of this code does not get optimized out into a no-op
 | |
|     // by actually running the code with reflection, then validating
 | |
|     // the result by asserting it against a string.
 | |
|     Class<? extends Test>[] testClasses = new Class[] {
 | |
|       TestOneFinal.class,
 | |
|       TestThreeFinal.class,
 | |
|       TestMultiAlloc.class,
 | |
|       TestThreeFinalTwice.class,
 | |
|       TestNonEscaping.Invoke.class,
 | |
|       TestNonEscaping.Store.class,
 | |
|       TestDontOptimizeAcrossBlocks.class,
 | |
|       TestDontOptimizeAcrossEscape.Invoke.class,
 | |
|       TestDontOptimizeAcrossEscape.StoreIput.class,
 | |
|       TestDontOptimizeAcrossEscape.StoreAput.class,
 | |
|       TestDontOptimizeAcrossEscape.StoreSput.class,
 | |
|       TestDontOptimizeAcrossEscape.Deopt.class,
 | |
|       TestDontOptimizeAcrossEscape.Select.class,
 | |
|       TestDontOptimizeAcrossEscape.MakeBoundTypeTest.class,
 | |
|     };
 | |
| 
 | |
|     for (Class<? extends Test> klass : testClasses) {
 | |
|       exerciseTestClass(klass);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Invoke Test#exercise(), then Test#check().
 | |
|    * @throws AssertionError if test fails.
 | |
|    */
 | |
|   private static void exerciseTestClass(Class<? extends Test> klass) throws Exception {
 | |
|     Test instance = klass.cast(klass.getDeclaredConstructor().newInstance());
 | |
| 
 | |
|     // Use reflection as a best-effort to avoid compiler optimizations (e.g. inlining).
 | |
|     instance.getClass().getDeclaredMethod("exercise").invoke(instance);
 | |
|     instance.getClass().getDeclaredMethod("check").invoke(instance);
 | |
|   }
 | |
| 
 | |
|   // Print an object, with special handling for array and null.
 | |
|   public static String valueToString(Object val) {
 | |
|     if (val == null) {
 | |
|       return "<null>";
 | |
|     }
 | |
|     if (val.getClass().isArray()) {
 | |
|       String fmt = "[";
 | |
|       int length = Array.getLength(val);
 | |
|       for (int i = 0; i < length; ++i) {
 | |
|         Object arrayElement = Array.get(val, i);
 | |
|         fmt += valueToString(arrayElement);
 | |
| 
 | |
|         if (i != length - 1) {
 | |
|           fmt += ",";
 | |
|         }
 | |
|       }
 | |
|       fmt += "]";
 | |
| 
 | |
|       return fmt;
 | |
|     }
 | |
| 
 | |
|     return val.toString();
 | |
|   }
 | |
| }
 |