Skip to content

Commit 4a9f8ef

Browse files
author
Ivan Walulya
committedApr 24, 2023
8057586: Explicit GC ignored if GCLocker is active
Reviewed-by: tschatzl, ayang
1 parent ce493dd commit 4a9f8ef

12 files changed

+328
-22
lines changed
 

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

+30-5
Original file line numberDiff line numberDiff line change
@@ -1928,6 +1928,35 @@ bool G1CollectedHeap::try_collect_concurrently(GCCause::Cause cause,
19281928
}
19291929
}
19301930

1931+
bool G1CollectedHeap::try_collect_fullgc(GCCause::Cause cause,
1932+
const G1GCCounters& counters_before) {
1933+
assert_heap_not_locked();
1934+
1935+
while(true) {
1936+
VM_G1CollectFull op(counters_before.total_collections(),
1937+
counters_before.total_full_collections(),
1938+
cause);
1939+
VMThread::execute(&op);
1940+
1941+
// Request is trivially finished.
1942+
if (!GCCause::is_explicit_full_gc(cause) || op.gc_succeeded()) {
1943+
return op.gc_succeeded();
1944+
}
1945+
1946+
{
1947+
MutexLocker ml(Heap_lock);
1948+
if (counters_before.total_full_collections() != total_full_collections()) {
1949+
return true;
1950+
}
1951+
}
1952+
1953+
if (GCLocker::is_active_and_needs_gc()) {
1954+
// If GCLocker is active, wait until clear before retrying.
1955+
GCLocker::stall_until_clear();
1956+
}
1957+
}
1958+
}
1959+
19311960
bool G1CollectedHeap::try_collect(GCCause::Cause cause,
19321961
const G1GCCounters& counters_before) {
19331962
if (should_do_concurrent_full_gc(cause)) {
@@ -1951,11 +1980,7 @@ bool G1CollectedHeap::try_collect(GCCause::Cause cause,
19511980
return op.gc_succeeded();
19521981
} else {
19531982
// Schedule a Full GC.
1954-
VM_G1CollectFull op(counters_before.total_collections(),
1955-
counters_before.total_full_collections(),
1956-
cause);
1957-
VMThread::execute(&op);
1958-
return op.gc_succeeded();
1983+
return try_collect_fullgc(cause, counters_before);
19591984
}
19601985
}
19611986

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

+3
Original file line numberDiff line numberDiff line change
@@ -283,6 +283,9 @@ class G1CollectedHeap : public CollectedHeap {
283283
uint gc_counter,
284284
uint old_marking_started_before);
285285

286+
bool try_collect_fullgc(GCCause::Cause cause,
287+
const G1GCCounters& counters_before);
288+
286289
// indicates whether we are in young or mixed GC mode
287290
G1CollectorState _collector_state;
288291

‎src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp

+20-2
Original file line numberDiff line numberDiff line change
@@ -549,8 +549,26 @@ void ParallelScavengeHeap::collect(GCCause::Cause cause) {
549549
return;
550550
}
551551

552-
VM_ParallelGCSystemGC op(gc_count, full_gc_count, cause);
553-
VMThread::execute(&op);
552+
while (true) {
553+
VM_ParallelGCSystemGC op(gc_count, full_gc_count, cause);
554+
VMThread::execute(&op);
555+
556+
if (!GCCause::is_explicit_full_gc(cause) || op.full_gc_succeeded()) {
557+
return;
558+
}
559+
560+
{
561+
MutexLocker ml(Heap_lock);
562+
if (full_gc_count != total_full_collections()) {
563+
return;
564+
}
565+
}
566+
567+
if (GCLocker::is_active_and_needs_gc()) {
568+
// If GCLocker is active, wait until clear before retrying.
569+
GCLocker::stall_until_clear();
570+
}
571+
}
554572
}
555573

