1 // Copyright 2016 the V8 project authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #ifndef V8_CODE_EVENTS_H_ 6 #define V8_CODE_EVENTS_H_ 7 8 #include <unordered_set> 9 10 #include "src/base/platform/mutex.h" 11 #include "src/globals.h" 12 13 namespace v8 { 14 namespace internal { 15 16 class AbstractCode; 17 class Name; 18 class SharedFunctionInfo; 19 class String; 20 21 #define LOG_EVENTS_AND_TAGS_LIST(V) \ 22 V(CODE_CREATION_EVENT, "code-creation") \ 23 V(CODE_DISABLE_OPT_EVENT, "code-disable-optimization") \ 24 V(CODE_MOVE_EVENT, "code-move") \ 25 V(CODE_DELETE_EVENT, "code-delete") \ 26 V(CODE_MOVING_GC, "code-moving-gc") \ 27 V(SHARED_FUNC_MOVE_EVENT, "sfi-move") \ 28 V(SNAPSHOT_CODE_NAME_EVENT, "snapshot-code-name") \ 29 V(TICK_EVENT, "tick") \ 30 V(REPEAT_META_EVENT, "repeat") \ 31 V(BUILTIN_TAG, "Builtin") \ 32 V(CALL_DEBUG_BREAK_TAG, "CallDebugBreak") \ 33 V(CALL_DEBUG_PREPARE_STEP_IN_TAG, "CallDebugPrepareStepIn") \ 34 V(CALL_INITIALIZE_TAG, "CallInitialize") \ 35 V(CALL_MEGAMORPHIC_TAG, "CallMegamorphic") \ 36 V(CALL_MISS_TAG, "CallMiss") \ 37 V(CALL_NORMAL_TAG, "CallNormal") \ 38 V(LOAD_INITIALIZE_TAG, "LoadInitialize") \ 39 V(LOAD_MEGAMORPHIC_TAG, "LoadMegamorphic") \ 40 V(STORE_INITIALIZE_TAG, "StoreInitialize") \ 41 V(STORE_GENERIC_TAG, "StoreGeneric") \ 42 V(STORE_MEGAMORPHIC_TAG, "StoreMegamorphic") \ 43 V(KEYED_CALL_DEBUG_BREAK_TAG, "KeyedCallDebugBreak") \ 44 V(KEYED_CALL_DEBUG_PREPARE_STEP_IN_TAG, "KeyedCallDebugPrepareStepIn") \ 45 V(KEYED_CALL_INITIALIZE_TAG, "KeyedCallInitialize") \ 46 V(KEYED_CALL_MEGAMORPHIC_TAG, "KeyedCallMegamorphic") \ 47 V(KEYED_CALL_MISS_TAG, "KeyedCallMiss") \ 48 V(KEYED_CALL_NORMAL_TAG, "KeyedCallNormal") \ 49 V(CALLBACK_TAG, "Callback") \ 50 V(EVAL_TAG, "Eval") \ 51 V(FUNCTION_TAG, "Function") \ 52 V(HANDLER_TAG, "Handler") \ 53 V(BYTECODE_HANDLER_TAG, "BytecodeHandler") \ 54 V(KEYED_LOAD_IC_TAG, "KeyedLoadIC") \ 55 V(KEYED_LOAD_POLYMORPHIC_IC_TAG, "KeyedLoadPolymorphicIC") \ 56 V(KEYED_EXTERNAL_ARRAY_LOAD_IC_TAG, "KeyedExternalArrayLoadIC") \ 57 V(KEYED_STORE_IC_TAG, "KeyedStoreIC") \ 58 V(KEYED_STORE_POLYMORPHIC_IC_TAG, "KeyedStorePolymorphicIC") \ 59 V(KEYED_EXTERNAL_ARRAY_STORE_IC_TAG, "KeyedExternalArrayStoreIC") \ 60 V(LAZY_COMPILE_TAG, "LazyCompile") \ 61 V(CALL_IC_TAG, "CallIC") \ 62 V(LOAD_IC_TAG, "LoadIC") \ 63 V(LOAD_GLOBAL_IC_TAG, "LoadGlobalIC") \ 64 V(LOAD_POLYMORPHIC_IC_TAG, "LoadPolymorphicIC") \ 65 V(REG_EXP_TAG, "RegExp") \ 66 V(SCRIPT_TAG, "Script") \ 67 V(STORE_IC_TAG, "StoreIC") \ 68 V(STORE_POLYMORPHIC_IC_TAG, "StorePolymorphicIC") \ 69 V(STUB_TAG, "Stub") \ 70 V(NATIVE_FUNCTION_TAG, "Function") \ 71 V(NATIVE_LAZY_COMPILE_TAG, "LazyCompile") \ 72 V(NATIVE_SCRIPT_TAG, "Script") 73 // Note that 'NATIVE_' cases for functions and scripts are mapped onto 74 // original tags when writing to the log. 75 76 #define PROFILE(the_isolate, Call) (the_isolate)->code_event_dispatcher()->Call; 77 78 class CodeEventListener { 79 public: 80 #define DECLARE_ENUM(enum_item, _) enum_item, 81 enum LogEventsAndTags { 82 LOG_EVENTS_AND_TAGS_LIST(DECLARE_ENUM) NUMBER_OF_LOG_EVENTS 83 }; 84 #undef DECLARE_ENUM 85 ~CodeEventListener()86 virtual ~CodeEventListener() {} 87 88 virtual void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 89 const char* comment) = 0; 90 virtual void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 91 Name* name) = 0; 92 virtual void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 93 SharedFunctionInfo* shared, Name* name) = 0; 94 virtual void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 95 SharedFunctionInfo* shared, Name* source, 96 int line, int column) = 0; 97 virtual void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 98 int args_count) = 0; 99 virtual void CallbackEvent(Name* name, Address entry_point) = 0; 100 virtual void GetterCallbackEvent(Name* name, Address entry_point) = 0; 101 virtual void SetterCallbackEvent(Name* name, Address entry_point) = 0; 102 virtual void RegExpCodeCreateEvent(AbstractCode* code, String* source) = 0; 103 virtual void CodeMoveEvent(AbstractCode* from, Address to) = 0; 104 virtual void SharedFunctionInfoMoveEvent(Address from, Address to) = 0; 105 virtual void CodeMovingGCEvent() = 0; 106 virtual void CodeDisableOptEvent(AbstractCode* code, 107 SharedFunctionInfo* shared) = 0; 108 virtual void CodeDeoptEvent(Code* code, Address pc, int fp_to_sp_delta) = 0; 109 }; 110 111 class CodeEventDispatcher { 112 public: 113 using LogEventsAndTags = CodeEventListener::LogEventsAndTags; 114 CodeEventDispatcher()115 CodeEventDispatcher() {} 116 AddListener(CodeEventListener * listener)117 bool AddListener(CodeEventListener* listener) { 118 base::LockGuard<base::Mutex> guard(&mutex_); 119 return listeners_.insert(listener).second; 120 } RemoveListener(CodeEventListener * listener)121 void RemoveListener(CodeEventListener* listener) { 122 base::LockGuard<base::Mutex> guard(&mutex_); 123 listeners_.erase(listener); 124 } 125 126 #define CODE_EVENT_DISPATCH(code) \ 127 base::LockGuard<base::Mutex> guard(&mutex_); \ 128 for (auto it = listeners_.begin(); it != listeners_.end(); ++it) (*it)->code 129 CodeCreateEvent(LogEventsAndTags tag,AbstractCode * code,const char * comment)130 void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 131 const char* comment) { 132 CODE_EVENT_DISPATCH(CodeCreateEvent(tag, code, comment)); 133 } CodeCreateEvent(LogEventsAndTags tag,AbstractCode * code,Name * name)134 void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, Name* name) { 135 CODE_EVENT_DISPATCH(CodeCreateEvent(tag, code, name)); 136 } CodeCreateEvent(LogEventsAndTags tag,AbstractCode * code,SharedFunctionInfo * shared,Name * name)137 void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 138 SharedFunctionInfo* shared, Name* name) { 139 CODE_EVENT_DISPATCH(CodeCreateEvent(tag, code, shared, name)); 140 } CodeCreateEvent(LogEventsAndTags tag,AbstractCode * code,SharedFunctionInfo * shared,Name * source,int line,int column)141 void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 142 SharedFunctionInfo* shared, Name* source, int line, 143 int column) { 144 CODE_EVENT_DISPATCH( 145 CodeCreateEvent(tag, code, shared, source, line, column)); 146 } CodeCreateEvent(LogEventsAndTags tag,AbstractCode * code,int args_count)147 void CodeCreateEvent(LogEventsAndTags tag, AbstractCode* code, 148 int args_count) { 149 CODE_EVENT_DISPATCH(CodeCreateEvent(tag, code, args_count)); 150 } CallbackEvent(Name * name,Address entry_point)151 void CallbackEvent(Name* name, Address entry_point) { 152 CODE_EVENT_DISPATCH(CallbackEvent(name, entry_point)); 153 } GetterCallbackEvent(Name * name,Address entry_point)154 void GetterCallbackEvent(Name* name, Address entry_point) { 155 CODE_EVENT_DISPATCH(GetterCallbackEvent(name, entry_point)); 156 } SetterCallbackEvent(Name * name,Address entry_point)157 void SetterCallbackEvent(Name* name, Address entry_point) { 158 CODE_EVENT_DISPATCH(SetterCallbackEvent(name, entry_point)); 159 } RegExpCodeCreateEvent(AbstractCode * code,String * source)160 void RegExpCodeCreateEvent(AbstractCode* code, String* source) { 161 CODE_EVENT_DISPATCH(RegExpCodeCreateEvent(code, source)); 162 } CodeMoveEvent(AbstractCode * from,Address to)163 void CodeMoveEvent(AbstractCode* from, Address to) { 164 CODE_EVENT_DISPATCH(CodeMoveEvent(from, to)); 165 } SharedFunctionInfoMoveEvent(Address from,Address to)166 void SharedFunctionInfoMoveEvent(Address from, Address to) { 167 CODE_EVENT_DISPATCH(SharedFunctionInfoMoveEvent(from, to)); 168 } CodeMovingGCEvent()169 void CodeMovingGCEvent() { CODE_EVENT_DISPATCH(CodeMovingGCEvent()); } CodeDisableOptEvent(AbstractCode * code,SharedFunctionInfo * shared)170 void CodeDisableOptEvent(AbstractCode* code, SharedFunctionInfo* shared) { 171 CODE_EVENT_DISPATCH(CodeDisableOptEvent(code, shared)); 172 } CodeDeoptEvent(Code * code,Address pc,int fp_to_sp_delta)173 void CodeDeoptEvent(Code* code, Address pc, int fp_to_sp_delta) { 174 CODE_EVENT_DISPATCH(CodeDeoptEvent(code, pc, fp_to_sp_delta)); 175 } 176 #undef CODE_EVENT_DISPATCH 177 178 private: 179 std::unordered_set<CodeEventListener*> listeners_; 180 base::Mutex mutex_; 181 182 DISALLOW_COPY_AND_ASSIGN(CodeEventDispatcher); 183 }; 184 185 } // namespace internal 186 } // namespace v8 187 188 #endif // V8_CODE_EVENTS_H_ 189