31
31
#include " gc/g1/g1RootClosures.hpp"
32
32
#include " gc/g1/g1StringDedup.hpp"
33
33
#include " gc/g1/g1Trace.hpp"
34
- #include " gc/shared/partialArrayTaskStepper.inline.hpp"
35
34
#include " gc/shared/taskqueue.inline.hpp"
36
35
#include " memory/allocation.inline.hpp"
37
36
#include " oops/access.inline.hpp"
38
37
#include " oops/oop.inline.hpp"
39
- #include " runtime/atomic.hpp"
40
38
#include " runtime/prefetch.inline.hpp"
41
- #include " utilities/globalDefinitions.hpp"
42
39
43
40
G1ParScanThreadState::G1ParScanThreadState (G1CollectedHeap* g1h,
44
41
G1RedirtyCardsQueueSet* rdcqs,
45
42
uint worker_id,
46
- uint n_workers,
47
43
size_t young_cset_length,
48
44
size_t optional_cset_length)
49
45
: _g1h(g1h),
@@ -64,8 +60,6 @@ G1ParScanThreadState::G1ParScanThreadState(G1CollectedHeap* g1h,
64
60
_surviving_young_words(NULL ),
65
61
_surviving_words_length(young_cset_length + 1 ),
66
62
_old_gen_is_full(false ),
67
- _partial_objarray_chunk_size(ParGCArrayScanChunk),
68
- _partial_array_stepper(n_workers),
69
63
_num_optional_regions(optional_cset_length),
70
64
_numa(g1h->numa ()),
71
65
_obj_alloc_stat(NULL )
@@ -205,59 +199,48 @@ void G1ParScanThreadState::do_partial_array(PartialArrayScanTask task) {
205
199
206
200
assert (_g1h->is_in_reserved (from_obj), " must be in heap." );
207
201
assert (from_obj->is_objArray (), " must be obj array" );
208
- assert (from_obj->is_forwarded (), " must be forwarded" );
202
+ objArrayOop from_obj_array = objArrayOop (from_obj);
203
+ // The from-space object contains the real length.
204
+ int length = from_obj_array->length ();
209
205
210
- oop to_obj = from_obj->forwardee ();
206
+ assert (from_obj->is_forwarded (), " must be forwarded" );
207
+ oop to_obj = from_obj->forwardee ();
211
208
assert (from_obj != to_obj, " should not be chunking self-forwarded objects" );
212
- assert (to_obj->is_objArray (), " must be obj array" );
213
- objArrayOop to_array = objArrayOop (to_obj);
214
-
215
- PartialArrayTaskStepper::Step step
216
- = _partial_array_stepper.next (objArrayOop (from_obj),
217
- to_array,
218
- _partial_objarray_chunk_size);
219
- for (uint i = 0 ; i < step._ncreate ; ++i) {
209
+ objArrayOop to_obj_array = objArrayOop (to_obj);
210
+ // We keep track of the next start index in the length field of the
211
+ // to-space object.
212
+ int next_index = to_obj_array->length ();
213
+ assert (0 <= next_index && next_index < length,
214
+ " invariant, next index: %d, length: %d" , next_index, length);
215
+
216
+ int start = next_index;
217
+ int end = length;
218
+ int remainder = end - start;
219
+ // We'll try not to push a range that's smaller than ParGCArrayScanChunk.
220
+ if (remainder > 2 * ParGCArrayScanChunk) {
221
+ end = start + ParGCArrayScanChunk;
222
+ to_obj_array->set_length (end);
223
+ // Push the remainder before we process the range in case another
224
+ // worker has run out of things to do and can steal it.
220
225
push_on_queue (ScannerTask (PartialArrayScanTask (from_obj)));
226
+ } else {
227
+ assert (length == end, " sanity" );
228
+ // We'll process the final range for this object. Restore the length
229
+ // so that the heap remains parsable in case of evacuation failure.
230
+ to_obj_array->set_length (end);
221
231
}
222
232
223
- HeapRegion* hr = _g1h->heap_region_containing (to_array );
233
+ HeapRegion* hr = _g1h->heap_region_containing (to_obj );
224
234
G1ScanInYoungSetter x (&_scanner, hr->is_young ());
225
- // Process claimed task. The length of to_array is not correct, but
226
- // fortunately the iteration ignores the length field and just relies
227
- // on start/end.
228
- to_array->oop_iterate_range (&_scanner,
229
- step._index ,
230
- step._index + _partial_objarray_chunk_size);
231
- }
232
-
233
- void G1ParScanThreadState::start_partial_objarray (G1HeapRegionAttr dest_attr,
234
- oop from_obj,
235
- oop to_obj) {
236
- assert (from_obj->is_objArray (), " precondition" );
237
- assert (from_obj->is_forwarded (), " precondition" );
238
- assert (from_obj->forwardee () == to_obj, " precondition" );
239
- assert (from_obj != to_obj, " should not be scanning self-forwarded objects" );
240
- assert (to_obj->is_objArray (), " precondition" );
241
-
242
- objArrayOop to_array = objArrayOop (to_obj);
243
-
244
- PartialArrayTaskStepper::Step step
245
- = _partial_array_stepper.start (objArrayOop (from_obj),
246
- to_array,
247
- _partial_objarray_chunk_size);
248
-
249
- // Push any needed partial scan tasks. Pushed before processing the
250
- // intitial chunk to allow other workers to steal while we're processing.
251
- for (uint i = 0 ; i < step._ncreate ; ++i) {
252
- push_on_queue (ScannerTask (PartialArrayScanTask (from_obj)));
253
- }
254
-
255
- G1ScanInYoungSetter x (&_scanner, dest_attr.is_young ());
256
- // Process the initial chunk. No need to process the type in the
257
- // klass, as it will already be handled by processing the built-in
258
- // module. The length of to_array is not correct, but fortunately
259
- // the iteration ignores that length field and relies on start/end.
260
- to_array->oop_iterate_range (&_scanner, 0 , step._index );
235
+ // Process indexes [start,end). It will also process the header
236
+ // along with the first chunk (i.e., the chunk with start == 0).
237
+ // Note that at this point the length field of to_obj_array is not
238
+ // correct given that we are using it to keep track of the next
239
+ // start index. oop_iterate_range() (thankfully!) ignores the length
240
+ // field and only relies on the start / end parameters. It does
241
+ // however return the size of the object which will be incorrect. So
242
+ // we have to ignore it even if we wanted to use it.
243
+ to_obj_array->oop_iterate_range (&_scanner, start, end);
261
244
}
262
245
263
246
void G1ParScanThreadState::dispatch_task (ScannerTask task) {
@@ -411,10 +394,7 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio
411
394
assert (region_attr.is_in_cset (),
412
395
" Unexpected region attr type: %s" , region_attr.get_type_str ());
413
396
414
- // Get the klass once. We'll need it again later, and this avoids
415
- // re-decoding when it's compressed.
416
- Klass* klass = old->klass ();
417
- const size_t word_sz = old->size_given_klass (klass);
397
+ const size_t word_sz = old->size ();
418
398
419
399
uint age = 0 ;
420
400
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
481
461
obj->set_mark_raw (old_mark);
482
462
}
483
463
484
- // Most objects are not arrays, so do one array check rather than
485
- // checking for each array category for each object.
486
- if (klass->is_array_klass ()) {
487
- if (klass->is_objArray_klass ()) {
488
- start_partial_objarray (dest_attr, old, obj);
489
- } else {
490
- // Nothing needs to be done for typeArrays. Body doesn't contain
491
- // any oops to scan, and the type in the klass will already be handled
492
- // by processing the built-in module.
493
- assert (klass->is_typeArray_klass (), " invariant" );
494
- }
495
- return obj;
496
- }
497
-
498
464
if (G1StringDedup::is_enabled ()) {
499
465
const bool is_from_young = region_attr.is_young ();
500
466
const bool is_to_young = dest_attr.is_young ();
@@ -508,10 +474,17 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio
508
474
obj);
509
475
}
510
476
511
- G1ScanInYoungSetter x (&_scanner, dest_attr.is_young ());
512
- obj->oop_iterate_backwards (&_scanner);
477
+ if (obj->is_objArray () && arrayOop (obj)->length () >= ParGCArrayScanChunk) {
478
+ // We keep track of the next start index in the length field of
479
+ // the to-space object. The actual length can be found in the
480
+ // length field of the from-space object.
481
+ arrayOop (obj)->set_length (0 );
482
+ do_partial_array (PartialArrayScanTask (old));
483
+ } else {
484
+ G1ScanInYoungSetter x (&_scanner, dest_attr.is_young ());
485
+ obj->oop_iterate_backwards (&_scanner);
486
+ }
513
487
return obj;
514
-
515
488
} else {
516
489
_plab_allocator->undo_allocation (dest_attr, obj_ptr, word_sz, node_index);
517
490
return forward_ptr;
@@ -530,9 +503,7 @@ G1ParScanThreadState* G1ParScanThreadStateSet::state_for_worker(uint worker_id)
530
503
assert (worker_id < _n_workers, " out of bounds access" );
531
504
if (_states[worker_id] == NULL ) {
532
505
_states[worker_id] =
533
- new G1ParScanThreadState (_g1h, _rdcqs,
534
- worker_id, _n_workers,
535
- _young_cset_length, _optional_cset_length);
506
+ new G1ParScanThreadState (_g1h, _rdcqs, worker_id, _young_cset_length, _optional_cset_length);
536
507
}
537
508
return _states[worker_id];
538
509
}
0 commit comments