javap/src/main/java/org/apidesign/javap/StackMapTableData.java
author Lubomir Nerad <lubomir.nerad@oracle.com>
Thu, 29 Nov 2012 20:19:00 +0100
branchregisters
changeset 221 3ee23267706c
parent 167 77f7135b6eb1
child 281 f2352e0b713e
permissions -rw-r--r--
Register based VM
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@221
    43
    abstract int getStackItemsCount();
lubomir@221
    44
jtulach@144
    45
    static class SameFrame extends StackMapTableData {
jtulach@144
    46
        SameFrame(int frameType, int offsetDelta) {
jtulach@144
    47
            super(frameType);
jtulach@144
    48
            this.offsetDelta = offsetDelta;
jtulach@144
    49
        }
lubomir@221
    50
lubomir@221
    51
        @Override
lubomir@221
    52
        int getStackItemsCount() {
lubomir@221
    53
            return 0;
lubomir@221
    54
        }
lubomir@221
    55
lubomir@221
    56
        @Override
lubomir@221
    57
        public String toString() {
lubomir@221
    58
            return "SAME"
lubomir@221
    59
                       + ((frameType == SAME_FRAME_EXTENDED)
lubomir@221
    60
                              ? "_FRAME_EXTENDED" : "")
lubomir@221
    61
                       + "(" + offsetDelta + ")";
lubomir@221
    62
        }
jtulach@144
    63
    }
jtulach@144
    64
jtulach@144
    65
    static class SameLocals1StackItem extends StackMapTableData {
jtulach@144
    66
        final int[] stack;
jtulach@144
    67
        SameLocals1StackItem(int frameType, int offsetDelta, int[] stack) {
jtulach@144
    68
            super(frameType);
jtulach@144
    69
            this.offsetDelta = offsetDelta;
jtulach@144
    70
            this.stack = stack;
jtulach@144
    71
        }
lubomir@221
    72
lubomir@221
    73
        @Override
lubomir@221
    74
        int getStackItemsCount() {
lubomir@221
    75
            return 1;
lubomir@221
    76
        }
lubomir@221
    77
lubomir@221
    78
        @Override
lubomir@221
    79
        public String toString() {
lubomir@221
    80
            return "SAME_LOCALS_1_STACK_ITEM"
lubomir@221
    81
                       + ((frameType == SAME_LOCALS_1_STACK_ITEM_EXTENDED)
lubomir@221
    82
                              ? "_EXTENDED" : "")
lubomir@221
    83
                       + "(" + offsetDelta + ")";
lubomir@221
    84
        }
jtulach@144
    85
    }
jtulach@144
    86
jtulach@144
    87
    static class ChopFrame extends StackMapTableData {
jtulach@144
    88
        ChopFrame(int frameType, int offsetDelta) {
jtulach@144
    89
            super(frameType);
jtulach@144
    90
            this.offsetDelta = offsetDelta;
jtulach@144
    91
        }
lubomir@221
    92
lubomir@221
    93
        @Override
lubomir@221
    94
        int getStackItemsCount() {
lubomir@221
    95
            return 0;
lubomir@221
    96
        }
lubomir@221
    97
lubomir@221
    98
        @Override
lubomir@221
    99
        public String toString() {
lubomir@221
   100
            return "CHOP(" + offsetDelta + ")";
lubomir@221
   101
        }
jtulach@144
   102
    }
jtulach@144
   103
jtulach@144
   104
    static class AppendFrame extends StackMapTableData {
jtulach@144
   105
        final int[] locals;
jtulach@144
   106
        AppendFrame(int frameType, int offsetDelta, int[] locals) {
jtulach@144
   107
            super(frameType);
jtulach@144
   108
            this.offsetDelta = offsetDelta;
jtulach@144
   109
            this.locals = locals;
jtulach@144
   110
        }
lubomir@221
   111
lubomir@221
   112
        @Override
lubomir@221
   113
        int getStackItemsCount() {
lubomir@221
   114
            return 0;
lubomir@221
   115
        }
lubomir@221
   116
lubomir@221
   117
        @Override
lubomir@221
   118
        public String toString() {
lubomir@221
   119
            return "APPEND(" + offsetDelta + ")";
lubomir@221
   120
        }
jtulach@144
   121
    }
