javap/src/main/java/org/apidesign/javap/StackMapTableData.java
author Lubomir Nerad <lubomir.nerad@oracle.com>
Fri, 07 Dec 2012 15:02:35 +0100
branchregisters
changeset 281 f2352e0b713e
parent 221 3ee23267706c
child 307 eaf4e8387065
permissions -rw-r--r--
Type specific stack variables
jtulach@144
     1
/*
jtulach@144
     2
 * Copyright (c) 2005, Oracle and/or its affiliates. All rights reserved.
jtulach@144
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
jtulach@144
     4
 *
jtulach@144
     5
 * This code is free software; you can redistribute it and/or modify it
jtulach@144
     6
 * under the terms of the GNU General Public License version 2 only, as
jtulach@144
     7
 * published by the Free Software Foundation.  Oracle designates this
jtulach@144
     8
 * particular file as subject to the "Classpath" exception as provided
jtulach@144
     9
 * by Oracle in the LICENSE file that accompanied this code.
jtulach@144
    10
 *
jtulach@144
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
jtulach@144
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
jtulach@144
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
jtulach@144
    14
 * version 2 for more details (a copy is included in the LICENSE file that
jtulach@144
    15
 * accompanied this code).
jtulach@144
    16
 *
jtulach@144
    17
 * You should have received a copy of the GNU General Public License version
jtulach@144
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
jtulach@144
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
jtulach@144
    20
 *
jtulach@144
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
jtulach@144
    22
 * or visit www.oracle.com if you need additional information or have any
jtulach@144
    23
 * questions.
jtulach@144
    24
 */
jtulach@144
    25
jtulach@144
    26
jtulach@167
    27
package org.apidesign.javap;
jtulach@144
    28
jtulach@144
    29
import java.io.*;
jtulach@144
    30
jtulach@167
    31
import static org.apidesign.javap.RuntimeConstants.*;
jtulach@144
    32
jtulach@144
    33
/* represents one entry of StackMapTable attribute
jtulach@144
    34
 */
lubomir@221
    35
