From 74851c507bed9eb2e1e0b0eac3b53a74decb2588 Mon Sep 17 00:00:00 2001 From: Kim Barrett Date: Tue, 15 Sep 2020 14:33:04 +0000 Subject: [PATCH] 8253169: [BACKOUT] Improve large object handling during evacuation Revert "8027545: Improve object array chunking test in G1's copy_to_survivor_space" Revert "8158045: Improve large object handling during evacuation" Revert "8027761: Investigate fast-path for scanning only objects with references during gc" Reviewed-by: tschatzl, sjohanss --- .../share/gc/g1/g1ParScanThreadState.cpp | 123 +++++++----------- .../share/gc/g1/g1ParScanThreadState.hpp | 9 +- .../gc/shared/partialArrayTaskStepper.cpp | 53 -------- .../gc/shared/partialArrayTaskStepper.hpp | 82 ------------ .../shared/partialArrayTaskStepper.inline.hpp | 117 ----------------- src/hotspot/share/oops/arrayOop.hpp | 25 ++-- src/hotspot/share/oops/objArrayOop.hpp | 3 +- .../shared/test_partialArrayTaskStepper.cpp | 87 ------------- 8 files changed, 61 insertions(+), 438 deletions(-) delete mode 100644 src/hotspot/share/gc/shared/partialArrayTaskStepper.cpp delete mode 100644 src/hotspot/share/gc/shared/partialArrayTaskStepper.hpp delete mode 100644 src/hotspot/share/gc/shared/partialArrayTaskStepper.inline.hpp delete mode 100644 test/hotspot/gtest/gc/shared/test_partialArrayTaskStepper.cpp diff --git a/src/hotspot/share/gc/g1/g1ParScanThreadState.cpp b/src/hotspot/share/gc/g1/g1ParScanThreadState.cpp index 15a2600837e..b92f649277f 100644 --- a/src/hotspot/share/gc/g1/g1ParScanThreadState.cpp +++ b/src/hotspot/share/gc/g1/g1ParScanThreadState.cpp @@ -31,19 +31,15 @@ #include "gc/g1/g1RootClosures.hpp" #include "gc/g1/g1StringDedup.hpp" #include "gc/g1/g1Trace.hpp" -#include "gc/shared/partialArrayTaskStepper.inline.hpp" #include "gc/shared/taskqueue.inline.hpp" #include "memory/allocation.inline.hpp" #include "oops/access.inline.hpp" #include "oops/oop.inline.hpp" -#include "runtime/atomic.hpp" #include "runtime/prefetch.inline.hpp" -#include "utilities/globalDefinitions.hpp" G1ParScanThreadState::G1ParScanThreadState(G1CollectedHeap* g1h, G1RedirtyCardsQueueSet* rdcqs, uint worker_id, - uint n_workers, size_t young_cset_length, size_t optional_cset_length) : _g1h(g1h), @@ -64,8 +60,6 @@ G1ParScanThreadState::G1ParScanThreadState(G1CollectedHeap* g1h, _surviving_young_words(NULL), _surviving_words_length(young_cset_length + 1), _old_gen_is_full(false), - _partial_objarray_chunk_size(ParGCArrayScanChunk), - _partial_array_stepper(n_workers), _num_optional_regions(optional_cset_length), _numa(g1h->numa()), _obj_alloc_stat(NULL) @@ -205,59 +199,48 @@ void G1ParScanThreadState::do_partial_array(PartialArrayScanTask task) { assert(_g1h->is_in_reserved(from_obj), "must be in heap."); assert(from_obj->is_objArray(), "must be obj array"); + objArrayOop from_obj_array = objArrayOop(from_obj); + // The from-space object contains the real length. + int length = from_obj_array->length(); + assert(from_obj->is_forwarded(), "must be forwarded"); - - oop to_obj = from_obj->forwardee(); + oop to_obj = from_obj->forwardee(); assert(from_obj != to_obj, "should not be chunking self-forwarded objects"); - assert(to_obj->is_objArray(), "must be obj array"); - objArrayOop to_array = objArrayOop(to_obj); + objArrayOop to_obj_array = objArrayOop(to_obj); + // We keep track of the next start index in the length field of the + // to-space object. + int next_index = to_obj_array->length(); + assert(0 <= next_index && next_index < length, + "invariant, next index: %d, length: %d", next_index, length); - PartialArrayTaskStepper::Step step - = _partial_array_stepper.next(objArrayOop(from_obj), - to_array, - _partial_objarray_chunk_size); - for (uint i = 0; i < step._ncreate; ++i) { + int start = next_index; + int end = length; + int remainder = end - start; + // We'll try not to push a range that's smaller than ParGCArrayScanChunk. + if (remainder > 2 * ParGCArrayScanChunk) { + end = start + ParGCArrayScanChunk; + to_obj_array->set_length(end); + // Push the remainder before we process the range in case another + // worker has run out of things to do and can steal it. push_on_queue(ScannerTask(PartialArrayScanTask(from_obj))); + } else { + assert(length == end, "sanity"); + // We'll process the final range for this object. Restore the length + // so that the heap remains parsable in case of evacuation failure. + to_obj_array->set_length(end); } - HeapRegion* hr = _g1h->heap_region_containing(to_array); + HeapRegion* hr = _g1h->heap_region_containing(to_obj); G1ScanInYoungSetter x(&_scanner, hr->is_young()); - // Process claimed task. The length of to_array is not correct, but - // fortunately the iteration ignores the length field and just relies - // on start/end. - to_array->oop_iterate_range(&_scanner, - step._index, - step._index + _partial_objarray_chunk_size); -} - -void G1ParScanThreadState::start_partial_objarray(G1HeapRegionAttr dest_attr, - oop from_obj, - oop to_obj) { - assert(from_obj->is_objArray(), "precondition"); - assert(from_obj->is_forwarded(), "precondition"); - assert(from_obj->forwardee() == to_obj, "precondition"); - assert(from_obj != to_obj, "should not be scanning self-forwarded objects"); - assert(to_obj->is_objArray(), "precondition"); - - objArrayOop to_array = objArrayOop(to_obj); - - PartialArrayTaskStepper::Step step - = _partial_array_stepper.start(objArrayOop(from_obj), - to_array, - _partial_objarray_chunk_size); - - // Push any needed partial scan tasks. Pushed before processing the - // intitial chunk to allow other workers to steal while we're processing. - for (uint i = 0; i < step._ncreate; ++i) { - push_on_queue(ScannerTask(PartialArrayScanTask(from_obj))); - } - - G1ScanInYoungSetter x(&_scanner, dest_attr.is_young()); - // Process the initial chunk. No need to process the type in the - // klass, as it will already be handled by processing the built-in - // module. The length of to_array is not correct, but fortunately - // the iteration ignores that length field and relies on start/end. - to_array->oop_iterate_range(&_scanner, 0, step._index); + // Process indexes [start,end). It will also process the header + // along with the first chunk (i.e., the chunk with start == 0). + // Note that at this point the length field of to_obj_array is not + // correct given that we are using it to keep track of the next + // start index. oop_iterate_range() (thankfully!) ignores the length + // field and only relies on the start / end parameters. It does + // however return the size of the object which will be incorrect. So + // we have to ignore it even if we wanted to use it. + to_obj_array->oop_iterate_range(&_scanner, start, end); } void G1ParScanThreadState::dispatch_task(ScannerTask task) { @@ -411,10 +394,7 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio assert(region_attr.is_in_cset(), "Unexpected region attr type: %s", region_attr.get_type_str()); - // Get the klass once. We'll need it again later, and this avoids - // re-decoding when it's compressed. - Klass* klass = old->klass(); - const size_t word_sz = old->size_given_klass(klass); + const size_t word_sz = old->size(); uint age = 0; G1HeapRegionAttr dest_attr = next_region_attr(region_attr, old_mark, age); @@ -481,20 +461,6 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio obj->set_mark_raw(old_mark); } - // Most objects are not arrays, so do one array check rather than - // checking for each array category for each object. - if (klass->is_array_klass()) { - if (klass->is_objArray_klass()) { - start_partial_objarray(dest_attr, old, obj); - } else { - // Nothing needs to be done for typeArrays. Body doesn't contain - // any oops to scan, and the type in the klass will already be handled - // by processing the built-in module. - assert(klass->is_typeArray_klass(), "invariant"); - } - return obj; - } - if (G1StringDedup::is_enabled()) { const bool is_from_young = region_attr.is_young(); const bool is_to_young = dest_attr.is_young(); @@ -508,10 +474,17 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio obj); } - G1ScanInYoungSetter x(&_scanner, dest_attr.is_young()); - obj->oop_iterate_backwards(&_scanner); + if (obj->is_objArray() && arrayOop(obj)->length() >= ParGCArrayScanChunk) { + // We keep track of the next start index in the length field of + // the to-space object. The actual length can be found in the + // length field of the from-space object. + arrayOop(obj)->set_length(0); + do_partial_array(PartialArrayScanTask(old)); + } else { + G1ScanInYoungSetter x(&_scanner, dest_attr.is_young()); + obj->oop_iterate_backwards(&_scanner); + } return obj; - } else { _plab_allocator->undo_allocation(dest_attr, obj_ptr, word_sz, node_index); return forward_ptr; @@ -530,9 +503,7 @@ G1ParScanThreadState* G1ParScanThreadStateSet::state_for_worker(uint worker_id) assert(worker_id < _n_workers, "out of bounds access"); if (_states[worker_id] == NULL) { _states[worker_id] = - new G1ParScanThreadState(_g1h, _rdcqs, - worker_id, _n_workers, - _young_cset_length, _optional_cset_length); + new G1ParScanThreadState(_g1h, _rdcqs, worker_id, _young_cset_length, _optional_cset_length); } return _states[worker_id]; } diff --git a/src/hotspot/share/gc/g1/g1ParScanThreadState.hpp b/src/hotspot/share/gc/g1/g1ParScanThreadState.hpp index a4b6761b32d..d57bb7dd88b 100644 --- a/src/hotspot/share/gc/g1/g1ParScanThreadState.hpp +++ b/src/hotspot/share/gc/g1/g1ParScanThreadState.hpp @@ -32,7 +32,6 @@ #include "gc/g1/g1RemSet.hpp" #include "gc/g1/heapRegionRemSet.hpp" #include "gc/shared/ageTable.hpp" -#include "gc/shared/partialArrayTaskStepper.hpp" #include "gc/shared/taskqueue.hpp" #include "memory/allocation.hpp" #include "oops/oop.hpp" @@ -80,9 +79,6 @@ class G1ParScanThreadState : public CHeapObj { // Indicates whether in the last generation (old) there is no more space // available for allocation. bool _old_gen_is_full; - // Size (in elements) of a partial objArray task chunk. - int _partial_objarray_chunk_size; - PartialArrayTaskStepper _partial_array_stepper; G1RedirtyCardsQueue& redirty_cards_queue() { return _rdcq; } G1CardTable* ct() { return _ct; } @@ -109,7 +105,6 @@ public: G1ParScanThreadState(G1CollectedHeap* g1h, G1RedirtyCardsQueueSet* rdcqs, uint worker_id, - uint n_workers, size_t young_cset_length, size_t optional_cset_length); virtual ~G1ParScanThreadState(); @@ -162,7 +157,6 @@ public: private: inline void do_partial_array(PartialArrayScanTask task); - inline void start_partial_objarray(G1HeapRegionAttr dest_dir, oop from, oop to); HeapWord* allocate_copy_slow(G1HeapRegionAttr* dest_attr, oop old, @@ -255,6 +249,9 @@ class G1ParScanThreadStateSet : public StackObj { G1ParScanThreadState* state_for_worker(uint worker_id); const size_t* surviving_young_words() const; + + private: + G1ParScanThreadState* new_par_scan_state(uint worker_id, size_t young_cset_length); }; #endif // SHARE_GC_G1_G1PARSCANTHREADSTATE_HPP diff --git a/src/hotspot/share/gc/shared/partialArrayTaskStepper.cpp b/src/hotspot/share/gc/shared/partialArrayTaskStepper.cpp deleted file mode 100644 index 7b60e1b0e11..00000000000 --- a/src/hotspot/share/gc/shared/partialArrayTaskStepper.cpp +++ /dev/null @@ -1,53 +0,0 @@ -/* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * 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. - * - */ - -#include "precompiled.hpp" -#include "gc/shared/partialArrayTaskStepper.hpp" -#include "oops/arrayOop.hpp" -#include "utilities/globalDefinitions.hpp" - -static uint compute_task_limit(uint n_workers) { - // Don't need more than n_workers tasks at a time. But allowing up to - // that maximizes available parallelism. - return n_workers; -} - -static uint compute_task_fanout(uint task_limit) { - assert(task_limit > 0, "precondition"); - // There is a tradeoff between providing parallelism more quickly and - // number of enqueued tasks. A constant fanout may be too slow when - // parallelism (and so task_limit) is large. A constant fraction might - // be overly eager. Using log2 attempts to balance between those. - uint result = log2_uint(task_limit); - // result must be > 0. result should be > 1 if task_limit > 1, to - // provide some potentially parallel tasks. But don't just +1 to - // avoid otherwise increasing rate of task generation. - if (result < 2) ++result; - return result; -} - -PartialArrayTaskStepper::PartialArrayTaskStepper(uint n_workers) : - _task_limit(compute_task_limit(n_workers)), - _task_fanout(compute_task_fanout(_task_limit)) -{} diff --git a/src/hotspot/share/gc/shared/partialArrayTaskStepper.hpp b/src/hotspot/share/gc/shared/partialArrayTaskStepper.hpp deleted file mode 100644 index aec993f907c..00000000000 --- a/src/hotspot/share/gc/shared/partialArrayTaskStepper.hpp +++ /dev/null @@ -1,82 +0,0 @@ -/* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * 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_SHARED_PARTIALARRAYTASKSTEPPER_HPP -#define SHARE_GC_SHARED_PARTIALARRAYTASKSTEPPER_HPP - -#include "oops/arrayOop.hpp" -#include "utilities/globalDefinitions.hpp" - -// Helper for handling PartialArrayTasks. -// -// When an array is large, we want to split it up into chunks that can be -// processed in parallel. Each task (implicitly) represents such a chunk. -// We can enqueue multiple tasks at the same time. We want to enqueue -// enough tasks to benefit from the available parallelism, while not so many -// as to substantially expand the task queues. -// -// A task directly refers to the from-space array. The from-space array's -// forwarding pointer refers to the associated to-space array, and its -// length is the actual length. The to-space array's length field is used to -// indicate processing progress. It is the starting index of the next chunk -// to process, or equals the actual length when there are no more chunks to -// be processed. -class PartialArrayTaskStepper { -public: - PartialArrayTaskStepper(uint n_workers); - - struct Step { - int _index; // Array index for the step. - uint _ncreate; // Number of new tasks to create. - }; - - // Set to's length to the end of the initial chunk, which is the start of - // the first partial task if the array is large enough to need splitting. - // Returns a Step with _index being that index and _ncreate being the - // initial number of partial tasks to enqueue. - inline Step start(arrayOop from, arrayOop to, int chunk_size) const; - - // Increment to's length by chunk_size to claim the next chunk. Returns a - // Step with _index being the starting index of the claimed chunk and - // _ncreate being the number of additional partial tasks to enqueue. - // precondition: chunk_size must be the same as used to start the task sequence. - inline Step next(arrayOop from, arrayOop to, int chunk_size) const; - - class TestSupport; // For unit tests - -private: - // Limit on the number of partial array tasks to create for a given array. - uint _task_limit; - // Maximum number of new tasks to create when processing an existing task. - uint _task_fanout; - - // Split start/next into public part dealing with oops and private - // impl dealing with lengths and pointers to lengths, for unit testing. - // length is the actual length obtained from the from-space object. - // to_length_addr is the address of the to-space object's length value. - inline Step start_impl(int length, int* to_length_addr, int chunk_size) const; - inline Step next_impl(int length, int* to_length_addr, int chunk_size) const; -}; - -#endif // SHARE_GC_SHARED_PARTIALARRAYTASKSTEPPER_HPP diff --git a/src/hotspot/share/gc/shared/partialArrayTaskStepper.inline.hpp b/src/hotspot/share/gc/shared/partialArrayTaskStepper.inline.hpp deleted file mode 100644 index de6ded3f19f..00000000000 --- a/src/hotspot/share/gc/shared/partialArrayTaskStepper.inline.hpp +++ /dev/null @@ -1,117 +0,0 @@ -/* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * 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_SHARED_PARTIALARRAYTASKSTEPPER_INLINE_HPP -#define SHARE_GC_SHARED_PARTIALARRAYTASKSTEPPER_INLINE_HPP - -#include "gc/shared/partialArrayTaskStepper.hpp" -#include "oops/arrayOop.hpp" -#include "runtime/atomic.hpp" - -PartialArrayTaskStepper::Step -PartialArrayTaskStepper::start_impl(int length, - int* to_length_addr, - int chunk_size) const { - assert(chunk_size > 0, "precondition"); - - int end = length % chunk_size; // End of initial chunk. - // Set to's length to end of initial chunk. Partial tasks use that length - // field as the start of the next chunk to process. Must be done before - // enqueuing partial scan tasks, in case other threads steal any of those - // tasks. - // - // The value of end can be 0, either because of a 0-length array or - // because length is a multiple of the chunk size. Both of those are - // relatively rare and handled in the normal course of the iteration, so - // not worth doing anything special about here. - *to_length_addr = end; - - // If the initial chunk is the complete array, then don't need any partial - // tasks. Otherwise, start with just one partial task; see new task - // calculation in next(). - Step result = { end, (length > end) ? 1u : 0u }; - return result; -} - -PartialArrayTaskStepper::Step -PartialArrayTaskStepper::start(arrayOop from, arrayOop to, int chunk_size) const { - return start_impl(from->length(), to->length_addr(), chunk_size); -} - -PartialArrayTaskStepper::Step -PartialArrayTaskStepper::next_impl(int length, - int* to_length_addr, - int chunk_size) const { - assert(chunk_size > 0, "precondition"); - - // The start of the next task is in the length field of the to-space object. - // Atomically increment by the chunk size to claim the associated chunk. - // Because we limit the number of enqueued tasks to being no more than the - // number of remaining chunks to process, we can use an atomic add for the - // claim, rather than a CAS loop. - int start = Atomic::fetch_and_add(to_length_addr, - chunk_size, - memory_order_relaxed); - - assert(start < length, "invariant: start %d, length %d", start, length); - assert(((length - start) % chunk_size) == 0, - "invariant: start %d, length %d, chunk size %d", - start, length, chunk_size); - - // Determine the number of new tasks to create. - // Zero-based index for this partial task. The initial task isn't counted. - uint task_num = (start / chunk_size); - // Number of tasks left to process, including this one. - uint remaining_tasks = (length - start) / chunk_size; - assert(remaining_tasks > 0, "invariant"); - // Compute number of pending tasks, including this one. The maximum number - // of tasks is a function of task_num (N) and _task_fanout (F). - // 1 : current task - // N : number of preceeding tasks - // F*N : maximum created for preceeding tasks - // => F*N - N + 1 : maximum number of tasks - // => (F-1)*N + 1 - assert(_task_limit > 0, "precondition"); - assert(_task_fanout > 0, "precondition"); - uint max_pending = (_task_fanout - 1) * task_num + 1; - - // The actual pending may be less than that. Bound by remaining_tasks to - // not overrun. Also bound by _task_limit to avoid spawning an excessive - // number of tasks for a large array. The +1 is to replace the current - // task with a new task when _task_limit limited. The pending value may - // not be what's actually in the queues, because of concurrent task - // processing. That's okay; we just need to determine the correct number - // of tasks to add for this task. - uint pending = MIN3(max_pending, remaining_tasks, _task_limit); - uint ncreate = MIN2(_task_fanout, MIN2(remaining_tasks, _task_limit + 1) - pending); - Step result = { start, ncreate }; - return result; -} - -PartialArrayTaskStepper::Step -PartialArrayTaskStepper::next(arrayOop from, arrayOop to, int chunk_size) const { - return next_impl(from->length(), to->length_addr(), chunk_size); -} - -#endif // SHARE_GC_SHARED_PARTIALARRAYTASKSTEPPER_INLINE_HPP diff --git a/src/hotspot/share/oops/arrayOop.hpp b/src/hotspot/share/oops/arrayOop.hpp index 42dc7b0ab51..8d3911a19e8 100644 --- a/src/hotspot/share/oops/arrayOop.hpp +++ b/src/hotspot/share/oops/arrayOop.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2019, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -61,12 +61,6 @@ class arrayOopDesc : public oopDesc { return (int)hs; } - // Returns the address of the length "field". See length_offset_in_bytes(). - static int* length_addr_impl(void* obj_ptr) { - char* ptr = static_cast(obj_ptr); - return reinterpret_cast(ptr + length_offset_in_bytes()); - } - // Check whether an element of a typeArrayOop with the given type must be // aligned 0 mod 8. The typeArrayOop itself must be aligned at least this // strongly. @@ -108,17 +102,16 @@ class arrayOopDesc : public oopDesc { // Tells whether index is within bounds. bool is_within_bounds(int index) const { return 0 <= index && index < length(); } - // Accessors for array length. There's not a member variable for - // it; see length_offset_in_bytes(). - int length() const { return *length_addr_impl(const_cast(this)); } - void set_length(int length) { *length_addr_impl(this) = length; } - - int* length_addr() { - return length_addr_impl(this); + // Accessors for instance variable which is not a C++ declared nonstatic + // field. + int length() const { + return *(int*)(((intptr_t)this) + length_offset_in_bytes()); + } + void set_length(int length) { + set_length((HeapWord*)this, length); } - static void set_length(HeapWord* mem, int length) { - *length_addr_impl(mem) = length; + *(int*)(((char*)mem) + length_offset_in_bytes()) = length; } // Should only be called with constants as argument diff --git a/src/hotspot/share/oops/objArrayOop.hpp b/src/hotspot/share/oops/objArrayOop.hpp index 07f89a6df3c..ab556d48aef 100644 --- a/src/hotspot/share/oops/objArrayOop.hpp +++ b/src/hotspot/share/oops/objArrayOop.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2019, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -38,6 +38,7 @@ class objArrayOopDesc : public arrayOopDesc { friend class Runtime1; friend class psPromotionManager; friend class CSetMarkWordClosure; + friend class G1ParScanPartialArrayClosure; template T* obj_at_addr(int index) const; template T* obj_at_addr_raw(int index) const; diff --git a/test/hotspot/gtest/gc/shared/test_partialArrayTaskStepper.cpp b/test/hotspot/gtest/gc/shared/test_partialArrayTaskStepper.cpp deleted file mode 100644 index fb797ba12c1..00000000000 --- a/test/hotspot/gtest/gc/shared/test_partialArrayTaskStepper.cpp +++ /dev/null @@ -1,87 +0,0 @@ -/* - * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * 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. - * - */ - -#include "precompiled.hpp" -#include "gc/shared/partialArrayTaskStepper.inline.hpp" -#include "memory/allStatic.hpp" -#include "unittest.hpp" - -using Step = PartialArrayTaskStepper::Step; -using Stepper = PartialArrayTaskStepper; - -class PartialArrayTaskStepper::TestSupport : AllStatic { -public: - static Step start(const Stepper* stepper, - int length, - int* to_length_addr, - uint chunk_size) { - return stepper->start_impl(length, to_length_addr, chunk_size); - } - - static Step next(const Stepper* stepper, - int length, - int* to_length_addr, - uint chunk_size) { - return stepper->next_impl(length, to_length_addr, chunk_size); - } -}; - -using StepperSupport = PartialArrayTaskStepper::TestSupport; - -static int simulate(const Stepper* stepper, - int length, - int* to_length_addr, - uint chunk_size) { - Step init = StepperSupport::start(stepper, length, to_length_addr, chunk_size); - uint queue_count = init._ncreate; - int task = 0; - for ( ; queue_count > 0; ++task) { - --queue_count; - Step step = StepperSupport::next(stepper, length, to_length_addr, chunk_size); - queue_count += step._ncreate; - } - return task; -} - -static void run_test(int length, int chunk_size, uint n_workers) { - const PartialArrayTaskStepper stepper(n_workers); - int to_length; - int tasks = simulate(&stepper, length, &to_length, chunk_size); - ASSERT_EQ(length, to_length); - ASSERT_EQ(tasks, length / chunk_size); -} - -TEST(PartialArrayTaskStepperTest, doit) { - for (int chunk_size = 50; chunk_size <= 500; chunk_size += 50) { - for (uint n_workers = 1; n_workers <= 256; n_workers = (n_workers * 3 / 2 + 1)) { - for (int length = 0; length <= 1000000; length = (length * 2 + 1)) { - run_test(length, chunk_size, n_workers); - } - // Ensure we hit boundary cases for length % chunk_size == 0. - for (uint i = 0; i < 2 * n_workers; ++i) { - run_test(i * chunk_size, chunk_size, n_workers); - } - } - } -}