@@ -3051,7 +3051,7 @@ bool G1ParEvacuateFollowersClosure::offer_termination() {
3051
3051
EventGCPhaseParallel event;
3052
3052
G1ParScanThreadState* const pss = par_scan_state ();
3053
3053
start_term_time ();
3054
- const bool res = terminator ()->offer_termination ();
3054
+ const bool res = ( terminator () == nullptr ) ? true : terminator ()->offer_termination ();
3055
3055
end_term_time ();
3056
3056
event.commit (GCId::current (), pss->worker_id (), G1GCPhaseTimes::phase_name (G1GCPhaseTimes::Termination));
3057
3057
return res;
@@ -3182,99 +3182,35 @@ class G1STWDrainQueueClosure: public VoidClosure {
3182
3182
}
3183
3183
};
3184
3184
3185
- // Parallel Reference Processing closures
3186
-
3187
- // Implementation of AbstractRefProcTaskExecutor for parallel reference
3188
- // processing during G1 evacuation pauses.
3189
-
3190
- class G1STWRefProcTaskExecutor : public AbstractRefProcTaskExecutor {
3191
- private:
3192
- G1CollectedHeap* _g1h;
3193
- G1ParScanThreadStateSet* _pss;
3194
- G1ScannerTasksQueueSet* _queues;
3195
- WorkGang* _workers;
3196
-
3197
- public:
3198
- G1STWRefProcTaskExecutor (G1CollectedHeap* g1h,
3199
- G1ParScanThreadStateSet* per_thread_states,
3200
- WorkGang* workers,
3201
- G1ScannerTasksQueueSet *task_queues) :
3202
- _g1h (g1h),
3203
- _pss (per_thread_states),
3204
- _queues (task_queues),
3205
- _workers (workers)
3206
- {
3207
- g1h->ref_processor_stw ()->set_active_mt_degree (workers->active_workers ());
3208
- }
3209
-
3210
- // Executes the given task using concurrent marking worker threads.
3211
- virtual void execute (ProcessTask& task, uint ergo_workers);
3212
- };
3213
-
3214
- // Gang task for possibly parallel reference processing
3215
-
3216
- class G1STWRefProcTaskProxy : public AbstractGangTask {
3217
- typedef AbstractRefProcTaskExecutor::ProcessTask ProcessTask;
3218
- ProcessTask& _proc_task;
3219
- G1CollectedHeap* _g1h;
3220
- G1ParScanThreadStateSet* _pss;
3221
- G1ScannerTasksQueueSet* _task_queues;
3222
- TaskTerminator* _terminator;
3185
+ class G1STWRefProcProxyTask : public RefProcProxyTask {
3186
+ G1CollectedHeap& _g1h;
3187
+ G1ParScanThreadStateSet& _pss;
3188
+ TaskTerminator _terminator;
3189
+ G1ScannerTasksQueueSet& _task_queues;
3223
3190
3224
3191
public:
3225
- G1STWRefProcTaskProxy (ProcessTask& proc_task,
3226
- G1CollectedHeap* g1h,
3227
- G1ParScanThreadStateSet* per_thread_states,
3228
- G1ScannerTasksQueueSet *task_queues,
3229
- TaskTerminator* terminator) :
3230
- AbstractGangTask (" Process reference objects in parallel" ),
3231
- _proc_task (proc_task),
3232
- _g1h (g1h),
3233
- _pss (per_thread_states),
3234
- _task_queues (task_queues),
3235
- _terminator (terminator)
3236
- {}
3237
-
3238
- virtual void work (uint worker_id) {
3239
- // The reference processing task executed by a single worker.
3240
- ResourceMark rm;
3241
-
3242
- G1STWIsAliveClosure is_alive (_g1h);
3243
-
3244
- G1ParScanThreadState* pss = _pss->state_for_worker (worker_id);
3245
- pss->set_ref_discoverer (NULL );
3246
-
3247
- // Keep alive closure.
3248
- G1CopyingKeepAliveClosure keep_alive (_g1h, pss);
3249
-
3250
- // Complete GC closure
3251
- G1ParEvacuateFollowersClosure drain_queue (_g1h, pss, _task_queues, _terminator, G1GCPhaseTimes::ObjCopy);
3192
+ G1STWRefProcProxyTask (uint max_workers, G1CollectedHeap& g1h, G1ParScanThreadStateSet& pss, G1ScannerTasksQueueSet& task_queues)
3193
+ : RefProcProxyTask(" G1STWRefProcProxyTask" , max_workers),
3194
+ _g1h (g1h),
3195
+ _pss(pss),
3196
+ _terminator(max_workers, &task_queues),
3197
+ _task_queues(task_queues) {}
3252
3198
3253
- // Call the reference processing task's work routine.
3254
- _proc_task.work (worker_id, is_alive, keep_alive, drain_queue);
3199
+ void work (uint worker_id) override {
3200
+ assert (worker_id < _max_workers, " sanity" );
3201
+ uint index = (_tm == RefProcThreadModel::Single) ? 0 : worker_id;
3202
+ _pss.state_for_worker (index )->set_ref_discoverer (nullptr );
3203
+ G1STWIsAliveClosure is_alive (&_g1h);
3204
+ G1CopyingKeepAliveClosure keep_alive (&_g1h, _pss.state_for_worker (index ));
3205
+ G1ParEvacuateFollowersClosure complete_gc (&_g1h, _pss.state_for_worker (index ), &_task_queues, _tm == RefProcThreadModel::Single ? nullptr : &_terminator, G1GCPhaseTimes::ObjCopy);
3206
+ _rp_task->rp_work (worker_id, &is_alive, &keep_alive, &complete_gc);
3207
+ }
3255
3208
3256
- // Note we cannot assert that the refs array is empty here as not all
3257
- // of the processing tasks (specifically phase2 - pp2_work) execute
3258
- // the complete_gc closure (which ordinarily would drain the queue) so
3259
- // the queue may not be empty.
3209
+ void prepare_run_task_hook () override {
3210
+ _terminator.reset_for_reuse (_queue_count);
3260
3211
}
3261
3212
};
3262
3213
3263
- // Driver routine for parallel reference processing.
3264
- // Creates an instance of the ref processing gang
3265
- // task and has the worker threads execute it.
3266
- void G1STWRefProcTaskExecutor::execute (ProcessTask& proc_task, uint ergo_workers) {
3267
- assert (_workers != NULL , " Need parallel worker threads." );
3268
-
3269
- assert (_workers->active_workers () >= ergo_workers,
3270
- " Ergonomically chosen workers (%u) should be less than or equal to active workers (%u)" ,
3271
- ergo_workers, _workers->active_workers ());
3272
- TaskTerminator terminator (ergo_workers, _queues);
3273
- G1STWRefProcTaskProxy proc_task_proxy (proc_task, _g1h, _pss, _queues, &terminator);
3274
-
3275
- _workers->run_task (&proc_task_proxy, ergo_workers);
3276
- }
3277
-
3278
3214
// End of weak reference support closures
3279
3215
3280
3216
void G1CollectedHeap::process_discovered_references (G1ParScanThreadStateSet* per_thread_states) {
@@ -3283,53 +3219,27 @@ void G1CollectedHeap::process_discovered_references(G1ParScanThreadStateSet* per
3283
3219
ReferenceProcessor* rp = _ref_processor_stw;
3284
3220
assert (rp->discovery_enabled (), " should have been enabled" );
3285
3221
3286
- // Closure to test whether a referent is alive.
3287
- G1STWIsAliveClosure is_alive (this );
3288
-
3289
- // Even when parallel reference processing is enabled, the processing
3290
- // of JNI refs is serial and performed serially by the current thread
3291
- // rather than by a worker. The following PSS will be used for processing
3292
- // JNI refs.
3293
-
3294
3222
// Use only a single queue for this PSS.
3295
3223
G1ParScanThreadState* pss = per_thread_states->state_for_worker (0 );
3296
3224
pss->set_ref_discoverer (NULL );
3297
3225
assert (pss->queue_is_empty (), " pre-condition" );
3298
3226
3299
- // Keep alive closure.
3300
- G1CopyingKeepAliveClosure keep_alive (this , pss);
3301
-
3302
- // Serial Complete GC closure
3303
- G1STWDrainQueueClosure drain_queue (this , pss);
3304
-
3305
3227
// Setup the soft refs policy...
3306
3228
rp->setup_policy (false );
3307
3229
3308
- ReferenceProcessorPhaseTimes* pt = phase_times ()->ref_phase_times ();
3230
+ ReferenceProcessorPhaseTimes& pt = * phase_times ()->ref_phase_times ();
3309
3231
3310
3232
ReferenceProcessorStats stats;
3311
- if (!rp->processing_is_mt ()) {
3312
- // Serial reference processing...
3313
- stats = rp->process_discovered_references (&is_alive,
3314
- &keep_alive,
3315
- &drain_queue,
3316
- NULL ,
3317
- pt);
3318
- } else {
3319
- uint no_of_gc_workers = workers ()->active_workers ();
3233
+ uint no_of_gc_workers = workers ()->active_workers ();
3320
3234
3321
- // Parallel reference processing
3322
- assert (no_of_gc_workers <= rp->max_num_queues (),
3323
- " Mismatch between the number of GC workers %u and the maximum number of Reference process queues %u" ,
3324
- no_of_gc_workers, rp->max_num_queues ());
3235
+ // Parallel reference processing
3236
+ assert (no_of_gc_workers <= rp->max_num_queues (),
3237
+ " Mismatch between the number of GC workers %u and the maximum number of Reference process queues %u" ,
3238
+ no_of_gc_workers, rp->max_num_queues ());
3325
3239
3326
- G1STWRefProcTaskExecutor par_task_executor (this , per_thread_states, workers (), _task_queues);
3327
- stats = rp->process_discovered_references (&is_alive,
3328
- &keep_alive,
3329
- &drain_queue,
3330
- &par_task_executor,
3331
- pt);
3332
- }
3240
+ rp->set_active_mt_degree (no_of_gc_workers);
3241
+ G1STWRefProcProxyTask task (rp->max_num_queues (), *this , *per_thread_states, *_task_queues);
3242
+ stats = rp->process_discovered_references (task, pt);
3333
3243
3334
3244
_gc_tracer_stw->report_gc_reference_stats (stats);
3335
3245
0 commit comments