hotspot/test/compiler/unsafe/UnsafeGetConstantField.java
author chegar
Tue, 27 Oct 2015 14:18:56 +0000
changeset 33606 af4ec8a4635b
parent 31857 adbf29d9ca43
child 33730 30e064828045
permissions -rw-r--r--
8139891: Prepare Unsafe for true encapsulation Reviewed-by: alanb, dholmes, jrose, psandoz, twisti
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
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    29
 * @library /testlibrary /../../test/lib
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    30
 * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    31
 *                   -Xbatch -XX:-TieredCompilation
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    32
 *                   -XX:+FoldStableValues
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    33
 *                   -XX:+UseUnalignedAccesses
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    34
 *                   java.lang.invoke.UnsafeGetConstantField
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    35
 * @run main/bootclasspath -XX:+UnlockDiagnosticVMOptions
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    36
 *                   -Xbatch -XX:-TieredCompilation
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    37
 *                   -XX:+FoldStableValues
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    38
 *                   -XX:-UseUnalignedAccesses
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    39
 *                   java.lang.invoke.UnsafeGetConstantField
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    40
 */
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    41
package java.lang.invoke;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    42
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    43
import jdk.internal.org.objectweb.asm.*;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    44
import jdk.test.lib.Asserts;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    45
import jdk.test.lib.Utils;
33606
af4ec8a4635b 8139891: Prepare Unsafe for true encapsulation
chegar
parents: 31857
diff changeset
    46
