annotate src/share/vm/runtime/reflection.cpp @ 1721:413ad0331a0c

6977924: Changes for 6975078 produce build error with certain gcc versions Summary: The changes introduced for 6975078 assign badHeapOopVal to the _allocation field in the ResourceObj class. In 32 bit linux builds with certain versions of gcc this assignment will be flagged as an error while compiling allocation.cpp. In 32 bit builds the constant value badHeapOopVal (which is cast to an intptr_t) is negative. The _allocation field is typed as an unsigned intptr_t and gcc catches this as an error. Reviewed-by: jcoomes, ysr, phh
author johnc
date Wed, 18 Aug 2010 10:59:06 -0700
parents c18cbe5936b8
children f95d63e2154a
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1152
diff changeset
2 * Copyright (c) 1997, 2010, Oracle and/or its affiliates. All rights reserved.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a61af66fc99e Initial load
duke
parents:
diff changeset
4 *
a61af66fc99e Initial load
duke
parents:
diff changeset
5 * This code is free software; you can redistribute it and/or modify it
a61af66fc99e Initial load
duke
parents:
diff changeset
6 * under the terms of the GNU General Public License version 2 only, as
a61af66fc99e Initial load
duke
parents:
diff changeset
7 * published by the Free Software Foundation.
a61af66fc99e Initial load
duke
parents:
diff changeset
8 *
a61af66fc99e Initial load
duke
parents:
diff changeset
9 * This code is distributed in the hope that it will be useful, but WITHOUT
a61af66fc99e Initial load
duke
parents:
diff changeset
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a61af66fc99e Initial load
duke
parents:
diff changeset
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
a61af66fc99e Initial load
duke
parents:
diff changeset
12 * version 2 for more details (a copy is included in the LICENSE file that
a61af66fc99e Initial load
duke
parents:
diff changeset
13 * accompanied this code).
a61af66fc99e Initial load
duke
parents:
diff changeset
14 *
a61af66fc99e Initial load
duke
parents:
diff changeset
15 * You should have received a copy of the GNU General Public License version
a61af66fc99e Initial load
duke
parents:
diff changeset
16 * 2 along with this work; if not, write to the Free Software Foundation,
a61af66fc99e Initial load
duke
parents:
diff changeset
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a61af66fc99e Initial load
duke
parents:
diff changeset
18 *
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1152
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1152
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: 1152
diff changeset
21 * questions.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
22 *
a61af66fc99e Initial load
duke
parents:
diff changeset
23 */
a61af66fc99e Initial load
duke
parents:
diff changeset
24
a61af66fc99e Initial load
duke
parents:
diff changeset
25 #include "incls/_precompiled.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
26 #include "incls/_reflection.cpp.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
27
a61af66fc99e Initial load
duke
parents:
diff changeset
28 #define JAVA_1_5_VERSION 49
a61af66fc99e Initial load
duke
parents:
diff changeset
29
a61af66fc99e Initial load
duke
parents:
diff changeset
30 static void trace_class_resolution(klassOop to_class) {
a61af66fc99e Initial load
duke
parents:
diff changeset
31 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
32 int line_number = -1;
a61af66fc99e Initial load
duke
parents:
diff changeset
33 const char * source_file = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
34 klassOop caller = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
35 JavaThread* jthread = JavaThread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
36 if (jthread->has_last_Java_frame()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
37 vframeStream vfst(jthread);
a61af66fc99e Initial load
duke
parents:
diff changeset
38 // skip over any frames belonging to java.lang.Class
a61af66fc99e Initial load
duke
parents:
diff changeset
39 while (!vfst.at_end() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
40 instanceKlass::cast(vfst.method()->method_holder())->name() == vmSymbols::java_lang_Class()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
41 vfst.next();
a61af66fc99e Initial load
duke
parents:
diff changeset
42 }
a61af66fc99e Initial load
duke
parents:
diff changeset
43 if (!vfst.at_end()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
44 // this frame is a likely suspect
a61af66fc99e Initial load
duke
parents:
diff changeset
45 caller = vfst.method()->method_holder();
a61af66fc99e Initial load
duke
parents:
diff changeset
46 line_number = vfst.method()->line_number_from_bci(vfst.bci());
a61af66fc99e Initial load
duke
parents:
diff changeset
47 symbolOop s = instanceKlass::cast(vfst.method()->method_holder())->source_file_name();
a61af66fc99e Initial load
duke
parents:
diff changeset
48 if (s != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
49 source_file = s->as_C_string();
a61af66fc99e Initial load
duke
parents:
diff changeset
50 }
a61af66fc99e Initial load
duke
parents:
diff changeset
51 }
a61af66fc99e Initial load
duke
parents:
diff changeset
52 }
a61af66fc99e Initial load
duke
parents:
diff changeset
53 if (caller != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
54 const char * from = Klass::cast(caller)->external_name();
a61af66fc99e Initial load
duke
parents:
diff changeset
55 const char * to = Klass::cast(to_class)->external_name();
a61af66fc99e Initial load
duke
parents:
diff changeset
56 // print in a single call to reduce interleaving between threads
a61af66fc99e Initial load
duke
parents:
diff changeset
57 if (source_file != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
58 tty->print("RESOLVE %s %s %s:%d (reflection)\n", from, to, source_file, line_number);
a61af66fc99e Initial load
duke
parents:
diff changeset
59 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
60 tty->print("RESOLVE %s %s (reflection)\n", from, to);
a61af66fc99e Initial load
duke
parents:
diff changeset
61 }
a61af66fc99e Initial load
duke
parents:
diff changeset
62 }
a61af66fc99e Initial load
duke
parents:
diff changeset
63 }
a61af66fc99e Initial load
duke
parents:
diff changeset
64
a61af66fc99e Initial load
duke
parents:
diff changeset
65
a61af66fc99e Initial load
duke
parents:
diff changeset
66 oop Reflection::box(jvalue* value, BasicType type, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
67 if (type == T_VOID) {
a61af66fc99e Initial load
duke
parents:
diff changeset
68 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
69 }
a61af66fc99e Initial load
duke
parents:
diff changeset
70 if (type == T_OBJECT || type == T_ARRAY) {
a61af66fc99e Initial load
duke
parents:
diff changeset
71 // regular objects are not boxed
a61af66fc99e Initial load
duke
parents:
diff changeset
72 return (oop) value->l;
a61af66fc99e Initial load
duke
parents:
diff changeset
73 }
a61af66fc99e Initial load
duke
parents:
diff changeset
74 oop result = java_lang_boxing_object::create(type, value, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
75 if (result == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
76 THROW_(vmSymbols::java_lang_IllegalArgumentException(), result);
a61af66fc99e Initial load
duke
parents:
diff changeset
77 }
a61af66fc99e Initial load
duke
parents:
diff changeset
78 return result;
a61af66fc99e Initial load
duke
parents:
diff changeset
79 }
a61af66fc99e Initial load
duke
parents:
diff changeset
80
a61af66fc99e Initial load
duke
parents:
diff changeset
81
a61af66fc99e Initial load
duke
parents:
diff changeset
82 BasicType Reflection::unbox_for_primitive(oop box, jvalue* value, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
83 if (box == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
84 THROW_(vmSymbols::java_lang_IllegalArgumentException(), T_ILLEGAL);
a61af66fc99e Initial load
duke
parents:
diff changeset
85 }
a61af66fc99e Initial load
duke
parents:
diff changeset
86 return java_lang_boxing_object::get_value(box, value);
a61af66fc99e Initial load
duke
parents:
diff changeset
87 }
a61af66fc99e Initial load
duke
parents:
diff changeset
88
a61af66fc99e Initial load
duke
parents:
diff changeset
89 BasicType Reflection::unbox_for_regular_object(oop box, jvalue* value) {
a61af66fc99e Initial load
duke
parents:
diff changeset
90 // Note: box is really the unboxed oop. It might even be a Short, etc.!
a61af66fc99e Initial load
duke
parents:
diff changeset
91 value->l = (jobject) box;
a61af66fc99e Initial load
duke
parents:
diff changeset
92 return T_OBJECT;
a61af66fc99e Initial load
duke
parents:
diff changeset
93 }
a61af66fc99e Initial load
duke
parents:
diff changeset
94
a61af66fc99e Initial load
duke
parents:
diff changeset
95
a61af66fc99e Initial load
duke
parents:
diff changeset
96 void Reflection::widen(jvalue* value, BasicType current_type, BasicType wide_type, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
97 assert(wide_type != current_type, "widen should not be called with identical types");
a61af66fc99e Initial load
duke
parents:
diff changeset
98 switch (wide_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
99 case T_BOOLEAN:
a61af66fc99e Initial load
duke
parents:
diff changeset
100 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
101 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
102 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
103 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
104 switch (current_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
105 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
106 value->s = (jshort) value->b;
a61af66fc99e Initial load
duke
parents:
diff changeset
107 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
108 }
a61af66fc99e Initial load
duke
parents:
diff changeset
109 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
110 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
111 switch (current_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
112 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
113 value->i = (jint) value->b;
a61af66fc99e Initial load
duke
parents:
diff changeset
114 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
115 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
116 value->i = (jint) value->c;
a61af66fc99e Initial load
duke
parents:
diff changeset
117 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
118 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
119 value->i = (jint) value->s;
a61af66fc99e Initial load
duke
parents:
diff changeset
120 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
121 }
a61af66fc99e Initial load
duke
parents:
diff changeset
122 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
123 case T_LONG:
a61af66fc99e Initial load
duke
parents:
diff changeset
124 switch (current_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
125 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
126 value->j = (jlong) value->b;
a61af66fc99e Initial load
duke
parents:
diff changeset
127 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
128 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
129 value->j = (jlong) value->c;
a61af66fc99e Initial load
duke
parents:
diff changeset
130 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
131 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
132 value->j = (jlong) value->s;
a61af66fc99e Initial load
duke
parents:
diff changeset
133 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
134 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
135 value->j = (jlong) value->i;
a61af66fc99e Initial load
duke
parents:
diff changeset
136 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
137 }
a61af66fc99e Initial load
duke
parents:
diff changeset
138 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
139 case T_FLOAT:
a61af66fc99e Initial load
duke
parents:
diff changeset
140 switch (current_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
141 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
142 value->f = (jfloat) value->b;
a61af66fc99e Initial load
duke
parents:
diff changeset
143 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
144 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
145 value->f = (jfloat) value->c;
a61af66fc99e Initial load
duke
parents:
diff changeset
146 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
147 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
148 value->f = (jfloat) value->s;
a61af66fc99e Initial load
duke
parents:
diff changeset
149 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
150 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
151 value->f = (jfloat) value->i;
a61af66fc99e Initial load
duke
parents:
diff changeset
152 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
153 case T_LONG:
a61af66fc99e Initial load
duke
parents:
diff changeset
154 value->f = (jfloat) value->j;
a61af66fc99e Initial load
duke
parents:
diff changeset
155 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
156 }
a61af66fc99e Initial load
duke
parents:
diff changeset
157 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
158 case T_DOUBLE:
a61af66fc99e Initial load
duke
parents:
diff changeset
159 switch (current_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
160 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
161 value->d = (jdouble) value->b;
a61af66fc99e Initial load
duke
parents:
diff changeset
162 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
163 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
164 value->d = (jdouble) value->c;
a61af66fc99e Initial load
duke
parents:
diff changeset
165 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
166 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
167 value->d = (jdouble) value->s;
a61af66fc99e Initial load
duke
parents:
diff changeset
168 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
169 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
170 value->d = (jdouble) value->i;
a61af66fc99e Initial load
duke
parents:
diff changeset
171 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
172 case T_FLOAT:
a61af66fc99e Initial load
duke
parents:
diff changeset
173 value->d = (jdouble) value->f;
a61af66fc99e Initial load
duke
parents:
diff changeset
174 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
175 case T_LONG:
a61af66fc99e Initial load
duke
parents:
diff changeset
176 value->d = (jdouble) value->j;
a61af66fc99e Initial load
duke
parents:
diff changeset
177 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
178 }
a61af66fc99e Initial load
duke
parents:
diff changeset
179 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
180 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
181 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
182 }
a61af66fc99e Initial load
duke
parents:
diff changeset
183 THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "argument type mismatch");
a61af66fc99e Initial load
duke
parents:
diff changeset
184 }
a61af66fc99e Initial load
duke
parents:
diff changeset
185
a61af66fc99e Initial load
duke
parents:
diff changeset
186
a61af66fc99e Initial load
duke
parents:
diff changeset
187 BasicType Reflection::array_get(jvalue* value, arrayOop a, int index, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
188 if (!a->is_within_bounds(index)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
189 THROW_(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), T_ILLEGAL);
a61af66fc99e Initial load
duke
parents:
diff changeset
190 }
a61af66fc99e Initial load
duke
parents:
diff changeset
191 if (a->is_objArray()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
192 value->l = (jobject) objArrayOop(a)->obj_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
193 return T_OBJECT;
a61af66fc99e Initial load
duke
parents:
diff changeset
194 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
195 assert(a->is_typeArray(), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
196 BasicType type = typeArrayKlass::cast(a->klass())->element_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
197 switch (type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
198 case T_BOOLEAN:
a61af66fc99e Initial load
duke
parents:
diff changeset
199 value->z = typeArrayOop(a)->bool_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
200 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
201 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
202 value->c = typeArrayOop(a)->char_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
203 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
204 case T_FLOAT:
a61af66fc99e Initial load
duke
parents:
diff changeset
205 value->f = typeArrayOop(a)->float_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
206 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
207 case T_DOUBLE:
a61af66fc99e Initial load
duke
parents:
diff changeset
208 value->d = typeArrayOop(a)->double_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
209 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
210 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
211 value->b = typeArrayOop(a)->byte_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
212 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
213 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
214 value->s = typeArrayOop(a)->short_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
215 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
216 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
217 value->i = typeArrayOop(a)->int_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
218 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
219 case T_LONG:
a61af66fc99e Initial load
duke
parents:
diff changeset
220 value->j = typeArrayOop(a)->long_at(index);
a61af66fc99e Initial load
duke
parents:
diff changeset
221 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
222 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
223 return T_ILLEGAL;
a61af66fc99e Initial load
duke
parents:
diff changeset
224 }
a61af66fc99e Initial load
duke
parents:
diff changeset
225 return type;
a61af66fc99e Initial load
duke
parents:
diff changeset
226 }
a61af66fc99e Initial load
duke
parents:
diff changeset
227 }
a61af66fc99e Initial load
duke
parents:
diff changeset
228
a61af66fc99e Initial load
duke
parents:
diff changeset
229
a61af66fc99e Initial load
duke
parents:
diff changeset
230 void Reflection::array_set(jvalue* value, arrayOop a, int index, BasicType value_type, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
231 if (!a->is_within_bounds(index)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
232 THROW(vmSymbols::java_lang_ArrayIndexOutOfBoundsException());
a61af66fc99e Initial load
duke
parents:
diff changeset
233 }
a61af66fc99e Initial load
duke
parents:
diff changeset
234 if (a->is_objArray()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
235 if (value_type == T_OBJECT) {
a61af66fc99e Initial load
duke
parents:
diff changeset
236 oop obj = (oop) value->l;
a61af66fc99e Initial load
duke
parents:
diff changeset
237 if (obj != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
238 klassOop element_klass = objArrayKlass::cast(a->klass())->element_klass();
a61af66fc99e Initial load
duke
parents:
diff changeset
239 if (!obj->is_a(element_klass)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
240 THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "array element type mismatch");
a61af66fc99e Initial load
duke
parents:
diff changeset
241 }
a61af66fc99e Initial load
duke
parents:
diff changeset
242 }
a61af66fc99e Initial load
duke
parents:
diff changeset
243 objArrayOop(a)->obj_at_put(index, obj);
a61af66fc99e Initial load
duke
parents:
diff changeset
244 }
a61af66fc99e Initial load
duke
parents:
diff changeset
245 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
246 assert(a->is_typeArray(), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
247 BasicType array_type = typeArrayKlass::cast(a->klass())->element_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
248 if (array_type != value_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
249 // The widen operation can potentially throw an exception, but cannot block,
a61af66fc99e Initial load
duke
parents:
diff changeset
250 // so typeArrayOop a is safe if the call succeeds.
a61af66fc99e Initial load
duke
parents:
diff changeset
251 widen(value, value_type, array_type, CHECK);
a61af66fc99e Initial load
duke
parents:
diff changeset
252 }
a61af66fc99e Initial load
duke
parents:
diff changeset
253 switch (array_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
254 case T_BOOLEAN:
a61af66fc99e Initial load
duke
parents:
diff changeset
255 typeArrayOop(a)->bool_at_put(index, value->z);
a61af66fc99e Initial load
duke
parents:
diff changeset
256 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
257 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
258 typeArrayOop(a)->char_at_put(index, value->c);
a61af66fc99e Initial load
duke
parents:
diff changeset
259 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
260 case T_FLOAT:
a61af66fc99e Initial load
duke
parents:
diff changeset
261 typeArrayOop(a)->float_at_put(index, value->f);
a61af66fc99e Initial load
duke
parents:
diff changeset
262 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
263 case T_DOUBLE:
a61af66fc99e Initial load
duke
parents:
diff changeset
264 typeArrayOop(a)->double_at_put(index, value->d);
a61af66fc99e Initial load
duke
parents:
diff changeset
265 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
266 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
267 typeArrayOop(a)->byte_at_put(index, value->b);
a61af66fc99e Initial load
duke
parents:
diff changeset
268 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
269 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
270 typeArrayOop(a)->short_at_put(index, value->s);
a61af66fc99e Initial load
duke
parents:
diff changeset
271 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
272 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
273 typeArrayOop(a)->int_at_put(index, value->i);
a61af66fc99e Initial load
duke
parents:
diff changeset
274 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
275 case T_LONG:
a61af66fc99e Initial load
duke
parents:
diff changeset
276 typeArrayOop(a)->long_at_put(index, value->j);
a61af66fc99e Initial load
duke
parents:
diff changeset
277 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
278 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
279 THROW(vmSymbols::java_lang_IllegalArgumentException());
a61af66fc99e Initial load
duke
parents:
diff changeset
280 }
a61af66fc99e Initial load
duke
parents:
diff changeset
281 }
a61af66fc99e Initial load
duke
parents:
diff changeset
282 }
a61af66fc99e Initial load
duke
parents:
diff changeset
283
a61af66fc99e Initial load
duke
parents:
diff changeset
284
a61af66fc99e Initial load
duke
parents:
diff changeset
285 klassOop Reflection::basic_type_mirror_to_arrayklass(oop basic_type_mirror, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
286 assert(java_lang_Class::is_primitive(basic_type_mirror), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
287 BasicType type = java_lang_Class::primitive_type(basic_type_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
288 if (type == T_VOID) {
a61af66fc99e Initial load
duke
parents:
diff changeset
289 THROW_0(vmSymbols::java_lang_IllegalArgumentException());
a61af66fc99e Initial load
duke
parents:
diff changeset
290 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
291 return Universe::typeArrayKlassObj(type);
a61af66fc99e Initial load
duke
parents:
diff changeset
292 }
a61af66fc99e Initial load
duke
parents:
diff changeset
293 }
a61af66fc99e Initial load
duke
parents:
diff changeset
294
a61af66fc99e Initial load
duke
parents:
diff changeset
295
a61af66fc99e Initial load
duke
parents:
diff changeset
296 oop Reflection:: basic_type_arrayklass_to_mirror(klassOop basic_type_arrayklass, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
297 BasicType type = typeArrayKlass::cast(basic_type_arrayklass)->element_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
298 return Universe::java_mirror(type);
a61af66fc99e Initial load
duke
parents:
diff changeset
299 }
a61af66fc99e Initial load
duke
parents:
diff changeset
300
a61af66fc99e Initial load
duke
parents:
diff changeset
301
a61af66fc99e Initial load
duke
parents:
diff changeset
302 arrayOop Reflection::reflect_new_array(oop element_mirror, jint length, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
303 if (element_mirror == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
304 THROW_0(vmSymbols::java_lang_NullPointerException());
a61af66fc99e Initial load
duke
parents:
diff changeset
305 }
a61af66fc99e Initial load
duke
parents:
diff changeset
306 if (length < 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
307 THROW_0(vmSymbols::java_lang_NegativeArraySizeException());
a61af66fc99e Initial load
duke
parents:
diff changeset
308 }
a61af66fc99e Initial load
duke
parents:
diff changeset
309 if (java_lang_Class::is_primitive(element_mirror)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
310 klassOop tak = basic_type_mirror_to_arrayklass(element_mirror, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
311 return typeArrayKlass::cast(tak)->allocate(length, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
312 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
313 klassOop k = java_lang_Class::as_klassOop(element_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
314 if (Klass::cast(k)->oop_is_array() && arrayKlass::cast(k)->dimension() >= MAX_DIM) {
a61af66fc99e Initial load
duke
parents:
diff changeset
315 THROW_0(vmSymbols::java_lang_IllegalArgumentException());
a61af66fc99e Initial load
duke
parents:
diff changeset
316 }
a61af66fc99e Initial load
duke
parents:
diff changeset
317 return oopFactory::new_objArray(k, length, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
318 }
a61af66fc99e Initial load
duke
parents:
diff changeset
319 }
a61af66fc99e Initial load
duke
parents:
diff changeset
320
a61af66fc99e Initial load
duke
parents:
diff changeset
321
a61af66fc99e Initial load
duke
parents:
diff changeset
322 arrayOop Reflection::reflect_new_multi_array(oop element_mirror, typeArrayOop dim_array, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
323 assert(dim_array->is_typeArray(), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
324 assert(typeArrayKlass::cast(dim_array->klass())->element_type() == T_INT, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
325
a61af66fc99e Initial load
duke
parents:
diff changeset
326 if (element_mirror == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
327 THROW_0(vmSymbols::java_lang_NullPointerException());
a61af66fc99e Initial load
duke
parents:
diff changeset
328 }
a61af66fc99e Initial load
duke
parents:
diff changeset
329
a61af66fc99e Initial load
duke
parents:
diff changeset
330 int len = dim_array->length();
a61af66fc99e Initial load
duke
parents:
diff changeset
331 if (len <= 0 || len > MAX_DIM) {
a61af66fc99e Initial load
duke
parents:
diff changeset
332 THROW_0(vmSymbols::java_lang_IllegalArgumentException());
a61af66fc99e Initial load
duke
parents:
diff changeset
333 }
a61af66fc99e Initial load
duke
parents:
diff changeset
334
a61af66fc99e Initial load
duke
parents:
diff changeset
335 jint dimensions[MAX_DIM]; // C array copy of intArrayOop
a61af66fc99e Initial load
duke
parents:
diff changeset
336 for (int i = 0; i < len; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
337 int d = dim_array->int_at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
338 if (d < 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
339 THROW_0(vmSymbols::java_lang_NegativeArraySizeException());
a61af66fc99e Initial load
duke
parents:
diff changeset
340 }
a61af66fc99e Initial load
duke
parents:
diff changeset
341 dimensions[i] = d;
a61af66fc99e Initial load
duke
parents:
diff changeset
342 }
a61af66fc99e Initial load
duke
parents:
diff changeset
343
a61af66fc99e Initial load
duke
parents:
diff changeset
344 klassOop klass;
a61af66fc99e Initial load
duke
parents:
diff changeset
345 int dim = len;
a61af66fc99e Initial load
duke
parents:
diff changeset
346 if (java_lang_Class::is_primitive(element_mirror)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
347 klass = basic_type_mirror_to_arrayklass(element_mirror, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
348 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
349 klass = java_lang_Class::as_klassOop(element_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
350 if (Klass::cast(klass)->oop_is_array()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
351 int k_dim = arrayKlass::cast(klass)->dimension();
a61af66fc99e Initial load
duke
parents:
diff changeset
352 if (k_dim + len > MAX_DIM) {
a61af66fc99e Initial load
duke
parents:
diff changeset
353 THROW_0(vmSymbols::java_lang_IllegalArgumentException());
a61af66fc99e Initial load
duke
parents:
diff changeset
354 }
a61af66fc99e Initial load
duke
parents:
diff changeset
355 dim += k_dim;
a61af66fc99e Initial load
duke
parents:
diff changeset
356 }
a61af66fc99e Initial load
duke
parents:
diff changeset
357 }
a61af66fc99e Initial load
duke
parents:
diff changeset
358 klass = Klass::cast(klass)->array_klass(dim, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
359 oop obj = arrayKlass::cast(klass)->multi_allocate(len, dimensions, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
360 assert(obj->is_array(), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
361 return arrayOop(obj);
a61af66fc99e Initial load
duke
parents:
diff changeset
362 }
a61af66fc99e Initial load
duke
parents:
diff changeset
363
a61af66fc99e Initial load
duke
parents:
diff changeset
364
a61af66fc99e Initial load
duke
parents:
diff changeset
365 oop Reflection::array_component_type(oop mirror, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
366 if (java_lang_Class::is_primitive(mirror)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
367 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
368 }
a61af66fc99e Initial load
duke
parents:
diff changeset
369
a61af66fc99e Initial load
duke
parents:
diff changeset
370 klassOop klass = java_lang_Class::as_klassOop(mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
371 if (!Klass::cast(klass)->oop_is_array()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
372 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
373 }
a61af66fc99e Initial load
duke
parents:
diff changeset
374
a61af66fc99e Initial load
duke
parents:
diff changeset
375 oop result = arrayKlass::cast(klass)->component_mirror();
a61af66fc99e Initial load
duke
parents:
diff changeset
376 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
377 oop result2 = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
378 if (arrayKlass::cast(klass)->dimension() == 1) {
a61af66fc99e Initial load
duke
parents:
diff changeset
379 if (Klass::cast(klass)->oop_is_typeArray()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
380 result2 = basic_type_arrayklass_to_mirror(klass, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
381 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
382 result2 = Klass::cast(objArrayKlass::cast(klass)->element_klass())->java_mirror();
a61af66fc99e Initial load
duke
parents:
diff changeset
383 }
a61af66fc99e Initial load
duke
parents:
diff changeset
384 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
385 klassOop lower_dim = arrayKlass::cast(klass)->lower_dimension();
a61af66fc99e Initial load
duke
parents:
diff changeset
386 assert(Klass::cast(lower_dim)->oop_is_array(), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
387 result2 = Klass::cast(lower_dim)->java_mirror();
a61af66fc99e Initial load
duke
parents:
diff changeset
388 }
a61af66fc99e Initial load
duke
parents:
diff changeset
389 assert(result == result2, "results must be consistent");
a61af66fc99e Initial load
duke
parents:
diff changeset
390 #endif //ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
391 return result;
a61af66fc99e Initial load
duke
parents:
diff changeset
392 }
a61af66fc99e Initial load
duke
parents:
diff changeset
393
a61af66fc99e Initial load
duke
parents:
diff changeset
394
a61af66fc99e Initial load
duke
parents:
diff changeset
395 bool Reflection::reflect_check_access(klassOop field_class, AccessFlags acc, klassOop target_class, bool is_method_invoke, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
396 // field_class : declaring class
a61af66fc99e Initial load
duke
parents:
diff changeset
397 // acc : declared field access
a61af66fc99e Initial load
duke
parents:
diff changeset
398 // target_class : for protected
a61af66fc99e Initial load
duke
parents:
diff changeset
399
a61af66fc99e Initial load
duke
parents:
diff changeset
400 // Check if field or method is accessible to client. Throw an
a61af66fc99e Initial load
duke
parents:
diff changeset
401 // IllegalAccessException and return false if not.
a61af66fc99e Initial load
duke
parents:
diff changeset
402
a61af66fc99e Initial load
duke
parents:
diff changeset
403 // The "client" is the class associated with the nearest real frame
a61af66fc99e Initial load
duke
parents:
diff changeset
404 // getCallerClass already skips Method.invoke frames, so pass 0 in
a61af66fc99e Initial load
duke
parents:
diff changeset
405 // that case (same as classic).
a61af66fc99e Initial load
duke
parents:
diff changeset
406 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
407 assert(THREAD->is_Java_thread(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
408 klassOop client_class = ((JavaThread *)THREAD)->security_get_caller_class(is_method_invoke ? 0 : 1);
a61af66fc99e Initial load
duke
parents:
diff changeset
409
a61af66fc99e Initial load
duke
parents:
diff changeset
410 if (client_class != field_class) {
a61af66fc99e Initial load
duke
parents:
diff changeset
411 if (!verify_class_access(client_class, field_class, false)
a61af66fc99e Initial load
duke
parents:
diff changeset
412 || !verify_field_access(client_class,
a61af66fc99e Initial load
duke
parents:
diff changeset
413 field_class,
a61af66fc99e Initial load
duke
parents:
diff changeset
414 field_class,
a61af66fc99e Initial load
duke
parents:
diff changeset
415 acc,
a61af66fc99e Initial load
duke
parents:
diff changeset
416 false)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
417 THROW_(vmSymbols::java_lang_IllegalAccessException(), false);
a61af66fc99e Initial load
duke
parents:
diff changeset
418 }
a61af66fc99e Initial load
duke
parents:
diff changeset
419 }
a61af66fc99e Initial load
duke
parents:
diff changeset
420
a61af66fc99e Initial load
duke
parents:
diff changeset
421 // Additional test for protected members: JLS 6.6.2
a61af66fc99e Initial load
duke
parents:
diff changeset
422
a61af66fc99e Initial load
duke
parents:
diff changeset
423 if (acc.is_protected()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
424 if (target_class != client_class) {
a61af66fc99e Initial load
duke
parents:
diff changeset
425 if (!is_same_class_package(client_class, field_class)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
426 if (!Klass::cast(target_class)->is_subclass_of(client_class)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
427 THROW_(vmSymbols::java_lang_IllegalAccessException(), false);
a61af66fc99e Initial load
duke
parents:
diff changeset
428 }
a61af66fc99e Initial load
duke
parents:
diff changeset
429 }
a61af66fc99e Initial load
duke
parents:
diff changeset
430 }
a61af66fc99e Initial load
duke
parents:
diff changeset
431 }
a61af66fc99e Initial load
duke
parents:
diff changeset
432
a61af66fc99e Initial load
duke
parents:
diff changeset
433 // Passed all tests
a61af66fc99e Initial load
duke
parents:
diff changeset
434 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
435 }
a61af66fc99e Initial load
duke
parents:
diff changeset
436
a61af66fc99e Initial load
duke
parents:
diff changeset
437
a61af66fc99e Initial load
duke
parents:
diff changeset
438 bool Reflection::verify_class_access(klassOop current_class, klassOop new_class, bool classloader_only) {
a61af66fc99e Initial load
duke
parents:
diff changeset
439 // Verify that current_class can access new_class. If the classloader_only
a61af66fc99e Initial load
duke
parents:
diff changeset
440 // flag is set, we automatically allow any accesses in which current_class
a61af66fc99e Initial load
duke
parents:
diff changeset
441 // doesn't have a classloader.
a61af66fc99e Initial load
duke
parents:
diff changeset
442 if ((current_class == NULL) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
443 (current_class == new_class) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
444 (instanceKlass::cast(new_class)->is_public()) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
445 is_same_class_package(current_class, new_class)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
446 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
447 }
a61af66fc99e Initial load
duke
parents:
diff changeset
448 // New (1.4) reflection implementation. Allow all accesses from
a61af66fc99e Initial load
duke
parents:
diff changeset
449 // sun/reflect/MagicAccessorImpl subclasses to succeed trivially.
a61af66fc99e Initial load
duke
parents:
diff changeset
450 if ( JDK_Version::is_gte_jdk14x_version()
a61af66fc99e Initial load
duke
parents:
diff changeset
451 && UseNewReflection
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
452 && Klass::cast(current_class)->is_subclass_of(SystemDictionary::reflect_MagicAccessorImpl_klass())) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
453 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
454 }
a61af66fc99e Initial load
duke
parents:
diff changeset
455
a61af66fc99e Initial load
duke
parents:
diff changeset
456 return can_relax_access_check_for(current_class, new_class, classloader_only);
a61af66fc99e Initial load
duke
parents:
diff changeset
457 }
a61af66fc99e Initial load
duke
parents:
diff changeset
458
431
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
459 static bool under_host_klass(instanceKlass* ik, klassOop host_klass) {
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
460 DEBUG_ONLY(int inf_loop_check = 1000 * 1000 * 1000);
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
461 for (;;) {
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
462 klassOop hc = (klassOop) ik->host_klass();
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
463 if (hc == NULL) return false;
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
464 if (hc == host_klass) return true;
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
465 ik = instanceKlass::cast(hc);
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
466
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
467 // There's no way to make a host class loop short of patching memory.
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
468 // Therefore there cannot be a loop here unles there's another bug.
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
469 // Still, let's check for it.
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
470 assert(--inf_loop_check > 0, "no host_klass loop");
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
471 }
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
472 }
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
473
0
a61af66fc99e Initial load
duke
parents:
diff changeset
474 bool Reflection::can_relax_access_check_for(
a61af66fc99e Initial load
duke
parents:
diff changeset
475 klassOop accessor, klassOop accessee, bool classloader_only) {
a61af66fc99e Initial load
duke
parents:
diff changeset
476 instanceKlass* accessor_ik = instanceKlass::cast(accessor);
a61af66fc99e Initial load
duke
parents:
diff changeset
477 instanceKlass* accessee_ik = instanceKlass::cast(accessee);
431
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
478
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
479 // If either is on the other's host_klass chain, access is OK,
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
480 // because one is inside the other.
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
481 if (under_host_klass(accessor_ik, accessee) ||
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
482 under_host_klass(accessee_ik, accessor))
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
483 return true;
a45484ea312d 6653858: dynamic languages need to be able to load anonymous classes
jrose
parents: 196
diff changeset
484
1152
cd37471eaecc 6914206: change way of permission checking for generated MethodHandle adapters
twisti
parents: 1142
diff changeset
485 // Adapter frames can access anything.
cd37471eaecc 6914206: change way of permission checking for generated MethodHandle adapters
twisti
parents: 1142
diff changeset
486 if (MethodHandleCompiler::klass_is_method_handle_adapter_holder(accessor))
cd37471eaecc 6914206: change way of permission checking for generated MethodHandle adapters
twisti
parents: 1142
diff changeset
487 // This is an internal adapter frame from the MethodHandleCompiler.
cd37471eaecc 6914206: change way of permission checking for generated MethodHandle adapters
twisti
parents: 1142
diff changeset
488 return true;
cd37471eaecc 6914206: change way of permission checking for generated MethodHandle adapters
twisti
parents: 1142
diff changeset
489
0
a61af66fc99e Initial load
duke
parents:
diff changeset
490 if (RelaxAccessControlCheck ||
a61af66fc99e Initial load
duke
parents:
diff changeset
491 (accessor_ik->major_version() < JAVA_1_5_VERSION &&
a61af66fc99e Initial load
duke
parents:
diff changeset
492 accessee_ik->major_version() < JAVA_1_5_VERSION)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
493 return classloader_only &&
a61af66fc99e Initial load
duke
parents:
diff changeset
494 Verifier::relax_verify_for(accessor_ik->class_loader()) &&
a61af66fc99e Initial load
duke
parents:
diff changeset
495 accessor_ik->protection_domain() == accessee_ik->protection_domain() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
496 accessor_ik->class_loader() == accessee_ik->class_loader();
a61af66fc99e Initial load
duke
parents:
diff changeset
497 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
498 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
499 }
a61af66fc99e Initial load
duke
parents:
diff changeset
500 }
a61af66fc99e Initial load
duke
parents:
diff changeset
501
a61af66fc99e Initial load
duke
parents:
diff changeset
502 bool Reflection::verify_field_access(klassOop current_class,
a61af66fc99e Initial load
duke
parents:
diff changeset
503 klassOop resolved_class,
a61af66fc99e Initial load
duke
parents:
diff changeset
504 klassOop field_class,
a61af66fc99e Initial load
duke
parents:
diff changeset
505 AccessFlags access,
a61af66fc99e Initial load
duke
parents:
diff changeset
506 bool classloader_only,
a61af66fc99e Initial load
duke
parents:
diff changeset
507 bool protected_restriction) {
a61af66fc99e Initial load
duke
parents:
diff changeset
508 // Verify that current_class can access a field of field_class, where that
a61af66fc99e Initial load
duke
parents:
diff changeset
509 // field's access bits are "access". We assume that we've already verified
a61af66fc99e Initial load
duke
parents:
diff changeset
510 // that current_class can access field_class.
a61af66fc99e Initial load
duke
parents:
diff changeset
511 //
a61af66fc99e Initial load
duke
parents:
diff changeset
512 // If the classloader_only flag is set, we automatically allow any accesses
a61af66fc99e Initial load
duke
parents:
diff changeset
513 // in which current_class doesn't have a classloader.
a61af66fc99e Initial load
duke
parents:
diff changeset
514 //
a61af66fc99e Initial load
duke
parents:
diff changeset
515 // "resolved_class" is the runtime type of "field_class". Sometimes we don't
a61af66fc99e Initial load
duke
parents:
diff changeset
516 // need this distinction (e.g. if all we have is the runtime type, or during
a61af66fc99e Initial load
duke
parents:
diff changeset
517 // class file parsing when we only care about the static type); in that case
a61af66fc99e Initial load
duke
parents:
diff changeset
518 // callers should ensure that resolved_class == field_class.
a61af66fc99e Initial load
duke
parents:
diff changeset
519 //
a61af66fc99e Initial load
duke
parents:
diff changeset
520 if ((current_class == NULL) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
521 (current_class == field_class) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
522 access.is_public()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
523 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
524 }
a61af66fc99e Initial load
duke
parents:
diff changeset
525
a61af66fc99e Initial load
duke
parents:
diff changeset
526 if (access.is_protected()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
527 if (!protected_restriction) {
a61af66fc99e Initial load
duke
parents:
diff changeset
528 // See if current_class is a subclass of field_class
a61af66fc99e Initial load
duke
parents:
diff changeset
529 if (Klass::cast(current_class)->is_subclass_of(field_class)) {
115
e7a91a357527 6622385: Accessing protected static methods
kamg
parents: 51
diff changeset
530 if (access.is_static() || // static fields are ok, see 6622385
e7a91a357527 6622385: Accessing protected static methods
kamg
parents: 51
diff changeset
531 current_class == resolved_class ||
0
a61af66fc99e Initial load
duke
parents:
diff changeset
532 field_class == resolved_class ||
a61af66fc99e Initial load
duke
parents:
diff changeset
533 Klass::cast(current_class)->is_subclass_of(resolved_class) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
534 Klass::cast(resolved_class)->is_subclass_of(current_class)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
535 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
536 }
a61af66fc99e Initial load
duke
parents:
diff changeset
537 }
a61af66fc99e Initial load
duke
parents:
diff changeset
538 }
a61af66fc99e Initial load
duke
parents:
diff changeset
539 }
a61af66fc99e Initial load
duke
parents:
diff changeset
540
a61af66fc99e Initial load
duke
parents:
diff changeset
541 if (!access.is_private() && is_same_class_package(current_class, field_class)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
542 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
543 }
a61af66fc99e Initial load
duke
parents:
diff changeset
544
a61af66fc99e Initial load
duke
parents:
diff changeset
545 // New (1.4) reflection implementation. Allow all accesses from
a61af66fc99e Initial load
duke
parents:
diff changeset
546 // sun/reflect/MagicAccessorImpl subclasses to succeed trivially.
a61af66fc99e Initial load
duke
parents:
diff changeset
547 if ( JDK_Version::is_gte_jdk14x_version()
a61af66fc99e Initial load
duke
parents:
diff changeset
548 && UseNewReflection
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
549 && Klass::cast(current_class)->is_subclass_of(SystemDictionary::reflect_MagicAccessorImpl_klass())) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
550 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
551 }
a61af66fc99e Initial load
duke
parents:
diff changeset
552
a61af66fc99e Initial load
duke
parents:
diff changeset
553 return can_relax_access_check_for(
a61af66fc99e Initial load
duke
parents:
diff changeset
554 current_class, field_class, classloader_only);
a61af66fc99e Initial load
duke
parents:
diff changeset
555 }
a61af66fc99e Initial load
duke
parents:
diff changeset
556
a61af66fc99e Initial load
duke
parents:
diff changeset
557
a61af66fc99e Initial load
duke
parents:
diff changeset
558 bool Reflection::is_same_class_package(klassOop class1, klassOop class2) {
a61af66fc99e Initial load
duke
parents:
diff changeset
559 return instanceKlass::cast(class1)->is_same_class_package(class2);
a61af66fc99e Initial load
duke
parents:
diff changeset
560 }
a61af66fc99e Initial load
duke
parents:
diff changeset
561
665
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
562 bool Reflection::is_same_package_member(klassOop class1, klassOop class2, TRAPS) {
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
563 return instanceKlass::cast(class1)->is_same_package_member(class2, THREAD);
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
564 }
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
565
0
a61af66fc99e Initial load
duke
parents:
diff changeset
566
a61af66fc99e Initial load
duke
parents:
diff changeset
567 // Checks that the 'outer' klass has declared 'inner' as being an inner klass. If not,
a61af66fc99e Initial load
duke
parents:
diff changeset
568 // throw an incompatible class change exception
665
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
569 // If inner_is_member, require the inner to be a member of the outer.
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
570 // If !inner_is_member, require the inner to be anonymous (a non-member).
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
571 // Caller is responsible for figuring out in advance which case must be true.
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
572 void Reflection::check_for_inner_class(instanceKlassHandle outer, instanceKlassHandle inner,
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
573 bool inner_is_member, TRAPS) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
574 const int inner_class_info_index = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
575 const int outer_class_info_index = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
576
a61af66fc99e Initial load
duke
parents:
diff changeset
577 typeArrayHandle icls (THREAD, outer->inner_classes());
a61af66fc99e Initial load
duke
parents:
diff changeset
578 constantPoolHandle cp (THREAD, outer->constants());
a61af66fc99e Initial load
duke
parents:
diff changeset
579 for(int i = 0; i < icls->length(); i += 4) {
a61af66fc99e Initial load
duke
parents:
diff changeset
580 int ioff = icls->ushort_at(i + inner_class_info_index);
a61af66fc99e Initial load
duke
parents:
diff changeset
581 int ooff = icls->ushort_at(i + outer_class_info_index);
a61af66fc99e Initial load
duke
parents:
diff changeset
582
665
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
583 if (inner_is_member && ioff != 0 && ooff != 0) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
584 klassOop o = cp->klass_at(ooff, CHECK);
a61af66fc99e Initial load
duke
parents:
diff changeset
585 if (o == outer()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
586 klassOop i = cp->klass_at(ioff, CHECK);
a61af66fc99e Initial load
duke
parents:
diff changeset
587 if (i == inner()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
588 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
589 }
a61af66fc99e Initial load
duke
parents:
diff changeset
590 }
a61af66fc99e Initial load
duke
parents:
diff changeset
591 }
665
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
592 if (!inner_is_member && ioff != 0 && ooff == 0 &&
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
593 cp->klass_name_at_matches(inner, ioff)) {
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
594 klassOop i = cp->klass_at(ioff, CHECK);
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
595 if (i == inner()) {
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
596 return;
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
597 }
c89f86385056 6814659: separable cleanups and subroutines for 6655638
jrose
parents: 431
diff changeset
598 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
599 }
a61af66fc99e Initial load
duke
parents:
diff changeset
600
a61af66fc99e Initial load
duke
parents:
diff changeset
601 // 'inner' not declared as an inner klass in outer
a61af66fc99e Initial load
duke
parents:
diff changeset
602 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
603 Exceptions::fthrow(
a61af66fc99e Initial load
duke
parents:
diff changeset
604 THREAD_AND_LOCATION,
a61af66fc99e Initial load
duke
parents:
diff changeset
605 vmSymbolHandles::java_lang_IncompatibleClassChangeError(),
a61af66fc99e Initial load
duke
parents:
diff changeset
606 "%s and %s disagree on InnerClasses attribute",
a61af66fc99e Initial load
duke
parents:
diff changeset
607 outer->external_name(),
a61af66fc99e Initial load
duke
parents:
diff changeset
608 inner->external_name()
a61af66fc99e Initial load
duke
parents:
diff changeset
609 );
a61af66fc99e Initial load
duke
parents:
diff changeset
610 }
a61af66fc99e Initial load
duke
parents:
diff changeset
611
a61af66fc99e Initial load
duke
parents:
diff changeset
612 // Utility method converting a single SignatureStream element into java.lang.Class instance
a61af66fc99e Initial load
duke
parents:
diff changeset
613
a61af66fc99e Initial load
duke
parents:
diff changeset
614 oop get_mirror_from_signature(methodHandle method, SignatureStream* ss, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
615 switch (ss->type()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
616 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
617 assert(ss->type() != T_VOID || ss->at_return_type(), "T_VOID should only appear as return type");
a61af66fc99e Initial load
duke
parents:
diff changeset
618 return java_lang_Class::primitive_mirror(ss->type());
a61af66fc99e Initial load
duke
parents:
diff changeset
619 case T_OBJECT:
a61af66fc99e Initial load
duke
parents:
diff changeset
620 case T_ARRAY:
a61af66fc99e Initial load
duke
parents:
diff changeset
621 symbolOop name = ss->as_symbol(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
622 oop loader = instanceKlass::cast(method->method_holder())->class_loader();
a61af66fc99e Initial load
duke
parents:
diff changeset
623 oop protection_domain = instanceKlass::cast(method->method_holder())->protection_domain();
a61af66fc99e Initial load
duke
parents:
diff changeset
624 klassOop k = SystemDictionary::resolve_or_fail(
a61af66fc99e Initial load
duke
parents:
diff changeset
625 symbolHandle(THREAD, name),
a61af66fc99e Initial load
duke
parents:
diff changeset
626 Handle(THREAD, loader),
a61af66fc99e Initial load
duke
parents:
diff changeset
627 Handle(THREAD, protection_domain),
a61af66fc99e Initial load
duke
parents:
diff changeset
628 true, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
629 if (TraceClassResolution) {
a61af66fc99e Initial load
duke
parents:
diff changeset
630 trace_class_resolution(k);
a61af66fc99e Initial load
duke
parents:
diff changeset
631 }
a61af66fc99e Initial load
duke
parents:
diff changeset
632 return k->klass_part()->java_mirror();
a61af66fc99e Initial load
duke
parents:
diff changeset
633 };
a61af66fc99e Initial load
duke
parents:
diff changeset
634 }
a61af66fc99e Initial load
duke
parents:
diff changeset
635
a61af66fc99e Initial load
duke
parents:
diff changeset
636
a61af66fc99e Initial load
duke
parents:
diff changeset
637 objArrayHandle Reflection::get_parameter_types(methodHandle method, int parameter_count, oop* return_type, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
638 // Allocate array holding parameter types (java.lang.Class instances)
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
639 objArrayOop m = oopFactory::new_objArray(SystemDictionary::Class_klass(), parameter_count, CHECK_(objArrayHandle()));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
640 objArrayHandle mirrors (THREAD, m);
a61af66fc99e Initial load
duke
parents:
diff changeset
641 int index = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
642 // Collect parameter types
a61af66fc99e Initial load
duke
parents:
diff changeset
643 symbolHandle signature (THREAD, method->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
644 SignatureStream ss(signature);
a61af66fc99e Initial load
duke
parents:
diff changeset
645 while (!ss.at_return_type()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
646 oop mirror = get_mirror_from_signature(method, &ss, CHECK_(objArrayHandle()));
a61af66fc99e Initial load
duke
parents:
diff changeset
647 mirrors->obj_at_put(index++, mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
648 ss.next();
a61af66fc99e Initial load
duke
parents:
diff changeset
649 }
a61af66fc99e Initial load
duke
parents:
diff changeset
650 assert(index == parameter_count, "invalid parameter count");
a61af66fc99e Initial load
duke
parents:
diff changeset
651 if (return_type != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
652 // Collect return type as well
a61af66fc99e Initial load
duke
parents:
diff changeset
653 assert(ss.at_return_type(), "return type should be present");
a61af66fc99e Initial load
duke
parents:
diff changeset
654 *return_type = get_mirror_from_signature(method, &ss, CHECK_(objArrayHandle()));
a61af66fc99e Initial load
duke
parents:
diff changeset
655 }
a61af66fc99e Initial load
duke
parents:
diff changeset
656 return mirrors;
a61af66fc99e Initial load
duke
parents:
diff changeset
657 }
a61af66fc99e Initial load
duke
parents:
diff changeset
658
a61af66fc99e Initial load
duke
parents:
diff changeset
659 objArrayHandle Reflection::get_exception_types(methodHandle method, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
660 return method->resolved_checked_exceptions(CHECK_(objArrayHandle()));
a61af66fc99e Initial load
duke
parents:
diff changeset
661 }
a61af66fc99e Initial load
duke
parents:
diff changeset
662
a61af66fc99e Initial load
duke
parents:
diff changeset
663
a61af66fc99e Initial load
duke
parents:
diff changeset
664 Handle Reflection::new_type(symbolHandle signature, KlassHandle k, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
665 // Basic types
a61af66fc99e Initial load
duke
parents:
diff changeset
666 BasicType type = vmSymbols::signature_type(signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
667 if (type != T_OBJECT) {
a61af66fc99e Initial load
duke
parents:
diff changeset
668 return Handle(THREAD, Universe::java_mirror(type));
a61af66fc99e Initial load
duke
parents:
diff changeset
669 }
a61af66fc99e Initial load
duke
parents:
diff changeset
670
a61af66fc99e Initial load
duke
parents:
diff changeset
671 oop loader = instanceKlass::cast(k())->class_loader();
a61af66fc99e Initial load
duke
parents:
diff changeset
672 oop protection_domain = Klass::cast(k())->protection_domain();
a61af66fc99e Initial load
duke
parents:
diff changeset
673 klassOop result = SystemDictionary::resolve_or_fail(signature,
a61af66fc99e Initial load
duke
parents:
diff changeset
674 Handle(THREAD, loader),
a61af66fc99e Initial load
duke
parents:
diff changeset
675 Handle(THREAD, protection_domain),
a61af66fc99e Initial load
duke
parents:
diff changeset
676 true, CHECK_(Handle()));
a61af66fc99e Initial load
duke
parents:
diff changeset
677
a61af66fc99e Initial load
duke
parents:
diff changeset
678 if (TraceClassResolution) {
a61af66fc99e Initial load
duke
parents:
diff changeset
679 trace_class_resolution(result);
a61af66fc99e Initial load
duke
parents:
diff changeset
680 }
a61af66fc99e Initial load
duke
parents:
diff changeset
681
a61af66fc99e Initial load
duke
parents:
diff changeset
682 oop nt = Klass::cast(result)->java_mirror();
a61af66fc99e Initial load
duke
parents:
diff changeset
683 return Handle(THREAD, nt);
a61af66fc99e Initial load
duke
parents:
diff changeset
684 }
a61af66fc99e Initial load
duke
parents:
diff changeset
685
a61af66fc99e Initial load
duke
parents:
diff changeset
686
a61af66fc99e Initial load
duke
parents:
diff changeset
687 oop Reflection::new_method(methodHandle method, bool intern_name, bool for_constant_pool_access, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
688 // In jdk1.2.x, getMethods on an interface erroneously includes <clinit>, thus the complicated assert.
a61af66fc99e Initial load
duke
parents:
diff changeset
689 // Also allow sun.reflect.ConstantPool to refer to <clinit> methods as java.lang.reflect.Methods.
a61af66fc99e Initial load
duke
parents:
diff changeset
690 assert(!method()->is_initializer() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
691 (for_constant_pool_access && method()->is_static()) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
692 (method()->name() == vmSymbols::class_initializer_name()
a61af66fc99e Initial load
duke
parents:
diff changeset
693 && Klass::cast(method()->method_holder())->is_interface() && JDK_Version::is_jdk12x_version()), "should call new_constructor instead");
a61af66fc99e Initial load
duke
parents:
diff changeset
694 instanceKlassHandle holder (THREAD, method->method_holder());
a61af66fc99e Initial load
duke
parents:
diff changeset
695 int slot = method->method_idnum();
a61af66fc99e Initial load
duke
parents:
diff changeset
696
a61af66fc99e Initial load
duke
parents:
diff changeset
697 symbolHandle signature (THREAD, method->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
698 int parameter_count = ArgumentCount(signature).size();
a61af66fc99e Initial load
duke
parents:
diff changeset
699 oop return_type_oop = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
700 objArrayHandle parameter_types = get_parameter_types(method, parameter_count, &return_type_oop, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
701 if (parameter_types.is_null() || return_type_oop == NULL) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
702
a61af66fc99e Initial load
duke
parents:
diff changeset
703 Handle return_type(THREAD, return_type_oop);
a61af66fc99e Initial load
duke
parents:
diff changeset
704
a61af66fc99e Initial load
duke
parents:
diff changeset
705 objArrayHandle exception_types = get_exception_types(method, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
706
a61af66fc99e Initial load
duke
parents:
diff changeset
707 if (exception_types.is_null()) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
708
a61af66fc99e Initial load
duke
parents:
diff changeset
709 symbolHandle method_name(THREAD, method->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
710 Handle name;
a61af66fc99e Initial load
duke
parents:
diff changeset
711 if (intern_name) {
a61af66fc99e Initial load
duke
parents:
diff changeset
712 // intern_name is only true with UseNewReflection
a61af66fc99e Initial load
duke
parents:
diff changeset
713 oop name_oop = StringTable::intern(method_name(), CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
714 name = Handle(THREAD, name_oop);
a61af66fc99e Initial load
duke
parents:
diff changeset
715 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
716 name = java_lang_String::create_from_symbol(method_name, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
717 }
a61af66fc99e Initial load
duke
parents:
diff changeset
718 if (name.is_null()) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
719
a61af66fc99e Initial load
duke
parents:
diff changeset
720 int modifiers = method->access_flags().as_int() & JVM_RECOGNIZED_METHOD_MODIFIERS;
a61af66fc99e Initial load
duke
parents:
diff changeset
721
a61af66fc99e Initial load
duke
parents:
diff changeset
722 Handle mh = java_lang_reflect_Method::create(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
723
a61af66fc99e Initial load
duke
parents:
diff changeset
724 java_lang_reflect_Method::set_clazz(mh(), holder->java_mirror());
a61af66fc99e Initial load
duke
parents:
diff changeset
725 java_lang_reflect_Method::set_slot(mh(), slot);
a61af66fc99e Initial load
duke
parents:
diff changeset
726 java_lang_reflect_Method::set_name(mh(), name());
a61af66fc99e Initial load
duke
parents:
diff changeset
727 java_lang_reflect_Method::set_return_type(mh(), return_type());
a61af66fc99e Initial load
duke
parents:
diff changeset
728 java_lang_reflect_Method::set_parameter_types(mh(), parameter_types());
a61af66fc99e Initial load
duke
parents:
diff changeset
729 java_lang_reflect_Method::set_exception_types(mh(), exception_types());
a61af66fc99e Initial load
duke
parents:
diff changeset
730 java_lang_reflect_Method::set_modifiers(mh(), modifiers);
a61af66fc99e Initial load
duke
parents:
diff changeset
731 java_lang_reflect_Method::set_override(mh(), false);
a61af66fc99e Initial load
duke
parents:
diff changeset
732 if (java_lang_reflect_Method::has_signature_field() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
733 method->generic_signature() != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
734 symbolHandle gs(THREAD, method->generic_signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
735 Handle sig = java_lang_String::create_from_symbol(gs, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
736 java_lang_reflect_Method::set_signature(mh(), sig());
a61af66fc99e Initial load
duke
parents:
diff changeset
737 }
a61af66fc99e Initial load
duke
parents:
diff changeset
738 if (java_lang_reflect_Method::has_annotations_field()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
739 java_lang_reflect_Method::set_annotations(mh(), method->annotations());
a61af66fc99e Initial load
duke
parents:
diff changeset
740 }
a61af66fc99e Initial load
duke
parents:
diff changeset
741 if (java_lang_reflect_Method::has_parameter_annotations_field()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
742 java_lang_reflect_Method::set_parameter_annotations(mh(), method->parameter_annotations());
a61af66fc99e Initial load
duke
parents:
diff changeset
743 }
a61af66fc99e Initial load
duke
parents:
diff changeset
744 if (java_lang_reflect_Method::has_annotation_default_field()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
745 java_lang_reflect_Method::set_annotation_default(mh(), method->annotation_default());
a61af66fc99e Initial load
duke
parents:
diff changeset
746 }
a61af66fc99e Initial load
duke
parents:
diff changeset
747 return mh();
a61af66fc99e Initial load
duke
parents:
diff changeset
748 }
a61af66fc99e Initial load
duke
parents:
diff changeset
749
a61af66fc99e Initial load
duke
parents:
diff changeset
750
a61af66fc99e Initial load
duke
parents:
diff changeset
751 oop Reflection::new_constructor(methodHandle method, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
752 assert(method()->is_initializer(), "should call new_method instead");
a61af66fc99e Initial load
duke
parents:
diff changeset
753
a61af66fc99e Initial load
duke
parents:
diff changeset
754 instanceKlassHandle holder (THREAD, method->method_holder());
a61af66fc99e Initial load
duke
parents:
diff changeset
755 int slot = method->method_idnum();
a61af66fc99e Initial load
duke
parents:
diff changeset
756
a61af66fc99e Initial load
duke
parents:
diff changeset
757 symbolHandle signature (THREAD, method->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
758 int parameter_count = ArgumentCount(signature).size();
a61af66fc99e Initial load
duke
parents:
diff changeset
759 objArrayHandle parameter_types = get_parameter_types(method, parameter_count, NULL, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
760 if (parameter_types.is_null()) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
761
a61af66fc99e Initial load
duke
parents:
diff changeset
762 objArrayHandle exception_types = get_exception_types(method, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
763 if (exception_types.is_null()) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
764
a61af66fc99e Initial load
duke
parents:
diff changeset
765 int modifiers = method->access_flags().as_int() & JVM_RECOGNIZED_METHOD_MODIFIERS;
a61af66fc99e Initial load
duke
parents:
diff changeset
766
a61af66fc99e Initial load
duke
parents:
diff changeset
767 Handle ch = java_lang_reflect_Constructor::create(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
768
a61af66fc99e Initial load
duke
parents:
diff changeset
769 java_lang_reflect_Constructor::set_clazz(ch(), holder->java_mirror());
a61af66fc99e Initial load
duke
parents:
diff changeset
770 java_lang_reflect_Constructor::set_slot(ch(), slot);
a61af66fc99e Initial load
duke
parents:
diff changeset
771 java_lang_reflect_Constructor::set_parameter_types(ch(), parameter_types());
a61af66fc99e Initial load
duke
parents:
diff changeset
772 java_lang_reflect_Constructor::set_exception_types(ch(), exception_types());
a61af66fc99e Initial load
duke
parents:
diff changeset
773 java_lang_reflect_Constructor::set_modifiers(ch(), modifiers);
a61af66fc99e Initial load
duke
parents:
diff changeset
774 java_lang_reflect_Constructor::set_override(ch(), false);
a61af66fc99e Initial load
duke
parents:
diff changeset
775 if (java_lang_reflect_Constructor::has_signature_field() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
776 method->generic_signature() != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
777 symbolHandle gs(THREAD, method->generic_signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
778 Handle sig = java_lang_String::create_from_symbol(gs, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
779 java_lang_reflect_Constructor::set_signature(ch(), sig());
a61af66fc99e Initial load
duke
parents:
diff changeset
780 }
a61af66fc99e Initial load
duke
parents:
diff changeset
781 if (java_lang_reflect_Constructor::has_annotations_field()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
782 java_lang_reflect_Constructor::set_annotations(ch(), method->annotations());
a61af66fc99e Initial load
duke
parents:
diff changeset
783 }
a61af66fc99e Initial load
duke
parents:
diff changeset
784 if (java_lang_reflect_Constructor::has_parameter_annotations_field()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
785 java_lang_reflect_Constructor::set_parameter_annotations(ch(), method->parameter_annotations());
a61af66fc99e Initial load
duke
parents:
diff changeset
786 }
a61af66fc99e Initial load
duke
parents:
diff changeset
787 return ch();
a61af66fc99e Initial load
duke
parents:
diff changeset
788 }
a61af66fc99e Initial load
duke
parents:
diff changeset
789
a61af66fc99e Initial load
duke
parents:
diff changeset
790
a61af66fc99e Initial load
duke
parents:
diff changeset
791 oop Reflection::new_field(fieldDescriptor* fd, bool intern_name, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
792 symbolHandle field_name(THREAD, fd->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
793 Handle name;
a61af66fc99e Initial load
duke
parents:
diff changeset
794 if (intern_name) {
a61af66fc99e Initial load
duke
parents:
diff changeset
795 // intern_name is only true with UseNewReflection
a61af66fc99e Initial load
duke
parents:
diff changeset
796 oop name_oop = StringTable::intern(field_name(), CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
797 name = Handle(THREAD, name_oop);
a61af66fc99e Initial load
duke
parents:
diff changeset
798 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
799 name = java_lang_String::create_from_symbol(field_name, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
800 }
a61af66fc99e Initial load
duke
parents:
diff changeset
801 symbolHandle signature (THREAD, fd->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
802 KlassHandle holder (THREAD, fd->field_holder());
a61af66fc99e Initial load
duke
parents:
diff changeset
803 Handle type = new_type(signature, holder, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
804 Handle rh = java_lang_reflect_Field::create(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
805
a61af66fc99e Initial load
duke
parents:
diff changeset
806 java_lang_reflect_Field::set_clazz(rh(), Klass::cast(fd->field_holder())->java_mirror());
a61af66fc99e Initial load
duke
parents:
diff changeset
807 java_lang_reflect_Field::set_slot(rh(), fd->index());
a61af66fc99e Initial load
duke
parents:
diff changeset
808 java_lang_reflect_Field::set_name(rh(), name());
a61af66fc99e Initial load
duke
parents:
diff changeset
809 java_lang_reflect_Field::set_type(rh(), type());
a61af66fc99e Initial load
duke
parents:
diff changeset
810 // Note the ACC_ANNOTATION bit, which is a per-class access flag, is never set here.
a61af66fc99e Initial load
duke
parents:
diff changeset
811 java_lang_reflect_Field::set_modifiers(rh(), fd->access_flags().as_int() & JVM_RECOGNIZED_FIELD_MODIFIERS);
a61af66fc99e Initial load
duke
parents:
diff changeset
812 java_lang_reflect_Field::set_override(rh(), false);
a61af66fc99e Initial load
duke
parents:
diff changeset
813 if (java_lang_reflect_Field::has_signature_field() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
814 fd->generic_signature() != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
815 symbolHandle gs(THREAD, fd->generic_signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
816 Handle sig = java_lang_String::create_from_symbol(gs, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
817 java_lang_reflect_Field::set_signature(rh(), sig());
a61af66fc99e Initial load
duke
parents:
diff changeset
818 }
a61af66fc99e Initial load
duke
parents:
diff changeset
819 if (java_lang_reflect_Field::has_annotations_field()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
820 java_lang_reflect_Field::set_annotations(rh(), fd->annotations());
a61af66fc99e Initial load
duke
parents:
diff changeset
821 }
a61af66fc99e Initial load
duke
parents:
diff changeset
822 return rh();
a61af66fc99e Initial load
duke
parents:
diff changeset
823 }
a61af66fc99e Initial load
duke
parents:
diff changeset
824
a61af66fc99e Initial load
duke
parents:
diff changeset
825
a61af66fc99e Initial load
duke
parents:
diff changeset
826 //---------------------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
827 //
a61af66fc99e Initial load
duke
parents:
diff changeset
828 // Supporting routines for old native code-based reflection (pre-JDK 1.4).
a61af66fc99e Initial load
duke
parents:
diff changeset
829 //
a61af66fc99e Initial load
duke
parents:
diff changeset
830 // See reflection.hpp for details.
a61af66fc99e Initial load
duke
parents:
diff changeset
831 //
a61af66fc99e Initial load
duke
parents:
diff changeset
832 //---------------------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
833
a61af66fc99e Initial load
duke
parents:
diff changeset
834 #ifdef SUPPORT_OLD_REFLECTION
a61af66fc99e Initial load
duke
parents:
diff changeset
835
a61af66fc99e Initial load
duke
parents:
diff changeset
836 methodHandle Reflection::resolve_interface_call(instanceKlassHandle klass, methodHandle method,
a61af66fc99e Initial load
duke
parents:
diff changeset
837 KlassHandle recv_klass, Handle receiver, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
838 assert(!method.is_null() , "method should not be null");
a61af66fc99e Initial load
duke
parents:
diff changeset
839
a61af66fc99e Initial load
duke
parents:
diff changeset
840 CallInfo info;
a61af66fc99e Initial load
duke
parents:
diff changeset
841 symbolHandle signature (THREAD, method->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
842 symbolHandle name (THREAD, method->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
843 LinkResolver::resolve_interface_call(info, receiver, recv_klass, klass,
a61af66fc99e Initial load
duke
parents:
diff changeset
844 name, signature,
a61af66fc99e Initial load
duke
parents:
diff changeset
845 KlassHandle(), false, true,
a61af66fc99e Initial load
duke
parents:
diff changeset
846 CHECK_(methodHandle()));
a61af66fc99e Initial load
duke
parents:
diff changeset
847 return info.selected_method();
a61af66fc99e Initial load
duke
parents:
diff changeset
848 }
a61af66fc99e Initial load
duke
parents:
diff changeset
849
a61af66fc99e Initial load
duke
parents:
diff changeset
850
a61af66fc99e Initial load
duke
parents:
diff changeset
851 oop Reflection::invoke(instanceKlassHandle klass, methodHandle reflected_method,
a61af66fc99e Initial load
duke
parents:
diff changeset
852 Handle receiver, bool override, objArrayHandle ptypes,
a61af66fc99e Initial load
duke
parents:
diff changeset
853 BasicType rtype, objArrayHandle args, bool is_method_invoke, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
854 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
855
a61af66fc99e Initial load
duke
parents:
diff changeset
856 methodHandle method; // actual method to invoke
a61af66fc99e Initial load
duke
parents:
diff changeset
857 KlassHandle target_klass; // target klass, receiver's klass for non-static
a61af66fc99e Initial load
duke
parents:
diff changeset
858
a61af66fc99e Initial load
duke
parents:
diff changeset
859 // Ensure klass is initialized
a61af66fc99e Initial load
duke
parents:
diff changeset
860 klass->initialize(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
861
a61af66fc99e Initial load
duke
parents:
diff changeset
862 bool is_static = reflected_method->is_static();
a61af66fc99e Initial load
duke
parents:
diff changeset
863 if (is_static) {
a61af66fc99e Initial load
duke
parents:
diff changeset
864 // ignore receiver argument
a61af66fc99e Initial load
duke
parents:
diff changeset
865 method = reflected_method;
a61af66fc99e Initial load
duke
parents:
diff changeset
866 target_klass = klass;
a61af66fc99e Initial load
duke
parents:
diff changeset
867 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
868 // check for null receiver
a61af66fc99e Initial load
duke
parents:
diff changeset
869 if (receiver.is_null()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
870 THROW_0(vmSymbols::java_lang_NullPointerException());
a61af66fc99e Initial load
duke
parents:
diff changeset
871 }
a61af66fc99e Initial load
duke
parents:
diff changeset
872 // Check class of receiver against class declaring method
a61af66fc99e Initial load
duke
parents:
diff changeset
873 if (!receiver->is_a(klass())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
874 THROW_MSG_0(vmSymbols::java_lang_IllegalArgumentException(), "object is not an instance of declaring class");
a61af66fc99e Initial load
duke
parents:
diff changeset
875 }
a61af66fc99e Initial load
duke
parents:
diff changeset
876 // target klass is receiver's klass
a61af66fc99e Initial load
duke
parents:
diff changeset
877 target_klass = KlassHandle(THREAD, receiver->klass());
a61af66fc99e Initial load
duke
parents:
diff changeset
878 // no need to resolve if method is private or <init>
a61af66fc99e Initial load
duke
parents:
diff changeset
879 if (reflected_method->is_private() || reflected_method->name() == vmSymbols::object_initializer_name()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
880 method = reflected_method;
a61af66fc99e Initial load
duke
parents:
diff changeset
881 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
882 // resolve based on the receiver
a61af66fc99e Initial load
duke
parents:
diff changeset
883 if (instanceKlass::cast(reflected_method->method_holder())->is_interface()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
884 // resolve interface call
a61af66fc99e Initial load
duke
parents:
diff changeset
885 if (ReflectionWrapResolutionErrors) {
a61af66fc99e Initial load
duke
parents:
diff changeset
886 // new default: 6531596
a61af66fc99e Initial load
duke
parents:
diff changeset
887 // Match resolution errors with those thrown due to reflection inlining
a61af66fc99e Initial load
duke
parents:
diff changeset
888 // Linktime resolution & IllegalAccessCheck already done by Class.getMethod()
a61af66fc99e Initial load
duke
parents:
diff changeset
889 method = resolve_interface_call(klass, reflected_method, target_klass, receiver, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
890 if (HAS_PENDING_EXCEPTION) {
a61af66fc99e Initial load
duke
parents:
diff changeset
891 // Method resolution threw an exception; wrap it in an InvocationTargetException
a61af66fc99e Initial load
duke
parents:
diff changeset
892 oop resolution_exception = PENDING_EXCEPTION;
a61af66fc99e Initial load
duke
parents:
diff changeset
893 CLEAR_PENDING_EXCEPTION;
a61af66fc99e Initial load
duke
parents:
diff changeset
894 JavaCallArguments args(Handle(THREAD, resolution_exception));
a61af66fc99e Initial load
duke
parents:
diff changeset
895 THROW_ARG_0(vmSymbolHandles::java_lang_reflect_InvocationTargetException(),
a61af66fc99e Initial load
duke
parents:
diff changeset
896 vmSymbolHandles::throwable_void_signature(),
a61af66fc99e Initial load
duke
parents:
diff changeset
897 &args);
a61af66fc99e Initial load
duke
parents:
diff changeset
898 }
a61af66fc99e Initial load
duke
parents:
diff changeset
899 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
900 method = resolve_interface_call(klass, reflected_method, target_klass, receiver, CHECK_(NULL));
a61af66fc99e Initial load
duke
parents:
diff changeset
901 }
a61af66fc99e Initial load
duke
parents:
diff changeset
902 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
903 // if the method can be overridden, we resolve using the vtable index.
a61af66fc99e Initial load
duke
parents:
diff changeset
904 int index = reflected_method->vtable_index();
a61af66fc99e Initial load
duke
parents:
diff changeset
905 method = reflected_method;
a61af66fc99e Initial load
duke
parents:
diff changeset
906 if (index != methodOopDesc::nonvirtual_vtable_index) {
a61af66fc99e Initial load
duke
parents:
diff changeset
907 // target_klass might be an arrayKlassOop but all vtables start at
a61af66fc99e Initial load
duke
parents:
diff changeset
908 // the same place. The cast is to avoid virtual call and assertion.
a61af66fc99e Initial load
duke
parents:
diff changeset
909 instanceKlass* inst = (instanceKlass*)target_klass()->klass_part();
a61af66fc99e Initial load
duke
parents:
diff changeset
910 method = methodHandle(THREAD, inst->method_at_vtable(index));
a61af66fc99e Initial load
duke
parents:
diff changeset
911 }
a61af66fc99e Initial load
duke
parents:
diff changeset
912 if (!method.is_null()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
913 // Check for abstract methods as well
a61af66fc99e Initial load
duke
parents:
diff changeset
914 if (method->is_abstract()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
915 // new default: 6531596
a61af66fc99e Initial load
duke
parents:
diff changeset
916 if (ReflectionWrapResolutionErrors) {
a61af66fc99e Initial load
duke
parents:
diff changeset
917 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
918 Handle h_origexception = Exceptions::new_exception(THREAD,
a61af66fc99e Initial load
duke
parents:
diff changeset
919 vmSymbols::java_lang_AbstractMethodError(),
a61af66fc99e Initial load
duke
parents:
diff changeset
920 methodOopDesc::name_and_sig_as_C_string(Klass::cast(target_klass()),
a61af66fc99e Initial load
duke
parents:
diff changeset
921 method->name(),
a61af66fc99e Initial load
duke
parents:
diff changeset
922 method->signature()));
a61af66fc99e Initial load
duke
parents:
diff changeset
923 JavaCallArguments args(h_origexception);
a61af66fc99e Initial load
duke
parents:
diff changeset
924 THROW_ARG_0(vmSymbolHandles::java_lang_reflect_InvocationTargetException(),
a61af66fc99e Initial load
duke
parents:
diff changeset
925 vmSymbolHandles::throwable_void_signature(),
a61af66fc99e Initial load
duke
parents:
diff changeset
926 &args);
a61af66fc99e Initial load
duke
parents:
diff changeset
927 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
928 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
929 THROW_MSG_0(vmSymbols::java_lang_AbstractMethodError(),
a61af66fc99e Initial load
duke
parents:
diff changeset
930 methodOopDesc::name_and_sig_as_C_string(Klass::cast(target_klass()),
a61af66fc99e Initial load
duke
parents:
diff changeset
931 method->name(),
a61af66fc99e Initial load
duke
parents:
diff changeset
932 method->signature()));
a61af66fc99e Initial load
duke
parents:
diff changeset
933 }
a61af66fc99e Initial load
duke
parents:
diff changeset
934 }
a61af66fc99e Initial load
duke
parents:
diff changeset
935 }
a61af66fc99e Initial load
duke
parents:
diff changeset
936 }
a61af66fc99e Initial load
duke
parents:
diff changeset
937 }
a61af66fc99e Initial load
duke
parents:
diff changeset
938 }
a61af66fc99e Initial load
duke
parents:
diff changeset
939
a61af66fc99e Initial load
duke
parents:
diff changeset
940 // I believe this is a ShouldNotGetHere case which requires
a61af66fc99e Initial load
duke
parents:
diff changeset
941 // an internal vtable bug. If you ever get this please let Karen know.
a61af66fc99e Initial load
duke
parents:
diff changeset
942 if (method.is_null()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
943 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
944 THROW_MSG_0(vmSymbols::java_lang_NoSuchMethodError(),
a61af66fc99e Initial load
duke
parents:
diff changeset
945 methodOopDesc::name_and_sig_as_C_string(Klass::cast(klass()),
a61af66fc99e Initial load
duke
parents:
diff changeset
946 reflected_method->name(),
a61af66fc99e Initial load
duke
parents:
diff changeset
947 reflected_method->signature()));
a61af66fc99e Initial load
duke
parents:
diff changeset
948 }
a61af66fc99e Initial load
duke
parents:
diff changeset
949
a61af66fc99e Initial load
duke
parents:
diff changeset
950 // In the JDK 1.4 reflection implementation, the security check is
a61af66fc99e Initial load
duke
parents:
diff changeset
951 // done at the Java level
a61af66fc99e Initial load
duke
parents:
diff changeset
952 if (!(JDK_Version::is_gte_jdk14x_version() && UseNewReflection)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
953
a61af66fc99e Initial load
duke
parents:
diff changeset
954 // Access checking (unless overridden by Method)
a61af66fc99e Initial load
duke
parents:
diff changeset
955 if (!override) {
a61af66fc99e Initial load
duke
parents:
diff changeset
956 if (!(klass->is_public() && reflected_method->is_public())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
957 bool access = Reflection::reflect_check_access(klass(), reflected_method->access_flags(), target_klass(), is_method_invoke, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
958 if (!access) {
a61af66fc99e Initial load
duke
parents:
diff changeset
959 return NULL; // exception
a61af66fc99e Initial load
duke
parents:
diff changeset
960 }
a61af66fc99e Initial load
duke
parents:
diff changeset
961 }
a61af66fc99e Initial load
duke
parents:
diff changeset
962 }
a61af66fc99e Initial load
duke
parents:
diff changeset
963
a61af66fc99e Initial load
duke
parents:
diff changeset
964 } // !(Universe::is_gte_jdk14x_version() && UseNewReflection)
a61af66fc99e Initial load
duke
parents:
diff changeset
965
a61af66fc99e Initial load
duke
parents:
diff changeset
966 assert(ptypes->is_objArray(), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
967 int args_len = args.is_null() ? 0 : args->length();
a61af66fc99e Initial load
duke
parents:
diff changeset
968 // Check number of arguments
a61af66fc99e Initial load
duke
parents:
diff changeset
969 if (ptypes->length() != args_len) {
a61af66fc99e Initial load
duke
parents:
diff changeset
970 THROW_MSG_0(vmSymbols::java_lang_IllegalArgumentException(), "wrong number of arguments");
a61af66fc99e Initial load
duke
parents:
diff changeset
971 }
a61af66fc99e Initial load
duke
parents:
diff changeset
972
a61af66fc99e Initial load
duke
parents:
diff changeset
973 // Create object to contain parameters for the JavaCall
a61af66fc99e Initial load
duke
parents:
diff changeset
974 JavaCallArguments java_args(method->size_of_parameters());
a61af66fc99e Initial load
duke
parents:
diff changeset
975
a61af66fc99e Initial load
duke
parents:
diff changeset
976 if (!is_static) {
a61af66fc99e Initial load
duke
parents:
diff changeset
977 java_args.push_oop(receiver);
a61af66fc99e Initial load
duke
parents:
diff changeset
978 }
a61af66fc99e Initial load
duke
parents:
diff changeset
979
a61af66fc99e Initial load
duke
parents:
diff changeset
980 for (int i = 0; i < args_len; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
981 oop type_mirror = ptypes->obj_at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
982 oop arg = args->obj_at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
983 if (java_lang_Class::is_primitive(type_mirror)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
984 jvalue value;
a61af66fc99e Initial load
duke
parents:
diff changeset
985 BasicType ptype = basic_type_mirror_to_basic_type(type_mirror, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
986 BasicType atype = unbox_for_primitive(arg, &value, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
987 if (ptype != atype) {
a61af66fc99e Initial load
duke
parents:
diff changeset
988 widen(&value, atype, ptype, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
989 }
a61af66fc99e Initial load
duke
parents:
diff changeset
990 switch (ptype) {
a61af66fc99e Initial load
duke
parents:
diff changeset
991 case T_BOOLEAN: java_args.push_int(value.z); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
992 case T_CHAR: java_args.push_int(value.c); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
993 case T_BYTE: java_args.push_int(value.b); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
994 case T_SHORT: java_args.push_int(value.s); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
995 case T_INT: java_args.push_int(value.i); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
996 case T_LONG: java_args.push_long(value.j); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
997 case T_FLOAT: java_args.push_float(value.f); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
998 case T_DOUBLE: java_args.push_double(value.d); break;
a61af66fc99e Initial load
duke
parents:
diff changeset
999 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
1000 THROW_MSG_0(vmSymbols::java_lang_IllegalArgumentException(), "argument type mismatch");
a61af66fc99e Initial load
duke
parents:
diff changeset
1001 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1002 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1003 if (arg != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1004 klassOop k = java_lang_Class::as_klassOop(type_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1005 if (!arg->is_a(k)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1006 THROW_MSG_0(vmSymbols::java_lang_IllegalArgumentException(), "argument type mismatch");
a61af66fc99e Initial load
duke
parents:
diff changeset
1007 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1008 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1009 Handle arg_handle(THREAD, arg); // Create handle for argument
a61af66fc99e Initial load
duke
parents:
diff changeset
1010 java_args.push_oop(arg_handle); // Push handle
a61af66fc99e Initial load
duke
parents:
diff changeset
1011 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1012 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1013
a61af66fc99e Initial load
duke
parents:
diff changeset
1014 assert(java_args.size_of_parameters() == method->size_of_parameters(), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
1015
a61af66fc99e Initial load
duke
parents:
diff changeset
1016 // All oops (including receiver) is passed in as Handles. An potential oop is returned as an
a61af66fc99e Initial load
duke
parents:
diff changeset
1017 // oop (i.e., NOT as an handle)
a61af66fc99e Initial load
duke
parents:
diff changeset
1018 JavaValue result(rtype);
a61af66fc99e Initial load
duke
parents:
diff changeset
1019 JavaCalls::call(&result, method, &java_args, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1020
a61af66fc99e Initial load
duke
parents:
diff changeset
1021 if (HAS_PENDING_EXCEPTION) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1022 // Method threw an exception; wrap it in an InvocationTargetException
a61af66fc99e Initial load
duke
parents:
diff changeset
1023 oop target_exception = PENDING_EXCEPTION;
a61af66fc99e Initial load
duke
parents:
diff changeset
1024 CLEAR_PENDING_EXCEPTION;
a61af66fc99e Initial load
duke
parents:
diff changeset
1025 JavaCallArguments args(Handle(THREAD, target_exception));
a61af66fc99e Initial load
duke
parents:
diff changeset
1026 THROW_ARG_0(vmSymbolHandles::java_lang_reflect_InvocationTargetException(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1027 vmSymbolHandles::throwable_void_signature(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1028 &args);
a61af66fc99e Initial load
duke
parents:
diff changeset
1029 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1030 if (rtype == T_BOOLEAN || rtype == T_BYTE || rtype == T_CHAR || rtype == T_SHORT)
a61af66fc99e Initial load
duke
parents:
diff changeset
1031 narrow((jvalue*) result.get_value_addr(), rtype, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1032 return box((jvalue*) result.get_value_addr(), rtype, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1033 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1034 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1035
a61af66fc99e Initial load
duke
parents:
diff changeset
1036
a61af66fc99e Initial load
duke
parents:
diff changeset
1037 void Reflection::narrow(jvalue* value, BasicType narrow_type, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1038 switch (narrow_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1039 case T_BOOLEAN:
a61af66fc99e Initial load
duke
parents:
diff changeset
1040 value->z = (jboolean) value->i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1041 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1042 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
1043 value->b = (jbyte) value->i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1044 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1045 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
1046 value->c = (jchar) value->i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1047 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1048 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1049 value->s = (jshort) value->i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1050 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1051 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
1052 break; // fail
a61af66fc99e Initial load
duke
parents:
diff changeset
1053 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1054 THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "argument type mismatch");
a61af66fc99e Initial load
duke
parents:
diff changeset
1055 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1056
a61af66fc99e Initial load
duke
parents:
diff changeset
1057
a61af66fc99e Initial load
duke
parents:
diff changeset
1058 BasicType Reflection::basic_type_mirror_to_basic_type(oop basic_type_mirror, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1059 assert(java_lang_Class::is_primitive(basic_type_mirror), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
1060 return java_lang_Class::primitive_type(basic_type_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1061 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1062
a61af66fc99e Initial load
duke
parents:
diff changeset
1063
a61af66fc99e Initial load
duke
parents:
diff changeset
1064 bool Reflection::match_parameter_types(methodHandle method, objArrayHandle types, int parameter_count, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1065 int types_len = types.is_null() ? 0 : types->length();
a61af66fc99e Initial load
duke
parents:
diff changeset
1066 if (types_len != parameter_count) return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
1067 if (parameter_count > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1068 objArrayHandle method_types = get_parameter_types(method, parameter_count, NULL, CHECK_false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1069 for (int index = 0; index < parameter_count; index++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1070 if (types->obj_at(index) != method_types->obj_at(index)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1071 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
1072 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1073 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1074 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1075 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
1076 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1077
a61af66fc99e Initial load
duke
parents:
diff changeset
1078
a61af66fc99e Initial load
duke
parents:
diff changeset
1079 oop Reflection::new_field(FieldStream* st, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1080 symbolHandle field_name(THREAD, st->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
1081 Handle name = java_lang_String::create_from_symbol(field_name, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1082 symbolHandle signature(THREAD, st->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
1083 Handle type = new_type(signature, st->klass(), CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1084 Handle rh = java_lang_reflect_Field::create(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1085 oop result = rh();
a61af66fc99e Initial load
duke
parents:
diff changeset
1086
a61af66fc99e Initial load
duke
parents:
diff changeset
1087 java_lang_reflect_Field::set_clazz(result, st->klass()->java_mirror());
a61af66fc99e Initial load
duke
parents:
diff changeset
1088 java_lang_reflect_Field::set_slot(result, st->index());
a61af66fc99e Initial load
duke
parents:
diff changeset
1089 java_lang_reflect_Field::set_name(result, name());
a61af66fc99e Initial load
duke
parents:
diff changeset
1090 java_lang_reflect_Field::set_type(result, type());
a61af66fc99e Initial load
duke
parents:
diff changeset
1091 // Note the ACC_ANNOTATION bit, which is a per-class access flag, is never set here.
a61af66fc99e Initial load
duke
parents:
diff changeset
1092 java_lang_reflect_Field::set_modifiers(result, st->access_flags().as_int() & JVM_RECOGNIZED_FIELD_MODIFIERS);
a61af66fc99e Initial load
duke
parents:
diff changeset
1093 java_lang_reflect_Field::set_override(result, false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1094 return result;
a61af66fc99e Initial load
duke
parents:
diff changeset
1095 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1096
a61af66fc99e Initial load
duke
parents:
diff changeset
1097
a61af66fc99e Initial load
duke
parents:
diff changeset
1098 bool Reflection::resolve_field(Handle field_mirror, Handle& receiver, fieldDescriptor* fd, bool check_final, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1099 if (field_mirror.is_null()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1100 THROW_(vmSymbols::java_lang_NullPointerException(), false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1101 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1102
a61af66fc99e Initial load
duke
parents:
diff changeset
1103 instanceKlassHandle klass (THREAD, java_lang_Class::as_klassOop(java_lang_reflect_Field::clazz(field_mirror())));
a61af66fc99e Initial load
duke
parents:
diff changeset
1104 int slot = java_lang_reflect_Field::slot(field_mirror());
a61af66fc99e Initial load
duke
parents:
diff changeset
1105
a61af66fc99e Initial load
duke
parents:
diff changeset
1106 // Ensure klass is initialized
a61af66fc99e Initial load
duke
parents:
diff changeset
1107 klass->initialize(CHECK_false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1108 fd->initialize(klass(), slot);
a61af66fc99e Initial load
duke
parents:
diff changeset
1109
a61af66fc99e Initial load
duke
parents:
diff changeset
1110 bool is_static = fd->is_static();
a61af66fc99e Initial load
duke
parents:
diff changeset
1111 KlassHandle receiver_klass;
a61af66fc99e Initial load
duke
parents:
diff changeset
1112
a61af66fc99e Initial load
duke
parents:
diff changeset
1113 if (is_static) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1114 receiver = KlassHandle(THREAD, klass());
a61af66fc99e Initial load
duke
parents:
diff changeset
1115 receiver_klass = klass;
a61af66fc99e Initial load
duke
parents:
diff changeset
1116 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1117 // Check object is a non-null instance of declaring class
a61af66fc99e Initial load
duke
parents:
diff changeset
1118 if (receiver.is_null()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1119 THROW_(vmSymbols::java_lang_NullPointerException(), false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1120 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1121 if (!receiver->is_a(klass())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1122 THROW_MSG_(vmSymbols::java_lang_IllegalArgumentException(), "object is not an instance of declaring class", false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1123 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1124 receiver_klass = KlassHandle(THREAD, receiver->klass());
a61af66fc99e Initial load
duke
parents:
diff changeset
1125 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1126
a61af66fc99e Initial load
duke
parents:
diff changeset
1127 // Access checking (unless overridden by Field)
a61af66fc99e Initial load
duke
parents:
diff changeset
1128 if (!java_lang_reflect_Field::override(field_mirror())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1129 if (!(klass->is_public() && fd->is_public())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1130 bool access_check = reflect_check_access(klass(), fd->access_flags(), receiver_klass(), false, CHECK_false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1131 if (!access_check) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1132 return false; // exception
a61af66fc99e Initial load
duke
parents:
diff changeset
1133 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1134 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1135 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1136
a61af66fc99e Initial load
duke
parents:
diff changeset
1137 if (check_final && fd->is_final()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1138 // In 1.3 we always throw an error when attempting to set a final field.
a61af66fc99e Initial load
duke
parents:
diff changeset
1139 // In 1.2.x, this was allowed in the override bit was set by calling Field.setAccessible(true).
a61af66fc99e Initial load
duke
parents:
diff changeset
1140 // We currently maintain backwards compatibility. See bug 4250960.
a61af66fc99e Initial load
duke
parents:
diff changeset
1141 bool strict_final_check = !JDK_Version::is_jdk12x_version();
a61af66fc99e Initial load
duke
parents:
diff changeset
1142 if (strict_final_check || !java_lang_reflect_Field::override(field_mirror())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1143 THROW_MSG_(vmSymbols::java_lang_IllegalAccessException(), "field is final", false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1144 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1145 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1146 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
1147 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1148
a61af66fc99e Initial load
duke
parents:
diff changeset
1149
a61af66fc99e Initial load
duke
parents:
diff changeset
1150 BasicType Reflection::field_get(jvalue* value, fieldDescriptor* fd, Handle receiver) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1151 BasicType field_type = fd->field_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
1152 int offset = fd->offset();
a61af66fc99e Initial load
duke
parents:
diff changeset
1153 switch (field_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1154 case T_BOOLEAN:
a61af66fc99e Initial load
duke
parents:
diff changeset
1155 value->z = receiver->bool_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1156 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1157 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
1158 value->c = receiver->char_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1159 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1160 case T_FLOAT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1161 value->f = receiver->float_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1162 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1163 case T_DOUBLE:
a61af66fc99e Initial load
duke
parents:
diff changeset
1164 value->d = receiver->double_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1165 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1166 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
1167 value->b = receiver->byte_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1168 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1169 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1170 value->s = receiver->short_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1171 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1172 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1173 value->i = receiver->int_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1174 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1175 case T_LONG:
a61af66fc99e Initial load
duke
parents:
diff changeset
1176 value->j = receiver->long_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1177 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1178 case T_OBJECT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1179 case T_ARRAY:
a61af66fc99e Initial load
duke
parents:
diff changeset
1180 value->l = (jobject) receiver->obj_field(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
1181 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1182 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
1183 return T_ILLEGAL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1184 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1185 return field_type;
a61af66fc99e Initial load
duke
parents:
diff changeset
1186 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1187
a61af66fc99e Initial load
duke
parents:
diff changeset
1188
a61af66fc99e Initial load
duke
parents:
diff changeset
1189 void Reflection::field_set(jvalue* value, fieldDescriptor* fd, Handle receiver, BasicType value_type, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1190 BasicType field_type = fd->field_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
1191 if (field_type != value_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1192 widen(value, value_type, field_type, CHECK);
a61af66fc99e Initial load
duke
parents:
diff changeset
1193 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1194
a61af66fc99e Initial load
duke
parents:
diff changeset
1195 int offset = fd->offset();
a61af66fc99e Initial load
duke
parents:
diff changeset
1196 switch (field_type) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1197 case T_BOOLEAN:
a61af66fc99e Initial load
duke
parents:
diff changeset
1198 receiver->bool_field_put(offset, value->z);
a61af66fc99e Initial load
duke
parents:
diff changeset
1199 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1200 case T_CHAR:
a61af66fc99e Initial load
duke
parents:
diff changeset
1201 receiver->char_field_put(offset, value->c);
a61af66fc99e Initial load
duke
parents:
diff changeset
1202 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1203 case T_FLOAT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1204 receiver->float_field_put(offset, value->f);
a61af66fc99e Initial load
duke
parents:
diff changeset
1205 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1206 case T_DOUBLE:
a61af66fc99e Initial load
duke
parents:
diff changeset
1207 receiver->double_field_put(offset, value->d);
a61af66fc99e Initial load
duke
parents:
diff changeset
1208 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1209 case T_BYTE:
a61af66fc99e Initial load
duke
parents:
diff changeset
1210 receiver->byte_field_put(offset, value->b);
a61af66fc99e Initial load
duke
parents:
diff changeset
1211 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1212 case T_SHORT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1213 receiver->short_field_put(offset, value->s);
a61af66fc99e Initial load
duke
parents:
diff changeset
1214 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1215 case T_INT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1216 receiver->int_field_put(offset, value->i);
a61af66fc99e Initial load
duke
parents:
diff changeset
1217 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1218 case T_LONG:
a61af66fc99e Initial load
duke
parents:
diff changeset
1219 receiver->long_field_put(offset, value->j);
a61af66fc99e Initial load
duke
parents:
diff changeset
1220 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1221 case T_OBJECT:
a61af66fc99e Initial load
duke
parents:
diff changeset
1222 case T_ARRAY: {
a61af66fc99e Initial load
duke
parents:
diff changeset
1223 Handle obj(THREAD, (oop) value->l);
a61af66fc99e Initial load
duke
parents:
diff changeset
1224 if (obj.not_null()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1225 symbolHandle signature(THREAD, fd->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
1226 Handle loader (THREAD, fd->loader());
a61af66fc99e Initial load
duke
parents:
diff changeset
1227 Handle protect (THREAD, Klass::cast(fd->field_holder())->protection_domain());
a61af66fc99e Initial load
duke
parents:
diff changeset
1228 klassOop k = SystemDictionary::resolve_or_fail(signature, loader, protect, true, CHECK); // may block
a61af66fc99e Initial load
duke
parents:
diff changeset
1229 if (!obj->is_a(k)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1230 THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "field type mismatch");
a61af66fc99e Initial load
duke
parents:
diff changeset
1231 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1232 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1233 receiver->obj_field_put(offset, obj());
a61af66fc99e Initial load
duke
parents:
diff changeset
1234 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1235 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1236 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
1237 THROW_MSG(vmSymbols::java_lang_IllegalArgumentException(), "field type mismatch");
a61af66fc99e Initial load
duke
parents:
diff changeset
1238 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1239 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1240
a61af66fc99e Initial load
duke
parents:
diff changeset
1241
a61af66fc99e Initial load
duke
parents:
diff changeset
1242 oop Reflection::reflect_field(oop mirror, symbolOop field_name, jint which, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1243 // Exclude primitive types and array types
a61af66fc99e Initial load
duke
parents:
diff changeset
1244 if (java_lang_Class::is_primitive(mirror)) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1245 if (Klass::cast(java_lang_Class::as_klassOop(mirror))->oop_is_array()) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1246
a61af66fc99e Initial load
duke
parents:
diff changeset
1247 instanceKlassHandle k(THREAD, java_lang_Class::as_klassOop(mirror));
a61af66fc99e Initial load
duke
parents:
diff changeset
1248 bool local_fields_only = (which == DECLARED);
a61af66fc99e Initial load
duke
parents:
diff changeset
1249
a61af66fc99e Initial load
duke
parents:
diff changeset
1250 // Ensure class is linked
a61af66fc99e Initial load
duke
parents:
diff changeset
1251 k->link_class(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1252
a61af66fc99e Initial load
duke
parents:
diff changeset
1253 // Search class and interface fields
a61af66fc99e Initial load
duke
parents:
diff changeset
1254 for (FieldStream st(k, local_fields_only, false); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1255 if (st.name() == field_name) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1256 if (local_fields_only || st.access_flags().is_public()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1257 return new_field(&st, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1258 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1259 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1260 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1261
a61af66fc99e Initial load
duke
parents:
diff changeset
1262 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1263 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1264
a61af66fc99e Initial load
duke
parents:
diff changeset
1265
a61af66fc99e Initial load
duke
parents:
diff changeset
1266 objArrayOop Reflection::reflect_fields(oop mirror, jint which, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1267 // Exclude primitive types and array types
a61af66fc99e Initial load
duke
parents:
diff changeset
1268 if (java_lang_Class::is_primitive(mirror)
a61af66fc99e Initial load
duke
parents:
diff changeset
1269 || Klass::cast(java_lang_Class::as_klassOop(mirror))->oop_is_array()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1270 symbolHandle name = vmSymbolHandles::java_lang_reflect_Field();
a61af66fc99e Initial load
duke
parents:
diff changeset
1271 klassOop klass = SystemDictionary::resolve_or_fail(name, true, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1272 return oopFactory::new_objArray(klass, 0, CHECK_NULL); // Return empty array
a61af66fc99e Initial load
duke
parents:
diff changeset
1273 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1274
a61af66fc99e Initial load
duke
parents:
diff changeset
1275 instanceKlassHandle k(THREAD, java_lang_Class::as_klassOop(mirror));
a61af66fc99e Initial load
duke
parents:
diff changeset
1276
a61af66fc99e Initial load
duke
parents:
diff changeset
1277 // Ensure class is linked
a61af66fc99e Initial load
duke
parents:
diff changeset
1278 k->link_class(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1279
a61af66fc99e Initial load
duke
parents:
diff changeset
1280 bool local_fields_only = (which == DECLARED);
a61af66fc99e Initial load
duke
parents:
diff changeset
1281 int count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1282 { // Compute fields count for class and interface fields
a61af66fc99e Initial load
duke
parents:
diff changeset
1283 for (FieldStream st(k, local_fields_only, false); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1284 if (local_fields_only || st.access_flags().is_public()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1285 count++;
a61af66fc99e Initial load
duke
parents:
diff changeset
1286 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1287 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1288 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1289
a61af66fc99e Initial load
duke
parents:
diff changeset
1290 // Allocate result
a61af66fc99e Initial load
duke
parents:
diff changeset
1291 symbolHandle name = vmSymbolHandles::java_lang_reflect_Field();
a61af66fc99e Initial load
duke
parents:
diff changeset
1292 klassOop klass = SystemDictionary::resolve_or_fail(name, true, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1293 objArrayOop r = oopFactory::new_objArray(klass, count, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1294 objArrayHandle result (THREAD, r);
a61af66fc99e Initial load
duke
parents:
diff changeset
1295
a61af66fc99e Initial load
duke
parents:
diff changeset
1296 // Fill in results backwards
a61af66fc99e Initial load
duke
parents:
diff changeset
1297 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1298 for (FieldStream st(k, local_fields_only, false); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1299 if (local_fields_only || st.access_flags().is_public()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1300 oop field = new_field(&st, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1301 result->obj_at_put(--count, field);
a61af66fc99e Initial load
duke
parents:
diff changeset
1302 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1303 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1304 assert(count == 0, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
1305 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1306 return result();
a61af66fc99e Initial load
duke
parents:
diff changeset
1307 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1308
a61af66fc99e Initial load
duke
parents:
diff changeset
1309
a61af66fc99e Initial load
duke
parents:
diff changeset
1310 oop Reflection::reflect_method(oop mirror, symbolHandle method_name, objArrayHandle types, jint which, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1311 if (java_lang_Class::is_primitive(mirror)) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1312 klassOop klass = java_lang_Class::as_klassOop(mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1313 if (Klass::cast(klass)->oop_is_array() && which == MEMBER_DECLARED) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1314
a61af66fc99e Initial load
duke
parents:
diff changeset
1315 if (Klass::cast(java_lang_Class::as_klassOop(mirror))->oop_is_array()) {
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
1316 klass = SystemDictionary::Object_klass();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1317 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1318 instanceKlassHandle h_k(THREAD, klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
1319
a61af66fc99e Initial load
duke
parents:
diff changeset
1320 // Ensure klass is linked (need not be initialized)
a61af66fc99e Initial load
duke
parents:
diff changeset
1321 h_k->link_class(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1322
a61af66fc99e Initial load
duke
parents:
diff changeset
1323 // For interfaces include static initializers under jdk1.2.x (since classic does that)
a61af66fc99e Initial load
duke
parents:
diff changeset
1324 bool include_clinit = JDK_Version::is_jdk12x_version() && h_k->is_interface();
a61af66fc99e Initial load
duke
parents:
diff changeset
1325
a61af66fc99e Initial load
duke
parents:
diff changeset
1326 switch (which) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1327 case MEMBER_PUBLIC:
a61af66fc99e Initial load
duke
parents:
diff changeset
1328 // First the public non-static methods (works if method holder is an interface)
a61af66fc99e Initial load
duke
parents:
diff changeset
1329 // Note that we can ignore checks for overridden methods, since we go up the hierarchy.
a61af66fc99e Initial load
duke
parents:
diff changeset
1330 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1331 for (MethodStream st(h_k, false, false); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1332 methodHandle m(THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1333 // For interfaces include static initializers since classic does that!
a61af66fc99e Initial load
duke
parents:
diff changeset
1334 if (method_name() == m->name() && (include_clinit || (m->is_public() && !m->is_static() && !m->is_initializer()))) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1335 symbolHandle signature(THREAD, m->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
1336 bool parameter_match = match_parameter_types(m, types, ArgumentCount(signature).size(), CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1337 if (parameter_match) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1338 return new_method(m, false, false, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1339 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1340 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1341 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1342 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1343 // Then the public static methods (works if method holder is an interface)
a61af66fc99e Initial load
duke
parents:
diff changeset
1344 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1345 for (MethodStream st(h_k, false, false); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1346 methodHandle m(THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1347 if (method_name() == m->name() && m->is_public() && m->is_static() && !m->is_initializer()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1348 symbolHandle signature(THREAD, m->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
1349 bool parameter_match = match_parameter_types(m, types, ArgumentCount(signature).size(), CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1350 if (parameter_match) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1351 return new_method(m, false, false, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1352 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1353 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1354 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1355 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1356 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1357 case MEMBER_DECLARED:
a61af66fc99e Initial load
duke
parents:
diff changeset
1358 // All local methods
a61af66fc99e Initial load
duke
parents:
diff changeset
1359 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1360 for (MethodStream st(h_k, true, true); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1361 methodHandle m(THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1362 if (method_name() == m->name() && !m->is_initializer()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1363 symbolHandle signature(THREAD, m->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
1364 bool parameter_match = match_parameter_types(m, types, ArgumentCount(signature).size(), CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1365 if (parameter_match) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1366 return new_method(m, false, false, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1367 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1368 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1369 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1370 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1371 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1372 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
1373 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1374 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1375 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1376 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1377
a61af66fc99e Initial load
duke
parents:
diff changeset
1378
a61af66fc99e Initial load
duke
parents:
diff changeset
1379 objArrayOop Reflection::reflect_methods(oop mirror, jint which, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1380 // Exclude primitive types
a61af66fc99e Initial load
duke
parents:
diff changeset
1381 if (java_lang_Class::is_primitive(mirror) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
1382 (Klass::cast(java_lang_Class::as_klassOop(mirror))->oop_is_array() && (which == MEMBER_DECLARED))) {
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
1383 klassOop klass = SystemDictionary::reflect_Method_klass();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1384 return oopFactory::new_objArray(klass, 0, CHECK_NULL); // Return empty array
a61af66fc99e Initial load
duke
parents:
diff changeset
1385 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1386
a61af66fc99e Initial load
duke
parents:
diff changeset
1387 klassOop klass = java_lang_Class::as_klassOop(mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1388 if (Klass::cast(java_lang_Class::as_klassOop(mirror))->oop_is_array()) {
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
1389 klass = SystemDictionary::Object_klass();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1390 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1391 instanceKlassHandle h_k(THREAD, klass);
a61af66fc99e Initial load
duke
parents:
diff changeset
1392
a61af66fc99e Initial load
duke
parents:
diff changeset
1393 // Ensure klass is linked (need not be initialized)
a61af66fc99e Initial load
duke
parents:
diff changeset
1394 h_k->link_class(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1395
a61af66fc99e Initial load
duke
parents:
diff changeset
1396 // We search the (super)interfaces only if h_k is an interface itself
a61af66fc99e Initial load
duke
parents:
diff changeset
1397 bool is_interface = h_k->is_interface();
a61af66fc99e Initial load
duke
parents:
diff changeset
1398
a61af66fc99e Initial load
duke
parents:
diff changeset
1399 // For interfaces include static initializers under jdk1.2.x (since classic does that)
a61af66fc99e Initial load
duke
parents:
diff changeset
1400 bool include_clinit = JDK_Version::is_jdk12x_version() && is_interface;
a61af66fc99e Initial load
duke
parents:
diff changeset
1401
a61af66fc99e Initial load
duke
parents:
diff changeset
1402 switch (which) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1403 case MEMBER_PUBLIC:
a61af66fc99e Initial load
duke
parents:
diff changeset
1404 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1405
a61af66fc99e Initial load
duke
parents:
diff changeset
1406 // Count public methods (non-static and static)
a61af66fc99e Initial load
duke
parents:
diff changeset
1407 int count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1408 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1409 for (MethodStream st(h_k, false, false); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1410 methodOop m = st.method();
a61af66fc99e Initial load
duke
parents:
diff changeset
1411 // For interfaces include static initializers since classic does that!
a61af66fc99e Initial load
duke
parents:
diff changeset
1412 if (include_clinit || (!m->is_initializer() && m->is_public() && !m->is_overridden_in(h_k()))) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1413 count++;
a61af66fc99e Initial load
duke
parents:
diff changeset
1414 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1415 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1416 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1417
a61af66fc99e Initial load
duke
parents:
diff changeset
1418 // Allocate result
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
1419 klassOop klass = SystemDictionary::reflect_Method_klass();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1420 objArrayOop r = oopFactory::new_objArray(klass, count, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1421 objArrayHandle h_result (THREAD, r);
a61af66fc99e Initial load
duke
parents:
diff changeset
1422
a61af66fc99e Initial load
duke
parents:
diff changeset
1423 // Fill in results backwards
a61af66fc99e Initial load
duke
parents:
diff changeset
1424 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1425 // First the non-static public methods
a61af66fc99e Initial load
duke
parents:
diff changeset
1426 for (MethodStream st(h_k, false, false); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1427 methodHandle m (THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1428 if (!m->is_static() && !m->is_initializer() && m->is_public() && !m->is_overridden_in(h_k())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1429 oop method = new_method(m, false, false, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1430 if (method == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1431 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1432 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1433 h_result->obj_at_put(--count, method);
a61af66fc99e Initial load
duke
parents:
diff changeset
1434 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1435 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1436 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1437 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1438 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1439 // Then the static public methods
a61af66fc99e Initial load
duke
parents:
diff changeset
1440 for (MethodStream st(h_k, false, !is_interface); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1441 methodHandle m (THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1442 if (m->is_static() && (include_clinit || (!m->is_initializer()) && m->is_public() && !m->is_overridden_in(h_k()))) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1443 oop method = new_method(m, false, false, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1444 if (method == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1445 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1446 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1447 h_result->obj_at_put(--count, method);
a61af66fc99e Initial load
duke
parents:
diff changeset
1448 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1449 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1450 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1451 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1452
a61af66fc99e Initial load
duke
parents:
diff changeset
1453 assert(count == 0, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
1454 return h_result();
a61af66fc99e Initial load
duke
parents:
diff changeset
1455 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1456
a61af66fc99e Initial load
duke
parents:
diff changeset
1457 case MEMBER_DECLARED:
a61af66fc99e Initial load
duke
parents:
diff changeset
1458 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1459 // Count all methods
a61af66fc99e Initial load
duke
parents:
diff changeset
1460 int count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1461 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1462 for (MethodStream st(h_k, true, !is_interface); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1463 methodOop m = st.method();
a61af66fc99e Initial load
duke
parents:
diff changeset
1464 if (!m->is_initializer()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1465 count++;
a61af66fc99e Initial load
duke
parents:
diff changeset
1466 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1467 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1468 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1469 // Allocate result
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
1470 klassOop klass = SystemDictionary::reflect_Method_klass();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1471 objArrayOop r = oopFactory::new_objArray(klass, count, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1472 objArrayHandle h_result (THREAD, r);
a61af66fc99e Initial load
duke
parents:
diff changeset
1473
a61af66fc99e Initial load
duke
parents:
diff changeset
1474 // Fill in results backwards
a61af66fc99e Initial load
duke
parents:
diff changeset
1475 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1476 for (MethodStream st(h_k, true, true); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1477 methodHandle m (THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1478 if (!m->is_initializer()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1479 oop method = new_method(m, false, false, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1480 if (method == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1481 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1482 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1483 h_result->obj_at_put(--count, method);
a61af66fc99e Initial load
duke
parents:
diff changeset
1484 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1485 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1486 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1487 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1488 assert(count == 0, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
1489 return h_result();
a61af66fc99e Initial load
duke
parents:
diff changeset
1490 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1491 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1492 ShouldNotReachHere();
a61af66fc99e Initial load
duke
parents:
diff changeset
1493 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1494 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1495
a61af66fc99e Initial load
duke
parents:
diff changeset
1496
a61af66fc99e Initial load
duke
parents:
diff changeset
1497 oop Reflection::reflect_constructor(oop mirror, objArrayHandle types, jint which, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1498
a61af66fc99e Initial load
duke
parents:
diff changeset
1499 // Exclude primitive, interface and array types
a61af66fc99e Initial load
duke
parents:
diff changeset
1500 bool prim = java_lang_Class::is_primitive(mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1501 Klass* klass = prim ? NULL : Klass::cast(java_lang_Class::as_klassOop(mirror));
a61af66fc99e Initial load
duke
parents:
diff changeset
1502 if (prim || klass->is_interface() || klass->oop_is_array()) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1503
a61af66fc99e Initial load
duke
parents:
diff changeset
1504 // Must be instance klass
a61af66fc99e Initial load
duke
parents:
diff changeset
1505 instanceKlassHandle h_k(THREAD, java_lang_Class::as_klassOop(mirror));
a61af66fc99e Initial load
duke
parents:
diff changeset
1506
a61af66fc99e Initial load
duke
parents:
diff changeset
1507 // Ensure klass is linked (need not be initialized)
a61af66fc99e Initial load
duke
parents:
diff changeset
1508 h_k->link_class(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1509
a61af66fc99e Initial load
duke
parents:
diff changeset
1510 bool local_only = (which == MEMBER_DECLARED);
a61af66fc99e Initial load
duke
parents:
diff changeset
1511 for (MethodStream st(h_k, true, true); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1512 methodHandle m(THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1513 if (m->name() == vmSymbols::object_initializer_name() && (local_only || m->is_public())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1514 symbolHandle signature(THREAD, m->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
1515 bool parameter_match = match_parameter_types(m, types, ArgumentCount(signature).size(), CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1516 if (parameter_match) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1517 return new_constructor(m, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1518 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1519 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1520 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1521
a61af66fc99e Initial load
duke
parents:
diff changeset
1522 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1523 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1524
a61af66fc99e Initial load
duke
parents:
diff changeset
1525
a61af66fc99e Initial load
duke
parents:
diff changeset
1526 objArrayOop Reflection::reflect_constructors(oop mirror, jint which, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1527 // Exclude primitive, interface and array types
a61af66fc99e Initial load
duke
parents:
diff changeset
1528 bool prim = java_lang_Class::is_primitive(mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1529 Klass* k = prim ? NULL : Klass::cast(java_lang_Class::as_klassOop(mirror));
a61af66fc99e Initial load
duke
parents:
diff changeset
1530 if (prim || k->is_interface() || k->oop_is_array()) {
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 665
diff changeset
1531 return oopFactory::new_objArray(SystemDictionary::reflect_Constructor_klass(), 0, CHECK_NULL); // Return empty array
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1532 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1533
a61af66fc99e Initial load
duke
parents:
diff changeset
1534 // Must be instanceKlass at this point
a61af66fc99e Initial load
duke
parents:
diff changeset
1535 instanceKlassHandle h_k(THREAD, java_lang_Class::as_klassOop(mirror));
a61af66fc99e Initial load
duke
parents:
diff changeset
1536
a61af66fc99e Initial load
duke
parents:
diff changeset
1537 // Ensure klass is linked (need not be initialized)
a61af66fc99e Initial load
duke
parents:
diff changeset
1538 h_k->link_class(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1539
a61af66fc99e Initial load
duke
parents:
diff changeset
1540 bool local_only = (which == MEMBER_DECLARED);
a61af66fc99e Initial load
duke
parents:
diff changeset
1541 int count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1542 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1543 for (MethodStream st(h_k, true, true); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1544 methodOop m = st.method();
a61af66fc99e Initial load
duke
parents:
diff changeset
1545 if (m->name() == vmSymbols::object_initializer_name() && (local_only || m->is_public())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1546 count++;
a61af66fc99e Initial load
duke
parents:
diff changeset
1547 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1548 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1549 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1550
a61af66fc99e Initial load
duke
parents:
diff changeset
1551 // Allocate result
a61af66fc99e Initial load
duke
parents:
diff changeset
1552 symbolHandle name = vmSymbolHandles::java_lang_reflect_Constructor();
a61af66fc99e Initial load
duke
parents:
diff changeset
1553 klassOop klass = SystemDictionary::resolve_or_fail(name, true, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1554 objArrayOop r = oopFactory::new_objArray(klass, count, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1555 objArrayHandle h_result (THREAD, r);
a61af66fc99e Initial load
duke
parents:
diff changeset
1556
a61af66fc99e Initial load
duke
parents:
diff changeset
1557 // Fill in results backwards
a61af66fc99e Initial load
duke
parents:
diff changeset
1558 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1559 for (MethodStream st(h_k, true, true); !st.eos(); st.next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1560 methodHandle m (THREAD, st.method());
a61af66fc99e Initial load
duke
parents:
diff changeset
1561 if (m->name() == vmSymbols::object_initializer_name() && (local_only || m->is_public())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1562 oop constr = new_constructor(m, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1563 if (constr == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1564 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1565 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1566 h_result->obj_at_put(--count, constr);
a61af66fc99e Initial load
duke
parents:
diff changeset
1567 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1568 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1569 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1570 assert(count == 0, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
1571 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1572 return h_result();
a61af66fc99e Initial load
duke
parents:
diff changeset
1573 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1574
a61af66fc99e Initial load
duke
parents:
diff changeset
1575
a61af66fc99e Initial load
duke
parents:
diff changeset
1576 // This would be nicer if, say, java.lang.reflect.Method was a subclass
a61af66fc99e Initial load
duke
parents:
diff changeset
1577 // of java.lang.reflect.Constructor
a61af66fc99e Initial load
duke
parents:
diff changeset
1578
a61af66fc99e Initial load
duke
parents:
diff changeset
1579 oop Reflection::invoke_method(oop method_mirror, Handle receiver, objArrayHandle args, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1580 oop mirror = java_lang_reflect_Method::clazz(method_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1581 int slot = java_lang_reflect_Method::slot(method_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1582 bool override = java_lang_reflect_Method::override(method_mirror) != 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1583 objArrayHandle ptypes(THREAD, objArrayOop(java_lang_reflect_Method::parameter_types(method_mirror)));
a61af66fc99e Initial load
duke
parents:
diff changeset
1584
a61af66fc99e Initial load
duke
parents:
diff changeset
1585 oop return_type_mirror = java_lang_reflect_Method::return_type(method_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1586 BasicType rtype;
a61af66fc99e Initial load
duke
parents:
diff changeset
1587 if (java_lang_Class::is_primitive(return_type_mirror)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1588 rtype = basic_type_mirror_to_basic_type(return_type_mirror, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1589 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1590 rtype = T_OBJECT;
a61af66fc99e Initial load
duke
parents:
diff changeset
1591 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1592
a61af66fc99e Initial load
duke
parents:
diff changeset
1593 instanceKlassHandle klass(THREAD, java_lang_Class::as_klassOop(mirror));
51
1ffa5cdd0b7e 6667089: 3/3 multiple redefinitions of a class break reflection
dcubed
parents: 0
diff changeset
1594 methodOop m = klass->method_with_idnum(slot);
1ffa5cdd0b7e 6667089: 3/3 multiple redefinitions of a class break reflection
dcubed
parents: 0
diff changeset
1595 if (m == NULL) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1596 THROW_MSG_0(vmSymbols::java_lang_InternalError(), "invoke");
a61af66fc99e Initial load
duke
parents:
diff changeset
1597 }
51
1ffa5cdd0b7e 6667089: 3/3 multiple redefinitions of a class break reflection
dcubed
parents: 0
diff changeset
1598 methodHandle method(THREAD, m);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1599
a61af66fc99e Initial load
duke
parents:
diff changeset
1600 return invoke(klass, method, receiver, override, ptypes, rtype, args, true, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1601 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1602
a61af66fc99e Initial load
duke
parents:
diff changeset
1603
a61af66fc99e Initial load
duke
parents:
diff changeset
1604 oop Reflection::invoke_constructor(oop constructor_mirror, objArrayHandle args, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1605 oop mirror = java_lang_reflect_Constructor::clazz(constructor_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1606 int slot = java_lang_reflect_Constructor::slot(constructor_mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
1607 bool override = java_lang_reflect_Constructor::override(constructor_mirror) != 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1608 objArrayHandle ptypes(THREAD, objArrayOop(java_lang_reflect_Constructor::parameter_types(constructor_mirror)));
a61af66fc99e Initial load
duke
parents:
diff changeset
1609
a61af66fc99e Initial load
duke
parents:
diff changeset
1610 instanceKlassHandle klass(THREAD, java_lang_Class::as_klassOop(mirror));
51
1ffa5cdd0b7e 6667089: 3/3 multiple redefinitions of a class break reflection
dcubed
parents: 0
diff changeset
1611 methodOop m = klass->method_with_idnum(slot);
1ffa5cdd0b7e 6667089: 3/3 multiple redefinitions of a class break reflection
dcubed
parents: 0
diff changeset
1612 if (m == NULL) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1613 THROW_MSG_0(vmSymbols::java_lang_InternalError(), "invoke");
a61af66fc99e Initial load
duke
parents:
diff changeset
1614 }
51
1ffa5cdd0b7e 6667089: 3/3 multiple redefinitions of a class break reflection
dcubed
parents: 0
diff changeset
1615 methodHandle method(THREAD, m);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1616 assert(method->name() == vmSymbols::object_initializer_name(), "invalid constructor");
a61af66fc99e Initial load
duke
parents:
diff changeset
1617
a61af66fc99e Initial load
duke
parents:
diff changeset
1618 // Make sure klass gets initialize
a61af66fc99e Initial load
duke
parents:
diff changeset
1619 klass->initialize(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1620
a61af66fc99e Initial load
duke
parents:
diff changeset
1621 // Create new instance (the receiver)
a61af66fc99e Initial load
duke
parents:
diff changeset
1622 klass->check_valid_for_instantiation(false, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1623 Handle receiver = klass->allocate_instance_handle(CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1624
a61af66fc99e Initial load
duke
parents:
diff changeset
1625 // Ignore result from call and return receiver
a61af66fc99e Initial load
duke
parents:
diff changeset
1626 invoke(klass, method, receiver, override, ptypes, T_VOID, args, false, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1627 return receiver();
a61af66fc99e Initial load
duke
parents:
diff changeset
1628 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1629
a61af66fc99e Initial load
duke
parents:
diff changeset
1630
a61af66fc99e Initial load
duke
parents:
diff changeset
1631 #endif /* SUPPORT_OLD_REFLECTION */