Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/mobile
Path: blob/master/src/hotspot/share/services/memoryManager.cpp
41145 views
1
/*
2
* Copyright (c) 2003, 2021, Oracle and/or its affiliates. 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 it
6
* under the terms of the GNU General Public License version 2 only, as
7
* published by the Free Software Foundation.
8
*
9
* This code is distributed in the hope that it will be useful, but WITHOUT
10
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12
* version 2 for more details (a copy is included in the LICENSE file that
13
* accompanied this code).
14
*
15
* You should have received a copy of the GNU General Public License version
16
* 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 USA
20
* or visit www.oracle.com if you need additional information or have any
21
* questions.
22
*
23
*/
24
25
#include "precompiled.hpp"
26
#include "classfile/javaClasses.hpp"
27
#include "classfile/vmSymbols.hpp"
28
#include "memory/allocation.inline.hpp"
29
#include "memory/universe.hpp"
30
#include "oops/oop.inline.hpp"
31
#include "oops/oopHandle.inline.hpp"
32
#include "runtime/atomic.hpp"
33
#include "runtime/handles.inline.hpp"
34
#include "runtime/javaCalls.hpp"
35
#include "services/lowMemoryDetector.hpp"
36
#include "services/management.hpp"
37
#include "services/memoryManager.hpp"
38
#include "services/memoryPool.hpp"
39
#include "services/memoryService.hpp"
40
#include "services/gcNotifier.hpp"
41
#include "utilities/dtrace.hpp"
42
43
MemoryManager::MemoryManager(const char* name) :
44
_num_pools(0), _name(name) {}
45
46
int MemoryManager::add_pool(MemoryPool* pool) {
47
int index = _num_pools;
48
assert(index < MemoryManager::max_num_pools, "_num_pools exceeds the max");
49
if (index < MemoryManager::max_num_pools) {
50
_pools[index] = pool;
51
_num_pools++;
52
}
53
pool->add_manager(this);
54
return index;
55
}
56
57
bool MemoryManager::is_manager(instanceHandle mh) const {
58
return mh() == Atomic::load(&_memory_mgr_obj).resolve();
59
}
60
61
MemoryManager* MemoryManager::get_code_cache_memory_manager() {
62
return new MemoryManager("CodeCacheManager");
63
}
64
65
MemoryManager* MemoryManager::get_metaspace_memory_manager() {
66
return new MemoryManager("Metaspace Manager");
67
}
68
69
instanceOop MemoryManager::get_memory_manager_instance(TRAPS) {
70
// Must do an acquire so as to force ordering of subsequent
71
// loads from anything _memory_mgr_obj points to or implies.
72
oop mgr_obj = Atomic::load_acquire(&_memory_mgr_obj).resolve();
73
if (mgr_obj == NULL) {
74
// It's ok for more than one thread to execute the code up to the locked region.
75
// Extra manager instances will just be gc'ed.
76
Klass* k = Management::sun_management_ManagementFactoryHelper_klass(CHECK_NULL);
77
78
Handle mgr_name = java_lang_String::create_from_str(name(), CHECK_NULL);
79
80
JavaValue result(T_OBJECT);
81
JavaCallArguments args;
82
args.push_oop(mgr_name); // Argument 1
83
84
Symbol* method_name = NULL;
85
Symbol* signature = NULL;
86
if (is_gc_memory_manager()) {
87
Klass* extKlass = Management::com_sun_management_internal_GarbageCollectorExtImpl_klass(CHECK_NULL);
88
// com.sun.management.GarbageCollectorMXBean is in jdk.management module which may not be present.
89
if (extKlass != NULL) {
90
k = extKlass;
91
}
92
93
method_name = vmSymbols::createGarbageCollector_name();
94
95
signature = vmSymbols::createGarbageCollector_signature();
96
args.push_oop(Handle()); // Argument 2 (for future extension)
97
} else {
98
method_name = vmSymbols::createMemoryManager_name();
99
signature = vmSymbols::createMemoryManager_signature();
100
}
101
102
InstanceKlass* ik = InstanceKlass::cast(k);
103
104
JavaCalls::call_static(&result,
105
ik,
106
method_name,
107
signature,
108
&args,
109
CHECK_NULL);
110
111
instanceOop m = (instanceOop) result.get_oop();
112
instanceHandle mgr(THREAD, m);
113
114
{
115
// Get lock before setting _memory_mgr_obj
116
// since another thread may have created the instance
117
MutexLocker ml(THREAD, Management_lock);
118
119
// Check if another thread has created the management object. We reload
120
// _memory_mgr_obj here because some other thread may have initialized
121
// it while we were executing the code before the lock.
122
mgr_obj = Atomic::load(&_memory_mgr_obj).resolve();
123
if (mgr_obj != NULL) {
124
return (instanceOop)mgr_obj;
125
}
126
127
// Get the address of the object we created via call_special.
128
mgr_obj = mgr();
129
130
// Use store barrier to make sure the memory accesses associated
131
// with creating the management object are visible before publishing
132
// its address. The unlock will publish the store to _memory_mgr_obj
133
// because it does a release first.
134
Atomic::release_store(&_memory_mgr_obj, OopHandle(Universe::vm_global(), mgr_obj));
135
}
136
}
137
138
return (instanceOop)mgr_obj;
139
}
140
141
GCStatInfo::GCStatInfo(int num_pools) {
142
// initialize the arrays for memory usage
143
_before_gc_usage_array = NEW_C_HEAP_ARRAY(MemoryUsage, num_pools, mtInternal);
144
_after_gc_usage_array = NEW_C_HEAP_ARRAY(MemoryUsage, num_pools, mtInternal);
145
_usage_array_size = num_pools;
146
clear();
147
}
148
149
GCStatInfo::~GCStatInfo() {
150
FREE_C_HEAP_ARRAY(MemoryUsage*, _before_gc_usage_array);
151
FREE_C_HEAP_ARRAY(MemoryUsage*, _after_gc_usage_array);
152
}
153
154
void GCStatInfo::set_gc_usage(int pool_index, MemoryUsage usage, bool before_gc) {
155
MemoryUsage* gc_usage_array;
156
if (before_gc) {
157
gc_usage_array = _before_gc_usage_array;
158
} else {
159
gc_usage_array = _after_gc_usage_array;
160
}
161
gc_usage_array[pool_index] = usage;
162
}
163
164
void GCStatInfo::clear() {
165
_index = 0;
166
_start_time = 0L;
167
_end_time = 0L;
168
for (int i = 0; i < _usage_array_size; i++) ::new (&_before_gc_usage_array[i]) MemoryUsage();
169
for (int i = 0; i < _usage_array_size; i++) ::new (&_after_gc_usage_array[i]) MemoryUsage();
170
}
171
172
173
GCMemoryManager::GCMemoryManager(const char* name, const char* gc_end_message) :
174
MemoryManager(name), _gc_end_message(gc_end_message) {
175
_num_collections = 0;
176
_last_gc_stat = NULL;
177
_last_gc_lock = new Mutex(Mutex::leaf, "_last_gc_lock", true,
178
Mutex::_safepoint_check_never);
179
_current_gc_stat = NULL;
180
_num_gc_threads = 1;
181
_notification_enabled = false;
182
}
183
184
GCMemoryManager::~GCMemoryManager() {
185
delete _last_gc_stat;
186
delete _last_gc_lock;
187
delete _current_gc_stat;
188
}
189
190
void GCMemoryManager::add_pool(MemoryPool* pool) {
191
add_pool(pool, true);
192
}
193
194
void GCMemoryManager::add_pool(MemoryPool* pool, bool always_affected_by_gc) {
195
int index = MemoryManager::add_pool(pool);
196
_pool_always_affected_by_gc[index] = always_affected_by_gc;
197
}
198
199
void GCMemoryManager::initialize_gc_stat_info() {
200
assert(MemoryService::num_memory_pools() > 0, "should have one or more memory pools");
201
_last_gc_stat = new(ResourceObj::C_HEAP, mtGC) GCStatInfo(MemoryService::num_memory_pools());
202
_current_gc_stat = new(ResourceObj::C_HEAP, mtGC) GCStatInfo(MemoryService::num_memory_pools());
203
// tracking concurrent collections we need two objects: one to update, and one to
204
// hold the publicly available "last (completed) gc" information.
205
}
206
207
void GCMemoryManager::gc_begin(bool recordGCBeginTime, bool recordPreGCUsage,
208
bool recordAccumulatedGCTime) {
209
assert(_last_gc_stat != NULL && _current_gc_stat != NULL, "Just checking");
210
if (recordAccumulatedGCTime) {
211
_accumulated_timer.start();
212
}
213
// _num_collections now increases in gc_end, to count completed collections
214
if (recordGCBeginTime) {
215
_current_gc_stat->set_index(_num_collections+1);
216
_current_gc_stat->set_start_time(Management::timestamp());
217
}
218
219
if (recordPreGCUsage) {
220
// Keep memory usage of all memory pools
221
for (int i = 0; i < MemoryService::num_memory_pools(); i++) {
222
MemoryPool* pool = MemoryService::get_memory_pool(i);
223
MemoryUsage usage = pool->get_memory_usage();
224
_current_gc_stat->set_before_gc_usage(i, usage);
225
HOTSPOT_MEM_POOL_GC_BEGIN(
226
(char *) name(), strlen(name()),
227
(char *) pool->name(), strlen(pool->name()),
228
usage.init_size(), usage.used(),
229
usage.committed(), usage.max_size());
230
}
231
}
232
}
233
234
// A collector MUST, even if it does not complete for some reason,
235
// make a TraceMemoryManagerStats object where countCollection is true,
236
// to ensure the current gc stat is placed in _last_gc_stat.
237
void GCMemoryManager::gc_end(bool recordPostGCUsage,
238
bool recordAccumulatedGCTime,
239
bool recordGCEndTime, bool countCollection,
240
GCCause::Cause cause,
241
bool allMemoryPoolsAffected) {
242
if (recordAccumulatedGCTime) {
243
_accumulated_timer.stop();
244
}
245
if (recordGCEndTime) {
246
_current_gc_stat->set_end_time(Management::timestamp());
247
}
248
249
if (recordPostGCUsage) {
250
int i;
251
// keep the last gc statistics for all memory pools
252
for (i = 0; i < MemoryService::num_memory_pools(); i++) {
253
MemoryPool* pool = MemoryService::get_memory_pool(i);
254
MemoryUsage usage = pool->get_memory_usage();
255
256
HOTSPOT_MEM_POOL_GC_END(
257
(char *) name(), strlen(name()),
258
(char *) pool->name(), strlen(pool->name()),
259
usage.init_size(), usage.used(),
260
usage.committed(), usage.max_size());
261
262
_current_gc_stat->set_after_gc_usage(i, usage);
263
}
264
265
// Set last collection usage of the memory pools managed by this collector
266
for (i = 0; i < num_memory_pools(); i++) {
267
MemoryPool* pool = get_memory_pool(i);
268
MemoryUsage usage = pool->get_memory_usage();
269
270
if (allMemoryPoolsAffected || pool_always_affected_by_gc(i)) {
271
// Compare with GC usage threshold
272
pool->set_last_collection_usage(usage);
273
LowMemoryDetector::detect_after_gc_memory(pool);
274
}
275
}
276
}
277
278
if (countCollection) {
279
_num_collections++;
280
// alternately update two objects making one public when complete
281
{
282
MutexLocker ml(_last_gc_lock, Mutex::_no_safepoint_check_flag);
283
GCStatInfo *tmp = _last_gc_stat;
284
_last_gc_stat = _current_gc_stat;
285
_current_gc_stat = tmp;
286
// reset the current stat for diagnosability purposes
287
_current_gc_stat->clear();
288
}
289
290
if (is_notification_enabled()) {
291
GCNotifier::pushNotification(this, _gc_end_message, GCCause::to_string(cause));
292
}
293
}
294
}
295
296
size_t GCMemoryManager::get_last_gc_stat(GCStatInfo* dest) {
297
MutexLocker ml(_last_gc_lock, Mutex::_no_safepoint_check_flag);
298
if (_last_gc_stat->gc_index() != 0) {
299
dest->set_index(_last_gc_stat->gc_index());
300
dest->set_start_time(_last_gc_stat->start_time());
301
dest->set_end_time(_last_gc_stat->end_time());
302
assert(dest->usage_array_size() == _last_gc_stat->usage_array_size(),
303
"Must have same array size");
304
size_t len = dest->usage_array_size() * sizeof(MemoryUsage);
305
memcpy(dest->before_gc_usage_array(), _last_gc_stat->before_gc_usage_array(), len);
306
memcpy(dest->after_gc_usage_array(), _last_gc_stat->after_gc_usage_array(), len);
307
}
308
return _last_gc_stat->gc_index();
309
}
310
311