1 /*
2 * Copyright (c) 1997, 2024, Oracle and/or its affiliates. All rights reserved.
3 * Copyright (c) 2012, 2024 Red Hat, Inc.
4 * Copyright (c) 2021, Azul Systems, Inc. All rights reserved.
5 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
6 *
7 * This code is free software; you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License version 2 only, as
9 * published by the Free Software Foundation.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22 * or visit www.oracle.com if you need additional information or have any
38 #include "classfile/symbolTable.hpp"
39 #include "classfile/systemDictionary.hpp"
40 #include "classfile/vmClasses.hpp"
41 #include "classfile/vmSymbols.hpp"
42 #include "compiler/compiler_globals.hpp"
43 #include "gc/shared/collectedHeap.hpp"
44 #include "gc/shared/gcLocker.inline.hpp"
45 #include "gc/shared/stringdedup/stringDedup.hpp"
46 #include "interpreter/linkResolver.hpp"
47 #include "jni.h"
48 #include "jvm.h"
49 #include "logging/log.hpp"
50 #include "memory/allocation.hpp"
51 #include "memory/allocation.inline.hpp"
52 #include "memory/oopFactory.hpp"
53 #include "memory/resourceArea.hpp"
54 #include "memory/universe.hpp"
55 #include "nmt/memTracker.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/arguments.hpp"
76 #include "runtime/atomic.hpp"
77 #include "runtime/fieldDescriptor.inline.hpp"
409 int modifiers = java_lang_reflect_Field::modifiers(reflected);
410
411 // Make sure class is initialized before handing id's out to fields
412 k1->initialize(CHECK_NULL);
413
414 // First check if this is a static field
415 if (modifiers & JVM_ACC_STATIC) {
416 int offset = InstanceKlass::cast(k1)->field_offset( slot );
417 JNIid* id = InstanceKlass::cast(k1)->jni_id_for(offset);
418 assert(id != nullptr, "corrupt Field object");
419 debug_only(id->set_is_static_field_id();)
420 // A jfieldID for a static field is a JNIid specifying the field holder and the offset within the Klass*
421 ret = jfieldIDWorkaround::to_static_jfieldID(id);
422 return ret;
423 }
424
425 // The slot is the index of the field description in the field-array
426 // The jfieldID is the offset of the field within the object
427 // It may also have hash bits for k, if VerifyJNIFields is turned on.
428 int offset = InstanceKlass::cast(k1)->field_offset( slot );
429 assert(InstanceKlass::cast(k1)->contains_field_offset(offset), "stay within object");
430 ret = jfieldIDWorkaround::to_instance_jfieldID(k1, offset);
431 return ret;
432 JNI_END
433
434
435 DT_RETURN_MARK_DECL(ToReflectedMethod, jobject
436 , HOTSPOT_JNI_TOREFLECTEDMETHOD_RETURN(_ret_ref));
437
438 JNI_ENTRY(jobject, jni_ToReflectedMethod(JNIEnv *env, jclass cls, jmethodID method_id, jboolean isStatic))
439 HOTSPOT_JNI_TOREFLECTEDMETHOD_ENTRY(env, cls, (uintptr_t) method_id, isStatic);
440
441 jobject ret = nullptr;
442 DT_RETURN_MARK(ToReflectedMethod, jobject, (const jobject&)ret);
443
444 methodHandle m (THREAD, Method::resolve_jmethod_id(method_id));
445 assert(m->is_static() == (isStatic != 0), "jni_ToReflectedMethod access flags doesn't match");
446 oop reflection_method;
447 if (m->is_object_initializer()) {
448 reflection_method = Reflection::new_constructor(m, CHECK_NULL);
449 } else {
450 // Note: Static initializers can theoretically be here, if JNI users manage
451 // to get their jmethodID. Record them as plain methods.
452 reflection_method = Reflection::new_method(m, false, CHECK_NULL);
453 }
454 ret = JNIHandles::make_local(THREAD, reflection_method);
455 return ret;
456 JNI_END
457
458 DT_RETURN_MARK_DECL(GetSuperclass, jclass
459 , HOTSPOT_JNI_GETSUPERCLASS_RETURN(_ret_ref));
460
461 JNI_ENTRY(jclass, jni_GetSuperclass(JNIEnv *env, jclass sub))
462 HOTSPOT_JNI_GETSUPERCLASS_ENTRY(env, sub);
463
464 jclass obj = nullptr;
465 DT_RETURN_MARK(GetSuperclass, jclass, (const jclass&)obj);
466
467 oop mirror = JNIHandles::resolve_non_null(sub);
787 }
788
789 friend class SignatureIterator; // so do_parameters_on can call do_type
790 void do_type(BasicType type) {
791 switch (type) {
792 // these are coerced to int when using va_arg
793 case T_BYTE:
794 case T_CHAR:
795 case T_SHORT:
796 case T_INT: push_int(va_arg(_ap, jint)); break;
797 case T_BOOLEAN: push_boolean((jboolean) va_arg(_ap, jint)); break;
798
799 // each of these paths is exercised by the various jck Call[Static,Nonvirtual,][Void,Int,..]Method[A,V,] tests
800
801 case T_LONG: push_long(va_arg(_ap, jlong)); break;
802 // float is coerced to double w/ va_arg
803 case T_FLOAT: push_float((jfloat) va_arg(_ap, jdouble)); break;
804 case T_DOUBLE: push_double(va_arg(_ap, jdouble)); break;
805
806 case T_ARRAY:
807 case T_OBJECT: push_object(va_arg(_ap, jobject)); break;
808 default: ShouldNotReachHere();
809 }
810 }
811
812 public:
813 JNI_ArgumentPusherVaArg(jmethodID method_id, va_list rap)
814 : JNI_ArgumentPusher(Method::resolve_jmethod_id(method_id)) {
815 set_ap(rap);
816 }
817
818 ~JNI_ArgumentPusherVaArg() {
819 va_end(_ap);
820 }
821
822 virtual void push_arguments_on(JavaCallArguments* arguments) {
823 _arguments = arguments;
824 do_parameters_on(this);
825 }
826 };
827
828
829 class JNI_ArgumentPusherArray : public JNI_ArgumentPusher {
830 protected:
831 const jvalue *_ap;
832
833 inline void set_ap(const jvalue *rap) { _ap = rap; }
834
835 friend class SignatureIterator; // so do_parameters_on can call do_type
836 void do_type(BasicType type) {
837 switch (type) {
838 case T_CHAR: push_int((_ap++)->c); break;
839 case T_SHORT: push_int((_ap++)->s); break;
840 case T_BYTE: push_int((_ap++)->b); break;
841 case T_INT: push_int((_ap++)->i); break;
842 case T_BOOLEAN: push_boolean((_ap++)->z); break;
843 case T_LONG: push_long((_ap++)->j); break;
844 case T_FLOAT: push_float((_ap++)->f); break;
845 case T_DOUBLE: push_double((_ap++)->d); break;
846 case T_ARRAY:
847 case T_OBJECT: push_object((_ap++)->l); break;
848 default: ShouldNotReachHere();
849 }
850 }
851
852 public:
853 JNI_ArgumentPusherArray(jmethodID method_id, const jvalue *rap)
854 : JNI_ArgumentPusher(Method::resolve_jmethod_id(method_id)) {
855 set_ap(rap);
856 }
857
858 virtual void push_arguments_on(JavaCallArguments* arguments) {
859 _arguments = arguments;
860 do_parameters_on(this);
861 }
862 };
863
864
865 enum JNICallType {
866 JNI_STATIC,
867 JNI_VIRTUAL,
954 result->set_type(args->return_type());
955
956 // Invoke the method. Result is returned as oop.
957 JavaCalls::call(result, method, &java_args, CHECK);
958
959 // Convert result
960 if (is_reference_type(result->get_type())) {
961 result->set_jobject(JNIHandles::make_local(THREAD, result->get_oop()));
962 }
963 }
964
965 DT_RETURN_MARK_DECL(AllocObject, jobject
966 , HOTSPOT_JNI_ALLOCOBJECT_RETURN(_ret_ref));
967
968 JNI_ENTRY(jobject, jni_AllocObject(JNIEnv *env, jclass clazz))
969 HOTSPOT_JNI_ALLOCOBJECT_ENTRY(env, clazz);
970
971 jobject ret = nullptr;
972 DT_RETURN_MARK(AllocObject, jobject, (const jobject&)ret);
973
974 instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);
975 ret = JNIHandles::make_local(THREAD, i);
976 return ret;
977 JNI_END
978
979 DT_RETURN_MARK_DECL(NewObjectA, jobject
980 , HOTSPOT_JNI_NEWOBJECTA_RETURN(_ret_ref));
981
982 JNI_ENTRY(jobject, jni_NewObjectA(JNIEnv *env, jclass clazz, jmethodID methodID, const jvalue *args))
983 HOTSPOT_JNI_NEWOBJECTA_ENTRY(env, clazz, (uintptr_t) methodID);
984
985 jobject obj = nullptr;
986 DT_RETURN_MARK(NewObjectA, jobject, (const jobject&)obj);
987
988 instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);
989 obj = JNIHandles::make_local(THREAD, i);
990 JavaValue jvalue(T_VOID);
991 JNI_ArgumentPusherArray ap(methodID, args);
992 jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);
993 return obj;
994 JNI_END
995
996
997 DT_RETURN_MARK_DECL(NewObjectV, jobject
998 , HOTSPOT_JNI_NEWOBJECTV_RETURN(_ret_ref));
999
1000 JNI_ENTRY(jobject, jni_NewObjectV(JNIEnv *env, jclass clazz, jmethodID methodID, va_list args))
1001 HOTSPOT_JNI_NEWOBJECTV_ENTRY(env, clazz, (uintptr_t) methodID);
1002
1003 jobject obj = nullptr;
1004 DT_RETURN_MARK(NewObjectV, jobject, (const jobject&)obj);
1005
1006 instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);
1007 obj = JNIHandles::make_local(THREAD, i);
1008 JavaValue jvalue(T_VOID);
1009 JNI_ArgumentPusherVaArg ap(methodID, args);
1010 jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);
1011 return obj;
1012 JNI_END
1013
1014
1015 DT_RETURN_MARK_DECL(NewObject, jobject
1016 , HOTSPOT_JNI_NEWOBJECT_RETURN(_ret_ref));
1017
1018 JNI_ENTRY(jobject, jni_NewObject(JNIEnv *env, jclass clazz, jmethodID methodID, ...))
1019 HOTSPOT_JNI_NEWOBJECT_ENTRY(env, clazz, (uintptr_t) methodID);
1020
1021 jobject obj = nullptr;
1022 DT_RETURN_MARK(NewObject, jobject, (const jobject&)obj);
1023
1024 instanceOop i = InstanceKlass::allocate_instance(JNIHandles::resolve_non_null(clazz), CHECK_NULL);
1025 obj = JNIHandles::make_local(THREAD, i);
1026 va_list args;
1027 va_start(args, methodID);
1028 JavaValue jvalue(T_VOID);
1029 JNI_ArgumentPusherVaArg ap(methodID, args);
1030 jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);
1031 va_end(args);
1032 return obj;
1033 JNI_END
1034
1035
1036 JNI_ENTRY(jclass, jni_GetObjectClass(JNIEnv *env, jobject obj))
1037 HOTSPOT_JNI_GETOBJECTCLASS_ENTRY(env, obj);
1038
1039 Klass* k = JNIHandles::resolve_non_null(obj)->klass();
1040 jclass ret =
1041 (jclass) JNIHandles::make_local(THREAD, k->java_mirror());
1042
1043 HOTSPOT_JNI_GETOBJECTCLASS_RETURN(ret);
1044 return ret;
1045 JNI_END
1046
1047 JNI_ENTRY_NO_PRESERVE(jboolean, jni_IsInstanceOf(JNIEnv *env, jobject obj, jclass clazz))
1048 HOTSPOT_JNI_ISINSTANCEOF_ENTRY(env, obj, clazz);
1049
1050 jboolean ret = JNI_TRUE;
1051 if (obj != nullptr) {
1762 // table. If they're not there, the field doesn't exist.
1763 TempNewSymbol fieldname = SymbolTable::probe(name, (int)strlen(name));
1764 TempNewSymbol signame = SymbolTable::probe(sig, (int)strlen(sig));
1765 if (fieldname == nullptr || signame == nullptr) {
1766 ResourceMark rm;
1767 THROW_MSG_NULL(vmSymbols::java_lang_NoSuchFieldError(), err_msg("%s.%s %s", k->external_name(), name, sig));
1768 }
1769
1770 // Make sure class is initialized before handing id's out to fields
1771 k->initialize(CHECK_NULL);
1772
1773 fieldDescriptor fd;
1774 if (!k->is_instance_klass() ||
1775 !InstanceKlass::cast(k)->find_field(fieldname, signame, false, &fd)) {
1776 ResourceMark rm;
1777 THROW_MSG_NULL(vmSymbols::java_lang_NoSuchFieldError(), err_msg("%s.%s %s", k->external_name(), name, sig));
1778 }
1779
1780 // A jfieldID for a non-static field is simply the offset of the field within the instanceOop
1781 // It may also have hash bits for k, if VerifyJNIFields is turned on.
1782 ret = jfieldIDWorkaround::to_instance_jfieldID(k, fd.offset());
1783 return ret;
1784 JNI_END
1785
1786
1787 JNI_ENTRY(jobject, jni_GetObjectField(JNIEnv *env, jobject obj, jfieldID fieldID))
1788 HOTSPOT_JNI_GETOBJECTFIELD_ENTRY(env, obj, (uintptr_t) fieldID);
1789 oop o = JNIHandles::resolve_non_null(obj);
1790 Klass* k = o->klass();
1791 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID);
1792 // Keep JVMTI addition small and only check enabled flag here.
1793 // jni_GetField_probe() assumes that is okay to create handles.
1794 if (JvmtiExport::should_post_field_access()) {
1795 o = JvmtiExport::jni_GetField_probe(thread, obj, o, k, fieldID, false);
1796 }
1797 oop loaded_obj = HeapAccess<ON_UNKNOWN_OOP_REF>::oop_load_at(o, offset);
1798 jobject ret = JNIHandles::make_local(THREAD, loaded_obj);
1799 HOTSPOT_JNI_GETOBJECTFIELD_RETURN(ret);
1800 return ret;
1801 JNI_END
1802
1803
1804
1805 #define DEFINE_GETFIELD(Return,Fieldname,Result \
1806 , EntryProbe, ReturnProbe) \
1807 \
1808 DT_RETURN_MARK_DECL_FOR(Result, Get##Result##Field, Return \
1809 , ReturnProbe); \
1810 \
1811 JNI_ENTRY_NO_PRESERVE(Return, jni_Get##Result##Field(JNIEnv *env, jobject obj, jfieldID fieldID)) \
1812 \
1813 EntryProbe; \
1814 Return ret = 0;\
1815 DT_RETURN_MARK_FOR(Result, Get##Result##Field, Return, (const Return&)ret);\
1816 \
1817 oop o = JNIHandles::resolve_non_null(obj); \
1818 Klass* k = o->klass(); \
1819 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID); \
1820 /* Keep JVMTI addition small and only check enabled flag here. */ \
1821 if (JvmtiExport::should_post_field_access()) { \
1822 o = JvmtiExport::jni_GetField_probe(thread, obj, o, k, fieldID, false); \
1823 } \
1824 ret = o->Fieldname##_field(offset); \
1870 return (address)jni_GetLongField;
1871 }
1872 address jni_GetFloatField_addr() {
1873 return (address)jni_GetFloatField;
1874 }
1875 address jni_GetDoubleField_addr() {
1876 return (address)jni_GetDoubleField;
1877 }
1878
1879 JNI_ENTRY_NO_PRESERVE(void, jni_SetObjectField(JNIEnv *env, jobject obj, jfieldID fieldID, jobject value))
1880 HOTSPOT_JNI_SETOBJECTFIELD_ENTRY(env, obj, (uintptr_t) fieldID, value);
1881 oop o = JNIHandles::resolve_non_null(obj);
1882 Klass* k = o->klass();
1883 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID);
1884 // Keep JVMTI addition small and only check enabled flag here.
1885 if (JvmtiExport::should_post_field_modification()) {
1886 jvalue field_value;
1887 field_value.l = value;
1888 o = JvmtiExport::jni_SetField_probe(thread, obj, o, k, fieldID, false, JVM_SIGNATURE_CLASS, (jvalue *)&field_value);
1889 }
1890 HeapAccess<ON_UNKNOWN_OOP_REF>::oop_store_at(o, offset, JNIHandles::resolve(value));
1891 HOTSPOT_JNI_SETOBJECTFIELD_RETURN();
1892 JNI_END
1893
1894 // TODO: make this a template
1895
1896 #define DEFINE_SETFIELD(Argument,Fieldname,Result,SigType,unionType \
1897 , EntryProbe, ReturnProbe) \
1898 \
1899 JNI_ENTRY_NO_PRESERVE(void, jni_Set##Result##Field(JNIEnv *env, jobject obj, jfieldID fieldID, Argument value)) \
1900 \
1901 EntryProbe; \
1902 \
1903 oop o = JNIHandles::resolve_non_null(obj); \
1904 Klass* k = o->klass(); \
1905 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID); \
1906 /* Keep JVMTI addition small and only check enabled flag here. */ \
1907 if (JvmtiExport::should_post_field_modification()) { \
1908 jvalue field_value; \
1909 field_value.unionType = value; \
1910 o = JvmtiExport::jni_SetField_probe(thread, obj, o, k, fieldID, false, SigType, (jvalue *)&field_value); \
2295 Klass* ak = ek->array_klass(CHECK_NULL);
2296 ObjArrayKlass::cast(ak)->initialize(CHECK_NULL);
2297 objArrayOop result = ObjArrayKlass::cast(ak)->allocate(length, CHECK_NULL);
2298 oop initial_value = JNIHandles::resolve(initialElement);
2299 if (initial_value != nullptr) { // array already initialized with null
2300 for (int index = 0; index < length; index++) {
2301 result->obj_at_put(index, initial_value);
2302 }
2303 }
2304 ret = (jobjectArray) JNIHandles::make_local(THREAD, result);
2305 return ret;
2306 JNI_END
2307
2308 DT_RETURN_MARK_DECL(GetObjectArrayElement, jobject
2309 , HOTSPOT_JNI_GETOBJECTARRAYELEMENT_RETURN(_ret_ref));
2310
2311 JNI_ENTRY(jobject, jni_GetObjectArrayElement(JNIEnv *env, jobjectArray array, jsize index))
2312 HOTSPOT_JNI_GETOBJECTARRAYELEMENT_ENTRY(env, array, index);
2313 jobject ret = nullptr;
2314 DT_RETURN_MARK(GetObjectArrayElement, jobject, (const jobject&)ret);
2315 objArrayOop a = objArrayOop(JNIHandles::resolve_non_null(array));
2316 if (a->is_within_bounds(index)) {
2317 ret = JNIHandles::make_local(THREAD, a->obj_at(index));
2318 return ret;
2319 } else {
2320 ResourceMark rm(THREAD);
2321 stringStream ss;
2322 ss.print("Index %d out of bounds for length %d", index, a->length());
2323 THROW_MSG_NULL(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());
2324 }
2325 JNI_END
2326
2327 DT_VOID_RETURN_MARK_DECL(SetObjectArrayElement
2328 , HOTSPOT_JNI_SETOBJECTARRAYELEMENT_RETURN());
2329
2330 JNI_ENTRY(void, jni_SetObjectArrayElement(JNIEnv *env, jobjectArray array, jsize index, jobject value))
2331 HOTSPOT_JNI_SETOBJECTARRAYELEMENT_ENTRY(env, array, index, value);
2332 DT_VOID_RETURN_MARK(SetObjectArrayElement);
2333
2334 objArrayOop a = objArrayOop(JNIHandles::resolve_non_null(array));
2335 oop v = JNIHandles::resolve(value);
2336 if (a->is_within_bounds(index)) {
2337 if (v == nullptr || v->is_a(ObjArrayKlass::cast(a->klass())->element_klass())) {
2338 a->obj_at_put(index, v);
2339 } else {
2340 ResourceMark rm(THREAD);
2341 stringStream ss;
2342 Klass *bottom_kl = ObjArrayKlass::cast(a->klass())->bottom_klass();
2343 ss.print("type mismatch: can not store %s to %s[%d]",
2344 v->klass()->external_name(),
2345 bottom_kl->is_typeArray_klass() ? type2name_tab[ArrayKlass::cast(bottom_kl)->element_type()] : bottom_kl->external_name(),
2346 index);
2347 for (int dims = ArrayKlass::cast(a->klass())->dimension(); dims > 1; --dims) {
2348 ss.print("[]");
2349 }
2350 THROW_MSG(vmSymbols::java_lang_ArrayStoreException(), ss.as_string());
2351 }
2352 } else {
2353 ResourceMark rm(THREAD);
2354 stringStream ss;
2355 ss.print("Index %d out of bounds for length %d", index, a->length());
2356 THROW_MSG(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());
2357 }
2358 JNI_END
2359
2360
2361
2362 #define DEFINE_NEWSCALARARRAY(Return,Allocator,Result \
2363 ,EntryProbe,ReturnProbe) \
2364 \
2365 DT_RETURN_MARK_DECL(New##Result##Array, Return \
2366 , ReturnProbe); \
2367 \
2368 JNI_ENTRY(Return, \
2369 jni_New##Result##Array(JNIEnv *env, jsize len)) \
2370 EntryProbe; \
2371 Return ret = nullptr;\
2372 DT_RETURN_MARK(New##Result##Array, Return, (const Return&)ret);\
2373 \
2374 oop obj= oopFactory::Allocator(len, CHECK_NULL); \
2375 ret = (Return) JNIHandles::make_local(THREAD, obj); \
2376 return ret;\
2377 JNI_END
|
1 /*
2 * Copyright (c) 1997, 2025, Oracle and/or its affiliates. All rights reserved.
3 * Copyright (c) 2012, 2024 Red Hat, Inc.
4 * Copyright (c) 2021, Azul Systems, Inc. All rights reserved.
5 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
6 *
7 * This code is free software; you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License version 2 only, as
9 * published by the Free Software Foundation.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22 * or visit www.oracle.com if you need additional information or have any
38 #include "classfile/symbolTable.hpp"
39 #include "classfile/systemDictionary.hpp"
40 #include "classfile/vmClasses.hpp"
41 #include "classfile/vmSymbols.hpp"
42 #include "compiler/compiler_globals.hpp"
43 #include "gc/shared/collectedHeap.hpp"
44 #include "gc/shared/gcLocker.inline.hpp"
45 #include "gc/shared/stringdedup/stringDedup.hpp"
46 #include "interpreter/linkResolver.hpp"
47 #include "jni.h"
48 #include "jvm.h"
49 #include "logging/log.hpp"
50 #include "memory/allocation.hpp"
51 #include "memory/allocation.inline.hpp"
52 #include "memory/oopFactory.hpp"
53 #include "memory/resourceArea.hpp"
54 #include "memory/universe.hpp"
55 #include "nmt/memTracker.hpp"
56 #include "oops/access.inline.hpp"
57 #include "oops/arrayOop.hpp"
58 #include "oops/flatArrayOop.inline.hpp"
59 #include "oops/inlineKlass.inline.hpp"
60 #include "oops/instanceKlass.inline.hpp"
61 #include "oops/instanceOop.hpp"
62 #include "oops/klass.inline.hpp"
63 #include "oops/markWord.hpp"
64 #include "oops/method.hpp"
65 #include "oops/objArrayKlass.hpp"
66 #include "oops/objArrayOop.inline.hpp"
67 #include "oops/oop.inline.hpp"
68 #include "oops/symbol.hpp"
69 #include "oops/typeArrayKlass.hpp"
70 #include "oops/typeArrayOop.inline.hpp"
71 #include "prims/jniCheck.hpp"
72 #include "prims/jniExport.hpp"
73 #include "prims/jniFastGetField.hpp"
74 #include "prims/jvm_misc.hpp"
75 #include "prims/jvmtiExport.hpp"
76 #include "prims/jvmtiThreadState.hpp"
77 #include "runtime/arguments.hpp"
78 #include "runtime/atomic.hpp"
79 #include "runtime/fieldDescriptor.inline.hpp"
411 int modifiers = java_lang_reflect_Field::modifiers(reflected);
412
413 // Make sure class is initialized before handing id's out to fields
414 k1->initialize(CHECK_NULL);
415
416 // First check if this is a static field
417 if (modifiers & JVM_ACC_STATIC) {
418 int offset = InstanceKlass::cast(k1)->field_offset( slot );
419 JNIid* id = InstanceKlass::cast(k1)->jni_id_for(offset);
420 assert(id != nullptr, "corrupt Field object");
421 debug_only(id->set_is_static_field_id();)
422 // A jfieldID for a static field is a JNIid specifying the field holder and the offset within the Klass*
423 ret = jfieldIDWorkaround::to_static_jfieldID(id);
424 return ret;
425 }
426
427 // The slot is the index of the field description in the field-array
428 // The jfieldID is the offset of the field within the object
429 // It may also have hash bits for k, if VerifyJNIFields is turned on.
430 int offset = InstanceKlass::cast(k1)->field_offset( slot );
431 bool is_flat = InstanceKlass::cast(k1)->field_is_flat(slot);
432 assert(InstanceKlass::cast(k1)->contains_field_offset(offset), "stay within object");
433 ret = jfieldIDWorkaround::to_instance_jfieldID(k1, offset, is_flat);
434 return ret;
435 JNI_END
436
437
438 DT_RETURN_MARK_DECL(ToReflectedMethod, jobject
439 , HOTSPOT_JNI_TOREFLECTEDMETHOD_RETURN(_ret_ref));
440
441 JNI_ENTRY(jobject, jni_ToReflectedMethod(JNIEnv *env, jclass cls, jmethodID method_id, jboolean isStatic))
442 HOTSPOT_JNI_TOREFLECTEDMETHOD_ENTRY(env, cls, (uintptr_t) method_id, isStatic);
443
444 jobject ret = nullptr;
445 DT_RETURN_MARK(ToReflectedMethod, jobject, (const jobject&)ret);
446
447 methodHandle m (THREAD, Method::resolve_jmethod_id(method_id));
448 assert(m->is_static() == (isStatic != 0), "jni_ToReflectedMethod access flags doesn't match");
449 oop reflection_method;
450 if (m->is_object_constructor()) {
451 reflection_method = Reflection::new_constructor(m, CHECK_NULL);
452 } else {
453 // Note: Static initializers can theoretically be here, if JNI users manage
454 // to get their jmethodID. Record them as plain methods.
455 reflection_method = Reflection::new_method(m, false, CHECK_NULL);
456 }
457 ret = JNIHandles::make_local(THREAD, reflection_method);
458 return ret;
459 JNI_END
460
461 DT_RETURN_MARK_DECL(GetSuperclass, jclass
462 , HOTSPOT_JNI_GETSUPERCLASS_RETURN(_ret_ref));
463
464 JNI_ENTRY(jclass, jni_GetSuperclass(JNIEnv *env, jclass sub))
465 HOTSPOT_JNI_GETSUPERCLASS_ENTRY(env, sub);
466
467 jclass obj = nullptr;
468 DT_RETURN_MARK(GetSuperclass, jclass, (const jclass&)obj);
469
470 oop mirror = JNIHandles::resolve_non_null(sub);
790 }
791
792 friend class SignatureIterator; // so do_parameters_on can call do_type
793 void do_type(BasicType type) {
794 switch (type) {
795 // these are coerced to int when using va_arg
796 case T_BYTE:
797 case T_CHAR:
798 case T_SHORT:
799 case T_INT: push_int(va_arg(_ap, jint)); break;
800 case T_BOOLEAN: push_boolean((jboolean) va_arg(_ap, jint)); break;
801
802 // each of these paths is exercised by the various jck Call[Static,Nonvirtual,][Void,Int,..]Method[A,V,] tests
803
804 case T_LONG: push_long(va_arg(_ap, jlong)); break;
805 // float is coerced to double w/ va_arg
806 case T_FLOAT: push_float((jfloat) va_arg(_ap, jdouble)); break;
807 case T_DOUBLE: push_double(va_arg(_ap, jdouble)); break;
808
809 case T_ARRAY:
810 case T_OBJECT: push_object(va_arg(_ap, jobject)); break;
811 default: ShouldNotReachHere();
812 }
813 }
814
815 public:
816 JNI_ArgumentPusherVaArg(jmethodID method_id, va_list rap)
817 : JNI_ArgumentPusher(Method::resolve_jmethod_id(method_id)) {
818 set_ap(rap);
819 }
820
821 ~JNI_ArgumentPusherVaArg() {
822 va_end(_ap);
823 }
824
825 virtual void push_arguments_on(JavaCallArguments* arguments) {
826 _arguments = arguments;
827 do_parameters_on(this);
828 }
829 };
830
831
832 class JNI_ArgumentPusherArray : public JNI_ArgumentPusher {
833 protected:
834 const jvalue *_ap;
835
836 inline void set_ap(const jvalue *rap) { _ap = rap; }
837
838 friend class SignatureIterator; // so do_parameters_on can call do_type
839 void do_type(BasicType type) {
840 switch (type) {
841 case T_CHAR: push_int((_ap++)->c); break;
842 case T_SHORT: push_int((_ap++)->s); break;
843 case T_BYTE: push_int((_ap++)->b); break;
844 case T_INT: push_int((_ap++)->i); break;
845 case T_BOOLEAN: push_boolean((_ap++)->z); break;
846 case T_LONG: push_long((_ap++)->j); break;
847 case T_FLOAT: push_float((_ap++)->f); break;
848 case T_DOUBLE: push_double((_ap++)->d); break;
849 case T_ARRAY:
850 case T_OBJECT:
851 case T_FLAT_ELEMENT: push_object((_ap++)->l); break;
852 default: ShouldNotReachHere();
853 }
854 }
855
856 public:
857 JNI_ArgumentPusherArray(jmethodID method_id, const jvalue *rap)
858 : JNI_ArgumentPusher(Method::resolve_jmethod_id(method_id)) {
859 set_ap(rap);
860 }
861
862 virtual void push_arguments_on(JavaCallArguments* arguments) {
863 _arguments = arguments;
864 do_parameters_on(this);
865 }
866 };
867
868
869 enum JNICallType {
870 JNI_STATIC,
871 JNI_VIRTUAL,
958 result->set_type(args->return_type());
959
960 // Invoke the method. Result is returned as oop.
961 JavaCalls::call(result, method, &java_args, CHECK);
962
963 // Convert result
964 if (is_reference_type(result->get_type())) {
965 result->set_jobject(JNIHandles::make_local(THREAD, result->get_oop()));
966 }
967 }
968
969 DT_RETURN_MARK_DECL(AllocObject, jobject
970 , HOTSPOT_JNI_ALLOCOBJECT_RETURN(_ret_ref));
971
972 JNI_ENTRY(jobject, jni_AllocObject(JNIEnv *env, jclass clazz))
973 HOTSPOT_JNI_ALLOCOBJECT_ENTRY(env, clazz);
974
975 jobject ret = nullptr;
976 DT_RETURN_MARK(AllocObject, jobject, (const jobject&)ret);
977
978 oop clazzoop = JNIHandles::resolve_non_null(clazz);
979 Klass* k = java_lang_Class::as_Klass(clazzoop);
980 if (k == nullptr || k->is_inline_klass()) {
981 ResourceMark rm(THREAD);
982 THROW_(vmSymbols::java_lang_InstantiationException(), nullptr);
983 }
984 instanceOop i = InstanceKlass::allocate_instance(clazzoop, CHECK_NULL);
985 ret = JNIHandles::make_local(THREAD, i);
986 return ret;
987 JNI_END
988
989 DT_RETURN_MARK_DECL(NewObjectA, jobject
990 , HOTSPOT_JNI_NEWOBJECTA_RETURN(_ret_ref));
991
992 JNI_ENTRY(jobject, jni_NewObjectA(JNIEnv *env, jclass clazz, jmethodID methodID, const jvalue *args))
993 HOTSPOT_JNI_NEWOBJECTA_ENTRY(env, clazz, (uintptr_t) methodID);
994
995 jobject obj = nullptr;
996 DT_RETURN_MARK(NewObjectA, jobject, (const jobject&)obj);
997
998 oop clazzoop = JNIHandles::resolve_non_null(clazz);
999 Klass* k = java_lang_Class::as_Klass(clazzoop);
1000 if (k == nullptr) {
1001 ResourceMark rm(THREAD);
1002 THROW_(vmSymbols::java_lang_InstantiationException(), nullptr);
1003 }
1004
1005 instanceOop i = InstanceKlass::allocate_instance(clazzoop, CHECK_NULL);
1006 obj = JNIHandles::make_local(THREAD, i);
1007 JavaValue jvalue(T_VOID);
1008 JNI_ArgumentPusherArray ap(methodID, args);
1009 jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);
1010
1011 return obj;
1012 JNI_END
1013
1014
1015 DT_RETURN_MARK_DECL(NewObjectV, jobject
1016 , HOTSPOT_JNI_NEWOBJECTV_RETURN(_ret_ref));
1017
1018 JNI_ENTRY(jobject, jni_NewObjectV(JNIEnv *env, jclass clazz, jmethodID methodID, va_list args))
1019 HOTSPOT_JNI_NEWOBJECTV_ENTRY(env, clazz, (uintptr_t) methodID);
1020
1021 jobject obj = nullptr;
1022 DT_RETURN_MARK(NewObjectV, jobject, (const jobject&)obj);
1023
1024 oop clazzoop = JNIHandles::resolve_non_null(clazz);
1025 Klass* k = java_lang_Class::as_Klass(clazzoop);
1026 if (k == nullptr) {
1027 ResourceMark rm(THREAD);
1028 THROW_(vmSymbols::java_lang_InstantiationException(), nullptr);
1029 }
1030
1031 instanceOop i = InstanceKlass::allocate_instance(clazzoop, CHECK_NULL);
1032 obj = JNIHandles::make_local(THREAD, i);
1033 JavaValue jvalue(T_VOID);
1034 JNI_ArgumentPusherVaArg ap(methodID, args);
1035 jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);
1036
1037 return obj;
1038 JNI_END
1039
1040
1041 DT_RETURN_MARK_DECL(NewObject, jobject
1042 , HOTSPOT_JNI_NEWOBJECT_RETURN(_ret_ref));
1043
1044 JNI_ENTRY(jobject, jni_NewObject(JNIEnv *env, jclass clazz, jmethodID methodID, ...))
1045 HOTSPOT_JNI_NEWOBJECT_ENTRY(env, clazz, (uintptr_t) methodID);
1046
1047 jobject obj = nullptr;
1048 DT_RETURN_MARK(NewObject, jobject, (const jobject&)obj);
1049
1050 oop clazzoop = JNIHandles::resolve_non_null(clazz);
1051 Klass* k = java_lang_Class::as_Klass(clazzoop);
1052 if (k == nullptr) {
1053 ResourceMark rm(THREAD);
1054 THROW_(vmSymbols::java_lang_InstantiationException(), nullptr);
1055 }
1056
1057 instanceOop i = InstanceKlass::allocate_instance(clazzoop, CHECK_NULL);
1058 obj = JNIHandles::make_local(THREAD, i);
1059 va_list args;
1060 va_start(args, methodID);
1061 JavaValue jvalue(T_VOID);
1062 JNI_ArgumentPusherVaArg ap(methodID, args);
1063 jni_invoke_nonstatic(env, &jvalue, obj, JNI_NONVIRTUAL, methodID, &ap, CHECK_NULL);
1064 va_end(args);
1065
1066 return obj;
1067 JNI_END
1068
1069
1070 JNI_ENTRY(jclass, jni_GetObjectClass(JNIEnv *env, jobject obj))
1071 HOTSPOT_JNI_GETOBJECTCLASS_ENTRY(env, obj);
1072
1073 Klass* k = JNIHandles::resolve_non_null(obj)->klass();
1074 jclass ret =
1075 (jclass) JNIHandles::make_local(THREAD, k->java_mirror());
1076
1077 HOTSPOT_JNI_GETOBJECTCLASS_RETURN(ret);
1078 return ret;
1079 JNI_END
1080
1081 JNI_ENTRY_NO_PRESERVE(jboolean, jni_IsInstanceOf(JNIEnv *env, jobject obj, jclass clazz))
1082 HOTSPOT_JNI_ISINSTANCEOF_ENTRY(env, obj, clazz);
1083
1084 jboolean ret = JNI_TRUE;
1085 if (obj != nullptr) {
1796 // table. If they're not there, the field doesn't exist.
1797 TempNewSymbol fieldname = SymbolTable::probe(name, (int)strlen(name));
1798 TempNewSymbol signame = SymbolTable::probe(sig, (int)strlen(sig));
1799 if (fieldname == nullptr || signame == nullptr) {
1800 ResourceMark rm;
1801 THROW_MSG_NULL(vmSymbols::java_lang_NoSuchFieldError(), err_msg("%s.%s %s", k->external_name(), name, sig));
1802 }
1803
1804 // Make sure class is initialized before handing id's out to fields
1805 k->initialize(CHECK_NULL);
1806
1807 fieldDescriptor fd;
1808 if (!k->is_instance_klass() ||
1809 !InstanceKlass::cast(k)->find_field(fieldname, signame, false, &fd)) {
1810 ResourceMark rm;
1811 THROW_MSG_NULL(vmSymbols::java_lang_NoSuchFieldError(), err_msg("%s.%s %s", k->external_name(), name, sig));
1812 }
1813
1814 // A jfieldID for a non-static field is simply the offset of the field within the instanceOop
1815 // It may also have hash bits for k, if VerifyJNIFields is turned on.
1816 ret = jfieldIDWorkaround::to_instance_jfieldID(k, fd.offset(), fd.is_flat());
1817 return ret;
1818 JNI_END
1819
1820
1821 JNI_ENTRY(jobject, jni_GetObjectField(JNIEnv *env, jobject obj, jfieldID fieldID))
1822 HOTSPOT_JNI_GETOBJECTFIELD_ENTRY(env, obj, (uintptr_t) fieldID);
1823 oop o = JNIHandles::resolve_non_null(obj);
1824 Klass* k = o->klass();
1825 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID);
1826 oop res = nullptr;
1827 // Keep JVMTI addition small and only check enabled flag here.
1828 // jni_GetField_probe() assumes that is okay to create handles.
1829 if (JvmtiExport::should_post_field_access()) {
1830 o = JvmtiExport::jni_GetField_probe(thread, obj, o, k, fieldID, false);
1831 }
1832 if (!jfieldIDWorkaround::is_flat_jfieldID(fieldID)) {
1833 res = HeapAccess<ON_UNKNOWN_OOP_REF>::oop_load_at(o, offset);
1834 } else {
1835 assert(k->is_instance_klass(), "Only instance can have flat fields");
1836 InstanceKlass* ik = InstanceKlass::cast(k);
1837 fieldDescriptor fd;
1838 bool found = ik->find_field_from_offset(offset, false, &fd); // performance bottleneck
1839 assert(found, "Field not found");
1840 InstanceKlass* holder = fd.field_holder();
1841 assert(holder->field_is_flat(fd.index()), "Must be");
1842 InlineLayoutInfo* li = holder->inline_layout_info_adr(fd.index());
1843 InlineKlass* field_vklass = li->klass();
1844 res = field_vklass->read_payload_from_addr(o, ik->field_offset(fd.index()), li->kind(), CHECK_NULL);
1845 }
1846 jobject ret = JNIHandles::make_local(THREAD, res);
1847 HOTSPOT_JNI_GETOBJECTFIELD_RETURN(ret);
1848 return ret;
1849 JNI_END
1850
1851 #define DEFINE_GETFIELD(Return,Fieldname,Result \
1852 , EntryProbe, ReturnProbe) \
1853 \
1854 DT_RETURN_MARK_DECL_FOR(Result, Get##Result##Field, Return \
1855 , ReturnProbe); \
1856 \
1857 JNI_ENTRY_NO_PRESERVE(Return, jni_Get##Result##Field(JNIEnv *env, jobject obj, jfieldID fieldID)) \
1858 \
1859 EntryProbe; \
1860 Return ret = 0;\
1861 DT_RETURN_MARK_FOR(Result, Get##Result##Field, Return, (const Return&)ret);\
1862 \
1863 oop o = JNIHandles::resolve_non_null(obj); \
1864 Klass* k = o->klass(); \
1865 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID); \
1866 /* Keep JVMTI addition small and only check enabled flag here. */ \
1867 if (JvmtiExport::should_post_field_access()) { \
1868 o = JvmtiExport::jni_GetField_probe(thread, obj, o, k, fieldID, false); \
1869 } \
1870 ret = o->Fieldname##_field(offset); \
1916 return (address)jni_GetLongField;
1917 }
1918 address jni_GetFloatField_addr() {
1919 return (address)jni_GetFloatField;
1920 }
1921 address jni_GetDoubleField_addr() {
1922 return (address)jni_GetDoubleField;
1923 }
1924
1925 JNI_ENTRY_NO_PRESERVE(void, jni_SetObjectField(JNIEnv *env, jobject obj, jfieldID fieldID, jobject value))
1926 HOTSPOT_JNI_SETOBJECTFIELD_ENTRY(env, obj, (uintptr_t) fieldID, value);
1927 oop o = JNIHandles::resolve_non_null(obj);
1928 Klass* k = o->klass();
1929 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID);
1930 // Keep JVMTI addition small and only check enabled flag here.
1931 if (JvmtiExport::should_post_field_modification()) {
1932 jvalue field_value;
1933 field_value.l = value;
1934 o = JvmtiExport::jni_SetField_probe(thread, obj, o, k, fieldID, false, JVM_SIGNATURE_CLASS, (jvalue *)&field_value);
1935 }
1936 if (!jfieldIDWorkaround::is_flat_jfieldID(fieldID)) {
1937 HeapAccess<ON_UNKNOWN_OOP_REF>::oop_store_at(o, offset, JNIHandles::resolve(value));
1938 } else {
1939 assert(k->is_instance_klass(), "Only instances can have flat fields");
1940 InstanceKlass* ik = InstanceKlass::cast(k);
1941 fieldDescriptor fd;
1942 ik->find_field_from_offset(offset, false, &fd);
1943 InstanceKlass* holder = fd.field_holder();
1944 InlineLayoutInfo* li = holder->inline_layout_info_adr(fd.index());
1945 InlineKlass* vklass = li->klass();
1946 oop v = JNIHandles::resolve_non_null(value);
1947 vklass->write_value_to_addr(v, ((char*)(oopDesc*)obj) + offset, li->kind(), true, CHECK);
1948 }
1949 HOTSPOT_JNI_SETOBJECTFIELD_RETURN();
1950 JNI_END
1951
1952 // TODO: make this a template
1953
1954 #define DEFINE_SETFIELD(Argument,Fieldname,Result,SigType,unionType \
1955 , EntryProbe, ReturnProbe) \
1956 \
1957 JNI_ENTRY_NO_PRESERVE(void, jni_Set##Result##Field(JNIEnv *env, jobject obj, jfieldID fieldID, Argument value)) \
1958 \
1959 EntryProbe; \
1960 \
1961 oop o = JNIHandles::resolve_non_null(obj); \
1962 Klass* k = o->klass(); \
1963 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, fieldID); \
1964 /* Keep JVMTI addition small and only check enabled flag here. */ \
1965 if (JvmtiExport::should_post_field_modification()) { \
1966 jvalue field_value; \
1967 field_value.unionType = value; \
1968 o = JvmtiExport::jni_SetField_probe(thread, obj, o, k, fieldID, false, SigType, (jvalue *)&field_value); \
2353 Klass* ak = ek->array_klass(CHECK_NULL);
2354 ObjArrayKlass::cast(ak)->initialize(CHECK_NULL);
2355 objArrayOop result = ObjArrayKlass::cast(ak)->allocate(length, CHECK_NULL);
2356 oop initial_value = JNIHandles::resolve(initialElement);
2357 if (initial_value != nullptr) { // array already initialized with null
2358 for (int index = 0; index < length; index++) {
2359 result->obj_at_put(index, initial_value);
2360 }
2361 }
2362 ret = (jobjectArray) JNIHandles::make_local(THREAD, result);
2363 return ret;
2364 JNI_END
2365
2366 DT_RETURN_MARK_DECL(GetObjectArrayElement, jobject
2367 , HOTSPOT_JNI_GETOBJECTARRAYELEMENT_RETURN(_ret_ref));
2368
2369 JNI_ENTRY(jobject, jni_GetObjectArrayElement(JNIEnv *env, jobjectArray array, jsize index))
2370 HOTSPOT_JNI_GETOBJECTARRAYELEMENT_ENTRY(env, array, index);
2371 jobject ret = nullptr;
2372 DT_RETURN_MARK(GetObjectArrayElement, jobject, (const jobject&)ret);
2373 oop res = nullptr;
2374 arrayOop arr((arrayOop)JNIHandles::resolve_non_null(array));
2375 if (arr->is_within_bounds(index)) {
2376 if (arr->is_flatArray()) {
2377 flatArrayOop a = flatArrayOop(JNIHandles::resolve_non_null(array));
2378 res = a->read_value_from_flat_array(index, CHECK_NULL);
2379 assert(res != nullptr, "Must be set in one of two paths above");
2380 } else {
2381 assert(arr->is_objArray(), "If not a valueArray. must be an objArray");
2382 objArrayOop a = objArrayOop(JNIHandles::resolve_non_null(array));
2383 res = a->obj_at(index);
2384 }
2385 } else {
2386 ResourceMark rm(THREAD);
2387 stringStream ss;
2388 ss.print("Index %d out of bounds for length %d", index,arr->length());
2389 THROW_MSG_NULL(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());
2390 }
2391 ret = JNIHandles::make_local(THREAD, res);
2392 return ret;
2393 JNI_END
2394
2395 DT_VOID_RETURN_MARK_DECL(SetObjectArrayElement
2396 , HOTSPOT_JNI_SETOBJECTARRAYELEMENT_RETURN());
2397
2398 JNI_ENTRY(void, jni_SetObjectArrayElement(JNIEnv *env, jobjectArray array, jsize index, jobject value))
2399 HOTSPOT_JNI_SETOBJECTARRAYELEMENT_ENTRY(env, array, index, value);
2400 DT_VOID_RETURN_MARK(SetObjectArrayElement);
2401
2402 bool oob = false;
2403 int length = -1;
2404 oop res = nullptr;
2405 arrayOop arr((arrayOop)JNIHandles::resolve_non_null(array));
2406 if (arr->is_within_bounds(index)) {
2407 if (arr->is_flatArray()) {
2408 flatArrayOop a = flatArrayOop(JNIHandles::resolve_non_null(array));
2409 oop v = JNIHandles::resolve(value);
2410 FlatArrayKlass* vaklass = FlatArrayKlass::cast(a->klass());
2411 InlineKlass* element_vklass = vaklass->element_klass();
2412 if (v != nullptr && v->is_a(element_vklass)) {
2413 a->write_value_to_flat_array(v, index, CHECK);
2414 } else {
2415 ResourceMark rm(THREAD);
2416 stringStream ss;
2417 Klass *kl = FlatArrayKlass::cast(a->klass());
2418 ss.print("type mismatch: can not store %s to %s[%d]",
2419 v->klass()->external_name(),
2420 kl->external_name(),
2421 index);
2422 for (int dims = ArrayKlass::cast(a->klass())->dimension(); dims > 1; --dims) {
2423 ss.print("[]");
2424 }
2425 THROW_MSG(vmSymbols::java_lang_ArrayStoreException(), ss.as_string());
2426 }
2427 } else {
2428 assert(arr->is_objArray(), "If not a valueArray. must be an objArray");
2429 objArrayOop a = objArrayOop(JNIHandles::resolve_non_null(array));
2430 oop v = JNIHandles::resolve(value);
2431 if (v == nullptr || v->is_a(ObjArrayKlass::cast(a->klass())->element_klass())) {
2432 a->obj_at_put(index, v);
2433 } else {
2434 ResourceMark rm(THREAD);
2435 stringStream ss;
2436 Klass *bottom_kl = ObjArrayKlass::cast(a->klass())->bottom_klass();
2437 ss.print("type mismatch: can not store %s to %s[%d]",
2438 v->klass()->external_name(),
2439 bottom_kl->is_typeArray_klass() ? type2name_tab[ArrayKlass::cast(bottom_kl)->element_type()] : bottom_kl->external_name(),
2440 index);
2441 for (int dims = ArrayKlass::cast(a->klass())->dimension(); dims > 1; --dims) {
2442 ss.print("[]");
2443 }
2444 THROW_MSG(vmSymbols::java_lang_ArrayStoreException(), ss.as_string());
2445 }
2446 }
2447 } else {
2448 ResourceMark rm(THREAD);
2449 stringStream ss;
2450 ss.print("Index %d out of bounds for length %d", index, arr->length());
2451 THROW_MSG(vmSymbols::java_lang_ArrayIndexOutOfBoundsException(), ss.as_string());
2452 }
2453 JNI_END
2454
2455
2456
2457 #define DEFINE_NEWSCALARARRAY(Return,Allocator,Result \
2458 ,EntryProbe,ReturnProbe) \
2459 \
2460 DT_RETURN_MARK_DECL(New##Result##Array, Return \
2461 , ReturnProbe); \
2462 \
2463 JNI_ENTRY(Return, \
2464 jni_New##Result##Array(JNIEnv *env, jsize len)) \
2465 EntryProbe; \
2466 Return ret = nullptr;\
2467 DT_RETURN_MARK(New##Result##Array, Return, (const Return&)ret);\
2468 \
2469 oop obj= oopFactory::Allocator(len, CHECK_NULL); \
2470 ret = (Return) JNIHandles::make_local(THREAD, obj); \
2471 return ret;\
2472 JNI_END
|