8292072: NMT: repurpose Tracking overhead counter as global malloc counter
Reviewed-by: iklam, clanger
This commit is contained in:
parent
d546d13861
commit
1c1c4410b2
@ -66,25 +66,6 @@ size_t MemoryCounter::peak_size() const {
|
||||
}
|
||||
#endif
|
||||
|
||||
// Total malloc invocation count
|
||||
size_t MallocMemorySnapshot::total_count() const {
|
||||
size_t amount = 0;
|
||||
for (int index = 0; index < mt_number_of_types; index ++) {
|
||||
amount += _malloc[index].malloc_count();
|
||||
}
|
||||
return amount;
|
||||
}
|
||||
|
||||
// Total malloc'd memory amount
|
||||
size_t MallocMemorySnapshot::total() const {
|
||||
size_t amount = 0;
|
||||
for (int index = 0; index < mt_number_of_types; index ++) {
|
||||
amount += _malloc[index].malloc_size();
|
||||
}
|
||||
amount += _tracking_header.size() + total_arena();
|
||||
return amount;
|
||||
}
|
||||
|
||||
// Total malloc'd memory used by arenas
|
||||
size_t MallocMemorySnapshot::total_arena() const {
|
||||
size_t amount = 0;
|
||||
@ -100,6 +81,7 @@ void MallocMemorySnapshot::make_adjustment() {
|
||||
size_t arena_size = total_arena();
|
||||
int chunk_idx = NMTUtil::flag_to_index(mtChunk);
|
||||
_malloc[chunk_idx].record_free(arena_size);
|
||||
_all_mallocs.deallocate(arena_size);
|
||||
}
|
||||
|
||||
void MallocMemorySummary::initialize() {
|
||||
@ -127,7 +109,6 @@ void* MallocTracker::record_malloc(void* malloc_base, size_t size, MEMFLAGS flag
|
||||
assert(malloc_base != NULL, "precondition");
|
||||
|
||||
MallocMemorySummary::record_malloc(size, flags);
|
||||
MallocMemorySummary::record_new_malloc_header(sizeof(MallocHeader));
|
||||
uint32_t mst_marker = 0;
|
||||
if (MemTracker::tracking_level() == NMT_detail) {
|
||||
MallocSiteTable::allocation_at(stack, size, &mst_marker, flags);
|
||||
@ -162,7 +143,6 @@ void* MallocTracker::record_free(void* memblock) {
|
||||
header->assert_block_integrity();
|
||||
|
||||
MallocMemorySummary::record_free(header->size(), header->flags());
|
||||
MallocMemorySummary::record_free_malloc_header(sizeof(MallocHeader));
|
||||
if (MemTracker::tracking_level() == NMT_detail) {
|
||||
MallocSiteTable::deallocation_at(header->size(), header->mst_marker());
|
||||
}
|
||||
|
@ -141,7 +141,7 @@ class MallocMemorySnapshot : public ResourceObj {
|
||||
|
||||
private:
|
||||
MallocMemory _malloc[mt_number_of_types];
|
||||
MemoryCounter _tracking_header;
|
||||
MemoryCounter _all_mallocs;
|
||||
|
||||
|
||||
public:
|
||||
@ -150,14 +150,20 @@ class MallocMemorySnapshot : public ResourceObj {
|
||||
return &_malloc[index];
|
||||
}
|
||||
|
||||
inline MemoryCounter* malloc_overhead() {
|
||||
return &_tracking_header;
|
||||
inline size_t malloc_overhead() const {
|
||||
return _all_mallocs.count() * sizeof(MallocHeader);
|
||||
}
|
||||
|
||||
// Total malloc invocation count
|
||||
size_t total_count() const;
|
||||
size_t total_count() const {
|
||||
return _all_mallocs.count();
|
||||
}
|
||||
|
||||
// Total malloc'd memory amount
|
||||
size_t total() const;
|
||||
size_t total() const {
|
||||
return _all_mallocs.size() + malloc_overhead() + total_arena();
|
||||
}
|
||||
|
||||
// Total malloc'd memory used by arenas
|
||||
size_t total_arena() const;
|
||||
|
||||
@ -171,7 +177,7 @@ class MallocMemorySnapshot : public ResourceObj {
|
||||
// copy is going on, because their size is adjusted using this
|
||||
// buffer in make_adjustment().
|
||||
ThreadCritical tc;
|
||||
s->_tracking_header = _tracking_header;
|
||||
s->_all_mallocs = _all_mallocs;
|
||||
for (int index = 0; index < mt_number_of_types; index ++) {
|
||||
s->_malloc[index] = _malloc[index];
|
||||
}
|
||||
@ -195,10 +201,12 @@ class MallocMemorySummary : AllStatic {
|
||||
|
||||
static inline void record_malloc(size_t size, MEMFLAGS flag) {
|
||||
as_snapshot()->by_type(flag)->record_malloc(size);
|
||||
as_snapshot()->_all_mallocs.allocate(size);
|
||||
}
|
||||
|
||||
static inline void record_free(size_t size, MEMFLAGS flag) {
|
||||
as_snapshot()->by_type(flag)->record_free(size);
|
||||
as_snapshot()->_all_mallocs.deallocate(size);
|
||||
}
|
||||
|
||||
static inline void record_new_arena(MEMFLAGS flag) {
|
||||
@ -218,18 +226,9 @@ class MallocMemorySummary : AllStatic {
|
||||
s->make_adjustment();
|
||||
}
|
||||
|
||||
// Record memory used by malloc tracking header
|
||||
static inline void record_new_malloc_header(size_t sz) {
|
||||
as_snapshot()->malloc_overhead()->allocate(sz);
|
||||
}
|
||||
|
||||
static inline void record_free_malloc_header(size_t sz) {
|
||||
as_snapshot()->malloc_overhead()->deallocate(sz);
|
||||
}
|
||||
|
||||
// The memory used by malloc tracking headers
|
||||
static inline size_t tracking_overhead() {
|
||||
return as_snapshot()->malloc_overhead()->size();
|
||||
return as_snapshot()->malloc_overhead();
|
||||
}
|
||||
|
||||
static MallocMemorySnapshot* as_snapshot() {
|
||||
|
@ -140,7 +140,7 @@ class MemBaseline {
|
||||
size_t malloc_tracking_overhead() const {
|
||||
assert(baseline_type() != Not_baselined, "Not yet baselined");
|
||||
MemBaseline* bl = const_cast<MemBaseline*>(this);
|
||||
return bl->_malloc_memory_snapshot.malloc_overhead()->size();
|
||||
return bl->_malloc_memory_snapshot.malloc_overhead();
|
||||
}
|
||||
|
||||
MallocMemory* malloc_memory(MEMFLAGS flag) {
|
||||
|
@ -157,8 +157,8 @@ void MemSummaryReporter::report_summary_of_type(MEMFLAGS flag,
|
||||
}
|
||||
} else if (flag == mtNMT) {
|
||||
// Count malloc headers in "NMT" category
|
||||
reserved_amount += _malloc_snapshot->malloc_overhead()->size();
|
||||
committed_amount += _malloc_snapshot->malloc_overhead()->size();
|
||||
reserved_amount += _malloc_snapshot->malloc_overhead();
|
||||
committed_amount += _malloc_snapshot->malloc_overhead();
|
||||
}
|
||||
|
||||
if (amount_in_current_scale(reserved_amount) > 0) {
|
||||
@ -210,9 +210,9 @@ void MemSummaryReporter::report_summary_of_type(MEMFLAGS flag,
|
||||
}
|
||||
|
||||
if (flag == mtNMT &&
|
||||
amount_in_current_scale(_malloc_snapshot->malloc_overhead()->size()) > 0) {
|
||||
amount_in_current_scale(_malloc_snapshot->malloc_overhead()) > 0) {
|
||||
out->print_cr("%27s (tracking overhead=" SIZE_FORMAT "%s)", " ",
|
||||
amount_in_current_scale(_malloc_snapshot->malloc_overhead()->size()), scale);
|
||||
amount_in_current_scale(_malloc_snapshot->malloc_overhead()), scale);
|
||||
} else if (flag == mtClass) {
|
||||
// Metadata information
|
||||
report_metadata(Metaspace::NonClassType);
|
||||
|
117
test/hotspot/gtest/nmt/test_nmt_totals.cpp
Normal file
117
test/hotspot/gtest/nmt/test_nmt_totals.cpp
Normal file
@ -0,0 +1,117 @@
|
||||
/*
|
||||
* Copyright (c) 2022 SAP SE. All rights reserved.
|
||||
* Copyright (c) 2022, Oracle and/or its affiliates. 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 "memory/allocation.hpp"
|
||||
#include "runtime/os.hpp"
|
||||
#include "services/mallocTracker.hpp"
|
||||
#include "services/memTracker.hpp"
|
||||
#include "unittest.hpp"
|
||||
|
||||
// convenience log. switch on if debugging tests. Don't use tty, plain stdio only.
|
||||
#define LOG(...) { printf(__VA_ARGS__); printf("\n"); fflush(stdout); }
|
||||
//#define LOG(...)
|
||||
|
||||
static size_t get_total_malloc_invocs() {
|
||||
return MallocMemorySummary::as_snapshot()->total_count();
|
||||
}
|
||||
|
||||
static size_t get_total_malloc_size() {
|
||||
return MallocMemorySummary::as_snapshot()->total();
|
||||
}
|
||||
|
||||
static size_t get_malloc_overhead() {
|
||||
return MallocMemorySummary::as_snapshot()->malloc_overhead();
|
||||
}
|
||||
|
||||
struct totals_t { size_t n; size_t s; size_t ovrh; };
|
||||
|
||||
static totals_t get_totals() {
|
||||
totals_t tot;
|
||||
tot.n = get_total_malloc_invocs();
|
||||
tot.s = get_total_malloc_size();
|
||||
tot.ovrh = get_malloc_overhead();
|
||||
return tot;
|
||||
}
|
||||
|
||||
// Concurrent code can malloc and free too, therefore we need to compare with a leeway factor
|
||||
#define compare_totals(t_real, t_expected) { \
|
||||
double leeway_factor = 0.33; \
|
||||
size_t leeway_n = (size_t)(((double)t_expected.n) * leeway_factor); \
|
||||
size_t leeway_s = (size_t)(((double)t_expected.s) * leeway_factor); \
|
||||
EXPECT_GE(t_real.n, t_expected.n - leeway_n); \
|
||||
EXPECT_LE(t_real.n, t_expected.n + leeway_n); \
|
||||
EXPECT_GE(t_real.s, t_expected.s - leeway_s); \
|
||||
EXPECT_LE(t_real.s, t_expected.s + leeway_s); \
|
||||
EXPECT_GE(t_real.ovrh, t_expected.ovrh - (leeway_n * sizeof(MallocHeader))); \
|
||||
EXPECT_LE(t_real.ovrh, t_expected.ovrh + (leeway_n * sizeof(MallocHeader))); \
|
||||
LOG("Deviation: n=" SSIZE_FORMAT ", s=" SSIZE_FORMAT ", ovrh=" SSIZE_FORMAT, \
|
||||
(ssize_t)t_real.n - (ssize_t)t_expected.n, \
|
||||
(ssize_t)t_real.s - (ssize_t)t_expected.s, \
|
||||
(ssize_t)t_real.ovrh - (ssize_t)t_expected.ovrh); \
|
||||
}
|
||||
|
||||
TEST_VM(NMTNumbers, totals) {
|
||||
|
||||
if (!MemTracker::enabled()) {
|
||||
// Skip test if NMT is disabled
|
||||
return;
|
||||
}
|
||||
|
||||
const totals_t t1 = get_totals();
|
||||
|
||||
LOG("t1: " SIZE_FORMAT " - " SIZE_FORMAT " - " SIZE_FORMAT, t1.n, t1.s, t1.ovrh);
|
||||
|
||||
static const int NUM_ALLOCS = 1024 * 16;
|
||||
static const int ALLOC_SIZE = 1024;
|
||||
|
||||
void* p[NUM_ALLOCS];
|
||||
for (int i = 0; i < NUM_ALLOCS; i ++) {
|
||||
// spread over categories
|
||||
int category = i % (mt_number_of_types - 1);
|
||||
p[i] = NEW_C_HEAP_ARRAY(char, ALLOC_SIZE, (MEMFLAGS)category);
|
||||
}
|
||||
|
||||
const totals_t t2 = get_totals();
|
||||
LOG("t2: " SIZE_FORMAT " - " SIZE_FORMAT " - " SIZE_FORMAT, t2.n, t2.s, t2.ovrh);
|
||||
|
||||
totals_t t2_expected;
|
||||
t2_expected.n = t1.n + NUM_ALLOCS;
|
||||
t2_expected.s = t1.s + ALLOC_SIZE * NUM_ALLOCS;
|
||||
t2_expected.ovrh = (t1.n + NUM_ALLOCS) * sizeof(MallocHeader);
|
||||
|
||||
LOG("t2 expected: " SIZE_FORMAT " - " SIZE_FORMAT " - " SIZE_FORMAT, t2_expected.n, t2_expected.s, t2_expected.ovrh);
|
||||
|
||||
compare_totals(t2, t2_expected);
|
||||
|
||||
for (int i = 0; i < NUM_ALLOCS; i ++) {
|
||||
os::free(p[i]);
|
||||
}
|
||||
|
||||
const totals_t t3 = get_totals();
|
||||
LOG("t3: " SIZE_FORMAT " - " SIZE_FORMAT " - " SIZE_FORMAT, t3.n, t3.s, t3.ovrh);
|
||||
|
||||
compare_totals(t3, t1);
|
||||
|
||||
}
|
Loading…
Reference in New Issue
Block a user