Skip to content

Commit faa8bde

Browse files
author
Markus Grönlund
committedNov 2, 2023
8319206: [REDO] Event NativeLibraryLoad breaks invariant by taking a stacktrace when thread is in state _thread_in_native
Reviewed-by: dholmes, egahlin
1 parent 4f808c6 commit faa8bde

File tree

10 files changed

+281
-152
lines changed

10 files changed

+281
-152
lines changed
 

‎src/hotspot/os/aix/os_aix.cpp

+5-20
Original file line numberDiff line numberDiff line change
@@ -81,7 +81,7 @@
8181
#include "utilities/growableArray.hpp"
8282
#include "utilities/vmError.hpp"
8383
#if INCLUDE_JFR
84-
#include "jfr/jfrEvents.hpp"
84+
#include "jfr/support/jfrNativeLibraryLoadEvent.hpp"
8585
#endif
8686

8787
// put OS-includes here (sorted alphabetically)
@@ -1118,11 +1118,6 @@ void *os::dll_load(const char *filename, char *ebuf, int ebuflen) {
11181118
return nullptr;
11191119
}
11201120

1121-
#if INCLUDE_JFR
1122-
EventNativeLibraryLoad event;
1123-
event.set_name(filename);
1124-
#endif
1125-
11261121
// RTLD_LAZY has currently the same behavior as RTLD_NOW
11271122
// The dl is loaded immediately with all its dependants.
11281123
int dflags = RTLD_LAZY;
@@ -1133,19 +1128,14 @@ void *os::dll_load(const char *filename, char *ebuf, int ebuflen) {
11331128
dflags |= RTLD_MEMBER;
11341129
}
11351130

1136-
void * result= ::dlopen(filename, dflags);
1131+
void* result;
1132+
JFR_ONLY(NativeLibraryLoadEvent load_event(filename, &result);)
1133+
result = ::dlopen(filename, dflags);
11371134
if (result != nullptr) {
11381135
Events::log_dll_message(nullptr, "Loaded shared library %s", filename);
11391136
// Reload dll cache. Don't do this in signal handling.
11401137
LoadedLibraries::reload();
11411138
log_info(os)("shared library load of %s was successful", filename);
1142-
1143-
#if INCLUDE_JFR
1144-
event.set_success(true);
1145-
event.set_errorMessage(nullptr);
1146-
event.commit();
1147-
#endif
1148-
11491139
return result;
11501140
} else {
11511141
// error analysis when dlopen fails
@@ -1159,12 +1149,7 @@ void *os::dll_load(const char *filename, char *ebuf, int ebuflen) {
11591149
}
11601150
Events::log_dll_message(nullptr, "Loading shared library %s failed, %s", filename, error_report);
11611151
log_info(os)("shared library load of %s failed, %s", filename, error_report);
1162-
1163-
#if INCLUDE_JFR
1164-
event.set_success(false);
1165-
event.set_errorMessage(error_report);
1166-
event.commit();
1167-
#endif
1152+
JFR_ONLY(load_event.set_error_msg(error_report);)
11681153
}
11691154
return nullptr;
11701155
}

‎src/hotspot/os/bsd/os_bsd.cpp

+9-32
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,7 @@
7171
#include "utilities/vmError.hpp"
7272
#if INCLUDE_JFR
7373
#include "jfr/jfrEvents.hpp"
74+
#include "jfr/support/jfrNativeLibraryLoadEvent.hpp"
7475
#endif
7576

