Skip to content

Commit 6016536

Browse files
committedNov 22, 2023
8314745: JFR: @StackFilter
Reviewed-by: mgronlun
1 parent aac4318 commit 6016536

34 files changed

+982
-52
lines changed
 

‎src/hotspot/share/jfr/jni/jfrJavaSupport.cpp

+27
Original file line numberDiff line numberDiff line change
@@ -525,6 +525,33 @@ const char* JfrJavaSupport::c_str(jstring string, Thread* thread, bool c_heap /*
525525
return string != nullptr ? c_str(resolve_non_null(string), thread, c_heap) : nullptr;
526526
}
527527

528+
static Symbol** allocate_symbol_array(bool c_heap, int length, Thread* thread) {
529+
return c_heap ?
530+
NEW_C_HEAP_ARRAY(Symbol*, length, mtTracing) :
531+
NEW_RESOURCE_ARRAY_IN_THREAD(thread, Symbol*, length);
532+
}
533+
534+
Symbol** JfrJavaSupport::symbol_array(jobjectArray string_array, JavaThread* thread, intptr_t* result_array_size, bool c_heap /* false */) {
535+
DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(thread));
536+
assert(string_array != nullptr, "invariant");
537+
assert(result_array_size != nullptr, "invariant");
538+
objArrayOop arrayOop = objArrayOop(resolve_non_null(string_array));
539+
const int length = arrayOop->length();
540+
*result_array_size = length;
541+
Symbol** result_array = allocate_symbol_array(c_heap, length, thread);
542+
assert(result_array != nullptr, "invariant");
543+
for (int i = 0; i < length; i++) {
544+
oop object = arrayOop->obj_at(i);
545+
Symbol* symbol = nullptr;
546+
if (object != nullptr) {
547+
const char* text = c_str(arrayOop->obj_at(i), thread, c_heap);
548+
symbol = SymbolTable::new_symbol(text);
549+
}
550+
result_array[i] = symbol;
551+
}
552+
return result_array;
553+
}
554+
528555
/*
529556
* Exceptions and errors
530557
*/

‎src/hotspot/share/jfr/jni/jfrJavaSupport.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,7 @@ class JfrJavaSupport : public AllStatic {
8686
static Klass* klass(const jobject handle);
8787
static const char* c_str(jstring string, Thread* thread, bool c_heap = false);
8888
static const char* c_str(oop string, Thread* thread, bool c_heap = false);
89+
static Symbol** symbol_array(jobjectArray string_array, JavaThread* thread, intptr_t* result_size, bool c_heap = false);
8990

9091
// exceptions
9192
static void throw_illegal_state_exception(const char* message, TRAPS);

‎src/hotspot/share/jfr/jni/jfrJniMethod.cpp

+12-2
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,8 @@
3636
#include "jfr/recorder/repository/jfrEmergencyDump.hpp"
3737
#include "jfr/recorder/service/jfrEventThrottler.hpp"
3838
#include "jfr/recorder/service/jfrOptionSet.hpp"
39+
#include "jfr/recorder/stacktrace/jfrStackFilter.hpp"
40+
#include "jfr/recorder/stacktrace/jfrStackFilterRegistry.hpp"
3941
#include "jfr/recorder/stacktrace/jfrStackTraceRepository.hpp"
4042
#include "jfr/recorder/stringpool/jfrStringPool.hpp"
4143
#include "jfr/jni/jfrJavaSupport.hpp"
@@ -239,8 +241,8 @@ JVM_ENTRY_NO_ENV(jlong, jfr_class_id(JNIEnv* env, jclass jvm, jclass jc))
239241
return JfrTraceId::load(jc);
240242
JVM_END
241243

242-
JVM_ENTRY_NO_ENV(jlong, jfr_stacktrace_id(JNIEnv* env, jclass jvm, jint skip))
243-
return JfrStackTraceRepository::record(thread, skip);
244+
JVM_ENTRY_NO_ENV(jlong, jfr_stacktrace_id(JNIEnv* env, jclass jvm, jint skip, jlong stack_filter_id))
245+
return JfrStackTraceRepository::record(thread, skip, stack_filter_id);
244246
JVM_END
245247

246248
JVM_ENTRY_NO_ENV(void, jfr_log(JNIEnv* env, jclass jvm, jint tag_set, jint level, jstring message))
@@ -397,3 +399,11 @@ JVM_END
397399
JVM_ENTRY_NO_ENV(void, jfr_emit_data_loss(JNIEnv* env, jclass jvm, jlong bytes))
398400
EventDataLoss::commit(bytes, min_jlong);
399401
JVM_END
402+
403+
JVM_ENTRY_NO_ENV(jlong, jfr_register_stack_filter(JNIEnv* env, jclass jvm, jobjectArray classes, jobjectArray methods))
404+
return JfrStackFilterRegistry::add(classes, methods, thread);
405+
JVM_END
406+
407+
JVM_ENTRY_NO_ENV(void, jfr_unregister_stack_filter(JNIEnv* env, jclass jvm, jlong id))
408+
JfrStackFilterRegistry::remove(id);
409+
JVM_END

‎src/hotspot/share/jfr/jni/jfrJniMethod.hpp

+5-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ jlong JNICALL jfr_class_id(JNIEnv* env, jclass jvm, jclass jc);
5757

5858
jstring JNICALL jfr_get_pid(JNIEnv* env, jclass jvm);
5959

60-
jlong JNICALL jfr_stacktrace_id(JNIEnv* env, jclass jvm, jint skip);
60+
jlong JNICALL jfr_stacktrace_id(JNIEnv* env, jclass jvm, jint skip, jlong stack_filter_id);
6161

6262
jlong JNICALL jfr_elapsed_frequency(JNIEnv* env, jclass jvm);
6363

@@ -159,6 +159,10 @@ jlong JNICALL jfr_host_total_memory(JNIEnv* env, jclass jvm);
159159

160160
void JNICALL jfr_emit_data_loss(JNIEnv* env, jclass jvm, jlong bytes);
161161

162+
jlong JNICALL jfr_register_stack_filter(JNIEnv* env, jobject classes, jobject methods);
163+
164+
jlong JNICALL jfr_unregister_stack_filter(JNIEnv* env, jlong start_filter_id);
165+
162166
#ifdef __cplusplus
163167
}
164168
#endif

