Skip to content

Commit 9160ef8

Browse files
committedMay 17, 2024
8332237: [nmt] Remove the need for ThreadStackTracker::track_as_vm()
Reviewed-by: jsjolen, azafari
1 parent 7c750fd commit 9160ef8

7 files changed

+57
-184
lines changed
 

‎src/hotspot/share/nmt/memBaseline.cpp

+1-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2012, 2023, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2012, 2024, 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
@@ -151,11 +151,6 @@ bool MemBaseline::baseline_allocation_sites() {
151151
return false;
152152
}
153153

154-
// Walk simple thread stacks
155-
if (!ThreadStackTracker::walk_simple_thread_stack_site(&malloc_walker)) {
156-
return false;
157-
}
158-
159154
_malloc_sites.move(malloc_walker.malloc_sites());
160155
// The malloc sites are collected in size order
161156
_malloc_sites_order = by_size;

‎src/hotspot/share/nmt/memReporter.cpp

+18-43
Original file line numberDiff line numberDiff line change
@@ -193,17 +193,10 @@ void MemSummaryReporter::report_summary_of_type(MEMFLAGS flag,
193193

194194
// Count thread's native stack in "Thread" category
195195
if (flag == mtThread) {
196-
if (ThreadStackTracker::track_as_vm()) {
197-
const VirtualMemory* thread_stack_usage =
198-
(const VirtualMemory*)_vm_snapshot->by_type(mtThreadStack);
199-
reserved_amount += thread_stack_usage->reserved();
200-
committed_amount += thread_stack_usage->committed();
201-
} else {
202-
const MallocMemory* thread_stack_usage =
203-
(const MallocMemory*)_malloc_snapshot->by_type(mtThreadStack);
204-
reserved_amount += thread_stack_usage->malloc_size();
205-
committed_amount += thread_stack_usage->malloc_size();
206-
}
196+
const VirtualMemory* thread_stack_usage =
197+
(const VirtualMemory*)_vm_snapshot->by_type(mtThreadStack);
198+
reserved_amount += thread_stack_usage->reserved();
199+
committed_amount += thread_stack_usage->committed();
207200
} else if (flag == mtNMT) {
208201
// Count malloc headers in "NMT" category
209202
reserved_amount += _malloc_snapshot->malloc_overhead();
@@ -240,21 +233,12 @@ void MemSummaryReporter::report_summary_of_type(MEMFLAGS flag,
240233
out->print_cr("%27s ( instance classes #" SIZE_FORMAT ", array classes #" SIZE_FORMAT ")",
241234
" ", _instance_class_count, _array_class_count);
242235
} else if (flag == mtThread) {
243-
if (ThreadStackTracker::track_as_vm()) {
244-
const VirtualMemory* thread_stack_usage =
245-
_vm_snapshot->by_type(mtThreadStack);
246-
// report thread count
247-
out->print_cr("%27s (threads #" SIZE_FORMAT ")", " ", ThreadStackTracker::thread_count());
248-
out->print("%27s (stack: ", " ");
249-
print_total(thread_stack_usage->reserved(), thread_stack_usage->committed(), thread_stack_usage->peak_size());
250-
} else {
251-
MallocMemory* thread_stack_memory = _malloc_snapshot->by_type(mtThreadStack);
252-
const char* scale = current_scale();
253-
// report thread count
254-
out->print_cr("%27s (threads #" SIZE_FORMAT ")", " ", thread_stack_memory->malloc_count());
255-
out->print("%27s (Stack: " SIZE_FORMAT "%s", " ",
256-
amount_in_current_scale(thread_stack_memory->malloc_size()), scale);
257-
}
236+
const VirtualMemory* thread_stack_usage =
237+
_vm_snapshot->by_type(mtThreadStack);
238+
// report thread count
239+
out->print_cr("%27s (threads #" SIZE_FORMAT ")", " ", ThreadStackTracker::thread_count());
240+
out->print("%27s (stack: ", " ");
241+
print_total(thread_stack_usage->reserved(), thread_stack_usage->committed(), thread_stack_usage->peak_size());
258242
out->print_cr(")");
259243
}
260244

@@ -627,24 +611,15 @@ void MemSummaryDiffReporter::diff_summary_of_type(MEMFLAGS flag,
627611
out->print_cr(")");
628612

629613
out->print("%27s (stack: ", " ");
630-
if (ThreadStackTracker::track_as_vm()) {
631-
// report thread stack
632-
const VirtualMemory* current_thread_stack =
633-
_current_baseline.virtual_memory(mtThreadStack);
634-
const VirtualMemory* early_thread_stack =
635-
_early_baseline.virtual_memory(mtThreadStack);
636-
637-
print_virtual_memory_diff(current_thread_stack->reserved(), current_thread_stack->committed(),
638-
early_thread_stack->reserved(), early_thread_stack->committed());
639-
} else {
640-
const MallocMemory* current_thread_stack =
641-
_current_baseline.malloc_memory(mtThreadStack);
642-
const MallocMemory* early_thread_stack =
643-
_early_baseline.malloc_memory(mtThreadStack);
614+
// report thread stack
615+
const VirtualMemory* current_thread_stack =
616+
_current_baseline.virtual_memory(mtThreadStack);
617+
const VirtualMemory* early_thread_stack =
618+
_early_baseline.virtual_memory(mtThreadStack);
619+
620+
print_virtual_memory_diff(current_thread_stack->reserved(), current_thread_stack->committed(),
621+
early_thread_stack->reserved(), early_thread_stack->committed());
644622

645-
print_malloc_diff(current_thread_stack->malloc_size(), current_thread_stack->malloc_count(),
646-
early_thread_stack->malloc_size(), early_thread_stack->malloc_count(), flag);
647-
}
648623
out->print_cr(")");
649624
}
650625

‎src/hotspot/share/nmt/memTracker.cpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -67,8 +67,7 @@ void MemTracker::initialize() {
6767

6868
if (level > NMT_off) {
6969
if (!MallocTracker::initialize(level) ||
70-
!VirtualMemoryTracker::initialize(level) ||
71-
!ThreadStackTracker::initialize(level)) {
70+
!VirtualMemoryTracker::initialize(level)) {
7271
assert(false, "NMT initialization failed");
7372
level = NMT_off;
7473
log_warning(nmt)("NMT initialization failed. NMT disabled.");

‎src/hotspot/share/nmt/nmtUsage.cpp

+6-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2022, 2024, 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
@@ -43,13 +43,11 @@ NMTUsage::NMTUsage(NMTUsageOptions options) :
4343
_usage_options(options) { }
4444

4545
void NMTUsage::walk_thread_stacks() {
46-
// If backed by virtual memory, snapping the thread stacks involves walking
47-
// them to to figure out how much memory is committed if they are backed by
48-
// virtual memory. This needs ot happen before we take the snapshot of the
49-
// virtual memory since it will update this information.
50-
if (ThreadStackTracker::track_as_vm()) {
51-
VirtualMemoryTracker::snapshot_thread_stacks();
52-
}
46+
// Snapping the thread stacks involves walking the areas to figure out how
47+
// much memory had been committed if they are backed by virtual memory. This
48+
// needs to happen before we take the snapshot of the virtual memory since it
49+
// will update this information.
50+
VirtualMemoryTracker::snapshot_thread_stacks();
5351
}
5452

5553
void NMTUsage::update_malloc_usage() {
+24-72
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
/*
2-
* Copyright (c) 2019, 2021, Red Hat, Inc. All rights reserved.
2+
* Copyright (c) 2019, 2024, Red Hat, Inc. All rights reserved.
3+
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
34
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
45
*
56
* This code is free software; you can redistribute it and/or modify it
@@ -24,95 +25,46 @@
2425

2526
#include "precompiled.hpp"
2627

27-
#include "nmt/mallocTracker.hpp"
2828
#include "nmt/memTracker.hpp"
2929
#include "nmt/threadStackTracker.hpp"
3030
#include "nmt/virtualMemoryTracker.hpp"
31+
#include "runtime/os.hpp"
3132
#include "runtime/threadCritical.hpp"
33+
#include "utilities/align.hpp"
34+
#include "utilities/debug.hpp"
35+
#include "utilities/globalDefinitions.hpp"
3236

3337
volatile size_t ThreadStackTracker::_thread_count = 0;
34-
SortedLinkedList<SimpleThreadStackSite, ThreadStackTracker::compare_thread_stack_base>* ThreadStackTracker::_simple_thread_stacks = nullptr;
3538

36-
bool ThreadStackTracker::initialize(NMT_TrackingLevel level) {
37-
if (level == NMT_detail && !track_as_vm()) {
38-
_simple_thread_stacks = new (std::nothrow, mtNMT)
39-
SortedLinkedList<SimpleThreadStackSite, ThreadStackTracker::compare_thread_stack_base>();
40-
return (_simple_thread_stacks != nullptr);
41-
}
42-
return true;
43-
}
44-
45-
int ThreadStackTracker::compare_thread_stack_base(const SimpleThreadStackSite& s1, const SimpleThreadStackSite& s2) {
46-
return primitive_compare(s1.base(), s2.base());
39+
static void align_thread_stack_boundaries_inward(void*& base, size_t& size) {
40+
// Thread stack boundaries don't have to be aligned to page boundaries. For cases where they
41+
// are not aligned (e.g. AIX, Alpine), this function corrects boundaries inward to the next
42+
// page boundaries. This ensures that we can track thread stacks piggybacking on the virtual
43+
// memory tracker.
44+
void* const base_aligned = align_up(base, os::vm_page_size());
45+
const size_t size_aligned = align_down(size, os::vm_page_size());
46+
assert(size_aligned > 0, "stack size less than a page?");
47+
base = base_aligned;
48+
size = size_aligned;
4749
}
4850

4951
void ThreadStackTracker::new_thread_stack(void* base, size_t size, const NativeCallStack& stack) {
50-
assert(MemTracker::tracking_level() >= NMT_summary, "Must be");
52+
assert(MemTracker::enabled(), "Must be");
5153
assert(base != nullptr, "Should have been filtered");
54+
align_thread_stack_boundaries_inward(base, size);
55+
5256
ThreadCritical tc;
53-
if (track_as_vm()) {
54-
VirtualMemoryTracker::add_reserved_region((address)base, size, stack, mtThreadStack);
55-
} else {
56-
// Use a slot in mallocMemorySummary for thread stack bookkeeping
57-
MallocMemorySummary::record_malloc(size, mtThreadStack);
58-
if (MemTracker::tracking_level() == NMT_detail) {
59-
assert(_simple_thread_stacks != nullptr, "Must be initialized");
60-
SimpleThreadStackSite site((address)base, size, stack);
61-
_simple_thread_stacks->add(site);
62-
}
63-
}
57+
VirtualMemoryTracker::add_reserved_region((address)base, size, stack, mtThreadStack);
6458
_thread_count++;
6559
}
6660

6761
void ThreadStackTracker::delete_thread_stack(void* base, size_t size) {
68-
assert(MemTracker::tracking_level() >= NMT_summary, "Must be");
62+
assert(MemTracker::enabled(), "Must be");
6963
assert(base != nullptr, "Should have been filtered");
64+
align_thread_stack_boundaries_inward(base, size);
65+
7066
ThreadCritical tc;
71-
if(track_as_vm()) {
72-
VirtualMemoryTracker::remove_released_region((address)base, size);
73-
} else {
74-
// Use a slot in mallocMemorySummary for thread stack bookkeeping
75-
MallocMemorySummary::record_free(size, mtThreadStack);
76-
if (MemTracker::tracking_level() == NMT_detail) {
77-
assert(_simple_thread_stacks != nullptr, "Must be initialized");
78-
SimpleThreadStackSite site((address)base, size, NativeCallStack::empty_stack()); // Fake object just to serve as compare target for delete
79-
bool removed = _simple_thread_stacks->remove(site);
80-
assert(removed, "Must exist");
81-
}
82-
}
67+
VirtualMemoryTracker::remove_released_region((address)base, size);
8368
_thread_count--;
8469
}
8570

86-
bool ThreadStackTracker::walk_simple_thread_stack_site(MallocSiteWalker* walker) {
87-
if (!track_as_vm()) {
88-
LinkedListImpl<MallocSite> _sites;
89-
{
90-
ThreadCritical tc;
91-
assert(_simple_thread_stacks != nullptr, "Must be initialized");
92-
LinkedListIterator<SimpleThreadStackSite> itr(_simple_thread_stacks->head());
93-
const SimpleThreadStackSite* ts = itr.next();
94-
// Consolidate sites and convert to MallocSites, so we can piggyback into
95-
// malloc snapshot
96-
while (ts != nullptr) {
97-
MallocSite site(*ts->call_stack(), mtThreadStack);
98-
MallocSite* exist = _sites.find(site);
99-
if (exist != nullptr) {
100-
exist->allocate(ts->size());
101-
} else {
102-
site.allocate(ts->size());
103-
_sites.add(site);
104-
}
105-
ts = itr.next();
106-
}
107-
}
108-
109-
// Piggyback to malloc snapshot
110-
LinkedListIterator<MallocSite> site_itr(_sites.head());
111-
const MallocSite* s = site_itr.next();
112-
while (s != nullptr) {
113-
walker->do_malloc_site(s);
114-
s = site_itr.next();
115-
}
116-
}
117-
return true;
118-
}

‎src/hotspot/share/nmt/threadStackTracker.hpp

+4-47
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
/*
2-
* Copyright (c) 2019, 2021, Red Hat, Inc. All rights reserved.
2+
* Copyright (c) 2019, 2024, Red Hat, Inc. All rights reserved.
3+
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
34
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
45
*
56
* This code is free software; you can redistribute it and/or modify it
@@ -25,61 +26,17 @@
2526
#ifndef SHARE_NMT_THREADSTACKTRACKER_HPP
2627
#define SHARE_NMT_THREADSTACKTRACKER_HPP
2728

28-
#include "nmt/allocationSite.hpp"
29-
#include "nmt/mallocSiteTable.hpp"
30-
#include "nmt/nmtCommon.hpp"
31-
#include "utilities/linkedlist.hpp"
29+
#include "memory/allStatic.hpp"
30+
#include "utilities/globalDefinitions.hpp"
3231
#include "utilities/nativeCallStack.hpp"
3332

34-
class SimpleThreadStackSite : public AllocationSite {
35-
const address _base;
36-
const size_t _size;
37-
public:
38-
SimpleThreadStackSite(address base, size_t size, const NativeCallStack& stack) :
39-
AllocationSite(stack, mtThreadStack),
40-
_base(base),
41-
_size(size) {}
42-
43-
bool equals(const SimpleThreadStackSite& mts) const {
44-
bool eq = base() == mts.base();
45-
assert(!eq || size() == mts.size(), "Must match");
46-
return eq;
47-
}
48-
49-
size_t size() const { return _size; }
50-
address base() const { return _base; }
51-
};
52-
53-
/*
54-
* Most of platforms, that hotspot support, have their thread stacks backed by
55-
* virtual memory by default. For these cases, thread stack tracker simply
56-
* delegates tracking to virtual memory tracker.
57-
* However, there are exceptions, (e.g. AIX), that platforms can provide stacks
58-
* that are not page aligned. A hypothetical VM implementation, it can provide
59-
* it own stacks. In these case, track_as_vm() should return false and manage
60-
* stack tracking by this tracker internally.
61-
* During memory snapshot, tracked thread stacks memory data is walked and stored
62-
* along with malloc'd data inside baseline. The regions are not scanned and assumed
63-
* all committed for now. Can add scanning phase when there is a need.
64-
*/
6533
class ThreadStackTracker : AllStatic {
6634
private:
6735
static volatile size_t _thread_count;
68-
69-
static int compare_thread_stack_base(const SimpleThreadStackSite& s1, const SimpleThreadStackSite& s2);
70-
static SortedLinkedList<SimpleThreadStackSite, compare_thread_stack_base>* _simple_thread_stacks;
7136
public:
72-
static bool initialize(NMT_TrackingLevel level);
73-
7437
static void new_thread_stack(void* base, size_t size, const NativeCallStack& stack);
7538
static void delete_thread_stack(void* base, size_t size);
76-
77-
static bool track_as_vm() { return AIX_ONLY(false) NOT_AIX(true); }
7839
static size_t thread_count() { return _thread_count; }
79-
80-
// Snapshot support. Piggyback thread stack data in malloc slot, NMT always handles
81-
// thread stack slot specially since beginning.
82-
static bool walk_simple_thread_stack_site(MallocSiteWalker* walker);
8340
};
8441

8542
#endif // SHARE_NMT_THREADSTACKTRACKER_HPP

‎src/hotspot/share/nmt/virtualMemoryTracker.cpp

+3-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2013, 2023, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2013, 2024, 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
@@ -47,11 +47,8 @@ void VirtualMemory::update_peak(size_t size) {
4747
}
4848

4949
void VirtualMemorySummary::snapshot(VirtualMemorySnapshot* s) {
50-
// Only if thread stack is backed by virtual memory
51-
if (ThreadStackTracker::track_as_vm()) {
52-
// Snapshot current thread stacks
53-
VirtualMemoryTracker::snapshot_thread_stacks();
54-
}
50+
// Snapshot current thread stacks
51+
VirtualMemoryTracker::snapshot_thread_stacks();
5552
as_snapshot()->copy_to(s);
5653
}
5754

0 commit comments

Comments
 (0)
Please sign in to comment.