jtulach@144
   122
jtulach@144
   123
    static class FullFrame extends StackMapTableData {
jtulach@144
   124
        final int[] locals;
jtulach@144
   125
        final int[] stack;
jtulach@144
   126
        FullFrame(int offsetDelta, int[] locals, int[] stack) {
jtulach@144
   127
            super(FULL_FRAME);
jtulach@144
   128
            this.offsetDelta = offsetDelta;
jtulach@144
   129
            this.locals = locals;
jtulach@144
   130
            this.stack = stack;
jtulach@144
   131
        }
lubomir@221
   132
lubomir@221
   133
        @Override
lubomir@221
   134
        int getStackItemsCount() {
lubomir@221
   135
            return stack.length;
lubomir@221
   136
        }
lubomir@221
   137
lubomir@221
   138
        @Override
lubomir@221
   139
        public String toString() {
lubomir@221
   140
            return "FULL(" + offsetDelta + ")";
lubomir@221
   141
        }
jtulach@144
   142
    }
jtulach@144
   143
jtulach@144
   144
    static StackMapTableData getInstance(DataInputStream in, MethodData method)
jtulach@144
   145
                  throws IOException {
jtulach@144
   146
        int frameType = in.readUnsignedByte();
jtulach@144
   147
jtulach@144
   148
        if (frameType < SAME_FRAME_BOUND) {
jtulach@144
   149
            // same_frame
jtulach@144
   150
            return new SameFrame(frameType, frameType);
jtulach@144
   151
        } else if (SAME_FRAME_BOUND <= frameType && frameType < SAME_LOCALS_1_STACK_ITEM_BOUND) {
jtulach@144
   152
            // same_locals_1_stack_item_frame
jtulach@144
   153
            // read additional single stack element
jtulach@144
   154
            return new SameLocals1StackItem(frameType,
jtulach@144
   155
                                            (frameType - SAME_FRAME_BOUND),
jtulach@144
   156
                                            StackMapData.readTypeArray(in, 1, method));
jtulach@144
   157
        } else if (frameType == SAME_LOCALS_1_STACK_ITEM_EXTENDED) {
jtulach@144
   158
            // same_locals_1_stack_item_extended
jtulach@144
   159
            return new SameLocals1StackItem(frameType,
jtulach@144
   160
                                            in.readUnsignedShort(),
jtulach@144
   161
                                            StackMapData.readTypeArray(in, 1, method));
jtulach@144
   162
        } else if (SAME_LOCALS_1_STACK_ITEM_EXTENDED < frameType  && frameType < SAME_FRAME_EXTENDED) {
jtulach@144
   163
            // chop_frame or same_frame_extended
jtulach@144
   164
            return new ChopFrame(frameType, in.readUnsignedShort());
jtulach@144
   165
        } else if (frameType == SAME_FRAME_EXTENDED) {
jtulach@144
   166
            // chop_frame or same_frame_extended
jtulach@144
   167
            return new SameFrame(frameType, in.readUnsignedShort());
jtulach@144
   168
        } else if (SAME_FRAME_EXTENDED < frameType  && frameType < FULL_FRAME) {
jtulach@144
   169
            // append_frame
jtulach@144
   170
            return new AppendFrame(frameType, in.readUnsignedShort(),
jtulach@144
   171
                                   StackMapData.readTypeArray(in, frameType - SAME_FRAME_EXTENDED, method));
jtulach@144
   172
        } else if (frameType == FULL_FRAME) {
jtulach@144
   173
            // full_frame
jtulach@144
   174
            int offsetDelta = in.readUnsignedShort();
jtulach@144
   175
            int locals_size = in.readUnsignedShort();
jtulach@144
   176
            int[] locals = StackMapData.readTypeArray(in, locals_size, method);
jtulach@144
   177
            int stack_size = in.readUnsignedShort();
jtulach@144
   178
            int[] stack = StackMapData.readTypeArray(in, stack_size, method);
jtulach@144
   179
            return new FullFrame(offsetDelta, locals, stack);
jtulach@144
   180
        } else {
jtulach@144
   181
            throw new ClassFormatError("unrecognized frame_type in StackMapTable");
jtulach@144
   182
        }
jtulach@144
   183
    }
jtulach@144
   184
}