2015-06-24 00:36:05 -06:00
|
|
|
// Copyright (c) 2015 GitHub, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2021-11-22 00:34:31 -07:00
|
|
|
#ifndef ELECTRON_SHELL_COMMON_GIN_HELPER_TRACKABLE_OBJECT_H_
|
|
|
|
#define ELECTRON_SHELL_COMMON_GIN_HELPER_TRACKABLE_OBJECT_H_
|
2015-06-24 00:36:05 -06:00
|
|
|
|
2015-06-24 03:58:12 -06:00
|
|
|
#include <vector>
|
|
|
|
|
2023-02-03 04:43:42 -07:00
|
|
|
#include "base/functional/bind.h"
|
2015-09-05 20:30:59 -06:00
|
|
|
#include "base/memory/weak_ptr.h"
|
2021-02-05 14:20:20 -07:00
|
|
|
#include "shell/common/gin_helper/cleaned_up_at_exit.h"
|
2019-10-25 07:03:28 -06:00
|
|
|
#include "shell/common/gin_helper/event_emitter.h"
|
2019-06-19 14:46:59 -06:00
|
|
|
#include "shell/common/key_weak_map.h"
|
2015-06-24 00:36:05 -06:00
|
|
|
|
2015-06-24 00:49:08 -06:00
|
|
|
namespace base {
|
|
|
|
class SupportsUserData;
|
|
|
|
}
|
|
|
|
|
2019-10-25 07:03:28 -06:00
|
|
|
namespace gin_helper {
|
2015-06-24 00:36:05 -06:00
|
|
|
|
2015-06-24 03:58:12 -06:00
|
|
|
// Users should use TrackableObject instead.
|
2021-02-05 14:20:20 -07:00
|
|
|
class TrackableObjectBase : public CleanedUpAtExit {
|
2015-06-24 00:36:05 -06:00
|
|
|
public:
|
2015-06-24 03:58:12 -06:00
|
|
|
TrackableObjectBase();
|
2015-06-24 00:36:05 -06:00
|
|
|
|
2021-11-03 05:41:45 -06:00
|
|
|
// disable copy
|
|
|
|
TrackableObjectBase(const TrackableObjectBase&) = delete;
|
|
|
|
TrackableObjectBase& operator=(const TrackableObjectBase&) = delete;
|
|
|
|
|
2015-06-24 03:58:12 -06:00
|
|
|
// The ID in weak map.
|
|
|
|
int32_t weak_map_id() const { return weak_map_id_; }
|
2015-06-24 00:49:08 -06:00
|
|
|
|
2015-06-24 03:58:12 -06:00
|
|
|
// Wrap TrackableObject into a class that SupportsUserData.
|
|
|
|
void AttachAsUserData(base::SupportsUserData* wrapped);
|
|
|
|
|
2017-09-12 06:17:11 -06:00
|
|
|
// Get the weak_map_id from SupportsUserData.
|
|
|
|
static int32_t GetIDFromWrappedClass(base::SupportsUserData* wrapped);
|
|
|
|
|
2015-06-24 03:58:12 -06:00
|
|
|
protected:
|
2021-02-05 14:20:20 -07:00
|
|
|
~TrackableObjectBase() override;
|
2015-12-03 00:38:43 -07:00
|
|
|
|
|
|
|
// Returns a closure that can destroy the native class.
|
2018-03-30 07:24:55 -06:00
|
|
|
base::OnceClosure GetDestroyClosure();
|
2015-06-24 03:58:12 -06:00
|
|
|
|
2018-05-21 16:18:38 -06:00
|
|
|
int32_t weak_map_id_ = 0;
|
2015-06-24 03:58:12 -06:00
|
|
|
|
|
|
|
private:
|
2015-12-03 00:38:43 -07:00
|
|
|
void Destroy();
|
|
|
|
|
2021-01-26 11:16:21 -07:00
|
|
|
base::WeakPtrFactory<TrackableObjectBase> weak_factory_{this};
|
2015-06-24 03:58:12 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
// All instances of TrackableObject will be kept in a weak map and can be got
|
|
|
|
// from its ID.
|
2019-10-25 07:03:28 -06:00
|
|
|
template <typename T>
|
|
|
|
class TrackableObject : public TrackableObjectBase, public EventEmitter<T> {
|
2015-06-24 03:58:12 -06:00
|
|
|
public:
|
2016-04-24 19:17:54 -06:00
|
|
|
// Mark the JS object as destroyed.
|
|
|
|
void MarkDestroyed() {
|
2020-03-10 19:16:58 -06:00
|
|
|
v8::HandleScope scope(gin_helper::Wrappable<T>::isolate());
|
2019-12-05 02:46:34 -07:00
|
|
|
v8::Local<v8::Object> wrapper = gin_helper::Wrappable<T>::GetWrapper();
|
2017-10-23 23:38:55 -06:00
|
|
|
if (!wrapper.IsEmpty()) {
|
2017-10-27 01:07:54 -06:00
|
|
|
wrapper->SetAlignedPointerInInternalField(0, nullptr);
|
2020-03-10 19:16:58 -06:00
|
|
|
gin_helper::WrappableBase::wrapper_.ClearWeak();
|
2017-10-23 23:38:55 -06:00
|
|
|
}
|
2016-04-24 19:17:54 -06:00
|
|
|
}
|
|
|
|
|
2016-12-20 17:51:03 -07:00
|
|
|
bool IsDestroyed() {
|
2020-03-10 19:16:58 -06:00
|
|
|
v8::HandleScope scope(gin_helper::Wrappable<T>::isolate());
|
2019-12-05 02:46:34 -07:00
|
|
|
v8::Local<v8::Object> wrapper = gin_helper::Wrappable<T>::GetWrapper();
|
2016-12-20 17:51:03 -07:00
|
|
|
return wrapper->InternalFieldCount() == 0 ||
|
|
|
|
wrapper->GetAlignedPointerFromInternalField(0) == nullptr;
|
|
|
|
}
|
|
|
|
|
2015-06-24 03:58:12 -06:00
|
|
|
// Finds out the TrackableObject from its ID in weak map.
|
|
|
|
static T* FromWeakMapID(v8::Isolate* isolate, int32_t id) {
|
2015-07-14 03:19:46 -06:00
|
|
|
if (!weak_map_)
|
|
|
|
return nullptr;
|
|
|
|
|
2020-03-10 19:16:58 -06:00
|
|
|
v8::HandleScope scope(isolate);
|
2015-07-14 03:19:46 -06:00
|
|
|
v8::MaybeLocal<v8::Object> object = weak_map_->Get(isolate, id);
|
2015-06-24 03:58:12 -06:00
|
|
|
if (object.IsEmpty())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
T* self = nullptr;
|
2019-10-25 07:03:28 -06:00
|
|
|
gin::ConvertFromV8(isolate, object.ToLocalChecked(), &self);
|
2015-06-24 03:58:12 -06:00
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finds out the TrackableObject from the class it wraps.
|
|
|
|
static T* FromWrappedClass(v8::Isolate* isolate,
|
|
|
|
base::SupportsUserData* wrapped) {
|
|
|
|
int32_t id = GetIDFromWrappedClass(wrapped);
|
|
|
|
if (!id)
|
|
|
|
return nullptr;
|
|
|
|
return FromWeakMapID(isolate, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns all objects in this class's weak map.
|
|
|
|
static std::vector<v8::Local<v8::Object>> GetAll(v8::Isolate* isolate) {
|
2015-07-14 03:19:46 -06:00
|
|
|
if (weak_map_)
|
|
|
|
return weak_map_->Values(isolate);
|
|
|
|
else
|
|
|
|
return std::vector<v8::Local<v8::Object>>();
|
2015-06-24 03:58:12 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Removes this instance from the weak map.
|
|
|
|
void RemoveFromWeakMap() {
|
2024-02-06 08:18:05 -07:00
|
|
|
if (weak_map_)
|
2015-07-14 03:19:46 -06:00
|
|
|
weak_map_->Remove(weak_map_id());
|
2015-06-24 03:58:12 -06:00
|
|
|
}
|
2015-06-24 00:36:05 -06:00
|
|
|
|
|
|
|
protected:
|
2018-04-17 19:44:10 -06:00
|
|
|
TrackableObject() { weak_map_id_ = ++next_id_; }
|
2016-04-24 19:17:54 -06:00
|
|
|
|
2018-04-17 19:44:10 -06:00
|
|
|
~TrackableObject() override { RemoveFromWeakMap(); }
|
2015-06-24 00:36:05 -06:00
|
|
|
|
2016-08-02 00:28:32 -06:00
|
|
|
void InitWith(v8::Isolate* isolate, v8::Local<v8::Object> wrapper) override {
|
2015-12-03 02:10:14 -07:00
|
|
|
if (!weak_map_) {
|
2019-06-19 15:23:04 -06:00
|
|
|
weak_map_ = new electron::KeyWeakMap<int32_t>;
|
2015-12-03 02:10:14 -07:00
|
|
|
}
|
2016-08-02 00:28:32 -06:00
|
|
|
weak_map_->Set(isolate, weak_map_id_, wrapper);
|
2022-03-16 17:23:14 -06:00
|
|
|
gin_helper::WrappableBase::InitWith(isolate, wrapper);
|
2015-06-24 03:58:12 -06:00
|
|
|
}
|
2015-06-24 00:49:08 -06:00
|
|
|
|
2015-06-24 00:36:05 -06:00
|
|
|
private:
|
2016-05-11 00:51:22 -06:00
|
|
|
static int32_t next_id_;
|
2019-06-19 15:23:04 -06:00
|
|
|
static electron::KeyWeakMap<int32_t>* weak_map_; // leaked on purpose
|
2015-06-24 00:36:05 -06:00
|
|
|
};
|
|
|
|
|
2019-10-25 07:03:28 -06:00
|
|
|
template <typename T>
|
|
|
|
int32_t TrackableObject<T>::next_id_ = 0;
|
2016-05-11 00:51:22 -06:00
|
|
|
|
2019-10-25 07:03:28 -06:00
|
|
|
template <typename T>
|
|
|
|
electron::KeyWeakMap<int32_t>* TrackableObject<T>::weak_map_ = nullptr;
|
2015-06-24 03:58:12 -06:00
|
|
|
|
2019-10-25 07:03:28 -06:00
|
|
|
} // namespace gin_helper
|
2015-06-24 00:36:05 -06:00
|
|
|
|
2021-11-22 00:34:31 -07:00
|
|
|
#endif // ELECTRON_SHELL_COMMON_GIN_HELPER_TRACKABLE_OBJECT_H_
|