Skip to content

Commit 945f408

Browse files
committedNov 9, 2021
8276098: Do precise BOT updates in G1 evacuation phase
Reviewed-by: tschatzl, ayang
1 parent 8747882 commit 945f408

15 files changed

+192
-82
lines changed
 

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

+4-7
Original file line numberDiff line numberDiff line change
@@ -43,10 +43,7 @@ void G1AllocRegion::setup(G1CollectedHeap* g1h, HeapRegion* dummy_region) {
4343

4444
// Make sure that any allocation attempt on this region will fail
4545
// and will not trigger any asserts.
46-
assert(dummy_region->allocate_no_bot_updates(1) == NULL, "should fail");
47-
assert(dummy_region->allocate(1) == NULL, "should fail");
4846
DEBUG_ONLY(size_t assert_tmp);
49-
assert(dummy_region->par_allocate_no_bot_updates(1, 1, &assert_tmp) == NULL, "should fail");
5047
assert(dummy_region->par_allocate(1, 1, &assert_tmp) == NULL, "should fail");
5148

5249
_g1h = g1h;
@@ -77,8 +74,9 @@ size_t G1AllocRegion::fill_up_remaining_space(HeapRegion* alloc_region) {
7774
while (free_word_size >= min_word_size_to_fill) {
7875
HeapWord* dummy = par_allocate(alloc_region, free_word_size);
7976
if (dummy != NULL) {
80-
// If the allocation was successful we should fill in the space.
81-
CollectedHeap::fill_with_object(dummy, free_word_size);
77+
// If the allocation was successful we should fill in the space. If the
78+
// allocation was in old any necessary BOT updates will be done.
79+
alloc_region->fill_with_dummy_object(dummy, free_word_size);
8280
alloc_region->set_pre_dummy_top(dummy);
8381
result += free_word_size * HeapWordSize;
8482
break;
@@ -256,7 +254,6 @@ G1AllocRegion::G1AllocRegion(const char* name,
256254
: _alloc_region(NULL),
257255
_count(0),
258256
_used_bytes_before(0),
259-
_bot_updates(bot_updates),
260257
_name(name),
261258
_node_index(node_index)
262259
{ }
@@ -389,7 +386,7 @@ HeapRegion* OldGCAllocRegion::release() {
389386
// original problem cannot occur.
390387
if (to_allocate_words >= G1CollectedHeap::min_fill_size()) {
391388
HeapWord* dummy = attempt_allocation(to_allocate_words);
392-
CollectedHeap::fill_with_object(dummy, to_allocate_words);
389+
cur->fill_with_dummy_object(dummy, to_allocate_words);
393390
}
394391
}
395392
}

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

-3
Original file line numberDiff line numberDiff line change
@@ -68,9 +68,6 @@ class G1AllocRegion : public CHeapObj<mtGC> {
6868
// we allocated in it.
6969
size_t _used_bytes_before;
7070

71-
// When true, indicates that allocate calls should do BOT updates.
72-
const bool _bot_updates;
73-
7471
// Useful for debugging and tracing.
7572
const char* _name;
7673

‎src/hotspot/share/gc/g1/g1AllocRegion.inline.hpp

+5-13
Original file line numberDiff line numberDiff line change
@@ -31,9 +31,9 @@
3131

3232
#define assert_alloc_region(p, message) \
3333
do { \
34-
assert((p), "[%s] %s c: %u b: %s r: " PTR_FORMAT " u: " SIZE_FORMAT, \
35-
_name, (message), _count, BOOL_TO_STR(_bot_updates), \
36-
p2i(_alloc_region), _used_bytes_before); \
34+
assert((p), "[%s] %s c: %u r: " PTR_FORMAT " u: " SIZE_FORMAT, \
35+
_name, (message), _count, p2i(_alloc_region), \
36+
_used_bytes_before); \
3737
} while (0)
3838

3939

@@ -45,11 +45,7 @@ inline HeapWord* G1AllocRegion::allocate(HeapRegion* alloc_region,
4545
size_t word_size) {
4646
assert(alloc_region != NULL, "pre-condition");
4747

48-
if (!_bot_updates) {
49-
return alloc_region->allocate_no_bot_updates(word_size);
50-
} else {
51-
return alloc_region->allocate(word_size);
52-
}
48+
return alloc_region->allocate(word_size);
5349
}
5450

5551
inline HeapWord* G1AllocRegion::par_allocate(HeapRegion* alloc_region, size_t word_size) {
@@ -64,11 +60,7 @@ inline HeapWord* G1AllocRegion::par_allocate(HeapRegion* alloc_region,
6460
assert(alloc_region != NULL, "pre-condition");
6561
assert(!alloc_region->is_empty(), "pre-condition");
6662

67-
if (!_bot_updates) {
68-
return alloc_region->par_allocate_no_bot_updates(min_word_size, desired_word_size, actual_word_size);
69-
} else {
70-
return alloc_region->par_allocate(min_word_size, desired_word_size, actual_word_size);
71-
}
63+
return alloc_region->par_allocate(min_word_size, desired_word_size, actual_word_size);
7264
}
7365

7466
inline HeapWord* G1AllocRegion::attempt_allocation(size_t word_size) {

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

+6-1
Original file line numberDiff line numberDiff line change
@@ -299,8 +299,12 @@ G1PLABAllocator::G1PLABAllocator(G1Allocator* allocator) :
299299
_direct_allocated[state] = 0;
300300
uint length = alloc_buffers_length(state);
301301
_alloc_buffers[state] = NEW_C_HEAP_ARRAY(PLAB*, length, mtGC);
302+
size_t word_sz = _g1h->desired_plab_sz(state);
302303
for (uint node_index = 0; node_index < length; node_index++) {
303-
_alloc_buffers[state][node_index] = new PLAB(_g1h->desired_plab_sz(state));
304+
// Specialized PLABs for old that handle BOT updates for object allocations.
305+
_alloc_buffers[state][node_index] = (state == G1HeapRegionAttr::Old)
306+
? new G1BotUpdatingPLAB(word_sz)
307+
: new PLAB(word_sz);
304308
}
305309
}
306310
}
@@ -360,6 +364,7 @@ HeapWord* G1PLABAllocator::allocate_direct_or_new_plab(G1HeapRegionAttr dest,
360364
// Try direct allocation.
361365
HeapWord* result = _allocator->par_allocate_during_gc(dest, word_sz, node_index);
362366
if (result != NULL) {
367+
update_bot_for_direct_allocation(dest, result, word_sz);
363368
_direct_allocated[dest.type()] += word_sz;
364369
}
365370
return result;

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

+26
Original file line numberDiff line numberDiff line change
@@ -145,6 +145,22 @@ class G1Allocator : public CHeapObj<mtGC> {
145145
uint node_index);
146146
};
147147

148+
// Specialized PLAB for old generation promotions. For old regions the
149+
// BOT needs to be updated and the relevant data to do this efficiently
150+
// is stored in the PLAB.
151+
class G1BotUpdatingPLAB : public PLAB {
152+
// An object spanning this threshold will cause a BOT update.
153+
HeapWord* _next_bot_threshold;
154+
// The region in which the PLAB resides.
155+
HeapRegion* _region;
156+
public:
157+
G1BotUpdatingPLAB(size_t word_sz) : PLAB(word_sz) { }
158+
// Sets the new PLAB buffer as well as updates the threshold and region.
159+
void set_buf(HeapWord* buf, size_t word_sz) override;
160+
// Updates the BOT if the last allocation crossed the threshold.
161+
inline void update_bot(size_t word_sz);
162+
};
163+
148164
// Manages the PLABs used during garbage collection. Interface for allocation from PLABs.
149165
// Needs to handle multiple contexts, extra alignment in any "survivor" area and some
150166
// statistics.
@@ -165,11 +181,18 @@ class G1PLABAllocator : public CHeapObj<mtGC> {
165181
inline PLAB* alloc_buffer(G1HeapRegionAttr dest, uint node_index) const;
166182
inline PLAB* alloc_buffer(region_type_t dest, uint node_index) const;
167183

184+
// Helpers to do explicit BOT updates for allocations in old generation regions.
185+
void update_bot_for_direct_allocation(G1HeapRegionAttr attr, HeapWord* addr, size_t size);
186+
168187
// Returns the number of allocation buffers for the given dest.
169188
// There is only 1 buffer for Old while Young may have multiple buffers depending on
170189
// active NUMA nodes.
171190
inline uint alloc_buffers_length(region_type_t dest) const;
172191

192+
// Returns if BOT updates are needed for the given destinaion. Currently we only have
193+
// two destinations and BOT updates are only needed for the old generation.
194+
inline bool needs_bot_update(G1HeapRegionAttr dest) const;
195+
173196
bool may_throw_away_buffer(size_t const allocation_word_sz, size_t const buffer_size) const;
174197
public:
175198
G1PLABAllocator(G1Allocator* allocator);
@@ -198,6 +221,9 @@ class G1PLABAllocator : public CHeapObj<mtGC> {
198221
bool* refill_failed,
199222
uint node_index);
200223

224+
// Update the BOT for the last PLAB allocation.
225+
inline void update_bot_for_plab_allocation(G1HeapRegionAttr dest, size_t word_sz, uint node_index);
226+
201227
void undo_allocation(G1HeapRegionAttr dest, HeapWord* obj, size_t word_sz, uint node_index);
202228
};
203229

‎src/hotspot/share/gc/g1/g1Allocator.inline.hpp

+45
Original file line numberDiff line numberDiff line change
@@ -133,4 +133,49 @@ inline HeapWord* G1PLABAllocator::allocate(G1HeapRegionAttr dest,
133133
return allocate_direct_or_new_plab(dest, word_sz, refill_failed, node_index);
134134
}
135135

136+
inline bool G1PLABAllocator::needs_bot_update(G1HeapRegionAttr dest) const {
137+
return dest.is_old();
138+
}
139+
140+
inline void G1PLABAllocator::update_bot_for_direct_allocation(G1HeapRegionAttr attr, HeapWord* addr, size_t size) {
141+
if (!needs_bot_update(attr)) {
142+
return;
143+
}
144+
145+
// Out of PLAB allocations in an old generation region. Update BOT.
146+
HeapRegion* region = _g1h->heap_region_containing(addr);
147+
region->update_bot_at(addr, size);
148+
}
149+
150+
inline void G1PLABAllocator::update_bot_for_plab_allocation(G1HeapRegionAttr dest, size_t word_sz, uint node_index) {
151+
assert(needs_bot_update(dest), "Wrong destination: %s", dest.get_type_str());
152+
G1BotUpdatingPLAB* plab = static_cast<G1BotUpdatingPLAB*>(alloc_buffer(dest, node_index));
153+
plab->update_bot(word_sz);
154+
}
155+
156+
inline void G1BotUpdatingPLAB::set_buf(HeapWord* buf, size_t word_sz) {
157+
PLAB::set_buf(buf, word_sz);
158+
// Update the region and threshold to allow efficient BOT updates.
159+
_region = G1CollectedHeap::heap()->heap_region_containing(buf);
160+
_next_bot_threshold = _region->bot_threshold_for_addr(buf);
161+
}
162+
163+
inline void G1BotUpdatingPLAB::update_bot(size_t word_sz) {
164+
// The last object end is at _top, if it did not cross the
165+
// threshold, there is nothing to do.
166+
if (_top <= _next_bot_threshold) {
167+
return;
168+
}
169+
170+
HeapWord* obj_start = _top - word_sz;
171+
assert(contains(obj_start),
172+
"Object start outside PLAB. bottom: " PTR_FORMAT " object: " PTR_FORMAT,
173+
p2i(_bottom), p2i(obj_start));
174+
assert(obj_start <= _next_bot_threshold,
175+
"Object start not below or at threshold. threshold: " PTR_FORMAT " object: " PTR_FORMAT,
176+
p2i(_next_bot_threshold), p2i(obj_start));
177+
178+
_region->update_bot_crossing_threshold(&_next_bot_threshold, obj_start, _top);
179+
}
180+
136181
#endif // SHARE_GC_G1_G1ALLOCATOR_INLINE_HPP

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

+9
Original file line numberDiff line numberDiff line change
@@ -108,6 +108,7 @@ class G1BlockOffsetTable: public CHeapObj<mtGC> {
108108

109109
class G1BlockOffsetTablePart {
110110
friend class G1BlockOffsetTable;
111+
friend class HeapRegion;
111112
friend class VMStructs;
112113
private:
113114
// allocation boundary at which offset array must be updated
@@ -181,6 +182,9 @@ class G1BlockOffsetTablePart {
181182

182183
void verify() const;
183184

185+
// Given an address calculate where the next threshold needing an update is.
186+
inline HeapWord* threshold_for_addr(const void* addr);
187+
184188
// Returns the address of the start of the block containing "addr", or
185189
// else "null" if it is covered by no block. (May have side effects,
186190
// namely updating of shared array entries that "point" too far
@@ -204,6 +208,11 @@ class G1BlockOffsetTablePart {
204208
// updated.
205209
HeapWord* threshold() const { return _next_offset_threshold; }
206210

211+
// Sets the threshold explicitly to keep it consistent with what has been
212+
// updated. This needs to be done when the threshold is not used for updating
213+
// the bot, for example when promoting to old in young collections.
214+
void set_threshold(HeapWord* threshold) { _next_offset_threshold = threshold; }
215+
207216
// These must be guaranteed to work properly (i.e., do nothing)
208217
// when "blk_start" ("blk" for second version) is "NULL". In this
209218
// implementation, that's true because NULL is represented as 0, and thus

‎src/hotspot/share/gc/g1/g1BlockOffsetTable.inline.hpp

+14-3
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,20 @@
3131
#include "gc/shared/memset_with_concurrent_readers.hpp"
3232
#include "runtime/atomic.hpp"
3333

34+
inline HeapWord* G1BlockOffsetTablePart::threshold_for_addr(const void* addr) {
35+
assert(addr >= _hr->bottom() && addr < _hr->top(), "invalid address");
36+
size_t index = _bot->index_for(addr);
37+
HeapWord* card_boundary = _bot->address_for_index(index);
38+
// Address at card boundary, use as threshold.
39+
if (card_boundary == addr) {
40+
return card_boundary;
41+
}
42+
43+
// Calculate next threshold.
44+
HeapWord* threshold = card_boundary + BOTConstants::N_words;
45+
return threshold;
46+
}
47+
3448
inline HeapWord* G1BlockOffsetTablePart::block_start(const void* addr) {
3549
assert(addr >= _hr->bottom() && addr < _hr->top(), "invalid address");
3650
HeapWord* q = block_at_or_preceding(addr);
@@ -109,9 +123,6 @@ inline HeapWord* G1BlockOffsetTablePart::block_at_or_preceding(const void* addr)
109123
"Object crossed region boundary, found offset %u instead of 0",
110124
(uint) _bot->offset_array(_bot->index_for(_hr->bottom())));
111125

112-
// We must make sure that the offset table entry we use is valid.
113-
assert(addr < _next_offset_threshold, "Precondition");
114-
115126
size_t index = _bot->index_for(addr);
116127

117128
HeapWord* q = _bot->address_for_index(index);

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

+6
Original file line numberDiff line numberDiff line change
@@ -3293,6 +3293,7 @@ void G1CollectedHeap::retire_gc_alloc_region(HeapRegion* alloc_region,
32933293
_bytes_used_during_gc += allocated_bytes;
32943294
if (dest.is_old()) {
32953295
old_set_add(alloc_region);
3296+
alloc_region->update_bot_threshold();
32963297
} else {
32973298
assert(dest.is_young(), "Retiring alloc region should be young (%d)", dest.type());
32983299
_survivor.add_used_bytes(allocated_bytes);
@@ -3445,3 +3446,8 @@ GrowableArray<GCMemoryManager*> G1CollectedHeap::memory_managers() {
34453446
GrowableArray<MemoryPool*> G1CollectedHeap::memory_pools() {
34463447
return _monitoring_support->memory_pools();
34473448
}
3449+
3450+
void G1CollectedHeap::fill_with_dummy_object(HeapWord* start, HeapWord* end, bool zap) {
3451+
HeapRegion* region = heap_region_containing(start);
3452+
region->fill_with_dummy_object(start, pointer_delta(end, start), zap);
3453+
}

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

+2
Original file line numberDiff line numberDiff line change
@@ -934,6 +934,8 @@ class G1CollectedHeap : public CollectedHeap {
934934
virtual GrowableArray<GCMemoryManager*> memory_managers();
935935
virtual GrowableArray<MemoryPool*> memory_pools();
936936

937+
virtual void fill_with_dummy_object(HeapWord* start, HeapWord* end, bool zap);
938+
937939
// Try to minimize the remembered set.
938940
void scrub_rem_set();
939941

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

+5
Original file line numberDiff line numberDiff line change
@@ -498,6 +498,11 @@ oop G1ParScanThreadState::do_copy_to_survivor_space(G1HeapRegionAttr const regio
498498
obj->incr_age();
499499
}
500500
_age_table.add(age, word_sz);
501+
} else {
502+
// Currently we only have two destinations and we only need BOT updates for
503+
// old. If the current allocation was done outside the PLAB this call will
504+
// have no effect since the _top of the PLAB has not changed.
505+
_plab_allocator->update_bot_for_plab_allocation(dest_attr, word_sz, node_index);
501506
}
502507

503508
// Most objects are not arrays, so do one array check rather than

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

+9-1
Original file line numberDiff line numberDiff line change
@@ -235,7 +235,6 @@ HeapRegion::HeapRegion(uint hrm_index,
235235
_top(NULL),
236236
_compaction_top(NULL),
237237
_bot_part(bot, this),
238-
_par_alloc_lock(Mutex::service-2, "HeapRegionParAlloc_lock"),
239238
_pre_dummy_top(NULL),
240239
_rem_set(NULL),
241240
_hrm_index(hrm_index),
@@ -818,3 +817,12 @@ void HeapRegion::object_iterate(ObjectClosure* blk) {
818817
p += block_size(p);
819818
}
820819
}
820+
821+
void HeapRegion::fill_with_dummy_object(HeapWord* address, size_t word_size, bool zap) {
822+
// Keep the BOT in sync for old generation regions.
823+
if (is_old()) {
824+
update_bot_at(address, word_size);
825+
}
826+
// Fill in the object.
827+
CollectedHeap::fill_with_object(address, word_size, zap);
828+
}

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

+23-13
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,7 @@ class HeapRegion : public CHeapObj<mtGC> {
7878
HeapWord* _compaction_top;
7979

8080
G1BlockOffsetTablePart _bot_part;
81-
Mutex _par_alloc_lock;
81+
8282
// When we need to retire an allocation region, while other threads
8383
// are also concurrently trying to allocate into it, we typically
8484
// allocate a dummy object at the end of the region to ensure that
@@ -153,18 +153,24 @@ class HeapRegion : public CHeapObj<mtGC> {
153153

154154
void object_iterate(ObjectClosure* blk);
155155

156-
// Allocation (return NULL if full). Assumes the caller has established
157-
// mutually exclusive access to the HeapRegion.
158-
HeapWord* allocate(size_t min_word_size, size_t desired_word_size, size_t* actual_word_size);
159-
// Allocation (return NULL if full). Enforces mutual exclusion internally.
160-
HeapWord* par_allocate(size_t min_word_size, size_t desired_word_size, size_t* actual_word_size);
161-
162-
HeapWord* allocate(size_t word_size);
163-
HeapWord* par_allocate(size_t word_size);
164-
165-
inline HeapWord* par_allocate_no_bot_updates(size_t min_word_size, size_t desired_word_size, size_t* word_size);
166-
inline HeapWord* allocate_no_bot_updates(size_t word_size);
167-
inline HeapWord* allocate_no_bot_updates(size_t min_word_size, size_t desired_word_size, size_t* actual_size);
156+
// At the given address create an object with the given size. If the region
157+
// is old the BOT will be updated if the object spans a threshold.
158+
void fill_with_dummy_object(HeapWord* address, size_t word_size, bool zap = true);
159+
160+
// All allocations are done without updating the BOT. The BOT
161+
// needs to be kept in sync for old generation regions and
162+
// this is done by explicit updates when crossing thresholds.
163+
inline HeapWord* par_allocate(size_t min_word_size, size_t desired_word_size, size_t* word_size);
164+
inline HeapWord* allocate(size_t word_size);
165+
inline HeapWord* allocate(size_t min_word_size, size_t desired_word_size, size_t* actual_size);
166+
167+
// Update the BOT for the given address if it crosses the next
168+
// BOT threshold at or after obj_start.
169+
inline void update_bot_at(HeapWord* obj_start, size_t obj_size);
170+
// Update BOT at the given threshold for the given object. The
171+
// given object must cross the threshold.
172+
inline void update_bot_crossing_threshold(HeapWord** threshold, HeapWord* obj_start, HeapWord* obj_end);
173+
inline HeapWord* bot_threshold_for_addr(const void* addr);
168174

169175
// Full GC support methods.
170176

@@ -201,6 +207,10 @@ class HeapRegion : public CHeapObj<mtGC> {
201207
_bot_part.update();
202208
}
203209

210+
void update_bot_threshold() {
211+
_bot_part.set_threshold(top());
212+
}
213+
204214
private:
205215
// The remembered set for this region.
206216
HeapRegionRemSet* _rem_set;

‎src/hotspot/share/gc/g1/heapRegion.inline.hpp

+37-40
Original file line numberDiff line numberDiff line change
@@ -79,36 +79,6 @@ inline HeapWord* HeapRegion::par_allocate_impl(size_t min_word_size,
7979
} while (true);
8080
}
8181

82-
inline HeapWord* HeapRegion::allocate(size_t min_word_size,
83-
size_t desired_word_size,
84-
size_t* actual_size) {
85-
HeapWord* res = allocate_impl(min_word_size, desired_word_size, actual_size);
86-
if (res != NULL) {
87-
_bot_part.alloc_block(res, *actual_size);
88-
}
89-
return res;
90-
}
91-
92-
inline HeapWord* HeapRegion::allocate(size_t word_size) {
93-
size_t temp;
94-
return allocate(word_size, word_size, &temp);
95-
}
96-
97-
inline HeapWord* HeapRegion::par_allocate(size_t word_size) {
98-
size_t temp;
99-
return par_allocate(word_size, word_size, &temp);
100-
}
101-
102-
// Because of the requirement of keeping "_offsets" up to date with the
103-
// allocations, we sequentialize these with a lock. Therefore, best if
104-
// this is used for larger LAB allocations only.
105-
inline HeapWord* HeapRegion::par_allocate(size_t min_word_size,
106-
size_t desired_word_size,
107-
size_t* actual_size) {
108-
MutexLocker x(&_par_alloc_lock, Mutex::_no_safepoint_check_flag);
109-
return allocate(min_word_size, desired_word_size, actual_size);
110-
}
111-
11282
inline HeapWord* HeapRegion::block_start(const void* p) {
11383
return _bot_part.block_start(p);
11484
}
@@ -252,25 +222,52 @@ inline void HeapRegion::apply_to_marked_objects(G1CMBitMap* bitmap, ApplyToMarke
252222
assert(next_addr == limit, "Should stop the scan at the limit.");
253223
}
254224

255-
inline HeapWord* HeapRegion::par_allocate_no_bot_updates(size_t min_word_size,
256-
size_t desired_word_size,
257-
size_t* actual_word_size) {
258-
assert(is_young(), "we can only skip BOT updates on young regions");
225+
inline HeapWord* HeapRegion::par_allocate(size_t min_word_size,
226+
size_t desired_word_size,
227+
size_t* actual_word_size) {
259228
return par_allocate_impl(min_word_size, desired_word_size, actual_word_size);
260229
}
261230

262-
inline HeapWord* HeapRegion::allocate_no_bot_updates(size_t word_size) {
231+
inline HeapWord* HeapRegion::allocate(size_t word_size) {
263232
size_t temp;
264-
return allocate_no_bot_updates(word_size, word_size, &temp);
233+
return allocate(word_size, word_size, &temp);
265234
}
266235

267-
inline HeapWord* HeapRegion::allocate_no_bot_updates(size_t min_word_size,
268-
size_t desired_word_size,
269-
size_t* actual_word_size) {
270-
assert(is_young(), "we can only skip BOT updates on young regions");
236+
inline HeapWord* HeapRegion::allocate(size_t min_word_size,
237+
size_t desired_word_size,
238+
size_t* actual_word_size) {
271239
return allocate_impl(min_word_size, desired_word_size, actual_word_size);
272240
}
273241

242+
inline HeapWord* HeapRegion::bot_threshold_for_addr(const void* addr) {
243+
HeapWord* threshold = _bot_part.threshold_for_addr(addr);
244+
assert(threshold >= addr,
245+
"threshold must be at or after given address. " PTR_FORMAT " >= " PTR_FORMAT,
246+
p2i(threshold), p2i(addr));
247+
assert(is_old(),
248+
"Should only calculate BOT threshold for old regions. addr: " PTR_FORMAT " region:" HR_FORMAT,
249+
p2i(addr), HR_FORMAT_PARAMS(this));
250+
return threshold;
251+
}
252+
253+
inline void HeapRegion::update_bot_crossing_threshold(HeapWord** threshold, HeapWord* obj_start, HeapWord* obj_end) {
254+
assert(is_old(), "should only do BOT updates for old regions");
255+
assert(is_in(obj_start), "obj_start must be in this region: " HR_FORMAT
256+
" obj_start " PTR_FORMAT " obj_end " PTR_FORMAT " threshold " PTR_FORMAT,
257+
HR_FORMAT_PARAMS(this),
258+
p2i(obj_start), p2i(obj_end), p2i(*threshold));
259+
_bot_part.alloc_block_work(threshold, obj_start, obj_end);
260+
}
261+
262+
inline void HeapRegion::update_bot_at(HeapWord* obj_start, size_t obj_size) {
263+
HeapWord* threshold = bot_threshold_for_addr(obj_start);
264+
HeapWord* obj_end = obj_start + obj_size;
265+
266+
if (obj_end > threshold) {
267+
update_bot_crossing_threshold(&threshold, obj_start, obj_end);
268+
}
269+
}
270+
274271
inline void HeapRegion::note_start_of_marking() {
275272
_next_marked_bytes = 0;
276273
_next_top_at_mark_start = top();

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -119,7 +119,7 @@ class PLAB: public CHeapObj<mtGC> {
119119
}
120120

121121
// Sets the space of the buffer to be [buf, space+word_sz()).
122-
void set_buf(HeapWord* buf, size_t new_word_sz) {
122+
virtual void set_buf(HeapWord* buf, size_t new_word_sz) {
123123
assert(new_word_sz > AlignmentReserve, "Too small");
124124
_word_sz = new_word_sz;
125125

0 commit comments

Comments
 (0)
Please sign in to comment.