jdk-24/test/hotspot/gtest/metaspace/test_freeblocks.cpp

242 lines
6.3 KiB
C++
Raw Normal View History

/*
* Copyright (c) 2020, 2024, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2020 SAP SE. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*
*/
#include "precompiled.hpp"
#include "memory/metaspace/counters.hpp"
#include "memory/metaspace/freeBlocks.hpp"
//#define LOG_PLEASE
#include "metaspaceGtestCommon.hpp"
using metaspace::FreeBlocks;
using metaspace::SizeCounter;
#define CHECK_CONTENT(fb, num_blocks_expected, word_size_expected) \
{ \
if (word_size_expected > 0) { \
EXPECT_FALSE(fb.is_empty()); \
} else { \
EXPECT_TRUE(fb.is_empty()); \
} \
EXPECT_EQ(fb.total_size(), (size_t)word_size_expected); \
EXPECT_EQ(fb.count(), (int)num_blocks_expected); \
}
class FreeBlocksTest {
FeederBuffer _fb;
FreeBlocks _freeblocks;
// random generator for block feeding
RandSizeGenerator _rgen_feeding;
// random generator for allocations (and, hence, deallocations)
RandSizeGenerator _rgen_allocations;
SizeCounter _allocated_words;
struct allocation_t {
allocation_t* next;
size_t word_size;
MetaWord* p;
};
// Array of the same size as the pool max capacity; holds the allocated elements.
allocation_t* _allocations;
int _num_allocs;
int _num_deallocs;
int _num_feeds;
bool feed_some() {
size_t word_size = _rgen_feeding.get();
MetaWord* p = _fb.get(word_size);
if (p != nullptr) {
_freeblocks.add_block(p, word_size);
return true;
}
return false;
}
bool deallocate_top() {
allocation_t* a = _allocations;
if (a != nullptr) {
_allocations = a->next;
check_marked_range(a->p, a->word_size);
_freeblocks.add_block(a->p, a->word_size);
delete a;
DEBUG_ONLY(_freeblocks.verify();)
return true;
}
return false;
}
void deallocate_all() {
while (deallocate_top());
}
bool allocate() {
size_t word_size = MAX2(_rgen_allocations.get(), _freeblocks.MinWordSize);
MetaWord* p = _freeblocks.remove_block(word_size);
if (p != nullptr) {
_allocated_words.increment_by(word_size);
allocation_t* a = new allocation_t;
a->p = p; a->word_size = word_size;
a->next = _allocations;
_allocations = a;
DEBUG_ONLY(_freeblocks.verify();)
mark_range(p, word_size);
return true;
}
return false;
}
void test_all_marked_ranges() {
for (allocation_t* a = _allocations; a != nullptr; a = a->next) {
check_marked_range(a->p, a->word_size);
}
}
void test_loop() {
// We loop and in each iteration execute one of three operations:
// - allocation from fbl
// - deallocation to fbl of a previously allocated block
// - feeding a new larger block into the fbl (mimicks chunk retiring)
// When we have fed all large blocks into the fbl (feedbuffer empty), we
// switch to draining the fbl completely (only allocs)
bool forcefeed = false;
bool draining = false;
bool stop = false;
int iter = 25000; // safety stop
while (!stop && iter > 0) {
iter --;
int surprise = (int)os::random() % 10;
if (!draining && (surprise >= 7 || forcefeed)) {
forcefeed = false;
if (feed_some()) {
_num_feeds++;
} else {
// We fed all input memory into the fbl. Now lets proceed until the fbl is drained.
draining = true;
}
} else if (!draining && surprise < 1) {
deallocate_top();
_num_deallocs++;
} else {
if (allocate()) {
_num_allocs++;
} else {
if (draining) {
stop = _freeblocks.total_size() < 512;
} else {
forcefeed = true;
}
}
}
if ((iter % 1000) == 0) {
DEBUG_ONLY(_freeblocks.verify();)
test_all_marked_ranges();
LOG("a %d (" SIZE_FORMAT "), d %d, f %d", _num_allocs, _allocated_words.get(), _num_deallocs, _num_feeds);
#ifdef LOG_PLEASE
_freeblocks.print(tty, true);
tty->cr();
#endif
}
}
// Drain
}
public:
FreeBlocksTest(size_t avg_alloc_size) :
_fb(512 * K), _freeblocks(),
_rgen_feeding(128, 4096),
_rgen_allocations(avg_alloc_size / 4, avg_alloc_size * 2, 0.01f, avg_alloc_size / 3, avg_alloc_size * 30),
_allocations(nullptr),
_num_allocs(0),
_num_deallocs(0),
_num_feeds(0)
{
CHECK_CONTENT(_freeblocks, 0, 0);
// some initial feeding
_freeblocks.add_block(_fb.get(1024), 1024);
CHECK_CONTENT(_freeblocks, 1, 1024);
}
~FreeBlocksTest() {
deallocate_all();
}
static void test_small_allocations() {
FreeBlocksTest test(10);
test.test_loop();
}
static void test_medium_allocations() {
FreeBlocksTest test(30);
test.test_loop();
}
static void test_large_allocations() {
FreeBlocksTest test(150);
test.test_loop();
}
};
TEST_VM(metaspace, freeblocks_basics) {
FreeBlocks fbl;
MetaWord tmp[1024];
CHECK_CONTENT(fbl, 0, 0);
fbl.add_block(tmp, 1024);
DEBUG_ONLY(fbl.verify();)
ASSERT_FALSE(fbl.is_empty());
CHECK_CONTENT(fbl, 1, 1024);
MetaWord* p = fbl.remove_block(1024);
EXPECT_EQ(p, tmp);
DEBUG_ONLY(fbl.verify();)
CHECK_CONTENT(fbl, 0, 0);
}
TEST_VM(metaspace, freeblocks_small) {
FreeBlocksTest::test_small_allocations();
}
TEST_VM(metaspace, freeblocks_medium) {
FreeBlocksTest::test_medium_allocations();
}
TEST_VM(metaspace, freeblocks_large) {
FreeBlocksTest::test_large_allocations();
}