/* * Copyright (c) 2013, 2015, Red Hat, Inc. and/or its affiliates. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. * */ #ifndef SHARE_VM_GC_SHENANDOAH_SHENANDOAHBARRIERSET_HPP #define SHARE_VM_GC_SHENANDOAH_SHENANDOAHBARRIERSET_HPP #include "gc/shared/accessBarrierSupport.hpp" #include "gc/shenandoah/shenandoahBaseBarrierSet.hpp" class ShenandoahBarrierSet: public ShenandoahBaseBarrierSet { public: ShenandoahBarrierSet(ShenandoahHeap* heap); void print_on(outputStream* st) const; bool is_a(BarrierSet::Name bsn); // We export this to make it available in cases where the static // type of the barrier set is known. Note that it is non-virtual. template inline void inline_write_ref_field_pre(T* field, oop new_val); // These are the more general virtual versions. void write_ref_field_pre_work(oop* field, oop new_val); void write_ref_field_pre_work(narrowOop* field, oop new_val); void write_ref_field_pre_work(void* field, oop new_val); void write_ref_field_work(void* v, oop o, bool release = false); public: // Callbacks for runtime accesses. template class AccessBarrier: public BarrierSet::AccessBarrier { typedef BarrierSet::AccessBarrier Raw; public: // Primitive heap accesses. These accessors get resolved when // IN_HEAP is set (e.g. when using the HeapAccess API), it is // not an oop_* overload, and the barrier strength is AS_NORMAL. template static T load_in_heap(T* addr) { ShouldNotReachHere(); return Raw::template load(addr); } template static T load_in_heap_at(oop base, ptrdiff_t offset) { base = ShenandoahBaseBarrierSet::resolve_forwarded(base); return Raw::template load_at(base, offset); } template static void store_in_heap(T* addr, T value) { ShouldNotReachHere(); Raw::store(addr, value); } template static void store_in_heap_at(oop base, ptrdiff_t offset, T value) { base = ShenandoahBaseBarrierSet::barrier_set()->write_barrier(base); Raw::store_at(base, offset, value); } template static T atomic_cmpxchg_in_heap(T new_value, T* addr, T compare_value) { ShouldNotReachHere(); return Raw::atomic_cmpxchg(new_value, addr, compare_value); } template static T atomic_cmpxchg_in_heap_at(T new_value, oop base, ptrdiff_t offset, T compare_value) { base = ShenandoahBaseBarrierSet::barrier_set()->write_barrier(base); return Raw::atomic_cmpxchg_at(new_value, base, offset, compare_value); } template static T atomic_xchg_in_heap(T new_value, T* addr) { ShouldNotReachHere(); return Raw::atomic_xchg(new_value, addr); } template static T atomic_xchg_in_heap_at(T new_value, oop base, ptrdiff_t offset) { base = ShenandoahBaseBarrierSet::barrier_set()->write_barrier(base); return Raw::atomic_xchg_at(new_value, base, offset); } template static void arraycopy_in_heap(arrayOop src_obj, size_t src_offset_in_bytes, T* src_raw, arrayOop dst_obj, size_t dst_offset_in_bytes, T* dst_raw, size_t length); // Heap oop accesses. These accessors get resolved when // IN_HEAP is set (e.g. when using the HeapAccess API), it is // an oop_* overload, and the barrier strength is AS_NORMAL. template static oop oop_load_in_heap(T* addr) { // ShouldNotReachHere(); oop value = Raw::template oop_load(addr); keep_alive_if_weak(decorators, value); return value; } static oop oop_load_in_heap_at(oop base, ptrdiff_t offset) { base = ShenandoahBaseBarrierSet::resolve_forwarded(base); oop value = Raw::template oop_load_at(base, offset); keep_alive_if_weak(AccessBarrierSupport::resolve_possibly_unknown_oop_ref_strength(base, offset), value); return value; } template static void oop_store_in_heap(T* addr, oop value) { ShenandoahBaseBarrierSet::barrier_set()->write_ref_field_pre_work(addr, value); Raw::oop_store(addr, value); } static void oop_store_in_heap_at(oop base, ptrdiff_t offset, oop value) { base = ShenandoahBaseBarrierSet::barrier_set()->write_barrier(base); value = ShenandoahBaseBarrierSet::barrier_set()->storeval_barrier(value); oop_store_in_heap(AccessInternal::oop_field_addr(base, offset), value); } template static oop oop_atomic_cmpxchg_in_heap(oop new_value, T* addr, oop compare_value); static oop oop_atomic_cmpxchg_in_heap_at(oop new_value, oop base, ptrdiff_t offset, oop compare_value) { base = ShenandoahBaseBarrierSet::barrier_set()->write_barrier(base); new_value = ShenandoahBaseBarrierSet::barrier_set()->storeval_barrier(new_value); return oop_atomic_cmpxchg_in_heap(new_value, AccessInternal::oop_field_addr(base, offset), compare_value); } template static oop oop_atomic_xchg_in_heap(oop new_value, T* addr); static oop oop_atomic_xchg_in_heap_at(oop new_value, oop base, ptrdiff_t offset) { base = ShenandoahBaseBarrierSet::barrier_set()->write_barrier(base); new_value = ShenandoahBaseBarrierSet::barrier_set()->storeval_barrier(new_value); return oop_atomic_xchg_in_heap(new_value, AccessInternal::oop_field_addr(base, offset)); } template static bool oop_arraycopy_in_heap(arrayOop src_obj, size_t src_offset_in_bytes, T* src_raw, arrayOop dst_obj, size_t dst_offset_in_bytes, T* dst_raw, size_t length); // Clone barrier support static void clone_in_heap(oop src, oop dst, size_t size); // Needed for loads on non-heap weak references template static oop oop_load_not_in_heap(T* addr) { oop value = Raw::oop_load_not_in_heap(addr); keep_alive_if_weak(decorators, value); return value; } static oop resolve(oop obj) { return ShenandoahBaseBarrierSet::barrier_set()->write_barrier(obj); } static bool equals(oop o1, oop o2) { return ShenandoahBaseBarrierSet::barrier_set()->obj_equals(o1, o2); } }; }; template<> struct BarrierSet::GetName { static const BarrierSet::Name value = BarrierSet::Shenandoah; }; template<> struct BarrierSet::GetType { typedef ::ShenandoahBarrierSet type; }; #endif //SHARE_VM_GC_SHENANDOAH_SHENANDOAHBARRIERSET_HPP