‎src/hotspot/share/jfr/jni/jfrJniMethodRegistration.cpp

+4-2
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ JfrJniMethodRegistration::JfrJniMethodRegistration(JNIEnv* env) {
4646
(char*)"getAllEventClasses", (char*)"()Ljava/util/List;", (void*)jfr_get_all_event_classes,
4747
(char*)"getClassId", (char*)"(Ljava/lang/Class;)J", (void*)jfr_class_id,
4848
(char*)"getPid", (char*)"()Ljava/lang/String;", (void*)jfr_get_pid,
49-
(char*)"getStackTraceId", (char*)"(I)J", (void*)jfr_stacktrace_id,
49+
(char*)"getStackTraceId", (char*)"(IJ)J", (void*)jfr_stacktrace_id,
5050
(char*)"getThreadId", (char*)"(Ljava/lang/Thread;)J", (void*)jfr_id_for_thread,
5151
(char*)"getTicksFrequency", (char*)"()J", (void*)jfr_elapsed_frequency,
5252
(char*)"subscribeLogLevel", (char*)"(Ljdk/jfr/internal/LogTag;I)V", (void*)jfr_subscribe_log_level,
@@ -97,7 +97,9 @@ JfrJniMethodRegistration::JfrJniMethodRegistration(JNIEnv* env) {
9797
(char*)"isInstrumented", (char*)"(Ljava/lang/Class;)Z", (void*) jfr_is_class_instrumented,
9898
(char*)"isContainerized", (char*)"()Z", (void*) jfr_is_containerized,
9999
(char*)"hostTotalMemory", (char*)"()J", (void*) jfr_host_total_memory,
100-
(char*)"emitDataLoss", (char*)"(J)V", (void*)jfr_emit_data_loss
100+
(char*)"emitDataLoss", (char*)"(J)V", (void*)jfr_emit_data_loss,
101+
(char*)"registerStackFilter", (char*)"([Ljava/lang/String;[Ljava/lang/String;)J", (void*)jfr_register_stack_filter,
102+
(char*)"unregisterStackFilter", (char*)"(J)V", (void*)jfr_unregister_stack_filter
101103
};
102104

103105
const size_t method_array_length = sizeof(method) / sizeof(JNINativeMethod);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
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/recorder/stacktrace/jfrStackFilter.hpp"
27+
#include "oops/method.hpp"
28+
#include "oops/symbol.hpp"
29+
30+
JfrStackFilter::JfrStackFilter(Symbol** class_names, Symbol** method_names, size_t count)
31+
: _count(count),
32+
_class_names(class_names),
33+
_method_names(method_names) {
34+
assert(_class_names != nullptr, "invariant");
35+
assert(_method_names != nullptr, "invariant");
36+
}
37+
38+
bool JfrStackFilter::match(const Method* method) const {
39+
assert(method != nullptr, "Invariant");
40+
const Symbol* const method_name = method->name();
41+
const Symbol* const klass_name = method->klass_name();
42+
for (size_t i = 0; i < _count; i++) {
43+
const Symbol* m = _method_names[i];
44+
if (m == nullptr || m == method_name) {
45+
const Symbol* c = _class_names[i];
46+
if (c == nullptr || c == klass_name) {
47+
return true;
48+
}
49+
}
50+
}
51+
return false;
52+
}
53+
54+
JfrStackFilter::~JfrStackFilter() {
55+
for (size_t i = 0; i < _count; i++) {
56+
Symbol::maybe_decrement_refcount(_method_names[i]);
57+
Symbol::maybe_decrement_refcount(_class_names[i]);
58+
}
59+
FREE_C_HEAP_ARRAY(Symbol*, _method_names);
60+
FREE_C_HEAP_ARRAY(Symbol*, _class_names);
61+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
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_RECORDER_STACKTRACE_JFRSTACKFILTER_HPP
26+
#define SHARE_JFR_RECORDER_STACKTRACE_JFRSTACKFILTER_HPP
27+
28+
#include "jfr/utilities/jfrAllocation.hpp"
29+
30+
class Mathod;
31+
class Symbol;
32+
33+
class JfrStackFilter : public JfrCHeapObj {
34+
private:
35+
size_t _count;
36+
Symbol** _class_names;
37+
Symbol** _method_names;
38+
39+
public:
40+
JfrStackFilter(Symbol** class_names, Symbol** method_names, size_t count);
41+
~JfrStackFilter();
42+
bool match(const Method* method) const;
43+
};
44+
45+
#endif // SHARE_JFR_RECORDER_STACKTRACE_JFRSTACKFILTER_HPP
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
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+
26+
#include "precompiled.hpp"
27+
#include "jfr/jni/jfrJavaSupport.hpp"
28+
#include "jfr/recorder/stacktrace/jfrStackFilter.hpp"
29+
#include "jfr/recorder/stacktrace/jfrStackFilterRegistry.hpp"
30+
#include "logging/log.hpp"
31+
32+
static const intptr_t STACK_FILTER_ELEMENTS_SIZE = 4096;
33+
static const intptr_t STACK_FILTER_ERROR_CODE = -1;
34+
static const JfrStackFilter* _elements[STACK_FILTER_ELEMENTS_SIZE];
35+
static intptr_t _free_list[STACK_FILTER_ELEMENTS_SIZE];
36+
static intptr_t _index = 0;
37+
static intptr_t _free_list_index = 0;
38+
39+
int64_t JfrStackFilterRegistry::add(jobjectArray classes, jobjectArray methods, JavaThread* jt) {
40+
intptr_t c_size = 0;
41+
Symbol** class_names = JfrJavaSupport::symbol_array(classes, jt, &c_size, true);
42+
assert(class_names != nullptr, "invariant");
43+
intptr_t m_size = 0;
44+
Symbol** method_names = JfrJavaSupport::symbol_array(methods, jt, &m_size, true);
45+
assert(method_names != nullptr, "invariant");
46+
if (c_size != m_size) {
47+
FREE_C_HEAP_ARRAY(Symbol*, class_names);
48+
FREE_C_HEAP_ARRAY(Symbol*, method_names);
49+
JfrJavaSupport::throw_internal_error("Method array size doesn't match class array size", jt);
50+
return STACK_FILTER_ERROR_CODE;
51+
}
52+
assert(c_size >= 0, "invariant");
53+
const JfrStackFilter* filter = new JfrStackFilter(class_names, method_names, static_cast<size_t>(c_size));
54+
return JfrStackFilterRegistry::add(filter);
55+
}
56+
57+
#ifdef ASSERT
58+
static bool range_check(int64_t idx) {
59+
return idx < STACK_FILTER_ELEMENTS_SIZE && idx >= 0;
60+
}
61+
#endif
62+
63+
int64_t JfrStackFilterRegistry::add(const JfrStackFilter* filter) {
64+
if (_free_list_index > 0) {
65+
assert(range_check(_free_list_index), "invariant");
66+
const intptr_t free_index = _free_list[_free_list_index - 1];
67+
_elements[free_index] = filter;
68+
_free_list_index--;
69+
return free_index;
70+
}
71+
if (_index >= STACK_FILTER_ELEMENTS_SIZE - 1) {
72+
log_warning(jfr)("Maximum number of @StackFrame in use has been reached.");
73+
return STACK_FILTER_ERROR_CODE;
74+
}
75+
assert(range_check(_index), "invariant");
76+
_elements[_index] = filter;
77+
return _index++;
78+
}
79+
80+
const JfrStackFilter* JfrStackFilterRegistry::lookup(int64_t id) {
81+
if (id < 0) {
82+
return nullptr;
83+
}
84+
assert(range_check(id), "invariant");
85+
return _elements[id];
86+
}
87+
88+
void JfrStackFilterRegistry::remove(int64_t index) {
89+
assert(range_check(index), "invariant");
90+
delete _elements[index];
91+
if (_free_list_index < STACK_FILTER_ELEMENTS_SIZE - 1) {
92+
assert(range_check(_free_list_index), "invariant");
93+
_free_list[_free_list_index++] = index;
94+
}
95+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
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_RECORDER_STACKTRACE_JFRSTCKFILTERREGISTRY_HPP
26+
#define SHARE_JFR_RECORDER_STACKTRACE_JFRSTCKFILTERREGISTRY_HPP
27+
28+
#include "jni.h"
29+
#include "jfr/utilities/jfrAllocation.hpp"
30+
31+
class JavaThread;
32+
class JfrStackFilter;
33+
34+
class JfrStackFilterRegistry : AllStatic {
35+
private:
36+
static int64_t add(const JfrStackFilter* frame);
37+
public:
38+
static int64_t add(jobjectArray classes, jobjectArray methods, JavaThread* jt);
39+
static void remove(int64_t id);
40+
static const JfrStackFilter* lookup(int64_t id);
41+
};
42+
43+
#endif // SHARE_JFR_RECORDER_STACKTRACE_JFRSTCKFILTERREGISTRY_HPP

0 commit comments

Comments
 (0)
Please sign in to comment.