Skip to content

Commit 19be497

Browse files
committedJun 10, 2020
8237354: Add option to jcmd to write a gzipped heap dump
Reviewed-by: rrich, clanger, goetz
1 parent 2e8356e commit 19be497

File tree

13 files changed

+1708
-121
lines changed

13 files changed

+1708
-121
lines changed
 

‎src/hotspot/share/gc/shared/workgroup.cpp

+19-4
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,17 @@ void AbstractWorkGang::threads_do(ThreadClosure* tc) const {
9494
}
9595
}
9696

97+
static void run_foreground_task_if_needed(AbstractGangTask* task, uint num_workers,
98+
bool add_foreground_work) {
99+
if (add_foreground_work) {
100+
log_develop_trace(gc, workgang)("Running work gang: %s task: %s worker: foreground",
101+
Thread::current()->name(), task->name());
102+
task->work(num_workers);
103+
log_develop_trace(gc, workgang)("Finished work gang: %s task: %s worker: foreground "
104+
"thread: " PTR_FORMAT, Thread::current()->name(), task->name(), p2i(Thread::current()));
105+
}
106+
}
107+
97108
// WorkGang dispatcher implemented with semaphores.
98109
//
99110
// Semaphores don't require the worker threads to re-claim the lock when they wake up.
@@ -124,14 +135,16 @@ class SemaphoreGangTaskDispatcher : public GangTaskDispatcher {
124135
delete _end_semaphore;
125136
}
126137

127-
void coordinator_execute_on_workers(AbstractGangTask* task, uint num_workers) {
138+
void coordinator_execute_on_workers(AbstractGangTask* task, uint num_workers, bool add_foreground_work) {
128139
// No workers are allowed to read the state variables until they have been signaled.
129140
_task = task;
130141
_not_finished = num_workers;
131142

132143
// Dispatch 'num_workers' number of tasks.
133144
_start_semaphore->signal(num_workers);
134145

146+
run_foreground_task_if_needed(task, num_workers, add_foreground_work);
147+
135148
// Wait for the last worker to signal the coordinator.
136149
_end_semaphore->wait();
137150

@@ -188,7 +201,7 @@ class MutexGangTaskDispatcher : public GangTaskDispatcher {
188201
delete _monitor;
189202
}
190203

191-
void coordinator_execute_on_workers(AbstractGangTask* task, uint num_workers) {
204+
void coordinator_execute_on_workers(AbstractGangTask* task, uint num_workers, bool add_foreground_work) {
192205
MonitorLocker ml(_monitor, Mutex::_no_safepoint_check_flag);
193206

194207
_task = task;
@@ -197,6 +210,8 @@ class MutexGangTaskDispatcher : public GangTaskDispatcher {
197210
// Tell the workers to get to work.
198211
_monitor->notify_all();
199212

213+
run_foreground_task_if_needed(task, num_workers, add_foreground_work);
214+
200215
// Wait for them to finish.
201216
while (_finished < _num_workers) {
202217
ml.wait();
@@ -263,14 +278,14 @@ void WorkGang::run_task(AbstractGangTask* task) {
263278
run_task(task, active_workers());
264279
}
265280

266-
void WorkGang::run_task(AbstractGangTask* task, uint num_workers) {
281+
void WorkGang::run_task(AbstractGangTask* task, uint num_workers, bool add_foreground_work) {
267282
guarantee(num_workers <= total_workers(),
268283
"Trying to execute task %s with %u workers which is more than the amount of total workers %u.",
269284
task->name(), num_workers, total_workers());
270285
guarantee(num_workers > 0, "Trying to execute task %s with zero workers", task->name());
271286
uint old_num_workers = _active_workers;
272287
update_active_workers(num_workers);
273-
_dispatcher->coordinator_execute_on_workers(task, num_workers);
288+
_dispatcher->coordinator_execute_on_workers(task, num_workers, add_foreground_work);
274289
update_active_workers(old_num_workers);
275290
}
276291

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

+5-3
Original file line numberDiff line numberDiff line change
@@ -90,7 +90,8 @@ class GangTaskDispatcher : public CHeapObj<mtGC> {
9090

9191
// Distributes the task out to num_workers workers.
9292
// Returns when the task has been completed by all workers.
93-
virtual void coordinator_execute_on_workers(AbstractGangTask* task, uint num_workers) = 0;
93+
virtual void coordinator_execute_on_workers(AbstractGangTask* task, uint num_workers,
94+
bool add_foreground_work) = 0;
9495

9596
// Worker API.
9697

@@ -215,8 +216,9 @@ class WorkGang: public AbstractWorkGang {
215216
// Run a task with the given number of workers, returns
216217
// when the task is done. The number of workers must be at most the number of
217218
// active workers. Additional workers may be created if an insufficient
218-
// number currently exists.
219-
void run_task(AbstractGangTask* task, uint num_workers);
219+
// number currently exists. If the add_foreground_work flag is true, the current thread
220+
// is used to run the task too.
221+
void run_task(AbstractGangTask* task, uint num_workers, bool add_foreground_work = false);
220222

221223
protected:
222224
virtual AbstractGangWorker* allocate_worker(uint which);

‎src/hotspot/share/services/diagnosticCommand.cpp

+17-2
Original file line numberDiff line numberDiff line change
@@ -504,17 +504,32 @@ HeapDumpDCmd::HeapDumpDCmd(outputStream* output, bool heap) :
504504
DCmdWithParser(output, heap),
505505
_filename("filename","Name of the dump file", "STRING",true),
506506
_all("-all", "Dump all objects, including unreachable objects",
507-
"BOOLEAN", false, "false") {
507+
"BOOLEAN", false, "false"),
508+
_gzip("-gz", "If specified, the heap dump is written in gzipped format "
509+
"using the given compression level. 1 (recommended) is the fastest, "
510+
"9 the strongest compression.", "INT", false, "1") {
508511
_dcmdparser.add_dcmd_option(&_all);
509512
_dcmdparser.add_dcmd_argument(&_filename);
513+
_dcmdparser.add_dcmd_option(&_gzip);
510514
}
511515

512516
void HeapDumpDCmd::execute(DCmdSource source, TRAPS) {
517+
jlong level = -1; // -1 means no compression.
518+
519+
if (_gzip.is_set()) {
520+
level = _gzip.value();
521+
522+
if (level < 1 || level > 9) {
523+
output()->print_cr("Compression level out of range (1-9): " JLONG_FORMAT, level);
524+
return;
525+
}
526+
}
527+
513528
// Request a full GC before heap dump if _all is false
514529
// This helps reduces the amount of unreachable objects in the dump
515530
// and makes it easier to browse.
516531
HeapDumper dumper(!_all.value() /* request GC if _all is false*/);
517-
dumper.dump(_filename.value(), output());
532+
dumper.dump(_filename.value(), output(), (int) level);
518533
}
519534

520535
int HeapDumpDCmd::num_arguments() {

‎src/hotspot/share/services/diagnosticCommand.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -330,6 +330,7 @@ class HeapDumpDCmd : public DCmdWithParser {
330330
protected:
331331
DCmdArgument<char*> _filename;
332332
DCmdArgument<bool> _all;
333+
DCmdArgument<jlong> _gzip;
333334
public:
334335
HeapDumpDCmd(outputStream* output, bool heap);
335336
static const char* name() {

0 commit comments

Comments
 (0)
Please sign in to comment.