abstract class StackMapTableData {
jtulach@144
    36
    final int frameType;
jtulach@144
    37
    int offsetDelta;
jtulach@144
    38
jtulach@144
    39
    StackMapTableData(int frameType) {
jtulach@144
    40
        this.frameType = frameType;
jtulach@144
    41
    }
jtulach@144
    42
lubomir@281
    43
    abstract void applyTo(TypeArray localTypes, TypeArray stackTypes);
lubomir@281
    44
lubomir@281
    45
    protected static String toString(
lubomir@281
    46
            final String frameType,
lubomir@281
    47
            final int offset,
lubomir@281
    48
            final int[] localTypes,
lubomir@281
    49
            final int[] stackTypes) {
lubomir@281
    50
        final StringBuilder sb = new StringBuilder(frameType);
lubomir@281
    51
lubomir@281
    52
        sb.append("(off: +").append(offset);
lubomir@281
    53
        if (localTypes != null) {
lubomir@281
    54
            sb.append(", locals: ");
lubomir@281
    55
            appendTypes(sb, localTypes);
lubomir@281
    56
        }
lubomir@281
    57
        if (stackTypes != null) {
lubomir@281
    58
            sb.append(", stack: ");
lubomir@281
    59
            appendTypes(sb, stackTypes);
lubomir@281
    60
        }
lubomir@281
    61
        sb.append(')');
lubomir@281
    62
lubomir@281
    63
        return sb.toString();
lubomir@281
    64
    }
lubomir@281
    65
lubomir@281
    66
    private static void appendTypes(final StringBuilder sb, final int[] types) {
lubomir@281
    67
        sb.append('[');
lubomir@281
    68
        if (types.length > 0) {
lubomir@281
    69
            appendType(sb, types[0]);
lubomir@281
    70
            for (int i = 1; i < types.length; ++i) {
lubomir@281
    71
                sb.append(", ");
lubomir@281
    72
                appendType(sb, types[i]);
lubomir@281
    73
            }
lubomir@281
    74
        }
lubomir@281
    75
        sb.append(']');
lubomir@281
    76
    }
lubomir@281
    77
lubomir@281
    78
    private static void appendType(final StringBuilder sb, final int type) {
lubomir@281
    79
        switch (type & 0xff) {
lubomir@281
    80
            case ITEM_Bogus:
lubomir@281
    81
                sb.append("_top_");
lubomir@281
    82
                break;
lubomir@281
    83
            case ITEM_Integer:
lubomir@281
    84
                sb.append("_int_");
lubomir@281
    85
                break;
lubomir@281
    86
            case ITEM_Float:
lubomir@281
    87
                sb.append("_float_");
lubomir@281
    88
                break;
lubomir@281
    89
            case ITEM_Double:
lubomir@281
    90
                sb.append("_double_");
lubomir@281
    91
                break;
lubomir@281
    92
            case ITEM_Long:
lubomir@281
    93
                sb.append("_long_");
lubomir@281
    94
                break;
lubomir@281
    95
            case ITEM_Null:
lubomir@281
    96
                sb.append("_null_");
lubomir@281
    97
                break;
lubomir@281
    98
            case ITEM_InitObject: // UninitializedThis
lubomir@281
    99
                sb.append("_init_");
lubomir@281
   100
                break;
lubomir@281
   101
            case ITEM_Object:
lubomir@281
   102
                sb.append("_object_");
lubomir@281
   103
                break;
lubomir@281
   104
            case ITEM_NewObject: // Uninitialized
lubomir@281
   105
                sb.append("_new_");
lubomir@281
   106
                break;
lubomir@281
   107
            default:
lubomir@281
   108
                sb.append("_unknown_");
lubomir@281
   109
                break;
lubomir@281
   110
        }
lubomir@281
   111
    }
lubomir@221
   112
jtulach@144
   113
    static class SameFrame extends StackMapTableData {
jtulach@144
   114
        SameFrame(int frameType, int offsetDelta) {
jtulach@144
   115
            super(frameType);
jtulach@144
   116
            this.offsetDelta = offsetDelta;
jtulach@144
   117
        }
lubomir@221
   118
lubomir@221
   119
        @Override
lubomir@281
   120
        void applyTo(TypeArray localTypes, TypeArray stackTypes) {
lubomir@281
   121
            stackTypes.clear();
lubomir@221
   122
        }
lubomir@221
   123
lubomir@221
   124
        @Override
lubomir@221
   125
        public String toString() {
lubomir@281
   126
            return toString("SAME" + ((frameType == SAME_FRAME_EXTENDED)
lubomir@281
   127
                                          ? "_FRAME_EXTENDED" : ""),
lubomir@281
   128
                            offsetDelta,
lubomir@281
   129
                            null, null);
lubomir@221
   130
        }
jtulach@144
   131
    }
jtulach@144
   132
jtulach@144
   133
    static class SameLocals1StackItem extends StackMapTableData {
jtulach@144
   134
        final int[] stack;
jtulach@144
   135
        SameLocals1StackItem(int frameType, int offsetDelta, int[] stack) {
jtulach@144
   136
            super(frameType);
jtulach@144
   137
            this.offsetDelta = offsetDelta;
jtulach@144
   138
            this.stack = stack;
jtulach@144
   139
        }
lubomir@221
   140
lubomir@221
   141
        @Override
lubomir@281
   142
        void applyTo(TypeArray localTypes, TypeArray stackTypes) {
lubomir@281
   143
            stackTypes.setAll(stack);
lubomir@221
   144
        }
lubomir@221
   145
lubomir@221
   146
        @Override
lubomir@221
   147
        public String toString() {
lubomir@281
   148
            return toString(
lubomir@281
   149
                       "SAME_LOCALS_1_STACK_ITEM"
lubomir@281
   150
                           + ((frameType == SAME_LOCALS_1_STACK_ITEM_EXTENDED)
lubomir@281
   151
                                  ? "_EXTENDED" : ""),
lubomir@281
   152
                       offsetDelta,
lubomir@281
   153
                       null, stack);
lubomir@221
   154
        }
jtulach@144
   155
    }
jtulach@144
   156
jtulach@144
   157
    static class ChopFrame extends StackMapTableData {
jtulach@144
   158
        ChopFrame(int frameType, int offsetDelta) {
jtulach@144
   159
            super(frameType);
jtulach@144
   160
            this.offsetDelta = offsetDelta;
jtulach@144
   161
        }
lubomir@221
   162
lubomir@221
   163
        @Override
lubomir@281
   164
        void applyTo(TypeArray localTypes, TypeArray stackTypes) {
lubomir@281
   165
            localTypes.setSize(localTypes.getSize()
lubomir@281
   166
                                   - (SAME_FRAME_EXTENDED - frameType));
lubomir@281
   167
            stackTypes.clear();
lubomir@221
   168
        }
lubomir@221
   169
lubomir@221
   170
        @Override
lubomir@221
   171
        public String toString() {
lubomir@281
   172
            return toString("CHOP", offsetDelta, null, null);
lubomir@221
   173
        }
jtulach@144
   174
    }
jtulach@144
   175
jtulach@144
   176
    static class AppendFrame extends StackMapTableData {
jtulach@144
   177
        final int[] locals;
jtulach@144
   178
        AppendFrame(int frameType, int offsetDelta, int[] locals) {
jtulach@144
   179
            super(frameType);
jtulach@144
   180
            this.offsetDelta = offsetDelta;
jtulach@144
   181
            this.locals = locals;
jtulach@144
   182
        }
lubomir@221
   183
lubomir@221
   184
        @Override
lubomir@281
   185
        void applyTo(TypeArray localTypes, TypeArray stackTypes) {
lubomir@281
   186
            localTypes.addAll(locals);
lubomir@281
   187
            stackTypes.clear();
lubomir@221
   188
        }
lubomir@221
   189
lubomir@221
   190
        @Override
lubomir@221
   191
        public String toString() {
lubomir@281
   192
            return toString("APPEND", offsetDelta, locals, null);
lubomir@221
   193
        }
jtulach@144
   194
    }
jtulach@144
   195
jtulach@144
   196
    static class FullFrame extends StackMapTableData {
jtulach@144
   197
        final int[] locals;
jtulach@144
   198
        final int[] stack;
jtulach@144
   199
        FullFrame(int offsetDelta, int[] locals, int[] stack) {
jtulach@144
   200
            super(FULL_FRAME);
jtulach@144
   201
            this.offsetDelta = offsetDelta;
jtulach@144
   202
            this.locals = locals;
jtulach@144
   203
            this.stack = stack;
jtulach@144
   204
        }
lubomir@221
   205
lubomir@221
   206
        @Override
lubomir@281
   207
        void applyTo(TypeArray localTypes, TypeArray stackTypes) {
lubomir@281
   208
            localTypes.setAll(locals);
lubomir@281
   209
            stackTypes.setAll(stack);
lubomir@221
   210
        }
lubomir@221
   211
lubomir@221
   212
        @Override
lubomir@221
   213
        public String toString() {
lubomir@281
   214
            return toString("FULL", offsetDelta, locals, stack);
lubomir@221
   215
        }
jtulach@144
   216
    }
jtulach@144
   217
jtulach@144
   218
    static StackMapTableData getInstance(DataInputStream in, MethodData method)
jtulach@144
   219
                  throws IOException {
jtulach@144
   220
        int frameType = in.readUnsignedByte();
jtulach@144
   221
jtulach@144
   222
        if (frameType < SAME_FRAME_BOUND) {
jtulach@144
   223
            // same_frame
jtulach@144
   224
            return new SameFrame(frameType, frameType);
jtulach@144
   225
        } else if (SAME_FRAME_BOUND <= frameType && frameType < SAME_LOCALS_1_STACK_ITEM_BOUND) {
jtulach@144
   226
            // same_locals_1_stack_item_frame
jtulach@144
   227
            // read additional single stack element
jtulach@144
   228
            return new SameLocals1StackItem(frameType,
jtulach@144
   229
                                            (frameType - SAME_FRAME_BOUND),
jtulach@144
   230
                                            StackMapData.readTypeArray(in, 1, method));
jtulach@144
   231
        } else if (frameType == SAME_LOCALS_1_STACK_ITEM_EXTENDED) {
jtulach@144
   232
            // same_locals_1_stack_item_extended
jtulach@144
   233
            return new SameLocals1StackItem(frameType,
jtulach@144
   234
                                            in.readUnsignedShort(),
jtulach@144
   235
                                            StackMapData.readTypeArray(in, 1, method));
jtulach@144
   236
        } else if (SAME_LOCALS_1_STACK_ITEM_EXTENDED < frameType  && frameType < SAME_FRAME_EXTENDED) {
jtulach@144
   237
            // chop_frame or same_frame_extended
jtulach@144
   238
            return new ChopFrame(frameType, in.readUnsignedShort());
jtulach@144
   239
        } else if (frameType == SAME_FRAME_EXTENDED) {
jtulach@144
   240
            // chop_frame or same_frame_extended
jtulach@144
   241
            return new SameFrame(frameType, in.readUnsignedShort());
jtulach@144
   242
        } else if (SAME_FRAME_EXTENDED < frameType  && frameType < FULL_FRAME) {
jtulach@144
   243
            // append_frame
jtulach@144
   244
            return new AppendFrame(frameType, in.readUnsignedShort(),
jtulach@144
   245
                                   StackMapData.readTypeArray(in, frameType - SAME_FRAME_EXTENDED, method));
jtulach@144
   246
        } else if (frameType == FULL_FRAME) {
jtulach@144
   247
            // full_frame
jtulach@144
   248
            int offsetDelta = in.readUnsignedShort();
jtulach@144
   249
            int locals_size = in.readUnsignedShort();
jtulach@144
   250
            int[] locals = StackMapData.readTypeArray(in, locals_size, method);
jtulach@144
   251
            int stack_size = in.readUnsignedShort();
jtulach@144
   252
            int[] stack = StackMapData.readTypeArray(in, stack_size, method);
jtulach@144
   253
            return new FullFrame(offsetDelta, locals, stack);
jtulach@144
   254
        } else {
jtulach@144
   255
            throw new ClassFormatError("unrecognized frame_type in StackMapTable");
jtulach@144
   256
        }
jtulach@144
   257
    }
jtulach@144
   258
}