2019-09-03 20:14:16 -06:00
|
|
|
// Copyright (c) 2019 GitHub, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "shell/common/gin_helper/destroyable.h"
|
|
|
|
|
2020-03-20 15:15:55 -06:00
|
|
|
#include "base/no_destructor.h"
|
2019-09-03 20:14:16 -06:00
|
|
|
#include "gin/converter.h"
|
2019-12-05 02:46:34 -07:00
|
|
|
#include "shell/common/gin_helper/wrappable_base.h"
|
2019-09-03 20:14:16 -06:00
|
|
|
|
|
|
|
namespace gin_helper {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2020-03-20 15:15:55 -06:00
|
|
|
v8::Global<v8::FunctionTemplate>* GetDestroyFunc() {
|
|
|
|
static base::NoDestructor<v8::Global<v8::FunctionTemplate>> destroy_func;
|
|
|
|
return destroy_func.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
v8::Global<v8::FunctionTemplate>* GetIsDestroyedFunc() {
|
|
|
|
static base::NoDestructor<v8::Global<v8::FunctionTemplate>> is_destroyed_func;
|
|
|
|
return is_destroyed_func.get();
|
|
|
|
}
|
2019-09-03 20:14:16 -06:00
|
|
|
|
|
|
|
void DestroyFunc(const v8::FunctionCallbackInfo<v8::Value>& info) {
|
|
|
|
v8::Local<v8::Object> holder = info.Holder();
|
|
|
|
if (Destroyable::IsDestroyed(holder))
|
|
|
|
return;
|
|
|
|
|
2019-12-05 02:46:34 -07:00
|
|
|
// TODO(zcbenz): gin_helper::Wrappable will be removed.
|
|
|
|
delete static_cast<gin_helper::WrappableBase*>(
|
2019-09-03 20:14:16 -06:00
|
|
|
holder->GetAlignedPointerFromInternalField(0));
|
|
|
|
holder->SetAlignedPointerInInternalField(0, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IsDestroyedFunc(const v8::FunctionCallbackInfo<v8::Value>& info) {
|
|
|
|
info.GetReturnValue().Set(gin::ConvertToV8(
|
|
|
|
info.GetIsolate(), Destroyable::IsDestroyed(info.Holder())));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
// static
|
|
|
|
bool Destroyable::IsDestroyed(v8::Local<v8::Object> object) {
|
|
|
|
// An object is considered destroyed if it has no internal pointer or its
|
|
|
|
// internal has been destroyed.
|
|
|
|
return object->InternalFieldCount() == 0 ||
|
|
|
|
object->GetAlignedPointerFromInternalField(0) == nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
void Destroyable::MakeDestroyable(v8::Isolate* isolate,
|
|
|
|
v8::Local<v8::FunctionTemplate> prototype) {
|
|
|
|
// Cache the FunctionTemplate of "destroy" and "isDestroyed".
|
2020-03-20 15:15:55 -06:00
|
|
|
if (GetDestroyFunc()->IsEmpty()) {
|
2019-09-03 20:14:16 -06:00
|
|
|
auto templ = v8::FunctionTemplate::New(isolate, DestroyFunc);
|
|
|
|
templ->RemovePrototype();
|
2020-03-20 15:15:55 -06:00
|
|
|
GetDestroyFunc()->Reset(isolate, templ);
|
2019-09-03 20:14:16 -06:00
|
|
|
templ = v8::FunctionTemplate::New(isolate, IsDestroyedFunc);
|
|
|
|
templ->RemovePrototype();
|
2020-03-20 15:15:55 -06:00
|
|
|
GetIsDestroyedFunc()->Reset(isolate, templ);
|
2019-09-03 20:14:16 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
auto proto_templ = prototype->PrototypeTemplate();
|
|
|
|
proto_templ->Set(
|
|
|
|
gin::StringToSymbol(isolate, "destroy"),
|
2020-03-20 15:15:55 -06:00
|
|
|
v8::Local<v8::FunctionTemplate>::New(isolate, *GetDestroyFunc()));
|
2019-09-03 20:14:16 -06:00
|
|
|
proto_templ->Set(
|
|
|
|
gin::StringToSymbol(isolate, "isDestroyed"),
|
2020-03-20 15:15:55 -06:00
|
|
|
v8::Local<v8::FunctionTemplate>::New(isolate, *GetIsDestroyedFunc()));
|
2019-09-03 20:14:16 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace gin_helper
|