annotate src/share/vm/prims/nativeLookup.cpp @ 3992:d1bdeef3e3e2

7098282: G1: assert(interval >= 0) failed: Sanity check, referencePolicy.cpp: 76 Summary: There is a race between one thread successfully forwarding and copying the klass mirror for the SoftReference class (including the static master clock) and another thread attempting to use the master clock while attempting to discover a soft reference object. Maintain a shadow copy of the soft reference master clock and use the shadow during reference discovery and reference processing. Reviewed-by: tonyp, brutisso, ysr
author johnc
date Wed, 12 Oct 2011 10:25:51 -0700
parents f08d439fab8c
children 0382d2b469b2
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
2356
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
2 * Copyright (c) 1997, 2011, 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: 1142
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1142
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: 1142
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
1972
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
25 #include "precompiled.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
26 #include "classfile/javaClasses.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
27 #include "classfile/systemDictionary.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
28 #include "classfile/vmSymbols.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
29 #include "memory/oopFactory.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
30 #include "memory/resourceArea.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
31 #include "memory/universe.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
32 #include "oops/instanceKlass.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
33 #include "oops/methodOop.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
34 #include "oops/oop.inline.hpp"
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
35 #include "oops/symbol.hpp"
1972
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
36 #include "prims/jvm_misc.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
37 #include "prims/nativeLookup.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
38 #include "runtime/arguments.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
39 #include "runtime/handles.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
40 #include "runtime/javaCalls.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
41 #include "runtime/sharedRuntime.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
42 #include "runtime/signature.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
43 #ifdef TARGET_OS_FAMILY_linux
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
44 # include "os_linux.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
45 #endif
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
46 #ifdef TARGET_OS_FAMILY_solaris
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
47 # include "os_solaris.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
48 #endif
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
49 #ifdef TARGET_OS_FAMILY_windows
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
50 # include "os_windows.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
51 #endif
3960
f08d439fab8c 7089790: integrate bsd-port changes
never
parents: 2460
diff changeset
52 #ifdef TARGET_OS_FAMILY_bsd
f08d439fab8c 7089790: integrate bsd-port changes
never
parents: 2460
diff changeset
53 # include "os_bsd.inline.hpp"
f08d439fab8c 7089790: integrate bsd-port changes
never
parents: 2460
diff changeset
54 #endif
0
a61af66fc99e Initial load
duke
parents:
diff changeset
55
a61af66fc99e Initial load
duke
parents:
diff changeset
56
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
57 static void mangle_name_on(outputStream* st, Symbol* name, int begin, int end) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
58 char* bytes = (char*)name->bytes() + begin;
a61af66fc99e Initial load
duke
parents:
diff changeset
59 char* end_bytes = (char*)name->bytes() + end;
a61af66fc99e Initial load
duke
parents:
diff changeset
60 while (bytes < end_bytes) {
a61af66fc99e Initial load
duke
parents:
diff changeset
61 jchar c;
a61af66fc99e Initial load
duke
parents:
diff changeset
62 bytes = UTF8::next(bytes, &c);
a61af66fc99e Initial load
duke
parents:
diff changeset
63 if (c <= 0x7f && isalnum(c)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
64 st->put((char) c);
a61af66fc99e Initial load
duke
parents:
diff changeset
65 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
66 if (c == '_') st->print("_1");
a61af66fc99e Initial load
duke
parents:
diff changeset
67 else if (c == '/') st->print("_");
a61af66fc99e Initial load
duke
parents:
diff changeset
68 else if (c == ';') st->print("_2");
a61af66fc99e Initial load
duke
parents:
diff changeset
69 else if (c == '[') st->print("_3");
a61af66fc99e Initial load
duke
parents:
diff changeset
70 else st->print("_%.5x", c);
a61af66fc99e Initial load
duke
parents:
diff changeset
71 }
a61af66fc99e Initial load
duke
parents:
diff changeset
72 }
a61af66fc99e Initial load
duke
parents:
diff changeset
73 }
a61af66fc99e Initial load
duke
parents:
diff changeset
74
a61af66fc99e Initial load
duke
parents:
diff changeset
75
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
76 static void mangle_name_on(outputStream* st, Symbol* name) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
77 mangle_name_on(st, name, 0, name->utf8_length());
a61af66fc99e Initial load
duke
parents:
diff changeset
78 }
a61af66fc99e Initial load
duke
parents:
diff changeset
79
a61af66fc99e Initial load
duke
parents:
diff changeset
80
a61af66fc99e Initial load
duke
parents:
diff changeset
81 char* NativeLookup::pure_jni_name(methodHandle method) {
a61af66fc99e Initial load
duke
parents:
diff changeset
82 stringStream st;
a61af66fc99e Initial load
duke
parents:
diff changeset
83 // Prefix
a61af66fc99e Initial load
duke
parents:
diff changeset
84 st.print("Java_");
a61af66fc99e Initial load
duke
parents:
diff changeset
85 // Klass name
a61af66fc99e Initial load
duke
parents:
diff changeset
86 mangle_name_on(&st, method->klass_name());
a61af66fc99e Initial load
duke
parents:
diff changeset
87 st.print("_");
a61af66fc99e Initial load
duke
parents:
diff changeset
88 // Method name
a61af66fc99e Initial load
duke
parents:
diff changeset
89 mangle_name_on(&st, method->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
90 return st.as_string();
a61af66fc99e Initial load
duke
parents:
diff changeset
91 }
a61af66fc99e Initial load
duke
parents:
diff changeset
92
a61af66fc99e Initial load
duke
parents:
diff changeset
93
a61af66fc99e Initial load
duke
parents:
diff changeset
94 char* NativeLookup::long_jni_name(methodHandle method) {
a61af66fc99e Initial load
duke
parents:
diff changeset
95 // Signature ignore the wrapping parenteses and the trailing return type
a61af66fc99e Initial load
duke
parents:
diff changeset
96 stringStream st;
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
97 Symbol* signature = method->signature();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
98 st.print("__");
a61af66fc99e Initial load
duke
parents:
diff changeset
99 // find ')'
a61af66fc99e Initial load
duke
parents:
diff changeset
100 int end;
a61af66fc99e Initial load
duke
parents:
diff changeset
101 for (end = 0; end < signature->utf8_length() && signature->byte_at(end) != ')'; end++);
a61af66fc99e Initial load
duke
parents:
diff changeset
102 // skip first '('
a61af66fc99e Initial load
duke
parents:
diff changeset
103 mangle_name_on(&st, signature, 1, end);
a61af66fc99e Initial load
duke
parents:
diff changeset
104 return st.as_string();
a61af66fc99e Initial load
duke
parents:
diff changeset
105 }
a61af66fc99e Initial load
duke
parents:
diff changeset
106
a61af66fc99e Initial load
duke
parents:
diff changeset
107 extern "C" {
a61af66fc99e Initial load
duke
parents:
diff changeset
108 void JNICALL JVM_RegisterUnsafeMethods(JNIEnv *env, jclass unsafecls);
710
e5b0439ef4ae 6655638: dynamic languages need method handles
jrose
parents: 0
diff changeset
109 void JNICALL JVM_RegisterMethodHandleMethods(JNIEnv *env, jclass unsafecls);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
110 void JNICALL JVM_RegisterPerfMethods(JNIEnv *env, jclass perfclass);
a61af66fc99e Initial load
duke
parents:
diff changeset
111 }
a61af66fc99e Initial load
duke
parents:
diff changeset
112
2356
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
113 #define CC (char*) /* cast a literal from (const char*) */
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
114 #define FN_PTR(f) CAST_FROM_FN_PTR(void*, &f)
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
115
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
116 static JNINativeMethod lookup_special_native_methods[] = {
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
117 // Next two functions only exist for compatibility with 1.3.1 and earlier.
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
118 { CC"Java_java_io_ObjectOutputStream_getPrimitiveFieldValues", NULL, FN_PTR(JVM_GetPrimitiveFieldValues) }, // intercept ObjectOutputStream getPrimitiveFieldValues for faster serialization
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
119 { CC"Java_java_io_ObjectInputStream_setPrimitiveFieldValues", NULL, FN_PTR(JVM_SetPrimitiveFieldValues) }, // intercept ObjectInputStream setPrimitiveFieldValues for faster serialization
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
120
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
121 { CC"Java_sun_misc_Unsafe_registerNatives", NULL, FN_PTR(JVM_RegisterUnsafeMethods) },
2357
8033953d67ff 7012648: move JSR 292 to package java.lang.invoke and adjust names
jrose
parents: 2356
diff changeset
122 { CC"Java_java_lang_invoke_MethodHandleNatives_registerNatives", NULL, FN_PTR(JVM_RegisterMethodHandleMethods) },
2356
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
123 { CC"Java_sun_misc_Perf_registerNatives", NULL, FN_PTR(JVM_RegisterPerfMethods) }
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
124 };
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
125
0
a61af66fc99e Initial load
duke
parents:
diff changeset
126 static address lookup_special_native(char* jni_name) {
2356
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
127 int i = !JDK_Version::is_gte_jdk14x_version() ? 0 : 2; // see comment in lookup_special_native_methods
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
128 int count = sizeof(lookup_special_native_methods) / sizeof(JNINativeMethod);
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
129 for (; i < count; i++) {
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
130 // NB: To ignore the jni prefix and jni postfix strstr is used matching.
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
131 if (strstr(jni_name, lookup_special_native_methods[i].name) != NULL) {
72dee110246f 6839872: remove implementation inheritance from JSR 292 APIs
jrose
parents: 2177
diff changeset
132 return CAST_FROM_FN_PTR(address, lookup_special_native_methods[i].fnPtr);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
133 }
a61af66fc99e Initial load
duke
parents:
diff changeset
134 }
a61af66fc99e Initial load
duke
parents:
diff changeset
135 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
136 }
a61af66fc99e Initial load
duke
parents:
diff changeset
137
a61af66fc99e Initial load
duke
parents:
diff changeset
138 address NativeLookup::lookup_style(methodHandle method, char* pure_name, const char* long_name, int args_size, bool os_style, bool& in_base_library, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
139 address entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
140 // Compute complete JNI name for style
a61af66fc99e Initial load
duke
parents:
diff changeset
141 stringStream st;
a61af66fc99e Initial load
duke
parents:
diff changeset
142 if (os_style) os::print_jni_name_prefix_on(&st, args_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
143 st.print_raw(pure_name);
a61af66fc99e Initial load
duke
parents:
diff changeset
144 st.print_raw(long_name);
a61af66fc99e Initial load
duke
parents:
diff changeset
145 if (os_style) os::print_jni_name_suffix_on(&st, args_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
146 char* jni_name = st.as_string();
a61af66fc99e Initial load
duke
parents:
diff changeset
147
a61af66fc99e Initial load
duke
parents:
diff changeset
148 // If the loader is null we have a system class, so we attempt a lookup in
a61af66fc99e Initial load
duke
parents:
diff changeset
149 // the native Java library. This takes care of any bootstrapping problems.
a61af66fc99e Initial load
duke
parents:
diff changeset
150 // Note: It is critical for bootstrapping that Java_java_lang_ClassLoader_00024NativeLibrary_find
a61af66fc99e Initial load
duke
parents:
diff changeset
151 // gets found the first time around - otherwise an infinite loop can occure. This is
a61af66fc99e Initial load
duke
parents:
diff changeset
152 // another VM/library dependency
a61af66fc99e Initial load
duke
parents:
diff changeset
153 Handle loader(THREAD,
a61af66fc99e Initial load
duke
parents:
diff changeset
154 instanceKlass::cast(method->method_holder())->class_loader());
a61af66fc99e Initial load
duke
parents:
diff changeset
155 if (loader.is_null()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
156 entry = lookup_special_native(jni_name);
a61af66fc99e Initial load
duke
parents:
diff changeset
157 if (entry == NULL) {
1980
828eafbd85cc 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 1972
diff changeset
158 entry = (address) os::dll_lookup(os::native_java_library(), jni_name);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
159 }
a61af66fc99e Initial load
duke
parents:
diff changeset
160 if (entry != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
161 in_base_library = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
162 return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
163 }
a61af66fc99e Initial load
duke
parents:
diff changeset
164 }
a61af66fc99e Initial load
duke
parents:
diff changeset
165
a61af66fc99e Initial load
duke
parents:
diff changeset
166 // Otherwise call static method findNative in ClassLoader
1142
4ce7240d622c 6914300: ciEnv should export all well known classes
never
parents: 710
diff changeset
167 KlassHandle klass (THREAD, SystemDictionary::ClassLoader_klass());
0
a61af66fc99e Initial load
duke
parents:
diff changeset
168 Handle name_arg = java_lang_String::create_from_str(jni_name, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
169
a61af66fc99e Initial load
duke
parents:
diff changeset
170 JavaValue result(T_LONG);
a61af66fc99e Initial load
duke
parents:
diff changeset
171 JavaCalls::call_static(&result,
a61af66fc99e Initial load
duke
parents:
diff changeset
172 klass,
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
173 vmSymbols::findNative_name(),
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
174 vmSymbols::classloader_string_long_signature(),
0
a61af66fc99e Initial load
duke
parents:
diff changeset
175 // Arguments
a61af66fc99e Initial load
duke
parents:
diff changeset
176 loader,
a61af66fc99e Initial load
duke
parents:
diff changeset
177 name_arg,
a61af66fc99e Initial load
duke
parents:
diff changeset
178 CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
179 entry = (address) (intptr_t) result.get_jlong();
a61af66fc99e Initial load
duke
parents:
diff changeset
180
a61af66fc99e Initial load
duke
parents:
diff changeset
181 if (entry == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
182 // findNative didn't find it, if there are any agent libraries look in them
a61af66fc99e Initial load
duke
parents:
diff changeset
183 AgentLibrary* agent;
a61af66fc99e Initial load
duke
parents:
diff changeset
184 for (agent = Arguments::agents(); agent != NULL; agent = agent->next()) {
1980
828eafbd85cc 6348631: remove the use of the HPI library from Hotspot
ikrylov
parents: 1972
diff changeset
185 entry = (address) os::dll_lookup(agent->os_lib(), jni_name);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
186 if (entry != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
187 return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
188 }
a61af66fc99e Initial load
duke
parents:
diff changeset
189 }
a61af66fc99e Initial load
duke
parents:
diff changeset
190 }
a61af66fc99e Initial load
duke
parents:
diff changeset
191
a61af66fc99e Initial load
duke
parents:
diff changeset
192 return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
193 }
a61af66fc99e Initial load
duke
parents:
diff changeset
194
a61af66fc99e Initial load
duke
parents:
diff changeset
195
a61af66fc99e Initial load
duke
parents:
diff changeset
196 // Check all the formats of native implementation name to see if there is one
a61af66fc99e Initial load
duke
parents:
diff changeset
197 // for the specified method.
a61af66fc99e Initial load
duke
parents:
diff changeset
198 address NativeLookup::lookup_entry(methodHandle method, bool& in_base_library, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
199 address entry = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
200 in_base_library = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
201 // Compute pure name
a61af66fc99e Initial load
duke
parents:
diff changeset
202 char* pure_name = pure_jni_name(method);
a61af66fc99e Initial load
duke
parents:
diff changeset
203
a61af66fc99e Initial load
duke
parents:
diff changeset
204 // Compute argument size
a61af66fc99e Initial load
duke
parents:
diff changeset
205 int args_size = 1 // JNIEnv
a61af66fc99e Initial load
duke
parents:
diff changeset
206 + (method->is_static() ? 1 : 0) // class for static methods
a61af66fc99e Initial load
duke
parents:
diff changeset
207 + method->size_of_parameters(); // actual parameters
a61af66fc99e Initial load
duke
parents:
diff changeset
208
a61af66fc99e Initial load
duke
parents:
diff changeset
209
a61af66fc99e Initial load
duke
parents:
diff changeset
210 // 1) Try JNI short style
a61af66fc99e Initial load
duke
parents:
diff changeset
211 entry = lookup_style(method, pure_name, "", args_size, true, in_base_library, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
212 if (entry != NULL) return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
213
a61af66fc99e Initial load
duke
parents:
diff changeset
214 // Compute long name
a61af66fc99e Initial load
duke
parents:
diff changeset
215 char* long_name = long_jni_name(method);
a61af66fc99e Initial load
duke
parents:
diff changeset
216
a61af66fc99e Initial load
duke
parents:
diff changeset
217 // 2) Try JNI long style
a61af66fc99e Initial load
duke
parents:
diff changeset
218 entry = lookup_style(method, pure_name, long_name, args_size, true, in_base_library, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
219 if (entry != NULL) return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
220
a61af66fc99e Initial load
duke
parents:
diff changeset
221 // 3) Try JNI short style without os prefix/suffix
a61af66fc99e Initial load
duke
parents:
diff changeset
222 entry = lookup_style(method, pure_name, "", args_size, false, in_base_library, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
223 if (entry != NULL) return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
224
a61af66fc99e Initial load
duke
parents:
diff changeset
225 // 4) Try JNI long style without os prefix/suffix
a61af66fc99e Initial load
duke
parents:
diff changeset
226 entry = lookup_style(method, pure_name, long_name, args_size, false, in_base_library, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
227
a61af66fc99e Initial load
duke
parents:
diff changeset
228 return entry; // NULL indicates not found
a61af66fc99e Initial load
duke
parents:
diff changeset
229 }
a61af66fc99e Initial load
duke
parents:
diff changeset
230
a61af66fc99e Initial load
duke
parents:
diff changeset
231 // Check if there are any JVM TI prefixes which have been applied to the native method name.
a61af66fc99e Initial load
duke
parents:
diff changeset
232 // If any are found, remove them before attemping the look up of the
a61af66fc99e Initial load
duke
parents:
diff changeset
233 // native implementation again.
a61af66fc99e Initial load
duke
parents:
diff changeset
234 // See SetNativeMethodPrefix in the JVM TI Spec for more details.
a61af66fc99e Initial load
duke
parents:
diff changeset
235 address NativeLookup::lookup_entry_prefixed(methodHandle method, bool& in_base_library, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
236 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
237
a61af66fc99e Initial load
duke
parents:
diff changeset
238 int prefix_count;
a61af66fc99e Initial load
duke
parents:
diff changeset
239 char** prefixes = JvmtiExport::get_all_native_method_prefixes(&prefix_count);
a61af66fc99e Initial load
duke
parents:
diff changeset
240 char* in_name = method->name()->as_C_string();
a61af66fc99e Initial load
duke
parents:
diff changeset
241 char* wrapper_name = in_name;
a61af66fc99e Initial load
duke
parents:
diff changeset
242 // last applied prefix will be first -- go backwards
a61af66fc99e Initial load
duke
parents:
diff changeset
243 for (int i = prefix_count-1; i >= 0; i--) {
a61af66fc99e Initial load
duke
parents:
diff changeset
244 char* prefix = prefixes[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
245 size_t prefix_len = strlen(prefix);
a61af66fc99e Initial load
duke
parents:
diff changeset
246 if (strncmp(prefix, wrapper_name, prefix_len) == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
247 // has this prefix remove it
a61af66fc99e Initial load
duke
parents:
diff changeset
248 wrapper_name += prefix_len;
a61af66fc99e Initial load
duke
parents:
diff changeset
249 }
a61af66fc99e Initial load
duke
parents:
diff changeset
250 }
a61af66fc99e Initial load
duke
parents:
diff changeset
251 if (wrapper_name != in_name) {
a61af66fc99e Initial load
duke
parents:
diff changeset
252 // we have a name for a wrapping method
a61af66fc99e Initial load
duke
parents:
diff changeset
253 int wrapper_name_len = (int)strlen(wrapper_name);
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
254 TempNewSymbol wrapper_symbol = SymbolTable::probe(wrapper_name, wrapper_name_len);
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
255 if (wrapper_symbol != NULL) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
256 KlassHandle kh(method->method_holder());
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
257 methodOop wrapper_method = Klass::cast(kh())->lookup_method(wrapper_symbol,
0
a61af66fc99e Initial load
duke
parents:
diff changeset
258 method->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
259 if (wrapper_method != NULL && !wrapper_method->is_native()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
260 // we found a wrapper method, use its native entry
a61af66fc99e Initial load
duke
parents:
diff changeset
261 method->set_is_prefixed_native();
a61af66fc99e Initial load
duke
parents:
diff changeset
262 return lookup_entry(wrapper_method, in_base_library, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
263 }
a61af66fc99e Initial load
duke
parents:
diff changeset
264 }
a61af66fc99e Initial load
duke
parents:
diff changeset
265 }
a61af66fc99e Initial load
duke
parents:
diff changeset
266 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
267 }
a61af66fc99e Initial load
duke
parents:
diff changeset
268
a61af66fc99e Initial load
duke
parents:
diff changeset
269 address NativeLookup::lookup_base(methodHandle method, bool& in_base_library, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
270 address entry = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
271 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
272
a61af66fc99e Initial load
duke
parents:
diff changeset
273 entry = lookup_entry(method, in_base_library, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
274 if (entry != NULL) return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
275
a61af66fc99e Initial load
duke
parents:
diff changeset
276 // standard native method resolution has failed. Check if there are any
a61af66fc99e Initial load
duke
parents:
diff changeset
277 // JVM TI prefixes which have been applied to the native method name.
a61af66fc99e Initial load
duke
parents:
diff changeset
278 entry = lookup_entry_prefixed(method, in_base_library, THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
279 if (entry != NULL) return entry;
a61af66fc99e Initial load
duke
parents:
diff changeset
280
a61af66fc99e Initial load
duke
parents:
diff changeset
281 // Native function not found, throw UnsatisfiedLinkError
a61af66fc99e Initial load
duke
parents:
diff changeset
282 THROW_MSG_0(vmSymbols::java_lang_UnsatisfiedLinkError(),
a61af66fc99e Initial load
duke
parents:
diff changeset
283 method->name_and_sig_as_C_string());
a61af66fc99e Initial load
duke
parents:
diff changeset
284 }
a61af66fc99e Initial load
duke
parents:
diff changeset
285
a61af66fc99e Initial load
duke
parents:
diff changeset
286
a61af66fc99e Initial load
duke
parents:
diff changeset
287 address NativeLookup::lookup(methodHandle method, bool& in_base_library, TRAPS) {
a61af66fc99e Initial load
duke
parents:
diff changeset
288 if (!method->has_native_function()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
289 address entry = lookup_base(method, in_base_library, CHECK_NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
290 method->set_native_function(entry,
a61af66fc99e Initial load
duke
parents:
diff changeset
291 methodOopDesc::native_bind_event_is_interesting);
a61af66fc99e Initial load
duke
parents:
diff changeset
292 // -verbose:jni printing
a61af66fc99e Initial load
duke
parents:
diff changeset
293 if (PrintJNIResolving) {
a61af66fc99e Initial load
duke
parents:
diff changeset
294 ResourceMark rm(THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
295 tty->print_cr("[Dynamic-linking native method %s.%s ... JNI]",
a61af66fc99e Initial load
duke
parents:
diff changeset
296 Klass::cast(method->method_holder())->external_name(),
a61af66fc99e Initial load
duke
parents:
diff changeset
297 method->name()->as_C_string());
a61af66fc99e Initial load
duke
parents:
diff changeset
298 }
a61af66fc99e Initial load
duke
parents:
diff changeset
299 }
a61af66fc99e Initial load
duke
parents:
diff changeset
300 return method->native_function();
a61af66fc99e Initial load
duke
parents:
diff changeset
301 }
a61af66fc99e Initial load
duke
parents:
diff changeset
302
a61af66fc99e Initial load
duke
parents:
diff changeset
303 address NativeLookup::base_library_lookup(const char* class_name, const char* method_name, const char* signature) {
a61af66fc99e Initial load
duke
parents:
diff changeset
304 EXCEPTION_MARK;
a61af66fc99e Initial load
duke
parents:
diff changeset
305 bool in_base_library = true; // SharedRuntime inits some math methods.
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
306 TempNewSymbol c_name = SymbolTable::new_symbol(class_name, CATCH);
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
307 TempNewSymbol m_name = SymbolTable::new_symbol(method_name, CATCH);
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
308 TempNewSymbol s_name = SymbolTable::new_symbol(signature, CATCH);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
309
a61af66fc99e Initial load
duke
parents:
diff changeset
310 // Find the class
a61af66fc99e Initial load
duke
parents:
diff changeset
311 klassOop k = SystemDictionary::resolve_or_fail(c_name, true, CATCH);
a61af66fc99e Initial load
duke
parents:
diff changeset
312 instanceKlassHandle klass (THREAD, k);
a61af66fc99e Initial load
duke
parents:
diff changeset
313
a61af66fc99e Initial load
duke
parents:
diff changeset
314 // Find method and invoke standard lookup
a61af66fc99e Initial load
duke
parents:
diff changeset
315 methodHandle method (THREAD,
2177
3582bf76420e 6990754: Use native memory and reference counting to implement SymbolTable
coleenp
parents: 1980
diff changeset
316 klass->uncached_lookup_method(m_name, s_name));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
317 address result = lookup(method, in_base_library, CATCH);
a61af66fc99e Initial load
duke
parents:
diff changeset
318 assert(in_base_library, "must be in basic library");
a61af66fc99e Initial load
duke
parents:
diff changeset
319 guarantee(result != NULL, "must be non NULL");
a61af66fc99e Initial load
duke
parents:
diff changeset
320 return result;
a61af66fc99e Initial load
duke
parents:
diff changeset
321 }