2020-10-20 06:48:09 +00:00
|
|
|
/*
|
2024-08-27 15:46:10 +00:00
|
|
|
* Copyright (c) 2020, 2024, Oracle and/or its affiliates. All rights reserved.
|
2020-10-20 06:48:09 +00:00
|
|
|
* 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 "classfile/classLoaderData.hpp"
|
|
|
|
#include "memory/classLoaderMetaspace.hpp"
|
|
|
|
#include "memory/metaspace/chunklevel.hpp"
|
|
|
|
#include "memory/metaspace/metaspaceSettings.hpp"
|
|
|
|
#include "utilities/powerOfTwo.hpp"
|
|
|
|
// #define LOG_PLEASE
|
|
|
|
#include "metaspaceGtestCommon.hpp"
|
|
|
|
|
|
|
|
using metaspace::chunklevel_t;
|
|
|
|
using namespace metaspace::chunklevel;
|
|
|
|
using metaspace::Settings;
|
|
|
|
|
|
|
|
TEST_VM(metaspace, misc_sizes) {
|
|
|
|
|
|
|
|
// Test test common sizes (seems primitive but breaks surprisingly often during development
|
|
|
|
// because of word vs byte confusion)
|
|
|
|
// Adjust this test if numbers change.
|
|
|
|
ASSERT_TRUE(Settings::commit_granule_bytes() == 16 * K ||
|
|
|
|
Settings::commit_granule_bytes() == 64 * K);
|
|
|
|
ASSERT_EQ(Settings::commit_granule_bytes(), Metaspace::commit_alignment());
|
|
|
|
ASSERT_TRUE(is_aligned(Settings::virtual_space_node_default_word_size(),
|
|
|
|
metaspace::chunklevel::MAX_CHUNK_WORD_SIZE));
|
2023-01-24 06:35:26 +00:00
|
|
|
ASSERT_EQ(Settings::virtual_space_node_default_word_size() * BytesPerWord, NOT_LP64(16) LP64_ONLY(64) * M);
|
2020-10-20 06:48:09 +00:00
|
|
|
ASSERT_EQ(Settings::virtual_space_node_reserve_alignment_words(),
|
|
|
|
Metaspace::reserve_alignment_words());
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_VM(metaspace, misc_max_alloc_size) {
|
|
|
|
|
2021-09-21 06:27:32 +00:00
|
|
|
// Make sure we can allocate what we promise to allocate...
|
2023-01-24 06:35:26 +00:00
|
|
|
for (int i = 0; i < 2; i ++) {
|
|
|
|
const bool in_class_space = (i == 0);
|
|
|
|
const Metaspace::MetadataType mdType = in_class_space ? Metaspace::ClassType : Metaspace::NonClassType;
|
|
|
|
const size_t sz = Metaspace::max_allocation_word_size();
|
|
|
|
ClassLoaderData* cld = ClassLoaderData::the_null_class_loader_data();
|
|
|
|
MetaWord* p = cld->metaspace_non_null()->allocate(sz, mdType);
|
|
|
|
if (p == nullptr) {
|
|
|
|
// Have we run into the GC threshold?
|
|
|
|
p = cld->metaspace_non_null()->expand_and_allocate(sz, mdType);
|
|
|
|
ASSERT_NOT_NULL(p);
|
|
|
|
}
|
|
|
|
// And also, successfully deallocate it.
|
2024-08-27 15:46:10 +00:00
|
|
|
cld->metaspace_non_null()->deallocate(p, sz);
|
2023-01-24 06:35:26 +00:00
|
|
|
}
|
2020-10-20 06:48:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_VM(metaspace, chunklevel_utils) {
|
|
|
|
|
|
|
|
// These tests seem to be really basic, but it is amazing what one can
|
|
|
|
// break accidentally...
|
|
|
|
LOG(SIZE_FORMAT, MAX_CHUNK_BYTE_SIZE);
|
|
|
|
LOG(SIZE_FORMAT, MIN_CHUNK_BYTE_SIZE);
|
|
|
|
LOG(SIZE_FORMAT, MIN_CHUNK_WORD_SIZE);
|
|
|
|
LOG(SIZE_FORMAT, MAX_CHUNK_WORD_SIZE);
|
|
|
|
LOG(SIZE_FORMAT, MAX_CHUNK_BYTE_SIZE);
|
|
|
|
LOG("%u", (unsigned)ROOT_CHUNK_LEVEL);
|
|
|
|
LOG("%u", (unsigned)HIGHEST_CHUNK_LEVEL);
|
|
|
|
LOG("%u", (unsigned)LOWEST_CHUNK_LEVEL);
|
|
|
|
|
|
|
|
static const chunklevel_t INVALID_CHUNK_LEVEL = (chunklevel_t) -1;
|
|
|
|
|
|
|
|
EXPECT_TRUE(is_power_of_2(MAX_CHUNK_WORD_SIZE));
|
|
|
|
EXPECT_TRUE(is_power_of_2(MIN_CHUNK_WORD_SIZE));
|
|
|
|
|
|
|
|
EXPECT_TRUE(is_valid_level(LOWEST_CHUNK_LEVEL));
|
|
|
|
EXPECT_TRUE(is_valid_level(HIGHEST_CHUNK_LEVEL));
|
|
|
|
EXPECT_FALSE(is_valid_level(HIGHEST_CHUNK_LEVEL + 1));
|
|
|
|
EXPECT_FALSE(is_valid_level(LOWEST_CHUNK_LEVEL - 1));
|
|
|
|
|
|
|
|
EXPECT_EQ(word_size_for_level(ROOT_CHUNK_LEVEL), MAX_CHUNK_WORD_SIZE);
|
|
|
|
EXPECT_EQ(word_size_for_level(HIGHEST_CHUNK_LEVEL), MIN_CHUNK_WORD_SIZE);
|
|
|
|
|
|
|
|
EXPECT_EQ(word_size_for_level(CHUNK_LEVEL_4K), (4 * K) / BytesPerWord);
|
|
|
|
EXPECT_EQ(word_size_for_level(CHUNK_LEVEL_64K), (64 * K) / BytesPerWord);
|
|
|
|
|
|
|
|
EXPECT_EQ(level_fitting_word_size(0), HIGHEST_CHUNK_LEVEL);
|
|
|
|
EXPECT_EQ(level_fitting_word_size(1), HIGHEST_CHUNK_LEVEL);
|
|
|
|
EXPECT_EQ(level_fitting_word_size(MIN_CHUNK_WORD_SIZE), HIGHEST_CHUNK_LEVEL);
|
|
|
|
EXPECT_EQ(level_fitting_word_size(MIN_CHUNK_WORD_SIZE + 1), HIGHEST_CHUNK_LEVEL - 1);
|
|
|
|
|
|
|
|
EXPECT_EQ(level_fitting_word_size(MAX_CHUNK_WORD_SIZE), ROOT_CHUNK_LEVEL);
|
|
|
|
EXPECT_EQ(level_fitting_word_size(MAX_CHUNK_WORD_SIZE - 1), ROOT_CHUNK_LEVEL);
|
|
|
|
EXPECT_EQ(level_fitting_word_size((MAX_CHUNK_WORD_SIZE / 2) + 1), ROOT_CHUNK_LEVEL);
|
|
|
|
EXPECT_EQ(level_fitting_word_size(MAX_CHUNK_WORD_SIZE / 2), ROOT_CHUNK_LEVEL + 1);
|
|
|
|
|
|
|
|
EXPECT_EQ(level_fitting_word_size(8 * K), LP64_ONLY(CHUNK_LEVEL_64K) NOT_LP64(CHUNK_LEVEL_32K));
|
|
|
|
EXPECT_EQ(level_fitting_word_size(8 * K + 13), LP64_ONLY(CHUNK_LEVEL_64K) NOT_LP64(CHUNK_LEVEL_32K) - 1);
|
|
|
|
EXPECT_EQ(level_fitting_word_size(8 * K - 13), LP64_ONLY(CHUNK_LEVEL_64K) NOT_LP64(CHUNK_LEVEL_32K));
|
|
|
|
|
|
|
|
}
|
|
|
|
|