7677
// put OS-includes here
@@ -981,21 +982,13 @@ void * os::dll_load(const char *filename, char *ebuf, int ebuflen) {
981982
#else
982983
log_info(os)("attempting shared library load of %s", filename);
983984

984-
#if INCLUDE_JFR
985-
EventNativeLibraryLoad event;
986-
event.set_name(filename);
987-
#endif
988-
989-
void * result= ::dlopen(filename, RTLD_LAZY);
985+
void* result;
986+
JFR_ONLY(NativeLibraryLoadEvent load_event(filename, &result);)
987+
result = ::dlopen(filename, RTLD_LAZY);
990988
if (result != nullptr) {
991989
Events::log_dll_message(nullptr, "Loaded shared library %s", filename);
992990
// Successful loading
993991
log_info(os)("shared library load of %s was successful", filename);
994-
#if INCLUDE_JFR
995-
event.set_success(true);
996-
event.set_errorMessage(nullptr);
997-
event.commit();
998-
#endif
999992
return result;
1000993
}
1001994

@@ -1010,11 +1003,7 @@ void * os::dll_load(const char *filename, char *ebuf, int ebuflen) {
10101003
}
10111004
Events::log_dll_message(nullptr, "Loading shared library %s failed, %s", filename, error_report);
10121005
log_info(os)("shared library load of %s failed, %s", filename, error_report);
1013-
#if INCLUDE_JFR
1014-
event.set_success(false);
1015-
event.set_errorMessage(error_report);
1016-
event.commit();
1017-
#endif
1006+
JFR_ONLY(load_event.set_error_msg(error_report);)
10181007

10191008
return nullptr;
10201009
#endif // STATIC_BUILD
@@ -1026,21 +1015,13 @@ void * os::dll_load(const char *filename, char *ebuf, int ebuflen) {
10261015
#else
10271016
log_info(os)("attempting shared library load of %s", filename);
10281017

1029-
#if INCLUDE_JFR
1030-
EventNativeLibraryLoad event;
1031-
event.set_name(filename);
1032-
#endif
1033-
1034-
void * result= ::dlopen(filename, RTLD_LAZY);
1018+
void* result;
1019+
JFR_ONLY(NativeLibraryLoadEvent load_event(filename, &result);)
1020+
result = ::dlopen(filename, RTLD_LAZY);
10351021
if (result != nullptr) {
10361022
Events::log_dll_message(nullptr, "Loaded shared library %s", filename);
10371023
// Successful loading
10381024
log_info(os)("shared library load of %s was successful", filename);
1039-
#if INCLUDE_JFR
1040-
event.set_success(true);
1041-
event.set_errorMessage(nullptr);
1042-
event.commit();
1043-
#endif
10441025
return result;
10451026
}
10461027

@@ -1057,11 +1038,7 @@ void * os::dll_load(const char *filename, char *ebuf, int ebuflen) {
10571038
}
10581039
Events::log_dll_message(nullptr, "Loading shared library %s failed, %s", filename, error_report);
10591040
log_info(os)("shared library load of %s failed, %s", filename, error_report);
1060-
#if INCLUDE_JFR
1061-
event.set_success(false);
1062-
event.set_errorMessage(error_report);
1063-
event.commit();
1064-
#endif
1041+
JFR_ONLY(load_event.set_error_msg(error_report);)
10651042
int diag_msg_max_length=ebuflen-strlen(ebuf);
10661043
char* diag_msg_buf=ebuf+strlen(ebuf);
10671044

‎src/hotspot/os/linux/os_linux.cpp

+6-19
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,7 @@
8282
#include "utilities/vmError.hpp"
8383
#if INCLUDE_JFR
8484
#include "jfr/jfrEvents.hpp"
85+
#include "jfr/support/jfrNativeLibraryLoadEvent.hpp"
8586
#endif
8687

8788
// put OS-includes here
@@ -1800,15 +1801,10 @@ void * os::dll_load(const char *filename, char *ebuf, int ebuflen) {
18001801
return nullptr;
18011802
}
18021803

1803-
void * os::Linux::dlopen_helper(const char *filename, char *ebuf,
1804-
int ebuflen) {
1805-
void * result = ::dlopen(filename, RTLD_LAZY);
1806-
1807-
#if INCLUDE_JFR
1808-
EventNativeLibraryLoad event;
1809-
event.set_name(filename);
1810-
#endif
1811-
1804+
void * os::Linux::dlopen_helper(const char *filename, char *ebuf, int ebuflen) {
1805+
void* result;
1806+
JFR_ONLY(NativeLibraryLoadEvent load_event(filename, &result);)
1807+
result = ::dlopen(filename, RTLD_LAZY);
18121808
if (result == nullptr) {
18131809
const char* error_report = ::dlerror();
18141810
if (error_report == nullptr) {
@@ -1820,19 +1816,10 @@ void * os::Linux::dlopen_helper(const char *filename, char *ebuf,
18201816
}
18211817
Events::log_dll_message(nullptr, "Loading shared library %s failed, %s", filename, error_report);
18221818
log_info(os)("shared library load of %s failed, %s", filename, error_report);
1823-
#if INCLUDE_JFR
1824-
event.set_success(false);
1825-
event.set_errorMessage(error_report);
1826-
event.commit();
1827-
#endif
1819+
JFR_ONLY(load_event.set_error_msg(error_report);)
18281820
} else {
18291821
Events::log_dll_message(nullptr, "Loaded shared library %s", filename);
18301822
log_info(os)("shared library load of %s was successful", filename);
1831-
#if INCLUDE_JFR
1832-
event.set_success(true);
1833-
event.set_errorMessage(nullptr);
1834-
event.commit();
1835-
#endif
18361823
}
18371824
return result;
18381825
}

‎src/hotspot/os/posix/os_posix.cpp

+4-15
Original file line numberDiff line numberDiff line change
@@ -51,7 +51,7 @@
5151
#include "utilities/macros.hpp"
5252
#include "utilities/vmError.hpp"
5353
#if INCLUDE_JFR
54-
#include "jfr/jfrEvents.hpp"
54+
#include "jfr/support/jfrNativeLibraryLoadEvent.hpp"
5555
#endif
5656

5757
#ifdef AIX
@@ -725,10 +725,7 @@ void os::dll_unload(void *lib) {
725725
}
726726
#endif // LINUX
727727

728-
#if INCLUDE_JFR
729-
EventNativeLibraryUnload event;
730-
event.set_name(l_path);
731-
#endif
728+
JFR_ONLY(NativeLibraryUnloadEvent unload_event(l_path);)
732729

733730
if (l_path == nullptr) {
734731
l_path = "<not available>";
@@ -739,11 +736,7 @@ void os::dll_unload(void *lib) {
739736
Events::log_dll_message(nullptr, "Unloaded shared library \"%s\" [" INTPTR_FORMAT "]",
740737
l_path, p2i(lib));
741738
log_info(os)("Unloaded shared library \"%s\" [" INTPTR_FORMAT "]", l_path, p2i(lib));
742-
#if INCLUDE_JFR
743-
event.set_success(true);
744-
event.set_errorMessage(nullptr);
745-
event.commit();
746-
#endif
739+
JFR_ONLY(unload_event.set_result(true);)
747740
} else {
748741
const char* error_report = ::dlerror();
749742
if (error_report == nullptr) {
@@ -754,11 +747,7 @@ void os::dll_unload(void *lib) {
754747
l_path, p2i(lib), error_report);
755748
log_info(os)("Attempt to unload shared library \"%s\" [" INTPTR_FORMAT "] failed, %s",
756749
l_path, p2i(lib), error_report);
757-
#if INCLUDE_JFR
758-
event.set_success(false);
759-
event.set_errorMessage(error_report);
760-
event.commit();
761-
#endif
750+
JFR_ONLY(unload_event.set_error_msg(error_report);)
762751
}
763752
// Update the dll cache
764753
AIX_ONLY(LoadedLibraries::reload());

‎src/hotspot/os/windows/os_windows.cpp

+12-50
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,7 @@
8080
#include "windbghelp.hpp"
8181
#if INCLUDE_JFR
8282
#include "jfr/jfrEvents.hpp"
83+
#include "jfr/support/jfrNativeLibraryLoadEvent.hpp"
8384
#endif
8485

8586
#ifdef _DEBUG
@@ -1246,33 +1247,22 @@ void os::dll_unload(void *lib) {
12461247
snprintf(name, MAX_PATH, "<not available>");
12471248
}
12481249

1249-
#if INCLUDE_JFR
1250-
EventNativeLibraryUnload event;
1251-
event.set_name(name);
1252-
#endif
1250+
JFR_ONLY(NativeLibraryUnloadEvent unload_event(name);)
12531251

12541252
if (::FreeLibrary((HMODULE)lib)) {
12551253
Events::log_dll_message(nullptr, "Unloaded dll \"%s\" [" INTPTR_FORMAT "]", name, p2i(lib));
12561254
log_info(os)("Unloaded dll \"%s\" [" INTPTR_FORMAT "]", name, p2i(lib));
1257-
#if INCLUDE_JFR
1258-
event.set_success(true);
1259-
event.set_errorMessage(nullptr);
1260-
event.commit();
1261-
#endif
1255+
JFR_ONLY(unload_event.set_result(true);)
12621256
} else {
12631257
const DWORD errcode = ::GetLastError();
12641258
char buf[500];
12651259
size_t tl = os::lasterror(buf, sizeof(buf));
12661260
Events::log_dll_message(nullptr, "Attempt to unload dll \"%s\" [" INTPTR_FORMAT "] failed (error code %d)", name, p2i(lib), errcode);
12671261
log_info(os)("Attempt to unload dll \"%s\" [" INTPTR_FORMAT "] failed (error code %d)", name, p2i(lib), errcode);
1268-
#if INCLUDE_JFR
1269-
event.set_success(false);
12701262
if (tl == 0) {
12711263
os::snprintf(buf, sizeof(buf), "Attempt to unload dll failed (error code %d)", (int) errcode);
12721264
}
1273-
event.set_errorMessage(buf);
1274-
event.commit();
1275-
#endif
1265+
JFR_ONLY(unload_event.set_error_msg(buf);)
12761266
}
12771267
}
12781268

@@ -1541,21 +1531,14 @@ static int _print_module(const char* fname, address base_address,
15411531
// same architecture as Hotspot is running on
15421532
void * os::dll_load(const char *name, char *ebuf, int ebuflen) {
15431533
log_info(os)("attempting shared library load of %s", name);
1544-
#if INCLUDE_JFR
1545-
EventNativeLibraryLoad event;
1546-
event.set_name(name);
1547-
#endif
1548-
void * result = LoadLibrary(name);
1534+
void* result;
1535+
JFR_ONLY(NativeLibraryLoadEvent load_event(name, &result);)
1536+
result = LoadLibrary(name);
15491537
if (result != nullptr) {
15501538
Events::log_dll_message(nullptr, "Loaded shared library %s", name);
15511539
// Recalculate pdb search path if a DLL was loaded successfully.
15521540
SymbolEngine::recalc_search_path();
15531541
log_info(os)("shared library load of %s was successful", name);
1554-
#if INCLUDE_JFR
1555-
event.set_success(true);
1556-
event.set_errorMessage(nullptr);
1557-
event.commit();
1558-
#endif
15591542
return result;
15601543
}
15611544
DWORD errcode = GetLastError();
@@ -1569,11 +1552,7 @@ void * os::dll_load(const char *name, char *ebuf, int ebuflen) {
15691552
if (errcode == ERROR_MOD_NOT_FOUND) {
15701553
strncpy(ebuf, "Can't find dependent libraries", ebuflen - 1);
15711554
ebuf[ebuflen - 1] = '\0';
1572-
#if INCLUDE_JFR
1573-
event.set_success(false);
1574-
event.set_errorMessage(ebuf);
1575-
event.commit();
1576-
#endif
1555+
JFR_ONLY(load_event.set_error_msg(ebuf);)
15771556
return nullptr;
15781557
}
15791558

@@ -1584,11 +1563,7 @@ void * os::dll_load(const char *name, char *ebuf, int ebuflen) {
15841563
// else call os::lasterror to obtain system error message
15851564
int fd = ::open(name, O_RDONLY | O_BINARY, 0);
15861565
if (fd < 0) {
1587-
#if INCLUDE_JFR
1588-
event.set_success(false);
1589-
event.set_errorMessage("open on dll file did not work");
1590-
event.commit();
1591-
#endif
1566+
JFR_ONLY(load_event.set_error_msg("open on dll file did not work");)
15921567
return nullptr;
15931568
}
15941569

@@ -1615,11 +1590,7 @@ void * os::dll_load(const char *name, char *ebuf, int ebuflen) {
16151590
::close(fd);
16161591
if (failed_to_get_lib_arch) {
16171592
// file i/o error - report os::lasterror(...) msg
1618-
#if INCLUDE_JFR
1619-
event.set_success(false);
1620-
event.set_errorMessage("failed to get lib architecture");
1621-
event.commit();
1622-
#endif
1593+
JFR_ONLY(load_event.set_error_msg("failed to get lib architecture");)
16231594
return nullptr;
16241595
}
16251596

@@ -1664,11 +1635,7 @@ void * os::dll_load(const char *name, char *ebuf, int ebuflen) {
16641635
// If the architecture is right
16651636
// but some other error took place - report os::lasterror(...) msg
16661637
if (lib_arch == running_arch) {
1667-
#if INCLUDE_JFR
1668-
event.set_success(false);
1669-
event.set_errorMessage("lib architecture matches, but other error occured");
1670-
event.commit();
1671-
#endif
1638+
JFR_ONLY(load_event.set_error_msg("lib architecture matches, but other error occured");)
16721639
return nullptr;
16731640
}
16741641

@@ -1682,12 +1649,7 @@ void * os::dll_load(const char *name, char *ebuf, int ebuflen) {
16821649
"Can't load this .dll (machine code=0x%x) on a %s-bit platform",
16831650
lib_arch, running_arch_str);
16841651
}
1685-
#if INCLUDE_JFR
1686-
event.set_success(false);
1687-
event.set_errorMessage(ebuf);
1688-
event.commit();
1689-
#endif
1690-
1652+
JFR_ONLY(load_event.set_error_msg(ebuf);)
16911653
return nullptr;
16921654
}
16931655

‎src/hotspot/share/jfr/metadata/metadata.xml

+2-2
Original file line numberDiff line numberDiff line change
@@ -939,14 +939,14 @@
939939
<Field type="ulong" contentType="address" name="topAddress" label="Top Address" description="Ending address of the module, if available" />
940940
</Event>
941941

942-
<Event name="NativeLibraryLoad" category="Java Virtual Machine, Runtime" label="Native Library Load" thread="false" stackTrace="true" startTime="true"
942+
<Event name="NativeLibraryLoad" category="Java Virtual Machine, Runtime" label="Native Library Load" thread="true" stackTrace="true" startTime="true"
943943
description="Information about a dynamic library or other native image load operation">
944944
<Field type="string" name="name" label="Name" />
945945
<Field type="boolean" name="success" label="Success" description="Success or failure of the load operation" />
946946
<Field type="string" name="errorMessage" label="Error Message" description="In case of a load error, error description" />
947947
</Event>
948948

949-
<Event name="NativeLibraryUnload" category="Java Virtual Machine, Runtime" label="Native Library Unload" thread="false" stackTrace="true" startTime="true"
949+
<Event name="NativeLibraryUnload" category="Java Virtual Machine, Runtime" label="Native Library Unload" thread="true" stackTrace="true" startTime="true"
950950
description="Information about a dynamic library or other native image unload operation">
951951
<Field type="string" name="name" label="Name" />
952952
<Field type="boolean" name="success" label="Success" description="Success or failure of the unload operation" />
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,123 @@
1+
/*
2+
* Copyright (c) 2023, 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+
#include "precompiled.hpp"
26+
#include "jfr/jfrEvents.hpp"
27+
#include "jfr/support/jfrNativeLibraryLoadEvent.hpp"
28+
#include "jfr/utilities/jfrTime.hpp"
29+
#include "runtime/interfaceSupport.inline.hpp"
30+
#include "runtime/thread.inline.hpp"
31+
32+
JfrNativeLibraryEventBase::JfrNativeLibraryEventBase(const char* name) : _name(name), _error_msg(nullptr), _start_time(nullptr) {}
33+
34+
JfrNativeLibraryEventBase::~JfrNativeLibraryEventBase() {
35+
delete _start_time;
36+
}
37+
38+
const char* JfrNativeLibraryEventBase::name() const {
39+
return _name;
40+
}
41+
42+
JfrTicksWrapper* JfrNativeLibraryEventBase::start_time() const {
43+
return _start_time;
44+
}
45+
46+
bool JfrNativeLibraryEventBase::has_start_time() const {
47+
return _start_time != nullptr;
48+
}
49+
50+
const char* JfrNativeLibraryEventBase::error_msg() const {
51+
return _error_msg;
52+
}
53+
54+
void JfrNativeLibraryEventBase::set_error_msg(const char* error_msg) {
55+
assert(_error_msg == nullptr, "invariant");
56+
_error_msg = error_msg;
57+
}
58+
59+
/*
60+
* The JfrTicks value is heap allocated inside an object of type JfrTicksWrapper.
61+
* The reason is that a raw value object of type Ticks is not possible at this
62+
* location because this code runs as part of early VM bootstrap, at a moment
63+
* where Ticks support is not yet initialized.
64+
*/
65+
template <typename EventType>
66+
static inline JfrTicksWrapper* allocate_start_time() {
67+
return EventType::is_enabled() ? new JfrTicksWrapper() : nullptr;
68+
}
69+
70+
NativeLibraryLoadEvent::NativeLibraryLoadEvent(const char* name, void** result) : JfrNativeLibraryEventBase(name), _result(result) {
71+
assert(_result != nullptr, "invariant");
72+
_start_time = allocate_start_time<EventNativeLibraryLoad>();
73+
}
74+
75+
bool NativeLibraryLoadEvent::success() const {
76+
return *_result != nullptr;
77+
}
78+
79+
NativeLibraryUnloadEvent::NativeLibraryUnloadEvent(const char* name) : JfrNativeLibraryEventBase(name), _result(false) {
80+
_start_time = allocate_start_time<EventNativeLibraryUnload>();
81+
}
82+
83+
bool NativeLibraryUnloadEvent::success() const {
84+
return _result;
85+
}
86+
87+
void NativeLibraryUnloadEvent::set_result(bool result) {
88+
_result = result;
89+
}
90+
91+
template <typename EventType, typename HelperType>
92+
static void commit(HelperType& helper) {
93+
if (!helper.has_start_time()) {
94+
return;
95+
}
96+
EventType event(UNTIMED);
97+
event.set_endtime(JfrTicks::now());
98+
event.set_starttime(*helper.start_time());
99+
event.set_name(helper.name());
100+
event.set_errorMessage(helper.error_msg());
101+
event.set_success(helper.success());
102+
Thread* thread = Thread::current();
103+
assert(thread != nullptr, "invariant");
104+
if (thread->is_Java_thread()) {
105+
JavaThread* jt = JavaThread::cast(thread);
106+
if (jt->thread_state() != _thread_in_vm) {
107+
assert(jt->thread_state() == _thread_in_native, "invariant");
108+
// For a JavaThread to take a JFR stacktrace, it must be in _thread_in_vm. Can safepoint here.
109+
ThreadInVMfromNative transition(jt);
110+
event.commit();
111+
return;
112+
}
113+
}
114+
event.commit();
115+
}
116+
117+
NativeLibraryLoadEvent::~NativeLibraryLoadEvent() {
118+
commit<EventNativeLibraryLoad>(*this);
119+
}
120+
121+
NativeLibraryUnloadEvent::~NativeLibraryUnloadEvent() {
122+
commit<EventNativeLibraryUnload>(*this);
123+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
/*
2+
* Copyright (c) 2023, 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_JFR_SUPPORT_JFRNATIVELIBRARYLOADEVENT_HPP
26+
#define SHARE_JFR_SUPPORT_JFRNATIVELIBRARYLOADEVENT_HPP
27+
28+
#include "memory/allocation.hpp"
29+
30+
class JfrTicksWrapper;
31+
32+
/*
33+
* Helper types for populating NativeLibrary events.
34+
* Event commit is run as part of destructors.
35+
*/
36+
37+
class JfrNativeLibraryEventBase : public StackObj {
38+
protected:
39+
const char* _name;
40+
const char* _error_msg;
41+
JfrTicksWrapper* _start_time;
42+
JfrNativeLibraryEventBase(const char* name);
43+
~JfrNativeLibraryEventBase();
44+
public:
45+
const char* name() const;
46+
const char* error_msg() const;
47+
void set_error_msg(const char* error_msg);
48+
JfrTicksWrapper* start_time() const;
49+
bool has_start_time() const;
50+
};
51+
52+
class NativeLibraryLoadEvent : public JfrNativeLibraryEventBase {
53+
private:
54+
void** _result;
55+
public:
56+
NativeLibraryLoadEvent(const char* name, void** result);
57+
~NativeLibraryLoadEvent();
58+
bool success() const;
59+
};
60+
61+
class NativeLibraryUnloadEvent : public JfrNativeLibraryEventBase {
62+
private:
63+
bool _result;
64+
public:
65+
NativeLibraryUnloadEvent(const char* name);
66+
~NativeLibraryUnloadEvent();
67+
bool success() const;
68+
void set_result(bool result);
69+
};
70+
71+
#endif // SHARE_JFR_SUPPORT_JFRNATIVELIBRARYLOADEVENT_HPP

‎src/hotspot/share/jfr/utilities/jfrTime.hpp

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2012, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2012, 2023, 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
@@ -25,6 +25,7 @@
2525
#ifndef SHARE_JFR_UTILITIES_JFRTIME_HPP
2626
#define SHARE_JFR_UTILITIES_JFRTIME_HPP
2727

28+
#include "jfr/utilities/jfrAllocation.hpp"
2829
#include "utilities/ticks.hpp"
2930

3031
typedef TimeInstant<CounterRepresentation, FastUnorderedElapsedCounterSource> JfrTicks;
@@ -41,4 +42,13 @@ class JfrTime {
4142
static const void* time_function();
4243
};
4344

45+
// For dynamically allocated Ticks values.
46+
class JfrTicksWrapper : public JfrCHeapObj {
47+
private:
48+
JfrTicks _ticks;
49+
public:
50+
JfrTicksWrapper() : _ticks(JfrTicks::now()) {}
51+
operator JfrTicks() const { return _ticks; }
52+
};
53+
4454
#endif // SHARE_JFR_UTILITIES_JFRTIME_HPP

‎test/jdk/jdk/jfr/event/runtime/TestNativeLibraryLoadEvent.java

+38-13
Original file line numberDiff line numberDiff line change
@@ -23,14 +23,15 @@
2323

2424
package jdk.jfr.event.runtime;
2525

26+
import static jdk.test.lib.Asserts.assertNotNull;
27+
import static jdk.test.lib.Asserts.assertNull;
2628
import static jdk.test.lib.Asserts.assertTrue;
2729

28-
import java.util.ArrayList;
29-
import java.util.List;
30-
import java.util.stream.Collectors;
31-
3230
import jdk.jfr.Recording;
3331
import jdk.jfr.consumer.RecordedEvent;
32+
import jdk.jfr.consumer.RecordedFrame;
33+
import jdk.jfr.consumer.RecordedMethod;
34+
import jdk.jfr.consumer.RecordedStackTrace;
3435
import jdk.test.lib.Platform;
3536
import jdk.test.lib.jfr.EventNames;
3637
import jdk.test.lib.jfr.Events;
@@ -46,25 +47,49 @@
4647
public class TestNativeLibraryLoadEvent {
4748

4849
private final static String EVENT_NAME = EventNames.NativeLibraryLoad;
50+
private final static String LOAD_CLASS_NAME = "java.lang.System";
51+
private final static String LOAD_METHOD_NAME = "loadLibrary";
52+
private final static String LIBRARY = "instrument";
53+
private final static String PLATFORM_LIBRARY_NAME = Platform.buildSharedLibraryName(LIBRARY);
4954

5055
public static void main(String[] args) throws Throwable {
5156
try (Recording recording = new Recording()) {
5257
recording.enable(EVENT_NAME);
5358
recording.start();
54-
System.loadLibrary("instrument");
59+
System.loadLibrary(LIBRARY);
5560
recording.stop();
5661

57-
String expectedLib = Platform.buildSharedLibraryName("instrument");
58-
boolean expectedLibFound = false;
5962
for (RecordedEvent event : Events.fromRecording(recording)) {
60-
System.out.println("Event:" + event);
61-
String lib = Events.assertField(event, "name").notEmpty().getValue();
62-
Events.assertField(event, "success");
63-
if (lib.contains(expectedLib)) {
64-
expectedLibFound = true;
63+
if (validate(event)) {
64+
return;
6565
}
6666
}
67-
assertTrue(expectedLibFound, "Missing library " + expectedLib);
67+
assertTrue(false, "Missing library " + PLATFORM_LIBRARY_NAME);
68+
}
69+
}
70+
71+
private static boolean validate(RecordedEvent event) {
72+
assertTrue(event.getEventType().getName().equals(EVENT_NAME));
73+
String lib = Events.assertField(event, "name").notEmpty().getValue();
74+
System.out.println(lib);
75+
if (!lib.endsWith(PLATFORM_LIBRARY_NAME)) {
76+
return false;
77+
}
78+
assertTrue(Events.assertField(event, "success").getValue());
79+
assertNull(Events.assertField(event, "errorMessage").getValue());
80+
RecordedStackTrace stacktrace = event.getStackTrace();
81+
assertNotNull(stacktrace);
82+
for (RecordedFrame f : stacktrace.getFrames()) {
83+
if (match(f.getMethod())) {
84+
return true;
85+
}
6886
}
87+
return false;
88+
}
89+
90+
private static boolean match(RecordedMethod method) {
91+
assertNotNull(method);
92+
System.out.println(method.getType().getName() + "." + method.getName());
93+
return method.getName().equals(LOAD_METHOD_NAME) && method.getType().getName().equals(LOAD_CLASS_NAME);
6994
}
7095
}

0 commit comments

Comments
 (0)
Please sign in to comment.