Path: blob/master/test/hotspot/gtest/metaspace/metaspaceGtestRangeHelpers.hpp
41144 views
/*1* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.2* Copyright (c) 2020 SAP SE. 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 it6* under the terms of the GNU General Public License version 2 only, as7* published by the Free Software Foundation.8*9* This code is distributed in the hope that it will be useful, but WITHOUT10* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or11* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License12* version 2 for more details (a copy is included in the LICENSE file that13* accompanied this code).14*15* You should have received a copy of the GNU General Public License version16* 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 USA20* or visit www.oracle.com if you need additional information or have any21* questions.22*23*/2425#ifndef GTEST_METASPACE_METASPACEGTESTRANGEHELPERS_HPP26#define GTEST_METASPACE_METASPACEGTESTRANGEHELPERS_HPP2728// We use ranges-of-things in these tests a lot so some helpers help29// keeping the code small.3031#include "memory/allocation.hpp"32#include "memory/metaspace/chunklevel.hpp"33#include "runtime/os.hpp" // For os::random34#include "utilities/align.hpp"35#include "utilities/debug.hpp"36#include "utilities/globalDefinitions.hpp"3738using metaspace::chunklevel_t;39using namespace metaspace::chunklevel;4041// A range of numerical values.42template <typename T, typename Td>43class Range : public StackObj {4445// start and size of range46T _start;47Td _size;4849static Td random_uncapped_offset() {50if (sizeof(Td) > 4) {51return (Td)((uint64_t)os::random() * os::random());52} else {53return (Td)os::random();54}55}5657protected:5859static void swap_if_needed(T& lo, T& hi) {60if (lo > hi) {61T v = lo;62lo = hi;63hi = v;64}65}6667public:6869// Lowest value in range70T lowest() const { return _start; }7172// Highest value in range (including)73T highest() const { return _start + (_size - 1); }7475T start() const { return _start; }76T end() const { return _start + _size; }7778// Number of values in range79Td size() const { return _size; }8081bool is_empty() const { return size() == 0; }8283bool contains(T v) const {84return v >= _start && v < end();85}8687bool contains(Range<T, Td> r) const {88return contains(r.lowest()) && contains(r.highest());89}9091// Create a range from [start, end)92Range(T start, T end) : _start(start), _size(end - start) {93assert(end >= start, "start and end reversed");94}9596// a range with a given size, starting at 097Range(Td size) : _start(0), _size(size) {}9899// Return a random offset100Td random_offset() const {101assert(!is_empty(), "Range too small");102Td v = random_uncapped_offset() % size();103return v;104}105106// Return a random value within the range107T random_value() const {108assert(!is_empty(), "Range too small");109T v = _start + random_offset();110assert(contains(v), "Sanity");111return v;112}113114// Return the head of this range up to but excluding <split_point>115Range<T, Td> head(Td split_point) const {116assert(_size >= split_point, "Sanity");117return Range<T, Td>(_start, _start + split_point);118}119120// Return the tail of this range, starting at <split_point>121Range<T, Td> tail(Td split_point) const {122assert(_size > split_point, "Sanity");123return Range<T, Td>(_start + split_point, end());124}125126// Return a non-empty random sub range.127Range<T, Td> random_subrange() const {128assert(size() > 1, "Range too small");129Td sz = MAX2((Td)1, random_offset());130return random_sized_subrange(sz);131}132133// Return a subrange of given size at a random start position134Range<T, Td> random_sized_subrange(Td subrange_size) const {135assert(subrange_size > 0 && subrange_size < _size, "invalid size");136T start = head(_size - subrange_size).random_value();137return Range<T, Td>(start, start + subrange_size);138}139140//// aligned ranges ////141142bool range_is_aligned(Td alignment) const {143return is_aligned(_size, alignment) && is_aligned(_start, alignment);144}145146// Return a non-empty aligned random sub range.147Range<T, Td> random_aligned_subrange(Td alignment) const {148assert(alignment > 0, "Sanity");149assert(range_is_aligned(alignment), "Outer range needs to be aligned"); // to keep matters simple150assert(_size >= alignment, "Outer range too small.");151Td sz = MAX2((Td)1, random_offset());152sz = align_up(sz, alignment);153return random_aligned_sized_subrange(sz, alignment);154}155156// Return a subrange of given size at a random aligned start position157Range<T, Td> random_aligned_sized_subrange(Td subrange_size, Td alignment) const {158assert(alignment > 0, "Sanity");159assert(range_is_aligned(alignment), "Outer range needs to be aligned"); // to keep matters simple160assert(subrange_size > 0 && subrange_size <= _size &&161is_aligned(subrange_size, alignment), "invalid subrange size");162if (_size == subrange_size) {163return *this;164}165T start = head(_size - subrange_size).random_value();166start = align_down(start, alignment);167return Range<T, Td>(start, start + subrange_size);168}169170};171172typedef Range<int, int> IntRange;173typedef Range<size_t, size_t> SizeRange;174typedef Range<chunklevel_t, int> ChunkLevelRange;175176struct ChunkLevelRanges : public AllStatic {177static ChunkLevelRange small_chunks() { return ChunkLevelRange(CHUNK_LEVEL_32K, CHUNK_LEVEL_1K + 1); }178static ChunkLevelRange medium_chunks() { return ChunkLevelRange(CHUNK_LEVEL_512K, CHUNK_LEVEL_32K + 1); }179static ChunkLevelRange large_chunks() { return ChunkLevelRange(CHUNK_LEVEL_4M, CHUNK_LEVEL_512K + 1); }180static ChunkLevelRange all_chunks() { return ChunkLevelRange(CHUNK_LEVEL_4M, CHUNK_LEVEL_1K + 1); }181};182183#endif // GTEST_METASPACE_METASPACEGTESTRANGEHELPERS_HPP184185186