import jdk.internal.misc.Unsafe;
31857
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    47
import static jdk.internal.org.objectweb.asm.Opcodes.*;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    48
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    49
public class UnsafeGetConstantField {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    50
    static final Class<?> THIS_CLASS = UnsafeGetConstantField.class;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    51
33606
af4ec8a4635b 8139891: Prepare Unsafe for true encapsulation
chegar
parents: 31857
diff changeset
    52
    static final Unsafe U = Unsafe.getUnsafe();
31857
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    53
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    54
    public static void main(String[] args) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    55
        testUnsafeGetAddress();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    56
        testUnsafeGetField();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    57
        testUnsafeGetFieldUnaligned();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    58
        System.out.println("TEST PASSED");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    59
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    60
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    61
    static final long nativeAddr = U.allocateMemory(16);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    62
    static void testUnsafeGetAddress() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    63
        long cookie = 0x12345678L;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    64
        U.putAddress(nativeAddr, cookie);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    65
        for (int i = 0; i < 20_000; i++) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    66
            Asserts.assertEquals(checkGetAddress(), cookie);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    67
        }
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
    @DontInline
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    70
    static long checkGetAddress() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    71
        return U.getAddress(nativeAddr);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    72
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    73
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    74
    static void testUnsafeGetField() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    75
        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
    76
        boolean[] boolValues = new boolean[] { false, true };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    77
        String[] modes = new String[] { "", "Volatile" };
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    78
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    79
        for (JavaType t : JavaType.values()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    80
            for (int flags : testedFlags) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    81
                for (boolean stable : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    82
                    for (boolean hasDefaultValue : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    83
                        for (String suffix : modes) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    84
                            runTest(t, flags, stable, hasDefaultValue, suffix);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    85
                        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    86
                    }
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
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    89
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    90
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    91
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    92
    static void testUnsafeGetFieldUnaligned() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    93
        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
    94
        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
    95
        boolean[] boolValues = new boolean[] { false, true };
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
        for (JavaType t : types) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    98
            for (int flags : testedFlags) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
    99
                for (boolean stable : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   100
                    for (boolean hasDefaultValue : boolValues) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   101
                        runTest(t, flags, stable, hasDefaultValue, "Unaligned");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   102
                    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   103
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   104
            }
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
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   107
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   108
    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
   109
        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
   110
        Test test = g.generate();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   111
        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
   112
                          t.typeName, flags, stable, hasDefaultValue, postfix);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   113
        // Trigger compilation
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   114
        for (int i = 0; i < 20_000; i++) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   115
            Asserts.assertEQ(test.testDirect(), test.testUnsafe());
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
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   118
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   119
    interface Test {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   120
        Object testDirect();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   121
        Object testUnsafe();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   122
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   123
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   124
    enum JavaType {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   125
        Z("Boolean", true),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   126
        B("Byte", new Byte((byte)-1)),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   127
        S("Short", new Short((short)-1)),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   128
        C("Char", Character.MAX_VALUE),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   129
        I("Int", -1),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   130
        J("Long", -1L),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   131
        F("Float", -1F),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   132
        D("Double", -1D),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   133
        L("Object", new Object());
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   134
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   135
        String typeName;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   136
        Object value;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   137
        String wrapper;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   138
        JavaType(String name, Object value) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   139
            this.typeName = name;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   140
            this.value = value;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   141
            this.wrapper = internalName(value.getClass());
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   142
        }
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 desc() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   145
            if (this == JavaType.L) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   146
                return "Ljava/lang/Object;";
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   147
            } else {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   148
                return toString();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   149
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   150
        }
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
    static String internalName(Class cls) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   154
        return cls.getName().replace('.', '/');
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   155
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   156
    static String descriptor(Class cls) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   157
        return String.format("L%s;", internalName(cls));
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
     * Sample generated class:
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   162
     * static class T1 implements Test {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   163
     *   final int f = -1;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   164
     *   static final long FIELD_OFFSET;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   165
     *   static final T1 t = new T1();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   166
     *   static {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   167
     *     FIELD_OFFSET = U.objectFieldOffset(T1.class.getDeclaredField("f"));
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
     *   public Object testDirect()  { return t.f; }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   170
     *   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
   171
     * }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   172
     */
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   173
    static class Generator {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   174
        static final String FIELD_NAME = "f";
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   175
        static final String UNSAFE_NAME = internalName(Unsafe.class);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   176
        static final String UNSAFE_DESC = descriptor(Unsafe.class);
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
        final JavaType type;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   179
        final int flags;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   180
        final boolean stable;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   181
        final boolean hasDefaultValue;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   182
        final String nameSuffix;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   183
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   184
        final String className;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   185
        final String classDesc;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   186
        final String fieldDesc;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   187
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   188
        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
   189
            this.type = t;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   190
            this.flags = flags;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   191
            this.stable = stable;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   192
            this.hasDefaultValue = hasDefaultValue;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   193
            this.nameSuffix = suffix;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   194
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   195
            fieldDesc = type.desc();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   196
            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
   197
                                      suffix, flags, stable, hasDefaultValue);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   198
            classDesc = String.format("L%s;", className);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   199
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   200
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   201
        byte[] generateClassFile() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   202
            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
   203
            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
   204
                    new String[]{ internalName(Test.class) });
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   205
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   206
            // Declare fields
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   207
            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
   208
            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
   209
            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
   210
            if (isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   211
                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
   212
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   213
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   214
            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
   215
            if (stable) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   216
                fv.visitAnnotation(descriptor(Stable.class), true);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   217
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   218
            fv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   219
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   220
            // Methods
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   221
            {   // <init>
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   222
                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
   223
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   224
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   225
                mv.visitVarInsn(ALOAD, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   226
                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
   227
                if (!isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   228
                    initField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   229
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   230
                mv.visitInsn(RETURN);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   231
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   232
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   233
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   234
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   235
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   236
            {   // public Object testDirect() { return t.f; }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   237
                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
   238
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   239
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   240
                getFieldValue(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   241
                wrapResult(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   242
                mv.visitInsn(ARETURN);
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
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   245
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   246
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   247
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   248
            {   // 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
   249
                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
   250
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   251
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   252
                getFieldValueUnsafe(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   253
                wrapResult(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   254
                mv.visitInsn(ARETURN);
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
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   257
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   258
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   259
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   260
            {   // <clinit>
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   261
                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
   262
                mv.visitCode();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   263
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   264
                // Cache Unsafe instance
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   265
                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
   266
                mv.visitFieldInsn(PUTSTATIC, className, "U", UNSAFE_DESC);
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
                // Create test object instance
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   269
                mv.visitTypeInsn(NEW, className);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   270
                mv.visitInsn(DUP);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   271
                mv.visitMethodInsn(INVOKESPECIAL, className, "<init>", "()V", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   272
                mv.visitFieldInsn(PUTSTATIC, className, "t", classDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   273
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   274
                // Compute field offset
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   275
                getUnsafe(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   276
                getField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   277
                mv.visitMethodInsn(INVOKEVIRTUAL, UNSAFE_NAME, (isStatic() ? "staticFieldOffset" : "objectFieldOffset"),
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   278
                        "(Ljava/lang/reflect/Field;)J", false);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   279
                mv.visitFieldInsn(PUTSTATIC, className, "FIELD_OFFSET", "J");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   280
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   281
                // Compute base offset for static field
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   282
                if (isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   283
                    getUnsafe(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   284
                    getField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   285
                    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
   286
                    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
   287
                    initField(mv);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   288
                }
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
                mv.visitInsn(RETURN);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   291
                mv.visitMaxs(0, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   292
                mv.visitEnd();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   293
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   294
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   295
            return cw.toByteArray();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   296
        }
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
        Test generate() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   299
            byte[] classFile = generateClassFile();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   300
            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
   301
            try {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   302
                return (Test) c.newInstance();
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   303
            } catch(Exception e) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   304
                throw new Error(e);
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
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   308
        boolean isStatic() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   309
            return (flags & ACC_STATIC) > 0;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   310
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   311
        boolean isFinal() {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   312
            return (flags & ACC_FINAL) > 0;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   313
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   314
        void getUnsafe(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   315
            mv.visitFieldInsn(GETSTATIC, className, "U", UNSAFE_DESC);
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
        void getField(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   318
            mv.visitLdcInsn(Type.getType(classDesc));
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   319
            mv.visitLdcInsn(FIELD_NAME);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   320
            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
   321
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   322
        void getFieldValue(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   323
            if (isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   324
                mv.visitFieldInsn(GETSTATIC, className, FIELD_NAME, fieldDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   325
            } else {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   326
                mv.visitFieldInsn(GETSTATIC, className, "t", classDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   327
                mv.visitFieldInsn(GETFIELD, className, FIELD_NAME, fieldDesc);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   328
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   329
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   330
        void getFieldValueUnsafe(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   331
            getUnsafe(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, "STATIC_BASE", "Ljava/lang/Object;");
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
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   337
            mv.visitFieldInsn(GETSTATIC, className, "FIELD_OFFSET", "J");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   338
            String name = "get" + type.typeName + nameSuffix;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   339
            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
   340
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   341
        void wrapResult(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   342
            if (type != JavaType.L) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   343
                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
   344
                mv.visitMethodInsn(INVOKESTATIC, type.wrapper, "valueOf", desc, false);
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
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   347
        void initField(MethodVisitor mv) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   348
            if (hasDefaultValue) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   349
                return; // Nothing to do
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   350
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   351
            if (!isStatic()) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   352
                mv.visitVarInsn(ALOAD, 0);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   353
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   354
            switch (type) {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   355
                case L: {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   356
                    mv.visitTypeInsn(NEW, "java/lang/Object");
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   357
                    mv.visitInsn(DUP);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   358
                    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
   359
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   360
                    break;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   361
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   362
                default: {
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   363
                    mv.visitLdcInsn(type.value);
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   364
                    break;
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   365
                }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   366
            }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   367
            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
   368
        }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   369
    }
adbf29d9ca43 8078629: VM should constant fold Unsafe.get*() loads from final fields
vlivanov
parents:
diff changeset
   370
}