blob: c9d587af94fa93f7428f4a9a50370826e055a7e1 [file] [log] [blame]
Andreas Gampe77708d92016-10-07 11:48:21 -07001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Andreas Gampe06c42a52017-07-26 14:17:14 -070017#ifndef ART_OPENJDKJVMTI_EVENTS_H_
18#define ART_OPENJDKJVMTI_EVENTS_H_
Andreas Gampe77708d92016-10-07 11:48:21 -070019
20#include <bitset>
Alex Lightb7c640d2019-03-20 15:52:13 -070021#include <unordered_map>
Andreas Gampe77708d92016-10-07 11:48:21 -070022#include <vector>
23
Andreas Gampe57943812017-12-06 21:39:13 -080024#include <android-base/logging.h>
Alex Light66834462019-04-08 16:28:29 +000025#include <android-base/thread_annotations.h>
Andreas Gampe57943812017-12-06 21:39:13 -080026
Alex Lightb7c640d2019-03-20 15:52:13 -070027#include "android-base/thread_annotations.h"
Andreas Gampe57943812017-12-06 21:39:13 -080028#include "base/macros.h"
29#include "base/mutex.h"
Andreas Gampe77708d92016-10-07 11:48:21 -070030#include "jvmti.h"
Alex Lightb7c640d2019-03-20 15:52:13 -070031#include "managed_stack.h"
Andreas Gampe77708d92016-10-07 11:48:21 -070032#include "thread.h"
33
34namespace openjdkjvmti {
35
36struct ArtJvmTiEnv;
Andreas Gampe27fa96c2016-10-07 15:05:24 -070037class JvmtiAllocationListener;
Alex Light8c2b9292017-11-09 13:21:01 -080038class JvmtiDdmChunkListener;
Andreas Gampe9b8c5882016-10-21 15:27:46 -070039class JvmtiGcPauseListener;
Alex Lightb7edcda2017-04-27 13:20:31 -070040class JvmtiMethodTraceListener;
Alex Light77fee872017-09-05 14:51:49 -070041class JvmtiMonitorListener;
Charles Munger5cc0e752018-11-09 12:30:46 -080042class JvmtiParkListener;
Andreas Gampe77708d92016-10-07 11:48:21 -070043
Alex Light73afd322017-01-18 11:17:47 -080044// an enum for ArtEvents. This differs from the JVMTI events only in that we distinguish between
45// retransformation capable and incapable loading
Alex Light8c2b9292017-11-09 13:21:01 -080046enum class ArtJvmtiEvent : jint {
Alex Light40d87f42017-01-18 10:27:06 -080047 kMinEventTypeVal = JVMTI_MIN_EVENT_TYPE_VAL,
48 kVmInit = JVMTI_EVENT_VM_INIT,
49 kVmDeath = JVMTI_EVENT_VM_DEATH,
50 kThreadStart = JVMTI_EVENT_THREAD_START,
51 kThreadEnd = JVMTI_EVENT_THREAD_END,
Alex Light73afd322017-01-18 11:17:47 -080052 kClassFileLoadHookNonRetransformable = JVMTI_EVENT_CLASS_FILE_LOAD_HOOK,
Alex Light40d87f42017-01-18 10:27:06 -080053 kClassLoad = JVMTI_EVENT_CLASS_LOAD,
54 kClassPrepare = JVMTI_EVENT_CLASS_PREPARE,
55 kVmStart = JVMTI_EVENT_VM_START,
56 kException = JVMTI_EVENT_EXCEPTION,
57 kExceptionCatch = JVMTI_EVENT_EXCEPTION_CATCH,
58 kSingleStep = JVMTI_EVENT_SINGLE_STEP,
59 kFramePop = JVMTI_EVENT_FRAME_POP,
60 kBreakpoint = JVMTI_EVENT_BREAKPOINT,
61 kFieldAccess = JVMTI_EVENT_FIELD_ACCESS,
62 kFieldModification = JVMTI_EVENT_FIELD_MODIFICATION,
63 kMethodEntry = JVMTI_EVENT_METHOD_ENTRY,
64 kMethodExit = JVMTI_EVENT_METHOD_EXIT,
65 kNativeMethodBind = JVMTI_EVENT_NATIVE_METHOD_BIND,
66 kCompiledMethodLoad = JVMTI_EVENT_COMPILED_METHOD_LOAD,
67 kCompiledMethodUnload = JVMTI_EVENT_COMPILED_METHOD_UNLOAD,
68 kDynamicCodeGenerated = JVMTI_EVENT_DYNAMIC_CODE_GENERATED,
69 kDataDumpRequest = JVMTI_EVENT_DATA_DUMP_REQUEST,
70 kMonitorWait = JVMTI_EVENT_MONITOR_WAIT,
71 kMonitorWaited = JVMTI_EVENT_MONITOR_WAITED,
72 kMonitorContendedEnter = JVMTI_EVENT_MONITOR_CONTENDED_ENTER,
73 kMonitorContendedEntered = JVMTI_EVENT_MONITOR_CONTENDED_ENTERED,
74 kResourceExhausted = JVMTI_EVENT_RESOURCE_EXHAUSTED,
75 kGarbageCollectionStart = JVMTI_EVENT_GARBAGE_COLLECTION_START,
76 kGarbageCollectionFinish = JVMTI_EVENT_GARBAGE_COLLECTION_FINISH,
77 kObjectFree = JVMTI_EVENT_OBJECT_FREE,
78 kVmObjectAlloc = JVMTI_EVENT_VM_OBJECT_ALLOC,
Alex Lightb7c640d2019-03-20 15:52:13 -070079 // Internal event to mark a ClassFileLoadHook as one created with the can_retransform_classes
80 // capability.
Alex Light73afd322017-01-18 11:17:47 -080081 kClassFileLoadHookRetransformable = JVMTI_MAX_EVENT_TYPE_VAL + 1,
Alex Light8c2b9292017-11-09 13:21:01 -080082 kDdmPublishChunk = JVMTI_MAX_EVENT_TYPE_VAL + 2,
Alex Light72d7e942019-07-23 13:10:20 -070083 kObsoleteObjectCreated = JVMTI_MAX_EVENT_TYPE_VAL + 3,
Alex Lightd55b8442019-10-15 15:46:07 -070084 kStructuralDexFileLoadHook = JVMTI_MAX_EVENT_TYPE_VAL + 4,
85 kMaxNormalEventTypeVal = kStructuralDexFileLoadHook,
Alex Lightb7c640d2019-03-20 15:52:13 -070086
87 // All that follow are events used to implement internal JVMTI functions. They are not settable
88 // directly by agents.
89 kMinInternalEventTypeVal = kMaxNormalEventTypeVal + 1,
90
91 // Internal event we use to implement the ForceEarlyReturn functions.
92 kForceEarlyReturnUpdateReturnValue = kMinInternalEventTypeVal,
93 kMaxInternalEventTypeVal = kForceEarlyReturnUpdateReturnValue,
94
95 kMaxEventTypeVal = kMaxInternalEventTypeVal,
Alex Light40d87f42017-01-18 10:27:06 -080096};
97
Alex Lightb7c640d2019-03-20 15:52:13 -070098constexpr jint kInternalEventCount = static_cast<jint>(ArtJvmtiEvent::kMaxInternalEventTypeVal) -
99 static_cast<jint>(ArtJvmtiEvent::kMinInternalEventTypeVal) + 1;
100
Alex Light8c2b9292017-11-09 13:21:01 -0800101using ArtJvmtiEventDdmPublishChunk = void (*)(jvmtiEnv *jvmti_env,
102 JNIEnv* jni_env,
103 jint data_type,
104 jint data_len,
105 const jbyte* data);
106
Alex Light72d7e942019-07-23 13:10:20 -0700107using ArtJvmtiEventObsoleteObjectCreated = void (*)(jvmtiEnv *jvmti_env,
108 jlong* obsolete_tag,
109 jlong* new_tag);
110
Alex Lightd55b8442019-10-15 15:46:07 -0700111using ArtJvmtiEventStructuralDexFileLoadHook = void (*)(jvmtiEnv *jvmti_env,
112 JNIEnv* jni_env,
113 jclass class_being_redefined,
114 jobject loader,
115 const char* name,
116 jobject protection_domain,
117 jint dex_data_len,
118 const unsigned char* dex_data,
119 jint* new_dex_data_len,
120 unsigned char** new_dex_data);
121
Alex Lightb7c640d2019-03-20 15:52:13 -0700122// It is not enough to store a Thread pointer, as these may be reused. Use the pointer and the
123// thread id.
124// Note: We could just use the tid like tracing does.
125using UniqueThread = std::pair<art::Thread*, uint32_t>;
126
127struct UniqueThreadHasher {
128 std::size_t operator()(const UniqueThread& k) const {
129 return std::hash<uint32_t>{}(k.second) ^ (std::hash<void*>{}(k.first) << 1);
130 }
131};
132
Alex Light8c2b9292017-11-09 13:21:01 -0800133struct ArtJvmtiEventCallbacks : jvmtiEventCallbacks {
Alex Lightd55b8442019-10-15 15:46:07 -0700134 ArtJvmtiEventCallbacks()
135 : DdmPublishChunk(nullptr),
136 ObsoleteObjectCreated(nullptr),
137 StructuralDexFileLoadHook(nullptr) {
Alex Light8c2b9292017-11-09 13:21:01 -0800138 memset(this, 0, sizeof(jvmtiEventCallbacks));
139 }
140
141 // Copies extension functions from other callback struct if it exists. There must not have been
142 // any modifications to this struct when it is called.
143 void CopyExtensionsFrom(const ArtJvmtiEventCallbacks* cb);
144
145 jvmtiError Set(jint index, jvmtiExtensionEvent cb);
146
147 ArtJvmtiEventDdmPublishChunk DdmPublishChunk;
Alex Light72d7e942019-07-23 13:10:20 -0700148 ArtJvmtiEventObsoleteObjectCreated ObsoleteObjectCreated;
Alex Lightd55b8442019-10-15 15:46:07 -0700149 ArtJvmtiEventStructuralDexFileLoadHook StructuralDexFileLoadHook;
Alex Light8c2b9292017-11-09 13:21:01 -0800150};
151
152bool IsExtensionEvent(jint e);
153bool IsExtensionEvent(ArtJvmtiEvent e);
154
Alex Light40d87f42017-01-18 10:27:06 -0800155// Convert a jvmtiEvent into a ArtJvmtiEvent
156ALWAYS_INLINE static inline ArtJvmtiEvent GetArtJvmtiEvent(ArtJvmTiEnv* env, jvmtiEvent e);
157
Alex Light73afd322017-01-18 11:17:47 -0800158static inline jvmtiEvent GetJvmtiEvent(ArtJvmtiEvent e) {
159 if (UNLIKELY(e == ArtJvmtiEvent::kClassFileLoadHookRetransformable)) {
160 return JVMTI_EVENT_CLASS_FILE_LOAD_HOOK;
161 } else {
162 return static_cast<jvmtiEvent>(e);
163 }
Alex Light40d87f42017-01-18 10:27:06 -0800164}
165
Andreas Gampe77708d92016-10-07 11:48:21 -0700166struct EventMask {
Alex Light40d87f42017-01-18 10:27:06 -0800167 static constexpr size_t kEventsSize =
168 static_cast<size_t>(ArtJvmtiEvent::kMaxEventTypeVal) -
169 static_cast<size_t>(ArtJvmtiEvent::kMinEventTypeVal) + 1;
Andreas Gampe77708d92016-10-07 11:48:21 -0700170 std::bitset<kEventsSize> bit_set;
171
Alex Light40d87f42017-01-18 10:27:06 -0800172 static bool EventIsInRange(ArtJvmtiEvent event) {
173 return event >= ArtJvmtiEvent::kMinEventTypeVal && event <= ArtJvmtiEvent::kMaxEventTypeVal;
Andreas Gampe77708d92016-10-07 11:48:21 -0700174 }
175
Alex Light40d87f42017-01-18 10:27:06 -0800176 void Set(ArtJvmtiEvent event, bool value = true) {
Andreas Gampe77708d92016-10-07 11:48:21 -0700177 DCHECK(EventIsInRange(event));
Alex Light40d87f42017-01-18 10:27:06 -0800178 bit_set.set(static_cast<size_t>(event) - static_cast<size_t>(ArtJvmtiEvent::kMinEventTypeVal),
179 value);
Andreas Gampe77708d92016-10-07 11:48:21 -0700180 }
181
Alex Light40d87f42017-01-18 10:27:06 -0800182 bool Test(ArtJvmtiEvent event) const {
Andreas Gampe77708d92016-10-07 11:48:21 -0700183 DCHECK(EventIsInRange(event));
Alex Light40d87f42017-01-18 10:27:06 -0800184 return bit_set.test(
185 static_cast<size_t>(event) - static_cast<size_t>(ArtJvmtiEvent::kMinEventTypeVal));
Andreas Gampe77708d92016-10-07 11:48:21 -0700186 }
187};
188
189struct EventMasks {
190 // The globally enabled events.
191 EventMask global_event_mask;
192
193 // The per-thread enabled events.
194
Andreas Gampe77708d92016-10-07 11:48:21 -0700195 // TODO: Native thread objects are immovable, so we can use them as keys in an (unordered) map,
196 // if necessary.
197 std::vector<std::pair<UniqueThread, EventMask>> thread_event_masks;
198
199 // A union of the per-thread events, for fast-pathing.
200 EventMask unioned_thread_event_mask;
201
202 EventMask& GetEventMask(art::Thread* thread);
203 EventMask* GetEventMaskOrNull(art::Thread* thread);
Alex Light74c84402017-11-29 15:26:38 -0800204 // Circular dependencies mean we cannot see the definition of ArtJvmTiEnv so the mutex is simply
205 // asserted in the function.
206 // Note that the 'env' passed in must be the same env this EventMasks is associated with.
207 void EnableEvent(ArtJvmTiEnv* env, art::Thread* thread, ArtJvmtiEvent event);
208 // REQUIRES(env->event_info_mutex_);
209 // Circular dependencies mean we cannot see the definition of ArtJvmTiEnv so the mutex is simply
210 // asserted in the function.
211 // Note that the 'env' passed in must be the same env this EventMasks is associated with.
212 void DisableEvent(ArtJvmTiEnv* env, art::Thread* thread, ArtJvmtiEvent event);
213 // REQUIRES(env->event_info_mutex_);
Alex Light73afd322017-01-18 11:17:47 -0800214 bool IsEnabledAnywhere(ArtJvmtiEvent event);
215 // Make any changes to event masks needed for the given capability changes. If caps_added is true
216 // then caps is all the newly set capabilities of the jvmtiEnv. If it is false then caps is the
217 // set of all capabilities that were removed from the jvmtiEnv.
218 void HandleChangedCapabilities(const jvmtiCapabilities& caps, bool caps_added);
Andreas Gampe77708d92016-10-07 11:48:21 -0700219};
220
Alex Lightb284f8d2017-11-21 00:00:48 +0000221namespace impl {
222template <ArtJvmtiEvent kEvent> struct EventHandlerFunc { };
223} // namespace impl
224
Andreas Gampe77708d92016-10-07 11:48:21 -0700225// Helper class for event handling.
Andreas Gampe27fa96c2016-10-07 15:05:24 -0700226class EventHandler {
227 public:
228 EventHandler();
229 ~EventHandler();
Andreas Gampe77708d92016-10-07 11:48:21 -0700230
Alex Lightb7edcda2017-04-27 13:20:31 -0700231 // do cleanup for the event handler.
232 void Shutdown();
233
Andreas Gampe77708d92016-10-07 11:48:21 -0700234 // Register an env. It is assumed that this happens on env creation, that is, no events are
235 // enabled, yet.
Alex Lightb284f8d2017-11-21 00:00:48 +0000236 void RegisterArtJvmTiEnv(ArtJvmTiEnv* env) REQUIRES(!envs_lock_);
Andreas Gampe77708d92016-10-07 11:48:21 -0700237
Andreas Gampe3a7eb142017-01-19 21:59:22 -0800238 // Remove an env.
Alex Lightb284f8d2017-11-21 00:00:48 +0000239 void RemoveArtJvmTiEnv(ArtJvmTiEnv* env) REQUIRES(!envs_lock_);
Andreas Gampe3a7eb142017-01-19 21:59:22 -0800240
Alex Light40d87f42017-01-18 10:27:06 -0800241 bool IsEventEnabledAnywhere(ArtJvmtiEvent event) const {
Andreas Gampe77708d92016-10-07 11:48:21 -0700242 if (!EventMask::EventIsInRange(event)) {
243 return false;
244 }
245 return global_mask.Test(event);
246 }
247
Alex Lightb7c640d2019-03-20 15:52:13 -0700248 // Sets an internal event. Unlike normal JVMTI events internal events are not associated with any
249 // particular jvmtiEnv and are refcounted. This refcounting is done to allow us to easily enable
250 // events during functions and disable them during the requested event callback. Since these are
251 // used to implement various JVMTI functions these events always have a single target thread. If
252 // target is null the current thread is used.
253 jvmtiError SetInternalEvent(jthread target,
254 ArtJvmtiEvent event,
255 jvmtiEventMode mode)
256 REQUIRES(!envs_lock_, !art::Locks::mutator_lock_);
257
Alex Light40d87f42017-01-18 10:27:06 -0800258 jvmtiError SetEvent(ArtJvmTiEnv* env,
Alex Light3dacdd62019-03-12 15:45:47 +0000259 jthread thread,
Alex Light40d87f42017-01-18 10:27:06 -0800260 ArtJvmtiEvent event,
Alex Lightb284f8d2017-11-21 00:00:48 +0000261 jvmtiEventMode mode)
262 REQUIRES(!envs_lock_);
Andreas Gampe77708d92016-10-07 11:48:21 -0700263
Alex Light9df79b72017-09-12 08:57:31 -0700264 // Dispatch event to all registered environments. Since this one doesn't have a JNIEnv* it doesn't
265 // matter if it has the mutator_lock.
Andreas Gampe983c1752017-01-23 19:46:56 -0800266 template <ArtJvmtiEvent kEvent, typename ...Args>
Alex Light40d87f42017-01-18 10:27:06 -0800267 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000268 inline void DispatchEvent(art::Thread* thread, Args... args) const
269 REQUIRES(!envs_lock_);
Alex Light9df79b72017-09-12 08:57:31 -0700270
Alex Lightb7edcda2017-04-27 13:20:31 -0700271 // Dispatch event to all registered environments stashing exceptions as needed. This works since
272 // JNIEnv* is always the second argument if it is passed to an event. Needed since C++ does not
273 // allow partial template function specialization.
Alex Light9df79b72017-09-12 08:57:31 -0700274 //
275 // We need both of these since we want to make sure to push a stack frame when it is possible for
276 // the event to allocate local references.
Alex Lightb7edcda2017-04-27 13:20:31 -0700277 template <ArtJvmtiEvent kEvent, typename ...Args>
278 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000279 inline void DispatchEvent(art::Thread* thread, JNIEnv* jnienv, Args... args) const
280 REQUIRES(!envs_lock_);
Andreas Gampe27fa96c2016-10-07 15:05:24 -0700281
Alex Light73afd322017-01-18 11:17:47 -0800282 // Tell the event handler capabilities were added/lost so it can adjust the sent events.If
283 // caps_added is true then caps is all the newly set capabilities of the jvmtiEnv. If it is false
284 // then caps is the set of all capabilities that were removed from the jvmtiEnv.
285 ALWAYS_INLINE
286 inline void HandleChangedCapabilities(ArtJvmTiEnv* env,
287 const jvmtiCapabilities& caps,
Alex Lightb284f8d2017-11-21 00:00:48 +0000288 bool added)
289 REQUIRES(!envs_lock_);
Alex Light73afd322017-01-18 11:17:47 -0800290
Alex Light9df79b72017-09-12 08:57:31 -0700291 // Dispatch event to the given environment, only.
292 template <ArtJvmtiEvent kEvent, typename ...Args>
293 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000294 inline void DispatchEventOnEnv(ArtJvmTiEnv* env,
295 art::Thread* thread,
296 JNIEnv* jnienv,
297 Args... args) const
298 REQUIRES(!envs_lock_);
Alex Light9df79b72017-09-12 08:57:31 -0700299
300 // Dispatch event to the given environment, only.
301 template <ArtJvmtiEvent kEvent, typename ...Args>
302 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000303 inline void DispatchEventOnEnv(ArtJvmTiEnv* env, art::Thread* thread, Args... args) const
304 REQUIRES(!envs_lock_);
Alex Light9df79b72017-09-12 08:57:31 -0700305
Alex Lightb7c640d2019-03-20 15:52:13 -0700306 void AddDelayedNonStandardExitEvent(const art::ShadowFrame* frame, bool is_object, jvalue val)
307 REQUIRES_SHARED(art::Locks::mutator_lock_)
308 REQUIRES(art::Locks::user_code_suspension_lock_, art::Locks::thread_list_lock_);
309
Alex Light72d7e942019-07-23 13:10:20 -0700310 template<typename Visitor>
311 void ForEachEnv(art::Thread* self, Visitor v) REQUIRES(!envs_lock_) {
312 art::ReaderMutexLock mu(self, envs_lock_);
313 for (ArtJvmTiEnv* e : envs) {
314 if (e != nullptr) {
315 v(e);
316 }
317 }
318 }
319
Andreas Gampe27fa96c2016-10-07 15:05:24 -0700320 private:
Alex Light40607862019-05-06 18:16:24 +0000321 void SetupTraceListener(JvmtiMethodTraceListener* listener, ArtJvmtiEvent event, bool enable);
Alex Lightf6df1b52017-11-29 14:46:53 -0800322
Alex Lightb7c640d2019-03-20 15:52:13 -0700323 uint32_t GetInstrumentationEventsFor(ArtJvmtiEvent event);
324
Alex Lightf5d5eb12018-03-06 15:13:59 -0800325 // Specifically handle the FramePop event which it might not always be possible to turn off.
Alex Light40607862019-05-06 18:16:24 +0000326 void SetupFramePopTraceListener(bool enable);
Alex Lightf5d5eb12018-03-06 15:13:59 -0800327
Alex Lightb284f8d2017-11-21 00:00:48 +0000328 template <ArtJvmtiEvent kEvent, typename ...Args>
329 ALWAYS_INLINE
330 inline std::vector<impl::EventHandlerFunc<kEvent>> CollectEvents(art::Thread* thread,
331 Args... args) const
332 REQUIRES(!envs_lock_);
333
Andreas Gampe983c1752017-01-23 19:46:56 -0800334 template <ArtJvmtiEvent kEvent>
Alex Light40d87f42017-01-18 10:27:06 -0800335 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000336 inline bool ShouldDispatchOnThread(ArtJvmTiEnv* env, art::Thread* thread) const;
Alex Light9df79b72017-09-12 08:57:31 -0700337
338 template <ArtJvmtiEvent kEvent, typename ...Args>
339 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000340 static inline void ExecuteCallback(impl::EventHandlerFunc<kEvent> handler,
341 JNIEnv* env,
342 Args... args)
343 REQUIRES(!envs_lock_);
Alex Light9df79b72017-09-12 08:57:31 -0700344
345 template <ArtJvmtiEvent kEvent, typename ...Args>
346 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000347 static inline void ExecuteCallback(impl::EventHandlerFunc<kEvent> handler, Args... args)
348 REQUIRES(!envs_lock_);
349
350 // Public for use to collect dispatches
351 template <ArtJvmtiEvent kEvent, typename ...Args>
352 ALWAYS_INLINE
Alex Light9df79b72017-09-12 08:57:31 -0700353 inline bool ShouldDispatch(ArtJvmTiEnv* env, art::Thread* thread, Args... args) const;
Alex Light40d87f42017-01-18 10:27:06 -0800354
Alex Light73afd322017-01-18 11:17:47 -0800355 ALWAYS_INLINE
356 inline bool NeedsEventUpdate(ArtJvmTiEnv* env,
357 const jvmtiCapabilities& caps,
358 bool added);
359
360 // Recalculates the event mask for the given event.
361 ALWAYS_INLINE
Alex Lightb284f8d2017-11-21 00:00:48 +0000362 inline void RecalculateGlobalEventMask(ArtJvmtiEvent event) REQUIRES(!envs_lock_);
363 ALWAYS_INLINE
Alex Light2a96fe82018-01-22 17:45:02 -0800364 inline void RecalculateGlobalEventMaskLocked(ArtJvmtiEvent event) REQUIRES_SHARED(envs_lock_);
Alex Light73afd322017-01-18 11:17:47 -0800365
Alex Light40607862019-05-06 18:16:24 +0000366 // Returns whether there are any active requests for the given event on the given thread. This
367 // should only be used while modifying the events for a thread.
368 bool GetThreadEventState(ArtJvmtiEvent event, art::Thread* thread)
369 REQUIRES(envs_lock_, art::Locks::thread_list_lock_);
370
Andreas Gampe983c1752017-01-23 19:46:56 -0800371 template <ArtJvmtiEvent kEvent>
Alex Light6ac57502017-01-19 15:05:06 -0800372 ALWAYS_INLINE inline void DispatchClassFileLoadHookEvent(art::Thread* thread,
Andreas Gampe983c1752017-01-23 19:46:56 -0800373 JNIEnv* jnienv,
374 jclass class_being_redefined,
375 jobject loader,
376 const char* name,
377 jobject protection_domain,
378 jint class_data_len,
379 const unsigned char* class_data,
380 jint* new_class_data_len,
Alex Lightb284f8d2017-11-21 00:00:48 +0000381 unsigned char** new_class_data) const
382 REQUIRES(!envs_lock_);
Alex Light6ac57502017-01-19 15:05:06 -0800383
Alex Light0aa7a5a2018-10-10 15:58:14 +0000384 template <ArtJvmtiEvent kEvent>
385 ALWAYS_INLINE inline void DispatchClassLoadOrPrepareEvent(art::Thread* thread,
386 JNIEnv* jnienv,
387 jthread jni_thread,
388 jclass klass) const
389 REQUIRES(!envs_lock_);
390
Alex Light40607862019-05-06 18:16:24 +0000391 // Sets up the global state needed for the first/last enable of an event across all threads
392 void HandleEventType(ArtJvmtiEvent event, bool enable);
393 // Perform deopts required for enabling the event on the given thread. Null thread indicates
394 // global event enabled.
395 jvmtiError HandleEventDeopt(ArtJvmtiEvent event, jthread thread, bool enable);
Alex Lightbebd7bd2017-07-25 14:05:52 -0700396 void HandleLocalAccessCapabilityAdded();
Alex Light0fa17862017-10-24 13:43:05 -0700397 void HandleBreakpointEventsChanged(bool enable);
Andreas Gampe27fa96c2016-10-07 15:05:24 -0700398
Alex Light77fee872017-09-05 14:51:49 -0700399 bool OtherMonitorEventsEnabledAnywhere(ArtJvmtiEvent event);
400
Alex Lightb7c640d2019-03-20 15:52:13 -0700401 int32_t GetInternalEventRefcount(ArtJvmtiEvent event) const REQUIRES(envs_lock_);
402 // Increment internal event refcount for the given event and return the new count.
403 int32_t IncrInternalEventRefcount(ArtJvmtiEvent event) REQUIRES(envs_lock_);
404 // Decrement internal event refcount for the given event and return the new count.
405 int32_t DecrInternalEventRefcount(ArtJvmtiEvent event) REQUIRES(envs_lock_);
406
407 int32_t& GetInternalEventThreadRefcount(ArtJvmtiEvent event, art::Thread* target)
408 REQUIRES(envs_lock_, art::Locks::thread_list_lock_);
409 // Increment internal event refcount for the given event and return the new count.
410 int32_t IncrInternalEventThreadRefcount(ArtJvmtiEvent event, art::Thread* target)
411 REQUIRES(envs_lock_, art::Locks::thread_list_lock_);
412 // Decrement internal event refcount for the given event and return the new count.
413 int32_t DecrInternalEventThreadRefcount(ArtJvmtiEvent event, art::Thread* target)
414 REQUIRES(envs_lock_, art::Locks::thread_list_lock_);
415
Alex Lightb284f8d2017-11-21 00:00:48 +0000416 // List of all JvmTiEnv objects that have been created, in their creation order. It is a std::list
417 // since we mostly access it by iterating over the entire thing, only ever append to the end, and
418 // need to be able to remove arbitrary elements from it.
419 std::list<ArtJvmTiEnv*> envs GUARDED_BY(envs_lock_);
420
Alex Light66834462019-04-08 16:28:29 +0000421 // Close to top level lock. Nothing should be held when we lock this (except for mutator_lock_
422 // which is needed when setting new events).
423 mutable art::ReaderWriterMutex envs_lock_ ACQUIRED_AFTER(art::Locks::mutator_lock_);
Andreas Gampe27fa96c2016-10-07 15:05:24 -0700424
425 // A union of all enabled events, anywhere.
426 EventMask global_mask;
427
428 std::unique_ptr<JvmtiAllocationListener> alloc_listener_;
Alex Light8c2b9292017-11-09 13:21:01 -0800429 std::unique_ptr<JvmtiDdmChunkListener> ddm_listener_;
Andreas Gampe9b8c5882016-10-21 15:27:46 -0700430 std::unique_ptr<JvmtiGcPauseListener> gc_pause_listener_;
Alex Lightb7edcda2017-04-27 13:20:31 -0700431 std::unique_ptr<JvmtiMethodTraceListener> method_trace_listener_;
Alex Light77fee872017-09-05 14:51:49 -0700432 std::unique_ptr<JvmtiMonitorListener> monitor_listener_;
Charles Munger5cc0e752018-11-09 12:30:46 -0800433 std::unique_ptr<JvmtiParkListener> park_listener_;
Alex Lighte814f9d2017-07-31 16:14:39 -0700434
435 // True if frame pop has ever been enabled. Since we store pointers to stack frames we need to
436 // continue to listen to this event even if it has been disabled.
437 // TODO We could remove the listeners once all jvmtiEnvs have drained their shadow-frame vectors.
438 bool frame_pop_enabled;
Alex Lightb7c640d2019-03-20 15:52:13 -0700439
440 // The overall refcount for each internal event across all threads.
441 std::array<int32_t, kInternalEventCount> internal_event_refcount_ GUARDED_BY(envs_lock_);
442 // The refcount for each thread for each internal event.
443 // TODO We should clean both this and the normal EventMask lists up when threads end.
444 std::array<std::unordered_map<UniqueThread, int32_t, UniqueThreadHasher>, kInternalEventCount>
445 internal_event_thread_refcount_
446 GUARDED_BY(envs_lock_) GUARDED_BY(art::Locks::thread_list_lock_);
447
448 friend class JvmtiMethodTraceListener;
Andreas Gampe77708d92016-10-07 11:48:21 -0700449};
450
451} // namespace openjdkjvmti
452
Andreas Gampe06c42a52017-07-26 14:17:14 -0700453#endif // ART_OPENJDKJVMTI_EVENTS_H_