hotspot/test/compiler/unsafe/UnsafeGetConstantField.java
author vlivanov
Mon, 29 Feb 2016 23:46:55 +0300
changeset 36337 d4b2f60ff5a9
parent 35099 982950884444
child 36508 5f9eee6b383b
permissions -rw-r--r--
8150543: Mismatched access detection is inaccurate Reviewed-by: kvn, shade
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
31857
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     1
/*
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     2
 * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     4
 *
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    10
 *
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    15
 * accompanied this code).
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    16
 *
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    20
 *
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    23
 * questions.
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    24
 */
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    25
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    26
/*
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    27
 * @test
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    28
 * @summary tests on constant folding of unsafe get operations
33730
30e064828045 8140189: [TESTBUG] Get rid of "@library /../../test/lib" in jtreg tests
cjplummer
parents: 33606
diff changeset
    29
 * @library /testlibrary /test/lib
36337
d4b2f60ff5a9 8150543: Mismatched access detection is inaccurate
vlivanov
parents: 35099
diff changeset
    30
 *
d4b2f60ff5a9 8150543: Mismatched access detection is inaccurate
vlivanov
parents: 35099
diff changeset
    31
 * @requires vm.flavor != "client"
d4b2f60ff5a9 8150543: Mismatched access detection is inaccurate
vlivanov
parents: 35099
diff changeset
    32
 *
31857
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    33
 * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    34
 *                   -Xbatch -XX:-TieredCompilation
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    35
 *                   -XX:+FoldStableValues
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    36
 *                   -XX:+UseUnalignedAccesses
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    37
 *                   java.lang.invoke.UnsafeGetConstantField
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    38
 * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    39
 *                   -Xbatch -XX:-TieredCompilation
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    40
 *                   -XX:+FoldStableValues
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    41
 *                   -XX:-UseUnalignedAccesses
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    42
 *                   java.lang.invoke.UnsafeGetConstantField
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    43
 */
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    44
package java.lang.invoke;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    45
35099
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    46
import jdk.internal.vm.annotation.DontInline;
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    47
import jdk.internal.vm.annotation.Stable;
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    48
import jdk.internal.misc.Unsafe;
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    49
import jdk.internal.org.objectweb.asm.ClassWriter;
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    50
import jdk.internal.org.objectweb.asm.FieldVisitor;
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    51
import jdk.internal.org.objectweb.asm.MethodVisitor;
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    52
import jdk.internal.org.objectweb.asm.Opcodes;
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    53
import jdk.internal.org.objectweb.asm.Type;
31857
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    54
import jdk.test.lib.Asserts;
35099
982950884444 8144223: Move j.l.invoke.{ForceInline, DontInline, Stable} to jdk.internal.vm.annotation package
psandoz
parents: 33730
diff changeset
    55
