1 /* 2 * Copyright (c) 2010-2011 Google Inc. All rights reserved. 3 * 4 * Redistribution and use in source and binary forms, with or without 5 * modification, are permitted provided that the following conditions are 6 * met: 7 * 8 * * Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * * Redistributions in binary form must reproduce the above 11 * copyright notice, this list of conditions and the following disclaimer 12 * in the documentation and/or other materials provided with the 13 * distribution. 14 * * Neither the name of Google Inc. nor the names of its 15 * contributors may be used to endorse or promote products derived from 16 * this software without specific prior written permission. 17 * 18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 19 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 20 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 21 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 22 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 23 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 24 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 29 */ 30 31 #include "src/inspector/v8-inspector-impl.h" 32 33 #include "src/inspector/inspected-context.h" 34 #include "src/inspector/string-util.h" 35 #include "src/inspector/v8-console-agent-impl.h" 36 #include "src/inspector/v8-console-message.h" 37 #include "src/inspector/v8-debugger-agent-impl.h" 38 #include "src/inspector/v8-debugger.h" 39 #include "src/inspector/v8-inspector-session-impl.h" 40 #include "src/inspector/v8-profiler-agent-impl.h" 41 #include "src/inspector/v8-runtime-agent-impl.h" 42 #include "src/inspector/v8-stack-trace-impl.h" 43 44 namespace v8_inspector { 45 46 std::unique_ptr<V8Inspector> V8Inspector::create(v8::Isolate* isolate, 47 V8InspectorClient* client) { 48 return wrapUnique(new V8InspectorImpl(isolate, client)); 49 } 50 51 V8InspectorImpl::V8InspectorImpl(v8::Isolate* isolate, 52 V8InspectorClient* client) 53 : m_isolate(isolate), 54 m_client(client), 55 m_debugger(new V8Debugger(isolate, this)), 56 m_capturingStackTracesCount(0), 57 m_lastExceptionId(0) {} 58 59 V8InspectorImpl::~V8InspectorImpl() {} 60 61 V8DebuggerAgentImpl* V8InspectorImpl::enabledDebuggerAgentForGroup( 62 int contextGroupId) { 63 V8InspectorSessionImpl* session = sessionForContextGroup(contextGroupId); 64 V8DebuggerAgentImpl* agent = session ? session->debuggerAgent() : nullptr; 65 return agent && agent->enabled() ? agent : nullptr; 66 } 67 68 V8RuntimeAgentImpl* V8InspectorImpl::enabledRuntimeAgentForGroup( 69 int contextGroupId) { 70 V8InspectorSessionImpl* session = sessionForContextGroup(contextGroupId); 71 V8RuntimeAgentImpl* agent = session ? session->runtimeAgent() : nullptr; 72 return agent && agent->enabled() ? agent : nullptr; 73 } 74 75 V8ProfilerAgentImpl* V8InspectorImpl::enabledProfilerAgentForGroup( 76 int contextGroupId) { 77 V8InspectorSessionImpl* session = sessionForContextGroup(contextGroupId); 78 V8ProfilerAgentImpl* agent = session ? session->profilerAgent() : nullptr; 79 return agent && agent->enabled() ? agent : nullptr; 80 } 81 82 v8::MaybeLocal<v8::Value> V8InspectorImpl::runCompiledScript( 83 v8::Local<v8::Context> context, v8::Local<v8::Script> script) { 84 v8::MicrotasksScope microtasksScope(m_isolate, 85 v8::MicrotasksScope::kRunMicrotasks); 86 int groupId = V8Debugger::getGroupId(context); 87 if (V8DebuggerAgentImpl* agent = enabledDebuggerAgentForGroup(groupId)) 88 agent->willExecuteScript(script->GetUnboundScript()->GetId()); 89 v8::MaybeLocal<v8::Value> result = script->Run(context); 90 // Get agent from the map again, since it could have detached during script 91 // execution. 92 if (V8DebuggerAgentImpl* agent = enabledDebuggerAgentForGroup(groupId)) 93 agent->didExecuteScript(); 94 return result; 95 } 96 97 v8::MaybeLocal<v8::Value> V8InspectorImpl::callFunction( 98 v8::Local<v8::Function> function, v8::Local<v8::Context> context, 99 v8::Local<v8::Value> receiver, int argc, v8::Local<v8::Value> info[]) { 100 v8::MicrotasksScope microtasksScope(m_isolate, 101 v8::MicrotasksScope::kRunMicrotasks); 102 int groupId = V8Debugger::getGroupId(context); 103 if (V8DebuggerAgentImpl* agent = enabledDebuggerAgentForGroup(groupId)) 104 agent->willExecuteScript(function->ScriptId()); 105 v8::MaybeLocal<v8::Value> result = 106 function->Call(context, receiver, argc, info); 107 // Get agent from the map again, since it could have detached during script 108 // execution. 109 if (V8DebuggerAgentImpl* agent = enabledDebuggerAgentForGroup(groupId)) 110 agent->didExecuteScript(); 111 return result; 112 } 113 114 v8::MaybeLocal<v8::Value> V8InspectorImpl::compileAndRunInternalScript( 115 v8::Local<v8::Context> context, v8::Local<v8::String> source) { 116 v8::Local<v8::Script> script = 117 compileScript(context, source, String16(), true); 118 if (script.IsEmpty()) return v8::MaybeLocal<v8::Value>(); 119 v8::MicrotasksScope microtasksScope(m_isolate, 120 v8::MicrotasksScope::kDoNotRunMicrotasks); 121 return script->Run(context); 122 } 123 124 v8::Local<v8::Script> V8InspectorImpl::compileScript( 125 v8::Local<v8::Context> context, v8::Local<v8::String> code, 126 const String16& fileName, bool markAsInternal) { 127 v8::ScriptOrigin origin( 128 toV8String(m_isolate, fileName), v8::Integer::New(m_isolate, 0), 129 v8::Integer::New(m_isolate, 0), 130 v8::False(m_isolate), // sharable 131 v8::Local<v8::Integer>(), 132 v8::Boolean::New(m_isolate, markAsInternal), // internal 133 toV8String(m_isolate, String16()), // sourceMap 134 v8::True(m_isolate)); // opaqueresource 135 v8::ScriptCompiler::Source source(code, origin); 136 v8::Local<v8::Script> script; 137 if (!v8::ScriptCompiler::Compile(context, &source, 138 v8::ScriptCompiler::kNoCompileOptions) 139 .ToLocal(&script)) 140 return v8::Local<v8::Script>(); 141 return script; 142 } 143 144 void V8InspectorImpl::enableStackCapturingIfNeeded() { 145 if (!m_capturingStackTracesCount) 146 V8StackTraceImpl::setCaptureStackTraceForUncaughtExceptions(m_isolate, 147 true); 148 ++m_capturingStackTracesCount; 149 } 150 151 void V8InspectorImpl::disableStackCapturingIfNeeded() { 152 if (!(--m_capturingStackTracesCount)) 153 V8StackTraceImpl::setCaptureStackTraceForUncaughtExceptions(m_isolate, 154 false); 155 } 156 157 void V8InspectorImpl::muteExceptions(int contextGroupId) { 158 m_muteExceptionsMap[contextGroupId]++; 159 } 160 161 void V8InspectorImpl::unmuteExceptions(int contextGroupId) { 162 m_muteExceptionsMap[contextGroupId]--; 163 } 164 165 V8ConsoleMessageStorage* V8InspectorImpl::ensureConsoleMessageStorage( 166 int contextGroupId) { 167 ConsoleStorageMap::iterator storageIt = 168 m_consoleStorageMap.find(contextGroupId); 169 if (storageIt == m_consoleStorageMap.end()) 170 storageIt = 171 m_consoleStorageMap 172 .insert(std::make_pair( 173 contextGroupId, 174 wrapUnique(new V8ConsoleMessageStorage(this, contextGroupId)))) 175 .first; 176 return storageIt->second.get(); 177 } 178 179 bool V8InspectorImpl::hasConsoleMessageStorage(int contextGroupId) { 180 ConsoleStorageMap::iterator storageIt = 181 m_consoleStorageMap.find(contextGroupId); 182 return storageIt != m_consoleStorageMap.end(); 183 } 184 185 std::unique_ptr<V8StackTrace> V8InspectorImpl::createStackTrace( 186 v8::Local<v8::StackTrace> stackTrace) { 187 return m_debugger->createStackTrace(stackTrace); 188 } 189 190 std::unique_ptr<V8InspectorSession> V8InspectorImpl::connect( 191 int contextGroupId, V8Inspector::Channel* channel, 192 const StringView& state) { 193 DCHECK(m_sessions.find(contextGroupId) == m_sessions.cend()); 194 std::unique_ptr<V8InspectorSessionImpl> session = 195 V8InspectorSessionImpl::create(this, contextGroupId, channel, state); 196 m_sessions[contextGroupId] = session.get(); 197 return std::move(session); 198 } 199 200 void V8InspectorImpl::disconnect(V8InspectorSessionImpl* session) { 201 DCHECK(m_sessions.find(session->contextGroupId()) != m_sessions.end()); 202 m_sessions.erase(session->contextGroupId()); 203 } 204 205 InspectedContext* V8InspectorImpl::getContext(int groupId, 206 int contextId) const { 207 if (!groupId || !contextId) return nullptr; 208 209 ContextsByGroupMap::const_iterator contextGroupIt = m_contexts.find(groupId); 210 if (contextGroupIt == m_contexts.end()) return nullptr; 211 212 ContextByIdMap::iterator contextIt = contextGroupIt->second->find(contextId); 213 if (contextIt == contextGroupIt->second->end()) return nullptr; 214 215 return contextIt->second.get(); 216 } 217 218 void V8InspectorImpl::contextCreated(const V8ContextInfo& info) { 219 int contextId = m_debugger->markContext(info); 220 221 ContextsByGroupMap::iterator contextIt = m_contexts.find(info.contextGroupId); 222 if (contextIt == m_contexts.end()) 223 contextIt = m_contexts 224 .insert(std::make_pair(info.contextGroupId, 225 wrapUnique(new ContextByIdMap()))) 226 .first; 227 228 const auto& contextById = contextIt->second; 229 230 DCHECK(contextById->find(contextId) == contextById->cend()); 231 InspectedContext* context = new InspectedContext(this, info, contextId); 232 (*contextById)[contextId] = wrapUnique(context); 233 SessionMap::iterator sessionIt = m_sessions.find(info.contextGroupId); 234 if (sessionIt != m_sessions.end()) 235 sessionIt->second->runtimeAgent()->reportExecutionContextCreated(context); 236 } 237 238 void V8InspectorImpl::contextDestroyed(v8::Local<v8::Context> context) { 239 int contextId = V8Debugger::contextId(context); 240 int contextGroupId = V8Debugger::getGroupId(context); 241 242 ConsoleStorageMap::iterator storageIt = 243 m_consoleStorageMap.find(contextGroupId); 244 if (storageIt != m_consoleStorageMap.end()) 245 storageIt->second->contextDestroyed(contextId); 246 247 InspectedContext* inspectedContext = getContext(contextGroupId, contextId); 248 if (!inspectedContext) return; 249 250 SessionMap::iterator iter = m_sessions.find(contextGroupId); 251 if (iter != m_sessions.end()) 252 iter->second->runtimeAgent()->reportExecutionContextDestroyed( 253 inspectedContext); 254 discardInspectedContext(contextGroupId, contextId); 255 } 256 257 void V8InspectorImpl::resetContextGroup(int contextGroupId) { 258 m_consoleStorageMap.erase(contextGroupId); 259 m_muteExceptionsMap.erase(contextGroupId); 260 SessionMap::iterator session = m_sessions.find(contextGroupId); 261 if (session != m_sessions.end()) session->second->reset(); 262 m_contexts.erase(contextGroupId); 263 } 264 265 void V8InspectorImpl::willExecuteScript(v8::Local<v8::Context> context, 266 int scriptId) { 267 if (V8DebuggerAgentImpl* agent = 268 enabledDebuggerAgentForGroup(V8Debugger::getGroupId(context))) 269 agent->willExecuteScript(scriptId); 270 } 271 272 void V8InspectorImpl::didExecuteScript(v8::Local<v8::Context> context) { 273 if (V8DebuggerAgentImpl* agent = 274 enabledDebuggerAgentForGroup(V8Debugger::getGroupId(context))) 275 agent->didExecuteScript(); 276 } 277 278 void V8InspectorImpl::idleStarted() { 279 for (auto it = m_sessions.begin(); it != m_sessions.end(); ++it) { 280 if (it->second->profilerAgent()->idleStarted()) return; 281 } 282 } 283 284 void V8InspectorImpl::idleFinished() { 285 for (auto it = m_sessions.begin(); it != m_sessions.end(); ++it) { 286 if (it->second->profilerAgent()->idleFinished()) return; 287 } 288 } 289 290 unsigned V8InspectorImpl::exceptionThrown( 291 v8::Local<v8::Context> context, const StringView& message, 292 v8::Local<v8::Value> exception, const StringView& detailedMessage, 293 const StringView& url, unsigned lineNumber, unsigned columnNumber, 294 std::unique_ptr<V8StackTrace> stackTrace, int scriptId) { 295 int contextGroupId = V8Debugger::getGroupId(context); 296 if (!contextGroupId || m_muteExceptionsMap[contextGroupId]) return 0; 297 std::unique_ptr<V8StackTraceImpl> stackTraceImpl = 298 wrapUnique(static_cast<V8StackTraceImpl*>(stackTrace.release())); 299 unsigned exceptionId = nextExceptionId(); 300 std::unique_ptr<V8ConsoleMessage> consoleMessage = 301 V8ConsoleMessage::createForException( 302 m_client->currentTimeMS(), toString16(detailedMessage), 303 toString16(url), lineNumber, columnNumber, std::move(stackTraceImpl), 304 scriptId, m_isolate, toString16(message), 305 V8Debugger::contextId(context), exception, exceptionId); 306 ensureConsoleMessageStorage(contextGroupId) 307 ->addMessage(std::move(consoleMessage)); 308 return exceptionId; 309 } 310 311 void V8InspectorImpl::exceptionRevoked(v8::Local<v8::Context> context, 312 unsigned exceptionId, 313 const StringView& message) { 314 int contextGroupId = V8Debugger::getGroupId(context); 315 if (!contextGroupId) return; 316 317 std::unique_ptr<V8ConsoleMessage> consoleMessage = 318 V8ConsoleMessage::createForRevokedException( 319 m_client->currentTimeMS(), toString16(message), exceptionId); 320 ensureConsoleMessageStorage(contextGroupId) 321 ->addMessage(std::move(consoleMessage)); 322 } 323 324 std::unique_ptr<V8StackTrace> V8InspectorImpl::captureStackTrace( 325 bool fullStack) { 326 return m_debugger->captureStackTrace(fullStack); 327 } 328 329 void V8InspectorImpl::asyncTaskScheduled(const StringView& taskName, void* task, 330 bool recurring) { 331 m_debugger->asyncTaskScheduled(taskName, task, recurring); 332 } 333 334 void V8InspectorImpl::asyncTaskCanceled(void* task) { 335 m_debugger->asyncTaskCanceled(task); 336 } 337 338 void V8InspectorImpl::asyncTaskStarted(void* task) { 339 m_debugger->asyncTaskStarted(task); 340 } 341 342 void V8InspectorImpl::asyncTaskFinished(void* task) { 343 m_debugger->asyncTaskFinished(task); 344 } 345 346 void V8InspectorImpl::allAsyncTasksCanceled() { 347 m_debugger->allAsyncTasksCanceled(); 348 } 349 350 v8::Local<v8::Context> V8InspectorImpl::regexContext() { 351 if (m_regexContext.IsEmpty()) 352 m_regexContext.Reset(m_isolate, v8::Context::New(m_isolate)); 353 return m_regexContext.Get(m_isolate); 354 } 355 356 void V8InspectorImpl::discardInspectedContext(int contextGroupId, 357 int contextId) { 358 if (!getContext(contextGroupId, contextId)) return; 359 m_contexts[contextGroupId]->erase(contextId); 360 if (m_contexts[contextGroupId]->empty()) m_contexts.erase(contextGroupId); 361 } 362 363 const V8InspectorImpl::ContextByIdMap* V8InspectorImpl::contextGroup( 364 int contextGroupId) { 365 ContextsByGroupMap::iterator iter = m_contexts.find(contextGroupId); 366 return iter == m_contexts.end() ? nullptr : iter->second.get(); 367 } 368 369 V8InspectorSessionImpl* V8InspectorImpl::sessionForContextGroup( 370 int contextGroupId) { 371 if (!contextGroupId) return nullptr; 372 SessionMap::iterator iter = m_sessions.find(contextGroupId); 373 return iter == m_sessions.end() ? nullptr : iter->second; 374 } 375 376 } // namespace v8_inspector 377