rt/emul/compact/src/main/java/java/lang/invoke/TypeConvertingMethodAdapter.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Sat, 09 Aug 2014 11:11:13 +0200
branchjdk8-b132
changeset 1646 c880a8a8803b
permissions -rw-r--r--
Batch of classes necessary to implement invoke dynamic interfaces. Taken from JDK8 build 132
jaroslav@1646
     1
/*
jaroslav@1646
     2
 * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved.
jaroslav@1646
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
jaroslav@1646
     4
 *
jaroslav@1646
     5
 * This code is free software; you can redistribute it and/or modify it
jaroslav@1646
     6
 * under the terms of the GNU General Public License version 2 only, as
jaroslav@1646
     7
 * published by the Free Software Foundation.  Oracle designates this
jaroslav@1646
     8
 * particular file as subject to the "Classpath" exception as provided
jaroslav@1646
     9
 * by Oracle in the LICENSE file that accompanied this code.
jaroslav@1646
    10
 *
jaroslav@1646
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
jaroslav@1646
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
jaroslav@1646
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
jaroslav@1646
    14
 * version 2 for more details (a copy is included in the LICENSE file that
jaroslav@1646
    15
 * accompanied this code).
jaroslav@1646
    16
 *
jaroslav@1646
    17
 * You should have received a copy of the GNU General Public License version
jaroslav@1646
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
jaroslav@1646
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
jaroslav@1646
    20
 *
jaroslav@1646
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
jaroslav@1646
    22
 * or visit www.oracle.com if you need additional information or have any
jaroslav@1646
    23
 * questions.
jaroslav@1646
    24
 */
jaroslav@1646
    25
jaroslav@1646
    26
package java.lang.invoke;
jaroslav@1646
    27
jaroslav@1646
    28
import jdk.internal.org.objectweb.asm.MethodVisitor;
jaroslav@1646
    29
import jdk.internal.org.objectweb.asm.Opcodes;
jaroslav@1646
    30
import jdk.internal.org.objectweb.asm.Type;
jaroslav@1646
    31
import sun.invoke.util.BytecodeDescriptor;
jaroslav@1646
    32
import sun.invoke.util.Wrapper;
jaroslav@1646
    33
import static sun.invoke.util.Wrapper.*;
jaroslav@1646
    34
jaroslav@1646
    35
