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 |
}
|