提交 f47626c1 编写于 作者: M mdoerr

8165489: Missing G1 barrier in Unsafe_GetObjectVolatile

Summary: Add missing barrier, sharing code with Unsafe_GetObject.
Reviewed-by: kbarrett, mgerdin, pliden, tschatzl
上级 cd6d97dc
...@@ -199,37 +199,40 @@ jint Unsafe_invocation_key_to_method_slot(jint key) { ...@@ -199,37 +199,40 @@ jint Unsafe_invocation_key_to_method_slot(jint key) {
// Get/SetObject must be special-cased, since it works with handles. // Get/SetObject must be special-cased, since it works with handles.
// The xxx140 variants for backward compatibility do not allow a full-width offset. // We could be accessing the referent field in a reference
UNSAFE_ENTRY(jobject, Unsafe_GetObject140(JNIEnv *env, jobject unsafe, jobject obj, jint offset)) // object. If G1 is enabled then we need to register non-null
UnsafeWrapper("Unsafe_GetObject"); // referent with the SATB barrier.
if (obj == NULL) THROW_0(vmSymbols::java_lang_NullPointerException());
GET_OOP_FIELD(obj, offset, v)
jobject ret = JNIHandles::make_local(env, v);
#if INCLUDE_ALL_GCS #if INCLUDE_ALL_GCS
// We could be accessing the referent field in a reference static bool is_java_lang_ref_Reference_access(oop o, jlong offset) {
// object. If G1 is enabled then we need to register a non-null if (offset == java_lang_ref_Reference::referent_offset && o != NULL) {
// referent with the SATB barrier.
if (UseG1GC) {
bool needs_barrier = false;
if (ret != NULL) {
if (offset == java_lang_ref_Reference::referent_offset) {
oop o = JNIHandles::resolve_non_null(obj);
Klass* k = o->klass(); Klass* k = o->klass();
if (InstanceKlass::cast(k)->reference_type() != REF_NONE) { if (InstanceKlass::cast(k)->reference_type() != REF_NONE) {
assert(InstanceKlass::cast(k)->is_subclass_of(SystemDictionary::Reference_klass()), "sanity"); assert(InstanceKlass::cast(k)->is_subclass_of(SystemDictionary::Reference_klass()), "sanity");
needs_barrier = true; return true;
}
} }
} }
return false;
}
#endif
if (needs_barrier) { static void ensure_satb_referent_alive(oop o, jlong offset, oop v) {
oop referent = JNIHandles::resolve(ret); #if INCLUDE_ALL_GCS
G1SATBCardTableModRefBS::enqueue(referent); if (UseG1GC && v != NULL && is_java_lang_ref_Reference_access(o, offset)) {
} G1SATBCardTableModRefBS::enqueue(v);
} }
#endif // INCLUDE_ALL_GCS #endif
return ret; }
// The xxx140 variants for backward compatibility do not allow a full-width offset.
UNSAFE_ENTRY(jobject, Unsafe_GetObject140(JNIEnv *env, jobject unsafe, jobject obj, jint offset))
UnsafeWrapper("Unsafe_GetObject");
if (obj == NULL) THROW_0(vmSymbols::java_lang_NullPointerException());
GET_OOP_FIELD(obj, offset, v)
ensure_satb_referent_alive(p, offset, v);
return JNIHandles::make_local(env, v);
UNSAFE_END UNSAFE_END
UNSAFE_ENTRY(void, Unsafe_SetObject140(JNIEnv *env, jobject unsafe, jobject obj, jint offset, jobject x_h)) UNSAFE_ENTRY(void, Unsafe_SetObject140(JNIEnv *env, jobject unsafe, jobject obj, jint offset, jobject x_h))
...@@ -262,32 +265,10 @@ UNSAFE_END ...@@ -262,32 +265,10 @@ UNSAFE_END
UNSAFE_ENTRY(jobject, Unsafe_GetObject(JNIEnv *env, jobject unsafe, jobject obj, jlong offset)) UNSAFE_ENTRY(jobject, Unsafe_GetObject(JNIEnv *env, jobject unsafe, jobject obj, jlong offset))
UnsafeWrapper("Unsafe_GetObject"); UnsafeWrapper("Unsafe_GetObject");
GET_OOP_FIELD(obj, offset, v) GET_OOP_FIELD(obj, offset, v)
jobject ret = JNIHandles::make_local(env, v);
#if INCLUDE_ALL_GCS
// We could be accessing the referent field in a reference
// object. If G1 is enabled then we need to register non-null
// referent with the SATB barrier.
if (UseG1GC) {
bool needs_barrier = false;
if (ret != NULL) {
if (offset == java_lang_ref_Reference::referent_offset && obj != NULL) {
oop o = JNIHandles::resolve(obj);
Klass* k = o->klass();
if (InstanceKlass::cast(k)->reference_type() != REF_NONE) {
assert(InstanceKlass::cast(k)->is_subclass_of(SystemDictionary::Reference_klass()), "sanity");
needs_barrier = true;
}
}
}
if (needs_barrier) { ensure_satb_referent_alive(p, offset, v);
oop referent = JNIHandles::resolve(ret);
G1SATBCardTableModRefBS::enqueue(referent); return JNIHandles::make_local(env, v);
}
}
#endif // INCLUDE_ALL_GCS
return ret;
UNSAFE_END UNSAFE_END
UNSAFE_ENTRY(void, Unsafe_SetObject(JNIEnv *env, jobject unsafe, jobject obj, jlong offset, jobject x_h)) UNSAFE_ENTRY(void, Unsafe_SetObject(JNIEnv *env, jobject unsafe, jobject obj, jlong offset, jobject x_h))
...@@ -312,6 +293,9 @@ UNSAFE_ENTRY(jobject, Unsafe_GetObjectVolatile(JNIEnv *env, jobject unsafe, jobj ...@@ -312,6 +293,9 @@ UNSAFE_ENTRY(jobject, Unsafe_GetObjectVolatile(JNIEnv *env, jobject unsafe, jobj
} else { } else {
(void)const_cast<oop&>(v = *(volatile oop*) addr); (void)const_cast<oop&>(v = *(volatile oop*) addr);
} }
ensure_satb_referent_alive(p, offset, v);
OrderAccess::acquire(); OrderAccess::acquire();
return JNIHandles::make_local(env, v); return JNIHandles::make_local(env, v);
UNSAFE_END UNSAFE_END
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册