Skip to content

Commit 3f0684d

Browse files
committedMar 8, 2022
8275775: Add jcmd VM.classes to print details of all classes
Reviewed-by: dholmes, iklam, stuefe
1 parent cde923d commit 3f0684d

File tree

6 files changed

+172
-1
lines changed

6 files changed

+172
-1
lines changed
 

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

+51-1
Original file line numberDiff line numberDiff line change
@@ -2074,6 +2074,52 @@ Method* InstanceKlass::lookup_method_in_all_interfaces(Symbol* name,
20742074
return NULL;
20752075
}
20762076

2077+
PrintClassClosure::PrintClassClosure(outputStream* st, bool verbose)
2078+
:_st(st), _verbose(verbose) {
2079+
ResourceMark rm;
2080+
_st->print("%-18s ", "KlassAddr");
2081+
_st->print("%-4s ", "Size");
2082+
_st->print("%-20s ", "State");
2083+
_st->print("%-7s ", "Flags");
2084+
_st->print("%-5s ", "ClassName");
2085+
_st->cr();
2086+
}
2087+
2088+
void PrintClassClosure::do_klass(Klass* k) {
2089+
ResourceMark rm;
2090+
// klass pointer
2091+
_st->print(INTPTR_FORMAT " ", p2i(k));
2092+
// klass size
2093+
_st->print("%4d ", k->size());
2094+
// initialization state
2095+
if (k->is_instance_klass()) {
2096+
_st->print("%-20s ",InstanceKlass::cast(k)->init_state_name());
2097+
} else {
2098+
_st->print("%-20s ","");
2099+
}
2100+
// misc flags(Changes should synced with ClassesDCmd::ClassesDCmd help doc)
2101+
char buf[10];
2102+
int i = 0;
2103+
if (k->has_finalizer()) buf[i++] = 'F';
2104+
if (k->has_final_method()) buf[i++] = 'f';
2105+
if (k->is_instance_klass()) {
2106+
InstanceKlass* ik = InstanceKlass::cast(k);
2107+
if (ik->is_rewritten()) buf[i++] = 'W';
2108+
if (ik->is_contended()) buf[i++] = 'C';
2109+
if (ik->has_been_redefined()) buf[i++] = 'R';
2110+
if (ik->is_shared()) buf[i++] = 'S';
2111+
}
2112+
buf[i++] = '\0';
2113+
_st->print("%-7s ", buf);
2114+
// klass name
2115+
_st->print("%-5s ", k->external_name());
2116+
// end
2117+
_st->cr();
2118+
if (_verbose) {
2119+
k->print_on(_st);
2120+
}
2121+
}
2122+
20772123
/* jni_id_for for jfieldIds only */
20782124
JNIid* InstanceKlass::jni_id_for(int offset) {
20792125
MutexLocker ml(JfieldIdCreation_lock);
@@ -3393,14 +3439,18 @@ static void print_vtable(vtableEntry* start, int len, outputStream* st) {
33933439
return print_vtable(reinterpret_cast<intptr_t*>(start), len, st);
33943440
}
33953441

3442+
const char* InstanceKlass::init_state_name() const {
3443+
return state_names[_init_state];
3444+
}
3445+
33963446
void InstanceKlass::print_on(outputStream* st) const {
33973447
assert(is_klass(), "must be klass");
33983448
Klass::print_on(st);
33993449

34003450
st->print(BULLET"instance size: %d", size_helper()); st->cr();
34013451
st->print(BULLET"klass size: %d", size()); st->cr();
34023452
st->print(BULLET"access: "); access_flags().print_on(st); st->cr();
3403-
st->print(BULLET"state: "); st->print_cr("%s", state_names[_init_state]);
3453+
st->print(BULLET"state: "); st->print_cr("%s", init_state_name());
34043454
st->print(BULLET"name: "); name()->print_value_on(st); st->cr();
34053455
st->print(BULLET"super: "); Metadata::print_value_on_maybe_null(st, super()); st->cr();
34063456
st->print(BULLET"sub: ");

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

+10
Original file line numberDiff line numberDiff line change
@@ -547,6 +547,7 @@ class InstanceKlass: public Klass {
547547
bool is_in_error_state() const { return _init_state == initialization_error; }
548548
bool is_reentrant_initialization(Thread *thread) { return thread == _init_thread; }
549549
ClassState init_state() { return (ClassState)_init_state; }
550+
const char* init_state_name() const;
550551
bool is_rewritten() const { return (_misc_flags & _misc_rewritten) != 0; }
551552

552553
// is this a sealed class
@@ -1277,6 +1278,15 @@ inline u2 InstanceKlass::next_method_idnum() {
12771278
}
12781279
}
12791280

1281+
class PrintClassClosure : public KlassClosure {
1282+
private:
1283+
outputStream* _st;
1284+
bool _verbose;
1285+
public:
1286+
PrintClassClosure(outputStream* st, bool verbose);
1287+
1288+
void do_klass(Klass* k);
1289+
};
12801290

12811291
/* JNIid class for jfieldIDs only */
12821292
class JNIid: public CHeapObj<mtClass> {

‎src/hotspot/share/runtime/vmOperation.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -95,6 +95,7 @@
9595
template(CleanClassLoaderDataMetaspaces) \
9696
template(PrintCompileQueue) \
9797
template(PrintClassHierarchy) \
98+
template(PrintClasses) \
9899
template(ICBufferFull) \
99100
template(PrintMetadata) \
100101
template(GTestExecuteAtSafepoint) \

‎src/hotspot/share/services/diagnosticCommand.cpp

+37
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
#include "jvm.h"
2727
#include "classfile/classLoaderHierarchyDCmd.hpp"
2828
#include "classfile/classLoaderStats.hpp"
29+
#include "classfile/classLoaderDataGraph.hpp"
2930
#include "classfile/javaClasses.hpp"
3031
#include "classfile/systemDictionary.hpp"
3132
#include "classfile/vmClasses.hpp"
@@ -102,6 +103,7 @@ void DCmdRegistrant::register_dcmds(){
102103
DCmdFactory::register_DCmdFactory(new DCmdFactoryImpl<ClassHistogramDCmd>(full_export, true, false));
103104
DCmdFactory::register_DCmdFactory(new DCmdFactoryImpl<SystemDictionaryDCmd>(full_export, true, false));
104105
DCmdFactory::register_DCmdFactory(new DCmdFactoryImpl<ClassHierarchyDCmd>(full_export, true, false));
106+
DCmdFactory::register_DCmdFactory(new DCmdFactoryImpl<ClassesDCmd>(full_export, true, false));
105107
DCmdFactory::register_DCmdFactory(new DCmdFactoryImpl<SymboltableDCmd>(full_export, true, false));
106108
DCmdFactory::register_DCmdFactory(new DCmdFactoryImpl<StringtableDCmd>(full_export, true, false));
107109
DCmdFactory::register_DCmdFactory(new DCmdFactoryImpl<metaspace::MetaspaceDCmd>(full_export, true, false));
@@ -954,6 +956,41 @@ void TouchedMethodsDCmd::execute(DCmdSource source, TRAPS) {
954956
VMThread::execute(&dumper);
955957
}
956958

959+
ClassesDCmd::ClassesDCmd(outputStream* output, bool heap) :
960+
DCmdWithParser(output, heap),
961+
_verbose("-verbose",
962+
"Dump the detailed content of a Java class. "
963+
"Some classes are annotated with flags: "
964+
"F = has, or inherits, a non-empty finalize method, "
965+
"f = has final method, "
966+
"W = methods rewritten, "
967+
"C = marked with @Contended annotation, "
968+
"R = has been redefined, "
969+
"S = is shared class",
970+
"BOOLEAN", false, "false") {
971+
_dcmdparser.add_dcmd_option(&_verbose);
972+
}
973+
974+
class VM_PrintClasses : public VM_Operation {
975+
private:
976+
outputStream* _out;
977+
bool _verbose;
978+
public:
979+
VM_PrintClasses(outputStream* out, bool verbose) : _out(out), _verbose(verbose) {}
980+
981+
virtual VMOp_Type type() const { return VMOp_PrintClasses; }
982+
983+
virtual void doit() {
984+
PrintClassClosure closure(_out, _verbose);
985+
ClassLoaderDataGraph::classes_do(&closure);
986+
}
987+
};
988+
989+
void ClassesDCmd::execute(DCmdSource source, TRAPS) {
990+
VM_PrintClasses vmop(output(), _verbose.is_set());
991+
VMThread::execute(&vmop);
992+
}
993+
957994
#if INCLUDE_CDS
958995
DumpSharedArchiveDCmd::DumpSharedArchiveDCmd(outputStream* output, bool heap) :
959996
DCmdWithParser(output, heap),

‎src/hotspot/share/services/diagnosticCommand.hpp

+22
Original file line numberDiff line numberDiff line change
@@ -858,6 +858,28 @@ class SystemDictionaryDCmd : public DCmdWithParser {
858858
virtual void execute(DCmdSource source, TRAPS);
859859
};
860860

861+
class ClassesDCmd : public DCmdWithParser {
862+
protected:
863+
DCmdArgument<bool> _verbose;
864+
public:
865+
ClassesDCmd(outputStream* output, bool heap);
866+
static const char* name() {
867+
return "VM.classes";
868+
}
869+
static const char* description() {
870+
return "Print all loaded classes";
871+
}
872+
static const char* impact() {
873+
return "Medium: Depends on number of loaded classes.";
874+
}
875+
static const JavaPermission permission() {
876+
JavaPermission p = {"java.lang.management.ManagementPermission",
877+
"monitor", NULL};
878+
return p;
879+
}
880+
virtual void execute(DCmdSource source, TRAPS);
881+
};
882+
861883
#if INCLUDE_JVMTI
862884
class DebugOnCmdStartDCmd : public DCmd {
863885
public:
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
/*
2+
* Copyright (c) 2022, Alibaba Group Holding Limited. 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+
/*
26+
* @test
27+
* @bug 8275775
28+
* @summary Test jcmd VM.classes
29+
* @library /test/lib
30+
* @run main/othervm PrintClasses
31+
*/
32+
33+
import jdk.test.lib.process.OutputAnalyzer;
34+
import jdk.test.lib.JDKToolFinder;
35+
36+
public class PrintClasses {
37+
public static void main(String args[]) throws Exception {
38+
var pid = Long.toString(ProcessHandle.current().pid());
39+
var pb = new ProcessBuilder();
40+
41+
pb.command(new String[] { JDKToolFinder.getJDKTool("jcmd"), pid, "VM.classes"});
42+
var output = new OutputAnalyzer(pb.start());
43+
output.shouldNotContain("instance size");
44+
output.shouldContain(PrintClasses.class.getSimpleName());
45+
46+
pb.command(new String[] { JDKToolFinder.getJDKTool("jcmd"), pid, "VM.classes", "-verbose"});
47+
output = new OutputAnalyzer(pb.start());
48+
output.shouldContain("instance size");
49+
output.shouldContain(PrintClasses.class.getSimpleName());
50+
}
51+
}

0 commit comments

Comments
 (0)
Please sign in to comment.