class TypeConvertingMethodAdapter extends MethodVisitor {
jaroslav@1646
    36
jaroslav@1646
    37
    TypeConvertingMethodAdapter(MethodVisitor mv) {
jaroslav@1646
    38
        super(Opcodes.ASM5, mv);
jaroslav@1646
    39
    }
jaroslav@1646
    40
jaroslav@1646
    41
    private static final int NUM_WRAPPERS = Wrapper.values().length;
jaroslav@1646
    42
jaroslav@1646
    43
    private static final String NAME_OBJECT = "java/lang/Object";
jaroslav@1646
    44
    private static final String WRAPPER_PREFIX = "Ljava/lang/";
jaroslav@1646
    45
jaroslav@1646
    46
    // Same for all primitives; name of the boxing method
jaroslav@1646
    47
    private static final String NAME_BOX_METHOD = "valueOf";
jaroslav@1646
    48
jaroslav@1646
    49
    // Table of opcodes for widening primitive conversions; NOP = no conversion
jaroslav@1646
    50
    private static final int[][] wideningOpcodes = new int[NUM_WRAPPERS][NUM_WRAPPERS];
jaroslav@1646
    51
jaroslav@1646
    52
    private static final Wrapper[] FROM_WRAPPER_NAME = new Wrapper[16];
jaroslav@1646
    53
jaroslav@1646
    54
    // Table of wrappers for primitives, indexed by ASM type sorts
jaroslav@1646
    55
    private static final Wrapper[] FROM_TYPE_SORT = new Wrapper[16];
jaroslav@1646
    56
jaroslav@1646
    57
    static {
jaroslav@1646
    58
        for (Wrapper w : Wrapper.values()) {
jaroslav@1646
    59
            if (w.basicTypeChar() != 'L') {
jaroslav@1646
    60
                int wi = hashWrapperName(w.wrapperSimpleName());
jaroslav@1646
    61
                assert (FROM_WRAPPER_NAME[wi] == null);
jaroslav@1646
    62
                FROM_WRAPPER_NAME[wi] = w;
jaroslav@1646
    63
            }
jaroslav@1646
    64
        }
jaroslav@1646
    65
jaroslav@1646
    66
        for (int i = 0; i < NUM_WRAPPERS; i++) {
jaroslav@1646
    67
            for (int j = 0; j < NUM_WRAPPERS; j++) {
jaroslav@1646
    68
                wideningOpcodes[i][j] = Opcodes.NOP;
jaroslav@1646
    69
            }
jaroslav@1646
    70
        }
jaroslav@1646
    71
jaroslav@1646
    72
        initWidening(LONG,   Opcodes.I2L, BYTE, SHORT, INT, CHAR);
jaroslav@1646
    73
        initWidening(LONG,   Opcodes.F2L, FLOAT);
jaroslav@1646
    74
        initWidening(FLOAT,  Opcodes.I2F, BYTE, SHORT, INT, CHAR);
jaroslav@1646
    75
        initWidening(FLOAT,  Opcodes.L2F, LONG);
jaroslav@1646
    76
        initWidening(DOUBLE, Opcodes.I2D, BYTE, SHORT, INT, CHAR);
jaroslav@1646
    77
        initWidening(DOUBLE, Opcodes.F2D, FLOAT);
jaroslav@1646
    78
        initWidening(DOUBLE, Opcodes.L2D, LONG);
jaroslav@1646
    79
jaroslav@1646
    80
        FROM_TYPE_SORT[Type.BYTE] = Wrapper.BYTE;
jaroslav@1646
    81
        FROM_TYPE_SORT[Type.SHORT] = Wrapper.SHORT;
jaroslav@1646
    82
        FROM_TYPE_SORT[Type.INT] = Wrapper.INT;
jaroslav@1646
    83
        FROM_TYPE_SORT[Type.LONG] = Wrapper.LONG;
jaroslav@1646
    84
        FROM_TYPE_SORT[Type.CHAR] = Wrapper.CHAR;
jaroslav@1646
    85
        FROM_TYPE_SORT[Type.FLOAT] = Wrapper.FLOAT;
jaroslav@1646
    86
        FROM_TYPE_SORT[Type.DOUBLE] = Wrapper.DOUBLE;
jaroslav@1646
    87
        FROM_TYPE_SORT[Type.BOOLEAN] = Wrapper.BOOLEAN;
jaroslav@1646
    88
    }
jaroslav@1646
    89
jaroslav@1646
    90
    private static void initWidening(Wrapper to, int opcode, Wrapper... from) {
jaroslav@1646
    91
        for (Wrapper f : from) {
jaroslav@1646
    92
            wideningOpcodes[f.ordinal()][to.ordinal()] = opcode;
jaroslav@1646
    93
        }
jaroslav@1646
    94
    }
jaroslav@1646
    95
jaroslav@1646
    96
    /**
jaroslav@1646
    97
     * Class name to Wrapper hash, derived from Wrapper.hashWrap()
jaroslav@1646
    98
     * @param xn
jaroslav@1646
    99
     * @return The hash code 0-15
jaroslav@1646
   100
     */
jaroslav@1646
   101
    private static int hashWrapperName(String xn) {
jaroslav@1646
   102
        if (xn.length() < 3) {
jaroslav@1646
   103
            return 0;
jaroslav@1646
   104
        }
jaroslav@1646
   105
        return (3 * xn.charAt(1) + xn.charAt(2)) % 16;
jaroslav@1646
   106
    }
jaroslav@1646
   107
jaroslav@1646
   108
    private Wrapper wrapperOrNullFromDescriptor(String desc) {
jaroslav@1646
   109
        if (!desc.startsWith(WRAPPER_PREFIX)) {
jaroslav@1646
   110
            // Not a class type (array or method), so not a boxed type
jaroslav@1646
   111
            // or not in the right package
jaroslav@1646
   112
            return null;
jaroslav@1646
   113
        }
jaroslav@1646
   114
        // Pare it down to the simple class name
jaroslav@1646
   115
        String cname = desc.substring(WRAPPER_PREFIX.length(), desc.length() - 1);
jaroslav@1646
   116
        // Hash to a Wrapper
jaroslav@1646
   117
        Wrapper w = FROM_WRAPPER_NAME[hashWrapperName(cname)];
jaroslav@1646
   118
        if (w == null || w.wrapperSimpleName().equals(cname)) {
jaroslav@1646
   119
            return w;
jaroslav@1646
   120
        } else {
jaroslav@1646
   121
            return null;
jaroslav@1646
   122
        }
jaroslav@1646
   123
    }
jaroslav@1646
   124
jaroslav@1646
   125
    private static String wrapperName(Wrapper w) {
jaroslav@1646
   126
        return "java/lang/" + w.wrapperSimpleName();
jaroslav@1646
   127
    }
jaroslav@1646
   128
jaroslav@1646
   129
    private static String unboxMethod(Wrapper w) {
jaroslav@1646
   130
        return w.primitiveSimpleName() + "Value";
jaroslav@1646
   131
    }
jaroslav@1646
   132
jaroslav@1646
   133
    private static String boxingDescriptor(Wrapper w) {
jaroslav@1646
   134
        return String.format("(%s)L%s;", w.basicTypeChar(), wrapperName(w));
jaroslav@1646
   135
    }
jaroslav@1646
   136
jaroslav@1646
   137
    private static String unboxingDescriptor(Wrapper w) {
jaroslav@1646
   138
        return "()" + w.basicTypeChar();
jaroslav@1646
   139
    }
jaroslav@1646
   140
jaroslav@1646
   141
    void boxIfTypePrimitive(Type t) {
jaroslav@1646
   142
        Wrapper w = FROM_TYPE_SORT[t.getSort()];
jaroslav@1646
   143
        if (w != null) {
jaroslav@1646
   144
            box(w);
jaroslav@1646
   145
        }
jaroslav@1646
   146
    }
jaroslav@1646
   147
jaroslav@1646
   148
    void widen(Wrapper ws, Wrapper wt) {
jaroslav@1646
   149
        if (ws != wt) {
jaroslav@1646
   150
            int opcode = wideningOpcodes[ws.ordinal()][wt.ordinal()];
jaroslav@1646
   151
            if (opcode != Opcodes.NOP) {
jaroslav@1646
   152
                visitInsn(opcode);
jaroslav@1646
   153
            }
jaroslav@1646
   154
        }
jaroslav@1646
   155
    }
jaroslav@1646
   156
jaroslav@1646
   157
    void box(Wrapper w) {
jaroslav@1646
   158
        visitMethodInsn(Opcodes.INVOKESTATIC,
jaroslav@1646
   159
                wrapperName(w),
jaroslav@1646
   160
                NAME_BOX_METHOD,
jaroslav@1646
   161
                boxingDescriptor(w));
jaroslav@1646
   162
    }
jaroslav@1646
   163
jaroslav@1646
   164
    /**
jaroslav@1646
   165
     * Convert types by unboxing. The source type is known to be a primitive wrapper.
jaroslav@1646
   166
     * @param ws A primitive wrapper corresponding to wrapped reference source type
jaroslav@1646
   167
     * @param wt A primitive wrapper being converted to
jaroslav@1646
   168
     */
jaroslav@1646
   169
    void unbox(String sname, Wrapper wt) {
jaroslav@1646
   170
        visitMethodInsn(Opcodes.INVOKEVIRTUAL,
jaroslav@1646
   171
                sname,
jaroslav@1646
   172
                unboxMethod(wt),
jaroslav@1646
   173
                unboxingDescriptor(wt));
jaroslav@1646
   174
    }
jaroslav@1646
   175
jaroslav@1646
   176
    private String descriptorToName(String desc) {
jaroslav@1646
   177
        int last = desc.length() - 1;
jaroslav@1646
   178
        if (desc.charAt(0) == 'L' && desc.charAt(last) == ';') {
jaroslav@1646
   179
            // In descriptor form
jaroslav@1646
   180
            return desc.substring(1, last);
jaroslav@1646
   181
        } else {
jaroslav@1646
   182
            // Already in internal name form
jaroslav@1646
   183
            return desc;
jaroslav@1646
   184
        }
jaroslav@1646
   185
    }
jaroslav@1646
   186
jaroslav@1646
   187
    void cast(String ds, String dt) {
jaroslav@1646
   188
        String ns = descriptorToName(ds);
jaroslav@1646
   189
        String nt = descriptorToName(dt);
jaroslav@1646
   190
        if (!nt.equals(ns) && !nt.equals(NAME_OBJECT)) {
jaroslav@1646
   191
            visitTypeInsn(Opcodes.CHECKCAST, nt);
jaroslav@1646
   192
        }
jaroslav@1646
   193
    }
jaroslav@1646
   194
jaroslav@1646
   195
    private boolean isPrimitive(Wrapper w) {
jaroslav@1646
   196
        return w != OBJECT;
jaroslav@1646
   197
    }
jaroslav@1646
   198
jaroslav@1646
   199
    private Wrapper toWrapper(String desc) {
jaroslav@1646
   200
        char first = desc.charAt(0);
jaroslav@1646
   201
        if (first == '[' || first == '(') {
jaroslav@1646
   202
            first = 'L';
jaroslav@1646
   203
        }
jaroslav@1646
   204
        return Wrapper.forBasicType(first);
jaroslav@1646
   205
    }
jaroslav@1646
   206
jaroslav@1646
   207
    /**
jaroslav@1646
   208
     * Convert an argument of type 'arg' to be passed to 'target' assuring that it is 'functional'.
jaroslav@1646
   209
     * Insert the needed conversion instructions in the method code.
jaroslav@1646
   210
     * @param arg
jaroslav@1646
   211
     * @param target
jaroslav@1646
   212
     * @param functional
jaroslav@1646
   213
     */
jaroslav@1646
   214
    void convertType(Class<?> arg, Class<?> target, Class<?> functional) {
jaroslav@1646
   215
        if (arg.equals(target) && arg.equals(functional)) {
jaroslav@1646
   216
            return;
jaroslav@1646
   217
        }
jaroslav@1646
   218
        if (arg == Void.TYPE || target == Void.TYPE) {
jaroslav@1646
   219
            return;
jaroslav@1646
   220
        }
jaroslav@1646
   221
        if (arg.isPrimitive()) {
jaroslav@1646
   222
            Wrapper wArg = Wrapper.forPrimitiveType(arg);
jaroslav@1646
   223
            if (target.isPrimitive()) {
jaroslav@1646
   224
                // Both primitives: widening
jaroslav@1646
   225
                widen(wArg, Wrapper.forPrimitiveType(target));
jaroslav@1646
   226
            } else {
jaroslav@1646
   227
                // Primitive argument to reference target
jaroslav@1646
   228
                String dTarget = BytecodeDescriptor.unparse(target);
jaroslav@1646
   229
                Wrapper wPrimTarget = wrapperOrNullFromDescriptor(dTarget);
jaroslav@1646
   230
                if (wPrimTarget != null) {
jaroslav@1646
   231
                    // The target is a boxed primitive type, widen to get there before boxing
jaroslav@1646
   232
                    widen(wArg, wPrimTarget);
jaroslav@1646
   233
                    box(wPrimTarget);
jaroslav@1646
   234
                } else {
jaroslav@1646
   235
                    // Otherwise, box and cast
jaroslav@1646
   236
                    box(wArg);
jaroslav@1646
   237
                    cast(wrapperName(wArg), dTarget);
jaroslav@1646
   238
                }
jaroslav@1646
   239
            }
jaroslav@1646
   240
        } else {
jaroslav@1646
   241
            String dArg = BytecodeDescriptor.unparse(arg);
jaroslav@1646
   242
            String dSrc;
jaroslav@1646
   243
            if (functional.isPrimitive()) {
jaroslav@1646
   244
                dSrc = dArg;
jaroslav@1646
   245
            } else {
jaroslav@1646
   246
                // Cast to convert to possibly more specific type, and generate CCE for invalid arg
jaroslav@1646
   247
                dSrc = BytecodeDescriptor.unparse(functional);
jaroslav@1646
   248
                cast(dArg, dSrc);
jaroslav@1646
   249
            }
jaroslav@1646
   250
            String dTarget = BytecodeDescriptor.unparse(target);
jaroslav@1646
   251
            if (target.isPrimitive()) {
jaroslav@1646
   252
                Wrapper wTarget = toWrapper(dTarget);
jaroslav@1646
   253
                // Reference argument to primitive target
jaroslav@1646
   254
                Wrapper wps = wrapperOrNullFromDescriptor(dSrc);
jaroslav@1646
   255
                if (wps != null) {
jaroslav@1646
   256
                    if (wps.isSigned() || wps.isFloating()) {
jaroslav@1646
   257
                        // Boxed number to primitive
jaroslav@1646
   258
                        unbox(wrapperName(wps), wTarget);
jaroslav@1646
   259
                    } else {
jaroslav@1646
   260
                        // Character or Boolean
jaroslav@1646
   261
                        unbox(wrapperName(wps), wps);
jaroslav@1646
   262
                        widen(wps, wTarget);
jaroslav@1646
   263
                    }
jaroslav@1646
   264
                } else {
jaroslav@1646
   265
                    // Source type is reference type, but not boxed type,
jaroslav@1646
   266
                    // assume it is super type of target type
jaroslav@1646
   267
                    String intermediate;
jaroslav@1646
   268
                    if (wTarget.isSigned() || wTarget.isFloating()) {
jaroslav@1646
   269
                        // Boxed number to primitive
jaroslav@1646
   270
                        intermediate = "java/lang/Number";
jaroslav@1646
   271
                    } else {
jaroslav@1646
   272
                        // Character or Boolean
jaroslav@1646
   273
                        intermediate = wrapperName(wTarget);
jaroslav@1646
   274
                    }
jaroslav@1646
   275
                    cast(dSrc, intermediate);
jaroslav@1646
   276
                    unbox(intermediate, wTarget);
jaroslav@1646
   277
                }
jaroslav@1646
   278
            } else {
jaroslav@1646
   279
                // Both reference types: just case to target type
jaroslav@1646
   280
                cast(dSrc, dTarget);
jaroslav@1646
   281
            }
jaroslav@1646
   282
        }
jaroslav@1646
   283
    }
jaroslav@1646
   284
jaroslav@1646
   285
    /**
jaroslav@1646
   286
     * The following method is copied from
jaroslav@1646
   287
     * org.objectweb.asm.commons.InstructionAdapter. Part of ASM: a very small
jaroslav@1646
   288
     * and fast Java bytecode manipulation framework.
jaroslav@1646
   289
     * Copyright (c) 2000-2005 INRIA, France Telecom All rights reserved.
jaroslav@1646
   290
     */
jaroslav@1646
   291
    void iconst(final int cst) {
jaroslav@1646
   292
        if (cst >= -1 && cst <= 5) {
jaroslav@1646
   293
            mv.visitInsn(Opcodes.ICONST_0 + cst);
jaroslav@1646
   294
        } else if (cst >= Byte.MIN_VALUE && cst <= Byte.MAX_VALUE) {
jaroslav@1646
   295
            mv.visitIntInsn(Opcodes.BIPUSH, cst);
jaroslav@1646
   296
        } else if (cst >= Short.MIN_VALUE && cst <= Short.MAX_VALUE) {
jaroslav@1646
   297
            mv.visitIntInsn(Opcodes.SIPUSH, cst);
jaroslav@1646
   298
        } else {
jaroslav@1646
   299
            mv.visitLdcInsn(cst);
jaroslav@1646
   300
        }
jaroslav@1646
   301
    }
jaroslav@1646
   302
}