Skip to content

Commit f393975

Browse files
committedJul 3, 2023
8310743: assert(reserved_rgn != nullptr) failed: Add committed region, No reserved region found
Reviewed-by: stefank, ayang
1 parent ba974d5 commit f393975

12 files changed

+445
-48
lines changed
 

‎src/hotspot/share/gc/z/zGlobals.hpp

+3
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,9 @@ const size_t ZGranuleSize = (size_t)1 << ZGranuleSizeShift
3838
// Virtual memory to physical memory ratio
3939
const size_t ZVirtualToPhysicalRatio = 16; // 16:1
4040

41+
// Max virtual memory ranges
42+
const size_t ZMaxVirtualReservations = 100; // Each reservation at least 1% of total
43+
4144
// Page size shifts
4245
const size_t ZPageSizeSmallShift = ZGranuleSizeShift;
4346
extern size_t ZPageSizeMediumShift;

‎src/hotspot/share/gc/z/zMemory.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,10 @@ ZMemoryManager::ZMemoryManager()
8181
: _freelist(),
8282
_callbacks() {}
8383

84+
bool ZMemoryManager::free_is_contiguous() const {
85+
return _freelist.size() == 1;
86+
}
87+
8488
void ZMemoryManager::register_callbacks(const Callbacks& callbacks) {
8589
_callbacks = callbacks;
8690
}

‎src/hotspot/share/gc/z/zMemory.hpp

