1 /*
2 * Copyright (c) 2020, 2026, 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 #ifndef SHARE_CDS_DUMPALLOCSTATS_HPP
26 #define SHARE_CDS_DUMPALLOCSTATS_HPP
27
28 #include "classfile/compactHashtable.hpp"
29 #include "memory/allocation.hpp"
30 #include "memory/metaspaceClosureType.hpp"
31
32 // This is for dumping detailed statistics for the allocations
33 // in the shared spaces.
34 class DumpAllocStats : public StackObj {
35 public:
36
37 #define DUMPED_OBJ_TYPES_DO(f) \
38 METASPACE_CLOSURE_TYPES_DO(f) \
39 f(SymbolHashentry) \
40 f(SymbolBucket) \
41 f(StringHashentry) \
42 f(StringBucket) \
43 f(CppVTables) \
44 f(Gap) \
45 f(Other)
46
47 #define DUMPED_TYPE_DECLARE(name) name ## Type,
48 #define DUMPED_TYPE_NAME_CASE(name) case name ## Type: return #name;
49
50 enum Type {
51 // Types are MetaspaceObj::ClassType, MetaspaceObj::SymbolType, etc
52 DUMPED_OBJ_TYPES_DO(DUMPED_TYPE_DECLARE)
53 _number_of_types
54 };
55
56 static const char* type_name(Type type) {
57 switch(type) {
58 DUMPED_OBJ_TYPES_DO(DUMPED_TYPE_NAME_CASE)
59 default:
60 ShouldNotReachHere();
61 return nullptr;
62 }
63 }
64
65 CompactHashtableStats _symbol_stats;
66 CompactHashtableStats _string_stats;
67
68 int _counts[2][_number_of_types];
69 int _bytes [2][_number_of_types];
70
71 int _num_field_cp_entries;
72 int _num_field_cp_entries_archived;
73 int _num_field_cp_entries_reverted;
74 int _num_indy_cp_entries;
75 int _num_indy_cp_entries_archived;
76 int _num_indy_cp_entries_reverted;
77 int _num_klass_cp_entries;
78 int _num_klass_cp_entries_archived;
79 int _num_klass_cp_entries_reverted;
80 int _num_method_cp_entries;
81 int _num_method_cp_entries_archived;
82 int _num_method_cp_entries_reverted;
83 int _num_dynamic_proxy_classes;
84
85 public:
86 enum { RO = 0, RW = 1 };
87
88 DumpAllocStats() {
89 memset(_counts, 0, sizeof(_counts));
90 memset(_bytes, 0, sizeof(_bytes));
91 _num_field_cp_entries = 0;
92 _num_field_cp_entries_archived = 0;
93 _num_field_cp_entries_reverted = 0;
94 _num_indy_cp_entries = 0;
95 _num_indy_cp_entries_archived = 0;
96 _num_indy_cp_entries_reverted = 0;
97 _num_klass_cp_entries = 0;
98 _num_klass_cp_entries_archived = 0;
99 _num_klass_cp_entries_reverted = 0;
100 _num_method_cp_entries = 0;
101 _num_method_cp_entries_archived = 0;
102 _num_method_cp_entries_reverted = 0;
103 _num_dynamic_proxy_classes = 0;
104 };
105
106 CompactHashtableStats* symbol_stats() { return &_symbol_stats; }
107 CompactHashtableStats* string_stats() { return &_string_stats; }
108
109 void record(MetaspaceClosureType type, int byte_size, bool read_only) {
110 int t = (int)type;
111 assert(t >= 0 && t < (int)MetaspaceClosureType::_number_of_types, "sanity");
112 int which = (read_only) ? RO : RW;
113 _counts[which][t] ++;
114 _bytes [which][t] += byte_size;
115 }
116
117 void record_gap(int byte_size) {
118 _counts[RW][GapType] += 1;
119 _bytes [RW][GapType] += byte_size;
120 }
121
122 void record_other_type(int byte_size, bool read_only) {
123 int which = (read_only) ? RO : RW;
124 _counts[which][OtherType] += 1;
125 _bytes [which][OtherType] += byte_size;
126 }
127
128 void record_cpp_vtables(int byte_size) {
129 _counts[RW][CppVTablesType] += 1;
130 _bytes[RW][CppVTablesType] += byte_size;
131 }
132
133 void record_field_cp_entry(bool archived, bool reverted) {
134 _num_field_cp_entries ++;
135 _num_field_cp_entries_archived += archived ? 1 : 0;
136 _num_field_cp_entries_reverted += reverted ? 1 : 0;
137 }
138
139 void record_indy_cp_entry(bool archived, bool reverted) {
140 _num_indy_cp_entries ++;
141 _num_indy_cp_entries_archived += archived ? 1 : 0;
142 _num_indy_cp_entries_reverted += reverted ? 1 : 0;
143 }
144
145 void record_klass_cp_entry(bool archived, bool reverted) {
146 _num_klass_cp_entries ++;
147 _num_klass_cp_entries_archived += archived ? 1 : 0;
148 _num_klass_cp_entries_reverted += reverted ? 1 : 0;
149 }
150
151 void record_method_cp_entry(bool archived, bool reverted) {
152 _num_method_cp_entries ++;
153 _num_method_cp_entries_archived += archived ? 1 : 0;
154 _num_method_cp_entries_reverted += reverted ? 1 : 0;
155 }
156
157 void record_dynamic_proxy_class() {
158 _num_dynamic_proxy_classes ++;
159 }
160
161 void print_stats(int ro_all, int rw_all);
162 };
163
164 #endif // SHARE_CDS_DUMPALLOCSTATS_HPP