Path: blob/master/src/hotspot/share/prims/jni.cpp
64440 views
/*1* Copyright (c) 1997, 2021, Oracle and/or its affiliates. All rights reserved.2* Copyright (c) 2012 Red Hat, Inc.3* Copyright (c) 2021, Azul Systems, Inc. All rights reserved.4* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.5*6* This code is free software; you can redistribute it and/or modify it7* under the terms of the GNU General Public License version 2 only, as8* published by the Free Software Foundation.9*10* This code is distributed in the hope that it will be useful, but WITHOUT11* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or12* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License13* version 2 for more details (a copy is included in the LICENSE file that14* accompanied this code).15*16* You should have received a copy of the GNU General Public License version17* 2 along with this work; if not, write to the Free Software Foundation,18* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.19*20* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA21* or visit www.oracle.com if you need additional information or have any22* questions.23*24*/2526#include "precompiled.hpp"27#include "jni.h"28#include "jvm.h"29#include "ci/ciReplay.hpp"30#include "classfile/altHashing.hpp"31#include "classfile/classFileStream.hpp"32#include "classfile/classLoader.hpp"33#include "classfile/classLoadInfo.hpp"34#include "classfile/javaClasses.hpp"35#include "classfile/javaClasses.inline.hpp"36#include "classfile/javaThreadStatus.hpp"37#include "classfile/moduleEntry.hpp"38#include "classfile/modules.hpp"39#include "classfile/symbolTable.hpp"40#include "classfile/systemDictionary.hpp"41#include "classfile/vmClasses.hpp"42#include "classfile/vmSymbols.hpp"43#include "compiler/compiler_globals.hpp"44#include "gc/shared/collectedHeap.hpp"45#include "gc/shared/gcLocker.inline.hpp"46#include "gc/shared/stringdedup/stringDedup.hpp"47#include "interpreter/linkResolver.hpp"48#include "jfr/jfrEvents.hpp"49#include "jfr/support/jfrThreadId.hpp"50#include "logging/log.hpp"51#include "memory/allocation.hpp"52#include "memory/allocation.inline.hpp"53#include "memory/oopFactory.hpp"54#include "memory/resourceArea.hpp"55#include "memory/universe.hpp"56#include "oops/access.inline.hpp"57#include "oops/arrayOop.hpp"58#include "oops/instanceKlass.inline.hpp"59#include "oops/instanceOop.hpp"60#include "oops/klass.inline.hpp"61#include "oops/markWord.hpp"62#include "oops/method.hpp"63#include "oops/objArrayKlass.hpp"64#include "oops/objArrayOop.inline.hpp"65#include "oops/oop.inline.hpp"66#include "oops/symbol.hpp"67#include "oops/typeArrayKlass.hpp"68#include "oops/typeArrayOop.inline.hpp"69#include "prims/jniCheck.hpp"70#include "prims/jniExport.hpp"71#include "prims/jniFastGetField.hpp"72#include "prims/jvm_misc.hpp"73#include "prims/jvmtiExport.hpp"74#include "prims/jvmtiThreadState.hpp"75#include "runtime/atomic.hpp"76#include "runtime/fieldDescriptor.inline.hpp"77#include "runtime/handles.inline.hpp"78#include "runtime/interfaceSupport.inline.hpp"79#include "runtime/java.hpp"80#include "runtime/javaCalls.hpp"81#include "runtime/jfieldIDWorkaround.hpp"82#include "runtime/jniHandles.inline.hpp"83#include "runtime/reflection.hpp"84#include "runtime/safepointVerifiers.hpp"85#include "runtime/sharedRuntime.hpp"86#include "runtime/signature.hpp"87#include "runtime/thread.inline.hpp"88#include "runtime/vmOperations.hpp"89#include "services/memTracker.hpp"90#include "services/runtimeService.hpp"91#include "utilities/defaultStream.hpp"92#include "utilities/dtrace.hpp"93#include "utilities/events.hpp"94#include "utilities/macros.hpp"95#include "utilities/vmError.hpp"96#if INCLUDE_JVMCI97#include "jvmci/jvmciCompiler.hpp"98#endif99100static jint CurrentVersion = JNI_VERSION_10;101102#if defined(_WIN32) && !defined(USE_VECTORED_EXCEPTION_HANDLING)103extern LONG WINAPI topLevelExceptionFilter(_EXCEPTION_POINTERS* );104#endif105106// The DT_RETURN_MARK macros create a scoped object to fire the dtrace107// '-return' probe regardless of the return path is taken out of the function.108// Methods that have multiple return paths use this to avoid having to109// instrument each return path. Methods that use CHECK or THROW must use this110// since those macros can cause an immedate uninstrumented return.111//112// In order to get the return value, a reference to the variable containing113// the return value must be passed to the contructor of the object, and114// the return value must be set before return (since the mark object has115// a reference to it).116//117// Example:118// DT_RETURN_MARK_DECL(SomeFunc, int);119// JNI_ENTRY(int, SomeFunc, ...)120// int return_value = 0;121// DT_RETURN_MARK(SomeFunc, int, (const int&)return_value);122// foo(CHECK_0)123// return_value = 5;124// return return_value;125// JNI_END126#define DT_RETURN_MARK_DECL(name, type, probe) \127DTRACE_ONLY( \128class DTraceReturnProbeMark_##name { \129public: \130const type& _ret_ref; \131DTraceReturnProbeMark_##name(const type& v) : _ret_ref(v) {} \132~DTraceReturnProbeMark_##name() { \133probe; \134} \135} \136)137// Void functions are simpler since there's no return value138#define DT_VOID_RETURN_MARK_DECL(name, probe) \139DTRACE_ONLY( \140class DTraceReturnProbeMark_##name { \141public: \142~DTraceReturnProbeMark_##name() { \143probe; \144} \145} \146)147148// Place these macros in the function to mark the return. Non-void149// functions need the type and address of the return value.150#define DT_RETURN_MARK(name, type, ref) \151DTRACE_ONLY( DTraceReturnProbeMark_##name dtrace_return_mark(ref) )152#define DT_VOID_RETURN_MARK(name) \153DTRACE_ONLY( DTraceReturnProbeMark_##name dtrace_return_mark )154155156// Use these to select distinct code for floating-point vs. non-floating point157// situations. Used from within common macros where we need slightly158// different behavior for Float/Double159#define FP_SELECT_Boolean(intcode, fpcode) intcode160#define FP_SELECT_Byte(intcode, fpcode) intcode161#define FP_SELECT_Char(intcode, fpcode) intcode162#define FP_SELECT_Short(intcode, fpcode) intcode163#define FP_SELECT_Object(intcode, fpcode) intcode164#define FP_SELECT_Int(intcode, fpcode) intcode165#define FP_SELECT_Long(intcode, fpcode) intcode166#define FP_SELECT_Float(intcode, fpcode) fpcode167#define FP_SELECT_Double(intcode, fpcode) fpcode168#define FP_SELECT(TypeName, intcode, fpcode) \169FP_SELECT_##TypeName(intcode, fpcode)170171// Choose DT_RETURN_MARK macros based on the type: float/double -> void172// (dtrace doesn't do FP yet)173#define DT_RETURN_MARK_DECL_FOR(TypeName, name, type, probe) \174FP_SELECT(TypeName, \175DT_RETURN_MARK_DECL(name, type, probe), DT_VOID_RETURN_MARK_DECL(name, probe) )176#define DT_RETURN_MARK_FOR(TypeName, name, type, ref) \177FP_SELECT(TypeName, \178DT_RETURN_MARK(name, type, ref), DT_VOID_RETURN_MARK(name) )179180181// out-of-line helpers for class jfieldIDWorkaround:182183bool jfieldIDWorkaround::is_valid_jfieldID(Klass* k, jfieldID id) {184if (jfieldIDWorkaround::is_instance_jfieldID(k, id)) {185uintptr_t as_uint = (uintptr_t) id;186intptr_t offset = raw_instance_offset(id);187if (is_checked_jfieldID(id)) {188if (!klass_hash_ok(k, id)) {189return false;190}191}192return InstanceKlass::cast(k)->contains_field_offset(offset);193} else {194JNIid* result = (JNIid*) id;195#ifdef ASSERT196return result != NULL && result->is_static_field_id();197#else198return result != NULL;199#endif200}201}202203204intptr_t jfieldIDWorkaround::encode_klass_hash(Klass* k, intptr_t offset) {205if (offset <= small_offset_mask) {206Klass* field_klass = k;207Klass* super_klass = field_klass->super();208// With compressed oops the most super class with nonstatic fields would209// be the owner of fields embedded in the header.210while (InstanceKlass::cast(super_klass)->has_nonstatic_fields() &&211InstanceKlass::cast(super_klass)->contains_field_offset(offset)) {212field_klass = super_klass; // super contains the field also213super_klass = field_klass->super();214}215debug_only(NoSafepointVerifier nosafepoint;)216uintptr_t klass_hash = field_klass->identity_hash();217return ((klass_hash & klass_mask) << klass_shift) | checked_mask_in_place;218} else {219#if 0220#ifndef PRODUCT221{222ResourceMark rm;223warning("VerifyJNIFields: long offset %d in %s", offset, k->external_name());224}225#endif226#endif227return 0;228}229}230231bool jfieldIDWorkaround::klass_hash_ok(Klass* k, jfieldID id) {232uintptr_t as_uint = (uintptr_t) id;233intptr_t klass_hash = (as_uint >> klass_shift) & klass_mask;234do {235debug_only(NoSafepointVerifier nosafepoint;)236// Could use a non-blocking query for identity_hash here...237if ((k->identity_hash() & klass_mask) == klass_hash)238return true;239k = k->super();240} while (k != NULL);241return false;242}243244void jfieldIDWorkaround::verify_instance_jfieldID(Klass* k, jfieldID id) {245guarantee(jfieldIDWorkaround::is_instance_jfieldID(k, id), "must be an instance field" );246uintptr_t as_uint = (uintptr_t) id;247intptr_t offset = raw_instance_offset(id);248if (VerifyJNIFields) {249if (is_checked_jfieldID(id)) {250guarantee(klass_hash_ok(k, id),251"Bug in native code: jfieldID class must match object");252} else {253#if 0254#ifndef PRODUCT255if (Verbose) {256ResourceMark rm;257warning("VerifyJNIFields: unverified offset %d for %s", offset, k->external_name());258}259#endif260#endif261}262}263guarantee(InstanceKlass::cast(k)->contains_field_offset(offset),264"Bug in native code: jfieldID offset must address interior of object");265}266267// Implementation of JNI entries268269DT_RETURN_MARK_DECL(DefineClass, jclass270, HOTSPOT_JNI_DEFINECLASS_RETURN(_ret_ref));271272JNI_ENTRY(jclass, jni_DefineClass(JNIEnv *env, const char *name, jobject loaderRef,273const jbyte *buf, jsize bufLen))274HOTSPOT_JNI_DEFINECLASS_ENTRY(275env, (char*) name, loaderRef, (char*) buf, bufLen);276277jclass cls = NULL;278DT_RETURN_MARK(DefineClass, jclass, (const jclass&)cls);279280// Class resolution will get the class name from the .class stream if the name is null.281TempNewSymbol class_name = name == NULL ? NULL :282SystemDictionary::class_name_symbol(name, vmSymbols::java_lang_NoClassDefFoundError(),283CHECK_NULL);284285ResourceMark rm(THREAD);286ClassFileStream st((u1*)buf, bufLen, NULL, ClassFileStream::verify);287Handle class_loader (THREAD, JNIHandles::resolve(loaderRef));288Handle protection_domain;289ClassLoadInfo cl_info(protection_domain);290Klass* k = SystemDictionary::resolve_from_stream(&st, class_name,291class_loader,292cl_info,293CHECK_NULL);294295if (log_is_enabled(Debug, class, resolve)) {296trace_class_resolution(k);297}298299cls = (jclass)JNIHandles::make_local(THREAD, k->java_mirror());300return cls;301JNI_END302303304305DT_RETURN_MARK_DECL(FindClass, jclass306, HOTSPOT_JNI_FINDCLASS_RETURN(_ret_ref));307308JNI_ENTRY(jclass, jni_FindClass(JNIEnv *env, const char *name))309HOTSPOT_JNI_FINDCLASS_ENTRY(env, (char *)name);310311jclass result = NULL;312DT_RETURN_MARK(FindClass, jclass, (const jclass&)result);313314// This should be ClassNotFoundException imo.315TempNewSymbol class_name =316SystemDictionary::class_name_symbol(name, vmSymbols::java_lang_NoClassDefFoundError(),317CHECK_NULL);318319//%note jni_3320Handle protection_domain;321// Find calling class322Klass* k = thread->security_get_caller_class(0);323// default to the system loader when no context324Handle loader(THREAD, SystemDictionary::java_system_loader());325if (k != NULL) {326// Special handling to make sure JNI_OnLoad and JNI_OnUnload are executed327// in the correct class context.328if (k->class_loader() == NULL &&329k->name() == vmSymbols::jdk_internal_loader_NativeLibraries()) {330JavaValue result(T_OBJECT);331JavaCalls::call_static(&result, k,332vmSymbols::getFromClass_name(),333vmSymbols::void_class_signature(),334CHECK_NULL);335// When invoked from JNI_OnLoad, NativeLibraries::getFromClass returns336// a non-NULL Class object. When invoked from JNI_OnUnload,337// it will return NULL to indicate no context.338oop mirror = result.get_oop();339if (mirror != NULL) {340Klass* fromClass = java_lang_Class::as_Klass(mirror);341loader = Handle(THREAD, fromClass->class_loader());342protection_domain = Handle(THREAD, fromClass->protection_domain());343}344} else {345loader = Handle(THREAD, k->class_loader());346}347}348349result = find_class_from_class_loader(env, class_name, true, loader,350protection_domain, true, thread);351352if (log_is_enabled(Debug, class, resolve) && result != NULL) {353trace_class_resolution(java_lang_Class::as_Klass(JNIHandles::resolve_non_null(result)));354}355356return result;357JNI_END358359DT_RETURN_MARK_DECL(FromReflectedMethod, jmethodID360, HOTSPOT_JNI_FROMREFLECTEDMETHOD_RETURN((uintptr_t)_ret_ref));361362JNI_ENTRY(jmethodID, jni_FromReflectedMethod(JNIEnv *env, jobject method))363HOTSPOT_JNI_FROMREFLECTEDMETHOD_ENTRY(env, method);364365jmethodID ret = NULL;366DT_RETURN_MARK(FromReflectedMethod, jmethodID, (const jmethodID&)ret);367368// method is a handle to a java.lang.reflect.Method object369oop reflected = JNIHandles::resolve_non_null(method);370oop mirror = NULL;371int slot = 0;372373if (reflected->klass() == vmClasses::reflect_Constructor_klass()) {374mirror = java_lang_reflect_Constructor::clazz(reflected);375slot = java_lang_reflect_Constructor::slot(reflected);376} else {377assert(reflected->klass() == vmClasses::reflect_Method_klass(), "wrong type");378mirror = java_lang_reflect_Method::clazz(reflected);379slot = java_lang_reflect_Method::slot(reflected);380}381Klass* k1 = java_lang_Class::as_Klass(mirror);382383// Make sure class is initialized before handing id's out to methods384k1->initialize(CHECK_NULL);385Method* m = InstanceKlass::cast(k1)->method_with_idnum(slot);386ret = m==NULL? NULL : m->jmethod_id(); // return NULL if reflected method deleted387return ret;388JNI_END389390DT_RETURN_MARK_DECL(FromReflectedField, jfieldID391, HOTSPOT_JNI_FROMREFLECTEDFIELD_RETURN((uintptr_t)_ret_ref));392393JNI_ENTRY(jfieldID, jni_FromReflectedField(JNIEnv *env, jobject field))394HOTSPOT_JNI_FROMREFLECTEDFIELD_ENTRY(env, field);395396jfieldID ret = NULL;397DT_RETURN_MARK(FromReflectedField, jfieldID, (const jfieldID&)ret);398399// field is a handle to a java.lang.reflect.Field object400oop reflected = JNIHandles::resolve_non_null(field);401oop mirror = java_lang_reflect_Field::clazz(reflected);402Klass* k1 = java_lang_Class::as_Klass(mirror);403int slot = java_lang_reflect_Field::slot(reflected);404int modifiers = java_lang_reflect_Field::modifiers(reflected);405406// Make sure class is initialized before handing id's out to fields407k1->initialize(CHECK_NULL);408409// First check if this is a static field410if (modifiers & JVM_ACC_STATIC) {411intptr_t offset = InstanceKlass::cast(k1)->field_offset( slot );412JNIid* id = InstanceKlass::cast(k1)->jni_id_for(offset);413assert(id != NULL, "corrupt Field object");414debug_only(id->set_is_static_field_id();)415// A jfieldID for a static field is a JNIid specifying the field holder and the offset within the Klass*416ret = jfieldIDWorkaround::to_static_jfieldID(id);417return ret;418}419420// The slot is the index of the field description in the field-array421// The jfieldID is the offset of the field within the object422// It may also have hash bits for k, if VerifyJNIFields is turned on.423intptr_t offset = InstanceKlass::cast(k1)->field_offset( slot );424assert(InstanceKlass::cast(k1)->contains_field_offset(offset), "stay within object");425ret = jfieldIDWorkaround::to_instance_jfieldID(k1, offset);426return ret;427JNI_END428429430DT_RETURN_MARK_DECL(ToReflectedMethod, jobject431, HOTSPOT_JNI_TOREFLECTEDMETHOD_RETURN(_ret_ref));432433JNI_ENTRY(jobject, jni_ToReflectedMethod(JNIEnv *env, jclass cls, jmethodID method_id, jboolean isStatic))434HOTSPOT_JNI_TOREFLECTEDMETHOD_ENTRY(env, cls, (uintptr_t) method_id, isStatic);435436jobject ret = NULL;437DT_RETURN_MARK(ToReflectedMethod, jobject, (const jobject&)ret);438439methodHandle m (THREAD, Method::resolve_jmethod_id(method_id));440assert(m->is_static() == (isStatic != 0), "jni_ToReflectedMethod access flags doesn't match");441oop reflection_method;442if (m->is_initializer()) {443reflection_method = Reflection::new_constructor(m, CHECK_NULL);444} else {445reflection_method = Reflection::new_method(m, false, CHECK_NULL);446}447ret = JNIHandles::make_local(THREAD, reflection_method);448return ret;449JNI_END450451DT_RETURN_MARK_DECL(GetSuperclass, jclass452, HOTSPOT_JNI_GETSUPERCLASS_RETURN(_ret_ref));453454JNI_ENTRY(jclass, jni_GetSuperclass(JNIEnv *env, jclass sub))455HOTSPOT_JNI_GETSUPERCLASS_ENTRY(env, sub);456457jclass obj = NULL;458DT_RETURN_MARK(GetSuperclass, jclass, (const jclass&)obj);459460oop mirror = JNIHandles::resolve_non_null(sub);461// primitive classes return NULL462if (java_lang_Class::is_primitive(mirror)) return NULL;463464// Rules of Class.getSuperClass as implemented by KLass::java_super:465// arrays return Object466// interfaces return NULL467// proper classes return Klass::super()468Klass* k = java_lang_Class::as_Klass(mirror);469if (k->is_interface()) return NULL;470471// return mirror for superclass472Klass* super = k->java_super();473// super2 is the value computed by the compiler's getSuperClass intrinsic:474debug_only(Klass* super2 = ( k->is_array_klass()475? vmClasses::Object_klass()476: k->super() ) );477assert(super == super2,478"java_super computation depends on interface, array, other super");479obj = (super == NULL) ? NULL : (jclass) JNIHandles::make_local(THREAD, super->java_mirror());480return obj;481JNI_END482483JNI_ENTRY_NO_PRESERVE(jboolean, jni_IsAssignableFrom(JNIEnv *env, jclass sub, jclass super))484HOTSPOT_JNI_ISASSIGNABLEFROM_ENTRY(env, sub, super);485486oop sub_mirror = JNIHandles::resolve_non_null(sub);487oop super_mirror = JNIHandles::resolve_non_null(super);488if (java_lang_Class::is_primitive(sub_mirror) ||489java_lang_Class::is_primitive(super_mirror)) {490jboolean ret = (sub_mirror == super_mirror);491492HOTSPOT_JNI_ISASSIGNABLEFROM_RETURN(ret);493return ret;494}495Klass* sub_klass = java_lang_Class::as_Klass(sub_mirror);496Klass* super_klass = java_lang_Class::as_Klass(super_mirror);497assert(sub_klass != NULL && super_klass != NULL, "invalid arguments to jni_IsAssignableFrom");498jboolean ret = sub_klass->is_subtype_of(super_klass) ?499JNI_TRUE : JNI_FALSE;500501HOTSPOT_JNI_ISASSIGNABLEFROM_RETURN(ret);502return ret;503JNI_END504505506DT_RETURN_MARK_DECL(Throw, jint507, HOTSPOT_JNI_THROW_RETURN(_ret_ref));508509JNI_ENTRY(jint, jni_Throw(JNIEnv *env, jthrowable obj))510HOTSPOT_JNI_THROW_ENTRY(env, obj);511512jint ret = JNI_OK;513DT_RETURN_MARK(Throw, jint, (const jint&)ret);514515THROW_OOP_(JNIHandles::resolve(obj), JNI_OK);516ShouldNotReachHere();517return 0; // Mute compiler.518JNI_END519520521DT_RETURN_MARK_DECL(ThrowNew, jint522, HOTSPOT_JNI_THROWNEW_RETURN(_ret_ref));523524JNI_ENTRY(jint, jni_ThrowNew(JNIEnv *env, jclass clazz, const char *message))525HOTSPOT_JNI_THROWNEW_ENTRY(env, clazz, (char *) message);526527jint ret = JNI_OK;528DT_RETURN_MARK(ThrowNew, jint, (const jint&)ret);529530InstanceKlass* k = InstanceKlass::cast(java_lang_Class::as_Klass(JNIHandles::resolve_non_null(clazz)));531Symbol* name = k->name();532Handle class_loader (THREAD, k->class_loader());533Handle protection_domain (THREAD, k->protection_domain());534THROW_MSG_LOADER_(name, (char *)message, class_loader, protection_domain, JNI_OK);535ShouldNotReachHere();536return 0; // Mute compiler.537JNI_END538539540// JNI functions only transform a pending async exception to a synchronous541// exception in ExceptionOccurred and ExceptionCheck calls, since542// delivering an async exception in other places won't change the native543// code's control flow and would be harmful when native code further calls544// JNI functions with a pending exception. Async exception is also checked545// during the call, so ExceptionOccurred/ExceptionCheck won't return546// false but deliver the async exception at the very end during547// state transition.548549static void jni_check_async_exceptions(JavaThread *thread) {550assert(thread == Thread::current(), "must be itself");551thread->check_and_handle_async_exceptions();552}553554JNI_ENTRY_NO_PRESERVE(jthrowable, jni_ExceptionOccurred(JNIEnv *env))555HOTSPOT_JNI_EXCEPTIONOCCURRED_ENTRY(env);556557jni_check_async_exceptions(thread);558oop exception = thread->pending_exception();559jthrowable ret = (jthrowable) JNIHandles::make_local(THREAD, exception);560561HOTSPOT_JNI_EXCEPTIONOCCURRED_RETURN(ret);562return ret;563JNI_END564565566JNI_ENTRY_NO_PRESERVE(void, jni_ExceptionDescribe(JNIEnv *env))567HOTSPOT_JNI_EXCEPTIONDESCRIBE_ENTRY(env);568569if (thread->has_pending_exception()) {570Handle ex(thread, thread->pending_exception());571thread->clear_pending_exception();572if (ex->is_a(vmClasses::ThreadDeath_klass())) {573// Don't print anything if we are being killed.574} else {575jio_fprintf(defaultStream::error_stream(), "Exception ");576if (thread != NULL && thread->threadObj() != NULL) {577ResourceMark rm(THREAD);578jio_fprintf(defaultStream::error_stream(),579"in thread \"%s\" ", thread->get_thread_name());580}581if (ex->is_a(vmClasses::Throwable_klass())) {582JavaValue result(T_VOID);583JavaCalls::call_virtual(&result,584ex,585vmClasses::Throwable_klass(),586vmSymbols::printStackTrace_name(),587vmSymbols::void_method_signature(),588THREAD);589// If an exception is thrown in the call it gets thrown away. Not much590// we can do with it. The native code that calls this, does not check591// for the exception - hence, it might still be in the thread when DestroyVM gets592// called, potentially causing a few asserts to trigger - since no pending exception593// is expected.594CLEAR_PENDING_EXCEPTION;595} else {596ResourceMark rm(THREAD);597jio_fprintf(defaultStream::error_stream(),598". Uncaught exception of type %s.",599ex->klass()->external_name());600}601}602}603604HOTSPOT_JNI_EXCEPTIONDESCRIBE_RETURN();605JNI_END606607608JNI_ENTRY_NO_PRESERVE(void, jni_ExceptionClear(JNIEnv *env))609HOTSPOT_JNI_EXCEPTIONCLEAR_ENTRY(env);610611// The jni code might be using this API to clear java thrown exception.612// So just mark jvmti thread exception state as exception caught.613JvmtiThreadState *state = JavaThread::current()->jvmti_thread_state();614if (state != NULL && state->is_exception_detected()) {615state->set_exception_caught();616}617thread->clear_pending_exception();618619HOTSPOT_JNI_EXCEPTIONCLEAR_RETURN();620JNI_END621622623JNI_ENTRY(void, jni_FatalError(JNIEnv *env, const char *msg))624HOTSPOT_JNI_FATALERROR_ENTRY(env, (char *) msg);625626tty->print_cr("FATAL ERROR in native method: %s", msg);627thread->print_stack();628os::abort(); // Dump core and abort629JNI_END630631632JNI_ENTRY(jint, jni_PushLocalFrame(JNIEnv *env, jint capacity))633HOTSPOT_JNI_PUSHLOCALFRAME_ENTRY(env, capacity);634635//%note jni_11636if (capacity < 0 ||637((MaxJNILocalCapacity > 0) && (capacity > MaxJNILocalCapacity))) {638HOTSPOT_JNI_PUSHLOCALFRAME_RETURN((uint32_t)JNI_ERR);639return JNI_ERR;640}641JNIHandleBlock* old_handles = thread->active_handles();642JNIHandleBlock* new_handles = JNIHandleBlock::allocate_block(thread);643assert(new_handles != NULL, "should not be NULL");644new_handles->set_pop_frame_link(old_handles);645thread->set_active_handles(new_handles);646jint ret = JNI_OK;647HOTSPOT_JNI_PUSHLOCALFRAME_RETURN(ret);648return ret;649JNI_END650651652JNI_ENTRY(jobject, jni_PopLocalFrame(JNIEnv *env, jobject result))653HOTSPOT_JNI_POPLOCALFRAME_ENTRY(env, result);654655//%note jni_11656Handle result_handle(thread, JNIHandles::resolve(result));657JNIHandleBlock* old_handles = thread->active_handles();658JNIHandleBlock* new_handles = old_handles->pop_frame_link();659if (new_handles != NULL) {660// As a sanity check we only release the handle blocks if the pop_frame_link is not NULL.661// This way code will still work if PopLocalFrame is called without a corresponding662// PushLocalFrame call. Note that we set the pop_frame_link to NULL explicitly, otherwise663// the release_block call will release the blocks.664thread->set_active_handles(new_handles);665old_handles->set_pop_frame_link(NULL); // clear link we won't release new_handles below666JNIHandleBlock::release_block(old_handles, thread); // may block667result = JNIHandles::make_local(thread, result_handle());668}669HOTSPOT_JNI_POPLOCALFRAME_RETURN(result);670return result;671JNI_END672673674JNI_ENTRY(jobject, jni_NewGlobalRef(JNIEnv *env, jobject ref))675HOTSPOT_JNI_NEWGLOBALREF_ENTRY(env, ref);676677Handle ref_handle(thread, JNIHandles::resolve(ref));678jobject ret = JNIHandles::make_global(ref_handle, AllocFailStrategy::RETURN_NULL);679680HOTSPOT_JNI_NEWGLOBALREF_RETURN(ret);681return ret;682JNI_END683684// Must be JNI_ENTRY (with HandleMark)685JNI_ENTRY_NO_PRESERVE(void, jni_DeleteGlobalRef(JNIEnv *env, jobject ref))686HOTSPOT_JNI_DELETEGLOBALREF_ENTRY(env, ref);687688JNIHandles::destroy_global(ref);689690HOTSPOT_JNI_DELETEGLOBALREF_RETURN();691JNI_END692693JNI_ENTRY_NO_PRESERVE(void, jni_DeleteLocalRef(JNIEnv *env, jobject obj))694HOTSPOT_JNI_DELETELOCALREF_ENTRY(env, obj);695696JNIHandles::destroy_local(obj);697698HOTSPOT_JNI_DELETELOCALREF_RETURN();699JNI_END700701JNI_ENTRY_NO_PRESERVE(jboolean, jni_IsSameObject(JNIEnv *env, jobject r1, jobject r2))702HOTSPOT_JNI_ISSAMEOBJECT_ENTRY(env, r1, r2);703704jboolean ret = JNIHandles::is_same_object(r1, r2) ? JNI_TRUE : JNI_FALSE;705706HOTSPOT_JNI_ISSAMEOBJECT_RETURN(ret);707return ret;708JNI_END709710711JNI_ENTRY(jobject, jni_NewLocalRef(JNIEnv *env, jobject ref))712HOTSPOT_JNI_NEWLOCALREF_ENTRY(env, ref);713714jobject ret = JNIHandles::make_local(THREAD, JNIHandles::resolve(ref),715AllocFailStrategy::RETURN_NULL);716717HOTSPOT_JNI_NEWLOCALREF_RETURN(ret);718return ret;719JNI_END720721JNI_LEAF(jint, jni_EnsureLocalCapacity(JNIEnv *env, jint capacity))722HOTSPOT_JNI_ENSURELOCALCAPACITY_ENTRY(env, capacity);723724jint ret;725if (capacity >= 0 &&726((MaxJNILocalCapacity <= 0) || (capacity <= MaxJNILocalCapacity))) {727ret = JNI_OK;728} else {729ret = JNI_ERR;730}731732HOTSPOT_JNI_ENSURELOCALCAPACITY_RETURN(ret);733return ret;734JNI_END735736// Return the Handle Type737JNI_LEAF(jobjectRefType, jni_GetObjectRefType(JNIEnv *env, jobject obj))738HOTSPOT_JNI_GETOBJECTREFTYPE_ENTRY(env, obj);739740jobjectRefType ret = JNIInvalidRefType;741if (obj != NULL) {742ret = JNIHandles::handle_type(thread, obj);743}744745HOTSPOT_JNI_GETOBJECTREFTYPE_RETURN((void *) ret);746return ret;747JNI_END748749750class JNI_ArgumentPusher : public SignatureIterator {751protected:752JavaCallArguments* _arguments;753754void push_int(jint x) { _arguments->push_int(x); }755void push_long(jlong x) { _arguments->push_long(x); }756void push_float(jfloat x) { _arguments->push_float(x); }757void push_double(jdouble x) { _arguments->push_double(x); }758void push_object(jobject x) { _arguments->push_jobject(x); }759760void push_boolean(jboolean b) {761// Normalize boolean arguments from native code by converting 1-255 to JNI_TRUE and762// 0 to JNI_FALSE. Boolean return values from native are normalized the same in763// TemplateInterpreterGenerator::generate_result_handler_for and764// SharedRuntime::generate_native_wrapper.765push_int(b == 0 ? JNI_FALSE : JNI_TRUE);766}767768JNI_ArgumentPusher(Method* method)769: SignatureIterator(method->signature(),770Fingerprinter(methodHandle(Thread::current(), method)).fingerprint())771{772_arguments = NULL;773}774775public:776virtual void push_arguments_on(JavaCallArguments* arguments) = 0;777};778779780class JNI_ArgumentPusherVaArg : public JNI_ArgumentPusher {781va_list _ap;782783void set_ap(va_list rap) {784va_copy(_ap, rap);785}786787friend class SignatureIterator; // so do_parameters_on can call do_type788void do_type(BasicType type) {789switch (type) {790// these are coerced to int when using va_arg791case T_BYTE:792case T_CHAR:793case T_SHORT:794case T_INT: push_int(va_arg(_ap, jint)); break;795case T_BOOLEAN: push_boolean((jboolean) va_arg(_ap, jint)); break;796797// each of these paths is exercised by the various jck Call[Static,Nonvirtual,][Void,Int,..]Method[A,V,] tests798799case T_LONG: push_long(va_arg(_ap, jlong)); break;800// float is coerced to double w/ va_arg801case T_FLOAT: push_float((jfloat) va_arg(_ap, jdouble)); break;802case T_DOUBLE: push_double(va_arg(_ap, jdouble)); break;803804case T_ARRAY:805case T_OBJECT: push_object(va_arg(_ap, jobject)); break;806default: ShouldNotReachHere();807}808}809810public:811JNI_ArgumentPusherVaArg(jmethodID method_id, va_list rap)812: JNI_ArgumentPusher(Method::resolve_jmethod_id(method_id)) {813set_ap(rap);814}815816~JNI_ArgumentPusherVaArg() {817va_end(_ap);818}819820virtual void push_arguments_on(JavaCallArguments* arguments) {821_arguments = arguments;822do_parameters_on(this);823}824};825826827class JNI_ArgumentPusherArray : public JNI_ArgumentPusher {828protected:829const jvalue *_ap;830831inline void set_ap(const jvalue *rap) { _ap = rap; }832833friend class SignatureIterator; // so do_parameters_on can call do_type834void do_type(BasicType type) {835switch (type) {836case T_CHAR: push_int((_ap++)->c); break;837case T_SHORT: push_int((_ap++)->s); break;838case T_BYTE: push_int((_ap++)->b); break;839case T_INT: push_int((_ap++)->i); break;840case T_BOOLEAN: push_boolean((_ap++)->z); break;841case T_LONG: push_long((_ap++)->j); break;842case T_FLOAT: push_float((_ap++)->f); break;843case T_DOUBLE: push_double((_ap++)->d); break;844case T_ARRAY:845case T_OBJECT: push_object((_ap++)->l); break;846default: ShouldNotReachHere();847}848}849850public:851JNI_ArgumentPusherArray(jmethodID method_id, const jvalue *rap)852: JNI_ArgumentPusher(Method::resolve_jmethod_id(method_id)) {853set_ap(rap);854}855856virtual void push_arguments_on(JavaCallArguments* arguments) {857_arguments = arguments;858do_parameters_on(this);859}860};861862863enum JNICallType {864JNI_STATIC,865JNI_VIRTUAL,866JNI_NONVIRTUAL867};868869870871static void jni_invoke_static(JNIEnv *env, JavaValue* result, jobject receiver, JNICallType call_type, jmethodID method_id, JNI_ArgumentPusher *args, TRAPS) {872methodHandle method(THREAD, Method::resolve_jmethod_id(method_id));873874// Create object to hold arguments for the JavaCall, and associate it with875// the jni parser876ResourceMark rm(THREAD);877int number_of_parameters = method->size_of_parameters();878JavaCallArguments java_args(number_of_parameters);879880assert(method->is_static(), "method should be static");881882// Fill out JavaCallArguments object883args->push_arguments_on(&java_args);884// Initialize result type885result->set_type(args->return_type());886887// Invoke the method. Result is returned as oop.888JavaCalls::call(result, method, &java_args, CHECK);889890// Convert result891if (is_reference_type(result->get_type())) {892result->set_jobject(JNIHandles::make_local(THREAD, result->get_oop()));893}894}895896897static void jni_invoke_nonstatic(JNIEnv *env, JavaValue* result, jobject receiver, JNICallType call_type, jmethodID method_id, JNI_ArgumentPusher *args, TRAPS) {898oop recv = JNIHandles::resolve(receiver);899if (recv == NULL) {900THROW(vmSymbols::java_lang_NullPointerException());901}902Handle h_recv(THREAD, recv);903904int number_of_parameters;905Method* selected_method;906{907Method* m = Method::resolve_jmethod_id(method_id);908number_of_parameters = m->size_of_parameters();909InstanceKlass* holder = m->method_holder();910if (call_type != JNI_VIRTUAL) {911selected_method = m;912} else if (!m->has_itable_index()) {913// non-interface call -- for that little speed boost, don't handlize914debug_only(NoSafepointVerifier nosafepoint;)915// jni_GetMethodID makes sure class is linked and initialized916// so m should have a valid vtable index.917assert(m->valid_vtable_index(), "no valid vtable index");918int vtbl_index = m->vtable_index();919if (vtbl_index != Method::nonvirtual_vtable_index) {920selected_method = h_recv->klass()->method_at_vtable(vtbl_index);921} else {922// final method923selected_method = m;924}925} else {926// interface call927int itbl_index = m->itable_index();928Klass* k = h_recv->klass();929selected_method = InstanceKlass::cast(k)->method_at_itable(holder, itbl_index, CHECK);930}931}932933methodHandle method(THREAD, selected_method);934935// Create object to hold arguments for the JavaCall, and associate it with936// the jni parser937ResourceMark rm(THREAD);938JavaCallArguments java_args(number_of_parameters);939940// handle arguments941assert(!method->is_static(), "method %s should not be static", method->name_and_sig_as_C_string());942java_args.push_oop(h_recv); // Push jobject handle943944// Fill out JavaCallArguments object945args->push_arguments_on(&java_args);946// Initialize result type947result->set_type(args->return_type());948949// Invoke the method. Result is returned as oop.950JavaCalls::call(result, method, &java_args, CHECK);951952// Convert result953if (is_reference_type(result->get_type())) {954result->set_jobject(JNIHandles::make_local(THREAD, result->get_oop()));955}956}957958DT_RETURN_MARK_DECL(AllocObject, jobject959, HOTSPOT_JNI_ALLOCOBJECT_RETURN(_ret_ref));960961JNI_ENTRY(jobject, jni_AllocObject(JNIEnv *env, jclass clazz))962HOTSPOT_JNI_ALLOCOBJECT_ENTRY(env, clazz);963964jobject ret = NULL;965DT_RETURN_MARK(AllocObject, jobject, (const jobject&)ret);966967instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);968ret = JNIHandles::make_local(THREAD, i);969return ret;970JNI_END971972DT_RETURN_MARK_DECL(NewObjectA, jobject973, HOTSPOT_JNI_NEWOBJECTA_RETURN(_ret_ref));974975JNI_ENTRY(jobject, jni_NewObjectA(JNIEnv *env, jclass clazz, jmethodID methodID, const jvalue *args))976HOTSPOT_JNI_NEWOBJECTA_ENTRY(env, clazz, (uintptr_t) methodID);977978jobject obj = NULL;979DT_RETURN_MARK(NewObjectA, jobject, (const jobject&)obj);980981instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);982obj = JNIHandles::make_local(THREAD, i);983JavaValue jvalue(T_VOID);984JNI_ArgumentPusherArray ap(methodID, args);985jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);986return obj;987JNI_END988989990DT_RETURN_MARK_DECL(NewObjectV, jobject991, HOTSPOT_JNI_NEWOBJECTV_RETURN(_ret_ref));992993JNI_ENTRY(jobject, jni_NewObjectV(JNIEnv *env, jclass clazz, jmethodID methodID, va_list args))994HOTSPOT_JNI_NEWOBJECTV_ENTRY(env, clazz, (uintptr_t) methodID);995996jobject obj = NULL;997DT_RETURN_MARK(NewObjectV, jobject, (const jobject&)obj);998999instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);1000obj = JNIHandles::make_local(THREAD, i);1001JavaValue jvalue(T_VOID);1002JNI_ArgumentPusherVaArg ap(methodID, args);1003jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);1004return obj;1005JNI_END100610071008DT_RETURN_MARK_DECL(NewObject, jobject1009, HOTSPOT_JNI_NEWOBJECT_RETURN(_ret_ref));10101011JNI_ENTRY(jobject, jni_NewObject(JNIEnv *env, jclass clazz, jmethodID methodID, ...))1012HOTSPOT_JNI_NEWOBJECT_ENTRY(env, clazz, (uintptr_t) methodID);10131014jobject obj = NULL;1015DT_RETURN_MARK(NewObject, jobject, (const jobject&)obj);10161017instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);1018obj = JNIHandles::make_local(THREAD, i);1019va_list args;1020va_start(args, methodID);1021JavaValue jvalue(T_VOID);1022JNI_ArgumentPusherVaArg ap(methodID, args);1023jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);1024va_end(args);1025return obj;1026JNI_END102710281029JNI_ENTRY(jclass, jni_GetObjectClass(JNIEnv *env, jobject obj))1030HOTSPOT_JNI_GETOBJECTCLASS_ENTRY(env, obj);10311032Klass* k = JNIHandles::resolve_non_null(obj)->klass();1033jclass ret =1034(jclass) JNIHandles::make_local(THREAD, k->java_mirror());10351036HOTSPOT_JNI_GETOBJECTCLASS_RETURN(ret);1037return ret;1038JNI_END10391040JNI_ENTRY_NO_PRESERVE(jboolean, jni_IsInstanceOf(JNIEnv *env, jobject obj, jclass clazz))1041HOTSPOT_JNI_ISINSTANCEOF_ENTRY(env, obj, clazz);10421043jboolean ret = JNI_TRUE;1044if (obj != NULL) {1045ret = JNI_FALSE;1046Klass* k = java_lang_Class::as_Klass(1047JNIHandles::resolve_non_null(clazz));1048if (k != NULL) {1049ret = JNIHandles::resolve_non_null(obj)->is_a(k) ? JNI_TRUE : JNI_FALSE;1050}1051}10521053HOTSPOT_JNI_ISINSTANCEOF_RETURN(ret);1054return ret;1055JNI_END105610571058static jmethodID get_method_id(JNIEnv *env, jclass clazz, const char *name_str,1059const char *sig, bool is_static, TRAPS) {1060// %%%% This code should probably just call into a method in the LinkResolver1061//1062// The class should have been loaded (we have an instance of the class1063// passed in) so the method and signature should already be in the symbol1064// table. If they're not there, the method doesn't exist.1065const char *name_to_probe = (name_str == NULL)1066? vmSymbols::object_initializer_name()->as_C_string()1067: name_str;1068TempNewSymbol name = SymbolTable::probe(name_to_probe, (int)strlen(name_to_probe));1069TempNewSymbol signature = SymbolTable::probe(sig, (int)strlen(sig));10701071if (name == NULL || signature == NULL) {1072THROW_MSG_0(vmSymbols::java_lang_NoSuchMethodError(), name_str);1073}10741075oop mirror = JNIHandles::resolve_non_null(clazz);1076Klass* klass = java_lang_Class::as_Klass(mirror);10771078// Throw a NoSuchMethodError exception if we have an instance of a1079// primitive java.lang.Class1080if (java_lang_Class::is_primitive(mirror)) {1081ResourceMark rm(THREAD);1082THROW_MSG_0(vmSymbols::java_lang_NoSuchMethodError(), err_msg("%s%s.%s%s", is_static ? "static " : "", klass->signature_name(), name_str, sig));1083}10841085// Make sure class is linked and initialized before handing id's out to1086// Method*s.1087klass->initialize(CHECK_NULL);10881089Method* m;1090if (name == vmSymbols::object_initializer_name() ||1091name == vmSymbols::class_initializer_name()) {1092// Never search superclasses for constructors1093if (klass->is_instance_klass()) {1094m = InstanceKlass::cast(klass)->find_method(name, signature);1095} else {1096m = NULL;1097}1098} else {1099m = klass->lookup_method(name, signature);1100if (m == NULL && klass->is_instance_klass()) {1101m = InstanceKlass::cast(klass)->lookup_method_in_ordered_interfaces(name, signature);1102}1103}1104if (m == NULL || (m->is_static() != is_static)) {1105ResourceMark rm(THREAD);1106THROW_MSG_0(vmSymbols::java_lang_NoSuchMethodError(), err_msg("%s%s.%s%s", is_static ? "static " : "", klass->signature_name(), name_str, sig));1107}1108return m->jmethod_id();1109}111011111112JNI_ENTRY(jmethodID, jni_GetMethodID(JNIEnv *env, jclass clazz,1113const char *name, const char *sig))1114HOTSPOT_JNI_GETMETHODID_ENTRY(env, clazz, (char *) name, (char *) sig);1115jmethodID ret = get_method_id(env, clazz, name, sig, false, thread);1116HOTSPOT_JNI_GETMETHODID_RETURN((uintptr_t) ret);1117return ret;1118JNI_END111911201121JNI_ENTRY(jmethodID, jni_GetStaticMethodID(JNIEnv *env, jclass clazz,1122const char *name, const char *sig))1123HOTSPOT_JNI_GETSTATICMETHODID_ENTRY(env, (char *) clazz, (char *) name, (char *)sig);1124jmethodID ret = get_method_id(env, clazz, name, sig, true, thread);1125HOTSPOT_JNI_GETSTATICMETHODID_RETURN((uintptr_t) ret);1126return ret;1127JNI_END1128112911301131//1132// Calling Methods1133//113411351136#define DEFINE_CALLMETHOD(ResultType, Result, Tag \1137, EntryProbe, ReturnProbe) \1138\1139DT_RETURN_MARK_DECL_FOR(Result, Call##Result##Method, ResultType \1140, ReturnProbe); \1141\1142JNI_ENTRY(ResultType, \1143jni_Call##Result##Method(JNIEnv *env, jobject obj, jmethodID methodID, ...)) \1144\1145EntryProbe; \1146ResultType ret = 0;\1147DT_RETURN_MARK_FOR(Result, Call##Result##Method, ResultType, \1148(const ResultType&)ret);\1149\1150va_list args; \1151va_start(args, methodID); \1152JavaValue jvalue(Tag); \1153JNI_ArgumentPusherVaArg ap(methodID, args); \1154jni_invoke_nonstatic(env, &jvalue, obj, JNI_VIRTUAL, methodID, &ap, CHECK_0); \1155va_end(args); \1156ret = jvalue.get_##ResultType(); \1157return ret;\1158JNI_END11591160// the runtime type of subword integral basic types is integer1161DEFINE_CALLMETHOD(jboolean, Boolean, T_BOOLEAN1162, HOTSPOT_JNI_CALLBOOLEANMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1163HOTSPOT_JNI_CALLBOOLEANMETHOD_RETURN(_ret_ref))1164DEFINE_CALLMETHOD(jbyte, Byte, T_BYTE1165, HOTSPOT_JNI_CALLBYTEMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1166HOTSPOT_JNI_CALLBYTEMETHOD_RETURN(_ret_ref))1167DEFINE_CALLMETHOD(jchar, Char, T_CHAR1168, HOTSPOT_JNI_CALLCHARMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1169HOTSPOT_JNI_CALLCHARMETHOD_RETURN(_ret_ref))1170DEFINE_CALLMETHOD(jshort, Short, T_SHORT1171, HOTSPOT_JNI_CALLSHORTMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1172HOTSPOT_JNI_CALLSHORTMETHOD_RETURN(_ret_ref))11731174DEFINE_CALLMETHOD(jobject, Object, T_OBJECT1175, HOTSPOT_JNI_CALLOBJECTMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1176HOTSPOT_JNI_CALLOBJECTMETHOD_RETURN(_ret_ref))1177DEFINE_CALLMETHOD(jint, Int, T_INT,1178HOTSPOT_JNI_CALLINTMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1179HOTSPOT_JNI_CALLINTMETHOD_RETURN(_ret_ref))1180DEFINE_CALLMETHOD(jlong, Long, T_LONG1181, HOTSPOT_JNI_CALLLONGMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1182HOTSPOT_JNI_CALLLONGMETHOD_RETURN(_ret_ref))1183// Float and double probes don't return value because dtrace doesn't currently support it1184DEFINE_CALLMETHOD(jfloat, Float, T_FLOAT1185, HOTSPOT_JNI_CALLFLOATMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1186HOTSPOT_JNI_CALLFLOATMETHOD_RETURN())1187DEFINE_CALLMETHOD(jdouble, Double, T_DOUBLE1188, HOTSPOT_JNI_CALLDOUBLEMETHOD_ENTRY(env, obj, (uintptr_t)methodID),1189HOTSPOT_JNI_CALLDOUBLEMETHOD_RETURN())11901191#define DEFINE_CALLMETHODV(ResultType, Result, Tag \1192, EntryProbe, ReturnProbe) \1193\1194DT_RETURN_MARK_DECL_FOR(Result, Call##Result##MethodV, ResultType \1195, ReturnProbe); \1196\1197JNI_ENTRY(ResultType, \1198jni_Call##Result##MethodV(JNIEnv *env, jobject obj, jmethodID methodID, va_list args)) \1199\1200EntryProbe;\1201ResultType ret = 0;\1202DT_RETURN_MARK_FOR(Result, Call##Result##MethodV, ResultType, \1203(const ResultType&)ret);\1204\1205JavaValue jvalue(Tag); \1206JNI_ArgumentPusherVaArg ap(methodID, args); \1207jni_invoke_nonstatic(env, &jvalue, obj, JNI_VIRTUAL, methodID, &ap, CHECK_0); \1208ret = jvalue.get_##ResultType(); \1209return ret;\1210JNI_END12111212// the runtime type of subword integral basic types is integer1213DEFINE_CALLMETHODV(jboolean, Boolean, T_BOOLEAN1214, HOTSPOT_JNI_CALLBOOLEANMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1215HOTSPOT_JNI_CALLBOOLEANMETHODV_RETURN(_ret_ref))1216DEFINE_CALLMETHODV(jbyte, Byte, T_BYTE1217, HOTSPOT_JNI_CALLBYTEMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1218HOTSPOT_JNI_CALLBYTEMETHODV_RETURN(_ret_ref))1219DEFINE_CALLMETHODV(jchar, Char, T_CHAR1220, HOTSPOT_JNI_CALLCHARMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1221HOTSPOT_JNI_CALLCHARMETHODV_RETURN(_ret_ref))1222DEFINE_CALLMETHODV(jshort, Short, T_SHORT1223, HOTSPOT_JNI_CALLSHORTMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1224HOTSPOT_JNI_CALLSHORTMETHODV_RETURN(_ret_ref))12251226DEFINE_CALLMETHODV(jobject, Object, T_OBJECT1227, HOTSPOT_JNI_CALLOBJECTMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1228HOTSPOT_JNI_CALLOBJECTMETHODV_RETURN(_ret_ref))1229DEFINE_CALLMETHODV(jint, Int, T_INT,1230HOTSPOT_JNI_CALLINTMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1231HOTSPOT_JNI_CALLINTMETHODV_RETURN(_ret_ref))1232DEFINE_CALLMETHODV(jlong, Long, T_LONG1233, HOTSPOT_JNI_CALLLONGMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1234HOTSPOT_JNI_CALLLONGMETHODV_RETURN(_ret_ref))1235// Float and double probes don't return value because dtrace doesn't currently support it1236DEFINE_CALLMETHODV(jfloat, Float, T_FLOAT1237, HOTSPOT_JNI_CALLFLOATMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1238HOTSPOT_JNI_CALLFLOATMETHODV_RETURN())1239DEFINE_CALLMETHODV(jdouble, Double, T_DOUBLE1240, HOTSPOT_JNI_CALLDOUBLEMETHODV_ENTRY(env, obj, (uintptr_t)methodID),1241HOTSPOT_JNI_CALLDOUBLEMETHODV_RETURN())12421243#define DEFINE_CALLMETHODA(ResultType, Result, Tag \1244, EntryProbe, ReturnProbe) \1245\1246DT_RETURN_MARK_DECL_FOR(Result, Call##Result##MethodA, ResultType \1247, ReturnProbe); \1248\1249JNI_ENTRY(ResultType, \1250jni_Call##Result##MethodA(JNIEnv *env, jobject obj, jmethodID methodID, const jvalue *args)) \1251EntryProbe; \1252ResultType ret = 0;\1253DT_RETURN_MARK_FOR(Result, Call##Result##MethodA, ResultType, \1254(const ResultType&)ret);\1255\1256JavaValue jvalue(Tag); \1257JNI_ArgumentPusherArray ap(methodID, args); \1258jni_invoke_nonstatic(env, &jvalue, obj, JNI_VIRTUAL, methodID, &ap, CHECK_0); \1259ret = jvalue.get_##ResultType(); \1260return ret;\1261JNI_END12621263// the runtime type of subword integral basic types is integer1264DEFINE_CALLMETHODA(jboolean, Boolean, T_BOOLEAN1265, HOTSPOT_JNI_CALLBOOLEANMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1266HOTSPOT_JNI_CALLBOOLEANMETHODA_RETURN(_ret_ref))1267DEFINE_CALLMETHODA(jbyte, Byte, T_BYTE1268, HOTSPOT_JNI_CALLBYTEMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1269HOTSPOT_JNI_CALLBYTEMETHODA_RETURN(_ret_ref))1270DEFINE_CALLMETHODA(jchar, Char, T_CHAR1271, HOTSPOT_JNI_CALLCHARMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1272HOTSPOT_JNI_CALLCHARMETHODA_RETURN(_ret_ref))1273DEFINE_CALLMETHODA(jshort, Short, T_SHORT1274, HOTSPOT_JNI_CALLSHORTMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1275HOTSPOT_JNI_CALLSHORTMETHODA_RETURN(_ret_ref))12761277DEFINE_CALLMETHODA(jobject, Object, T_OBJECT1278, HOTSPOT_JNI_CALLOBJECTMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1279HOTSPOT_JNI_CALLOBJECTMETHODA_RETURN(_ret_ref))1280DEFINE_CALLMETHODA(jint, Int, T_INT,1281HOTSPOT_JNI_CALLINTMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1282HOTSPOT_JNI_CALLINTMETHODA_RETURN(_ret_ref))1283DEFINE_CALLMETHODA(jlong, Long, T_LONG1284, HOTSPOT_JNI_CALLLONGMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1285HOTSPOT_JNI_CALLLONGMETHODA_RETURN(_ret_ref))1286// Float and double probes don't return value because dtrace doesn't currently support it1287DEFINE_CALLMETHODA(jfloat, Float, T_FLOAT1288, HOTSPOT_JNI_CALLFLOATMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1289HOTSPOT_JNI_CALLFLOATMETHODA_RETURN())1290DEFINE_CALLMETHODA(jdouble, Double, T_DOUBLE1291, HOTSPOT_JNI_CALLDOUBLEMETHODA_ENTRY(env, obj, (uintptr_t)methodID),1292HOTSPOT_JNI_CALLDOUBLEMETHODA_RETURN())12931294DT_VOID_RETURN_MARK_DECL(CallVoidMethod, HOTSPOT_JNI_CALLVOIDMETHOD_RETURN());1295DT_VOID_RETURN_MARK_DECL(CallVoidMethodV, HOTSPOT_JNI_CALLVOIDMETHODV_RETURN());1296DT_VOID_RETURN_MARK_DECL(CallVoidMethodA, HOTSPOT_JNI_CALLVOIDMETHODA_RETURN());129712981299JNI_ENTRY(void, jni_CallVoidMethod(JNIEnv *env, jobject obj, jmethodID methodID, ...))1300HOTSPOT_JNI_CALLVOIDMETHOD_ENTRY(env, obj, (uintptr_t) methodID);1301DT_VOID_RETURN_MARK(CallVoidMethod);13021303va_list args;1304va_start(args, methodID);1305JavaValue jvalue(T_VOID);1306JNI_ArgumentPusherVaArg ap(methodID, args);1307jni_invoke_nonstatic(env, &jvalue, obj, JNI_VIRTUAL, methodID, &ap, CHECK);1308va_end(args);1309JNI_END131013111312JNI_ENTRY(void, jni_CallVoidMethodV(JNIEnv *env, jobject obj, jmethodID methodID, va_list args))1313HOTSPOT_JNI_CALLVOIDMETHODV_ENTRY(env, obj, (uintptr_t) methodID);1314DT_VOID_RETURN_MARK(CallVoidMethodV);13151316JavaValue jvalue(T_VOID);1317JNI_ArgumentPusherVaArg ap(methodID, args);1318jni_invoke_nonstatic(env, &jvalue, obj, JNI_VIRTUAL, methodID, &ap, CHECK);1319JNI_END132013211322JNI_ENTRY(void, jni_CallVoidMethodA(JNIEnv *env, jobject obj, jmethodID methodID, const jvalue *args))1323HOTSPOT_JNI_CALLVOIDMETHODA_ENTRY(env, obj, (uintptr_t) methodID);1324DT_VOID_RETURN_MARK(CallVoidMethodA);13251326JavaValue jvalue(T_VOID);1327JNI_ArgumentPusherArray ap(methodID, args);1328jni_invoke_nonstatic(env, &jvalue, obj, JNI_VIRTUAL, methodID, &ap, CHECK);1329JNI_END1330133113321333#define DEFINE_CALLNONVIRTUALMETHOD(ResultType, Result, Tag \1334, EntryProbe, ReturnProbe) \1335\1336DT_RETURN_MARK_DECL_FOR(Result, CallNonvirtual##Result##Method, ResultType \1337, ReturnProbe);\1338\1339JNI_ENTRY(ResultType, \1340jni_CallNonvirtual##Result##Method(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, ...)) \1341\1342EntryProbe;\1343ResultType ret;\1344DT_RETURN_MARK_FOR(Result, CallNonvirtual##Result##Method, ResultType, \1345(const ResultType&)ret);\1346\1347va_list args; \1348va_start(args, methodID); \1349JavaValue jvalue(Tag); \1350JNI_ArgumentPusherVaArg ap(methodID, args); \1351jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_0); \1352va_end(args); \1353ret = jvalue.get_##ResultType(); \1354return ret;\1355JNI_END13561357// the runtime type of subword integral basic types is integer1358DEFINE_CALLNONVIRTUALMETHOD(jboolean, Boolean, T_BOOLEAN1359, HOTSPOT_JNI_CALLNONVIRTUALBOOLEANMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1360HOTSPOT_JNI_CALLNONVIRTUALBOOLEANMETHOD_RETURN(_ret_ref))1361DEFINE_CALLNONVIRTUALMETHOD(jbyte, Byte, T_BYTE1362, HOTSPOT_JNI_CALLNONVIRTUALBYTEMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1363HOTSPOT_JNI_CALLNONVIRTUALBYTEMETHOD_RETURN(_ret_ref))1364DEFINE_CALLNONVIRTUALMETHOD(jchar, Char, T_CHAR1365, HOTSPOT_JNI_CALLNONVIRTUALCHARMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1366HOTSPOT_JNI_CALLNONVIRTUALCHARMETHOD_RETURN(_ret_ref))1367DEFINE_CALLNONVIRTUALMETHOD(jshort, Short, T_SHORT1368, HOTSPOT_JNI_CALLNONVIRTUALSHORTMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1369HOTSPOT_JNI_CALLNONVIRTUALSHORTMETHOD_RETURN(_ret_ref))13701371DEFINE_CALLNONVIRTUALMETHOD(jobject, Object, T_OBJECT1372, HOTSPOT_JNI_CALLNONVIRTUALOBJECTMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1373HOTSPOT_JNI_CALLNONVIRTUALOBJECTMETHOD_RETURN(_ret_ref))1374DEFINE_CALLNONVIRTUALMETHOD(jint, Int, T_INT1375, HOTSPOT_JNI_CALLNONVIRTUALINTMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1376HOTSPOT_JNI_CALLNONVIRTUALINTMETHOD_RETURN(_ret_ref))1377DEFINE_CALLNONVIRTUALMETHOD(jlong, Long, T_LONG1378, HOTSPOT_JNI_CALLNONVIRTUALLONGMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1379// Float and double probes don't return value because dtrace doesn't currently support it1380HOTSPOT_JNI_CALLNONVIRTUALLONGMETHOD_RETURN(_ret_ref))1381DEFINE_CALLNONVIRTUALMETHOD(jfloat, Float, T_FLOAT1382, HOTSPOT_JNI_CALLNONVIRTUALFLOATMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1383HOTSPOT_JNI_CALLNONVIRTUALFLOATMETHOD_RETURN())1384DEFINE_CALLNONVIRTUALMETHOD(jdouble, Double, T_DOUBLE1385, HOTSPOT_JNI_CALLNONVIRTUALDOUBLEMETHOD_ENTRY(env, obj, cls, (uintptr_t)methodID),1386HOTSPOT_JNI_CALLNONVIRTUALDOUBLEMETHOD_RETURN())13871388#define DEFINE_CALLNONVIRTUALMETHODV(ResultType, Result, Tag \1389, EntryProbe, ReturnProbe) \1390\1391DT_RETURN_MARK_DECL_FOR(Result, CallNonvirtual##Result##MethodV, ResultType \1392, ReturnProbe);\1393\1394JNI_ENTRY(ResultType, \1395jni_CallNonvirtual##Result##MethodV(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, va_list args)) \1396\1397EntryProbe;\1398ResultType ret;\1399DT_RETURN_MARK_FOR(Result, CallNonvirtual##Result##MethodV, ResultType, \1400(const ResultType&)ret);\1401\1402JavaValue jvalue(Tag); \1403JNI_ArgumentPusherVaArg ap(methodID, args); \1404jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_0); \1405ret = jvalue.get_##ResultType(); \1406return ret;\1407JNI_END14081409// the runtime type of subword integral basic types is integer1410DEFINE_CALLNONVIRTUALMETHODV(jboolean, Boolean, T_BOOLEAN1411, HOTSPOT_JNI_CALLNONVIRTUALBOOLEANMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1412HOTSPOT_JNI_CALLNONVIRTUALBOOLEANMETHODV_RETURN(_ret_ref))1413DEFINE_CALLNONVIRTUALMETHODV(jbyte, Byte, T_BYTE1414, HOTSPOT_JNI_CALLNONVIRTUALBYTEMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1415HOTSPOT_JNI_CALLNONVIRTUALBYTEMETHODV_RETURN(_ret_ref))1416DEFINE_CALLNONVIRTUALMETHODV(jchar, Char, T_CHAR1417, HOTSPOT_JNI_CALLNONVIRTUALCHARMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1418HOTSPOT_JNI_CALLNONVIRTUALCHARMETHODV_RETURN(_ret_ref))1419DEFINE_CALLNONVIRTUALMETHODV(jshort, Short, T_SHORT1420, HOTSPOT_JNI_CALLNONVIRTUALSHORTMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1421HOTSPOT_JNI_CALLNONVIRTUALSHORTMETHODV_RETURN(_ret_ref))14221423DEFINE_CALLNONVIRTUALMETHODV(jobject, Object, T_OBJECT1424, HOTSPOT_JNI_CALLNONVIRTUALOBJECTMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1425HOTSPOT_JNI_CALLNONVIRTUALOBJECTMETHODV_RETURN(_ret_ref))1426DEFINE_CALLNONVIRTUALMETHODV(jint, Int, T_INT1427, HOTSPOT_JNI_CALLNONVIRTUALINTMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1428HOTSPOT_JNI_CALLNONVIRTUALINTMETHODV_RETURN(_ret_ref))1429DEFINE_CALLNONVIRTUALMETHODV(jlong, Long, T_LONG1430, HOTSPOT_JNI_CALLNONVIRTUALLONGMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1431// Float and double probes don't return value because dtrace doesn't currently support it1432HOTSPOT_JNI_CALLNONVIRTUALLONGMETHODV_RETURN(_ret_ref))1433DEFINE_CALLNONVIRTUALMETHODV(jfloat, Float, T_FLOAT1434, HOTSPOT_JNI_CALLNONVIRTUALFLOATMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1435HOTSPOT_JNI_CALLNONVIRTUALFLOATMETHODV_RETURN())1436DEFINE_CALLNONVIRTUALMETHODV(jdouble, Double, T_DOUBLE1437, HOTSPOT_JNI_CALLNONVIRTUALDOUBLEMETHODV_ENTRY(env, obj, cls, (uintptr_t)methodID),1438HOTSPOT_JNI_CALLNONVIRTUALDOUBLEMETHODV_RETURN())14391440#define DEFINE_CALLNONVIRTUALMETHODA(ResultType, Result, Tag \1441, EntryProbe, ReturnProbe) \1442\1443DT_RETURN_MARK_DECL_FOR(Result, CallNonvirtual##Result##MethodA, ResultType \1444, ReturnProbe);\1445\1446JNI_ENTRY(ResultType, \1447jni_CallNonvirtual##Result##MethodA(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, const jvalue *args)) \1448\1449EntryProbe;\1450ResultType ret;\1451DT_RETURN_MARK_FOR(Result, CallNonvirtual##Result##MethodA, ResultType, \1452(const ResultType&)ret);\1453\1454JavaValue jvalue(Tag); \1455JNI_ArgumentPusherArray ap(methodID, args); \1456jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_0); \1457ret = jvalue.get_##ResultType(); \1458return ret;\1459JNI_END14601461// the runtime type of subword integral basic types is integer1462DEFINE_CALLNONVIRTUALMETHODA(jboolean, Boolean, T_BOOLEAN1463, HOTSPOT_JNI_CALLNONVIRTUALBOOLEANMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1464HOTSPOT_JNI_CALLNONVIRTUALBOOLEANMETHODA_RETURN(_ret_ref))1465DEFINE_CALLNONVIRTUALMETHODA(jbyte, Byte, T_BYTE1466, HOTSPOT_JNI_CALLNONVIRTUALBYTEMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1467HOTSPOT_JNI_CALLNONVIRTUALBYTEMETHODA_RETURN(_ret_ref))1468DEFINE_CALLNONVIRTUALMETHODA(jchar, Char, T_CHAR1469, HOTSPOT_JNI_CALLNONVIRTUALCHARMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1470HOTSPOT_JNI_CALLNONVIRTUALCHARMETHODA_RETURN(_ret_ref))1471DEFINE_CALLNONVIRTUALMETHODA(jshort, Short, T_SHORT1472, HOTSPOT_JNI_CALLNONVIRTUALSHORTMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1473HOTSPOT_JNI_CALLNONVIRTUALSHORTMETHODA_RETURN(_ret_ref))14741475DEFINE_CALLNONVIRTUALMETHODA(jobject, Object, T_OBJECT1476, HOTSPOT_JNI_CALLNONVIRTUALOBJECTMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1477HOTSPOT_JNI_CALLNONVIRTUALOBJECTMETHODA_RETURN(_ret_ref))1478DEFINE_CALLNONVIRTUALMETHODA(jint, Int, T_INT1479, HOTSPOT_JNI_CALLNONVIRTUALINTMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1480HOTSPOT_JNI_CALLNONVIRTUALINTMETHODA_RETURN(_ret_ref))1481DEFINE_CALLNONVIRTUALMETHODA(jlong, Long, T_LONG1482, HOTSPOT_JNI_CALLNONVIRTUALLONGMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1483// Float and double probes don't return value because dtrace doesn't currently support it1484HOTSPOT_JNI_CALLNONVIRTUALLONGMETHODA_RETURN(_ret_ref))1485DEFINE_CALLNONVIRTUALMETHODA(jfloat, Float, T_FLOAT1486, HOTSPOT_JNI_CALLNONVIRTUALFLOATMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1487HOTSPOT_JNI_CALLNONVIRTUALFLOATMETHODA_RETURN())1488DEFINE_CALLNONVIRTUALMETHODA(jdouble, Double, T_DOUBLE1489, HOTSPOT_JNI_CALLNONVIRTUALDOUBLEMETHODA_ENTRY(env, obj, cls, (uintptr_t)methodID),1490HOTSPOT_JNI_CALLNONVIRTUALDOUBLEMETHODA_RETURN())14911492DT_VOID_RETURN_MARK_DECL(CallNonvirtualVoidMethod1493, HOTSPOT_JNI_CALLNONVIRTUALVOIDMETHOD_RETURN());1494DT_VOID_RETURN_MARK_DECL(CallNonvirtualVoidMethodV1495, HOTSPOT_JNI_CALLNONVIRTUALVOIDMETHODV_RETURN());1496DT_VOID_RETURN_MARK_DECL(CallNonvirtualVoidMethodA1497, HOTSPOT_JNI_CALLNONVIRTUALVOIDMETHODA_RETURN());14981499JNI_ENTRY(void, jni_CallNonvirtualVoidMethod(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, ...))1500HOTSPOT_JNI_CALLNONVIRTUALVOIDMETHOD_ENTRY(env, obj, cls, (uintptr_t) methodID);1501DT_VOID_RETURN_MARK(CallNonvirtualVoidMethod);15021503va_list args;1504va_start(args, methodID);1505JavaValue jvalue(T_VOID);1506JNI_ArgumentPusherVaArg ap(methodID, args);1507jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK);1508va_end(args);1509JNI_END151015111512JNI_ENTRY(void, jni_CallNonvirtualVoidMethodV(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, va_list args))1513HOTSPOT_JNI_CALLNONVIRTUALVOIDMETHODV_ENTRY(1514env, obj, cls, (uintptr_t) methodID);1515DT_VOID_RETURN_MARK(CallNonvirtualVoidMethodV);15161517JavaValue jvalue(T_VOID);1518JNI_ArgumentPusherVaArg ap(methodID, args);1519jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK);1520JNI_END152115221523JNI_ENTRY(void, jni_CallNonvirtualVoidMethodA(JNIEnv *env, jobject obj, jclass cls, jmethodID methodID, const jvalue *args))1524HOTSPOT_JNI_CALLNONVIRTUALVOIDMETHODA_ENTRY(1525env, obj, cls, (uintptr_t) methodID);1526DT_VOID_RETURN_MARK(CallNonvirtualVoidMethodA);1527JavaValue jvalue(T_VOID);1528JNI_ArgumentPusherArray ap(methodID, args);1529jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK);1530JNI_END1531153215331534#define DEFINE_CALLSTATICMETHOD(ResultType, Result, Tag \1535, EntryProbe, ResultProbe) \1536\1537DT_RETURN_MARK_DECL_FOR(Result, CallStatic##Result##Method, ResultType \1538, ResultProbe); \1539\1540JNI_ENTRY(ResultType, \1541jni_CallStatic##Result##Method(JNIEnv *env, jclass cls, jmethodID methodID, ...)) \1542\1543EntryProbe; \1544ResultType ret = 0;\1545DT_RETURN_MARK_FOR(Result, CallStatic##Result##Method, ResultType, \1546(const ResultType&)ret);\1547\1548va_list args; \1549va_start(args, methodID); \1550JavaValue jvalue(Tag); \1551JNI_ArgumentPusherVaArg ap(methodID, args); \1552jni_invoke_static(env, &jvalue, NULL, JNI_STATIC, methodID, &ap, CHECK_0); \1553va_end(args); \1554ret = jvalue.get_##ResultType(); \1555return ret;\1556JNI_END15571558// the runtime type of subword integral basic types is integer1559DEFINE_CALLSTATICMETHOD(jboolean, Boolean, T_BOOLEAN1560, HOTSPOT_JNI_CALLSTATICBOOLEANMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1561HOTSPOT_JNI_CALLSTATICBOOLEANMETHOD_RETURN(_ret_ref));1562DEFINE_CALLSTATICMETHOD(jbyte, Byte, T_BYTE1563, HOTSPOT_JNI_CALLSTATICBYTEMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1564HOTSPOT_JNI_CALLSTATICBYTEMETHOD_RETURN(_ret_ref));1565DEFINE_CALLSTATICMETHOD(jchar, Char, T_CHAR1566, HOTSPOT_JNI_CALLSTATICCHARMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1567HOTSPOT_JNI_CALLSTATICCHARMETHOD_RETURN(_ret_ref));1568DEFINE_CALLSTATICMETHOD(jshort, Short, T_SHORT1569, HOTSPOT_JNI_CALLSTATICSHORTMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1570HOTSPOT_JNI_CALLSTATICSHORTMETHOD_RETURN(_ret_ref));15711572DEFINE_CALLSTATICMETHOD(jobject, Object, T_OBJECT1573, HOTSPOT_JNI_CALLSTATICOBJECTMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1574HOTSPOT_JNI_CALLSTATICOBJECTMETHOD_RETURN(_ret_ref));1575DEFINE_CALLSTATICMETHOD(jint, Int, T_INT1576, HOTSPOT_JNI_CALLSTATICINTMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1577HOTSPOT_JNI_CALLSTATICINTMETHOD_RETURN(_ret_ref));1578DEFINE_CALLSTATICMETHOD(jlong, Long, T_LONG1579, HOTSPOT_JNI_CALLSTATICLONGMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1580HOTSPOT_JNI_CALLSTATICLONGMETHOD_RETURN(_ret_ref));1581// Float and double probes don't return value because dtrace doesn't currently support it1582DEFINE_CALLSTATICMETHOD(jfloat, Float, T_FLOAT1583, HOTSPOT_JNI_CALLSTATICFLOATMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1584HOTSPOT_JNI_CALLSTATICFLOATMETHOD_RETURN());1585DEFINE_CALLSTATICMETHOD(jdouble, Double, T_DOUBLE1586, HOTSPOT_JNI_CALLSTATICDOUBLEMETHOD_ENTRY(env, cls, (uintptr_t)methodID),1587HOTSPOT_JNI_CALLSTATICDOUBLEMETHOD_RETURN());15881589#define DEFINE_CALLSTATICMETHODV(ResultType, Result, Tag \1590, EntryProbe, ResultProbe) \1591\1592DT_RETURN_MARK_DECL_FOR(Result, CallStatic##Result##MethodV, ResultType \1593, ResultProbe); \1594\1595JNI_ENTRY(ResultType, \1596jni_CallStatic##Result##MethodV(JNIEnv *env, jclass cls, jmethodID methodID, va_list args)) \1597\1598EntryProbe; \1599ResultType ret = 0;\1600DT_RETURN_MARK_FOR(Result, CallStatic##Result##MethodV, ResultType, \1601(const ResultType&)ret);\1602\1603JavaValue jvalue(Tag); \1604JNI_ArgumentPusherVaArg ap(methodID, args); \1605/* Make sure class is initialized before trying to invoke its method */ \1606Klass* k = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(cls)); \1607k->initialize(CHECK_0); \1608jni_invoke_static(env, &jvalue, NULL, JNI_STATIC, methodID, &ap, CHECK_0); \1609va_end(args); \1610ret = jvalue.get_##ResultType(); \1611return ret;\1612JNI_END16131614// the runtime type of subword integral basic types is integer1615DEFINE_CALLSTATICMETHODV(jboolean, Boolean, T_BOOLEAN1616, HOTSPOT_JNI_CALLSTATICBOOLEANMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1617HOTSPOT_JNI_CALLSTATICBOOLEANMETHODV_RETURN(_ret_ref));1618DEFINE_CALLSTATICMETHODV(jbyte, Byte, T_BYTE1619, HOTSPOT_JNI_CALLSTATICBYTEMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1620HOTSPOT_JNI_CALLSTATICBYTEMETHODV_RETURN(_ret_ref));1621DEFINE_CALLSTATICMETHODV(jchar, Char, T_CHAR1622, HOTSPOT_JNI_CALLSTATICCHARMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1623HOTSPOT_JNI_CALLSTATICCHARMETHODV_RETURN(_ret_ref));1624DEFINE_CALLSTATICMETHODV(jshort, Short, T_SHORT1625, HOTSPOT_JNI_CALLSTATICSHORTMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1626HOTSPOT_JNI_CALLSTATICSHORTMETHODV_RETURN(_ret_ref));16271628DEFINE_CALLSTATICMETHODV(jobject, Object, T_OBJECT1629, HOTSPOT_JNI_CALLSTATICOBJECTMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1630HOTSPOT_JNI_CALLSTATICOBJECTMETHODV_RETURN(_ret_ref));1631DEFINE_CALLSTATICMETHODV(jint, Int, T_INT1632, HOTSPOT_JNI_CALLSTATICINTMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1633HOTSPOT_JNI_CALLSTATICINTMETHODV_RETURN(_ret_ref));1634DEFINE_CALLSTATICMETHODV(jlong, Long, T_LONG1635, HOTSPOT_JNI_CALLSTATICLONGMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1636HOTSPOT_JNI_CALLSTATICLONGMETHODV_RETURN(_ret_ref));1637// Float and double probes don't return value because dtrace doesn't currently support it1638DEFINE_CALLSTATICMETHODV(jfloat, Float, T_FLOAT1639, HOTSPOT_JNI_CALLSTATICFLOATMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1640HOTSPOT_JNI_CALLSTATICFLOATMETHODV_RETURN());1641DEFINE_CALLSTATICMETHODV(jdouble, Double, T_DOUBLE1642, HOTSPOT_JNI_CALLSTATICDOUBLEMETHODV_ENTRY(env, cls, (uintptr_t)methodID),1643HOTSPOT_JNI_CALLSTATICDOUBLEMETHODV_RETURN());16441645#define DEFINE_CALLSTATICMETHODA(ResultType, Result, Tag \1646, EntryProbe, ResultProbe) \1647\1648DT_RETURN_MARK_DECL_FOR(Result, CallStatic##Result##MethodA, ResultType \1649, ResultProbe); \1650\1651JNI_ENTRY(ResultType, \1652jni_CallStatic##Result##MethodA(JNIEnv *env, jclass cls, jmethodID methodID, const jvalue *args)) \1653\1654EntryProbe; \1655ResultType ret = 0;\1656DT_RETURN_MARK_FOR(Result, CallStatic##Result##MethodA, ResultType, \1657(const ResultType&)ret);\1658\1659JavaValue jvalue(Tag); \1660JNI_ArgumentPusherArray ap(methodID, args); \1661jni_invoke_static(env, &jvalue, NULL, JNI_STATIC, methodID, &ap, CHECK_0); \1662ret = jvalue.get_##ResultType(); \1663return ret;\1664JNI_END16651666// the runtime type of subword integral basic types is integer1667DEFINE_CALLSTATICMETHODA(jboolean, Boolean, T_BOOLEAN1668, HOTSPOT_JNI_CALLSTATICBOOLEANMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1669HOTSPOT_JNI_CALLSTATICBOOLEANMETHODA_RETURN(_ret_ref));1670DEFINE_CALLSTATICMETHODA(jbyte, Byte, T_BYTE1671, HOTSPOT_JNI_CALLSTATICBYTEMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1672HOTSPOT_JNI_CALLSTATICBYTEMETHODA_RETURN(_ret_ref));1673DEFINE_CALLSTATICMETHODA(jchar, Char, T_CHAR1674, HOTSPOT_JNI_CALLSTATICCHARMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1675HOTSPOT_JNI_CALLSTATICCHARMETHODA_RETURN(_ret_ref));1676DEFINE_CALLSTATICMETHODA(jshort, Short, T_SHORT1677, HOTSPOT_JNI_CALLSTATICSHORTMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1678HOTSPOT_JNI_CALLSTATICSHORTMETHODA_RETURN(_ret_ref));16791680DEFINE_CALLSTATICMETHODA(jobject, Object, T_OBJECT1681, HOTSPOT_JNI_CALLSTATICOBJECTMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1682HOTSPOT_JNI_CALLSTATICOBJECTMETHODA_RETURN(_ret_ref));1683DEFINE_CALLSTATICMETHODA(jint, Int, T_INT1684, HOTSPOT_JNI_CALLSTATICINTMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1685HOTSPOT_JNI_CALLSTATICINTMETHODA_RETURN(_ret_ref));1686DEFINE_CALLSTATICMETHODA(jlong, Long, T_LONG1687, HOTSPOT_JNI_CALLSTATICLONGMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1688HOTSPOT_JNI_CALLSTATICLONGMETHODA_RETURN(_ret_ref));1689// Float and double probes don't return value because dtrace doesn't currently support it1690DEFINE_CALLSTATICMETHODA(jfloat, Float, T_FLOAT1691, HOTSPOT_JNI_CALLSTATICFLOATMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1692HOTSPOT_JNI_CALLSTATICFLOATMETHODA_RETURN());1693DEFINE_CALLSTATICMETHODA(jdouble, Double, T_DOUBLE1694, HOTSPOT_JNI_CALLSTATICDOUBLEMETHODA_ENTRY(env, cls, (uintptr_t)methodID),1695HOTSPOT_JNI_CALLSTATICDOUBLEMETHODA_RETURN());16961697DT_VOID_RETURN_MARK_DECL(CallStaticVoidMethod1698, HOTSPOT_JNI_CALLSTATICVOIDMETHOD_RETURN());1699DT_VOID_RETURN_MARK_DECL(CallStaticVoidMethodV1700, HOTSPOT_JNI_CALLSTATICVOIDMETHODV_RETURN());1701DT_VOID_RETURN_MARK_DECL(CallStaticVoidMethodA1702, HOTSPOT_JNI_CALLSTATICVOIDMETHODA_RETURN());17031704JNI_ENTRY(void, jni_CallStaticVoidMethod(JNIEnv *env, jclass cls, jmethodID methodID, ...))1705HOTSPOT_JNI_CALLSTATICVOIDMETHOD_ENTRY(env, cls, (uintptr_t) methodID);1706DT_VOID_RETURN_MARK(CallStaticVoidMethod);17071708va_list args;1709va_start(args, methodID);1710JavaValue jvalue(T_VOID);1711JNI_ArgumentPusherVaArg ap(methodID, args);1712jni_invoke_static(env, &jvalue, NULL, JNI_STATIC, methodID, &ap, CHECK);1713va_end(args);1714JNI_END171517161717JNI_ENTRY(void, jni_CallStaticVoidMethodV(JNIEnv *env, jclass cls, jmethodID methodID, va_list args))1718HOTSPOT_JNI_CALLSTATICVOIDMETHODV_ENTRY(env, cls, (uintptr_t) methodID);1719DT_VOID_RETURN_MARK(CallStaticVoidMethodV);17201721JavaValue jvalue(T_VOID);1722JNI_ArgumentPusherVaArg ap(methodID, args);1723jni_invoke_static(env, &jvalue, NULL, JNI_STATIC, methodID, &ap, CHECK);1724JNI_END172517261727JNI_ENTRY(void, jni_CallStaticVoidMethodA(JNIEnv *env, jclass cls, jmethodID methodID, const jvalue *args))1728HOTSPOT_JNI_CALLSTATICVOIDMETHODA_ENTRY(env, cls, (uintptr_t) methodID);1729DT_VOID_RETURN_MARK(CallStaticVoidMethodA);17301731JavaValue jvalue(T_VOID);1732JNI_ArgumentPusherArray ap(methodID, args);1733jni_invoke_static(env, &jvalue, NULL, JNI_STATIC, methodID, &ap, CHECK);1734JNI_END173517361737//1738// Accessing Fields1739//174017411742DT_RETURN_MARK_DECL(GetFieldID, jfieldID1743, HOTSPOT_JNI_GETFIELDID_RETURN((uintptr_t)_ret_ref));17441745JNI_ENTRY(jfieldID, jni_GetFieldID(JNIEnv *env, jclass clazz,1746const char *name, const char *sig))1747HOTSPOT_JNI_GETFIELDID_ENTRY(env, clazz, (char *) name, (char *) sig);1748jfieldID ret = 0;1749DT_RETURN_MARK(GetFieldID, jfieldID, (const jfieldID&)ret);17501751Klass* k = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(clazz));17521753// The class should have been loaded (we have an instance of the class1754// passed in) so the field and signature should already be in the symbol1755// table. If they're not there, the field doesn't exist.1756TempNewSymbol fieldname = SymbolTable::probe(name, (int)strlen(name));1757TempNewSymbol signame = SymbolTable::probe(sig, (int)strlen(sig));1758if (fieldname == NULL || signame == NULL) {1759ResourceMark rm;1760THROW_MSG_0(vmSymbols::java_lang_NoSuchFieldError(), err_msg("%s.%s %s", k->external_name(), name, sig));1761}17621763// Make sure class is initialized before handing id's out to fields1764k->initialize(CHECK_NULL);17651766fieldDescriptor fd;1767if (!k->is_instance_klass() ||1768!InstanceKlass::cast(k)->find_field(fieldname, signame, false, &fd)) {1769ResourceMark rm;1770THROW_MSG_0(vmSymbols::java_lang_NoSuchFieldError(), err_msg("%s.%s %s", k->external_name(), name, sig));1771}17721773// A jfieldID for a non-static field is simply the offset of the field within the instanceOop1774// It may also have hash bits for k, if VerifyJNIFields is turned on.1775ret = jfieldIDWorkaround::to_instance_jfieldID(k, fd.offset());1776return ret;1777JNI_END177817791780JNI_ENTRY(jobject, jni_GetObjectField(JNIEnv *env, jobject obj, jfieldID fieldID))1781HOTSPOT_JNI_GETOBJECTFIELD_ENTRY(env, obj, (uintptr_t) fieldID);1782oop o = JNIHandles::resolve_non_null(obj);1783Klass* k = o->klass();1784int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID);1785// Keep JVMTI addition small and only check enabled flag here.1786// jni_GetField_probe() assumes that is okay to create handles.1787if (JvmtiExport::should_post_field_access()) {1788o = JvmtiExport::jni_GetField_probe(thread, obj, o, k, fieldID, false);1789}1790oop loaded_obj = HeapAccess<ON_UNKNOWN_OOP_REF>::oop_load_at(o, offset);1791jobject ret = JNIHandles::make_local(THREAD, loaded_obj);1792HOTSPOT_JNI_GETOBJECTFIELD_RETURN(ret);1793return ret;1794JNI_END1795179617971798#define DEFINE_GETFIELD(Return,Fieldname,Result \1799, EntryProbe, ReturnProbe) \1800\1801DT_RETURN_MARK_DECL_FOR(Result, Get##Result##Field, Return \1802, ReturnProbe); \1803\1804JNI_ENTRY_NO_PRESERVE(Return, jni_Get##Result##Field(JNIEnv *env, jobject obj, jfieldID fieldID)) \1805\1806EntryProbe; \1807Return ret = 0;\1808DT_RETURN_MARK_FOR(Result, Get##Result##Field, Return, (const Return&)ret);\1809\1810oop o = JNIHandles::resolve_non_null(obj); \1811Klass* k = o->klass(); \1812int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID); \1813/* Keep JVMTI addition small and only check enabled flag here. */ \1814if (JvmtiExport::should_post_field_access()) { \1815o = JvmtiExport::jni_GetField_probe(thread, obj, o, k, fieldID, false); \1816} \1817ret = o->Fieldname##_field(offset); \1818return ret; \1819JNI_END18201821DEFINE_GETFIELD(jboolean, bool, Boolean1822, HOTSPOT_JNI_GETBOOLEANFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1823HOTSPOT_JNI_GETBOOLEANFIELD_RETURN(_ret_ref))1824DEFINE_GETFIELD(jbyte, byte, Byte1825, HOTSPOT_JNI_GETBYTEFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1826HOTSPOT_JNI_GETBYTEFIELD_RETURN(_ret_ref))1827DEFINE_GETFIELD(jchar, char, Char1828, HOTSPOT_JNI_GETCHARFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1829HOTSPOT_JNI_GETCHARFIELD_RETURN(_ret_ref))1830DEFINE_GETFIELD(jshort, short, Short1831, HOTSPOT_JNI_GETSHORTFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1832HOTSPOT_JNI_GETSHORTFIELD_RETURN(_ret_ref))1833DEFINE_GETFIELD(jint, int, Int1834, HOTSPOT_JNI_GETINTFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1835HOTSPOT_JNI_GETINTFIELD_RETURN(_ret_ref))1836DEFINE_GETFIELD(jlong, long, Long1837, HOTSPOT_JNI_GETLONGFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1838HOTSPOT_JNI_GETLONGFIELD_RETURN(_ret_ref))1839// Float and double probes don't return value because dtrace doesn't currently support it1840DEFINE_GETFIELD(jfloat, float, Float1841, HOTSPOT_JNI_GETFLOATFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1842HOTSPOT_JNI_GETFLOATFIELD_RETURN())1843DEFINE_GETFIELD(jdouble, double, Double1844, HOTSPOT_JNI_GETDOUBLEFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1845HOTSPOT_JNI_GETDOUBLEFIELD_RETURN())18461847address jni_GetBooleanField_addr() {1848return (address)jni_GetBooleanField;1849}1850address jni_GetByteField_addr() {1851return (address)jni_GetByteField;1852}1853address jni_GetCharField_addr() {1854return (address)jni_GetCharField;1855}1856address jni_GetShortField_addr() {1857return (address)jni_GetShortField;1858}1859address jni_GetIntField_addr() {1860return (address)jni_GetIntField;1861}1862address jni_GetLongField_addr() {1863return (address)jni_GetLongField;1864}1865address jni_GetFloatField_addr() {1866return (address)jni_GetFloatField;1867}1868address jni_GetDoubleField_addr() {1869return (address)jni_GetDoubleField;1870}18711872JNI_ENTRY_NO_PRESERVE(void, jni_SetObjectField(JNIEnv *env, jobject obj, jfieldID fieldID, jobject value))1873HOTSPOT_JNI_SETOBJECTFIELD_ENTRY(env, obj, (uintptr_t) fieldID, value);1874oop o = JNIHandles::resolve_non_null(obj);1875Klass* k = o->klass();1876int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID);1877// Keep JVMTI addition small and only check enabled flag here.1878if (JvmtiExport::should_post_field_modification()) {1879jvalue field_value;1880field_value.l = value;1881o = JvmtiExport::jni_SetField_probe(thread, obj, o, k, fieldID, false, JVM_SIGNATURE_CLASS, (jvalue *)&field_value);1882}1883HeapAccess<ON_UNKNOWN_OOP_REF>::oop_store_at(o, offset, JNIHandles::resolve(value));1884HOTSPOT_JNI_SETOBJECTFIELD_RETURN();1885JNI_END188618871888#define DEFINE_SETFIELD(Argument,Fieldname,Result,SigType,unionType \1889, EntryProbe, ReturnProbe) \1890\1891JNI_ENTRY_NO_PRESERVE(void, jni_Set##Result##Field(JNIEnv *env, jobject obj, jfieldID fieldID, Argument value)) \1892\1893EntryProbe; \1894\1895oop o = JNIHandles::resolve_non_null(obj); \1896Klass* k = o->klass(); \1897int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID); \1898/* Keep JVMTI addition small and only check enabled flag here. */ \1899if (JvmtiExport::should_post_field_modification()) { \1900jvalue field_value; \1901field_value.unionType = value; \1902o = JvmtiExport::jni_SetField_probe(thread, obj, o, k, fieldID, false, SigType, (jvalue *)&field_value); \1903} \1904if (SigType == JVM_SIGNATURE_BOOLEAN) { value = ((jboolean)value) & 1; } \1905o->Fieldname##_field_put(offset, value); \1906ReturnProbe; \1907JNI_END19081909DEFINE_SETFIELD(jboolean, bool, Boolean, JVM_SIGNATURE_BOOLEAN, z1910, HOTSPOT_JNI_SETBOOLEANFIELD_ENTRY(env, obj, (uintptr_t)fieldID, value),1911HOTSPOT_JNI_SETBOOLEANFIELD_RETURN())1912DEFINE_SETFIELD(jbyte, byte, Byte, JVM_SIGNATURE_BYTE, b1913, HOTSPOT_JNI_SETBYTEFIELD_ENTRY(env, obj, (uintptr_t)fieldID, value),1914HOTSPOT_JNI_SETBYTEFIELD_RETURN())1915DEFINE_SETFIELD(jchar, char, Char, JVM_SIGNATURE_CHAR, c1916, HOTSPOT_JNI_SETCHARFIELD_ENTRY(env, obj, (uintptr_t)fieldID, value),1917HOTSPOT_JNI_SETCHARFIELD_RETURN())1918DEFINE_SETFIELD(jshort, short, Short, JVM_SIGNATURE_SHORT, s1919, HOTSPOT_JNI_SETSHORTFIELD_ENTRY(env, obj, (uintptr_t)fieldID, value),1920HOTSPOT_JNI_SETSHORTFIELD_RETURN())1921DEFINE_SETFIELD(jint, int, Int, JVM_SIGNATURE_INT, i1922, HOTSPOT_JNI_SETINTFIELD_ENTRY(env, obj, (uintptr_t)fieldID, value),1923HOTSPOT_JNI_SETINTFIELD_RETURN())1924DEFINE_SETFIELD(jlong, long, Long, JVM_SIGNATURE_LONG, j1925, HOTSPOT_JNI_SETLONGFIELD_ENTRY(env, obj, (uintptr_t)fieldID, value),1926HOTSPOT_JNI_SETLONGFIELD_RETURN())1927// Float and double probes don't return value because dtrace doesn't currently support it1928DEFINE_SETFIELD(jfloat, float, Float, JVM_SIGNATURE_FLOAT, f1929, HOTSPOT_JNI_SETFLOATFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1930HOTSPOT_JNI_SETFLOATFIELD_RETURN())1931DEFINE_SETFIELD(jdouble, double, Double, JVM_SIGNATURE_DOUBLE, d1932, HOTSPOT_JNI_SETDOUBLEFIELD_ENTRY(env, obj, (uintptr_t)fieldID),1933HOTSPOT_JNI_SETDOUBLEFIELD_RETURN())19341935DT_RETURN_MARK_DECL(ToReflectedField, jobject1936, HOTSPOT_JNI_TOREFLECTEDFIELD_RETURN(_ret_ref));19371938JNI_ENTRY(jobject, jni_ToReflectedField(JNIEnv *env, jclass cls, jfieldID fieldID, jboolean isStatic))1939HOTSPOT_JNI_TOREFLECTEDFIELD_ENTRY(env, cls, (uintptr_t) fieldID, isStatic);1940jobject ret = NULL;1941DT_RETURN_MARK(ToReflectedField, jobject, (const jobject&)ret);19421943fieldDescriptor fd;1944bool found = false;1945Klass* k = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(cls));19461947assert(jfieldIDWorkaround::is_static_jfieldID(fieldID) == (isStatic != 0), "invalid fieldID");19481949if (isStatic) {1950// Static field. The fieldID a JNIid specifying the field holder and the offset within the Klass*.1951JNIid* id = jfieldIDWorkaround::from_static_jfieldID(fieldID);1952assert(id->is_static_field_id(), "invalid static field id");1953found = id->find_local_field(&fd);1954} else {1955// Non-static field. The fieldID is really the offset of the field within the instanceOop.1956int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID);1957found = InstanceKlass::cast(k)->find_field_from_offset(offset, false, &fd);1958}1959assert(found, "bad fieldID passed into jni_ToReflectedField");1960oop reflected = Reflection::new_field(&fd, CHECK_NULL);1961ret = JNIHandles::make_local(THREAD, reflected);1962return ret;1963JNI_END196419651966//1967// Accessing Static Fields1968//1969DT_RETURN_MARK_DECL(GetStaticFieldID, jfieldID1970, HOTSPOT_JNI_GETSTATICFIELDID_RETURN((uintptr_t)_ret_ref));19711972JNI_ENTRY(jfieldID, jni_GetStaticFieldID(JNIEnv *env, jclass clazz,1973const char *name, const char *sig))1974HOTSPOT_JNI_GETSTATICFIELDID_ENTRY(env, clazz, (char *) name, (char *) sig);1975jfieldID ret = NULL;1976DT_RETURN_MARK(GetStaticFieldID, jfieldID, (const jfieldID&)ret);19771978// The class should have been loaded (we have an instance of the class1979// passed in) so the field and signature should already be in the symbol1980// table. If they're not there, the field doesn't exist.1981TempNewSymbol fieldname = SymbolTable::probe(name, (int)strlen(name));1982TempNewSymbol signame = SymbolTable::probe(sig, (int)strlen(sig));1983if (fieldname == NULL || signame == NULL) {1984THROW_MSG_0(vmSymbols::java_lang_NoSuchFieldError(), (char*) name);1985}1986Klass* k = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(clazz));1987// Make sure class is initialized before handing id's out to static fields1988k->initialize(CHECK_NULL);19891990fieldDescriptor fd;1991if (!k->is_instance_klass() ||1992!InstanceKlass::cast(k)->find_field(fieldname, signame, true, &fd)) {1993THROW_MSG_0(vmSymbols::java_lang_NoSuchFieldError(), (char*) name);1994}19951996// A jfieldID for a static field is a JNIid specifying the field holder and the offset within the Klass*1997JNIid* id = fd.field_holder()->jni_id_for(fd.offset());1998debug_only(id->set_is_static_field_id();)19992000debug_only(id->verify(fd.field_holder()));20012002ret = jfieldIDWorkaround::to_static_jfieldID(id);2003return ret;2004JNI_END200520062007JNI_ENTRY(jobject, jni_GetStaticObjectField(JNIEnv *env, jclass clazz, jfieldID fieldID))2008HOTSPOT_JNI_GETSTATICOBJECTFIELD_ENTRY(env, clazz, (uintptr_t) fieldID);2009#if INCLUDE_JNI_CHECK2010DEBUG_ONLY(Klass* param_k = jniCheck::validate_class(thread, clazz);)2011#endif // INCLUDE_JNI_CHECK2012JNIid* id = jfieldIDWorkaround::from_static_jfieldID(fieldID);2013assert(id->is_static_field_id(), "invalid static field id");2014// Keep JVMTI addition small and only check enabled flag here.2015// jni_GetField_probe() assumes that is okay to create handles.2016if (JvmtiExport::should_post_field_access()) {2017JvmtiExport::jni_GetField_probe(thread, NULL, NULL, id->holder(), fieldID, true);2018}2019jobject ret = JNIHandles::make_local(THREAD, id->holder()->java_mirror()->obj_field(id->offset()));2020HOTSPOT_JNI_GETSTATICOBJECTFIELD_RETURN(ret);2021return ret;2022JNI_END202320242025#define DEFINE_GETSTATICFIELD(Return,Fieldname,Result \2026, EntryProbe, ReturnProbe) \2027\2028DT_RETURN_MARK_DECL_FOR(Result, GetStatic##Result##Field, Return \2029, ReturnProbe); \2030\2031JNI_ENTRY(Return, jni_GetStatic##Result##Field(JNIEnv *env, jclass clazz, jfieldID fieldID)) \2032EntryProbe; \2033Return ret = 0;\2034DT_RETURN_MARK_FOR(Result, GetStatic##Result##Field, Return, \2035(const Return&)ret);\2036JNIid* id = jfieldIDWorkaround::from_static_jfieldID(fieldID); \2037assert(id->is_static_field_id(), "invalid static field id"); \2038/* Keep JVMTI addition small and only check enabled flag here. */ \2039/* jni_GetField_probe() assumes that is okay to create handles. */ \2040if (JvmtiExport::should_post_field_access()) { \2041JvmtiExport::jni_GetField_probe(thread, NULL, NULL, id->holder(), fieldID, true); \2042} \2043ret = id->holder()->java_mirror()-> Fieldname##_field (id->offset()); \2044return ret;\2045JNI_END20462047DEFINE_GETSTATICFIELD(jboolean, bool, Boolean2048, HOTSPOT_JNI_GETSTATICBOOLEANFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICBOOLEANFIELD_RETURN(_ret_ref))2049DEFINE_GETSTATICFIELD(jbyte, byte, Byte2050, HOTSPOT_JNI_GETSTATICBYTEFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICBYTEFIELD_RETURN(_ret_ref) )2051DEFINE_GETSTATICFIELD(jchar, char, Char2052, HOTSPOT_JNI_GETSTATICCHARFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICCHARFIELD_RETURN(_ret_ref) )2053DEFINE_GETSTATICFIELD(jshort, short, Short2054, HOTSPOT_JNI_GETSTATICSHORTFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICSHORTFIELD_RETURN(_ret_ref) )2055DEFINE_GETSTATICFIELD(jint, int, Int2056, HOTSPOT_JNI_GETSTATICINTFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICINTFIELD_RETURN(_ret_ref) )2057DEFINE_GETSTATICFIELD(jlong, long, Long2058, HOTSPOT_JNI_GETSTATICLONGFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICLONGFIELD_RETURN(_ret_ref) )2059// Float and double probes don't return value because dtrace doesn't currently support it2060DEFINE_GETSTATICFIELD(jfloat, float, Float2061, HOTSPOT_JNI_GETSTATICFLOATFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICFLOATFIELD_RETURN() )2062DEFINE_GETSTATICFIELD(jdouble, double, Double2063, HOTSPOT_JNI_GETSTATICDOUBLEFIELD_ENTRY(env, clazz, (uintptr_t) fieldID), HOTSPOT_JNI_GETSTATICDOUBLEFIELD_RETURN() )20642065JNI_ENTRY(void, jni_SetStaticObjectField(JNIEnv *env, jclass clazz, jfieldID fieldID, jobject value))2066HOTSPOT_JNI_SETSTATICOBJECTFIELD_ENTRY(env, clazz, (uintptr_t) fieldID, value);2067JNIid* id = jfieldIDWorkaround::from_static_jfieldID(fieldID);2068assert(id->is_static_field_id(), "invalid static field id");2069// Keep JVMTI addition small and only check enabled flag here.2070// jni_SetField_probe() assumes that is okay to create handles.2071if (JvmtiExport::should_post_field_modification()) {2072jvalue field_value;2073field_value.l = value;2074JvmtiExport::jni_SetField_probe(thread, NULL, NULL, id->holder(), fieldID, true, JVM_SIGNATURE_CLASS, (jvalue *)&field_value);2075}2076id->holder()->java_mirror()->obj_field_put(id->offset(), JNIHandles::resolve(value));2077HOTSPOT_JNI_SETSTATICOBJECTFIELD_RETURN();2078JNI_END2079208020812082#define DEFINE_SETSTATICFIELD(Argument,Fieldname,Result,SigType,unionType \2083, EntryProbe, ReturnProbe) \2084\2085JNI_ENTRY(void, jni_SetStatic##Result##Field(JNIEnv *env, jclass clazz, jfieldID fieldID, Argument value)) \2086EntryProbe; \2087\2088JNIid* id = jfieldIDWorkaround::from_static_jfieldID(fieldID); \2089assert(id->is_static_field_id(), "invalid static field id"); \2090/* Keep JVMTI addition small and only check enabled flag here. */ \2091/* jni_SetField_probe() assumes that is okay to create handles. */ \2092if (JvmtiExport::should_post_field_modification()) { \2093jvalue field_value; \2094field_value.unionType = value; \2095JvmtiExport::jni_SetField_probe(thread, NULL, NULL, id->holder(), fieldID, true, SigType, (jvalue *)&field_value); \2096} \2097if (SigType == JVM_SIGNATURE_BOOLEAN) { value = ((jboolean)value) & 1; } \2098id->holder()->java_mirror()-> Fieldname##_field_put (id->offset(), value); \2099ReturnProbe;\2100JNI_END21012102DEFINE_SETSTATICFIELD(jboolean, bool, Boolean, JVM_SIGNATURE_BOOLEAN, z2103, HOTSPOT_JNI_SETSTATICBOOLEANFIELD_ENTRY(env, clazz, (uintptr_t)fieldID, value),2104HOTSPOT_JNI_SETSTATICBOOLEANFIELD_RETURN())2105DEFINE_SETSTATICFIELD(jbyte, byte, Byte, JVM_SIGNATURE_BYTE, b2106, HOTSPOT_JNI_SETSTATICBYTEFIELD_ENTRY(env, clazz, (uintptr_t) fieldID, value),2107HOTSPOT_JNI_SETSTATICBYTEFIELD_RETURN())2108DEFINE_SETSTATICFIELD(jchar, char, Char, JVM_SIGNATURE_CHAR, c2109, HOTSPOT_JNI_SETSTATICCHARFIELD_ENTRY(env, clazz, (uintptr_t) fieldID, value),2110HOTSPOT_JNI_SETSTATICCHARFIELD_RETURN())2111DEFINE_SETSTATICFIELD(jshort, short, Short, JVM_SIGNATURE_SHORT, s2112, HOTSPOT_JNI_SETSTATICSHORTFIELD_ENTRY(env, clazz, (uintptr_t) fieldID, value),2113HOTSPOT_JNI_SETSTATICSHORTFIELD_RETURN())2114DEFINE_SETSTATICFIELD(jint, int, Int, JVM_SIGNATURE_INT, i2115, HOTSPOT_JNI_SETSTATICINTFIELD_ENTRY(env, clazz, (uintptr_t) fieldID, value),2116HOTSPOT_JNI_SETSTATICINTFIELD_RETURN())2117DEFINE_SETSTATICFIELD(jlong, long, Long, JVM_SIGNATURE_LONG, j2118, HOTSPOT_JNI_SETSTATICLONGFIELD_ENTRY(env, clazz, (uintptr_t) fieldID, value),2119HOTSPOT_JNI_SETSTATICLONGFIELD_RETURN())2120// Float and double probes don't return value because dtrace doesn't currently support it2121DEFINE_SETSTATICFIELD(jfloat, float, Float, JVM_SIGNATURE_FLOAT, f2122, HOTSPOT_JNI_SETSTATICFLOATFIELD_ENTRY(env, clazz, (uintptr_t) fieldID),2123HOTSPOT_JNI_SETSTATICFLOATFIELD_RETURN())2124DEFINE_SETSTATICFIELD(jdouble, double, Double, JVM_SIGNATURE_DOUBLE, d2125, HOTSPOT_JNI_SETSTATICDOUBLEFIELD_ENTRY(env, clazz, (uintptr_t) fieldID),2126HOTSPOT_JNI_SETSTATICDOUBLEFIELD_RETURN())21272128//2129// String Operations2130//21312132// Unicode Interface21332134DT_RETURN_MARK_DECL(NewString, jstring2135, HOTSPOT_JNI_NEWSTRING_RETURN(_ret_ref));21362137JNI_ENTRY(jstring, jni_NewString(JNIEnv *env, const jchar *unicodeChars, jsize len))2138HOTSPOT_JNI_NEWSTRING_ENTRY(env, (uint16_t *) unicodeChars, len);2139jstring ret = NULL;2140DT_RETURN_MARK(NewString, jstring, (const jstring&)ret);2141oop string=java_lang_String::create_oop_from_unicode((jchar*) unicodeChars, len, CHECK_NULL);2142ret = (jstring) JNIHandles::make_local(THREAD, string);2143return ret;2144JNI_END214521462147JNI_ENTRY_NO_PRESERVE(jsize, jni_GetStringLength(JNIEnv *env, jstring string))2148HOTSPOT_JNI_GETSTRINGLENGTH_ENTRY(env, string);2149jsize ret = 0;2150oop s = JNIHandles::resolve_non_null(string);2151ret = java_lang_String::length(s);2152HOTSPOT_JNI_GETSTRINGLENGTH_RETURN(ret);2153return ret;2154JNI_END215521562157JNI_ENTRY_NO_PRESERVE(const jchar*, jni_GetStringChars(2158JNIEnv *env, jstring string, jboolean *isCopy))2159HOTSPOT_JNI_GETSTRINGCHARS_ENTRY(env, string, (uintptr_t *) isCopy);2160jchar* buf = NULL;2161oop s = JNIHandles::resolve_non_null(string);2162typeArrayOop s_value = java_lang_String::value(s);2163if (s_value != NULL) {2164int s_len = java_lang_String::length(s, s_value);2165bool is_latin1 = java_lang_String::is_latin1(s);2166buf = NEW_C_HEAP_ARRAY_RETURN_NULL(jchar, s_len + 1, mtInternal); // add one for zero termination2167/* JNI Specification states return NULL on OOM */2168if (buf != NULL) {2169if (s_len > 0) {2170if (!is_latin1) {2171ArrayAccess<>::arraycopy_to_native(s_value, (size_t) typeArrayOopDesc::element_offset<jchar>(0),2172buf, s_len);2173} else {2174for (int i = 0; i < s_len; i++) {2175buf[i] = ((jchar) s_value->byte_at(i)) & 0xff;2176}2177}2178}2179buf[s_len] = 0;2180//%note jni_52181if (isCopy != NULL) {2182*isCopy = JNI_TRUE;2183}2184}2185}2186HOTSPOT_JNI_GETSTRINGCHARS_RETURN(buf);2187return buf;2188JNI_END218921902191JNI_ENTRY_NO_PRESERVE(void, jni_ReleaseStringChars(JNIEnv *env, jstring str, const jchar *chars))2192HOTSPOT_JNI_RELEASESTRINGCHARS_ENTRY(env, str, (uint16_t *) chars);2193//%note jni_62194if (chars != NULL) {2195// Since String objects are supposed to be immutable, don't copy any2196// new data back. A bad user will have to go after the char array.2197FreeHeap((void*) chars);2198}2199HOTSPOT_JNI_RELEASESTRINGCHARS_RETURN();2200JNI_END220122022203// UTF Interface22042205DT_RETURN_MARK_DECL(NewStringUTF, jstring2206, HOTSPOT_JNI_NEWSTRINGUTF_RETURN(_ret_ref));22072208JNI_ENTRY(jstring, jni_NewStringUTF(JNIEnv *env, const char *bytes))2209HOTSPOT_JNI_NEWSTRINGUTF_ENTRY(env, (char *) bytes);2210jstring ret;2211DT_RETURN_MARK(NewStringUTF, jstring, (const jstring&)ret);22122213oop result = java_lang_String::create_oop_from_str((char*) bytes, CHECK_NULL);2214ret = (jstring) JNIHandles::make_local(THREAD, result);2215return ret;2216JNI_END221722182219JNI_ENTRY(jsize, jni_GetStringUTFLength(JNIEnv *env, jstring string))2220HOTSPOT_JNI_GETSTRINGUTFLENGTH_ENTRY(env, string);2221oop java_string = JNIHandles::resolve_non_null(string);2222jsize ret = java_lang_String::utf8_length(java_string);2223HOTSPOT_JNI_GETSTRINGUTFLENGTH_RETURN(ret);2224return ret;2225JNI_END222622272228JNI_ENTRY(const char*, jni_GetStringUTFChars(JNIEnv *env, jstring string, jboolean *isCopy))2229HOTSPOT_JNI_GETSTRINGUTFCHARS_ENTRY(env, string, (uintptr_t *) isCopy);2230char* result = NULL;2231oop java_string = JNIHandles::resolve_non_null(string);2232typeArrayOop s_value = java_lang_String::value(java_string);2233if (s_value != NULL) {2234size_t length = java_lang_String::utf8_length(java_string, s_value);2235/* JNI Specification states return NULL on OOM */2236result = AllocateHeap(length + 1, mtInternal, 0, AllocFailStrategy::RETURN_NULL);2237if (result != NULL) {2238java_lang_String::as_utf8_string(java_string, s_value, result, (int) length + 1);2239if (isCopy != NULL) {2240*isCopy = JNI_TRUE;2241}2242}2243}2244HOTSPOT_JNI_GETSTRINGUTFCHARS_RETURN(result);2245return result;2246JNI_END224722482249JNI_LEAF(void, jni_ReleaseStringUTFChars(JNIEnv *env, jstring str, const char *chars))2250HOTSPOT_JNI_RELEASESTRINGUTFCHARS_ENTRY(env, str, (char *) chars);2251if (chars != NULL) {2252FreeHeap((char*) chars);2253}2254HOTSPOT_JNI_RELEASESTRINGUTFCHARS_RETURN();2255JNI_END225622572258JNI_ENTRY_NO_PRESERVE(jsize, jni_GetArrayLength(JNIEnv *env, jarray array))2259HOTSPOT_JNI_GETARRAYLENGTH_ENTRY(env, array);2260arrayOop a = arrayOop(JNIHandles::resolve_non_null(array));2261assert(a->is_array(), "must be array");2262jsize ret = a->length();2263HOTSPOT_JNI_GETARRAYLENGTH_RETURN(ret);2264return ret;2265JNI_END226622672268//2269// Object Array Operations2270//22712272DT_RETURN_MARK_DECL(NewObjectArray, jobjectArray2273, HOTSPOT_JNI_NEWOBJECTARRAY_RETURN(_ret_ref));22742275JNI_ENTRY(jobjectArray, jni_NewObjectArray(JNIEnv *env, jsize length, jclass elementClass, jobject initialElement))2276HOTSPOT_JNI_NEWOBJECTARRAY_ENTRY(env, length, elementClass, initialElement);2277jobjectArray ret = NULL;2278DT_RETURN_MARK(NewObjectArray, jobjectArray, (const jobjectArray&)ret);2279Klass* ek = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(elementClass));2280Klass* ak = ek->array_klass(CHECK_NULL);2281ObjArrayKlass::cast(ak)->initialize(CHECK_NULL);2282objArrayOop result = ObjArrayKlass::cast(ak)->allocate(length, CHECK_NULL);2283oop initial_value = JNIHandles::resolve(initialElement);2284if (initial_value != NULL) { // array already initialized with NULL2285for (int index = 0; index < length; index++) {2286result->obj_at_put(index, initial_value);2287}2288}2289ret = (jobjectArray) JNIHandles::make_local(THREAD, result);2290return ret;2291JNI_END22922293DT_RETURN_MARK_DECL(GetObjectArrayElement, jobject2294, HOTSPOT_JNI_GETOBJECTARRAYELEMENT_RETURN(_ret_ref));22952296JNI_ENTRY(jobject, jni_GetObjectArrayElement(JNIEnv *env, jobjectArray array, jsize index))2297HOTSPOT_JNI_GETOBJECTARRAYELEMENT_ENTRY(env, array, index);2298jobject ret = NULL;2299DT_RETURN_MARK(GetObjectArrayElement, jobject, (const jobject&)ret);2300objArrayOop a = objArrayOop(JNIHandles::resolve_non_null(array));2301if (a->is_within_bounds(index)) {2302ret = JNIHandles::make_local(THREAD, a->obj_at(index));2303return ret;2304} else {2305ResourceMark rm(THREAD);2306stringStream ss;2307ss.print("Index %d out of bounds for length %d", index, a->length());2308THROW_MSG_0(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());2309}2310JNI_END23112312DT_VOID_RETURN_MARK_DECL(SetObjectArrayElement2313, HOTSPOT_JNI_SETOBJECTARRAYELEMENT_RETURN());23142315JNI_ENTRY(void, jni_SetObjectArrayElement(JNIEnv *env, jobjectArray array, jsize index, jobject value))2316HOTSPOT_JNI_SETOBJECTARRAYELEMENT_ENTRY(env, array, index, value);2317DT_VOID_RETURN_MARK(SetObjectArrayElement);23182319objArrayOop a = objArrayOop(JNIHandles::resolve_non_null(array));2320oop v = JNIHandles::resolve(value);2321if (a->is_within_bounds(index)) {2322if (v == NULL || v->is_a(ObjArrayKlass::cast(a->klass())->element_klass())) {2323a->obj_at_put(index, v);2324} else {2325ResourceMark rm(THREAD);2326stringStream ss;2327Klass *bottom_kl = ObjArrayKlass::cast(a->klass())->bottom_klass();2328ss.print("type mismatch: can not store %s to %s[%d]",2329v->klass()->external_name(),2330bottom_kl->is_typeArray_klass() ? type2name_tab[ArrayKlass::cast(bottom_kl)->element_type()] : bottom_kl->external_name(),2331index);2332for (int dims = ArrayKlass::cast(a->klass())->dimension(); dims > 1; --dims) {2333ss.print("[]");2334}2335THROW_MSG(vmSymbols::java_lang_ArrayStoreException(), ss.as_string());2336}2337} else {2338ResourceMark rm(THREAD);2339stringStream ss;2340ss.print("Index %d out of bounds for length %d", index, a->length());2341THROW_MSG(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());2342}2343JNI_END2344234523462347#define DEFINE_NEWSCALARARRAY(Return,Allocator,Result \2348,EntryProbe,ReturnProbe) \2349\2350DT_RETURN_MARK_DECL(New##Result##Array, Return \2351, ReturnProbe); \2352\2353JNI_ENTRY(Return, \2354jni_New##Result##Array(JNIEnv *env, jsize len)) \2355EntryProbe; \2356Return ret = NULL;\2357DT_RETURN_MARK(New##Result##Array, Return, (const Return&)ret);\2358\2359oop obj= oopFactory::Allocator(len, CHECK_NULL); \2360ret = (Return) JNIHandles::make_local(THREAD, obj); \2361return ret;\2362JNI_END23632364DEFINE_NEWSCALARARRAY(jbooleanArray, new_boolArray, Boolean,2365HOTSPOT_JNI_NEWBOOLEANARRAY_ENTRY(env, len),2366HOTSPOT_JNI_NEWBOOLEANARRAY_RETURN(_ret_ref))2367DEFINE_NEWSCALARARRAY(jbyteArray, new_byteArray, Byte,2368HOTSPOT_JNI_NEWBYTEARRAY_ENTRY(env, len),2369HOTSPOT_JNI_NEWBYTEARRAY_RETURN(_ret_ref))2370DEFINE_NEWSCALARARRAY(jshortArray, new_shortArray, Short,2371HOTSPOT_JNI_NEWSHORTARRAY_ENTRY(env, len),2372HOTSPOT_JNI_NEWSHORTARRAY_RETURN(_ret_ref))2373DEFINE_NEWSCALARARRAY(jcharArray, new_charArray, Char,2374HOTSPOT_JNI_NEWCHARARRAY_ENTRY(env, len),2375HOTSPOT_JNI_NEWCHARARRAY_RETURN(_ret_ref))2376DEFINE_NEWSCALARARRAY(jintArray, new_intArray, Int,2377HOTSPOT_JNI_NEWINTARRAY_ENTRY(env, len),2378HOTSPOT_JNI_NEWINTARRAY_RETURN(_ret_ref))2379DEFINE_NEWSCALARARRAY(jlongArray, new_longArray, Long,2380HOTSPOT_JNI_NEWLONGARRAY_ENTRY(env, len),2381HOTSPOT_JNI_NEWLONGARRAY_RETURN(_ret_ref))2382DEFINE_NEWSCALARARRAY(jfloatArray, new_floatArray, Float,2383HOTSPOT_JNI_NEWFLOATARRAY_ENTRY(env, len),2384HOTSPOT_JNI_NEWFLOATARRAY_RETURN(_ret_ref))2385DEFINE_NEWSCALARARRAY(jdoubleArray, new_doubleArray, Double,2386HOTSPOT_JNI_NEWDOUBLEARRAY_ENTRY(env, len),2387HOTSPOT_JNI_NEWDOUBLEARRAY_RETURN(_ret_ref))23882389// Return an address which will fault if the caller writes to it.23902391static char* get_bad_address() {2392static char* bad_address = NULL;2393if (bad_address == NULL) {2394size_t size = os::vm_allocation_granularity();2395bad_address = os::reserve_memory(size);2396if (bad_address != NULL) {2397os::protect_memory(bad_address, size, os::MEM_PROT_READ,2398/*is_committed*/false);2399MemTracker::record_virtual_memory_type((void*)bad_address, mtInternal);2400}2401}2402return bad_address;2403}2404240524062407#define DEFINE_GETSCALARARRAYELEMENTS(ElementTag,ElementType,Result, Tag \2408, EntryProbe, ReturnProbe) \2409\2410JNI_ENTRY_NO_PRESERVE(ElementType*, \2411jni_Get##Result##ArrayElements(JNIEnv *env, ElementType##Array array, jboolean *isCopy)) \2412EntryProbe; \2413/* allocate an chunk of memory in c land */ \2414typeArrayOop a = typeArrayOop(JNIHandles::resolve_non_null(array)); \2415ElementType* result; \2416int len = a->length(); \2417if (len == 0) { \2418if (isCopy != NULL) { \2419*isCopy = JNI_FALSE; \2420} \2421/* Empty array: legal but useless, can't return NULL. \2422* Return a pointer to something useless. \2423* Avoid asserts in typeArrayOop. */ \2424result = (ElementType*)get_bad_address(); \2425} else { \2426/* JNI Specification states return NULL on OOM */ \2427result = NEW_C_HEAP_ARRAY_RETURN_NULL(ElementType, len, mtInternal); \2428if (result != NULL) { \2429/* copy the array to the c chunk */ \2430ArrayAccess<>::arraycopy_to_native(a, typeArrayOopDesc::element_offset<ElementType>(0), \2431result, len); \2432if (isCopy) { \2433*isCopy = JNI_TRUE; \2434} \2435} \2436} \2437ReturnProbe; \2438return result; \2439JNI_END24402441DEFINE_GETSCALARARRAYELEMENTS(T_BOOLEAN, jboolean, Boolean, bool2442, HOTSPOT_JNI_GETBOOLEANARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) isCopy),2443HOTSPOT_JNI_GETBOOLEANARRAYELEMENTS_RETURN((uintptr_t*)result))2444DEFINE_GETSCALARARRAYELEMENTS(T_BYTE, jbyte, Byte, byte2445, HOTSPOT_JNI_GETBYTEARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) isCopy),2446HOTSPOT_JNI_GETBYTEARRAYELEMENTS_RETURN((char*)result))2447DEFINE_GETSCALARARRAYELEMENTS(T_SHORT, jshort, Short, short2448, HOTSPOT_JNI_GETSHORTARRAYELEMENTS_ENTRY(env, (uint16_t*) array, (uintptr_t *) isCopy),2449HOTSPOT_JNI_GETSHORTARRAYELEMENTS_RETURN((uint16_t*)result))2450DEFINE_GETSCALARARRAYELEMENTS(T_CHAR, jchar, Char, char2451, HOTSPOT_JNI_GETCHARARRAYELEMENTS_ENTRY(env, (uint16_t*) array, (uintptr_t *) isCopy),2452HOTSPOT_JNI_GETCHARARRAYELEMENTS_RETURN(result))2453DEFINE_GETSCALARARRAYELEMENTS(T_INT, jint, Int, int2454, HOTSPOT_JNI_GETINTARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) isCopy),2455HOTSPOT_JNI_GETINTARRAYELEMENTS_RETURN((uint32_t*)result))2456DEFINE_GETSCALARARRAYELEMENTS(T_LONG, jlong, Long, long2457, HOTSPOT_JNI_GETLONGARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) isCopy),2458HOTSPOT_JNI_GETLONGARRAYELEMENTS_RETURN(((uintptr_t*)result)))2459// Float and double probes don't return value because dtrace doesn't currently support it2460DEFINE_GETSCALARARRAYELEMENTS(T_FLOAT, jfloat, Float, float2461, HOTSPOT_JNI_GETFLOATARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) isCopy),2462HOTSPOT_JNI_GETFLOATARRAYELEMENTS_RETURN(result))2463DEFINE_GETSCALARARRAYELEMENTS(T_DOUBLE, jdouble, Double, double2464, HOTSPOT_JNI_GETDOUBLEARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) isCopy),2465HOTSPOT_JNI_GETDOUBLEARRAYELEMENTS_RETURN(result))246624672468#define DEFINE_RELEASESCALARARRAYELEMENTS(ElementTag,ElementType,Result,Tag \2469, EntryProbe, ReturnProbe);\2470\2471JNI_ENTRY_NO_PRESERVE(void, \2472jni_Release##Result##ArrayElements(JNIEnv *env, ElementType##Array array, \2473ElementType *buf, jint mode)) \2474EntryProbe; \2475typeArrayOop a = typeArrayOop(JNIHandles::resolve_non_null(array)); \2476int len = a->length(); \2477if (len != 0) { /* Empty array: nothing to free or copy. */ \2478if ((mode == 0) || (mode == JNI_COMMIT)) { \2479ArrayAccess<>::arraycopy_from_native(buf, a, typeArrayOopDesc::element_offset<ElementType>(0), len); \2480} \2481if ((mode == 0) || (mode == JNI_ABORT)) { \2482FreeHeap(buf); \2483} \2484} \2485ReturnProbe; \2486JNI_END24872488DEFINE_RELEASESCALARARRAYELEMENTS(T_BOOLEAN, jboolean, Boolean, bool2489, HOTSPOT_JNI_RELEASEBOOLEANARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) buf, mode),2490HOTSPOT_JNI_RELEASEBOOLEANARRAYELEMENTS_RETURN())2491DEFINE_RELEASESCALARARRAYELEMENTS(T_BYTE, jbyte, Byte, byte2492, HOTSPOT_JNI_RELEASEBYTEARRAYELEMENTS_ENTRY(env, array, (char *) buf, mode),2493HOTSPOT_JNI_RELEASEBYTEARRAYELEMENTS_RETURN())2494DEFINE_RELEASESCALARARRAYELEMENTS(T_SHORT, jshort, Short, short2495, HOTSPOT_JNI_RELEASESHORTARRAYELEMENTS_ENTRY(env, array, (uint16_t *) buf, mode),2496HOTSPOT_JNI_RELEASESHORTARRAYELEMENTS_RETURN())2497DEFINE_RELEASESCALARARRAYELEMENTS(T_CHAR, jchar, Char, char2498, HOTSPOT_JNI_RELEASECHARARRAYELEMENTS_ENTRY(env, array, (uint16_t *) buf, mode),2499HOTSPOT_JNI_RELEASECHARARRAYELEMENTS_RETURN())2500DEFINE_RELEASESCALARARRAYELEMENTS(T_INT, jint, Int, int2501, HOTSPOT_JNI_RELEASEINTARRAYELEMENTS_ENTRY(env, array, (uint32_t *) buf, mode),2502HOTSPOT_JNI_RELEASEINTARRAYELEMENTS_RETURN())2503DEFINE_RELEASESCALARARRAYELEMENTS(T_LONG, jlong, Long, long2504, HOTSPOT_JNI_RELEASELONGARRAYELEMENTS_ENTRY(env, array, (uintptr_t *) buf, mode),2505HOTSPOT_JNI_RELEASELONGARRAYELEMENTS_RETURN())2506DEFINE_RELEASESCALARARRAYELEMENTS(T_FLOAT, jfloat, Float, float2507, HOTSPOT_JNI_RELEASEFLOATARRAYELEMENTS_ENTRY(env, array, (float *) buf, mode),2508HOTSPOT_JNI_RELEASEFLOATARRAYELEMENTS_RETURN())2509DEFINE_RELEASESCALARARRAYELEMENTS(T_DOUBLE, jdouble, Double, double2510, HOTSPOT_JNI_RELEASEDOUBLEARRAYELEMENTS_ENTRY(env, array, (double *) buf, mode),2511HOTSPOT_JNI_RELEASEDOUBLEARRAYELEMENTS_RETURN())25122513static void check_bounds(jsize start, jsize copy_len, jsize array_len, TRAPS) {2514ResourceMark rm(THREAD);2515if (copy_len < 0) {2516stringStream ss;2517ss.print("Length %d is negative", copy_len);2518THROW_MSG(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());2519} else if (start < 0 || (start > array_len - copy_len)) {2520stringStream ss;2521ss.print("Array region %d.." INT64_FORMAT " out of bounds for length %d",2522start, (int64_t)start+(int64_t)copy_len, array_len);2523THROW_MSG(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());2524}2525}25262527#define DEFINE_GETSCALARARRAYREGION(ElementTag,ElementType,Result, Tag \2528, EntryProbe, ReturnProbe); \2529DT_VOID_RETURN_MARK_DECL(Get##Result##ArrayRegion \2530, ReturnProbe); \2531\2532JNI_ENTRY(void, \2533jni_Get##Result##ArrayRegion(JNIEnv *env, ElementType##Array array, jsize start, \2534jsize len, ElementType *buf)) \2535EntryProbe; \2536DT_VOID_RETURN_MARK(Get##Result##ArrayRegion); \2537typeArrayOop src = typeArrayOop(JNIHandles::resolve_non_null(array)); \2538check_bounds(start, len, src->length(), CHECK); \2539if (len > 0) { \2540ArrayAccess<>::arraycopy_to_native(src, typeArrayOopDesc::element_offset<ElementType>(start), buf, len); \2541} \2542JNI_END25432544DEFINE_GETSCALARARRAYREGION(T_BOOLEAN, jboolean,Boolean, bool2545, HOTSPOT_JNI_GETBOOLEANARRAYREGION_ENTRY(env, array, start, len, (uintptr_t *) buf),2546HOTSPOT_JNI_GETBOOLEANARRAYREGION_RETURN());2547DEFINE_GETSCALARARRAYREGION(T_BYTE, jbyte, Byte, byte2548, HOTSPOT_JNI_GETBYTEARRAYREGION_ENTRY(env, array, start, len, (char *) buf),2549HOTSPOT_JNI_GETBYTEARRAYREGION_RETURN());2550DEFINE_GETSCALARARRAYREGION(T_SHORT, jshort, Short, short2551, HOTSPOT_JNI_GETSHORTARRAYREGION_ENTRY(env, array, start, len, (uint16_t *) buf),2552HOTSPOT_JNI_GETSHORTARRAYREGION_RETURN());2553DEFINE_GETSCALARARRAYREGION(T_CHAR, jchar, Char, char2554, HOTSPOT_JNI_GETCHARARRAYREGION_ENTRY(env, array, start, len, (uint16_t*) buf),2555HOTSPOT_JNI_GETCHARARRAYREGION_RETURN());2556DEFINE_GETSCALARARRAYREGION(T_INT, jint, Int, int2557, HOTSPOT_JNI_GETINTARRAYREGION_ENTRY(env, array, start, len, (uint32_t*) buf),2558HOTSPOT_JNI_GETINTARRAYREGION_RETURN());2559DEFINE_GETSCALARARRAYREGION(T_LONG, jlong, Long, long2560, HOTSPOT_JNI_GETLONGARRAYREGION_ENTRY(env, array, start, len, (uintptr_t *) buf),2561HOTSPOT_JNI_GETLONGARRAYREGION_RETURN());2562DEFINE_GETSCALARARRAYREGION(T_FLOAT, jfloat, Float, float2563, HOTSPOT_JNI_GETFLOATARRAYREGION_ENTRY(env, array, start, len, (float *) buf),2564HOTSPOT_JNI_GETFLOATARRAYREGION_RETURN());2565DEFINE_GETSCALARARRAYREGION(T_DOUBLE, jdouble, Double, double2566, HOTSPOT_JNI_GETDOUBLEARRAYREGION_ENTRY(env, array, start, len, (double *) buf),2567HOTSPOT_JNI_GETDOUBLEARRAYREGION_RETURN());256825692570#define DEFINE_SETSCALARARRAYREGION(ElementTag,ElementType,Result, Tag \2571, EntryProbe, ReturnProbe); \2572DT_VOID_RETURN_MARK_DECL(Set##Result##ArrayRegion \2573,ReturnProbe); \2574\2575JNI_ENTRY(void, \2576jni_Set##Result##ArrayRegion(JNIEnv *env, ElementType##Array array, jsize start, \2577jsize len, const ElementType *buf)) \2578EntryProbe; \2579DT_VOID_RETURN_MARK(Set##Result##ArrayRegion); \2580typeArrayOop dst = typeArrayOop(JNIHandles::resolve_non_null(array)); \2581check_bounds(start, len, dst->length(), CHECK); \2582if (len > 0) { \2583ArrayAccess<>::arraycopy_from_native(buf, dst, typeArrayOopDesc::element_offset<ElementType>(start), len); \2584} \2585JNI_END25862587DEFINE_SETSCALARARRAYREGION(T_BOOLEAN, jboolean, Boolean, bool2588, HOTSPOT_JNI_SETBOOLEANARRAYREGION_ENTRY(env, array, start, len, (uintptr_t *)buf),2589HOTSPOT_JNI_SETBOOLEANARRAYREGION_RETURN())2590DEFINE_SETSCALARARRAYREGION(T_BYTE, jbyte, Byte, byte2591, HOTSPOT_JNI_SETBYTEARRAYREGION_ENTRY(env, array, start, len, (char *) buf),2592HOTSPOT_JNI_SETBYTEARRAYREGION_RETURN())2593DEFINE_SETSCALARARRAYREGION(T_SHORT, jshort, Short, short2594, HOTSPOT_JNI_SETSHORTARRAYREGION_ENTRY(env, array, start, len, (uint16_t *) buf),2595HOTSPOT_JNI_SETSHORTARRAYREGION_RETURN())2596DEFINE_SETSCALARARRAYREGION(T_CHAR, jchar, Char, char2597, HOTSPOT_JNI_SETCHARARRAYREGION_ENTRY(env, array, start, len, (uint16_t *) buf),2598HOTSPOT_JNI_SETCHARARRAYREGION_RETURN())2599DEFINE_SETSCALARARRAYREGION(T_INT, jint, Int, int2600, HOTSPOT_JNI_SETINTARRAYREGION_ENTRY(env, array, start, len, (uint32_t *) buf),2601HOTSPOT_JNI_SETINTARRAYREGION_RETURN())2602DEFINE_SETSCALARARRAYREGION(T_LONG, jlong, Long, long2603, HOTSPOT_JNI_SETLONGARRAYREGION_ENTRY(env, array, start, len, (uintptr_t *) buf),2604HOTSPOT_JNI_SETLONGARRAYREGION_RETURN())2605DEFINE_SETSCALARARRAYREGION(T_FLOAT, jfloat, Float, float2606, HOTSPOT_JNI_SETFLOATARRAYREGION_ENTRY(env, array, start, len, (float *) buf),2607HOTSPOT_JNI_SETFLOATARRAYREGION_RETURN())2608DEFINE_SETSCALARARRAYREGION(T_DOUBLE, jdouble, Double, double2609, HOTSPOT_JNI_SETDOUBLEARRAYREGION_ENTRY(env, array, start, len, (double *) buf),2610HOTSPOT_JNI_SETDOUBLEARRAYREGION_RETURN())261126122613DT_RETURN_MARK_DECL(RegisterNatives, jint2614, HOTSPOT_JNI_REGISTERNATIVES_RETURN(_ret_ref));26152616JNI_ENTRY(jint, jni_RegisterNatives(JNIEnv *env, jclass clazz,2617const JNINativeMethod *methods,2618jint nMethods))2619HOTSPOT_JNI_REGISTERNATIVES_ENTRY(env, clazz, (void *) methods, nMethods);2620jint ret = 0;2621DT_RETURN_MARK(RegisterNatives, jint, (const jint&)ret);26222623Klass* k = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(clazz));26242625// There are no restrictions on native code registering native methods,2626// which allows agents to redefine the bindings to native methods, however2627// we issue a warning if any code running outside of the boot/platform2628// loader is rebinding any native methods in classes loaded by the2629// boot/platform loader that are in named modules. That will catch changes2630// to platform classes while excluding classes added to the bootclasspath.2631bool do_warning = false;26322633// Only instanceKlasses can have native methods2634if (k->is_instance_klass()) {2635oop cl = k->class_loader();2636InstanceKlass* ik = InstanceKlass::cast(k);2637// Check for a platform class2638if ((cl == NULL || SystemDictionary::is_platform_class_loader(cl)) &&2639ik->module()->is_named()) {2640Klass* caller = thread->security_get_caller_class(1);2641// If no caller class, or caller class has a different loader, then2642// issue a warning below.2643do_warning = (caller == NULL) || caller->class_loader() != cl;2644}2645}264626472648for (int index = 0; index < nMethods; index++) {2649const char* meth_name = methods[index].name;2650const char* meth_sig = methods[index].signature;2651int meth_name_len = (int)strlen(meth_name);26522653// The class should have been loaded (we have an instance of the class2654// passed in) so the method and signature should already be in the symbol2655// table. If they're not there, the method doesn't exist.2656TempNewSymbol name = SymbolTable::probe(meth_name, meth_name_len);2657TempNewSymbol signature = SymbolTable::probe(meth_sig, (int)strlen(meth_sig));26582659if (name == NULL || signature == NULL) {2660ResourceMark rm(THREAD);2661stringStream st;2662st.print("Method %s.%s%s not found", k->external_name(), meth_name, meth_sig);2663// Must return negative value on failure2664THROW_MSG_(vmSymbols::java_lang_NoSuchMethodError(), st.as_string(), -1);2665}26662667if (do_warning) {2668ResourceMark rm(THREAD);2669log_warning(jni, resolve)("Re-registering of platform native method: %s.%s%s "2670"from code in a different classloader", k->external_name(), meth_name, meth_sig);2671}26722673bool res = Method::register_native(k, name, signature,2674(address) methods[index].fnPtr, THREAD);2675if (!res) {2676ret = -1;2677break;2678}2679}2680return ret;2681JNI_END268226832684JNI_ENTRY(jint, jni_UnregisterNatives(JNIEnv *env, jclass clazz))2685HOTSPOT_JNI_UNREGISTERNATIVES_ENTRY(env, clazz);2686Klass* k = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(clazz));2687//%note jni_22688if (k->is_instance_klass()) {2689for (int index = 0; index < InstanceKlass::cast(k)->methods()->length(); index++) {2690Method* m = InstanceKlass::cast(k)->methods()->at(index);2691if (m->is_native()) {2692m->clear_native_function();2693m->set_signature_handler(NULL);2694}2695}2696}2697HOTSPOT_JNI_UNREGISTERNATIVES_RETURN(0);2698return 0;2699JNI_END27002701//2702// Monitor functions2703//27042705DT_RETURN_MARK_DECL(MonitorEnter, jint2706, HOTSPOT_JNI_MONITORENTER_RETURN(_ret_ref));27072708JNI_ENTRY(jint, jni_MonitorEnter(JNIEnv *env, jobject jobj))2709HOTSPOT_JNI_MONITORENTER_ENTRY(env, jobj);2710jint ret = JNI_ERR;2711DT_RETURN_MARK(MonitorEnter, jint, (const jint&)ret);27122713// If the object is null, we can't do anything with it2714if (jobj == NULL) {2715THROW_(vmSymbols::java_lang_NullPointerException(), JNI_ERR);2716}27172718Handle obj(thread, JNIHandles::resolve_non_null(jobj));2719ObjectSynchronizer::jni_enter(obj, thread);2720ret = JNI_OK;2721return ret;2722JNI_END27232724DT_RETURN_MARK_DECL(MonitorExit, jint2725, HOTSPOT_JNI_MONITOREXIT_RETURN(_ret_ref));27262727JNI_ENTRY(jint, jni_MonitorExit(JNIEnv *env, jobject jobj))2728HOTSPOT_JNI_MONITOREXIT_ENTRY(env, jobj);2729jint ret = JNI_ERR;2730DT_RETURN_MARK(MonitorExit, jint, (const jint&)ret);27312732// Don't do anything with a null object2733if (jobj == NULL) {2734THROW_(vmSymbols::java_lang_NullPointerException(), JNI_ERR);2735}27362737Handle obj(THREAD, JNIHandles::resolve_non_null(jobj));2738ObjectSynchronizer::jni_exit(obj(), CHECK_(JNI_ERR));27392740ret = JNI_OK;2741return ret;2742JNI_END27432744//2745// Extensions2746//27472748DT_VOID_RETURN_MARK_DECL(GetStringRegion2749, HOTSPOT_JNI_GETSTRINGREGION_RETURN());27502751JNI_ENTRY(void, jni_GetStringRegion(JNIEnv *env, jstring string, jsize start, jsize len, jchar *buf))2752HOTSPOT_JNI_GETSTRINGREGION_ENTRY(env, string, start, len, buf);2753DT_VOID_RETURN_MARK(GetStringRegion);2754oop s = JNIHandles::resolve_non_null(string);2755typeArrayOop s_value = java_lang_String::value(s);2756int s_len = java_lang_String::length(s, s_value);2757if (start < 0 || len < 0 || start > s_len - len) {2758THROW(vmSymbols::java_lang_StringIndexOutOfBoundsException());2759} else {2760if (len > 0) {2761bool is_latin1 = java_lang_String::is_latin1(s);2762if (!is_latin1) {2763ArrayAccess<>::arraycopy_to_native(s_value, typeArrayOopDesc::element_offset<jchar>(start),2764buf, len);2765} else {2766for (int i = 0; i < len; i++) {2767buf[i] = ((jchar) s_value->byte_at(i + start)) & 0xff;2768}2769}2770}2771}2772JNI_END27732774DT_VOID_RETURN_MARK_DECL(GetStringUTFRegion2775, HOTSPOT_JNI_GETSTRINGUTFREGION_RETURN());27762777JNI_ENTRY(void, jni_GetStringUTFRegion(JNIEnv *env, jstring string, jsize start, jsize len, char *buf))2778HOTSPOT_JNI_GETSTRINGUTFREGION_ENTRY(env, string, start, len, buf);2779DT_VOID_RETURN_MARK(GetStringUTFRegion);2780oop s = JNIHandles::resolve_non_null(string);2781typeArrayOop s_value = java_lang_String::value(s);2782int s_len = java_lang_String::length(s, s_value);2783if (start < 0 || len < 0 || start > s_len - len) {2784THROW(vmSymbols::java_lang_StringIndexOutOfBoundsException());2785} else {2786//%note jni_72787if (len > 0) {2788// Assume the buffer is large enough as the JNI spec. does not require user error checking2789java_lang_String::as_utf8_string(s, s_value, start, len, buf, INT_MAX);2790// as_utf8_string null-terminates the result string2791} else {2792// JDK null-terminates the buffer even in len is zero2793if (buf != NULL) {2794buf[0] = 0;2795}2796}2797}2798JNI_END27992800static oop lock_gc_or_pin_object(JavaThread* thread, jobject obj) {2801if (Universe::heap()->supports_object_pinning()) {2802const oop o = JNIHandles::resolve_non_null(obj);2803return Universe::heap()->pin_object(thread, o);2804} else {2805GCLocker::lock_critical(thread);2806return JNIHandles::resolve_non_null(obj);2807}2808}28092810static void unlock_gc_or_unpin_object(JavaThread* thread, jobject obj) {2811if (Universe::heap()->supports_object_pinning()) {2812const oop o = JNIHandles::resolve_non_null(obj);2813return Universe::heap()->unpin_object(thread, o);2814} else {2815GCLocker::unlock_critical(thread);2816}2817}28182819JNI_ENTRY(void*, jni_GetPrimitiveArrayCritical(JNIEnv *env, jarray array, jboolean *isCopy))2820HOTSPOT_JNI_GETPRIMITIVEARRAYCRITICAL_ENTRY(env, array, (uintptr_t *) isCopy);2821if (isCopy != NULL) {2822*isCopy = JNI_FALSE;2823}2824oop a = lock_gc_or_pin_object(thread, array);2825assert(a->is_array(), "just checking");2826BasicType type;2827if (a->is_objArray()) {2828type = T_OBJECT;2829} else {2830type = TypeArrayKlass::cast(a->klass())->element_type();2831}2832void* ret = arrayOop(a)->base(type);2833HOTSPOT_JNI_GETPRIMITIVEARRAYCRITICAL_RETURN(ret);2834return ret;2835JNI_END283628372838JNI_ENTRY(void, jni_ReleasePrimitiveArrayCritical(JNIEnv *env, jarray array, void *carray, jint mode))2839HOTSPOT_JNI_RELEASEPRIMITIVEARRAYCRITICAL_ENTRY(env, array, carray, mode);2840unlock_gc_or_unpin_object(thread, array);2841HOTSPOT_JNI_RELEASEPRIMITIVEARRAYCRITICAL_RETURN();2842JNI_END284328442845static typeArrayOop lock_gc_or_pin_string_value(JavaThread* thread, oop str) {2846if (Universe::heap()->supports_object_pinning()) {2847// Forbid deduplication before obtaining the value array, to prevent2848// deduplication from replacing the value array while setting up or in2849// the critical section. That would lead to the release operation2850// unpinning the wrong object.2851if (StringDedup::is_enabled()) {2852NoSafepointVerifier nsv;2853StringDedup::forbid_deduplication(str);2854}2855typeArrayOop s_value = java_lang_String::value(str);2856return (typeArrayOop) Universe::heap()->pin_object(thread, s_value);2857} else {2858Handle h(thread, str); // Handlize across potential safepoint.2859GCLocker::lock_critical(thread);2860return java_lang_String::value(h());2861}2862}28632864static void unlock_gc_or_unpin_string_value(JavaThread* thread, oop str) {2865if (Universe::heap()->supports_object_pinning()) {2866typeArrayOop s_value = java_lang_String::value(str);2867Universe::heap()->unpin_object(thread, s_value);2868} else {2869GCLocker::unlock_critical(thread);2870}2871}28722873JNI_ENTRY(const jchar*, jni_GetStringCritical(JNIEnv *env, jstring string, jboolean *isCopy))2874HOTSPOT_JNI_GETSTRINGCRITICAL_ENTRY(env, string, (uintptr_t *) isCopy);2875oop s = JNIHandles::resolve_non_null(string);2876jchar* ret;2877if (!java_lang_String::is_latin1(s)) {2878typeArrayOop s_value = lock_gc_or_pin_string_value(thread, s);2879ret = (jchar*) s_value->base(T_CHAR);2880if (isCopy != NULL) *isCopy = JNI_FALSE;2881} else {2882// Inflate latin1 encoded string to UTF162883typeArrayOop s_value = java_lang_String::value(s);2884int s_len = java_lang_String::length(s, s_value);2885ret = NEW_C_HEAP_ARRAY_RETURN_NULL(jchar, s_len + 1, mtInternal); // add one for zero termination2886/* JNI Specification states return NULL on OOM */2887if (ret != NULL) {2888for (int i = 0; i < s_len; i++) {2889ret[i] = ((jchar) s_value->byte_at(i)) & 0xff;2890}2891ret[s_len] = 0;2892}2893if (isCopy != NULL) *isCopy = JNI_TRUE;2894}2895HOTSPOT_JNI_GETSTRINGCRITICAL_RETURN((uint16_t *) ret);2896return ret;2897JNI_END289828992900JNI_ENTRY(void, jni_ReleaseStringCritical(JNIEnv *env, jstring str, const jchar *chars))2901HOTSPOT_JNI_RELEASESTRINGCRITICAL_ENTRY(env, str, (uint16_t *) chars);2902oop s = JNIHandles::resolve_non_null(str);2903bool is_latin1 = java_lang_String::is_latin1(s);2904if (is_latin1) {2905// For latin1 string, free jchar array allocated by earlier call to GetStringCritical.2906// This assumes that ReleaseStringCritical bookends GetStringCritical.2907FREE_C_HEAP_ARRAY(jchar, chars);2908} else {2909// For non-latin1 string, drop the associated gc-locker/pin.2910unlock_gc_or_unpin_string_value(thread, s);2911}2912HOTSPOT_JNI_RELEASESTRINGCRITICAL_RETURN();2913JNI_END291429152916JNI_ENTRY(jweak, jni_NewWeakGlobalRef(JNIEnv *env, jobject ref))2917HOTSPOT_JNI_NEWWEAKGLOBALREF_ENTRY(env, ref);2918Handle ref_handle(thread, JNIHandles::resolve(ref));2919jweak ret = JNIHandles::make_weak_global(ref_handle, AllocFailStrategy::RETURN_NULL);2920if (ret == NULL) {2921THROW_OOP_(Universe::out_of_memory_error_c_heap(), NULL);2922}2923HOTSPOT_JNI_NEWWEAKGLOBALREF_RETURN(ret);2924return ret;2925JNI_END29262927// Must be JNI_ENTRY (with HandleMark)2928JNI_ENTRY(void, jni_DeleteWeakGlobalRef(JNIEnv *env, jweak ref))2929HOTSPOT_JNI_DELETEWEAKGLOBALREF_ENTRY(env, ref);2930JNIHandles::destroy_weak_global(ref);2931HOTSPOT_JNI_DELETEWEAKGLOBALREF_RETURN();2932JNI_END293329342935JNI_ENTRY_NO_PRESERVE(jboolean, jni_ExceptionCheck(JNIEnv *env))2936HOTSPOT_JNI_EXCEPTIONCHECK_ENTRY(env);2937jni_check_async_exceptions(thread);2938jboolean ret = (thread->has_pending_exception()) ? JNI_TRUE : JNI_FALSE;2939HOTSPOT_JNI_EXCEPTIONCHECK_RETURN(ret);2940return ret;2941JNI_END294229432944// Initialization state for three routines below relating to2945// java.nio.DirectBuffers2946static int directBufferSupportInitializeStarted = 0;2947static volatile int directBufferSupportInitializeEnded = 0;2948static volatile int directBufferSupportInitializeFailed = 0;2949static jclass bufferClass = NULL;2950static jclass directBufferClass = NULL;2951static jclass directByteBufferClass = NULL;2952static jmethodID directByteBufferConstructor = NULL;2953static jfieldID directBufferAddressField = NULL;2954static jfieldID bufferCapacityField = NULL;29552956static jclass lookupOne(JNIEnv* env, const char* name, TRAPS) {2957Handle loader; // null (bootstrap) loader2958Handle protection_domain; // null protection domain29592960TempNewSymbol sym = SymbolTable::new_symbol(name);2961jclass result = find_class_from_class_loader(env, sym, true, loader, protection_domain, true, CHECK_NULL);29622963if (log_is_enabled(Debug, class, resolve) && result != NULL) {2964trace_class_resolution(java_lang_Class::as_Klass(JNIHandles::resolve_non_null(result)));2965}2966return result;2967}29682969// These lookups are done with the NULL (bootstrap) ClassLoader to2970// circumvent any security checks that would be done by jni_FindClass.2971JNI_ENTRY(bool, lookupDirectBufferClasses(JNIEnv* env))2972{2973if ((bufferClass = lookupOne(env, "java/nio/Buffer", thread)) == NULL) { return false; }2974if ((directBufferClass = lookupOne(env, "sun/nio/ch/DirectBuffer", thread)) == NULL) { return false; }2975if ((directByteBufferClass = lookupOne(env, "java/nio/DirectByteBuffer", thread)) == NULL) { return false; }2976return true;2977}2978JNI_END297929802981static bool initializeDirectBufferSupport(JNIEnv* env, JavaThread* thread) {2982if (directBufferSupportInitializeFailed) {2983return false;2984}29852986if (Atomic::cmpxchg(&directBufferSupportInitializeStarted, 0, 1) == 0) {2987if (!lookupDirectBufferClasses(env)) {2988directBufferSupportInitializeFailed = 1;2989return false;2990}29912992// Make global references for these2993bufferClass = (jclass) env->NewGlobalRef(bufferClass);2994directBufferClass = (jclass) env->NewGlobalRef(directBufferClass);2995directByteBufferClass = (jclass) env->NewGlobalRef(directByteBufferClass);29962997// Global refs will be NULL if out-of-memory (no exception is pending)2998if (bufferClass == NULL || directBufferClass == NULL || directByteBufferClass == NULL) {2999directBufferSupportInitializeFailed = 1;3000return false;3001}30023003// Get needed field and method IDs3004directByteBufferConstructor = env->GetMethodID(directByteBufferClass, "<init>", "(JI)V");3005if (env->ExceptionCheck()) {3006env->ExceptionClear();3007directBufferSupportInitializeFailed = 1;3008return false;3009}3010directBufferAddressField = env->GetFieldID(bufferClass, "address", "J");3011if (env->ExceptionCheck()) {3012env->ExceptionClear();3013directBufferSupportInitializeFailed = 1;3014return false;3015}3016bufferCapacityField = env->GetFieldID(bufferClass, "capacity", "I");3017if (env->ExceptionCheck()) {3018env->ExceptionClear();3019directBufferSupportInitializeFailed = 1;3020return false;3021}30223023if ((directByteBufferConstructor == NULL) ||3024(directBufferAddressField == NULL) ||3025(bufferCapacityField == NULL)) {3026directBufferSupportInitializeFailed = 1;3027return false;3028}30293030directBufferSupportInitializeEnded = 1;3031} else {3032while (!directBufferSupportInitializeEnded && !directBufferSupportInitializeFailed) {3033os::naked_yield();3034}3035}30363037return !directBufferSupportInitializeFailed;3038}30393040extern "C" jobject JNICALL jni_NewDirectByteBuffer(JNIEnv *env, void* address, jlong capacity)3041{3042// thread_from_jni_environment() will block if VM is gone.3043JavaThread* thread = JavaThread::thread_from_jni_environment(env);30443045HOTSPOT_JNI_NEWDIRECTBYTEBUFFER_ENTRY(env, address, capacity);30463047if (!directBufferSupportInitializeEnded) {3048if (!initializeDirectBufferSupport(env, thread)) {3049HOTSPOT_JNI_NEWDIRECTBYTEBUFFER_RETURN(NULL);3050return NULL;3051}3052}30533054// Being paranoid about accidental sign extension on address3055jlong addr = (jlong) ((uintptr_t) address);3056// NOTE that package-private DirectByteBuffer constructor currently3057// takes int capacity3058jint cap = (jint) capacity;3059jobject ret = env->NewObject(directByteBufferClass, directByteBufferConstructor, addr, cap);3060HOTSPOT_JNI_NEWDIRECTBYTEBUFFER_RETURN(ret);3061return ret;3062}30633064DT_RETURN_MARK_DECL(GetDirectBufferAddress, void*3065, HOTSPOT_JNI_GETDIRECTBUFFERADDRESS_RETURN((void*) _ret_ref));30663067extern "C" void* JNICALL jni_GetDirectBufferAddress(JNIEnv *env, jobject buf)3068{3069// thread_from_jni_environment() will block if VM is gone.3070JavaThread* thread = JavaThread::thread_from_jni_environment(env);30713072HOTSPOT_JNI_GETDIRECTBUFFERADDRESS_ENTRY(env, buf);3073void* ret = NULL;3074DT_RETURN_MARK(GetDirectBufferAddress, void*, (const void*&)ret);30753076if (!directBufferSupportInitializeEnded) {3077if (!initializeDirectBufferSupport(env, thread)) {3078return 0;3079}3080}30813082if ((buf != NULL) && (!env->IsInstanceOf(buf, directBufferClass))) {3083return 0;3084}30853086ret = (void*)(intptr_t)env->GetLongField(buf, directBufferAddressField);3087return ret;3088}30893090DT_RETURN_MARK_DECL(GetDirectBufferCapacity, jlong3091, HOTSPOT_JNI_GETDIRECTBUFFERCAPACITY_RETURN(_ret_ref));30923093extern "C" jlong JNICALL jni_GetDirectBufferCapacity(JNIEnv *env, jobject buf)3094{3095// thread_from_jni_environment() will block if VM is gone.3096JavaThread* thread = JavaThread::thread_from_jni_environment(env);30973098HOTSPOT_JNI_GETDIRECTBUFFERCAPACITY_ENTRY(env, buf);3099jlong ret = -1;3100DT_RETURN_MARK(GetDirectBufferCapacity, jlong, (const jlong&)ret);31013102if (!directBufferSupportInitializeEnded) {3103if (!initializeDirectBufferSupport(env, thread)) {3104ret = 0;3105return ret;3106}3107}31083109if (buf == NULL) {3110return -1;3111}31123113if (!env->IsInstanceOf(buf, directBufferClass)) {3114return -1;3115}31163117// NOTE that capacity is currently an int in the implementation3118ret = env->GetIntField(buf, bufferCapacityField);3119return ret;3120}312131223123JNI_LEAF(jint, jni_GetVersion(JNIEnv *env))3124HOTSPOT_JNI_GETVERSION_ENTRY(env);3125HOTSPOT_JNI_GETVERSION_RETURN(CurrentVersion);3126return CurrentVersion;3127JNI_END31283129extern struct JavaVM_ main_vm;31303131JNI_LEAF(jint, jni_GetJavaVM(JNIEnv *env, JavaVM **vm))3132HOTSPOT_JNI_GETJAVAVM_ENTRY(env, (void **) vm);3133*vm = (JavaVM *)(&main_vm);3134HOTSPOT_JNI_GETJAVAVM_RETURN(JNI_OK);3135return JNI_OK;3136JNI_END313731383139JNI_ENTRY(jobject, jni_GetModule(JNIEnv* env, jclass clazz))3140return Modules::get_module(clazz, THREAD);3141JNI_END314231433144// Structure containing all jni functions3145struct JNINativeInterface_ jni_NativeInterface = {3146NULL,3147NULL,3148NULL,31493150NULL,31513152jni_GetVersion,31533154jni_DefineClass,3155jni_FindClass,31563157jni_FromReflectedMethod,3158jni_FromReflectedField,31593160jni_ToReflectedMethod,31613162jni_GetSuperclass,3163jni_IsAssignableFrom,31643165jni_ToReflectedField,31663167jni_Throw,3168jni_ThrowNew,3169jni_ExceptionOccurred,3170jni_ExceptionDescribe,3171jni_ExceptionClear,3172jni_FatalError,31733174jni_PushLocalFrame,3175jni_PopLocalFrame,31763177jni_NewGlobalRef,3178jni_DeleteGlobalRef,3179jni_DeleteLocalRef,3180jni_IsSameObject,31813182jni_NewLocalRef,3183jni_EnsureLocalCapacity,31843185jni_AllocObject,3186jni_NewObject,3187jni_NewObjectV,3188jni_NewObjectA,31893190jni_GetObjectClass,3191jni_IsInstanceOf,31923193jni_GetMethodID,31943195jni_CallObjectMethod,3196jni_CallObjectMethodV,3197jni_CallObjectMethodA,3198jni_CallBooleanMethod,3199jni_CallBooleanMethodV,3200jni_CallBooleanMethodA,3201jni_CallByteMethod,3202jni_CallByteMethodV,3203jni_CallByteMethodA,3204jni_CallCharMethod,3205jni_CallCharMethodV,3206jni_CallCharMethodA,3207jni_CallShortMethod,3208jni_CallShortMethodV,3209jni_CallShortMethodA,3210jni_CallIntMethod,3211jni_CallIntMethodV,3212jni_CallIntMethodA,3213jni_CallLongMethod,3214jni_CallLongMethodV,3215jni_CallLongMethodA,3216jni_CallFloatMethod,3217jni_CallFloatMethodV,3218jni_CallFloatMethodA,3219jni_CallDoubleMethod,3220jni_CallDoubleMethodV,3221jni_CallDoubleMethodA,3222jni_CallVoidMethod,3223jni_CallVoidMethodV,3224jni_CallVoidMethodA,32253226jni_CallNonvirtualObjectMethod,3227jni_CallNonvirtualObjectMethodV,3228jni_CallNonvirtualObjectMethodA,3229jni_CallNonvirtualBooleanMethod,3230jni_CallNonvirtualBooleanMethodV,3231jni_CallNonvirtualBooleanMethodA,3232jni_CallNonvirtualByteMethod,3233jni_CallNonvirtualByteMethodV,3234jni_CallNonvirtualByteMethodA,3235jni_CallNonvirtualCharMethod,3236jni_CallNonvirtualCharMethodV,3237jni_CallNonvirtualCharMethodA,3238jni_CallNonvirtualShortMethod,3239jni_CallNonvirtualShortMethodV,3240jni_CallNonvirtualShortMethodA,3241jni_CallNonvirtualIntMethod,3242jni_CallNonvirtualIntMethodV,3243jni_CallNonvirtualIntMethodA,3244jni_CallNonvirtualLongMethod,3245jni_CallNonvirtualLongMethodV,3246jni_CallNonvirtualLongMethodA,3247jni_CallNonvirtualFloatMethod,3248jni_CallNonvirtualFloatMethodV,3249jni_CallNonvirtualFloatMethodA,3250jni_CallNonvirtualDoubleMethod,3251jni_CallNonvirtualDoubleMethodV,3252jni_CallNonvirtualDoubleMethodA,3253jni_CallNonvirtualVoidMethod,3254jni_CallNonvirtualVoidMethodV,3255jni_CallNonvirtualVoidMethodA,32563257jni_GetFieldID,32583259jni_GetObjectField,3260jni_GetBooleanField,3261jni_GetByteField,3262jni_GetCharField,3263jni_GetShortField,3264jni_GetIntField,3265jni_GetLongField,3266jni_GetFloatField,3267jni_GetDoubleField,32683269jni_SetObjectField,3270jni_SetBooleanField,3271jni_SetByteField,3272jni_SetCharField,3273jni_SetShortField,3274jni_SetIntField,3275jni_SetLongField,3276jni_SetFloatField,3277jni_SetDoubleField,32783279jni_GetStaticMethodID,32803281jni_CallStaticObjectMethod,3282jni_CallStaticObjectMethodV,3283jni_CallStaticObjectMethodA,3284jni_CallStaticBooleanMethod,3285jni_CallStaticBooleanMethodV,3286jni_CallStaticBooleanMethodA,3287jni_CallStaticByteMethod,3288jni_CallStaticByteMethodV,3289jni_CallStaticByteMethodA,3290jni_CallStaticCharMethod,3291jni_CallStaticCharMethodV,3292jni_CallStaticCharMethodA,3293jni_CallStaticShortMethod,3294jni_CallStaticShortMethodV,3295jni_CallStaticShortMethodA,3296jni_CallStaticIntMethod,3297jni_CallStaticIntMethodV,3298jni_CallStaticIntMethodA,3299jni_CallStaticLongMethod,3300jni_CallStaticLongMethodV,3301jni_CallStaticLongMethodA,3302jni_CallStaticFloatMethod,3303jni_CallStaticFloatMethodV,3304jni_CallStaticFloatMethodA,3305jni_CallStaticDoubleMethod,3306jni_CallStaticDoubleMethodV,3307jni_CallStaticDoubleMethodA,3308jni_CallStaticVoidMethod,3309jni_CallStaticVoidMethodV,3310jni_CallStaticVoidMethodA,33113312jni_GetStaticFieldID,33133314jni_GetStaticObjectField,3315jni_GetStaticBooleanField,3316jni_GetStaticByteField,3317jni_GetStaticCharField,3318jni_GetStaticShortField,3319jni_GetStaticIntField,3320jni_GetStaticLongField,3321jni_GetStaticFloatField,3322jni_GetStaticDoubleField,33233324jni_SetStaticObjectField,3325jni_SetStaticBooleanField,3326jni_SetStaticByteField,3327jni_SetStaticCharField,3328jni_SetStaticShortField,3329jni_SetStaticIntField,3330jni_SetStaticLongField,3331jni_SetStaticFloatField,3332jni_SetStaticDoubleField,33333334jni_NewString,3335jni_GetStringLength,3336jni_GetStringChars,3337jni_ReleaseStringChars,33383339jni_NewStringUTF,3340jni_GetStringUTFLength,3341jni_GetStringUTFChars,3342jni_ReleaseStringUTFChars,33433344jni_GetArrayLength,33453346jni_NewObjectArray,3347jni_GetObjectArrayElement,3348jni_SetObjectArrayElement,33493350jni_NewBooleanArray,3351jni_NewByteArray,3352jni_NewCharArray,3353jni_NewShortArray,3354jni_NewIntArray,3355jni_NewLongArray,3356jni_NewFloatArray,3357jni_NewDoubleArray,33583359jni_GetBooleanArrayElements,3360jni_GetByteArrayElements,3361jni_GetCharArrayElements,3362jni_GetShortArrayElements,3363jni_GetIntArrayElements,3364jni_GetLongArrayElements,3365jni_GetFloatArrayElements,3366jni_GetDoubleArrayElements,33673368jni_ReleaseBooleanArrayElements,3369jni_ReleaseByteArrayElements,3370jni_ReleaseCharArrayElements,3371jni_ReleaseShortArrayElements,3372jni_ReleaseIntArrayElements,3373jni_ReleaseLongArrayElements,3374jni_ReleaseFloatArrayElements,3375jni_ReleaseDoubleArrayElements,33763377jni_GetBooleanArrayRegion,3378jni_GetByteArrayRegion,3379jni_GetCharArrayRegion,3380jni_GetShortArrayRegion,3381jni_GetIntArrayRegion,3382jni_GetLongArrayRegion,3383jni_GetFloatArrayRegion,3384jni_GetDoubleArrayRegion,33853386jni_SetBooleanArrayRegion,3387jni_SetByteArrayRegion,3388jni_SetCharArrayRegion,3389jni_SetShortArrayRegion,3390jni_SetIntArrayRegion,3391jni_SetLongArrayRegion,3392jni_SetFloatArrayRegion,3393jni_SetDoubleArrayRegion,33943395jni_RegisterNatives,3396jni_UnregisterNatives,33973398jni_MonitorEnter,3399jni_MonitorExit,34003401jni_GetJavaVM,34023403jni_GetStringRegion,3404jni_GetStringUTFRegion,34053406jni_GetPrimitiveArrayCritical,3407jni_ReleasePrimitiveArrayCritical,34083409jni_GetStringCritical,3410jni_ReleaseStringCritical,34113412jni_NewWeakGlobalRef,3413jni_DeleteWeakGlobalRef,34143415jni_ExceptionCheck,34163417jni_NewDirectByteBuffer,3418jni_GetDirectBufferAddress,3419jni_GetDirectBufferCapacity,34203421// New 1_6 features34223423jni_GetObjectRefType,34243425// Module features34263427jni_GetModule3428};342934303431// For jvmti use to modify jni function table.3432// Java threads in native contiues to run until it is transitioned3433// to VM at safepoint. Before the transition or before it is blocked3434// for safepoint it may access jni function table. VM could crash if3435// any java thread access the jni function table in the middle of memcpy.3436// To avoid this each function pointers are copied automically.3437void copy_jni_function_table(const struct JNINativeInterface_ *new_jni_NativeInterface) {3438assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint");3439intptr_t *a = (intptr_t *) jni_functions();3440intptr_t *b = (intptr_t *) new_jni_NativeInterface;3441for (uint i=0; i < sizeof(struct JNINativeInterface_)/sizeof(void *); i++) {3442Atomic::store(a++, *b++);3443}3444}34453446void quicken_jni_functions() {3447// Replace Get<Primitive>Field with fast versions3448if (UseFastJNIAccessors && !VerifyJNIFields && !CheckJNICalls) {3449address func;3450func = JNI_FastGetField::generate_fast_get_boolean_field();3451if (func != (address)-1) {3452jni_NativeInterface.GetBooleanField = (GetBooleanField_t)func;3453}3454func = JNI_FastGetField::generate_fast_get_byte_field();3455if (func != (address)-1) {3456jni_NativeInterface.GetByteField = (GetByteField_t)func;3457}3458func = JNI_FastGetField::generate_fast_get_char_field();3459if (func != (address)-1) {3460jni_NativeInterface.GetCharField = (GetCharField_t)func;3461}3462func = JNI_FastGetField::generate_fast_get_short_field();3463if (func != (address)-1) {3464jni_NativeInterface.GetShortField = (GetShortField_t)func;3465}3466func = JNI_FastGetField::generate_fast_get_int_field();3467if (func != (address)-1) {3468jni_NativeInterface.GetIntField = (GetIntField_t)func;3469}3470func = JNI_FastGetField::generate_fast_get_long_field();3471if (func != (address)-1) {3472jni_NativeInterface.GetLongField = (GetLongField_t)func;3473}3474func = JNI_FastGetField::generate_fast_get_float_field();3475if (func != (address)-1) {3476jni_NativeInterface.GetFloatField = (GetFloatField_t)func;3477}3478func = JNI_FastGetField::generate_fast_get_double_field();3479if (func != (address)-1) {3480jni_NativeInterface.GetDoubleField = (GetDoubleField_t)func;3481}3482}3483}34843485// Returns the function structure3486struct JNINativeInterface_* jni_functions() {3487#if INCLUDE_JNI_CHECK3488if (CheckJNICalls) return jni_functions_check();3489#endif // INCLUDE_JNI_CHECK3490return &jni_NativeInterface;3491}34923493// Returns the function structure3494struct JNINativeInterface_* jni_functions_nocheck() {3495return &jni_NativeInterface;3496}34973498static void post_thread_start_event(const JavaThread* jt) {3499assert(jt != NULL, "invariant");3500EventThreadStart event;3501if (event.should_commit()) {3502event.set_thread(JFR_THREAD_ID(jt));3503event.set_parentThread((traceid)0);3504#if INCLUDE_JFR3505if (EventThreadStart::is_stacktrace_enabled()) {3506jt->jfr_thread_local()->set_cached_stack_trace_id((traceid)0);3507event.commit();3508jt->jfr_thread_local()->clear_cached_stack_trace();3509} else3510#endif3511{3512event.commit();3513}3514}3515}35163517// Invocation API351835193520// Forward declaration3521extern const struct JNIInvokeInterface_ jni_InvokeInterface;35223523// Global invocation API vars3524volatile int vm_created = 0;3525// Indicate whether it is safe to recreate VM. Recreation is only3526// possible after a failed initial creation attempt in some cases.3527volatile int safe_to_recreate_vm = 1;3528struct JavaVM_ main_vm = {&jni_InvokeInterface};352935303531#define JAVASTACKSIZE (400 * 1024) /* Default size of a thread java stack */3532enum { VERIFY_NONE, VERIFY_REMOTE, VERIFY_ALL };35333534DT_RETURN_MARK_DECL(GetDefaultJavaVMInitArgs, jint3535, HOTSPOT_JNI_GETDEFAULTJAVAVMINITARGS_RETURN(_ret_ref));35363537_JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_GetDefaultJavaVMInitArgs(void *args_) {3538HOTSPOT_JNI_GETDEFAULTJAVAVMINITARGS_ENTRY(args_);3539JDK1_1InitArgs *args = (JDK1_1InitArgs *)args_;3540jint ret = JNI_ERR;3541DT_RETURN_MARK(GetDefaultJavaVMInitArgs, jint, (const jint&)ret);35423543if (Threads::is_supported_jni_version(args->version)) {3544ret = JNI_OK;3545}3546// 1.1 style no longer supported in hotspot.3547// According the JNI spec, we should update args->version on return.3548// We also use the structure to communicate with launcher about default3549// stack size.3550if (args->version == JNI_VERSION_1_1) {3551args->version = JNI_VERSION_1_2;3552// javaStackSize is int in arguments structure3553assert(jlong(ThreadStackSize) * K < INT_MAX, "integer overflow");3554args->javaStackSize = (jint)(ThreadStackSize * K);3555}3556return ret;3557}35583559DT_RETURN_MARK_DECL(CreateJavaVM, jint3560, HOTSPOT_JNI_CREATEJAVAVM_RETURN(_ret_ref));35613562static jint JNI_CreateJavaVM_inner(JavaVM **vm, void **penv, void *args) {3563HOTSPOT_JNI_CREATEJAVAVM_ENTRY((void **) vm, penv, args);35643565jint result = JNI_ERR;3566DT_RETURN_MARK(CreateJavaVM, jint, (const jint&)result);35673568// We're about to use Atomic::xchg for synchronization. Some Zero3569// platforms use the GCC builtin __sync_lock_test_and_set for this,3570// but __sync_lock_test_and_set is not guaranteed to do what we want3571// on all architectures. So we check it works before relying on it.3572#if defined(ZERO) && defined(ASSERT)3573{3574jint a = 0xcafebabe;3575jint b = Atomic::xchg(&a, (jint) 0xdeadbeef);3576void *c = &a;3577void *d = Atomic::xchg(&c, &b);3578assert(a == (jint) 0xdeadbeef && b == (jint) 0xcafebabe, "Atomic::xchg() works");3579assert(c == &b && d == &a, "Atomic::xchg() works");3580}3581#endif // ZERO && ASSERT35823583// At the moment it's only possible to have one Java VM,3584// since some of the runtime state is in global variables.35853586// We cannot use our mutex locks here, since they only work on3587// Threads. We do an atomic compare and exchange to ensure only3588// one thread can call this method at a time35893590// We use Atomic::xchg rather than Atomic::add/dec since on some platforms3591// the add/dec implementations are dependent on whether we are running3592// on a multiprocessor Atomic::xchg does not have this problem.3593if (Atomic::xchg(&vm_created, 1) == 1) {3594return JNI_EEXIST; // already created, or create attempt in progress3595}35963597// If a previous creation attempt failed but can be retried safely,3598// then safe_to_recreate_vm will have been reset to 1 after being3599// cleared here. If a previous creation attempt succeeded and we then3600// destroyed that VM, we will be prevented from trying to recreate3601// the VM in the same process, as the value will still be 0.3602if (Atomic::xchg(&safe_to_recreate_vm, 0) == 0) {3603return JNI_ERR;3604}36053606assert(vm_created == 1, "vm_created is true during the creation");36073608/**3609* Certain errors during initialization are recoverable and do not3610* prevent this method from being called again at a later time3611* (perhaps with different arguments). However, at a certain3612* point during initialization if an error occurs we cannot allow3613* this function to be called again (or it will crash). In those3614* situations, the 'canTryAgain' flag is set to false, which atomically3615* sets safe_to_recreate_vm to 1, such that any new call to3616* JNI_CreateJavaVM will immediately fail using the above logic.3617*/3618bool can_try_again = true;36193620result = Threads::create_vm((JavaVMInitArgs*) args, &can_try_again);3621if (result == JNI_OK) {3622JavaThread *thread = JavaThread::current();3623assert(!thread->has_pending_exception(), "should have returned not OK");3624/* thread is thread_in_vm here */3625*vm = (JavaVM *)(&main_vm);3626*(JNIEnv**)penv = thread->jni_environment();36273628#if INCLUDE_JVMCI3629if (EnableJVMCI) {3630if (UseJVMCICompiler) {3631// JVMCI is initialized on a CompilerThread3632if (BootstrapJVMCI) {3633JavaThread* THREAD = thread; // For exception macros.3634JVMCICompiler* compiler = JVMCICompiler::instance(true, CATCH);3635compiler->bootstrap(THREAD);3636if (HAS_PENDING_EXCEPTION) {3637HandleMark hm(THREAD);3638vm_exit_during_initialization(Handle(THREAD, PENDING_EXCEPTION));3639}3640}3641}3642}3643#endif36443645// Notify JVMTI3646if (JvmtiExport::should_post_thread_life()) {3647JvmtiExport::post_thread_start(thread);3648}36493650post_thread_start_event(thread);36513652#ifndef PRODUCT3653if (ReplayCompiles) ciReplay::replay(thread);3654#endif36553656#ifdef ASSERT3657// Some platforms (like Win*) need a wrapper around these test3658// functions in order to properly handle error conditions.3659if (ErrorHandlerTest != 0) {3660VMError::controlled_crash(ErrorHandlerTest);3661}3662#endif36633664// Since this is not a JVM_ENTRY we have to set the thread state manually before leaving.3665ThreadStateTransition::transition(thread, _thread_in_vm, _thread_in_native);3666MACOS_AARCH64_ONLY(thread->enable_wx(WXExec));3667} else {3668// If create_vm exits because of a pending exception, exit with that3669// exception. In the future when we figure out how to reclaim memory,3670// we may be able to exit with JNI_ERR and allow the calling application3671// to continue.3672if (Universe::is_fully_initialized()) {3673// otherwise no pending exception possible - VM will already have aborted3674JavaThread* THREAD = JavaThread::current(); // For exception macros.3675if (HAS_PENDING_EXCEPTION) {3676HandleMark hm(THREAD);3677vm_exit_during_initialization(Handle(THREAD, PENDING_EXCEPTION));3678}3679}36803681if (can_try_again) {3682// reset safe_to_recreate_vm to 1 so that retrial would be possible3683safe_to_recreate_vm = 1;3684}36853686// Creation failed. We must reset vm_created3687*vm = 0;3688*(JNIEnv**)penv = 0;3689// reset vm_created last to avoid race condition. Use OrderAccess to3690// control both compiler and architectural-based reordering.3691Atomic::release_store(&vm_created, 0);3692}36933694// Flush stdout and stderr before exit.3695fflush(stdout);3696fflush(stderr);36973698return result;36993700}37013702_JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_CreateJavaVM(JavaVM **vm, void **penv, void *args) {3703jint result = JNI_ERR;3704// On Windows, let CreateJavaVM run with SEH protection3705#if defined(_WIN32) && !defined(USE_VECTORED_EXCEPTION_HANDLING)3706__try {3707#endif3708result = JNI_CreateJavaVM_inner(vm, penv, args);3709#if defined(_WIN32) && !defined(USE_VECTORED_EXCEPTION_HANDLING)3710} __except(topLevelExceptionFilter((_EXCEPTION_POINTERS*)_exception_info())) {3711// Nothing to do.3712}3713#endif3714return result;3715}37163717_JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_GetCreatedJavaVMs(JavaVM **vm_buf, jsize bufLen, jsize *numVMs) {3718HOTSPOT_JNI_GETCREATEDJAVAVMS_ENTRY((void **) vm_buf, bufLen, (uintptr_t *) numVMs);37193720if (vm_created == 1) {3721if (numVMs != NULL) *numVMs = 1;3722if (bufLen > 0) *vm_buf = (JavaVM *)(&main_vm);3723} else {3724if (numVMs != NULL) *numVMs = 0;3725}3726HOTSPOT_JNI_GETCREATEDJAVAVMS_RETURN(JNI_OK);3727return JNI_OK;3728}37293730extern "C" {37313732DT_RETURN_MARK_DECL(DestroyJavaVM, jint3733, HOTSPOT_JNI_DESTROYJAVAVM_RETURN(_ret_ref));37343735static jint JNICALL jni_DestroyJavaVM_inner(JavaVM *vm) {3736HOTSPOT_JNI_DESTROYJAVAVM_ENTRY(vm);3737jint res = JNI_ERR;3738DT_RETURN_MARK(DestroyJavaVM, jint, (const jint&)res);37393740if (vm_created == 0) {3741res = JNI_ERR;3742return res;3743}37443745JNIEnv *env;3746JavaVMAttachArgs destroyargs;3747destroyargs.version = CurrentVersion;3748destroyargs.name = (char *)"DestroyJavaVM";3749destroyargs.group = NULL;3750res = vm->AttachCurrentThread((void **)&env, (void *)&destroyargs);3751if (res != JNI_OK) {3752return res;3753}37543755// Since this is not a JVM_ENTRY we have to set the thread state manually before entering.3756JavaThread* thread = JavaThread::current();37573758// We are going to VM, change W^X state to the expected one.3759MACOS_AARCH64_ONLY(WXMode oldmode = thread->enable_wx(WXWrite));37603761ThreadStateTransition::transition_from_native(thread, _thread_in_vm);3762Threads::destroy_vm();3763// Don't bother restoring thread state, VM is gone.3764vm_created = 0;3765return JNI_OK;3766}37673768jint JNICALL jni_DestroyJavaVM(JavaVM *vm) {3769jint result = JNI_ERR;3770// On Windows, we need SEH protection3771#if defined(_WIN32) && !defined(USE_VECTORED_EXCEPTION_HANDLING)3772__try {3773#endif3774result = jni_DestroyJavaVM_inner(vm);3775#if defined(_WIN32) && !defined(USE_VECTORED_EXCEPTION_HANDLING)3776} __except(topLevelExceptionFilter((_EXCEPTION_POINTERS*)_exception_info())) {3777// Nothing to do.3778}3779#endif3780return result;3781}37823783static jint attach_current_thread(JavaVM *vm, void **penv, void *_args, bool daemon) {3784JavaVMAttachArgs *args = (JavaVMAttachArgs *) _args;37853786// Check below commented out from JDK1.2fcs as well3787/*3788if (args && (args->version != JNI_VERSION_1_1 || args->version != JNI_VERSION_1_2)) {3789return JNI_EVERSION;3790}3791*/37923793Thread* t = Thread::current_or_null();3794if (t != NULL) {3795// If executing from an atexit hook we may be in the VMThread.3796if (t->is_Java_thread()) {3797// If the thread has been attached this operation is a no-op3798*(JNIEnv**)penv = t->as_Java_thread()->jni_environment();3799return JNI_OK;3800} else {3801return JNI_ERR;3802}3803}38043805// Create a thread and mark it as attaching so it will be skipped by the3806// ThreadsListEnumerator - see CR 64043063807JavaThread* thread = new JavaThread(true);38083809// Set correct safepoint info. The thread is going to call into Java when3810// initializing the Java level thread object. Hence, the correct state must3811// be set in order for the Safepoint code to deal with it correctly.3812thread->set_thread_state(_thread_in_vm);3813thread->record_stack_base_and_size();3814thread->register_thread_stack_with_NMT();3815thread->initialize_thread_current();3816MACOS_AARCH64_ONLY(thread->init_wx());38173818if (!os::create_attached_thread(thread)) {3819thread->smr_delete();3820return JNI_ERR;3821}3822// Enable stack overflow checks3823thread->stack_overflow_state()->create_stack_guard_pages();38243825thread->initialize_tlab();38263827thread->cache_global_variables();38283829// This thread will not do a safepoint check, since it has3830// not been added to the Thread list yet.3831{ MutexLocker ml(Threads_lock);3832// This must be inside this lock in order to get FullGCALot to work properly, i.e., to3833// avoid this thread trying to do a GC before it is added to the thread-list3834thread->set_active_handles(JNIHandleBlock::allocate_block());3835Threads::add(thread, daemon);3836}3837// Create thread group and name info from attach arguments3838oop group = NULL;3839char* thread_name = NULL;3840if (args != NULL && Threads::is_supported_jni_version(args->version)) {3841group = JNIHandles::resolve(args->group);3842thread_name = args->name; // may be NULL3843}3844if (group == NULL) group = Universe::main_thread_group();38453846// Create Java level thread object and attach it to this thread3847bool attach_failed = false;3848{3849EXCEPTION_MARK;3850HandleMark hm(THREAD);3851Handle thread_group(THREAD, group);3852thread->allocate_threadObj(thread_group, thread_name, daemon, THREAD);3853if (HAS_PENDING_EXCEPTION) {3854CLEAR_PENDING_EXCEPTION;3855// cleanup outside the handle mark.3856attach_failed = true;3857}3858}38593860if (attach_failed) {3861// Added missing cleanup3862thread->cleanup_failed_attach_current_thread(daemon);3863return JNI_ERR;3864}38653866// mark the thread as no longer attaching3867// this uses a fence to push the change through so we don't have3868// to regrab the threads_lock3869thread->set_done_attaching_via_jni();38703871// Set java thread status.3872java_lang_Thread::set_thread_status(thread->threadObj(),3873JavaThreadStatus::RUNNABLE);38743875// Notify the debugger3876if (JvmtiExport::should_post_thread_life()) {3877JvmtiExport::post_thread_start(thread);3878}38793880post_thread_start_event(thread);38813882*(JNIEnv**)penv = thread->jni_environment();38833884// Now leaving the VM, so change thread_state. This is normally automatically taken care3885// of in the JVM_ENTRY. But in this situation we have to do it manually. Notice, that by3886// using ThreadStateTransition::transition, we do a callback to the safepoint code if3887// needed.38883889ThreadStateTransition::transition(thread, _thread_in_vm, _thread_in_native);3890MACOS_AARCH64_ONLY(thread->enable_wx(WXExec));38913892// Perform any platform dependent FPU setup3893os::setup_fpu();38943895return JNI_OK;3896}389738983899jint JNICALL jni_AttachCurrentThread(JavaVM *vm, void **penv, void *_args) {3900HOTSPOT_JNI_ATTACHCURRENTTHREAD_ENTRY(vm, penv, _args);3901if (vm_created == 0) {3902HOTSPOT_JNI_ATTACHCURRENTTHREAD_RETURN((uint32_t) JNI_ERR);3903return JNI_ERR;3904}39053906jint ret = attach_current_thread(vm, penv, _args, false);3907HOTSPOT_JNI_ATTACHCURRENTTHREAD_RETURN(ret);3908return ret;3909}391039113912jint JNICALL jni_DetachCurrentThread(JavaVM *vm) {3913HOTSPOT_JNI_DETACHCURRENTTHREAD_ENTRY(vm);3914if (vm_created == 0) {3915HOTSPOT_JNI_DETACHCURRENTTHREAD_RETURN(JNI_ERR);3916return JNI_ERR;3917}39183919Thread* current = Thread::current_or_null();39203921// If the thread has already been detached the operation is a no-op3922if (current == NULL) {3923HOTSPOT_JNI_DETACHCURRENTTHREAD_RETURN(JNI_OK);3924return JNI_OK;3925}39263927// If executing from an atexit hook we may be in the VMThread.3928if (!current->is_Java_thread()) {3929HOTSPOT_JNI_DETACHCURRENTTHREAD_RETURN((uint32_t) JNI_ERR);3930return JNI_ERR;3931}39323933VM_Exit::block_if_vm_exited();39343935JavaThread* thread = current->as_Java_thread();3936if (thread->has_last_Java_frame()) {3937HOTSPOT_JNI_DETACHCURRENTTHREAD_RETURN((uint32_t) JNI_ERR);3938// Can't detach a thread that's running java, that can't work.3939return JNI_ERR;3940}39413942// We are going to VM, change W^X state to the expected one.3943MACOS_AARCH64_ONLY(thread->enable_wx(WXWrite));39443945// Safepoint support. Have to do call-back to safepoint code, if in the3946// middle of a safepoint operation3947ThreadStateTransition::transition_from_native(thread, _thread_in_vm);39483949// XXX: Note that JavaThread::exit() call below removes the guards on the3950// stack pages set up via enable_stack_{red,yellow}_zone() calls3951// above in jni_AttachCurrentThread. Unfortunately, while the setting3952// of the guards is visible in jni_AttachCurrentThread above,3953// the removal of the guards is buried below in JavaThread::exit()3954// here. The abstraction should be more symmetrically either exposed3955// or hidden (e.g. it could probably be hidden in the same3956// (platform-dependent) methods where we do alternate stack3957// maintenance work?)3958thread->exit(false, JavaThread::jni_detach);3959thread->smr_delete();39603961// Go to the execute mode, the initial state of the thread on creation.3962// Use os interface as the thread is not a JavaThread anymore.3963MACOS_AARCH64_ONLY(os::current_thread_enable_wx(WXExec));39643965HOTSPOT_JNI_DETACHCURRENTTHREAD_RETURN(JNI_OK);3966return JNI_OK;3967}39683969DT_RETURN_MARK_DECL(GetEnv, jint3970, HOTSPOT_JNI_GETENV_RETURN(_ret_ref));39713972jint JNICALL jni_GetEnv(JavaVM *vm, void **penv, jint version) {3973HOTSPOT_JNI_GETENV_ENTRY(vm, penv, version);3974jint ret = JNI_ERR;3975DT_RETURN_MARK(GetEnv, jint, (const jint&)ret);39763977if (vm_created == 0) {3978*penv = NULL;3979ret = JNI_EDETACHED;3980return ret;3981}39823983if (JniExportedInterface::GetExportedInterface(vm, penv, version, &ret)) {3984return ret;3985}39863987#ifndef JVMPI_VERSION_13988// need these in order to be polite about older agents3989#define JVMPI_VERSION_1 ((jint)0x10000001)3990#define JVMPI_VERSION_1_1 ((jint)0x10000002)3991#define JVMPI_VERSION_1_2 ((jint)0x10000003)3992#endif // !JVMPI_VERSION_139933994Thread* thread = Thread::current_or_null();3995if (thread != NULL && thread->is_Java_thread()) {3996if (Threads::is_supported_jni_version_including_1_1(version)) {3997*(JNIEnv**)penv = thread->as_Java_thread()->jni_environment();3998ret = JNI_OK;3999return ret;40004001} else if (version == JVMPI_VERSION_1 ||4002version == JVMPI_VERSION_1_1 ||4003version == JVMPI_VERSION_1_2) {4004tty->print_cr("ERROR: JVMPI, an experimental interface, is no longer supported.");4005tty->print_cr("Please use the supported interface: the JVM Tool Interface (JVM TI).");4006ret = JNI_EVERSION;4007return ret;4008} else if (JvmtiExport::is_jvmdi_version(version)) {4009tty->print_cr("FATAL ERROR: JVMDI is no longer supported.");4010tty->print_cr("Please use the supported interface: the JVM Tool Interface (JVM TI).");4011ret = JNI_EVERSION;4012return ret;4013} else {4014*penv = NULL;4015ret = JNI_EVERSION;4016return ret;4017}4018} else {4019*penv = NULL;4020ret = JNI_EDETACHED;4021return ret;4022}4023}402440254026jint JNICALL jni_AttachCurrentThreadAsDaemon(JavaVM *vm, void **penv, void *_args) {4027HOTSPOT_JNI_ATTACHCURRENTTHREADASDAEMON_ENTRY(vm, penv, _args);4028if (vm_created == 0) {4029HOTSPOT_JNI_ATTACHCURRENTTHREADASDAEMON_RETURN((uint32_t) JNI_ERR);4030return JNI_ERR;4031}40324033jint ret = attach_current_thread(vm, penv, _args, true);4034HOTSPOT_JNI_ATTACHCURRENTTHREADASDAEMON_RETURN(ret);4035return ret;4036}403740384039} // End extern "C"40404041const struct JNIInvokeInterface_ jni_InvokeInterface = {4042NULL,4043NULL,4044NULL,40454046jni_DestroyJavaVM,4047jni_AttachCurrentThread,4048jni_DetachCurrentThread,4049jni_GetEnv,4050jni_AttachCurrentThreadAsDaemon4051};405240534054