blob: ed1d55744c79e14fd7cce51b25b65b7209068582 [file] [log] [blame]
// Copyright 2020 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.
#include "test/inspector/tasks.h"
#include <vector>
#include "include/v8-isolate.h"
#include "include/v8-script.h"
#include "test/inspector/isolate-data.h"
#include "test/inspector/utils.h"
namespace v8 {
namespace internal {
void RunSyncTask(TaskRunner* task_runner,
std::function<void(InspectorIsolateData*)> callback) {
class SyncTask : public TaskRunner::Task {
public:
SyncTask(v8::base::Semaphore* ready_semaphore,
std::function<void(InspectorIsolateData*)> callback)
: ready_semaphore_(ready_semaphore), callback_(callback) {}
~SyncTask() override = default;
bool is_priority_task() final { return true; }
private:
void Run(InspectorIsolateData* data) override {
callback_(data);
if (ready_semaphore_) ready_semaphore_->Signal();
}
v8::base::Semaphore* ready_semaphore_;
std::function<void(InspectorIsolateData*)> callback_;
};
v8::base::Semaphore ready_semaphore(0);
task_runner->Append(std::make_unique<SyncTask>(&ready_semaphore, callback));
ready_semaphore.Wait();
}
void RunSimpleAsyncTask(TaskRunner* task_runner,
std::function<void(InspectorIsolateData* data)> task,
v8::Local<v8::Function> callback) {
class DispatchResponseTask : public TaskRunner::Task {
public:
explicit DispatchResponseTask(v8::Local<v8::Function> callback)
: context_(callback->GetIsolate(),
callback->GetIsolate()->GetCurrentContext()),
client_callback_(callback->GetIsolate(), callback) {}
~DispatchResponseTask() override = default;
private:
bool is_priority_task() final { return true; }
void Run(InspectorIsolateData* data) override {
v8::HandleScope handle_scope(data->isolate());
v8::Local<v8::Context> context = context_.Get(data->isolate());
v8::MicrotasksScope microtasks_scope(context,
v8::MicrotasksScope::kRunMicrotasks);
v8::Context::Scope context_scope(context);
USE(client_callback_.Get(data->isolate())
->Call(context, context->Global(), 0, nullptr));
}
v8::Global<v8::Context> context_;
v8::Global<v8::Function> client_callback_;
};
using TaskCallback = std::function<void(InspectorIsolateData * data)>;
class TaskWrapper : public TaskRunner::Task {
public:
TaskWrapper(TaskCallback task, TaskRunner* client_task_runner,
std::unique_ptr<TaskRunner::Task> response_task)
: task_(std::move(task)),
client_task_runner_(client_task_runner),
response_task_(std::move(response_task)) {}
~TaskWrapper() override = default;
private:
bool is_priority_task() final { return true; }
void Run(InspectorIsolateData* data) override {
task_(data);
client_task_runner_->Append(std::move(response_task_));
}
TaskCallback task_;
TaskRunner* client_task_runner_;
std::unique_ptr<TaskRunner::Task> response_task_;
};
v8::Local<v8::Context> context = callback->GetIsolate()->GetCurrentContext();
TaskRunner* response_task_runner =
InspectorIsolateData::FromContext(context)->task_runner();
auto response_task = std::make_unique<DispatchResponseTask>(callback);
task_runner->Append(std::make_unique<TaskWrapper>(
std::move(task), response_task_runner, std::move(response_task)));
}
void ExecuteStringTask::Run(InspectorIsolateData* data) {
v8::HandleScope handle_scope(data->isolate());
v8::Local<v8::Context> context = data->GetDefaultContext(context_group_id_);
v8::MicrotasksScope microtasks_scope(context,
v8::MicrotasksScope::kRunMicrotasks);
v8::Context::Scope context_scope(context);
v8::ScriptOrigin origin(ToV8String(data->isolate(), name_), line_offset_,
column_offset_,
/* resource_is_shared_cross_origin */ false,
/* script_id */ -1,
/* source_map_url */ v8::Local<v8::Value>(),
/* resource_is_opaque */ false,
/* is_wasm */ false, is_module_);
v8::Local<v8::String> source;
if (expression_.size() != 0)
source = ToV8String(data->isolate(), expression_);
else
source = ToV8String(data->isolate(), expression_utf8_);
v8::ScriptCompiler::Source scriptSource(source, origin);
if (!is_module_) {
v8::Local<v8::Script> script;
if (!v8::ScriptCompiler::Compile(context, &scriptSource).ToLocal(&script))
return;
v8::MaybeLocal<v8::Value> result;
result = script->Run(context);
} else {
data->RegisterModule(context, name_, &scriptSource);
}
}
} // namespace internal
} // namespace v8