Skip to content

Commit 4a1fcb6

Browse files
author
Doug Simon
committedJul 8, 2023
8193513: add support for printing a stack trace on class loading
Reviewed-by: dholmes, iklam, coleenp
1 parent 292ee63 commit 4a1fcb6

File tree

6 files changed

+110
-5
lines changed

6 files changed

+110
-5
lines changed
 

‎src/hotspot/share/logging/logTag.hpp

+2
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,7 @@ class outputStream;
4545
LOG_TAG(bot) \
4646
LOG_TAG(breakpoint) \
4747
LOG_TAG(bytecode) \
48+
LOG_TAG(cause) \
4849
LOG_TAG(cds) \
4950
LOG_TAG(census) \
5051
LOG_TAG(class) \
@@ -124,6 +125,7 @@ class outputStream;
124125
LOG_TAG(module) \
125126
LOG_TAG(monitorinflation) \
126127
LOG_TAG(monitormismatch) \
128+
LOG_TAG(native) \
127129
LOG_TAG(nestmates) \
128130
LOG_TAG(nmethod) \
129131
LOG_TAG(nmt) \

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

+75-5
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,7 @@
8383
#include "runtime/javaThread.inline.hpp"
8484
#include "runtime/mutexLocker.hpp"
8585
#include "runtime/orderAccess.hpp"
86+
#include "runtime/os.inline.hpp"
8687
#include "runtime/reflectionUtils.hpp"
8788
#include "runtime/threads.hpp"
8889
#include "services/classLoadingService.hpp"
@@ -3786,10 +3787,19 @@ const char* InstanceKlass::internal_name() const {
37863787
void InstanceKlass::print_class_load_logging(ClassLoaderData* loader_data,
37873788
const ModuleEntry* module_entry,
37883789
const ClassFileStream* cfs) const {
3790+
37893791
if (ClassListWriter::is_enabled()) {
37903792
ClassListWriter::write(this, cfs);
37913793
}
37923794

3795+
print_class_load_helper(loader_data, module_entry, cfs);
3796+
print_class_load_cause_logging();
3797+
}
3798+
3799+
void InstanceKlass::print_class_load_helper(ClassLoaderData* loader_data,
3800+
const ModuleEntry* module_entry,
3801+
const ClassFileStream* cfs) const {
3802+
37933803
if (!log_is_enabled(Info, class, load)) {
37943804
return;
37953805
}
@@ -3847,15 +3857,15 @@ void InstanceKlass::print_class_load_logging(ClassLoaderData* loader_data,
38473857

38483858
// Class hierarchy info
38493859
debug_stream.print(" klass: " PTR_FORMAT " super: " PTR_FORMAT,
3850-
p2i(this), p2i(superklass()));
3860+
p2i(this), p2i(superklass()));
38513861

38523862
// Interfaces
38533863
if (local_interfaces() != nullptr && local_interfaces()->length() > 0) {
38543864
debug_stream.print(" interfaces:");
38553865
int length = local_interfaces()->length();
38563866
for (int i = 0; i < length; i++) {
38573867
debug_stream.print(" " PTR_FORMAT,
3858-
p2i(InstanceKlass::cast(local_interfaces()->at(i))));
3868+
p2i(InstanceKlass::cast(local_interfaces()->at(i))));
38593869
}
38603870
}
38613871

@@ -3867,15 +3877,75 @@ void InstanceKlass::print_class_load_logging(ClassLoaderData* loader_data,
38673877
// Classfile checksum
38683878
if (cfs) {
38693879
debug_stream.print(" bytes: %d checksum: %08x",
3870-
cfs->length(),
3871-
ClassLoader::crc32(0, (const char*)cfs->buffer(),
3872-
cfs->length()));
3880+
cfs->length(),
3881+
ClassLoader::crc32(0, (const char*)cfs->buffer(),
3882+
cfs->length()));
38733883
}
38743884

38753885
msg.debug("%s", debug_stream.as_string());
38763886
}
38773887
}
38783888

3889+
void InstanceKlass::print_class_load_cause_logging() const {
3890+
bool log_cause_native = log_is_enabled(Info, class, load, cause, native);
3891+
if (log_cause_native || log_is_enabled(Info, class, load, cause)) {
3892+
JavaThread* current = JavaThread::current();
3893+
ResourceMark rm(current);
3894+
const char* name = external_name();
3895+
3896+
if (LogClassLoadingCauseFor == nullptr ||
3897+
(strcmp("*", LogClassLoadingCauseFor) != 0 &&
3898+
strstr(name, LogClassLoadingCauseFor) == nullptr)) {
3899+
return;
3900+
}
3901+
3902+
// Log Java stack first
3903+
{
3904+
LogMessage(class, load, cause) msg;
3905+
NonInterleavingLogStream info_stream{LogLevelType::Info, msg};
3906+
3907+
info_stream.print_cr("Java stack when loading %s:", name);
3908+
current->print_stack_on(&info_stream);
3909+
}
3910+
3911+
// Log native stack second
3912+
if (log_cause_native) {
3913+
// Log to string first so that lines can be indented
3914+
stringStream stack_stream;
3915+
char buf[O_BUFLEN];
3916+
address lastpc = nullptr;
3917+
if (os::platform_print_native_stack(&stack_stream, nullptr, buf, O_BUFLEN, lastpc)) {
3918+
// We have printed the native stack in platform-specific code,
3919+
// so nothing else to do in this case.
3920+
} else {
3921+
frame f = os::current_frame();
3922+
VMError::print_native_stack(&stack_stream, f, current, true /*print_source_info */,
3923+
-1 /* max stack_stream */, buf, O_BUFLEN);
3924+
}
3925+
3926+
LogMessage(class, load, cause, native) msg;
3927+
NonInterleavingLogStream info_stream{LogLevelType::Info, msg};
3928+
info_stream.print_cr("Native stack when loading %s:", name);
3929+
3930+
// Print each native stack line to the log
3931+
int size = (int) stack_stream.size();
3932+
char* stack = stack_stream.as_string();
3933+
char* stack_end = stack + size;
3934+
char* line_start = stack;
3935+
for (char* p = stack; p < stack_end; p++) {
3936+
if (*p == '\n') {
3937+
*p = '\0';
3938+
info_stream.print_cr("\t%s", line_start);
3939+
line_start = p + 1;
3940+
}
3941+
}
3942+
if (line_start < stack_end) {
3943+
info_stream.print_cr("\t%s", line_start);
3944+
}
3945+
}
3946+
}
3947+
}
3948+
38793949
// Verification
38803950