+2
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,8 @@ class ZMemoryManager {
8181
public:
8282
ZMemoryManager();
8383

84+
bool free_is_contiguous() const;
85+
8486
void register_callbacks(const Callbacks& callbacks);
8587

8688
zoffset peek_low_address() const;

‎src/hotspot/share/gc/z/zNMT.cpp

+121
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,121 @@
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+
#include "precompiled.hpp"
25+
#include "gc/z/zAddress.inline.hpp"
26+
#include "gc/z/zGlobals.hpp"
27+
#include "gc/z/zNMT.hpp"
28+
#include "gc/z/zVirtualMemory.hpp"
29+
#include "memory/allocation.hpp"
30+
#include "services/memTracker.hpp"
31+
#include "utilities/nativeCallStack.hpp"
32+
33+
ZNMT::Reservation ZNMT::_reservations[ZMaxVirtualReservations] = {};
34+
size_t ZNMT::_num_reservations = 0;
35+
36+
size_t ZNMT::reservation_index(zoffset offset, size_t* offset_in_reservation) {
37+
assert(_num_reservations > 0, "at least one reservation must exist");
38+
39+
size_t index = 0;
40+
*offset_in_reservation = untype(offset);
41+
for (; index < _num_reservations; ++index) {
42+
const size_t reservation_size = _reservations[index]._size;
43+
if (*offset_in_reservation < reservation_size) {
44+
break;
45+
}
46+
*offset_in_reservation -= reservation_size;
47+
}
48+
49+
assert(index != _num_reservations, "failed to find reservation index");
50+
return index;
51+
}
52+
53+
void ZNMT::process_fake_mapping(zoffset offset, size_t size, bool commit) {
54+
// In order to satisfy NTM's requirement of an 1:1 mapping between committed
55+
// and reserved addresses, a fake mapping from the offset into the reservation
56+
// is used.
57+
//
58+
// These mappings from
59+
// [offset, offset + size) -> {[virtual address range], ...}
60+
// are stable after the heap has been reserved. No commits proceed any
61+
// reservations. Committing and uncommitting the same [offset, offset + size)
62+
// range will result in same virtual memory ranges.
63+
64+
size_t left_to_process = size;
65+
size_t offset_in_reservation;
66+
for (size_t i = reservation_index(offset, &offset_in_reservation); i < _num_reservations; ++i) {
67+
const zaddress_unsafe reservation_start = _reservations[i]._start;
68+
const size_t reservation_size = _reservations[i]._size;
69+
const size_t sub_range_size = MIN2(left_to_process, reservation_size - offset_in_reservation);
70+
const uintptr_t sub_range_addr = untype(reservation_start) + offset_in_reservation;
71+
72+
// commit / uncommit memory
73+
if (commit) {
74+
MemTracker::record_virtual_memory_commit((void*)sub_range_addr, sub_range_size, CALLER_PC);
75+
} else {
76+
if (MemTracker::enabled()) {
77+
Tracker tracker(Tracker::uncommit);
78+
tracker.record((address)sub_range_addr, sub_range_size);
79+
}
80+
}
81+
82+
left_to_process -= sub_range_size;
83+
if (left_to_process == 0) {
84+
// Processed all nmt registrations
85+
return;
86+
}
87+
88+
offset_in_reservation = 0;
89+
}
90+
91+
assert(left_to_process == 0, "everything was not commited");
92+
}
93+
94+
void ZNMT::reserve(zaddress_unsafe start, size_t size) {
95+
assert(_num_reservations < ZMaxVirtualReservations, "too many reservations");
96+
// Keep track of the reservations made in order to create fake mappings
97+
// between the reserved and commited memory.
98+
// See details in ZNMT::process_fake_mapping
99+
_reservations[_num_reservations++] = {start, size};
100+
101+
MemTracker::record_virtual_memory_reserve((void*)untype(start), size, CALLER_PC, mtJavaHeap);
102+
}
103+
104+
void ZNMT::commit(zoffset offset, size_t size) {
105+
// NMT expects a 1-to-1 mapping between virtual and physical memory.
106+
// ZGC can temporarily have multiple virtual addresses pointing to
107+
// the same physical memory.
108+
//
109+
// When this function is called we don't know where in the virtual memory
110+
// this physical memory will be mapped. So we fake the virtual memory
111+
// address by mapping the physical offset into offsets in the reserved
112+
// memory space.
113+
process_fake_mapping(offset, size, true);
114+
}
115+
116+
void ZNMT::uncommit(zoffset offset, size_t size) {
117+
// We fake the virtual memory address by mapping the physical offset
118+
// into offsets in the reserved memory space.
119+
// See comment in ZNMT::commit
120+
process_fake_mapping(offset, size, false);
121+
}

‎src/hotspot/share/gc/z/zNMT.hpp

+53
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
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+
#ifndef SHARE_GC_Z_ZNMT_HPP
25+
#define SHARE_GC_Z_ZNMT_HPP
26+
27+
#include "gc/z/zAddress.hpp"
28+
#include "gc/z/zGlobals.hpp"
29+
#include "gc/z/zMemory.hpp"
30+
#include "gc/z/zVirtualMemory.hpp"
31+
#include "memory/allStatic.hpp"
32+
#include "utilities/globalDefinitions.hpp"
33+
#include "utilities/nativeCallStack.hpp"
34+
35+
class ZNMT : public AllStatic {
36+
private:
37+
struct Reservation {
38+
zaddress_unsafe _start;
39+
size_t _size;
40+
};
41+
static Reservation _reservations[ZMaxVirtualReservations];
42+
static size_t _num_reservations;
43+
44+
static size_t reservation_index(zoffset offset, size_t* offset_in_reservation);
45+
static void process_fake_mapping(zoffset offset, size_t size, bool commit);
46+
47+
public:
48+
static void reserve(zaddress_unsafe start, size_t size);
49+
static void commit(zoffset offset, size_t size);
50+
static void uncommit(zoffset offset, size_t size);
51+
};
52+
53+
#endif // SHARE_GC_Z_ZNMT_HPP

‎src/hotspot/share/gc/z/zPhysicalMemory.cpp

+3-23
Original file line numberDiff line numberDiff line change
@@ -28,14 +28,14 @@
2828
#include "gc/z/zGlobals.hpp"
2929
#include "gc/z/zLargePages.inline.hpp"
3030
#include "gc/z/zList.inline.hpp"
31+
#include "gc/z/zNMT.hpp"
3132
#include "gc/z/zNUMA.inline.hpp"
3233
#include "gc/z/zPhysicalMemory.inline.hpp"
3334
#include "logging/log.hpp"
3435
#include "runtime/globals.hpp"
3536
#include "runtime/globals_extension.hpp"
3637
#include "runtime/init.hpp"
3738
#include "runtime/os.hpp"
38-
#include "services/memTracker.hpp"
3939
#include "utilities/align.hpp"
4040
#include "utilities/debug.hpp"
4141
#include "utilities/globalDefinitions.hpp"
@@ -276,26 +276,6 @@ void ZPhysicalMemoryManager::try_enable_uncommit(size_t min_capacity, size_t max
276276
log_info_p(gc, init)("Uncommit Delay: " UINTX_FORMAT "s", ZUncommitDelay);
277277
}
278278

279-
void ZPhysicalMemoryManager::nmt_commit(zoffset offset, size_t size) const {
280-
// NMT expects a 1-to-1 mapping between virtual and physical memory.
281-
// ZGC can temporarily have multiple virtual addresses pointing to
282-
// the same physical memory.
283-
//
284-
// When this function is called we don't know where in the virtual memory
285-
// this physical memory will be mapped. So we fake that the virtual memory
286-
// address is the heap base + the given offset.
287-
const uintptr_t addr = ZAddressHeapBase + untype(offset);
288-
MemTracker::record_virtual_memory_commit((void*)addr, size, CALLER_PC);
289-
}
290-
291-
void ZPhysicalMemoryManager::nmt_uncommit(zoffset offset, size_t size) const {
292-
if (MemTracker::enabled()) {
293-
const uintptr_t addr = ZAddressHeapBase + untype(offset);
294-
Tracker tracker(Tracker::uncommit);
295-
tracker.record((address)addr, size);
296-
}
297-
}
298-
299279
void ZPhysicalMemoryManager::alloc(ZPhysicalMemory& pmem, size_t size) {
300280
assert(is_aligned(size, ZGranuleSize), "Invalid size");
301281

@@ -330,7 +310,7 @@ bool ZPhysicalMemoryManager::commit(ZPhysicalMemory& pmem) {
330310
const size_t committed = _backing.commit(segment.start(), segment.size());
331311

332312
// Register with NMT
333-
nmt_commit(segment.start(), committed);
313+
ZNMT::commit(segment.start(), committed);
334314

335315
// Register committed segment
336316
if (!pmem.commit_segment(i, committed)) {
@@ -356,7 +336,7 @@ bool ZPhysicalMemoryManager::uncommit(ZPhysicalMemory& pmem) {
356336
const size_t uncommitted = _backing.uncommit(segment.start(), segment.size());
357337

358338
// Unregister with NMT
359-
nmt_uncommit(segment.start(), uncommitted);
339+
ZNMT::uncommit(segment.start(), uncommitted);
360340

361341
// Deregister uncommitted segment
362342
if (!pmem.uncommit_segment(i, uncommitted)) {

‎src/hotspot/share/gc/z/zPhysicalMemory.hpp

-3
Original file line numberDiff line numberDiff line change
@@ -84,9 +84,6 @@ class ZPhysicalMemoryManager {
8484
ZPhysicalMemoryBacking _backing;
8585
ZMemoryManager _manager;
8686

87-
void nmt_commit(zoffset offset, size_t size) const;
88-
void nmt_uncommit(zoffset offset, size_t size) const;
89-
9087
void pretouch_view(zaddress addr, size_t size) const;
9188
void map_view(zaddress_unsafe addr, const ZPhysicalMemory& pmem) const;
9289
void unmap_view(zaddress_unsafe addr, size_t size) const;

‎src/hotspot/share/gc/z/zVirtualMemory.cpp

+67-19
Original file line numberDiff line numberDiff line change
@@ -22,12 +22,13 @@
2222
*/
2323

2424
#include "precompiled.hpp"
25+
#include "gc/shared/gc_globals.hpp"
2526
#include "gc/shared/gcLogPrecious.hpp"
2627
#include "gc/z/zAddress.inline.hpp"
2728
#include "gc/z/zAddressSpaceLimit.hpp"
2829
#include "gc/z/zGlobals.hpp"
30+
#include "gc/z/zNMT.hpp"
2931
#include "gc/z/zVirtualMemory.inline.hpp"
30-
#include "services/memTracker.hpp"
3132
#include "utilities/align.hpp"
3233
#include "utilities/debug.hpp"
3334

@@ -54,6 +55,39 @@ ZVirtualMemoryManager::ZVirtualMemoryManager(size_t max_capacity)
5455
_initialized = true;
5556
}
5657

58+
#ifdef ASSERT
59+
size_t ZVirtualMemoryManager::force_reserve_discontiguous(size_t size) {
60+
const size_t min_range = calculate_min_range(size);
61+
const size_t max_range = MAX2(align_down(size / ZForceDiscontiguousHeapReservations, ZGranuleSize), min_range);
62+
size_t reserved = 0;
63+
64+
// Try to reserve ZForceDiscontiguousHeapReservations number of virtual memory
65+
// ranges. Starting with higher addresses.
66+
uintptr_t end = ZAddressOffsetMax;
67+
while (reserved < size && end >= max_range) {
68+
const size_t remaining = size - reserved;
69+
const size_t reserve_size = MIN2(max_range, remaining);
70+
const uintptr_t reserve_start = end - reserve_size;
71+
72+
if (reserve_contiguous(to_zoffset(reserve_start), reserve_size)) {
73+
reserved += reserve_size;
74+
}
75+
76+
end -= reserve_size * 2;
77+
}
78+
79+
// If (reserved < size) attempt to reserve the rest via normal divide and conquer
80+
uintptr_t start = 0;
81+
while (reserved < size && start < ZAddressOffsetMax) {
82+
const size_t remaining = MIN2(size - reserved, ZAddressOffsetMax - start);
83+
reserved += reserve_discontiguous(to_zoffset(start), remaining, min_range);
84+
start += remaining;
85+
}
86+
87+
return reserved;
88+
}
89+
#endif
90+
5791
size_t ZVirtualMemoryManager::reserve_discontiguous(zoffset start, size_t size, size_t min_range) {
5892
if (size < min_range) {
5993
// Too small
@@ -75,15 +109,20 @@ size_t ZVirtualMemoryManager::reserve_discontiguous(zoffset start, size_t size,
75109
// Divide and conquer
76110
const size_t first_part = align_down(half, ZGranuleSize);
77111
const size_t second_part = size - first_part;
78-
return reserve_discontiguous(start, first_part, min_range) +
79-
reserve_discontiguous(start + first_part, second_part, min_range);
112+
const size_t first_size = reserve_discontiguous(start, first_part, min_range);
113+
const size_t second_size = reserve_discontiguous(start + first_part, second_part, min_range);
114+
return first_size + second_size;
80115
}
81116

82-
size_t ZVirtualMemoryManager::reserve_discontiguous(size_t size) {
117+
size_t ZVirtualMemoryManager::calculate_min_range(size_t size) {
83118
// Don't try to reserve address ranges smaller than 1% of the requested size.
84119
// This avoids an explosion of reservation attempts in case large parts of the
85120
// address space is already occupied.
86-
const size_t min_range = align_up(size / 100, ZGranuleSize);
121+
return align_up(size / ZMaxVirtualReservations, ZGranuleSize);
122+
}
123+
124+
size_t ZVirtualMemoryManager::reserve_discontiguous(size_t size) {
125+
const size_t min_range = calculate_min_range(size);
87126
uintptr_t start = 0;
88127
size_t reserved = 0;
89128

@@ -98,7 +137,7 @@ size_t ZVirtualMemoryManager::reserve_discontiguous(size_t size) {
98137
}
99138

100139
bool ZVirtualMemoryManager::reserve_contiguous(zoffset start, size_t size) {
101-
assert(is_aligned(size, ZGranuleSize), "Must be granule aligned");
140+
assert(is_aligned(size, ZGranuleSize), "Must be granule aligned " SIZE_FORMAT_X, size);
102141

103142
// Reserve address views
104143
const zaddress_unsafe addr = ZOffset::address_unsafe(start);
@@ -109,7 +148,7 @@ bool ZVirtualMemoryManager::reserve_contiguous(zoffset start, size_t size) {
109148
}
110149

111150
// Register address views with native memory tracker
112-
nmt_reserve(addr, size);
151+
ZNMT::reserve(addr, size);
113152

114153
// Make the address range free
115154
_manager.free(start, size);
@@ -137,15 +176,25 @@ bool ZVirtualMemoryManager::reserve(size_t max_capacity) {
137176
const size_t limit = MIN2(ZAddressOffsetMax, ZAddressSpaceLimit::heap());
138177
const size_t size = MIN2(max_capacity * ZVirtualToPhysicalRatio, limit);
139178

140-
size_t reserved = size;
141-
bool contiguous = true;
179+
auto do_reserve = [&]() {
180+
#ifdef ASSERT
181+
if (ZForceDiscontiguousHeapReservations > 0) {
182+
return force_reserve_discontiguous(size);
183+
}
184+
#endif
185+
186+
// Prefer a contiguous address space
187+
if (reserve_contiguous(size)) {
188+
return size;
189+
}
142190

143-
// Prefer a contiguous address space
144-
if (!reserve_contiguous(size)) {
145191
// Fall back to a discontiguous address space
146-
reserved = reserve_discontiguous(size);
147-
contiguous = false;
148-
}
192+
return reserve_discontiguous(size);
193+
};
194+
195+
const size_t reserved = do_reserve();
196+
197+
const bool contiguous = _manager.free_is_contiguous();
149198

150199
log_info_p(gc, init)("Address Space Type: %s/%s/%s",
151200
(contiguous ? "Contiguous" : "Discontiguous"),
@@ -159,11 +208,6 @@ bool ZVirtualMemoryManager::reserve(size_t max_capacity) {
159208
return reserved >= max_capacity;
160209
}
161210

162-
void ZVirtualMemoryManager::nmt_reserve(zaddress_unsafe start, size_t size) {
163-
MemTracker::record_virtual_memory_reserve((void*)untype(start), size, CALLER_PC);
164-
MemTracker::record_virtual_memory_type((void*)untype(start), mtJavaHeap);
165-
}
166-
167211
bool ZVirtualMemoryManager::is_initialized() const {
168212
return _initialized;
169213
}
@@ -179,6 +223,10 @@ ZVirtualMemory ZVirtualMemoryManager::alloc(size_t size, bool force_low_address)
179223
start = _manager.alloc_high_address(size);
180224
}
181225

226+
if (start == zoffset(UINTPTR_MAX)) {
227+
return ZVirtualMemory();
228+
}
229+
182230
return ZVirtualMemory(start, size);
183231
}
184232

‎src/hotspot/share/gc/z/zVirtualMemory.hpp

+4-1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
#ifndef SHARE_GC_Z_ZVIRTUALMEMORY_HPP
2525
#define SHARE_GC_Z_ZVIRTUALMEMORY_HPP
2626

27+
#include "gc/z/zAddress.hpp"
2728
#include "gc/z/zMemory.hpp"
2829

2930
class ZVirtualMemory {
@@ -47,6 +48,8 @@ class ZVirtualMemory {
4748

4849
class ZVirtualMemoryManager {
4950
private:
51+
static size_t calculate_min_range(size_t size);
52+
5053
ZMemoryManager _manager;
5154
size_t _reserved;
5255
bool _initialized;
@@ -63,7 +66,7 @@ class ZVirtualMemoryManager {
6366
size_t reserve_discontiguous(size_t size);
6467
bool reserve(size_t max_capacity);
6568

66-
void nmt_reserve(zaddress_unsafe start, size_t size);
69+
DEBUG_ONLY(size_t force_reserve_discontiguous(size_t size);)
6770

6871
public:
6972
ZVirtualMemoryManager(size_t max_capacity);

‎src/hotspot/share/gc/z/z_globals.hpp

+13-2
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,8 @@
2424
#ifndef SHARE_GC_Z_Z_GLOBALS_HPP
2525
#define SHARE_GC_Z_Z_GLOBALS_HPP
2626

27-
#include "zPageAge.hpp"
27+
#include "gc/z/zGlobals.hpp"
28+
#include "gc/z/zPageAge.hpp"
2829

2930
#define GC_Z_FLAGS(develop, \
3031
develop_pd, \
@@ -68,7 +69,17 @@
6869
\
6970
product(int, ZTenuringThreshold, -1, DIAGNOSTIC, \
7071
"Young generation tenuring threshold, -1 for dynamic computation")\
71-
range(-1, static_cast<int>(ZPageAgeMax))
72+
range(-1, static_cast<int>(ZPageAgeMax)) \
73+
\
74+
develop(size_t, ZForceDiscontiguousHeapReservations, 0, \
75+
"The gc will attempt to split the heap reservation into this " \
76+
"many reservations, subject to available virtual address space " \
77+
"and invariant restrictions. Higher virtual addresses are " \
78+
"preferred " \
79+
"0: Disabled " \
80+
"1: Attempt contiguous reservation starting at a higher address " \
81+
"N: Force that many reservations, if possible") \
82+
range(0, ZMaxVirtualReservations)
7283

7384
// end of GC_Z_FLAGS
7485

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
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+
package gc.z;
25+
26+
/**
27+
* @test TestZForceDiscontiguousHeapReservations
28+
* @requires vm.gc.ZGenerational & vm.debug
29+
* @summary Test the ZForceDiscontiguousHeapReservations development flag
30+
* @library /test/lib
31+
* @run driver gc.z.TestZForceDiscontiguousHeapReservations
32+
*/
33+
34+
import jdk.test.lib.process.OutputAnalyzer;
35+
import jdk.test.lib.process.ProcessTools;
36+
37+
public class TestZForceDiscontiguousHeapReservations {
38+
39+
private static void testValue(int n) throws Exception {
40+
/**
41+
* Xmx is picked so that it is divisible by 'ZForceDiscontiguousHeapReservations * ZGranuleSize'
42+
* Xms is picked so that it is less than '16 * Xmx / ZForceDiscontiguousHeapReservations' as ZGC
43+
* cannot currently handle a discontiguous heap with an initial size larger than the individual
44+
* reservations.
45+
*/
46+
final int XmxInM = 2000;
47+
final int XmsInM = Math.min(16 * XmxInM / (n + 1), XmxInM);
48+
OutputAnalyzer oa = ProcessTools.executeProcess(ProcessTools.createTestJvm(
49+
"-XX:+UseZGC",
50+
"-XX:+ZGenerational",
51+
"-Xms" + XmsInM + "M",
52+
"-Xmx" + XmxInM + "M",
53+
"-Xlog:gc,gc+init",
54+
"-XX:ZForceDiscontiguousHeapReservations=" + n,
55+
"-version"))
56+
.outputTo(System.out)
57+
.errorTo(System.out)
58+
.shouldHaveExitValue(0);
59+
if (n > 1) {
60+
oa.shouldContain("Address Space Type: Discontiguous");
61+
}
62+
}
63+
64+
public static void main(String[] args) throws Exception {
65+
testValue(0);
66+
testValue(1);
67+
testValue(2);
68+
testValue(100);
69+
}
70+
}

‎test/hotspot/jtreg/gc/z/TestZNMT.java

+105
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,105 @@
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+
package gc.z;
25+
26+
/**
27+
* @test TestZNMT
28+
* @bug 8310743
29+
* @requires vm.gc.ZGenerational & vm.debug
30+
* @summary Test NMT and ZGenerational heap reservation / commits interactions.
31+
* @library / /test/lib
32+
* @run driver gc.z.TestZNMT
33+
*/
34+
35+
import static gc.testlibrary.Allocation.blackHole;
36+
import java.util.ArrayList;
37+
import jdk.test.lib.process.OutputAnalyzer;
38+
import jdk.test.lib.process.ProcessTools;
39+
40+
public class TestZNMT {
41+
private static final int XmxInM = 2000;
42+
static class Test {
43+
private static final int K = 1024;
44+
private static final int M = K * K;
45+
public static void main(String[] args) throws Exception {
46+
final int zForceDiscontiguousHeapReservations = Integer.parseInt(args[0]);
47+
final int XmsInM = Integer.parseInt(args[1]);
48+
// 75% of the largest allocation that fits within one reservation
49+
// (or Xmx / zForceDiscontiguousHeapReservations), whichever is smallest
50+
final int allocationInM = (int)(Math.min(zForceDiscontiguousHeapReservations == 0
51+
? XmxInM
52+
: XmxInM / zForceDiscontiguousHeapReservations,
53+
XmsInM) * 0.75);
54+
ArrayList<byte[]> list = new ArrayList<>(zForceDiscontiguousHeapReservations);
55+
for (int i = 0; i < zForceDiscontiguousHeapReservations; i++) {
56+
list.add(new byte[allocationInM * M]);
57+
}
58+
blackHole(list);
59+
}
60+
}
61+
62+
63+
private static void testValue(int zForceDiscontiguousHeapReservations) throws Exception {
64+
/**
65+
* Xmx is picked so that it is divisible by 'ZForceDiscontiguousHeapReservations * ZGranuleSize'
66+
* Xms is picked so that it is less than '16 * Xmx / ZForceDiscontiguousHeapReservations' as ZGC
67+
* cannot currently handle a discontiguous heap with an initial size larger than the individual
68+
* reservations.
69+
*/
70+
final int XmsInM = Math.min(16 * XmxInM / (zForceDiscontiguousHeapReservations + 1), XmxInM);
71+
OutputAnalyzer oa = ProcessTools.executeProcess(ProcessTools.createTestJvm(
72+
"-XX:+UseZGC",
73+
"-XX:+ZGenerational",
74+
"-Xms" + XmsInM + "M",
75+
"-Xmx" + XmxInM + "M",
76+
"-Xlog:gc,gc+init",
77+
"-XX:ZForceDiscontiguousHeapReservations=" + zForceDiscontiguousHeapReservations,
78+
"-XX:NativeMemoryTracking=detail",
79+
"-XX:+PrintNMTStatistics",
80+
Test.class.getName(),
81+
Integer.toString(zForceDiscontiguousHeapReservations),
82+
Integer.toString(XmxInM)))
83+
.outputTo(System.out)
84+
.errorTo(System.out)
85+
.shouldHaveExitValue(0);
86+
if (zForceDiscontiguousHeapReservations > 1) {
87+
oa.shouldContain("Address Space Type: Discontiguous");
88+
}
89+
90+
if (XmsInM < XmxInM) {
91+
// There will be reservations which are smaller than the total
92+
// memory allocated in TestZNMT.Test.main. This means that some
93+
// reservation will be completely committed and print the following
94+
// in the NMT statistics.
95+
oa.shouldMatch("reserved and committed \\d+ for Java Heap");
96+
}
97+
}
98+
99+
public static void main(String[] args) throws Exception {
100+
testValue(0);
101+
testValue(1);
102+
testValue(2);
103+
testValue(100);
104+
}
105+
}

3 commit comments

Comments
 (3)

openjdk-notifier[bot] commented on Jul 3, 2023

@openjdk-notifier[bot]

xmas92 commented on Jul 3, 2023

@xmas92
MemberAuthor

/backport jdk21

openjdk[bot] commented on Jul 3, 2023

@openjdk[bot]

@xmas92 Could not automatically backport f393975d to openjdk/jdk21 due to conflicts in the following files:

  • src/hotspot/share/gc/z/zPhysicalMemory.cpp

Please fetch the appropriate branch/commit and manually resolve these conflicts by using the following commands in your personal fork of openjdk/jdk21. Note: these commands are just some suggestions and you can use other equivalent commands you know.

# Fetch the up-to-date version of the target branch
$ git fetch --no-tags https://git.openjdk.org/jdk21.git master:master

# Check out the target branch and create your own branch to backport
$ git checkout master
$ git checkout -b xmas92-backport-f393975d

# Fetch the commit you want to backport
$ git fetch --no-tags https://git.openjdk.org/jdk.git f393975d1b9423835a7aa42f6ffc0656499701d7

# Backport the commit
$ git cherry-pick --no-commit f393975d1b9423835a7aa42f6ffc0656499701d7
# Resolve conflicts now

# Commit the files you have modified
$ git add files/with/resolved/conflicts
$ git commit -m 'Backport f393975d1b9423835a7aa42f6ffc0656499701d7'

Once you have resolved the conflicts as explained above continue with creating a pull request towards the openjdk/jdk21 with the title Backport f393975d1b9423835a7aa42f6ffc0656499701d7.

Please sign in to comment.