556574
void ParallelScavengeHeap::object_iterate(ObjectClosure* cl) {

‎src/hotspot/share/gc/parallel/parallelScavengeHeap.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -211,7 +211,7 @@ class ParallelScavengeHeap : public CollectedHeap {
211211
// will then attempt a full gc. The second collects the entire heap; if
212212
// maximum_compaction is true, it will compact everything and clear all soft
213213
// references.
214-
inline void invoke_scavenge();
214+
inline bool invoke_scavenge();
215215

216216
// Perform a full collection
217217
void do_full_collection(bool clear_all_soft_refs) override;

‎src/hotspot/share/gc/parallel/parallelScavengeHeap.inline.hpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2006, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2006, 2023, 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
@@ -35,8 +35,8 @@ inline bool ParallelScavengeHeap::should_alloc_in_eden(const size_t size) const
3535
return size < eden_size / 2;
3636
}
3737

38-
inline void ParallelScavengeHeap::invoke_scavenge() {
39-
PSScavenge::invoke();
38+
inline bool ParallelScavengeHeap::invoke_scavenge() {
39+
return PSScavenge::invoke();
4040
}
4141

4242
inline bool ParallelScavengeHeap::is_in_young(const void* p) const {

‎src/hotspot/share/gc/parallel/psParallelCompact.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -1678,7 +1678,7 @@ void PSParallelCompact::summary_phase(bool maximum_compaction)
16781678
// may be true because this method can be called without intervening
16791679
// activity. For example when the heap space is tight and full measure
16801680
// are being taken to free space.
1681-
void PSParallelCompact::invoke(bool maximum_heap_compaction) {
1681+
bool PSParallelCompact::invoke(bool maximum_heap_compaction) {
16821682
assert(SafepointSynchronize::is_at_safepoint(), "should be at safepoint");
16831683
assert(Thread::current() == (Thread*)VMThread::vm_thread(),
16841684
"should be in vm thread");
@@ -1695,8 +1695,8 @@ void PSParallelCompact::invoke(bool maximum_heap_compaction) {
16951695
const bool clear_all_soft_refs =
16961696
heap->soft_ref_policy()->should_clear_all_soft_refs();
16971697

1698-
PSParallelCompact::invoke_no_policy(clear_all_soft_refs ||
1699-
maximum_heap_compaction);
1698+
return PSParallelCompact::invoke_no_policy(clear_all_soft_refs ||
1699+
maximum_heap_compaction);
17001700
}
17011701

17021702
// This method contains no policy. You should probably

‎src/hotspot/share/gc/parallel/psParallelCompact.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -1141,7 +1141,7 @@ class PSParallelCompact : AllStatic {
11411141

11421142
PSParallelCompact();
11431143

1144-
static void invoke(bool maximum_heap_compaction);
1144+
static bool invoke(bool maximum_heap_compaction);
11451145
static bool invoke_no_policy(bool maximum_heap_compaction);
11461146

11471147
static void post_initialize();

‎src/hotspot/share/gc/parallel/psVMOperations.cpp

+4-3
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,8 @@ static bool is_cause_full(GCCause::Cause cause) {
5858
VM_ParallelGCSystemGC::VM_ParallelGCSystemGC(uint gc_count,
5959
uint full_gc_count,
6060
GCCause::Cause gc_cause) :
61-
VM_GC_Operation(gc_count, gc_cause, full_gc_count, is_cause_full(gc_cause))
61+
VM_GC_Operation(gc_count, gc_cause, full_gc_count, is_cause_full(gc_cause)),
62+
_full_gc_succeeded(false)
6263
{
6364
}
6465

@@ -70,8 +71,8 @@ void VM_ParallelGCSystemGC::doit() {
7071
GCCauseSetter gccs(heap, _gc_cause);
7172
if (!_full) {
7273
// If (and only if) the scavenge fails, this will invoke a full gc.
73-
heap->invoke_scavenge();
74+
_full_gc_succeeded = heap->invoke_scavenge();
7475
} else {
75-
heap->do_full_collection(false);
76+
_full_gc_succeeded = PSParallelCompact::invoke(false);
7677
}
7778
}

‎src/hotspot/share/gc/parallel/psVMOperations.hpp

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2007, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2007, 2023, 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
@@ -40,10 +40,12 @@ class VM_ParallelGCFailedAllocation : public VM_CollectForAllocation {
4040
};
4141

4242
class VM_ParallelGCSystemGC: public VM_GC_Operation {
43+
bool _full_gc_succeeded;
4344
public:
4445
VM_ParallelGCSystemGC(uint gc_count, uint full_gc_count, GCCause::Cause gc_cause);
4546
virtual VMOp_Type type() const { return VMOp_ParallelGCSystemGC; }
4647
virtual void doit();
48+
bool full_gc_succeeded() const { return _full_gc_succeeded; }
4749
};
4850

4951
#endif // SHARE_GC_PARALLEL_PSVMOPERATIONS_HPP

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

+6
Original file line numberDiff line numberDiff line change
@@ -95,6 +95,12 @@ class GCCause : public AllStatic {
9595
cause == GCCause::_dcmd_gc_run);
9696
}
9797

98+
inline static bool is_explicit_full_gc(GCCause::Cause cause) {
99+
return (is_user_requested_gc(cause) ||
100+
is_serviceability_requested_gc(cause) ||
101+
cause == GCCause::_wb_full_gc);
102+
}
103+
98104
inline static bool is_serviceability_requested_gc(GCCause::Cause
99105
cause) {
100106
return (cause == GCCause::_jvmti_force_gc ||

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

+22-3
Original file line numberDiff line numberDiff line change
@@ -796,9 +796,28 @@ void GenCollectedHeap::collect(GCCause::Cause cause) {
796796
? YoungGen
797797
: OldGen;
798798

799-
VM_GenCollectFull op(gc_count_before, full_gc_count_before,
800-
cause, max_generation);
801-
VMThread::execute(&op);
799+
while (true) {
800+
VM_GenCollectFull op(gc_count_before, full_gc_count_before,
801+
cause, max_generation);
802+
VMThread::execute(&op);
803+
804+
if (!GCCause::is_explicit_full_gc(cause)) {
805+
return;
806+
}
807+
808+
{
809+
MutexLocker ml(Heap_lock);
810+
// Read the GC count while holding the Heap_lock
811+
if (full_gc_count_before != total_full_collections()) {
812+
return;
813+
}
814+
}
815+
816+
if (GCLocker::is_active_and_needs_gc()) {
817+
// If GCLocker is active, wait until clear before retrying.
818+
GCLocker::stall_until_clear();
819+
}
820+
}
802821
}
803822

804823
void GenCollectedHeap::do_full_collection(bool clear_all_soft_refs) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,232 @@
1+
/*
2+
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test G1 Full GC execution when JNICritical is active
26+
* @summary Check that Full GC calls are not ignored if concurrent with an active GCLocker.
27+
* @bug 8057586
28+
* @requires vm.gc.G1
29+
* @modules java.base/jdk.internal.misc
30+
* @library /test/lib
31+
* @build jdk.test.whitebox.WhiteBox
32+
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
33+
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xms3g -Xmx3g -Xmn2g -Xlog:gc TestJNICriticalStressTest 30 4 4 G1
34+
*/
35+
36+
/*
37+
* @test Parallel Full GC execution when JNICritical is active
38+
* @bug 8057586
39+
* @requires vm.gc.Parallel
40+
* @modules java.base/jdk.internal.misc
41+
* @library /test/lib
42+
* @build jdk.test.whitebox.WhiteBox
43+
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
44+
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -XX:+UseParallelGC -Xms3g -Xmx3g -Xmn2g -Xlog:gc TestJNICriticalStressTest 30 4 4 Parallel
45+
*/
46+
47+
/*
48+
* @test Serial Full GC execution when JNICritical is active
49+
* @bug 8057586
50+
* @requires vm.gc.Serial
51+
* @modules java.base/jdk.internal.misc
52+
* @library /test/lib
53+
* @build jdk.test.whitebox.WhiteBox
54+
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
55+
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -XX:+UseSerialGC -Xms3g -Xmx3g -Xmn2g -Xlog:gc TestJNICriticalStressTest 30 4 4 Serial
56+
*/
57+
58+
import jdk.test.lib.Asserts;
59+
import jdk.test.whitebox.WhiteBox;
60+
61+
import java.lang.management.GarbageCollectorMXBean;
62+
import java.lang.management.ManagementFactory;
63+
import java.util.List;
64+
import java.util.HashMap;
65+
import java.util.Map;
66+
import java.util.zip.Deflater;
67+
68+
/**
69+
* Test verifies that Full GC calls are not ignored if concurrent with an active GCLocker.
70+
*
71+
* The test checks that at least a full gc is executed in the duration of a WhiteBox.fullGC() call;
72+
* either by the calling thread or a concurrent thread.
73+
*/
74+
public class TestJNICriticalStressTest {
75+
private static final WhiteBox wb = WhiteBox.getWhiteBox();
76+
77+
static private final int LARGE_MAP_SIZE = 64 * 1024;
78+
79+
static private final int MAP_ARRAY_LENGTH = 4;
80+
static private final int MAP_SIZE = 1024;
81+
82+
static private final int BYTE_ARRAY_LENGTH = 16 * 1024;
83+
84+
static private final long SYSTEM_GC_PERIOD_MS = 5 * 1000;
85+
static private long gcCountBefore = 0;
86+
static private GarbageCollectorMXBean collector = null;
87+
88+
static private void println(String str) { System.out.println(str); }
89+
static private void exit(int code) { System.exit(code); }
90+
91+
static Map<Integer,String> populateMap(int size) {
92+
Map<Integer,String> map = new HashMap<Integer,String>();
93+
for (int i = 0; i < size; i += 1) {
94+
Integer keyInt = Integer.valueOf(i);
95+
String valStr = "value is [" + i + "]";
96+
map.put(keyInt,valStr);
97+
}
98+
return map;
99+
}
100+
101+
static private class AllocatingWorker implements Runnable {
102+
private final Object[] array = new Object[MAP_ARRAY_LENGTH];
103+
private int arrayIndex = 0;
104+
105+
private void doStep() {
106+
Map<Integer,String> map = populateMap(MAP_SIZE);
107+
array[arrayIndex] = map;
108+
arrayIndex = (arrayIndex + 1) % MAP_ARRAY_LENGTH;
109+
}
110+
111+
public void run() {
112+
while (true) {
113+
doStep();
114+
}
115+
}
116+
}
117+
118+
static private class JNICriticalWorker implements Runnable {
119+
private int count;
120+
121+
private void doStep() {
122+
byte[] inputArray = new byte[BYTE_ARRAY_LENGTH];
123+
for (int i = 0; i < inputArray.length; i += 1) {
124+
inputArray[i] = (byte) (count + i);
125+
}
126+
127+
Deflater deflater = new Deflater();
128+
deflater.setInput(inputArray);
129+
deflater.finish();
130+
131+
byte[] outputArray = new byte[2 * inputArray.length];
132+
deflater.deflate(outputArray);
133+
deflater.end();
134+
135+
count += 1;
136+
}
137+
138+
public void run() {
139+
while (true) {
140+
doStep();
141+
}
142+
}
143+
}
144+
145+
static private class SystemGCWorker implements Runnable {
146+
public void run() {
147+
long fullGcCounts = 0;
148+
while (true) {
149+
try {
150+
Thread.sleep(SYSTEM_GC_PERIOD_MS);
151+
} catch (InterruptedException e) {
152+
e.printStackTrace();
153+
return;
154+
}
155+
156+
long gcCountBefore = collector.getCollectionCount();
157+
wb.fullGC();
158+
long gcCountAfter = collector.getCollectionCount();
159+
160+
Asserts.assertLessThan(gcCountBefore, gcCountAfter, "Triggered more Full GCs than expected");
161+
}
162+
}
163+
}
164+
165+
static public Map<Integer,String> largeMap;
166+
167+
public static void main(String... args) throws Exception {
168+
if (args.length < 4) {
169+
println("usage: JNICriticalStressTest <duration sec> <alloc threads> <jni critical threads> <gc name>");
170+
exit(-1);
171+
}
172+
173+
long durationSec = Long.parseLong(args[0]);
174+
int allocThreadNum = Integer.parseInt(args[1]);
175+
int jniCriticalThreadNum = Integer.parseInt(args[2]);
176+
177+
StringBuilder OldGCName = new StringBuilder();
178+
switch (args[3]) {
179+
case "G1":
180+
OldGCName.append("G1 Old Generation");
181+
break;
182+
case "Parallel":
183+
OldGCName.append("PS MarkSweep");
184+
break;
185+
case "Serial":
186+
OldGCName.append("MarkSweepCompact");
187+
break;
188+
default:
189+
throw new RuntimeException("Unsupported GC selected");
190+
}
191+
192+
List<GarbageCollectorMXBean> collectors = ManagementFactory.getGarbageCollectorMXBeans();
193+
194+
for (int i = 0; i < collectors.size(); i++) {
195+
if (collectors.get(i).getName().contains(OldGCName.toString())) {
196+
collector = collectors.get(i);
197+
break;
198+
}
199+
}
200+
201+
if (collector == null) {
202+
throw new RuntimeException(OldGCName.toString() + " not found");
203+
}
204+
205+
println("Running for " + durationSec + " secs");
206+
207+
largeMap = populateMap(LARGE_MAP_SIZE);
208+
209+
// Start threads to allocate memory, this will trigger both GCLocker initiated
210+
// garbage collections (GCs) and regular GCs. Thus increasing the likelihood of
211+
// having different types of GCs happening concurrently with the System.gc call.
212+
println("Starting " + allocThreadNum + " allocating threads");
213+
for (int i = 0; i < allocThreadNum; i += 1) {
214+
new Thread(new AllocatingWorker()).start();
215+
}
216+
217+
println("Starting " + jniCriticalThreadNum + " jni critical threads");
218+
for (int i = 0; i < jniCriticalThreadNum; i += 1) {
219+
new Thread(new JNICriticalWorker()).start();
220+
}
221+
222+
new Thread(new SystemGCWorker()).start();
223+
224+
long durationMS = (long) (1000 * durationSec);
225+
try {
226+
Thread.sleep(durationMS);
227+
} catch (InterruptedException e) {
228+
e.printStackTrace();
229+
exit(-1);
230+
}
231+
}
232+
}

0 commit comments

Comments
 (0)
Please sign in to comment.