38813951
class VerifyFieldClosure: public BasicOopIterateClosure {

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

+5
Original file line numberDiff line numberDiff line change
@@ -1164,6 +1164,11 @@ class InstanceKlass: public Klass {
11641164
void print_class_load_logging(ClassLoaderData* loader_data,
11651165
const ModuleEntry* module_entry,
11661166
const ClassFileStream* cfs) const;
1167+
private:
1168+
void print_class_load_cause_logging() const;
1169+
void print_class_load_helper(ClassLoaderData* loader_data,
1170+
const ModuleEntry* module_entry,
1171+
const ClassFileStream* cfs) const;
11671172
};
11681173

11691174
// for adding methods

‎src/hotspot/share/runtime/arguments.cpp

+6
Original file line numberDiff line numberDiff line change
@@ -3961,6 +3961,12 @@ jint Arguments::parse(const JavaVMInitArgs* initial_cmd_args) {
39613961
warning("dependency logging results may be inflated by VerifyDependencies");
39623962
}
39633963

3964+
bool log_class_load_cause = log_is_enabled(Info, class, load, cause, native) ||
3965+
log_is_enabled(Info, class, load, cause);
3966+
if (log_class_load_cause && LogClassLoadingCauseFor == nullptr) {
3967+
warning("class load cause logging will not produce output without LogClassLoadingCauseFor");
3968+
}
3969+
39643970
apply_debugger_ergo();
39653971

39663972
if (log_is_enabled(Info, arguments)) {

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

+5
Original file line numberDiff line numberDiff line change
@@ -933,6 +933,11 @@ const int ObjectAlignmentInBytes = 8;
933933
product(bool, TraceCompilerThreads, false, DIAGNOSTIC, \
934934
"Trace creation and removal of compiler threads") \
935935
\
936+
product(ccstr, LogClassLoadingCauseFor, nullptr, \
937+
"Apply -Xlog:class+load+cause* to classes whose fully " \
938+
"qualified name contains this string (\"*\" matches " \
939+
"any class).") \
940+
\
936941
develop(bool, InjectCompilerCreationFailure, false, \
937942
"Inject thread creation failures for " \
938943
"UseDynamicNumberOfCompilerThreads") \

‎test/hotspot/jtreg/runtime/logging/ClassLoadUnloadTest.java

+17
Original file line numberDiff line numberDiff line change
@@ -118,5 +118,22 @@ public static void main(String... args) throws Exception {
118118
pb = exec("-Xlog:class+loader+data=trace");
119119
checkFor("[class,loader,data]", "create loader data");
120120

121+
// -Xlog:class+load+cause
122+
pb = exec("-Xlog:class+load+cause");
123+
checkAbsent("[class,load,cause]");
124+
checkFor("class load cause logging will not produce output without LogClassLoadingCauseFor");
125+
126+
// -Xlog:class+load+cause -XX:LogClassLoadingCauseFor=java.lang.StringCoding
127+
pb = exec("-Xlog:class+load+cause", "-XX:LogClassLoadingCauseFor=java.lang.StringCoding");
128+
checkFor("[class,load,cause]", "Java stack when loading java.lang.StringCoding:");
129+
130+
// -Xlog:class+load+cause -XX:LogClassLoadingCauseFor=java.lang.StringCoding
131+
pb = exec("-Xlog:class+load+cause+native", "-XX:LogClassLoadingCauseFor=java.lang.StringCoding");
132+
checkFor("[class,load,cause,native]", "Native stack when loading java.lang.StringCoding:");
133+
134+
// -Xlog:class+load+cause* -XX:LogClassLoadingCauseFor=java.lang.StringCoding
135+
pb = exec("-Xlog:class+load+cause*", "-XX:LogClassLoadingCauseFor=java.lang.StringCoding");
136+
checkFor("[class,load,cause] Java stack when loading java.lang.StringCoding:");
137+
checkFor("[class,load,cause,native] Native stack when loading java.lang.StringCoding:");
121138
}
122139
}

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Jul 8, 2023

@openjdk-notifier[bot]
Please sign in to comment.