Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/mobile
Path: blob/master/test/hotspot/gtest/metaspace/metaspaceGtestRangeHelpers.hpp
41144 views
1
/*
2
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
3
* Copyright (c) 2020 SAP SE. All rights reserved.
4
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5
*
6
* This code is free software; you can redistribute it and/or modify it
7
* under the terms of the GNU General Public License version 2 only, as
8
* published by the Free Software Foundation.
9
*
10
* This code is distributed in the hope that it will be useful, but WITHOUT
11
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
13
* version 2 for more details (a copy is included in the LICENSE file that
14
* accompanied this code).
15
*
16
* You should have received a copy of the GNU General Public License version
17
* 2 along with this work; if not, write to the Free Software Foundation,
18
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
19
*
20
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
21
* or visit www.oracle.com if you need additional information or have any
22
* questions.
23
*
24
*/
25
26
#ifndef GTEST_METASPACE_METASPACEGTESTRANGEHELPERS_HPP
27
#define GTEST_METASPACE_METASPACEGTESTRANGEHELPERS_HPP
28
29
// We use ranges-of-things in these tests a lot so some helpers help
30
// keeping the code small.
31
32
#include "memory/allocation.hpp"
33
#include "memory/metaspace/chunklevel.hpp"
34
#include "runtime/os.hpp" // For os::random
35
#include "utilities/align.hpp"
36
#include "utilities/debug.hpp"
37
#include "utilities/globalDefinitions.hpp"
38
39
using metaspace::chunklevel_t;
40
using namespace metaspace::chunklevel;
41
42
// A range of numerical values.
43
template <typename T, typename Td>
44
class Range : public StackObj {
45
46
// start and size of range
47
T _start;
48
Td _size;
49
50
static Td random_uncapped_offset() {
51
if (sizeof(Td) > 4) {
52
return (Td)((uint64_t)os::random() * os::random());
53
} else {
54
return (Td)os::random();
55
}
56
}
57
58
protected:
59
60
static void swap_if_needed(T& lo, T& hi) {
61
if (lo > hi) {
62
T v = lo;
63
lo = hi;
64
hi = v;
65
}
66
}
67
68
public:
69
70
// Lowest value in range
71
T lowest() const { return _start; }
72
73
// Highest value in range (including)
74
T highest() const { return _start + (_size - 1); }
75
76
T start() const { return _start; }
77
T end() const { return _start + _size; }
78
79
// Number of values in range
80
Td size() const { return _size; }
81
82
bool is_empty() const { return size() == 0; }
83
84
bool contains(T v) const {
85
return v >= _start && v < end();
86
}
87
88
bool contains(Range<T, Td> r) const {
89
return contains(r.lowest()) && contains(r.highest());
90
}
91
92
// Create a range from [start, end)
93
Range(T start, T end) : _start(start), _size(end - start) {
94
assert(end >= start, "start and end reversed");
95
}
96
97
// a range with a given size, starting at 0
98
Range(Td size) : _start(0), _size(size) {}
99
100
// Return a random offset
101
Td random_offset() const {
102
assert(!is_empty(), "Range too small");
103
Td v = random_uncapped_offset() % size();
104
return v;
105
}
106
107
// Return a random value within the range
108
T random_value() const {
109
assert(!is_empty(), "Range too small");
110
T v = _start + random_offset();
111
assert(contains(v), "Sanity");
112
return v;
113
}
114
115
// Return the head of this range up to but excluding <split_point>
116
Range<T, Td> head(Td split_point) const {
117
assert(_size >= split_point, "Sanity");
118
return Range<T, Td>(_start, _start + split_point);
119
}
120
121
// Return the tail of this range, starting at <split_point>
122
Range<T, Td> tail(Td split_point) const {
123
assert(_size > split_point, "Sanity");
124
return Range<T, Td>(_start + split_point, end());
125
}
126
127
// Return a non-empty random sub range.
128
Range<T, Td> random_subrange() const {
129
assert(size() > 1, "Range too small");
130
Td sz = MAX2((Td)1, random_offset());
131
return random_sized_subrange(sz);
132
}
133
134
// Return a subrange of given size at a random start position
135
Range<T, Td> random_sized_subrange(Td subrange_size) const {
136
assert(subrange_size > 0 && subrange_size < _size, "invalid size");
137
T start = head(_size - subrange_size).random_value();
138
return Range<T, Td>(start, start + subrange_size);
139
}
140
141
//// aligned ranges ////
142
143
bool range_is_aligned(Td alignment) const {
144
return is_aligned(_size, alignment) && is_aligned(_start, alignment);
145
}
146
147
// Return a non-empty aligned random sub range.
148
Range<T, Td> random_aligned_subrange(Td alignment) const {
149
assert(alignment > 0, "Sanity");
150
assert(range_is_aligned(alignment), "Outer range needs to be aligned"); // to keep matters simple
151
assert(_size >= alignment, "Outer range too small.");
152
Td sz = MAX2((Td)1, random_offset());
153
sz = align_up(sz, alignment);
154
return random_aligned_sized_subrange(sz, alignment);
155
}
156
157
// Return a subrange of given size at a random aligned start position
158
Range<T, Td> random_aligned_sized_subrange(Td subrange_size, Td alignment) const {
159
assert(alignment > 0, "Sanity");
160
assert(range_is_aligned(alignment), "Outer range needs to be aligned"); // to keep matters simple
161
assert(subrange_size > 0 && subrange_size <= _size &&
162
is_aligned(subrange_size, alignment), "invalid subrange size");
163
if (_size == subrange_size) {
164
return *this;
165
}
166
T start = head(_size - subrange_size).random_value();
167
start = align_down(start, alignment);
168
return Range<T, Td>(start, start + subrange_size);
169
}
170
171
};
172
173
typedef Range<int, int> IntRange;
174
typedef Range<size_t, size_t> SizeRange;
175
typedef Range<chunklevel_t, int> ChunkLevelRange;
176
177
struct ChunkLevelRanges : public AllStatic {
178
static ChunkLevelRange small_chunks() { return ChunkLevelRange(CHUNK_LEVEL_32K, CHUNK_LEVEL_1K + 1); }
179
static ChunkLevelRange medium_chunks() { return ChunkLevelRange(CHUNK_LEVEL_512K, CHUNK_LEVEL_32K + 1); }
180
static ChunkLevelRange large_chunks() { return ChunkLevelRange(CHUNK_LEVEL_4M, CHUNK_LEVEL_512K + 1); }
181
static ChunkLevelRange all_chunks() { return ChunkLevelRange(CHUNK_LEVEL_4M, CHUNK_LEVEL_1K + 1); }
182
};
183
184
#endif // GTEST_METASPACE_METASPACEGTESTRANGEHELPERS_HPP
185
186