< prev index next >
src/hotspot/share/runtime/javaCalls.cpp
Print this page
@@ -28,10 +28,13 @@
#include "code/nmethod.hpp"
#include "compiler/compilationPolicy.hpp"
#include "compiler/compileBroker.hpp"
#include "interpreter/interpreter.hpp"
#include "interpreter/linkResolver.hpp"
+#if INCLUDE_JVMCI
+#include "jvmci/jvmciJavaClasses.hpp"
+#endif
#include "memory/universe.hpp"
#include "oops/method.inline.hpp"
#include "oops/oop.inline.hpp"
#include "prims/jniCheck.hpp"
#include "runtime/handles.inline.hpp"
@@ -344,31 +347,19 @@
assert(thread->is_Java_thread(), "must be called by a java thread");
assert(method.not_null(), "must have a method to call");
assert(!SafepointSynchronize::is_at_safepoint(), "call to Java code during VM operation");
assert(!thread->handle_area()->no_handle_mark_active(), "cannot call out to Java here");
-#if INCLUDE_JVMCI
- // Gets the nmethod (if any) that should be called instead of normal target
- nmethod* alternative_target = args->alternative_target();
- if (alternative_target == NULL) {
-#endif
-// Verify the arguments
-
- if (CheckJNICalls) {
+ // Verify the arguments
+ if (JVMCI_ONLY(args->alternative_target().is_null() &&) (DEBUG_ONLY(true ||) CheckJNICalls)) {
args->verify(method, result->get_type());
}
- else debug_only(args->verify(method, result->get_type()));
-#if INCLUDE_JVMCI
- }
-#else
-
// Ignore call if method is empty
- if (method->is_empty_method()) {
+ if (JVMCI_ONLY(args->alternative_target().is_null() &&) method->is_empty_method()) {
assert(result->get_type() == T_VOID, "an empty method must return a void value");
return;
}
-#endif
#ifdef ASSERT
{ InstanceKlass* holder = method->method_holder();
// A klass might not be initialized since JavaCall's might be used during the executing of
// the <clinit>. For example, a Thread.start might start executing on an object that is
@@ -412,30 +403,32 @@
} else {
// Touch pages checked if the OS needs them to be touched to be mapped.
os::map_stack_shadow_pages(sp);
}
-#if INCLUDE_JVMCI
- if (alternative_target != NULL) {
- if (alternative_target->is_alive() && !alternative_target->is_unloading()) {
- thread->set_jvmci_alternate_call_target(alternative_target->verified_entry_point());
- entry_point = method->adapter()->get_i2c_entry();
- } else {
- THROW(vmSymbols::jdk_vm_ci_code_InvalidInstalledCodeException());
- }
- }
-#endif
-
// do call
{ JavaCallWrapper link(method, receiver, result, CHECK);
{ HandleMark hm(thread); // HandleMark used by HandleMarkCleaner
// NOTE: if we move the computation of the result_val_address inside
// the call to call_stub, the optimizer produces wrong code.
intptr_t* result_val_address = (intptr_t*)(result->get_value_addr());
intptr_t* parameter_address = args->parameters();
-
+#if INCLUDE_JVMCI
+ // Gets the alternative target (if any) that should be called
+ Handle alternative_target = args->alternative_target();
+ if (!alternative_target.is_null()) {
+ // Must extract verified entry point from HotSpotNmethod after VM to Java
+ // transition in JavaCallWrapper constructor so that it is safe with
+ // respect to nmethod sweeping.
+ address verified_entry_point = (address) HotSpotJVMCI::InstalledCode::entryPoint(NULL, alternative_target());
+ if (verified_entry_point != NULL) {
+ thread->set_jvmci_alternate_call_target(verified_entry_point);
+ entry_point = method->adapter()->get_i2c_entry();
+ }
+ }
+#endif
StubRoutines::call_stub()(
(address)&link,
// (intptr_t*)&(result->_value), // see NOTE above (compiler problem)
result_val_address, // see NOTE above (compiler problem)
result_type,
< prev index next >