v8 :: Locker在隔离中退出当前上下文

时间:2019-12-18 16:16:55

标签: v8

我正在尝试将V8嵌入多线程应用程序中。我的计划是在当前需要执行JS代码的任何线程中以内联方式执行V8代码。 因此,在执行上下文中,我要获取一个v8::Locker并在 random 线程中输入隔离。我注意到,获取v8::Locker时,隔离丢失是CurrentContext。 在下面的代码片段中,我将其归结为问题的实质。如果我在v8 :: Locker下Enter使用上下文,退出该储物柜并输入一个新的储物柜,则隔离将丢失其Context。请注意,我从不明确Exit上下文。 这是正确用法吗?我想念什么吗?

在下面的代码片段中,您可以看到问题的单线程虚拟示例。 断言 1 2 通过并断言 3 失败。

v8::V8::InitializeICU(V8_ICU_FILE);
v8::V8::InitializeExternalStartupData(V8_SNAPSHOT_BIN_FILE);

std::unique_ptr<v8::Platform> platform = v8::platform::NewDefaultPlatform();
v8::V8::InitializePlatform(platform.get());
v8::V8::Initialize();
// Create a new Isolate and make it the current one.
v8::Isolate::CreateParams create_params;
auto allocator =  std::unique_ptr<v8::ArrayBuffer::Allocator>(v8::ArrayBuffer::Allocator::NewDefaultAllocator());
create_params.array_buffer_allocator = allocator.get();
v8::Isolate* isolate = v8::Isolate::New(create_params);


v8::HandleScope handle_scope(isolate);
// Create a new context.
v8::Local<v8::ObjectTemplate> global = v8::ObjectTemplate::New(isolate);
global->Set(isolate, "log", v8::FunctionTemplate::New(isolate, logCallback));

v8::Local<v8::Context> context = v8::Context::New(isolate, nullptr, global);
// Enter the context for compiling and running the hello world script.
{
    v8::Locker locker(isolate);
    v8::Isolate::Scope scope(isolate);
    context->Enter(); // Explicitly Enter the context (Which makes it the CurrentContext)
    assert(!isolate->GetCurrentContext().IsEmpty()); // 1 - Succeeds

    isolate->Exit();
    isolate->Enter();
    assert(!isolate->GetCurrentContext().IsEmpty()); // 2 - Succeeds
}
{
    v8::Locker locker(isolate);
    v8::Isolate::Scope scope(isolate);
    assert(!isolate->GetCurrentContext().IsEmpty()); // 3 - Assertion Fails
}
// Create a string containing the JavaScript source code.
isolate->Dispose();
v8::V8::Dispose();
v8::V8::ShutdownPlatform();

1 个答案:

答案 0 :(得分:0)

v8::Locker析构函数假定当前线程已完成所有工作,然后进行清理;特别是这意味着它将取消当前上下文。在下一个线程上创建下一个Locker之后,您必须(重新)输入正确的上下文。 v8::Context::Scope方便了。