Mercurial > hg > truffle
annotate src/share/vm/prims/jvmtiClassFileReconstituter.cpp @ 8733:9def4075da6d
8008079: G1: Add nextObject routine to CMBitMapRO and replace nextWord
Summary: Update the task local finger to the start of the next object when marking aborts, in order to avoid the redundant scanning of all 0's when the marking task restarts, if otherwise updating to the next word. In addition, reuse the routine nextObject() in routine iterate().
Reviewed-by: johnc, ysr
Contributed-by: tamao <tao.mao@oracle.com>
author | tamao |
---|---|
date | Tue, 05 Mar 2013 15:36:56 -0800 |
parents | 8aaef2cee3b2 |
children | c456f4510385 |
rev | line source |
---|---|
0 | 1 /* |
5922
ff29ce866f23
7118280: The gbyc00102 JCK7 test causes an assert in JVM 7.0 fastdebug mode
dsamersoff
parents:
4718
diff
changeset
|
2 * Copyright (c) 2005, 2012, Oracle and/or its affiliates. All rights reserved. |
0 | 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
4 * | |
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. | |
8 * | |
9 * This code is distributed in the hope that it will be useful, but WITHOUT | |
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | |
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License | |
12 * version 2 for more details (a copy is included in the LICENSE file that | |
13 * accompanied this code). | |
14 * | |
15 * You should have received a copy of the GNU General Public License version | |
16 * 2 along with this work; if not, write to the Free Software Foundation, | |
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. | |
18 * | |
1552
c18cbe5936b8
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
844
diff
changeset
|
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
c18cbe5936b8
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
844
diff
changeset
|
20 * or visit www.oracle.com if you need additional information or have any |
c18cbe5936b8
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
844
diff
changeset
|
21 * questions. |
0 | 22 * |
23 */ | |
24 | |
1972 | 25 #include "precompiled.hpp" |
26 #include "classfile/symbolTable.hpp" | |
27 #include "interpreter/bytecodeStream.hpp" | |
3938 | 28 #include "oops/fieldStreams.hpp" |
1972 | 29 #include "prims/jvmtiClassFileReconstituter.hpp" |
30 #include "runtime/signature.hpp" | |
31 #ifdef TARGET_ARCH_x86 | |
32 # include "bytes_x86.hpp" | |
33 #endif | |
34 #ifdef TARGET_ARCH_sparc | |
35 # include "bytes_sparc.hpp" | |
36 #endif | |
37 #ifdef TARGET_ARCH_zero | |
38 # include "bytes_zero.hpp" | |
39 #endif | |
2192
b92c45f2bc75
7016023: Enable building ARM and PPC from src/closed repository
bobv
parents:
2177
diff
changeset
|
40 #ifdef TARGET_ARCH_arm |
b92c45f2bc75
7016023: Enable building ARM and PPC from src/closed repository
bobv
parents:
2177
diff
changeset
|
41 # include "bytes_arm.hpp" |
b92c45f2bc75
7016023: Enable building ARM and PPC from src/closed repository
bobv
parents:
2177
diff
changeset
|
42 #endif |
b92c45f2bc75
7016023: Enable building ARM and PPC from src/closed repository
bobv
parents:
2177
diff
changeset
|
43 #ifdef TARGET_ARCH_ppc |
b92c45f2bc75
7016023: Enable building ARM and PPC from src/closed repository
bobv
parents:
2177
diff
changeset
|
44 # include "bytes_ppc.hpp" |
b92c45f2bc75
7016023: Enable building ARM and PPC from src/closed repository
bobv
parents:
2177
diff
changeset
|
45 #endif |
6772
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
46 // FIXME: add Deprecated attribute |
0 | 47 // FIXME: fix Synthetic attribute |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
48 // FIXME: per Serguei, add error return handling for ConstantPool::copy_cpool_bytes() |
0 | 49 |
50 | |
51 // Write the field information portion of ClassFile structure | |
52 // JVMSpec| u2 fields_count; | |
53 // JVMSpec| field_info fields[fields_count]; | |
54 void JvmtiClassFileReconstituter::write_field_infos() { | |
55 HandleMark hm(thread()); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
56 Array<AnnotationArray*>* fields_anno = ikh()->fields_annotations(); |
0 | 57 |
3938 | 58 // Compute the real number of Java fields |
59 int java_fields = ikh()->java_fields_count(); | |
60 | |
3944 | 61 write_u2(java_fields); |
3938 | 62 for (JavaFieldStream fs(ikh()); !fs.done(); fs.next()) { |
63 AccessFlags access_flags = fs.access_flags(); | |
64 int name_index = fs.name_index(); | |
65 int signature_index = fs.signature_index(); | |
66 int initial_value_index = fs.initval_index(); | |
0 | 67 guarantee(name_index != 0 && signature_index != 0, "bad constant pool index for field"); |
3938 | 68 // int offset = ikh()->field_offset( index ); |
69 int generic_signature_index = fs.generic_signature_index(); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
70 AnnotationArray* anno = fields_anno == NULL ? NULL : fields_anno->at(fs.index()); |
0 | 71 |
72 // JVMSpec| field_info { | |
73 // JVMSpec| u2 access_flags; | |
74 // JVMSpec| u2 name_index; | |
75 // JVMSpec| u2 descriptor_index; | |
76 // JVMSpec| u2 attributes_count; | |
77 // JVMSpec| attribute_info attributes[attributes_count]; | |
78 // JVMSpec| } | |
79 | |
3938 | 80 write_u2(access_flags.as_int() & JVM_RECOGNIZED_FIELD_MODIFIERS); |
0 | 81 write_u2(name_index); |
82 write_u2(signature_index); | |
83 int attr_count = 0; | |
84 if (initial_value_index != 0) { | |
85 ++attr_count; | |
86 } | |
87 if (access_flags.is_synthetic()) { | |
88 // ++attr_count; | |
89 } | |
90 if (generic_signature_index != 0) { | |
91 ++attr_count; | |
92 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
93 if (anno != NULL) { |
0 | 94 ++attr_count; // has RuntimeVisibleAnnotations attribute |
95 } | |
96 | |
97 write_u2(attr_count); | |
98 | |
99 if (initial_value_index != 0) { | |
100 write_attribute_name_index("ConstantValue"); | |
101 write_u4(2); //length always 2 | |
102 write_u2(initial_value_index); | |
103 } | |
104 if (access_flags.is_synthetic()) { | |
105 // write_synthetic_attribute(); | |
106 } | |
107 if (generic_signature_index != 0) { | |
108 write_signature_attribute(generic_signature_index); | |
109 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
110 if (anno != NULL) { |
0 | 111 write_annotations_attribute("RuntimeVisibleAnnotations", anno); |
112 } | |
113 } | |
114 } | |
115 | |
116 // Write Code attribute | |
117 // JVMSpec| Code_attribute { | |
118 // JVMSpec| u2 attribute_name_index; | |
119 // JVMSpec| u4 attribute_length; | |
120 // JVMSpec| u2 max_stack; | |
121 // JVMSpec| u2 max_locals; | |
122 // JVMSpec| u4 code_length; | |
123 // JVMSpec| u1 code[code_length]; | |
124 // JVMSpec| u2 exception_table_length; | |
125 // JVMSpec| { u2 start_pc; | |
126 // JVMSpec| u2 end_pc; | |
127 // JVMSpec| u2 handler_pc; | |
128 // JVMSpec| u2 catch_type; | |
129 // JVMSpec| } exception_table[exception_table_length]; | |
130 // JVMSpec| u2 attributes_count; | |
131 // JVMSpec| attribute_info attributes[attributes_count]; | |
132 // JVMSpec| } | |
133 void JvmtiClassFileReconstituter::write_code_attribute(methodHandle method) { | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
134 ConstMethod* const_method = method->constMethod(); |
0 | 135 u2 line_num_cnt = 0; |
136 int stackmap_len = 0; | |
4718
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
137 int local_variable_table_length = 0; |
6772
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
138 int local_variable_type_table_length = 0; |
0 | 139 |
4718
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
140 // compute number and length of attributes |
0 | 141 int attr_count = 0; |
142 int attr_size = 0; | |
143 if (const_method->has_linenumber_table()) { | |
144 line_num_cnt = line_number_table_entries(method); | |
145 if (line_num_cnt != 0) { | |
146 ++attr_count; | |
147 // Compute the complete size of the line number table attribute: | |
148 // LineNumberTable_attribute { | |
149 // u2 attribute_name_index; | |
150 // u4 attribute_length; | |
151 // u2 line_number_table_length; | |
152 // { u2 start_pc; | |
153 // u2 line_number; | |
154 // } line_number_table[line_number_table_length]; | |
155 // } | |
156 attr_size += 2 + 4 + 2 + line_num_cnt * (2 + 2); | |
157 } | |
158 } | |
159 if (method->has_stackmap_table()) { | |
160 stackmap_len = method->stackmap_data()->length(); | |
161 if (stackmap_len != 0) { | |
162 ++attr_count; | |
163 // Compute the size of the stack map table attribute (VM stores raw): | |
164 // StackMapTable_attribute { | |
165 // u2 attribute_name_index; | |
166 // u4 attribute_length; | |
167 // u2 number_of_entries; | |
168 // stack_map_frame_entries[number_of_entries]; | |
169 // } | |
170 attr_size += 2 + 4 + stackmap_len; | |
171 } | |
172 } | |
4718
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
173 if (method->has_localvariable_table()) { |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
174 local_variable_table_length = method->localvariable_table_length(); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
175 if (local_variable_table_length != 0) { |
6772
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
176 ++attr_count; |
4718
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
177 // Compute the size of the local variable table attribute (VM stores raw): |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
178 // LocalVariableTable_attribute { |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
179 // u2 attribute_name_index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
180 // u4 attribute_length; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
181 // u2 local_variable_table_length; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
182 // { |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
183 // u2 start_pc; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
184 // u2 length; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
185 // u2 name_index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
186 // u2 descriptor_index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
187 // u2 index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
188 // } |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
189 attr_size += 2 + 4 + 2 + local_variable_table_length * (2 + 2 + 2 + 2 + 2); |
6772
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
190 |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
191 // Local variables with generic signatures must have LVTT entries |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
192 LocalVariableTableElement *elem = method->localvariable_table_start(); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
193 for (int idx = 0; idx < local_variable_table_length; idx++) { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
194 if (elem[idx].signature_cp_index != 0) { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
195 local_variable_type_table_length++; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
196 } |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
197 } |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
198 |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
199 if (local_variable_type_table_length != 0) { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
200 ++attr_count; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
201 // Compute the size of the local variable type table attribute (VM stores raw): |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
202 // LocalVariableTypeTable_attribute { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
203 // u2 attribute_name_index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
204 // u4 attribute_length; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
205 // u2 local_variable_type_table_length; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
206 // { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
207 // u2 start_pc; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
208 // u2 length; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
209 // u2 name_index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
210 // u2 signature_index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
211 // u2 index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
212 // } |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
213 attr_size += 2 + 4 + 2 + local_variable_type_table_length * (2 + 2 + 2 + 2 + 2); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
214 } |
4718
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
215 } |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
216 } |
0 | 217 |
6213
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
218 ExceptionTable exception_table(method()); |
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
219 int exception_table_length = exception_table.length(); |
0 | 220 int code_size = const_method->code_size(); |
221 int size = | |
222 2+2+4 + // max_stack, max_locals, code_length | |
223 code_size + // code | |
224 2 + // exception_table_length | |
6213
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
225 (2+2+2+2) * exception_table_length + // exception_table |
0 | 226 2 + // attributes_count |
227 attr_size; // attributes | |
228 | |
229 write_attribute_name_index("Code"); | |
230 write_u4(size); | |
6975
64812523d72e
7194607: VerifyLocalVariableTableOnRetransformTest.sh fails after JSR-292 merge
sspitsyn
parents:
6772
diff
changeset
|
231 write_u2(method->verifier_max_stack()); |
0 | 232 write_u2(method->max_locals()); |
233 write_u4(code_size); | |
234 copy_bytecodes(method, (unsigned char*)writeable_address(code_size)); | |
6213
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
235 write_u2(exception_table_length); |
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
236 for (int index = 0; index < exception_table_length; index++) { |
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
237 write_u2(exception_table.start_pc(index)); |
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
238 write_u2(exception_table.end_pc(index)); |
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
239 write_u2(exception_table.handler_pc(index)); |
8150fa46d2ed
7178145: Change constMethodOop::_exception_table to optionally inlined u2 table.
jiangli
parents:
5967
diff
changeset
|
240 write_u2(exception_table.catch_type_index(index)); |
0 | 241 } |
242 write_u2(attr_count); | |
243 if (line_num_cnt != 0) { | |
244 write_line_number_table_attribute(method, line_num_cnt); | |
245 } | |
246 if (stackmap_len != 0) { | |
247 write_stackmap_table_attribute(method, stackmap_len); | |
248 } | |
4718
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
249 if (local_variable_table_length != 0) { |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
250 write_local_variable_table_attribute(method, local_variable_table_length); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
251 } |
6772
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
252 if (local_variable_type_table_length != 0) { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
253 write_local_variable_type_table_attribute(method, local_variable_type_table_length); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
254 } |
0 | 255 } |
256 | |
257 // Write Exceptions attribute | |
258 // JVMSpec| Exceptions_attribute { | |
259 // JVMSpec| u2 attribute_name_index; | |
260 // JVMSpec| u4 attribute_length; | |
261 // JVMSpec| u2 number_of_exceptions; | |
262 // JVMSpec| u2 exception_index_table[number_of_exceptions]; | |
263 // JVMSpec| } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
264 void JvmtiClassFileReconstituter::write_exceptions_attribute(ConstMethod* const_method) { |
0 | 265 CheckedExceptionElement* checked_exceptions = const_method->checked_exceptions_start(); |
266 int checked_exceptions_length = const_method->checked_exceptions_length(); | |
267 int size = | |
268 2 + // number_of_exceptions | |
269 2 * checked_exceptions_length; // exception_index_table | |
270 | |
271 write_attribute_name_index("Exceptions"); | |
272 write_u4(size); | |
273 write_u2(checked_exceptions_length); | |
274 for (int index = 0; index < checked_exceptions_length; index++) { | |
275 write_u2(checked_exceptions[index].class_cp_index); | |
276 } | |
277 } | |
278 | |
279 // Write SourceFile attribute | |
280 // JVMSpec| SourceFile_attribute { | |
281 // JVMSpec| u2 attribute_name_index; | |
282 // JVMSpec| u4 attribute_length; | |
283 // JVMSpec| u2 sourcefile_index; | |
284 // JVMSpec| } | |
285 void JvmtiClassFileReconstituter::write_source_file_attribute() { | |
286 assert(ikh()->source_file_name() != NULL, "caller must check"); | |
287 | |
288 write_attribute_name_index("SourceFile"); | |
289 write_u4(2); // always length 2 | |
290 write_u2(symbol_to_cpool_index(ikh()->source_file_name())); | |
291 } | |
292 | |
293 // Write SourceDebugExtension attribute | |
294 // JSR45| SourceDebugExtension_attribute { | |
295 // JSR45| u2 attribute_name_index; | |
296 // JSR45| u4 attribute_length; | |
6203
04ade88d9712
6294277: java -Xdebug crashes on SourceDebugExtension attribute larger than 64K
fparain
parents:
5967
diff
changeset
|
297 // JSR45| u1 debug_extension[attribute_length]; |
0 | 298 // JSR45| } |
299 void JvmtiClassFileReconstituter::write_source_debug_extension_attribute() { | |
300 assert(ikh()->source_debug_extension() != NULL, "caller must check"); | |
301 | |
302 write_attribute_name_index("SourceDebugExtension"); | |
6203
04ade88d9712
6294277: java -Xdebug crashes on SourceDebugExtension attribute larger than 64K
fparain
parents:
5967
diff
changeset
|
303 int len = (int)strlen(ikh()->source_debug_extension()); |
04ade88d9712
6294277: java -Xdebug crashes on SourceDebugExtension attribute larger than 64K
fparain
parents:
5967
diff
changeset
|
304 write_u4(len); |
04ade88d9712
6294277: java -Xdebug crashes on SourceDebugExtension attribute larger than 64K
fparain
parents:
5967
diff
changeset
|
305 u1* ext = (u1*)ikh()->source_debug_extension(); |
04ade88d9712
6294277: java -Xdebug crashes on SourceDebugExtension attribute larger than 64K
fparain
parents:
5967
diff
changeset
|
306 for (int i=0; i<len; i++) { |
04ade88d9712
6294277: java -Xdebug crashes on SourceDebugExtension attribute larger than 64K
fparain
parents:
5967
diff
changeset
|
307 write_u1(ext[i]); |
04ade88d9712
6294277: java -Xdebug crashes on SourceDebugExtension attribute larger than 64K
fparain
parents:
5967
diff
changeset
|
308 } |
0 | 309 } |
310 | |
311 // Write (generic) Signature attribute | |
312 // JVMSpec| Signature_attribute { | |
313 // JVMSpec| u2 attribute_name_index; | |
314 // JVMSpec| u4 attribute_length; | |
315 // JVMSpec| u2 signature_index; | |
316 // JVMSpec| } | |
317 void JvmtiClassFileReconstituter::write_signature_attribute(u2 generic_signature_index) { | |
318 write_attribute_name_index("Signature"); | |
319 write_u4(2); // always length 2 | |
320 write_u2(generic_signature_index); | |
321 } | |
322 | |
323 // Compute the number of entries in the InnerClasses attribute | |
324 u2 JvmtiClassFileReconstituter::inner_classes_attribute_length() { | |
5967
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
325 InnerClassesIterator iter(ikh()); |
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
326 return iter.length(); |
0 | 327 } |
328 | |
329 // Write an annotation attribute. The VM stores them in raw form, so all we need | |
330 // to do is add the attrubute name and fill in the length. | |
331 // JSR202| *Annotations_attribute { | |
332 // JSR202| u2 attribute_name_index; | |
333 // JSR202| u4 attribute_length; | |
334 // JSR202| ... | |
335 // JSR202| } | |
336 void JvmtiClassFileReconstituter::write_annotations_attribute(const char* attr_name, | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
337 AnnotationArray* annos) { |
0 | 338 u4 length = annos->length(); |
339 write_attribute_name_index(attr_name); | |
340 write_u4(length); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
341 memcpy(writeable_address(length), annos->adr_at(0), length); |
0 | 342 } |
343 | |
344 | |
345 // Write InnerClasses attribute | |
346 // JVMSpec| InnerClasses_attribute { | |
347 // JVMSpec| u2 attribute_name_index; | |
348 // JVMSpec| u4 attribute_length; | |
349 // JVMSpec| u2 number_of_classes; | |
350 // JVMSpec| { u2 inner_class_info_index; | |
351 // JVMSpec| u2 outer_class_info_index; | |
352 // JVMSpec| u2 inner_name_index; | |
353 // JVMSpec| u2 inner_class_access_flags; | |
354 // JVMSpec| } classes[number_of_classes]; | |
355 // JVMSpec| } | |
356 void JvmtiClassFileReconstituter::write_inner_classes_attribute(int length) { | |
5967
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
357 InnerClassesIterator iter(ikh()); |
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
358 guarantee(iter.length() != 0 && iter.length() == length, |
0 | 359 "caller must check"); |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
360 u2 entry_count = length / InstanceKlass::inner_class_next_offset; |
0 | 361 u4 size = 2 + entry_count * (2+2+2+2); |
362 | |
363 write_attribute_name_index("InnerClasses"); | |
364 write_u4(size); | |
365 write_u2(entry_count); | |
5967
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
366 for (; !iter.done(); iter.next()) { |
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
367 write_u2(iter.inner_class_info_index()); |
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
368 write_u2(iter.outer_class_info_index()); |
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
369 write_u2(iter.inner_name_index()); |
f7c4174b33ba
7109878: The instanceKlass EnclosingMethhod attribute fields can be folded into the _inner_class field.
jiangli
parents:
5922
diff
changeset
|
370 write_u2(iter.inner_access_flags()); |
0 | 371 } |
372 } | |
373 | |
374 // Write Synthetic attribute | |
375 // JVMSpec| Synthetic_attribute { | |
376 // JVMSpec| u2 attribute_name_index; | |
377 // JVMSpec| u4 attribute_length; | |
378 // JVMSpec| } | |
379 void JvmtiClassFileReconstituter::write_synthetic_attribute() { | |
380 write_attribute_name_index("Synthetic"); | |
381 write_u4(0); //length always zero | |
382 } | |
383 | |
384 // Compute size of LineNumberTable | |
385 u2 JvmtiClassFileReconstituter::line_number_table_entries(methodHandle method) { | |
386 // The line number table is compressed so we don't know how big it is until decompressed. | |
387 // Decompression is really fast so we just do it twice. | |
388 u2 num_entries = 0; | |
389 CompressedLineNumberReadStream stream(method->compressed_linenumber_table()); | |
390 while (stream.read_pair()) { | |
391 num_entries++; | |
392 } | |
393 return num_entries; | |
394 } | |
395 | |
396 // Write LineNumberTable attribute | |
397 // JVMSpec| LineNumberTable_attribute { | |
398 // JVMSpec| u2 attribute_name_index; | |
399 // JVMSpec| u4 attribute_length; | |
400 // JVMSpec| u2 line_number_table_length; | |
401 // JVMSpec| { u2 start_pc; | |
402 // JVMSpec| u2 line_number; | |
403 // JVMSpec| } line_number_table[line_number_table_length]; | |
404 // JVMSpec| } | |
405 void JvmtiClassFileReconstituter::write_line_number_table_attribute(methodHandle method, | |
406 u2 num_entries) { | |
407 | |
408 write_attribute_name_index("LineNumberTable"); | |
409 write_u4(2 + num_entries * (2 + 2)); | |
410 write_u2(num_entries); | |
411 | |
412 CompressedLineNumberReadStream stream(method->compressed_linenumber_table()); | |
413 while (stream.read_pair()) { | |
414 write_u2(stream.bci()); | |
415 write_u2(stream.line()); | |
416 } | |
417 } | |
418 | |
6772
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
419 // Write LocalVariableTable attribute |
4718
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
420 // JVMSpec| LocalVariableTable_attribute { |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
421 // JVMSpec| u2 attribute_name_index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
422 // JVMSpec| u4 attribute_length; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
423 // JVMSpec| u2 local_variable_table_length; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
424 // JVMSpec| { u2 start_pc; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
425 // JVMSpec| u2 length; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
426 // JVMSpec| u2 name_index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
427 // JVMSpec| u2 descriptor_index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
428 // JVMSpec| u2 index; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
429 // JVMSpec| } local_variable_table[local_variable_table_length]; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
430 // JVMSpec| } |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
431 void JvmtiClassFileReconstituter::write_local_variable_table_attribute(methodHandle method, u2 num_entries) { |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
432 write_attribute_name_index("LocalVariableTable"); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
433 write_u4(2 + num_entries * (2 + 2 + 2 + 2 + 2)); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
434 write_u2(num_entries); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
435 |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
436 assert(method->localvariable_table_length() == num_entries, "just checking"); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
437 |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
438 LocalVariableTableElement *elem = method->localvariable_table_start(); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
439 for (int j=0; j<method->localvariable_table_length(); j++) { |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
440 write_u2(elem->start_bci); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
441 write_u2(elem->length); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
442 write_u2(elem->name_cp_index); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
443 write_u2(elem->descriptor_cp_index); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
444 write_u2(elem->slot); |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
445 elem++; |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
446 } |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
447 } |
c01e115b095e
7064927: retransformClasses() does not pass in LocalVariableTable of a method
coleenp
parents:
3944
diff
changeset
|
448 |
6772
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
449 // Write LocalVariableTypeTable attribute |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
450 // JVMSpec| LocalVariableTypeTable_attribute { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
451 // JVMSpec| u2 attribute_name_index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
452 // JVMSpec| u4 attribute_length; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
453 // JVMSpec| u2 local_variable_type_table_length; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
454 // JVMSpec| { u2 start_pc; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
455 // JVMSpec| u2 length; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
456 // JVMSpec| u2 name_index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
457 // JVMSpec| u2 signature_index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
458 // JVMSpec| u2 index; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
459 // JVMSpec| } local_variable_type_table[local_variable_type_table_length]; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
460 // JVMSpec| } |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
461 void JvmtiClassFileReconstituter::write_local_variable_type_table_attribute(methodHandle method, u2 num_entries) { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
462 write_attribute_name_index("LocalVariableTypeTable"); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
463 write_u4(2 + num_entries * (2 + 2 + 2 + 2 + 2)); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
464 write_u2(num_entries); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
465 |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
466 LocalVariableTableElement *elem = method->localvariable_table_start(); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
467 for (int j=0; j<method->localvariable_table_length(); j++) { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
468 if (elem->signature_cp_index > 0) { |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
469 // Local variable has a generic signature - write LVTT attribute entry |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
470 write_u2(elem->start_bci); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
471 write_u2(elem->length); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
472 write_u2(elem->name_cp_index); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
473 write_u2(elem->signature_cp_index); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
474 write_u2(elem->slot); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
475 num_entries--; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
476 } |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
477 elem++; |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
478 } |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
479 assert(num_entries == 0, "just checking"); |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
480 } |
1cb8583c3da8
7191786: retransformClasses() does not pass in LocalVariableTypeTable of a method
minqi
parents:
6725
diff
changeset
|
481 |
0 | 482 // Write stack map table attribute |
483 // JSR-202| StackMapTable_attribute { | |
484 // JSR-202| u2 attribute_name_index; | |
485 // JSR-202| u4 attribute_length; | |
486 // JSR-202| u2 number_of_entries; | |
487 // JSR-202| stack_map_frame_entries[number_of_entries]; | |
488 // JSR-202| } | |
489 void JvmtiClassFileReconstituter::write_stackmap_table_attribute(methodHandle method, | |
490 int stackmap_len) { | |
491 | |
492 write_attribute_name_index("StackMapTable"); | |
493 write_u4(stackmap_len); | |
494 memcpy( | |
495 writeable_address(stackmap_len), | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
496 (void*)(method->stackmap_data()->adr_at(0)), |
0 | 497 stackmap_len); |
498 } | |
499 | |
500 // Write one method_info structure | |
501 // JVMSpec| method_info { | |
502 // JVMSpec| u2 access_flags; | |
503 // JVMSpec| u2 name_index; | |
504 // JVMSpec| u2 descriptor_index; | |
505 // JVMSpec| u2 attributes_count; | |
506 // JVMSpec| attribute_info attributes[attributes_count]; | |
507 // JVMSpec| } | |
508 void JvmtiClassFileReconstituter::write_method_info(methodHandle method) { | |
509 AccessFlags access_flags = method->access_flags(); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
510 ConstMethod* const_method = method->constMethod(); |
0 | 511 u2 generic_signature_index = const_method->generic_signature_index(); |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
512 AnnotationArray* anno = method->annotations(); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
513 AnnotationArray* param_anno = method->parameter_annotations(); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
514 AnnotationArray* default_anno = method->annotation_default(); |
0 | 515 |
516 write_u2(access_flags.get_flags() & JVM_RECOGNIZED_METHOD_MODIFIERS); | |
517 write_u2(const_method->name_index()); | |
518 write_u2(const_method->signature_index()); | |
519 | |
520 // write attributes in the same order javac does, so we can test with byte for | |
521 // byte comparison | |
522 int attr_count = 0; | |
523 if (const_method->code_size() != 0) { | |
524 ++attr_count; // has Code attribute | |
525 } | |
526 if (const_method->has_checked_exceptions()) { | |
527 ++attr_count; // has Exceptions attribute | |
528 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
529 if (default_anno != NULL) { |
0 | 530 ++attr_count; // has AnnotationDefault attribute |
531 } | |
532 // Deprecated attribute would go here | |
533 if (access_flags.is_synthetic()) { // FIXME | |
534 // ++attr_count; | |
535 } | |
536 if (generic_signature_index != 0) { | |
537 ++attr_count; | |
538 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
539 if (anno != NULL) { |
0 | 540 ++attr_count; // has RuntimeVisibleAnnotations attribute |
541 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
542 if (param_anno != NULL) { |
0 | 543 ++attr_count; // has RuntimeVisibleParameterAnnotations attribute |
544 } | |
545 | |
546 write_u2(attr_count); | |
547 if (const_method->code_size() > 0) { | |
548 write_code_attribute(method); | |
549 } | |
550 if (const_method->has_checked_exceptions()) { | |
551 write_exceptions_attribute(const_method); | |
552 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
553 if (default_anno != NULL) { |
0 | 554 write_annotations_attribute("AnnotationDefault", default_anno); |
555 } | |
556 // Deprecated attribute would go here | |
557 if (access_flags.is_synthetic()) { | |
558 // write_synthetic_attribute(); | |
559 } | |
560 if (generic_signature_index != 0) { | |
561 write_signature_attribute(generic_signature_index); | |
562 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
563 if (anno != NULL) { |
0 | 564 write_annotations_attribute("RuntimeVisibleAnnotations", anno); |
565 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
566 if (param_anno != NULL) { |
0 | 567 write_annotations_attribute("RuntimeVisibleParameterAnnotations", param_anno); |
568 } | |
569 } | |
570 | |
571 // Write the class attributes portion of ClassFile structure | |
572 // JVMSpec| u2 attributes_count; | |
573 // JVMSpec| attribute_info attributes[attributes_count]; | |
574 void JvmtiClassFileReconstituter::write_class_attributes() { | |
575 u2 inner_classes_length = inner_classes_attribute_length(); | |
2177
3582bf76420e
6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents:
1972
diff
changeset
|
576 Symbol* generic_signature = ikh()->generic_signature(); |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
577 AnnotationArray* anno = ikh()->class_annotations(); |
0 | 578 |
579 int attr_count = 0; | |
2177
3582bf76420e
6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents:
1972
diff
changeset
|
580 if (generic_signature != NULL) { |
0 | 581 ++attr_count; |
582 } | |
583 if (ikh()->source_file_name() != NULL) { | |
584 ++attr_count; | |
585 } | |
586 if (ikh()->source_debug_extension() != NULL) { | |
587 ++attr_count; | |
588 } | |
589 if (inner_classes_length > 0) { | |
590 ++attr_count; | |
591 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
592 if (anno != NULL) { |
0 | 593 ++attr_count; // has RuntimeVisibleAnnotations attribute |
594 } | |
595 | |
596 write_u2(attr_count); | |
597 | |
2177
3582bf76420e
6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents:
1972
diff
changeset
|
598 if (generic_signature != NULL) { |
3582bf76420e
6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents:
1972
diff
changeset
|
599 write_signature_attribute(symbol_to_cpool_index(generic_signature)); |
0 | 600 } |
601 if (ikh()->source_file_name() != NULL) { | |
602 write_source_file_attribute(); | |
603 } | |
604 if (ikh()->source_debug_extension() != NULL) { | |
605 write_source_debug_extension_attribute(); | |
606 } | |
607 if (inner_classes_length > 0) { | |
608 write_inner_classes_attribute(inner_classes_length); | |
609 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
610 if (anno != NULL) { |
0 | 611 write_annotations_attribute("RuntimeVisibleAnnotations", anno); |
612 } | |
613 } | |
614 | |
615 // Write the method information portion of ClassFile structure | |
616 // JVMSpec| u2 methods_count; | |
617 // JVMSpec| method_info methods[methods_count]; | |
618 void JvmtiClassFileReconstituter::write_method_infos() { | |
619 HandleMark hm(thread()); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
620 Array<Method*>* methods = ikh()->methods(); |
0 | 621 int num_methods = methods->length(); |
622 | |
623 write_u2(num_methods); | |
624 if (JvmtiExport::can_maintain_original_method_order()) { | |
625 int index; | |
626 int original_index; | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
627 intArray method_order(num_methods, 0); |
0 | 628 |
629 // invert the method order mapping | |
630 for (index = 0; index < num_methods; index++) { | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
631 original_index = ikh()->method_ordering()->at(index); |
0 | 632 assert(original_index >= 0 && original_index < num_methods, |
633 "invalid original method index"); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
634 method_order.at_put(original_index, index); |
0 | 635 } |
636 | |
637 // write in original order | |
638 for (original_index = 0; original_index < num_methods; original_index++) { | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
639 index = method_order.at(original_index); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
640 methodHandle method(thread(), methods->at(index)); |
0 | 641 write_method_info(method); |
642 } | |
643 } else { | |
644 // method order not preserved just dump the method infos | |
645 for (int index = 0; index < num_methods; index++) { | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
646 methodHandle method(thread(), methods->at(index)); |
0 | 647 write_method_info(method); |
648 } | |
649 } | |
650 } | |
651 | |
652 void JvmtiClassFileReconstituter::write_class_file_format() { | |
653 ReallocMark(); | |
654 | |
655 // JVMSpec| ClassFile { | |
656 // JVMSpec| u4 magic; | |
657 write_u4(0xCAFEBABE); | |
658 | |
659 // JVMSpec| u2 minor_version; | |
660 // JVMSpec| u2 major_version; | |
661 write_u2(ikh()->minor_version()); | |
662 u2 major = ikh()->major_version(); | |
663 write_u2(major); | |
664 | |
665 // JVMSpec| u2 constant_pool_count; | |
666 // JVMSpec| cp_info constant_pool[constant_pool_count-1]; | |
667 write_u2(cpool()->length()); | |
668 copy_cpool_bytes(writeable_address(cpool_size())); | |
669 | |
670 // JVMSpec| u2 access_flags; | |
671 write_u2(ikh()->access_flags().get_flags() & JVM_RECOGNIZED_CLASS_MODIFIERS); | |
672 | |
673 // JVMSpec| u2 this_class; | |
674 // JVMSpec| u2 super_class; | |
675 write_u2(class_symbol_to_cpool_index(ikh()->name())); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
676 Klass* super_class = ikh()->super(); |
0 | 677 write_u2(super_class == NULL? 0 : // zero for java.lang.Object |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
678 class_symbol_to_cpool_index(super_class->name())); |
0 | 679 |
680 // JVMSpec| u2 interfaces_count; | |
681 // JVMSpec| u2 interfaces[interfaces_count]; | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
682 Array<Klass*>* interfaces = ikh()->local_interfaces(); |
0 | 683 int num_interfaces = interfaces->length(); |
684 write_u2(num_interfaces); | |
685 for (int index = 0; index < num_interfaces; index++) { | |
686 HandleMark hm(thread()); | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
687 instanceKlassHandle iikh(thread(), interfaces->at(index)); |
0 | 688 write_u2(class_symbol_to_cpool_index(iikh->name())); |
689 } | |
690 | |
691 // JVMSpec| u2 fields_count; | |
692 // JVMSpec| field_info fields[fields_count]; | |
693 write_field_infos(); | |
694 | |
695 // JVMSpec| u2 methods_count; | |
696 // JVMSpec| method_info methods[methods_count]; | |
697 write_method_infos(); | |
698 | |
699 // JVMSpec| u2 attributes_count; | |
700 // JVMSpec| attribute_info attributes[attributes_count]; | |
701 // JVMSpec| } /* end ClassFile 8? | |
702 write_class_attributes(); | |
703 } | |
704 | |
705 address JvmtiClassFileReconstituter::writeable_address(size_t size) { | |
706 size_t used_size = _buffer_ptr - _buffer; | |
707 if (size + used_size >= _buffer_size) { | |
708 // compute the new buffer size: must be at least twice as big as before | |
709 // plus whatever new is being used; then convert to nice clean block boundary | |
710 size_t new_buffer_size = (size + _buffer_size*2 + 1) / initial_buffer_size | |
711 * initial_buffer_size; | |
712 | |
713 // VM goes belly-up if the memory isn't available, so cannot do OOM processing | |
714 _buffer = REALLOC_RESOURCE_ARRAY(u1, _buffer, _buffer_size, new_buffer_size); | |
715 _buffer_size = new_buffer_size; | |
716 _buffer_ptr = _buffer + used_size; | |
717 } | |
718 u1* ret_ptr = _buffer_ptr; | |
719 _buffer_ptr += size; | |
720 return ret_ptr; | |
721 } | |
722 | |
723 void JvmtiClassFileReconstituter::write_attribute_name_index(const char* name) { | |
2177
3582bf76420e
6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents:
1972
diff
changeset
|
724 TempNewSymbol sym = SymbolTable::probe(name, (int)strlen(name)); |
0 | 725 assert(sym != NULL, "attribute name symbol not found"); |
726 u2 attr_name_index = symbol_to_cpool_index(sym); | |
727 assert(attr_name_index != 0, "attribute name symbol not in constant pool"); | |
728 write_u2(attr_name_index); | |
729 } | |
730 | |
731 void JvmtiClassFileReconstituter::write_u1(u1 x) { | |
732 *writeable_address(1) = x; | |
733 } | |
734 | |
735 void JvmtiClassFileReconstituter::write_u2(u2 x) { | |
736 Bytes::put_Java_u2(writeable_address(2), x); | |
737 } | |
738 | |
739 void JvmtiClassFileReconstituter::write_u4(u4 x) { | |
740 Bytes::put_Java_u4(writeable_address(4), x); | |
741 } | |
742 | |
743 void JvmtiClassFileReconstituter::write_u8(u8 x) { | |
744 Bytes::put_Java_u8(writeable_address(8), x); | |
745 } | |
746 | |
747 void JvmtiClassFileReconstituter::copy_bytecodes(methodHandle mh, | |
748 unsigned char* bytecodes) { | |
749 // use a BytecodeStream to iterate over the bytecodes. JVM/fast bytecodes | |
750 // and the breakpoint bytecode are converted to their original bytecodes. | |
751 | |
752 BytecodeStream bs(mh); | |
753 | |
754 unsigned char* p = bytecodes; | |
755 Bytecodes::Code code; | |
6940
18fb7da42534
8000725: NPG: method_holder() and pool_holder() and pool_holder field should be InstanceKlass
coleenp
parents:
6772
diff
changeset
|
756 bool is_rewritten = mh->method_holder()->is_rewritten(); |
0 | 757 |
758 while ((code = bs.next()) >= 0) { | |
759 assert(Bytecodes::is_java_code(code), "sanity check"); | |
760 assert(code != Bytecodes::_breakpoint, "sanity check"); | |
761 | |
762 // length of bytecode (mnemonic + operands) | |
763 address bcp = bs.bcp(); | |
1565 | 764 int len = bs.instruction_size(); |
0 | 765 assert(len > 0, "length must be > 0"); |
766 | |
767 // copy the bytecodes | |
768 *p = (unsigned char) (bs.is_wide()? Bytecodes::_wide : code); | |
769 if (len > 1) { | |
770 memcpy(p+1, bcp+1, len-1); | |
771 } | |
772 | |
773 // During linking the get/put and invoke instructions are rewritten | |
774 // with an index into the constant pool cache. The original constant | |
775 // pool index must be returned to caller. Rewrite the index. | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
776 if (is_rewritten && len > 1) { |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
777 bool is_wide = false; |
0 | 778 switch (code) { |
779 case Bytecodes::_getstatic : // fall through | |
780 case Bytecodes::_putstatic : // fall through | |
781 case Bytecodes::_getfield : // fall through | |
782 case Bytecodes::_putfield : // fall through | |
783 case Bytecodes::_invokevirtual : // fall through | |
784 case Bytecodes::_invokespecial : // fall through | |
785 case Bytecodes::_invokestatic : // fall through | |
726
be93aad57795
6655646: dynamic languages need dynamically linked call sites
jrose
parents:
0
diff
changeset
|
786 case Bytecodes::_invokedynamic : // fall through |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
787 case Bytecodes::_invokeinterface : { |
5922
ff29ce866f23
7118280: The gbyc00102 JCK7 test causes an assert in JVM 7.0 fastdebug mode
dsamersoff
parents:
4718
diff
changeset
|
788 assert(len == 3 || |
ff29ce866f23
7118280: The gbyc00102 JCK7 test causes an assert in JVM 7.0 fastdebug mode
dsamersoff
parents:
4718
diff
changeset
|
789 (code == Bytecodes::_invokeinterface && len == 5) || |
ff29ce866f23
7118280: The gbyc00102 JCK7 test causes an assert in JVM 7.0 fastdebug mode
dsamersoff
parents:
4718
diff
changeset
|
790 (code == Bytecodes::_invokedynamic && len == 5), |
0 | 791 "sanity check"); |
5922
ff29ce866f23
7118280: The gbyc00102 JCK7 test causes an assert in JVM 7.0 fastdebug mode
dsamersoff
parents:
4718
diff
changeset
|
792 |
726
be93aad57795
6655646: dynamic languages need dynamically linked call sites
jrose
parents:
0
diff
changeset
|
793 int cpci = Bytes::get_native_u2(bcp+1); |
be93aad57795
6655646: dynamic languages need dynamically linked call sites
jrose
parents:
0
diff
changeset
|
794 bool is_invokedynamic = (EnableInvokeDynamic && code == Bytecodes::_invokedynamic); |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
795 ConstantPoolCacheEntry* entry; |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
796 if (is_invokedynamic) { |
726
be93aad57795
6655646: dynamic languages need dynamically linked call sites
jrose
parents:
0
diff
changeset
|
797 cpci = Bytes::get_native_u4(bcp+1); |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
798 entry = mh->constants()->invokedynamic_cp_cache_entry_at(cpci); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
799 } else { |
0 | 800 // cache cannot be pre-fetched since some classes won't have it yet |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
801 entry = mh->constants()->cache()->entry_at(cpci); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
802 } |
0 | 803 int i = entry->constant_pool_index(); |
804 assert(i < mh->constants()->length(), "sanity check"); | |
805 Bytes::put_Java_u2((address)(p+1), (u2)i); // java byte ordering | |
726
be93aad57795
6655646: dynamic languages need dynamically linked call sites
jrose
parents:
0
diff
changeset
|
806 if (is_invokedynamic) *(p+3) = *(p+4) = 0; |
0 | 807 break; |
808 } | |
6725
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
809 case Bytecodes::_ldc_w: |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
810 is_wide = true; // fall through |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
811 case Bytecodes::_ldc: { |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
812 if (bs.raw_code() == Bytecodes::_fast_aldc || bs.raw_code() == Bytecodes::_fast_aldc_w) { |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
813 int cpci = is_wide ? Bytes::get_native_u2(bcp+1) : (u1)(*(bcp+1)); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
814 int i = mh->constants()->object_to_cp_index(cpci); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
815 assert(i < mh->constants()->length(), "sanity check"); |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
816 if (is_wide) { |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
817 Bytes::put_Java_u2((address)(p+1), (u2)i); // java byte ordering |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
818 } else { |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
819 *(p+1) = (u1)i; |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
820 } |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
821 } |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
822 break; |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
823 } |
da91efe96a93
6964458: Reimplement class meta-data storage to use native memory
coleenp
parents:
6217
diff
changeset
|
824 } |
0 | 825 } |
826 | |
827 p += len; | |
828 } | |
829 } |