%PDF- <> %âãÏÓ endobj 2 0 obj <> endobj 3 0 obj <>/ExtGState<>/ProcSet[/PDF/Text/ImageB/ImageC/ImageI] >>/Annots[ 28 0 R 29 0 R] /MediaBox[ 0 0 595.5 842.25] /Contents 4 0 R/Group<>/Tabs/S>> endobj ºaâÚÎΞ-ÌE1ÍØÄ÷{òò2ÿ ÛÖ^ÔÀá TÎ{¦?§®¥kuµùÕ5sLOšuY>endobj 2 0 obj<>endobj 2 0 obj<>endobj 2 0 obj<>endobj 2 0 obj<> endobj 2 0 obj<>endobj 2 0 obj<>es 3 0 R>> endobj 2 0 obj<> ox[ 0.000000 0.000000 609.600000 935.600000]/Fi endobj 3 0 obj<> endobj 7 1 obj<>/ProcSet[/PDF/Text/ImageB/ImageC/ImageI]>>/Subtype/Form>> stream
// Copyright 2016 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_LOGGING_CODE_EVENTS_H_ #define V8_LOGGING_CODE_EVENTS_H_ #include <unordered_set> #include "src/base/platform/mutex.h" #include "src/base/vector.h" #include "src/common/globals.h" #include "src/objects/code.h" #include "src/objects/name.h" #include "src/objects/shared-function-info.h" #include "src/objects/string.h" namespace v8 { namespace internal { class AbstractCode; class Name; class SharedFunctionInfo; class String; namespace wasm { class WasmCode; using WasmName = base::Vector<const char>; } // namespace wasm // clang-format off #define LOG_EVENTS_LIST(V) \ V(CODE_CREATION_EVENT, code-creation) \ V(CODE_DISABLE_OPT_EVENT, code-disable-optimization) \ V(CODE_MOVE_EVENT, code-move) \ V(CODE_DELETE_EVENT, code-delete) \ V(CODE_MOVING_GC, code-moving-gc) \ V(SHARED_FUNC_MOVE_EVENT, sfi-move) \ V(SNAPSHOT_CODE_NAME_EVENT, snapshot-code-name) \ V(TICK_EVENT, tick) // clang-format on #define TAGS_LIST(V) \ V(BUILTIN_TAG, Builtin) \ V(CALLBACK_TAG, Callback) \ V(EVAL_TAG, Eval) \ V(FUNCTION_TAG, Function) \ V(HANDLER_TAG, Handler) \ V(BYTECODE_HANDLER_TAG, BytecodeHandler) \ V(LAZY_COMPILE_TAG, LazyCompile) \ V(REG_EXP_TAG, RegExp) \ V(SCRIPT_TAG, Script) \ V(STUB_TAG, Stub) \ V(NATIVE_FUNCTION_TAG, Function) \ V(NATIVE_LAZY_COMPILE_TAG, LazyCompile) \ V(NATIVE_SCRIPT_TAG, Script) // Note that 'NATIVE_' cases for functions and scripts are mapped onto // original tags when writing to the log. #define LOG_EVENTS_AND_TAGS_LIST(V) \ LOG_EVENTS_LIST(V) \ TAGS_LIST(V) #define PROFILE(the_isolate, Call) (the_isolate)->code_event_dispatcher()->Call; class CodeEventListener { public: #define DECLARE_ENUM(enum_item, _) enum_item, enum LogEventsAndTags { LOG_EVENTS_AND_TAGS_LIST(DECLARE_ENUM) NUMBER_OF_LOG_EVENTS }; #undef DECLARE_ENUM virtual ~CodeEventListener() = default; virtual void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, const char* name) = 0; virtual void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, Handle<Name> name) = 0; virtual void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, Handle<SharedFunctionInfo> shared, Handle<Name> script_name) = 0; virtual void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, Handle<SharedFunctionInfo> shared, Handle<Name> script_name, int line, int column) = 0; #if V8_ENABLE_WEBASSEMBLY virtual void CodeCreateEvent(LogEventsAndTags tag, const wasm::WasmCode* code, wasm::WasmName name, const char* source_url, int code_offset, int script_id) = 0; #endif // V8_ENABLE_WEBASSEMBLY virtual void CallbackEvent(Handle<Name> name, Address entry_point) = 0; virtual void GetterCallbackEvent(Handle<Name> name, Address entry_point) = 0; virtual void SetterCallbackEvent(Handle<Name> name, Address entry_point) = 0; virtual void RegExpCodeCreateEvent(Handle<AbstractCode> code, Handle<String> source) = 0; // Not handlified as this happens during GC. No allocation allowed. virtual void CodeMoveEvent(AbstractCode from, AbstractCode to) = 0; virtual void SharedFunctionInfoMoveEvent(Address from, Address to) = 0; virtual void CodeMovingGCEvent() = 0; virtual void CodeDisableOptEvent(Handle<AbstractCode> code, Handle<SharedFunctionInfo> shared) = 0; virtual void CodeDeoptEvent(Handle<Code> code, DeoptimizeKind kind, Address pc, int fp_to_sp_delta, bool reuse_code) = 0; // These events can happen when 1. an assumption made by optimized code fails // or 2. a weakly embedded object dies. virtual void CodeDependencyChangeEvent(Handle<Code> code, Handle<SharedFunctionInfo> shared, const char* reason) = 0; // Called during GC shortly after any weak references to code objects are // cleared. virtual void WeakCodeClearEvent() = 0; virtual bool is_listening_to_code_events() { return false; } }; // Dispatches code events to a set of registered listeners. class CodeEventDispatcher : public CodeEventListener { public: using LogEventsAndTags = CodeEventListener::LogEventsAndTags; CodeEventDispatcher() = default; CodeEventDispatcher(const CodeEventDispatcher&) = delete; CodeEventDispatcher& operator=(const CodeEventDispatcher&) = delete; bool AddListener(CodeEventListener* listener) { base::MutexGuard guard(&mutex_); return listeners_.insert(listener).second; } void RemoveListener(CodeEventListener* listener) { base::MutexGuard guard(&mutex_); listeners_.erase(listener); } bool IsListeningToCodeEvents() { for (auto it : listeners_) { if (it->is_listening_to_code_events()) { return true; } } return false; } void DispatchEventToListeners( std::function<void(CodeEventListener*)> callback) { base::MutexGuard guard(&mutex_); for (CodeEventListener* listener : listeners_) { callback(listener); } } void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, const char* comment) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeCreateEvent(tag, code, comment); }); } void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, Handle<Name> name) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeCreateEvent(tag, code, name); }); } void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, Handle<SharedFunctionInfo> shared, Handle<Name> name) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeCreateEvent(tag, code, shared, name); }); } void CodeCreateEvent(LogEventsAndTags tag, Handle<AbstractCode> code, Handle<SharedFunctionInfo> shared, Handle<Name> source, int line, int column) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeCreateEvent(tag, code, shared, source, line, column); }); } #if V8_ENABLE_WEBASSEMBLY void CodeCreateEvent(LogEventsAndTags tag, const wasm::WasmCode* code, wasm::WasmName name, const char* source_url, int code_offset, int script_id) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeCreateEvent(tag, code, name, source_url, code_offset, script_id); }); } #endif // V8_ENABLE_WEBASSEMBLY void CallbackEvent(Handle<Name> name, Address entry_point) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CallbackEvent(name, entry_point); }); } void GetterCallbackEvent(Handle<Name> name, Address entry_point) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->GetterCallbackEvent(name, entry_point); }); } void SetterCallbackEvent(Handle<Name> name, Address entry_point) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->SetterCallbackEvent(name, entry_point); }); } void RegExpCodeCreateEvent(Handle<AbstractCode> code, Handle<String> source) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->RegExpCodeCreateEvent(code, source); }); } void CodeMoveEvent(AbstractCode from, AbstractCode to) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeMoveEvent(from, to); }); } void SharedFunctionInfoMoveEvent(Address from, Address to) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->SharedFunctionInfoMoveEvent(from, to); }); } void CodeMovingGCEvent() override { DispatchEventToListeners( [](CodeEventListener* listener) { listener->CodeMovingGCEvent(); }); } void CodeDisableOptEvent(Handle<AbstractCode> code, Handle<SharedFunctionInfo> shared) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeDisableOptEvent(code, shared); }); } void CodeDeoptEvent(Handle<Code> code, DeoptimizeKind kind, Address pc, int fp_to_sp_delta, bool reuse_code) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeDeoptEvent(code, kind, pc, fp_to_sp_delta, reuse_code); }); } void CodeDependencyChangeEvent(Handle<Code> code, Handle<SharedFunctionInfo> sfi, const char* reason) override { DispatchEventToListeners([=](CodeEventListener* listener) { listener->CodeDependencyChangeEvent(code, sfi, reason); }); } void WeakCodeClearEvent() override { DispatchEventToListeners( [=](CodeEventListener* listener) { listener->WeakCodeClearEvent(); }); } private: std::unordered_set<CodeEventListener*> listeners_; base::Mutex mutex_; }; } // namespace internal } // namespace v8 #endif // V8_LOGGING_CODE_EVENTS_H_