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