/* * Copyright (c) 2018, Red Hat, Inc. All rights reserved. * * 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_GC_SHENANDOAH_SHENANDOAHTHREADLOCALDATA_HPP #define SHARE_GC_SHENANDOAH_SHENANDOAHTHREADLOCALDATA_HPP #include "gc/shared/plab.hpp" #include "gc/shenandoah/shenandoahBarrierSet.hpp" #include "gc/shenandoah/shenandoahSATBMarkQueueSet.hpp" #include "runtime/thread.hpp" #include "utilities/debug.hpp" #include "utilities/sizes.hpp" class ShenandoahThreadLocalData { public: static const uint INVALID_WORKER_ID = uint(-1); private: char _gc_state; char _oom_during_evac; ShenandoahSATBMarkQueue _satb_mark_queue; PLAB* _gclab; size_t _gclab_size; uint _worker_id; bool _force_satb_flush; ShenandoahThreadLocalData() : _gc_state(0), _oom_during_evac(0), _satb_mark_queue(&ShenandoahBarrierSet::satb_mark_queue_set()), _gclab(NULL), _gclab_size(0), _worker_id(INVALID_WORKER_ID), _force_satb_flush(false) { } ~ShenandoahThreadLocalData() { if (_gclab != NULL) { delete _gclab; } } static ShenandoahThreadLocalData* data(Thread* thread) { assert(UseShenandoahGC, "Sanity"); return thread->gc_data(); } static ByteSize satb_mark_queue_offset() { return Thread::gc_data_offset() + byte_offset_of(ShenandoahThreadLocalData, _satb_mark_queue); } public: static void create(Thread* thread) { new (data(thread)) ShenandoahThreadLocalData(); } static void destroy(Thread* thread) { data(thread)->~ShenandoahThreadLocalData(); } static SATBMarkQueue& satb_mark_queue(Thread* thread) { return data(thread)->_satb_mark_queue; } static bool is_oom_during_evac(Thread* thread) { return (data(thread)->_oom_during_evac & 1) == 1; } static void set_oom_during_evac(Thread* thread, bool oom) { if (oom) { data(thread)->_oom_during_evac |= 1; } else { data(thread)->_oom_during_evac &= ~1; } } static void set_gc_state(Thread* thread, char gc_state) { data(thread)->_gc_state = gc_state; } static char gc_state(Thread* thread) { return data(thread)->_gc_state; } static void set_worker_id(Thread* thread, uint id) { assert(thread->is_Worker_thread(), "Must be a worker thread"); data(thread)->_worker_id = id; } static uint worker_id(Thread* thread) { assert(thread->is_Worker_thread(), "Must be a worker thread"); return data(thread)->_worker_id; } static void set_force_satb_flush(Thread* thread, bool v) { data(thread)->_force_satb_flush = v; } static bool is_force_satb_flush(Thread* thread) { return data(thread)->_force_satb_flush; } static void initialize_gclab(Thread* thread) { assert (thread->is_Java_thread() || thread->is_Worker_thread(), "Only Java and GC worker threads are allowed to get GCLABs"); data(thread)->_gclab = new PLAB(PLAB::min_size()); data(thread)->_gclab_size = 0; } static PLAB* gclab(Thread* thread) { return data(thread)->_gclab; } static size_t gclab_size(Thread* thread) { return data(thread)->_gclab_size; } static void set_gclab_size(Thread* thread, size_t v) { data(thread)->_gclab_size = v; } #ifdef ASSERT static void set_evac_allowed(Thread* thread, bool evac_allowed) { if (evac_allowed) { data(thread)->_oom_during_evac |= 2; } else { data(thread)->_oom_during_evac &= ~2; } } static bool is_evac_allowed(Thread* thread) { return (data(thread)->_oom_during_evac & 2) == 2; } #endif // Offsets static ByteSize satb_mark_queue_active_offset() { return satb_mark_queue_offset() + SATBMarkQueue::byte_offset_of_active(); } static ByteSize satb_mark_queue_index_offset() { return satb_mark_queue_offset() + SATBMarkQueue::byte_offset_of_index(); } static ByteSize satb_mark_queue_buffer_offset() { return satb_mark_queue_offset() + SATBMarkQueue::byte_offset_of_buf(); } static ByteSize gc_state_offset() { return Thread::gc_data_offset() + byte_offset_of(ShenandoahThreadLocalData, _gc_state); } }; #endif // SHARE_GC_SHENANDOAH_SHENANDOAHTHREADLOCALDATA_HPP