Skip to content

Commit 1856ff8

Browse files
author
Kim Barrett
committedMay 14, 2020
8244684: G1 abuses StarTask to also include partial objarray scan tasks
New ScannerTask and PartialArrayScanTask, initially used by G1 Reviewed-by: tschatzl, sjohanss
1 parent 5b6f81d commit 1856ff8

8 files changed

+179
-135
lines changed
 

‎src/hotspot/share/gc/g1/g1CollectedHeap.cpp

+14-11
Original file line numberDiff line numberDiff line change
@@ -1542,12 +1542,12 @@ G1CollectedHeap::G1CollectedHeap() :
15421542
_filler_array_max_size = _humongous_object_threshold_in_words;
15431543

15441544
uint n_queues = ParallelGCThreads;
1545-
_task_queues = new RefToScanQueueSet(n_queues);
1545+
_task_queues = new ScannerTasksQueueSet(n_queues);
15461546

15471547
_evacuation_failed_info_array = NEW_C_HEAP_ARRAY(EvacuationFailedInfo, n_queues, mtGC);
15481548

15491549
for (uint i = 0; i < n_queues; i++) {
1550-
RefToScanQueue* q = new RefToScanQueue();
1550+
ScannerTasksQueue* q = new ScannerTasksQueue();
15511551
q->initialize();
15521552
_task_queues->register_queue(i, q);
15531553
::new (&_evacuation_failed_info_array[i]) EvacuationFailedInfo();
@@ -3399,7 +3399,7 @@ class G1CopyingKeepAliveClosure: public OopClosure {
33993399
// When the queue is drained (after each phase of reference processing)
34003400
// the object and it's followers will be copied, the reference field set
34013401
// to point to the new location, and the RSet updated.
3402-
_par_scan_state->push_on_queue(p);
3402+
_par_scan_state->push_on_queue(ScannerTask(p));
34033403
}
34043404
}
34053405
};
@@ -3436,14 +3436,14 @@ class G1STWRefProcTaskExecutor: public AbstractRefProcTaskExecutor {
34363436
private:
34373437
G1CollectedHeap* _g1h;
34383438
G1ParScanThreadStateSet* _pss;
3439-
RefToScanQueueSet* _queues;
3439+
ScannerTasksQueueSet* _queues;
34403440
WorkGang* _workers;
34413441

34423442
public:
34433443
G1STWRefProcTaskExecutor(G1CollectedHeap* g1h,
34443444
G1ParScanThreadStateSet* per_thread_states,
34453445
WorkGang* workers,
3446-
RefToScanQueueSet *task_queues) :
3446+
ScannerTasksQueueSet *task_queues) :
34473447
_g1h(g1h),
34483448
_pss(per_thread_states),
34493449
_queues(task_queues),
@@ -3463,14 +3463,14 @@ class G1STWRefProcTaskProxy: public AbstractGangTask {
34633463
ProcessTask& _proc_task;
34643464
G1CollectedHeap* _g1h;
34653465
G1ParScanThreadStateSet* _pss;
3466-
RefToScanQueueSet* _task_queues;
3466+
ScannerTasksQueueSet* _task_queues;
34673467
TaskTerminator* _terminator;
34683468

34693469
public:
34703470
G1STWRefProcTaskProxy(ProcessTask& proc_task,
34713471
G1CollectedHeap* g1h,
34723472
G1ParScanThreadStateSet* per_thread_states,
3473-
RefToScanQueueSet *task_queues,
3473+
ScannerTasksQueueSet *task_queues,
34743474
TaskTerminator* terminator) :
34753475
AbstractGangTask("Process reference objects in parallel"),
34763476
_proc_task(proc_task),
@@ -3801,7 +3801,7 @@ class G1EvacuateRegionsBaseTask : public AbstractGangTask {
38013801
protected:
38023802
G1CollectedHeap* _g1h;
38033803
G1ParScanThreadStateSet* _per_thread_states;
3804-
RefToScanQueueSet* _task_queues;
3804+
ScannerTasksQueueSet* _task_queues;
38053805
TaskTerminator _terminator;
38063806
uint _num_workers;
38073807

@@ -3839,7 +3839,10 @@ class G1EvacuateRegionsBaseTask : public AbstractGangTask {
38393839
virtual void evacuate_live_objects(G1ParScanThreadState* pss, uint worker_id) = 0;
38403840

38413841
public:
3842-
G1EvacuateRegionsBaseTask(const char* name, G1ParScanThreadStateSet* per_thread_states, RefToScanQueueSet* task_queues, uint num_workers) :
3842+
G1EvacuateRegionsBaseTask(const char* name,
3843+
G1ParScanThreadStateSet* per_thread_states,
3844+
ScannerTasksQueueSet* task_queues,
3845+
uint num_workers) :
38433846
AbstractGangTask(name),
38443847
_g1h(G1CollectedHeap::heap()),
38453848
_per_thread_states(per_thread_states),
@@ -3890,7 +3893,7 @@ class G1EvacuateRegionsTask : public G1EvacuateRegionsBaseTask {
38903893
public:
38913894
G1EvacuateRegionsTask(G1CollectedHeap* g1h,
38923895
G1ParScanThreadStateSet* per_thread_states,
3893-
RefToScanQueueSet* task_queues,
3896+
ScannerTasksQueueSet* task_queues,
38943897
G1RootProcessor* root_processor,
38953898
uint num_workers) :
38963899
G1EvacuateRegionsBaseTask("G1 Evacuate Regions", per_thread_states, task_queues, num_workers),
@@ -3938,7 +3941,7 @@ class G1EvacuateOptionalRegionsTask : public G1EvacuateRegionsBaseTask {
39383941

39393942
public:
39403943
G1EvacuateOptionalRegionsTask(G1ParScanThreadStateSet* per_thread_states,
3941-
RefToScanQueueSet* queues,
3944+
ScannerTasksQueueSet* queues,
39423945
uint num_workers) :
39433946
G1EvacuateRegionsBaseTask("G1 Evacuate Optional Regions", per_thread_states, queues, num_workers) {
39443947
}

‎src/hotspot/share/gc/g1/g1CollectedHeap.hpp

+8-7
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@
5454
#include "gc/shared/plab.hpp"
5555
#include "gc/shared/preservedMarks.hpp"
5656
#include "gc/shared/softRefPolicy.hpp"
57+
#include "gc/shared/taskqueue.hpp"
5758
#include "memory/memRegion.hpp"
5859
#include "utilities/stack.hpp"
5960

@@ -97,8 +98,8 @@ class G1HeapSizingPolicy;
9798
class G1HeapSummary;
9899
class G1EvacSummary;
99100

100-
typedef OverflowTaskQueue<StarTask, mtGC> RefToScanQueue;
101-
typedef GenericTaskQueueSet<RefToScanQueue, mtGC> RefToScanQueueSet;
101+
typedef OverflowTaskQueue<ScannerTask, mtGC> ScannerTasksQueue;
102+
typedef GenericTaskQueueSet<ScannerTasksQueue, mtGC> ScannerTasksQueueSet;
102103

103104
typedef int RegionIdx_t; // needs to hold [ 0..max_regions() )
104105
typedef int CardIdx_t; // needs to hold [ 0..CardsPerRegion )
@@ -814,7 +815,7 @@ class G1CollectedHeap : public CollectedHeap {
814815
G1ConcurrentRefine* _cr;
815816

816817
// The parallel task queues
817-
RefToScanQueueSet *_task_queues;
818+
ScannerTasksQueueSet *_task_queues;
818819

819820
// True iff a evacuation has failed in the current collection.
820821
bool _evacuation_failed;
@@ -951,7 +952,7 @@ class G1CollectedHeap : public CollectedHeap {
951952
G1CMSubjectToDiscoveryClosure _is_subject_to_discovery_cm;
952953
public:
953954

954-
RefToScanQueue *task_queue(uint i) const;
955+
ScannerTasksQueue* task_queue(uint i) const;
955956

956957
uint num_task_queues() const;
957958

@@ -1478,18 +1479,18 @@ class G1ParEvacuateFollowersClosure : public VoidClosure {
14781479
protected:
14791480
G1CollectedHeap* _g1h;
14801481
G1ParScanThreadState* _par_scan_state;
1481-
RefToScanQueueSet* _queues;
1482+
ScannerTasksQueueSet* _queues;
14821483
TaskTerminator* _terminator;
14831484
G1GCPhaseTimes::GCParPhases _phase;
14841485

14851486
G1ParScanThreadState* par_scan_state() { return _par_scan_state; }
1486-
RefToScanQueueSet* queues() { return _queues; }
1487+
ScannerTasksQueueSet* queues() { return _queues; }
14871488
TaskTerminator* terminator() { return _terminator; }
14881489

14891490
public:
14901491
G1ParEvacuateFollowersClosure(G1CollectedHeap* g1h,
14911492
G1ParScanThreadState* par_scan_state,
1492-
RefToScanQueueSet* queues,
1493+
ScannerTasksQueueSet* queues,
14931494
TaskTerminator* terminator,
14941495
G1GCPhaseTimes::GCParPhases phase)
14951496
: _start_term(0.0), _term_time(0.0), _term_attempts(0),

‎src/hotspot/share/gc/g1/g1CollectedHeap.inline.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2001, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2001, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -139,7 +139,7 @@ G1CollectedHeap::dirty_young_block(HeapWord* start, size_t word_size) {
139139
card_table()->g1_mark_as_young(mr);
140140
}
141141

142-
inline RefToScanQueue* G1CollectedHeap::task_queue(uint i) const {
142+
inline ScannerTasksQueue* G1CollectedHeap::task_queue(uint i) const {
143143
return _task_queues->queue(i);
144144
}
145145

‎src/hotspot/share/gc/g1/g1OopClosures.inline.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2001, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2001, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -58,7 +58,7 @@ inline void G1ScanClosureBase::prefetch_and_push(T* p, const oop obj) {
5858
obj->forwardee() == RawAccess<>::oop_load(p)),
5959
"p should still be pointing to obj or to its forwardee");
6060

61-
_par_scan_state->push_on_queue(p);
61+
_par_scan_state->push_on_queue(ScannerTask(p));
6262
}
6363

6464
template <class T>

‎src/hotspot/share/gc/g1/g1ParScanThreadState.cpp

+27-29
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -43,7 +43,7 @@ G1ParScanThreadState::G1ParScanThreadState(G1CollectedHeap* g1h,
4343
size_t young_cset_length,
4444
size_t optional_cset_length)
4545
: _g1h(g1h),
46-
_refs(g1h->task_queue(worker_id)),
46+
_task_queue(g1h->task_queue(worker_id)),
4747
_rdcq(rdcqs),
4848
_ct(g1h->card_table()),
4949
_closures(NULL),
@@ -119,46 +119,45 @@ size_t G1ParScanThreadState::lab_undo_waste_words() const {
119119
}
120120

121121
#ifdef ASSERT
122-
bool G1ParScanThreadState::verify_ref(narrowOop* ref) const {
123-
assert(ref != NULL, "invariant");
122+
void G1ParScanThreadState::verify_task(narrowOop* task) const {
123+
assert(task != NULL, "invariant");
124124
assert(UseCompressedOops, "sanity");
125-
assert(!has_partial_array_mask(ref), "ref=" PTR_FORMAT, p2i(ref));
126-
oop p = RawAccess<>::oop_load(ref);
125+
oop p = RawAccess<>::oop_load(task);
127126
assert(_g1h->is_in_g1_reserved(p),
128-
"ref=" PTR_FORMAT " p=" PTR_FORMAT, p2i(ref), p2i(p));
129-
return true;
127+
"task=" PTR_FORMAT " p=" PTR_FORMAT, p2i(task), p2i(p));
130128
}
131129

132-
bool G1ParScanThreadState::verify_ref(oop* ref) const {
133-
assert(ref != NULL, "invariant");
134-
if (has_partial_array_mask(ref)) {
135-
// Must be in the collection set--it's already been copied.
136-
oop p = clear_partial_array_mask(ref);
137-
assert(_g1h->is_in_cset(p),
138-
"ref=" PTR_FORMAT " p=" PTR_FORMAT, p2i(ref), p2i(p));
139-
} else {
140-
oop p = RawAccess<>::oop_load(ref);
141-
assert(_g1h->is_in_g1_reserved(p),
142-
"ref=" PTR_FORMAT " p=" PTR_FORMAT, p2i(ref), p2i(p));
143-
}
144-
return true;
130+
void G1ParScanThreadState::verify_task(oop* task) const {
131+
assert(task != NULL, "invariant");
132+
oop p = RawAccess<>::oop_load(task);
133+
assert(_g1h->is_in_g1_reserved(p),
134+
"task=" PTR_FORMAT " p=" PTR_FORMAT, p2i(task), p2i(p));
135+
}
136+
137+
void G1ParScanThreadState::verify_task(PartialArrayScanTask task) const {
138+
// Must be in the collection set--it's already been copied.
139+
oop p = task.to_source_array();
140+
assert(_g1h->is_in_cset(p), "p=" PTR_FORMAT, p2i(p));
145141
}
146142

147-
bool G1ParScanThreadState::verify_task(StarTask ref) const {
148-
if (ref.is_narrow()) {
149-
return verify_ref((narrowOop*) ref);
143+
void G1ParScanThreadState::verify_task(ScannerTask task) const {
144+
if (task.is_narrow_oop_ptr()) {
145+
verify_task(task.to_narrow_oop_ptr());
146+
} else if (task.is_oop_ptr()) {
147+
verify_task(task.to_oop_ptr());
148+
} else if (task.is_partial_array_task()) {
149+
verify_task(task.to_partial_array_task());
150150
} else {
151-
return verify_ref((oop*) ref);
151+
ShouldNotReachHere();
152152
}
153153
}
154154
#endif // ASSERT
155155

156156
void G1ParScanThreadState::trim_queue() {
157-
StarTask ref;
158157
do {
159158
// Fully drain the queue.
160159
trim_queue_to_threshold(0);
161-
} while (!_refs->is_empty());
160+
} while (!_task_queue->is_empty());
162161
}
163162

164163
HeapWord* G1ParScanThreadState::allocate_in_next_plab(G1HeapRegionAttr* dest,
@@ -330,8 +329,7 @@ oop G1ParScanThreadState::copy_to_survivor_space(G1HeapRegionAttr const region_a
330329
// the to-space object. The actual length can be found in the
331330
// length field of the from-space object.
332331
arrayOop(obj)->set_length(0);
333-
oop* old_p = set_partial_array_mask(old);
334-
do_oop_partial_array(old_p);
332+
do_partial_array(PartialArrayScanTask(old));
335333
} else {
336334
G1ScanInYoungSetter x(&_scanner, dest_attr.is_young());
337335
obj->oop_iterate_backwards(&_scanner);

‎src/hotspot/share/gc/g1/g1ParScanThreadState.hpp

+13-43
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -33,6 +33,7 @@
3333
#include "gc/g1/g1RemSet.hpp"
3434
#include "gc/g1/heapRegionRemSet.hpp"
3535
#include "gc/shared/ageTable.hpp"
36+
#include "gc/shared/taskqueue.hpp"
3637
#include "memory/allocation.hpp"
3738
#include "oops/oop.hpp"
3839
#include "utilities/ticks.hpp"
@@ -45,7 +46,7 @@ class outputStream;
4546

4647
class G1ParScanThreadState : public CHeapObj<mtGC> {
4748
G1CollectedHeap* _g1h;
48-
RefToScanQueue* _refs;
49+
ScannerTasksQueue* _task_queue;
4950
G1RedirtyCardsQueue _rdcq;
5051
G1CardTable* _ct;
5152
G1EvacuationRootClosures* _closures;
@@ -114,15 +115,15 @@ class G1ParScanThreadState : public CHeapObj<mtGC> {
114115
void set_ref_discoverer(ReferenceDiscoverer* rd) { _scanner.set_ref_discoverer(rd); }
115116

116117
#ifdef ASSERT
117-
bool queue_is_empty() const { return _refs->is_empty(); }
118+
bool queue_is_empty() const { return _task_queue->is_empty(); }
119+
#endif
118120

119-
bool verify_ref(narrowOop* ref) const;
120-
bool verify_ref(oop* ref) const;
121-
bool verify_task(StarTask ref) const;
122-
#endif // ASSERT
121+
void verify_task(narrowOop* task) const NOT_DEBUG_RETURN;
122+
void verify_task(oop* task) const NOT_DEBUG_RETURN;
123+
void verify_task(PartialArrayScanTask task) const NOT_DEBUG_RETURN;
124+
void verify_task(ScannerTask task) const NOT_DEBUG_RETURN;
123125

124-
template <class T> void do_oop_ext(T* ref);
125-
template <class T> void push_on_queue(T* ref);
126+
void push_on_queue(ScannerTask task);
126127

127128
template <class T> void enqueue_card_if_tracked(G1HeapRegionAttr region_attr, T* p, oop o) {
128129
assert(!HeapRegion::is_in_same_region(p, o), "Should have filtered out cross-region references already.");
@@ -158,43 +159,12 @@ class G1ParScanThreadState : public CHeapObj<mtGC> {
158159
size_t flush(size_t* surviving_young_words);
159160

160161
private:
161-
#define G1_PARTIAL_ARRAY_MASK 0x2
162-
163-
inline bool has_partial_array_mask(oop* ref) const {
164-
return ((uintptr_t)ref & G1_PARTIAL_ARRAY_MASK) == G1_PARTIAL_ARRAY_MASK;
165-
}
166-
167-
// We never encode partial array oops as narrowOop*, so return false immediately.
168-
// This allows the compiler to create optimized code when popping references from
169-
// the work queue.
170-
inline bool has_partial_array_mask(narrowOop* ref) const {
171-
assert(((uintptr_t)ref & G1_PARTIAL_ARRAY_MASK) != G1_PARTIAL_ARRAY_MASK, "Partial array oop reference encoded as narrowOop*");
172-
return false;
173-
}
174-
175-
// Only implement set_partial_array_mask() for regular oops, not for narrowOops.
176-
// We always encode partial arrays as regular oop, to allow the
177-
// specialization for has_partial_array_mask() for narrowOops above.
178-
// This means that unintentional use of this method with narrowOops are caught
179-
// by the compiler.
180-
inline oop* set_partial_array_mask(oop obj) const {
181-
assert(((uintptr_t)(void *)obj & G1_PARTIAL_ARRAY_MASK) == 0, "Information loss!");
182-
return (oop*) ((uintptr_t)(void *)obj | G1_PARTIAL_ARRAY_MASK);
183-
}
184-
185-
inline oop clear_partial_array_mask(oop* ref) const {
186-
return cast_to_oop((intptr_t)ref & ~G1_PARTIAL_ARRAY_MASK);
187-
}
188-
189-
inline void do_oop_partial_array(oop* p);
162+
inline void do_partial_array(PartialArrayScanTask task);
190163

191164
// This method is applied to the fields of the objects that have just been copied.
192165
template <class T> inline void do_oop_evac(T* p);
193166

194-
inline void deal_with_reference(oop* ref_to_scan);
195-
inline void deal_with_reference(narrowOop* ref_to_scan);
196-
197-
inline void dispatch_reference(StarTask ref);
167+
inline void dispatch_task(ScannerTask task);
198168

199169
// Tries to allocate word_sz in the PLAB of the next "generation" after trying to
200170
// allocate into dest. Previous_plab_refill_failed indicates whether previous
@@ -232,7 +202,7 @@ class G1ParScanThreadState : public CHeapObj<mtGC> {
232202
Tickspan trim_ticks() const;
233203
void reset_trim_ticks();
234204

235-
inline void steal_and_trim_queue(RefToScanQueueSet *task_queues);
205+
inline void steal_and_trim_queue(ScannerTasksQueueSet *task_queues);
236206

237207
// An attempt to evacuate "obj" has failed; take necessary steps.
238208
oop handle_evacuation_failure_par(oop obj, markWord m);

‎src/hotspot/share/gc/g1/g1ParScanThreadState.inline.hpp

+29-41
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2020, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -71,14 +71,13 @@ template <class T> void G1ParScanThreadState::do_oop_evac(T* p) {
7171
}
7272
}
7373

74-
template <class T> inline void G1ParScanThreadState::push_on_queue(T* ref) {
75-
assert(verify_ref(ref), "sanity");
76-
_refs->push(ref);
74+
inline void G1ParScanThreadState::push_on_queue(ScannerTask task) {
75+
verify_task(task);
76+
_task_queue->push(task);
7777
}
7878

79-
inline void G1ParScanThreadState::do_oop_partial_array(oop* p) {
80-
assert(has_partial_array_mask(p), "invariant");
81-
oop from_obj = clear_partial_array_mask(p);
79+
inline void G1ParScanThreadState::do_partial_array(PartialArrayScanTask task) {
80+
oop from_obj = task.to_source_array();
8281

8382
assert(_g1h->is_in_reserved(from_obj), "must be in heap.");
8483
assert(from_obj->is_objArray(), "must be obj array");
@@ -105,8 +104,7 @@ inline void G1ParScanThreadState::do_oop_partial_array(oop* p) {
105104
to_obj_array->set_length(end);
106105
// Push the remainder before we process the range in case another
107106
// worker has run out of things to do and can steal it.
108-
oop* from_obj_p = set_partial_array_mask(from_obj);
109-
push_on_queue(from_obj_p);
107+
push_on_queue(ScannerTask(PartialArrayScanTask(from_obj)));
110108
} else {
111109
assert(length == end, "sanity");
112110
// We'll process the final range for this object. Restore the length
@@ -127,60 +125,50 @@ inline void G1ParScanThreadState::do_oop_partial_array(oop* p) {
127125
to_obj_array->oop_iterate_range(&_scanner, start, end);
128126
}
129127

130-
inline void G1ParScanThreadState::deal_with_reference(oop* ref_to_scan) {
131-
if (!has_partial_array_mask(ref_to_scan)) {
132-
do_oop_evac(ref_to_scan);
128+
inline void G1ParScanThreadState::dispatch_task(ScannerTask task) {
129+
verify_task(task);
130+
if (task.is_narrow_oop_ptr()) {
131+
do_oop_evac(task.to_narrow_oop_ptr());
132+
} else if (task.is_oop_ptr()) {
133+
do_oop_evac(task.to_oop_ptr());
133134
} else {
134-
do_oop_partial_array(ref_to_scan);
135+
do_partial_array(task.to_partial_array_task());
135136
}
136137
}
137138

138-
inline void G1ParScanThreadState::deal_with_reference(narrowOop* ref_to_scan) {
139-
assert(!has_partial_array_mask(ref_to_scan), "NarrowOop* elements should never be partial arrays.");
140-
do_oop_evac(ref_to_scan);
141-
}
142-
143-
inline void G1ParScanThreadState::dispatch_reference(StarTask ref) {
144-
assert(verify_task(ref), "sanity");
145-
if (ref.is_narrow()) {
146-
deal_with_reference((narrowOop*)ref);
147-
} else {
148-
deal_with_reference((oop*)ref);
149-
}
150-
}
151-
152-
void G1ParScanThreadState::steal_and_trim_queue(RefToScanQueueSet *task_queues) {
153-
StarTask stolen_task;
139+
void G1ParScanThreadState::steal_and_trim_queue(ScannerTasksQueueSet *task_queues) {
140+
ScannerTask stolen_task;
154141
while (task_queues->steal(_worker_id, stolen_task)) {
155-
assert(verify_task(stolen_task), "sanity");
156-
dispatch_reference(stolen_task);
142+
dispatch_task(stolen_task);
157143

158-
// We've just processed a reference and we might have made
144+
// We've just processed a task and we might have made
159145
// available new entries on the queues. So we have to make sure
160146
// we drain the queues as necessary.
161147
trim_queue();
162148
}
163149
}
164150

165151
inline bool G1ParScanThreadState::needs_partial_trimming() const {
166-
return !_refs->overflow_empty() || _refs->size() > _stack_trim_upper_threshold;
152+
return !_task_queue->overflow_empty() ||
153+
(_task_queue->size() > _stack_trim_upper_threshold);
167154
}
168155

169156
inline bool G1ParScanThreadState::is_partially_trimmed() const {
170-
return _refs->overflow_empty() && _refs->size() <= _stack_trim_lower_threshold;
157+
return _task_queue->overflow_empty() &&
158+
(_task_queue->size() <= _stack_trim_lower_threshold);
171159
}
172160

173161
inline void G1ParScanThreadState::trim_queue_to_threshold(uint threshold) {
174-
StarTask ref;
162+
ScannerTask task;
175163
// Drain the overflow stack first, so other threads can potentially steal.
176-
while (_refs->pop_overflow(ref)) {
177-
if (!_refs->try_push_to_taskqueue(ref)) {
178-
dispatch_reference(ref);
164+
while (_task_queue->pop_overflow(task)) {
165+
if (!_task_queue->try_push_to_taskqueue(task)) {
166+
dispatch_task(task);
179167
}
180168
}
181169

182-
while (_refs->pop_local(ref, threshold)) {
183-
dispatch_reference(ref);
170+
while (_task_queue->pop_local(task, threshold)) {
171+
dispatch_task(task);
184172
}
185173
}
186174

@@ -220,7 +208,7 @@ inline void G1ParScanThreadState::remember_reference_into_optional_region(T* p)
220208
assert(index < _num_optional_regions,
221209
"Trying to access optional region idx %u beyond " SIZE_FORMAT, index, _num_optional_regions);
222210
_oops_into_optional_regions[index].push_oop(p);
223-
DEBUG_ONLY(verify_ref(p);)
211+
verify_task(p);
224212
}
225213

226214
G1OopStarChunkedList* G1ParScanThreadState::oops_into_optional_region(const HeapRegion* hr) {

‎src/hotspot/share/gc/shared/taskqueue.hpp

+84
Original file line numberDiff line numberDiff line change
@@ -564,4 +564,88 @@ class ObjArrayTask
564564
int _index;
565565
};
566566

567+
// Wrapper over an oop that is a partially scanned array.
568+
// Can be converted to a ScannerTask for placement in associated task queues.
569+
// Refers to the partially copied source array oop.
570+
class PartialArrayScanTask {
571+
oop _src;
572+
573+
public:
574+
PartialArrayScanTask() : _src() {}
575+
explicit PartialArrayScanTask(oop src_array) : _src(src_array) {}
576+
// Trivially copyable.
577+
578+
oop to_source_array() const { return _src; }
579+
};
580+
581+
// Discriminated union over oop*, narrowOop*, and PartialArrayScanTask.
582+
// Uses a low tag in the associated pointer to identify the category.
583+
// Used as a task queue element type.
584+
class ScannerTask {
585+
void* _p;
586+
587+
static const uintptr_t OopTag = 0;
588+
static const uintptr_t NarrowOopTag = 1;
589+
static const uintptr_t PartialArrayTag = 2;
590+
static const uintptr_t TagSize = 2;
591+
static const uintptr_t TagAlignment = 1 << TagSize;
592+
static const uintptr_t TagMask = TagAlignment - 1;
593+
594+
static void* encode(void* p, uintptr_t tag) {
595+
assert(is_aligned(p, TagAlignment), "misaligned: " PTR_FORMAT, p2i(p));
596+
return static_cast<char*>(p) + tag;
597+
}
598+
599+
uintptr_t raw_value() const {
600+
return reinterpret_cast<uintptr_t>(_p);
601+
}
602+
603+
bool has_tag(uintptr_t tag) const {
604+
return (raw_value() & TagMask) == tag;
605+
}
606+
607+
void* decode(uintptr_t tag) const {
608+
assert(has_tag(tag), "precondition");
609+
return static_cast<char*>(_p) - tag;
610+
}
611+
612+
public:
613+
ScannerTask() : _p(NULL) {}
614+
615+
explicit ScannerTask(oop* p) : _p(encode(p, OopTag)) {}
616+
617+
explicit ScannerTask(narrowOop* p) : _p(encode(p, NarrowOopTag)) {}
618+
619+
explicit ScannerTask(PartialArrayScanTask t) :
620+
_p(encode(t.to_source_array(), PartialArrayTag)) {}
621+
622+
// Trivially copyable.
623+
624+
// Predicate implementations assume OopTag == 0, others are powers of 2.
625+
626+
bool is_oop_ptr() const {
627+
return (raw_value() & (NarrowOopTag | PartialArrayTag)) == 0;
628+
}
629+
630+
bool is_narrow_oop_ptr() const {
631+
return (raw_value() & NarrowOopTag) != 0;
632+
}
633+
634+
bool is_partial_array_task() const {
635+
return (raw_value() & PartialArrayTag) != 0;
636+
}
637+
638+
oop* to_oop_ptr() const {
639+
return static_cast<oop*>(decode(OopTag));
640+
}
641+
642+
narrowOop* to_narrow_oop_ptr() const {
643+
return static_cast<narrowOop*>(decode(NarrowOopTag));
644+
}
645+
646+
PartialArrayScanTask to_partial_array_task() const {
647+
return PartialArrayScanTask(oop(decode(PartialArrayTag)));
648+
}
649+
};
650+
567651
#endif // SHARE_GC_SHARED_TASKQUEUE_HPP

0 commit comments

Comments
 (0)
Please sign in to comment.