31857
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    56
import static jdk.internal.org.objectweb.asm.Opcodes.*;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    57
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    58
public class UnsafeGetConstantField {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    59
    static final Class<?> THIS_CLASS = UnsafeGetConstantField.class;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    60
33606
af4ec8a4635b 8139891: Prepare Unsafe for true encapsulation
chegar
parents: 31857
diff changeset
    61
    static final Unsafe U = Unsafe.getUnsafe();
31857
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    62
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    63
    public static void main(String[] args) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    64
        testUnsafeGetAddress();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    65
        testUnsafeGetField();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    66
        testUnsafeGetFieldUnaligned();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    67
        System.out.println("TEST PASSED");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    68
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    69
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    70
    static final long nativeAddr = U.allocateMemory(16);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    71
    static void testUnsafeGetAddress() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    72
        long cookie = 0x12345678L;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    73
        U.putAddress(nativeAddr, cookie);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    74
        for (int i = 0; i < 20_000; i++) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    75
            Asserts.assertEquals(checkGetAddress(), cookie);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    76
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    77
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    78
    @DontInline
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    79
    static long checkGetAddress() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    80
        return U.getAddress(nativeAddr);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    81
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    82
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    83
    static void testUnsafeGetField() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    84
        int[] testedFlags = new int[] { 0, ACC_STATIC, ACC_FINAL, (ACC_STATIC | ACC_FINAL) };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    85
        boolean[] boolValues = new boolean[] { false, true };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    86
        String[] modes = new String[] { "", "Volatile" };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    87
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    88
        for (JavaType t : JavaType.values()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    89
            for (int flags : testedFlags) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    90
                for (boolean stable : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    91
                    for (boolean hasDefaultValue : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    92
                        for (String suffix : modes) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    93
                            runTest(t, flags, stable, hasDefaultValue, suffix);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    94
                        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    95
                    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    96
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    97
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    98
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    99
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   100
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   101
    static void testUnsafeGetFieldUnaligned() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   102
        JavaType[] types = new JavaType[] { JavaType.S, JavaType.C, JavaType.I, JavaType.J };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   103
        int[] testedFlags = new int[] { 0, ACC_STATIC, ACC_FINAL, (ACC_STATIC | ACC_FINAL) };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   104
        boolean[] boolValues = new boolean[] { false, true };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   105
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   106
        for (JavaType t : types) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   107
            for (int flags : testedFlags) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   108
                for (boolean stable : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   109
                    for (boolean hasDefaultValue : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   110
                        runTest(t, flags, stable, hasDefaultValue, "Unaligned");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   111
                    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   112
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   113
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   114
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   115
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   116
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   117
    static void runTest(JavaType t, int flags, boolean stable, boolean hasDefaultValue, String postfix) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   118
        Generator g = new Generator(t, flags, stable, hasDefaultValue, postfix);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   119
        Test test = g.generate();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   120
        System.out.printf("type=%s flags=%d stable=%b default=%b post=%s\n",
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   121
                          t.typeName, flags, stable, hasDefaultValue, postfix);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   122
        // Trigger compilation
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   123
        for (int i = 0; i < 20_000; i++) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   124
            Asserts.assertEQ(test.testDirect(), test.testUnsafe());
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   125
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   126
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   127
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   128
    interface Test {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   129
        Object testDirect();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   130
        Object testUnsafe();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   131
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   132
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   133
    enum JavaType {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   134
        Z("Boolean", true),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   135
        B("Byte", new Byte((byte)-1)),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   136
        S("Short", new Short((short)-1)),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   137
        C("Char", Character.MAX_VALUE),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   138
        I("Int", -1),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   139
        J("Long", -1L),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   140
        F("Float", -1F),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   141
        D("Double", -1D),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   142
        L("Object", new Object());
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   143
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   144
        String typeName;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   145
        Object value;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   146
        String wrapper;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   147
        JavaType(String name, Object value) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   148
            this.typeName = name;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   149
            this.value = value;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   150
            this.wrapper = internalName(value.getClass());
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   151
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   152
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   153
        String desc() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   154
            if (this == JavaType.L) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   155
                return "Ljava/lang/Object;";
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   156
            } else {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   157
                return toString();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   158
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   159
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   160
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   161
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   162
    static String internalName(Class cls) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   163
        return cls.getName().replace('.', '/');
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   164
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   165
    static String descriptor(Class cls) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   166
        return String.format("L%s;", internalName(cls));
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   167
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   168
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   169
    /**
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   170
     * Sample generated class:
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   171
     * static class T1 implements Test {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   172
     *   final int f = -1;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   173
     *   static final long FIELD_OFFSET;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   174
     *   static final T1 t = new T1();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   175
     *   static {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   176
     *     FIELD_OFFSET = U.objectFieldOffset(T1.class.getDeclaredField("f"));
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   177
     *   }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   178
     *   public Object testDirect()  { return t.f; }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   179
     *   public Object testUnsafe()  { return U.getInt(t, FIELD_OFFSET); }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   180
     * }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   181
     */
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   182
    static class Generator {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   183
        static final String FIELD_NAME = "f";
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   184
        static final String UNSAFE_NAME = internalName(Unsafe.class);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   185
        static final String UNSAFE_DESC = descriptor(Unsafe.class);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   186
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   187
        final JavaType type;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   188
        final int flags;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   189
        final boolean stable;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   190
        final boolean hasDefaultValue;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   191
        final String nameSuffix;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   192
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   193
        final String className;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   194
        final String classDesc;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   195
        final String fieldDesc;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   196
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   197
        Generator(JavaType t, int flags, boolean stable, boolean hasDefaultValue, String suffix) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   198
            this.type = t;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   199
            this.flags = flags;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   200
            this.stable = stable;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   201
            this.hasDefaultValue = hasDefaultValue;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   202
            this.nameSuffix = suffix;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   203
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   204
            fieldDesc = type.desc();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   205
            className = String.format("%s$Test%s%s__f=%d__s=%b__d=%b", internalName(THIS_CLASS), type.typeName,
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   206
                                      suffix, flags, stable, hasDefaultValue);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   207
            classDesc = String.format("L%s;", className);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   208
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   209
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   210
        byte[] generateClassFile() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   211
            ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS | ClassWriter.COMPUTE_FRAMES);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   212
            cw.visit(Opcodes.V1_8, Opcodes.ACC_PUBLIC | Opcodes.ACC_SUPER, className, null, "java/lang/Object",
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   213
                    new String[]{ internalName(Test.class) });
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   214
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   215
            // Declare fields
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   216
            cw.visitField(ACC_FINAL | ACC_STATIC, "t", classDesc, null, null).visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   217
            cw.visitField(ACC_FINAL | ACC_STATIC, "FIELD_OFFSET", "J", null, null).visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   218
            cw.visitField(ACC_FINAL | ACC_STATIC, "U", UNSAFE_DESC, null, null).visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   219
            if (isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   220
                cw.visitField(ACC_FINAL | ACC_STATIC, "STATIC_BASE", "Ljava/lang/Object;", null, null).visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   221
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   222
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   223
            FieldVisitor fv = cw.visitField(flags, FIELD_NAME, fieldDesc, null, null);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   224
            if (stable) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   225
                fv.visitAnnotation(descriptor(Stable.class), true);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   226
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   227
            fv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   228
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   229
            // Methods
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   230
            {   // <init>
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   231
                MethodVisitor mv = cw.visitMethod(0, "<init>", "()V", null, null);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   232
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   233
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   234
                mv.visitVarInsn(ALOAD, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   235
                mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   236
                if (!isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   237
                    initField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   238
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   239
                mv.visitInsn(RETURN);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   240
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   241
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   242
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   243
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   244
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   245
            {   // public Object testDirect() { return t.f; }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   246
                MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "testDirect", "()Ljava/lang/Object;", null, null);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   247
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   248
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   249
                getFieldValue(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   250
                wrapResult(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   251
                mv.visitInsn(ARETURN);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   252
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   253
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   254
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   255
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   256
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   257
            {   // public Object testUnsafe() { return U.getInt(t, FIELD_OFFSET); }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   258
                MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "testUnsafe", "()Ljava/lang/Object;", null, null);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   259
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   260
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   261
                getFieldValueUnsafe(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   262
                wrapResult(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   263
                mv.visitInsn(ARETURN);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   264
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   265
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   266
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   267
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   268
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   269
            {   // <clinit>
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   270
                MethodVisitor mv = cw.visitMethod(ACC_STATIC, "<clinit>", "()V", null, null);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   271
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   272
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   273
                // Cache Unsafe instance
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   274
                mv.visitMethodInsn(INVOKESTATIC, UNSAFE_NAME, "getUnsafe", "()"+UNSAFE_DESC, false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   275
                mv.visitFieldInsn(PUTSTATIC, className, "U", UNSAFE_DESC);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   276
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   277
                // Create test object instance
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   278
                mv.visitTypeInsn(NEW, className);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   279
                mv.visitInsn(DUP);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   280
                mv.visitMethodInsn(INVOKESPECIAL, className, "<init>", "()V", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   281
                mv.visitFieldInsn(PUTSTATIC, className, "t", classDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   282
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   283
                // Compute field offset
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   284
                getUnsafe(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   285
                getField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   286
                mv.visitMethodInsn(INVOKEVIRTUAL, UNSAFE_NAME, (isStatic() ? "staticFieldOffset" : "objectFieldOffset"),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   287
                        "(Ljava/lang/reflect/Field;)J", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   288
                mv.visitFieldInsn(PUTSTATIC, className, "FIELD_OFFSET", "J");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   289
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   290
                // Compute base offset for static field
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   291
                if (isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   292
                    getUnsafe(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   293
                    getField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   294
                    mv.visitMethodInsn(INVOKEVIRTUAL, UNSAFE_NAME, "staticFieldBase", "(Ljava/lang/reflect/Field;)Ljava/lang/Object;", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   295
                    mv.visitFieldInsn(PUTSTATIC, className, "STATIC_BASE", "Ljava/lang/Object;");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   296
                    initField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   297
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   298
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   299
                mv.visitInsn(RETURN);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   300
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   301
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   302
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   303
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   304
            return cw.toByteArray();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   305
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   306
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   307
        Test generate() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   308
            byte[] classFile = generateClassFile();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   309
            Class<?> c = U.defineClass(className, classFile, 0, classFile.length, THIS_CLASS.getClassLoader(), null);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   310
            try {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   311
                return (Test) c.newInstance();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   312
            } catch(Exception e) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   313
                throw new Error(e);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   314
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   315
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   316
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   317
        boolean isStatic() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   318
            return (flags & ACC_STATIC) > 0;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   319
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   320
        boolean isFinal() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   321
            return (flags & ACC_FINAL) > 0;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   322
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   323
        void getUnsafe(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   324
            mv.visitFieldInsn(GETSTATIC, className, "U", UNSAFE_DESC);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   325
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   326
        void getField(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   327
            mv.visitLdcInsn(Type.getType(classDesc));
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   328
            mv.visitLdcInsn(FIELD_NAME);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   329
            mv.visitMethodInsn(INVOKEVIRTUAL, "java/lang/Class", "getDeclaredField", "(Ljava/lang/String;)Ljava/lang/reflect/Field;", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   330
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   331
        void getFieldValue(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   332
            if (isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   333
                mv.visitFieldInsn(GETSTATIC, className, FIELD_NAME, fieldDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   334
            } else {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   335
                mv.visitFieldInsn(GETSTATIC, className, "t", classDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   336
                mv.visitFieldInsn(GETFIELD, className, FIELD_NAME, fieldDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   337
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   338
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   339
        void getFieldValueUnsafe(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   340
            getUnsafe(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   341
            if (isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   342
                mv.visitFieldInsn(GETSTATIC, className, "STATIC_BASE", "Ljava/lang/Object;");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   343
            } else {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   344
                mv.visitFieldInsn(GETSTATIC, className, "t", classDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   345
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   346
            mv.visitFieldInsn(GETSTATIC, className, "FIELD_OFFSET", "J");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   347
            String name = "get" + type.typeName + nameSuffix;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   348
            mv.visitMethodInsn(INVOKEVIRTUAL, UNSAFE_NAME, name, "(Ljava/lang/Object;J)" + type.desc(), false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   349
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   350
        void wrapResult(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   351
            if (type != JavaType.L) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   352
                String desc = String.format("(%s)L%s;", type.desc(), type.wrapper);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   353
                mv.visitMethodInsn(INVOKESTATIC, type.wrapper, "valueOf", desc, false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   354
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   355
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   356
        void initField(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   357
            if (hasDefaultValue) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   358
                return; // Nothing to do
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   359
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   360
            if (!isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   361
                mv.visitVarInsn(ALOAD, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   362
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   363
            switch (type) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   364
                case L: {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   365
                    mv.visitTypeInsn(NEW, "java/lang/Object");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   366
                    mv.visitInsn(DUP);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   367
                    mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   368
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   369
                    break;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   370
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   371
                default: {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   372
                    mv.visitLdcInsn(type.value);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   373
                    break;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   374
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   375
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   376
            mv.visitFieldInsn((isStatic() ? PUTSTATIC : PUTFIELD), className, FIELD_NAME, fieldDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   377
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   378
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   379
}