// Copyright 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE.chromium file. #include "shell/common/gin_helper/wrappable.h" #include "base/logging.h" #include "base/threading/thread_task_runner_handle.h" #include "shell/common/gin_helper/dictionary.h" namespace gin_helper { WrappableBase::WrappableBase() = default; WrappableBase::~WrappableBase() { if (wrapper_.IsEmpty()) return; GetWrapper()->SetAlignedPointerInInternalField(0, nullptr); wrapper_.ClearWeak(); wrapper_.Reset(); } v8::Local WrappableBase::GetWrapper() const { if (!wrapper_.IsEmpty()) return v8::Local::New(isolate_, wrapper_); else return v8::Local(); } v8::MaybeLocal WrappableBase::GetWrapper( v8::Isolate* isolate) const { if (!wrapper_.IsEmpty()) return v8::MaybeLocal( v8::Local::New(isolate, wrapper_)); else return v8::MaybeLocal(); } void WrappableBase::InitWithArgs(gin::Arguments* args) { v8::Local holder; args->GetHolder(&holder); InitWith(args->isolate(), holder); } void WrappableBase::InitWith(v8::Isolate* isolate, v8::Local wrapper) { CHECK(wrapper_.IsEmpty()); isolate_ = isolate; wrapper->SetAlignedPointerInInternalField(0, this); wrapper_.Reset(isolate, wrapper); wrapper_.SetWeak(this, FirstWeakCallback, v8::WeakCallbackType::kParameter); // Call object._init if we have one. v8::Local init; if (Dictionary(isolate, wrapper).Get("_init", &init)) init->Call(isolate->GetCurrentContext(), wrapper, 0, nullptr).IsEmpty(); AfterInit(isolate); } // static void WrappableBase::FirstWeakCallback( const v8::WeakCallbackInfo& data) { WrappableBase* wrappable = data.GetParameter(); wrappable->wrapper_.Reset(); data.SetSecondPassCallback(SecondWeakCallback); } // static void WrappableBase::SecondWeakCallback( const v8::WeakCallbackInfo& data) { // Certain classes (for example api::WebContents and api::WebContentsView) // are running JS code in the destructor, while V8 may crash when JS code // runs inside weak callback. // // We work around this problem by delaying the deletion to next tick where // garbage collection is done. base::ThreadTaskRunnerHandle::Get()->PostNonNestableTask( FROM_HERE, base::BindOnce([](WrappableBase* wrappable) { delete wrappable; }, base::Unretained(data.GetParameter()))); } namespace internal { void* FromV8Impl(v8::Isolate* isolate, v8::Local val) { if (!val->IsObject()) return nullptr; v8::Local obj = v8::Local::Cast(val); if (obj->InternalFieldCount() != 1) return nullptr; return obj->GetAlignedPointerFromInternalField(0); } } // namespace internal } // namespace gin_helper