Skip to content

Commit 04b8d0c

Browse files
coleenpjcking
andcommittedDec 13, 2022
8298084: Memory leak in Method::build_profiling_method_data
Co-authored-by: Justin King <jcking@openjdk.org> Reviewed-by: kbarrett, eosterlund, dholmes, jcking, thartmann
1 parent d453190 commit 04b8d0c

File tree

6 files changed

+38
-18
lines changed

6 files changed

+38
-18
lines changed
 

‎src/hotspot/share/memory/metadataFactory.hpp

+11-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2010, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2010, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -30,6 +30,7 @@
3030
#include "oops/array.inline.hpp"
3131
#include "utilities/exceptions.hpp"
3232
#include "utilities/globalDefinitions.hpp"
33+
#include <type_traits>
3334

3435
class MetadataFactory : AllStatic {
3536
public:
@@ -61,14 +62,21 @@ class MetadataFactory : AllStatic {
6162

6263
// Deallocation method for metadata
6364
template <class T>
64-
static void free_metadata(ClassLoaderData* loader_data, T md) {
65+
static void free_metadata(ClassLoaderData* loader_data, T* md) {
6566
if (md != NULL) {
6667
assert(loader_data != NULL, "shouldn't pass null");
6768
int size = md->size();
68-
// Call metadata's deallocate function which will call deallocate fields
69+
// Call metadata's deallocate function which will deallocate fields and release_C_heap_structures
6970
assert(!md->on_stack(), "can't deallocate things on stack");
7071
assert(!md->is_shared(), "cannot deallocate if in shared spaces");
7172
md->deallocate_contents(loader_data);
73+
// Call the destructor. This is currently used for MethodData which has a member
74+
// that needs to be destructed to release resources. Most Metadata derived classes have noop
75+
// destructors and/or cleanup using deallocate_contents.
76+
// T is a potentially const or volatile qualified pointer. Remove any const
77+
// or volatile so we can call the destructor of the type T points to.
78+
using U = std::remove_cv_t<T>;
79+
md->~U();
7280
loader_data->metaspace_non_null()->deallocate((MetaWord*)md, size, md->is_klass());
7381
}
7482
}

‎src/hotspot/share/oops/instanceKlass.cpp

+10-8
Original file line numberDiff line numberDiff line change
@@ -595,10 +595,10 @@ void InstanceKlass::deallocate_contents(ClassLoaderData* loader_data) {
595595

596596
// Release C heap allocated data that this points to, which includes
597597
// reference counting symbol names.
598-
// Can't release the constant pool here because the constant pool can be
599-
// deallocated separately from the InstanceKlass for default methods and
600-
// redefine classes.
601-
release_C_heap_structures(/* release_constant_pool */ false);
598+
// Can't release the constant pool or MethodData C heap data here because the constant
599+
// pool can be deallocated separately from the InstanceKlass for default methods and
600+
// redefine classes. MethodData can also be released separately.
601+
release_C_heap_structures(/* release_sub_metadata */ false);
602602

603603
deallocate_methods(loader_data, methods());
604604
set_methods(NULL);
@@ -2650,13 +2650,15 @@ static void method_release_C_heap_structures(Method* m) {
26502650
m->release_C_heap_structures();
26512651
}
26522652

2653-
// Called also by InstanceKlass::deallocate_contents, with false for release_constant_pool.
2654-
void InstanceKlass::release_C_heap_structures(bool release_constant_pool) {
2653+
// Called also by InstanceKlass::deallocate_contents, with false for release_sub_metadata.
2654+
void InstanceKlass::release_C_heap_structures(bool release_sub_metadata) {
26552655
// Clean up C heap
26562656
Klass::release_C_heap_structures();
26572657

26582658
// Deallocate and call destructors for MDO mutexes
2659-
methods_do(method_release_C_heap_structures);
2659+
if (release_sub_metadata) {
2660+
methods_do(method_release_C_heap_structures);
2661+
}
26602662

26612663
// Destroy the init_monitor
26622664
delete _init_monitor;
@@ -2696,7 +2698,7 @@ void InstanceKlass::release_C_heap_structures(bool release_constant_pool) {
26962698

26972699
FREE_C_HEAP_ARRAY(char, _source_debug_extension);
26982700

2699-
if (release_constant_pool) {
2701+
if (release_sub_metadata) {
27002702
constants()->release_C_heap_structures();
27012703
}
27022704
}

‎src/hotspot/share/oops/instanceKlass.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -1012,7 +1012,7 @@ class InstanceKlass: public Klass {
10121012
// callbacks for actions during class unloading
10131013
static void unload_class(InstanceKlass* ik);
10141014

1015-
virtual void release_C_heap_structures(bool release_constant_pool = true);
1015+
virtual void release_C_heap_structures(bool release_sub_metadata = true);
10161016

10171017
// Naming
10181018
const char* signature_name() const;

‎src/hotspot/share/oops/method.cpp

+3-4
Original file line numberDiff line numberDiff line change
@@ -141,10 +141,9 @@ void Method::deallocate_contents(ClassLoaderData* loader_data) {
141141

142142
void Method::release_C_heap_structures() {
143143
if (method_data()) {
144-
#if INCLUDE_JVMCI
145-
FailedSpeculation::free_failed_speculations(method_data()->get_failed_speculations_address());
146-
#endif
147-
// Destroy MethodData
144+
method_data()->release_C_heap_structures();
145+
146+
// Destroy MethodData embedded lock
148147
method_data()->~MethodData();
149148
}
150149
}

‎src/hotspot/share/oops/methodData.cpp

+10
Original file line numberDiff line numberDiff line change
@@ -1821,3 +1821,13 @@ void MethodData::clean_weak_method_links() {
18211821
clean_extra_data(&cl);
18221822
verify_extra_data_clean(&cl);
18231823
}
1824+
1825+
void MethodData::deallocate_contents(ClassLoaderData* loader_data) {
1826+
release_C_heap_structures();
1827+
}
1828+
1829+
void MethodData::release_C_heap_structures() {
1830+
#if INCLUDE_JVMCI
1831+
FailedSpeculation::free_failed_speculations(get_failed_speculations_address());
1832+
#endif
1833+
}

‎src/hotspot/share/oops/methodData.hpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -2449,8 +2449,9 @@ class MethodData : public Metadata {
24492449
virtual void metaspace_pointers_do(MetaspaceClosure* iter);
24502450
virtual MetaspaceObj::Type type() const { return MethodDataType; }
24512451

2452-
// Deallocation support - no metaspace pointer fields to deallocate
2453-
void deallocate_contents(ClassLoaderData* loader_data) {}
2452+
// Deallocation support
2453+
void deallocate_contents(ClassLoaderData* loader_data);
2454+
void release_C_heap_structures();
24542455

24552456
// GC support
24562457
void set_size(int object_size_in_bytes) { _size = object_size_in_bytes; }

0 commit comments

Comments
 (0)
Please sign in to comment.