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 <vector>
34 
35 #include "src/base/platform/mutex.h"
36 #include "src/inspector/inspected-context.h"
37 #include "src/inspector/string-util.h"
38 #include "src/inspector/v8-console-agent-impl.h"
39 #include "src/inspector/v8-console-message.h"
40 #include "src/inspector/v8-console.h"
41 #include "src/inspector/v8-debugger-agent-impl.h"
42 #include "src/inspector/v8-debugger.h"
43 #include "src/inspector/v8-inspector-session-impl.h"
44 #include "src/inspector/v8-profiler-agent-impl.h"
45 #include "src/inspector/v8-runtime-agent-impl.h"
46 #include "src/inspector/v8-stack-trace-impl.h"
47 
48 #include "include/v8-platform.h"
49 
50 namespace v8_inspector {
51 
create(v8::Isolate * isolate,V8InspectorClient * client)52 std::unique_ptr<V8Inspector> V8Inspector::create(v8::Isolate* isolate,
53                                                  V8InspectorClient* client) {
54   return std::unique_ptr<V8Inspector>(new V8InspectorImpl(isolate, client));
55 }
56 
V8InspectorImpl(v8::Isolate * isolate,V8InspectorClient * client)57 V8InspectorImpl::V8InspectorImpl(v8::Isolate* isolate,
58                                  V8InspectorClient* client)
59     : m_isolate(isolate),
60       m_client(client),
61       m_debugger(new V8Debugger(isolate, this)),
62       m_capturingStackTracesCount(0),
63       m_lastExceptionId(0),
64       m_lastContextId(0),
65       m_isolateId(v8::debug::GetNextRandomInt64(m_isolate)) {
66   v8::debug::SetInspector(m_isolate, this);
67   v8::debug::SetConsoleDelegate(m_isolate, console());
68 }
69 
~V8InspectorImpl()70 V8InspectorImpl::~V8InspectorImpl() {
71   v8::debug::SetInspector(m_isolate, nullptr);
72   v8::debug::SetConsoleDelegate(m_isolate, nullptr);
73 }
74 
contextGroupId(v8::Local<v8::Context> context) const75 int V8InspectorImpl::contextGroupId(v8::Local<v8::Context> context) const {
76   return contextGroupId(InspectedContext::contextId(context));
77 }
78 
contextGroupId(int contextId) const79 int V8InspectorImpl::contextGroupId(int contextId) const {
80   auto it = m_contextIdToGroupIdMap.find(contextId);
81   return it != m_contextIdToGroupIdMap.end() ? it->second : 0;
82 }
83 
compileAndRunInternalScript(v8::Local<v8::Context> context,v8::Local<v8::String> source)84 v8::MaybeLocal<v8::Value> V8InspectorImpl::compileAndRunInternalScript(
85     v8::Local<v8::Context> context, v8::Local<v8::String> source) {
86   v8::Local<v8::UnboundScript> unboundScript;
87   if (!v8::debug::CompileInspectorScript(m_isolate, source)
88            .ToLocal(&unboundScript))
89     return v8::MaybeLocal<v8::Value>();
90   v8::MicrotasksScope microtasksScope(m_isolate,
91                                       v8::MicrotasksScope::kDoNotRunMicrotasks);
92   v8::Context::Scope contextScope(context);
93   v8::Isolate::SafeForTerminationScope allowTermination(m_isolate);
94   return unboundScript->BindToCurrentContext()->Run(context);
95 }
96 
compileScript(v8::Local<v8::Context> context,const String16 & code,const String16 & fileName)97 v8::MaybeLocal<v8::Script> V8InspectorImpl::compileScript(
98     v8::Local<v8::Context> context, const String16& code,
99     const String16& fileName) {
100   v8::ScriptOrigin origin(
101       toV8String(m_isolate, fileName), v8::Integer::New(m_isolate, 0),
102       v8::Integer::New(m_isolate, 0),
103       v8::False(m_isolate),                                         // sharable
104       v8::Local<v8::Integer>(), toV8String(m_isolate, String16()),  // sourceMap
105       v8::True(m_isolate));  // opaqueresource
106   v8::ScriptCompiler::Source source(toV8String(m_isolate, code), origin);
107   return v8::ScriptCompiler::Compile(context, &source,
108                                      v8::ScriptCompiler::kNoCompileOptions);
109 }
110 
enableStackCapturingIfNeeded()111 void V8InspectorImpl::enableStackCapturingIfNeeded() {
112   if (!m_capturingStackTracesCount)
113     V8StackTraceImpl::setCaptureStackTraceForUncaughtExceptions(m_isolate,
114                                                                 true);
115   ++m_capturingStackTracesCount;
116 }
117 
disableStackCapturingIfNeeded()118 void V8InspectorImpl::disableStackCapturingIfNeeded() {
119   if (!(--m_capturingStackTracesCount))
120     V8StackTraceImpl::setCaptureStackTraceForUncaughtExceptions(m_isolate,
121                                                                 false);
122 }
123 
muteExceptions(int contextGroupId)124 void V8InspectorImpl::muteExceptions(int contextGroupId) {
125   m_muteExceptionsMap[contextGroupId]++;
126 }
127 
unmuteExceptions(int contextGroupId)128 void V8InspectorImpl::unmuteExceptions(int contextGroupId) {
129   m_muteExceptionsMap[contextGroupId]--;
130 }
131 
ensureConsoleMessageStorage(int contextGroupId)132 V8ConsoleMessageStorage* V8InspectorImpl::ensureConsoleMessageStorage(
133     int contextGroupId) {
134   ConsoleStorageMap::iterator storageIt =
135       m_consoleStorageMap.find(contextGroupId);
136   if (storageIt == m_consoleStorageMap.end())
137     storageIt = m_consoleStorageMap
138                     .insert(std::make_pair(
139                         contextGroupId,
140                         std::unique_ptr<V8ConsoleMessageStorage>(
141                             new V8ConsoleMessageStorage(this, contextGroupId))))
142                     .first;
143   return storageIt->second.get();
144 }
145 
hasConsoleMessageStorage(int contextGroupId)146 bool V8InspectorImpl::hasConsoleMessageStorage(int contextGroupId) {
147   ConsoleStorageMap::iterator storageIt =
148       m_consoleStorageMap.find(contextGroupId);
149   return storageIt != m_consoleStorageMap.end();
150 }
151 
createStackTrace(v8::Local<v8::StackTrace> stackTrace)152 std::unique_ptr<V8StackTrace> V8InspectorImpl::createStackTrace(
153     v8::Local<v8::StackTrace> stackTrace) {
154   return m_debugger->createStackTrace(stackTrace);
155 }
156 
connect(int contextGroupId,V8Inspector::Channel * channel,const StringView & state)157 std::unique_ptr<V8InspectorSession> V8InspectorImpl::connect(
158     int contextGroupId, V8Inspector::Channel* channel,
159     const StringView& state) {
160   int sessionId = ++m_lastSessionId;
161   std::unique_ptr<V8InspectorSessionImpl> session =
162       V8InspectorSessionImpl::create(this, contextGroupId, sessionId, channel,
163                                      state);
164   m_sessions[contextGroupId][sessionId] = session.get();
165   return std::move(session);
166 }
167 
disconnect(V8InspectorSessionImpl * session)168 void V8InspectorImpl::disconnect(V8InspectorSessionImpl* session) {
169   auto& map = m_sessions[session->contextGroupId()];
170   map.erase(session->sessionId());
171   if (map.empty()) m_sessions.erase(session->contextGroupId());
172 }
173 
getContext(int groupId,int contextId) const174 InspectedContext* V8InspectorImpl::getContext(int groupId,
175                                               int contextId) const {
176   if (!groupId || !contextId) return nullptr;
177 
178   ContextsByGroupMap::const_iterator contextGroupIt = m_contexts.find(groupId);
179   if (contextGroupIt == m_contexts.end()) return nullptr;
180 
181   ContextByIdMap::iterator contextIt = contextGroupIt->second->find(contextId);
182   if (contextIt == contextGroupIt->second->end()) return nullptr;
183 
184   return contextIt->second.get();
185 }
186 
getContext(int contextId) const187 InspectedContext* V8InspectorImpl::getContext(int contextId) const {
188   return getContext(contextGroupId(contextId), contextId);
189 }
190 
contextById(int groupId,v8::Maybe<int> contextId)191 v8::MaybeLocal<v8::Context> V8InspectorImpl::contextById(
192     int groupId, v8::Maybe<int> contextId) {
193   if (contextId.IsNothing()) {
194     v8::Local<v8::Context> context =
195         client()->ensureDefaultContextInGroup(groupId);
196     return context.IsEmpty() ? v8::MaybeLocal<v8::Context>() : context;
197   }
198   InspectedContext* context = getContext(contextId.FromJust());
199   return context ? context->context() : v8::MaybeLocal<v8::Context>();
200 }
201 
contextCreated(const V8ContextInfo & info)202 void V8InspectorImpl::contextCreated(const V8ContextInfo& info) {
203   int contextId = ++m_lastContextId;
204   InspectedContext* context = new InspectedContext(this, info, contextId);
205   m_contextIdToGroupIdMap[contextId] = info.contextGroupId;
206 
207   ContextsByGroupMap::iterator contextIt = m_contexts.find(info.contextGroupId);
208   if (contextIt == m_contexts.end())
209     contextIt = m_contexts
210                     .insert(std::make_pair(
211                         info.contextGroupId,
212                         std::unique_ptr<ContextByIdMap>(new ContextByIdMap())))
213                     .first;
214   const auto& contextById = contextIt->second;
215 
216   DCHECK(contextById->find(contextId) == contextById->cend());
217   (*contextById)[contextId].reset(context);
218   forEachSession(
219       info.contextGroupId, [&context](V8InspectorSessionImpl* session) {
220         session->runtimeAgent()->addBindings(context);
221         session->runtimeAgent()->reportExecutionContextCreated(context);
222       });
223 }
224 
contextDestroyed(v8::Local<v8::Context> context)225 void V8InspectorImpl::contextDestroyed(v8::Local<v8::Context> context) {
226   int contextId = InspectedContext::contextId(context);
227   int groupId = contextGroupId(context);
228   contextCollected(groupId, contextId);
229 }
230 
contextCollected(int groupId,int contextId)231 void V8InspectorImpl::contextCollected(int groupId, int contextId) {
232   m_contextIdToGroupIdMap.erase(contextId);
233 
234   ConsoleStorageMap::iterator storageIt = m_consoleStorageMap.find(groupId);
235   if (storageIt != m_consoleStorageMap.end())
236     storageIt->second->contextDestroyed(contextId);
237 
238   InspectedContext* inspectedContext = getContext(groupId, contextId);
239   if (!inspectedContext) return;
240 
241   forEachSession(groupId, [&inspectedContext](V8InspectorSessionImpl* session) {
242     session->runtimeAgent()->reportExecutionContextDestroyed(inspectedContext);
243   });
244   discardInspectedContext(groupId, contextId);
245 }
246 
resetContextGroup(int contextGroupId)247 void V8InspectorImpl::resetContextGroup(int contextGroupId) {
248   m_consoleStorageMap.erase(contextGroupId);
249   m_muteExceptionsMap.erase(contextGroupId);
250   forEachSession(contextGroupId,
251                  [](V8InspectorSessionImpl* session) { session->reset(); });
252   m_contexts.erase(contextGroupId);
253   m_debugger->wasmTranslation()->Clear();
254 }
255 
idleStarted()256 void V8InspectorImpl::idleStarted() { m_isolate->SetIdle(true); }
257 
idleFinished()258 void V8InspectorImpl::idleFinished() { m_isolate->SetIdle(false); }
259 
exceptionThrown(v8::Local<v8::Context> context,const StringView & message,v8::Local<v8::Value> exception,const StringView & detailedMessage,const StringView & url,unsigned lineNumber,unsigned columnNumber,std::unique_ptr<V8StackTrace> stackTrace,int scriptId)260 unsigned V8InspectorImpl::exceptionThrown(
261     v8::Local<v8::Context> context, const StringView& message,
262     v8::Local<v8::Value> exception, const StringView& detailedMessage,
263     const StringView& url, unsigned lineNumber, unsigned columnNumber,
264     std::unique_ptr<V8StackTrace> stackTrace, int scriptId) {
265   int groupId = contextGroupId(context);
266   if (!groupId || m_muteExceptionsMap[groupId]) return 0;
267   std::unique_ptr<V8StackTraceImpl> stackTraceImpl(
268       static_cast<V8StackTraceImpl*>(stackTrace.release()));
269   unsigned exceptionId = nextExceptionId();
270   std::unique_ptr<V8ConsoleMessage> consoleMessage =
271       V8ConsoleMessage::createForException(
272           m_client->currentTimeMS(), toString16(detailedMessage),
273           toString16(url), lineNumber, columnNumber, std::move(stackTraceImpl),
274           scriptId, m_isolate, toString16(message),
275           InspectedContext::contextId(context), exception, exceptionId);
276   ensureConsoleMessageStorage(groupId)->addMessage(std::move(consoleMessage));
277   return exceptionId;
278 }
279 
exceptionRevoked(v8::Local<v8::Context> context,unsigned exceptionId,const StringView & message)280 void V8InspectorImpl::exceptionRevoked(v8::Local<v8::Context> context,
281                                        unsigned exceptionId,
282                                        const StringView& message) {
283   int groupId = contextGroupId(context);
284   if (!groupId) return;
285 
286   std::unique_ptr<V8ConsoleMessage> consoleMessage =
287       V8ConsoleMessage::createForRevokedException(
288           m_client->currentTimeMS(), toString16(message), exceptionId);
289   ensureConsoleMessageStorage(groupId)->addMessage(std::move(consoleMessage));
290 }
291 
captureStackTrace(bool fullStack)292 std::unique_ptr<V8StackTrace> V8InspectorImpl::captureStackTrace(
293     bool fullStack) {
294   return m_debugger->captureStackTrace(fullStack);
295 }
296 
storeCurrentStackTrace(const StringView & description)297 V8StackTraceId V8InspectorImpl::storeCurrentStackTrace(
298     const StringView& description) {
299   return m_debugger->storeCurrentStackTrace(description);
300 }
301 
externalAsyncTaskStarted(const V8StackTraceId & parent)302 void V8InspectorImpl::externalAsyncTaskStarted(const V8StackTraceId& parent) {
303   m_debugger->externalAsyncTaskStarted(parent);
304 }
305 
externalAsyncTaskFinished(const V8StackTraceId & parent)306 void V8InspectorImpl::externalAsyncTaskFinished(const V8StackTraceId& parent) {
307   m_debugger->externalAsyncTaskFinished(parent);
308 }
309 
asyncTaskScheduled(const StringView & taskName,void * task,bool recurring)310 void V8InspectorImpl::asyncTaskScheduled(const StringView& taskName, void* task,
311                                          bool recurring) {
312   if (!task) return;
313   m_debugger->asyncTaskScheduled(taskName, task, recurring);
314 }
315 
asyncTaskCanceled(void * task)316 void V8InspectorImpl::asyncTaskCanceled(void* task) {
317   if (!task) return;
318   m_debugger->asyncTaskCanceled(task);
319 }
320 
asyncTaskStarted(void * task)321 void V8InspectorImpl::asyncTaskStarted(void* task) {
322   if (!task) return;
323   m_debugger->asyncTaskStarted(task);
324 }
325 
asyncTaskFinished(void * task)326 void V8InspectorImpl::asyncTaskFinished(void* task) {
327   if (!task) return;
328   m_debugger->asyncTaskFinished(task);
329 }
330 
allAsyncTasksCanceled()331 void V8InspectorImpl::allAsyncTasksCanceled() {
332   m_debugger->allAsyncTasksCanceled();
333 }
334 
regexContext()335 v8::Local<v8::Context> V8InspectorImpl::regexContext() {
336   if (m_regexContext.IsEmpty())
337     m_regexContext.Reset(m_isolate, v8::Context::New(m_isolate));
338   return m_regexContext.Get(m_isolate);
339 }
340 
discardInspectedContext(int contextGroupId,int contextId)341 void V8InspectorImpl::discardInspectedContext(int contextGroupId,
342                                               int contextId) {
343   if (!getContext(contextGroupId, contextId)) return;
344   m_contexts[contextGroupId]->erase(contextId);
345   if (m_contexts[contextGroupId]->empty()) m_contexts.erase(contextGroupId);
346 }
347 
sessionById(int contextGroupId,int sessionId)348 V8InspectorSessionImpl* V8InspectorImpl::sessionById(int contextGroupId,
349                                                      int sessionId) {
350   auto it = m_sessions.find(contextGroupId);
351   if (it == m_sessions.end()) return nullptr;
352   auto it2 = it->second.find(sessionId);
353   return it2 == it->second.end() ? nullptr : it2->second;
354 }
355 
console()356 V8Console* V8InspectorImpl::console() {
357   if (!m_console) m_console.reset(new V8Console(this));
358   return m_console.get();
359 }
360 
forEachContext(int contextGroupId,std::function<void (InspectedContext *)> callback)361 void V8InspectorImpl::forEachContext(
362     int contextGroupId, std::function<void(InspectedContext*)> callback) {
363   auto it = m_contexts.find(contextGroupId);
364   if (it == m_contexts.end()) return;
365   std::vector<int> ids;
366   ids.reserve(it->second->size());
367   for (auto& contextIt : *(it->second)) ids.push_back(contextIt.first);
368 
369   // Retrieve by ids each time since |callback| may destroy some contexts.
370   for (auto& contextId : ids) {
371     it = m_contexts.find(contextGroupId);
372     if (it == m_contexts.end()) continue;
373     auto contextIt = it->second->find(contextId);
374     if (contextIt != it->second->end()) callback(contextIt->second.get());
375   }
376 }
377 
forEachSession(int contextGroupId,std::function<void (V8InspectorSessionImpl *)> callback)378 void V8InspectorImpl::forEachSession(
379     int contextGroupId, std::function<void(V8InspectorSessionImpl*)> callback) {
380   auto it = m_sessions.find(contextGroupId);
381   if (it == m_sessions.end()) return;
382   std::vector<int> ids;
383   ids.reserve(it->second.size());
384   for (auto& sessionIt : it->second) ids.push_back(sessionIt.first);
385 
386   // Retrieve by ids each time since |callback| may destroy some contexts.
387   for (auto& sessionId : ids) {
388     it = m_sessions.find(contextGroupId);
389     if (it == m_sessions.end()) continue;
390     auto sessionIt = it->second.find(sessionId);
391     if (sessionIt != it->second.end()) callback(sessionIt->second);
392   }
393 }
394 
EvaluateScope(v8::Isolate * isolate)395 V8InspectorImpl::EvaluateScope::EvaluateScope(v8::Isolate* isolate)
396     : m_isolate(isolate), m_safeForTerminationScope(isolate) {}
397 
398 struct V8InspectorImpl::EvaluateScope::CancelToken {
399   v8::base::Mutex m_mutex;
400   bool m_canceled = false;
401 };
402 
~EvaluateScope()403 V8InspectorImpl::EvaluateScope::~EvaluateScope() {
404   if (m_cancelToken) {
405     v8::base::LockGuard<v8::base::Mutex> lock(&m_cancelToken->m_mutex);
406     m_cancelToken->m_canceled = true;
407     m_isolate->CancelTerminateExecution();
408   }
409 }
410 
411 class V8InspectorImpl::EvaluateScope::TerminateTask : public v8::Task {
412  public:
TerminateTask(v8::Isolate * isolate,std::shared_ptr<CancelToken> token)413   TerminateTask(v8::Isolate* isolate, std::shared_ptr<CancelToken> token)
414       : m_isolate(isolate), m_token(token) {}
415 
Run()416   void Run() {
417     // CancelToken contains m_canceled bool which may be changed from main
418     // thread, so lock mutex first.
419     v8::base::LockGuard<v8::base::Mutex> lock(&m_token->m_mutex);
420     if (m_token->m_canceled) return;
421     m_isolate->TerminateExecution();
422   }
423 
424  private:
425   v8::Isolate* m_isolate;
426   std::shared_ptr<CancelToken> m_token;
427 };
428 
setTimeout(double timeout)429 protocol::Response V8InspectorImpl::EvaluateScope::setTimeout(double timeout) {
430   if (m_isolate->IsExecutionTerminating()) {
431     return protocol::Response::Error("Execution was terminated");
432   }
433   m_cancelToken.reset(new CancelToken());
434   v8::debug::GetCurrentPlatform()->CallDelayedOnWorkerThread(
435       v8::base::make_unique<TerminateTask>(m_isolate, m_cancelToken), timeout);
436   return protocol::Response::OK();
437 }
438 
439 }  // namespace v8_inspector
440