2 * Copyright (c) 2005, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation. Oracle designates this
8 * particular file as subject to the "Classpath" exception as provided
9 * by Oracle in the LICENSE file that accompanied this code.
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22 * or visit www.oracle.com if you need additional information or have any
27 package org.apidesign.javap;
31 import static org.apidesign.javap.RuntimeConstants.*;
33 /* represents one entry of StackMapTable attribute
35 abstract class StackMapTableData {
39 StackMapTableData(int frameType) {
40 this.frameType = frameType;
43 abstract int getStackItemsCount();
45 static class SameFrame extends StackMapTableData {
46 SameFrame(int frameType, int offsetDelta) {
48 this.offsetDelta = offsetDelta;
52 int getStackItemsCount() {
57 public String toString() {
59 + ((frameType == SAME_FRAME_EXTENDED)
60 ? "_FRAME_EXTENDED" : "")
61 + "(" + offsetDelta + ")";
65 static class SameLocals1StackItem extends StackMapTableData {
67 SameLocals1StackItem(int frameType, int offsetDelta, int[] stack) {
69 this.offsetDelta = offsetDelta;
74 int getStackItemsCount() {
79 public String toString() {
80 return "SAME_LOCALS_1_STACK_ITEM"
81 + ((frameType == SAME_LOCALS_1_STACK_ITEM_EXTENDED)
83 + "(" + offsetDelta + ")";
87 static class ChopFrame extends StackMapTableData {
88 ChopFrame(int frameType, int offsetDelta) {
90 this.offsetDelta = offsetDelta;
94 int getStackItemsCount() {
99 public String toString() {
100 return "CHOP(" + offsetDelta + ")";
104 static class AppendFrame extends StackMapTableData {
106 AppendFrame(int frameType, int offsetDelta, int[] locals) {
108 this.offsetDelta = offsetDelta;
109 this.locals = locals;
113 int getStackItemsCount() {
118 public String toString() {
119 return "APPEND(" + offsetDelta + ")";
123 static class FullFrame extends StackMapTableData {
126 FullFrame(int offsetDelta, int[] locals, int[] stack) {
128 this.offsetDelta = offsetDelta;
129 this.locals = locals;
134 int getStackItemsCount() {
139 public String toString() {
140 return "FULL(" + offsetDelta + ")";
144 static StackMapTableData getInstance(DataInputStream in, MethodData method)
146 int frameType = in.readUnsignedByte();
148 if (frameType < SAME_FRAME_BOUND) {
150 return new SameFrame(frameType, frameType);
151 } else if (SAME_FRAME_BOUND <= frameType && frameType < SAME_LOCALS_1_STACK_ITEM_BOUND) {
152 // same_locals_1_stack_item_frame
153 // read additional single stack element
154 return new SameLocals1StackItem(frameType,
155 (frameType - SAME_FRAME_BOUND),
156 StackMapData.readTypeArray(in, 1, method));
157 } else if (frameType == SAME_LOCALS_1_STACK_ITEM_EXTENDED) {
158 // same_locals_1_stack_item_extended
159 return new SameLocals1StackItem(frameType,
160 in.readUnsignedShort(),
161 StackMapData.readTypeArray(in, 1, method));
162 } else if (SAME_LOCALS_1_STACK_ITEM_EXTENDED < frameType && frameType < SAME_FRAME_EXTENDED) {
163 // chop_frame or same_frame_extended
164 return new ChopFrame(frameType, in.readUnsignedShort());
165 } else if (frameType == SAME_FRAME_EXTENDED) {
166 // chop_frame or same_frame_extended
167 return new SameFrame(frameType, in.readUnsignedShort());
168 } else if (SAME_FRAME_EXTENDED < frameType && frameType < FULL_FRAME) {
170 return new AppendFrame(frameType, in.readUnsignedShort(),
171 StackMapData.readTypeArray(in, frameType - SAME_FRAME_EXTENDED, method));
172 } else if (frameType == FULL_FRAME) {
174 int offsetDelta = in.readUnsignedShort();
175 int locals_size = in.readUnsignedShort();
176 int[] locals = StackMapData.readTypeArray(in, locals_size, method);
177 int stack_size = in.readUnsignedShort();
178 int[] stack = StackMapData.readTypeArray(in, stack_size, method);
179 return new FullFrame(offsetDelta, locals, stack);
181 throw new ClassFormatError("unrecognized frame_type in StackMapTable");