Use DoubleIDWeakMap for |rendererFunctions|

This commit is contained in:
Cheng Zhao 2016-05-11 20:40:48 +09:00
parent 49ac160ff7
commit 759a46f3d6
3 changed files with 54 additions and 16 deletions

View file

@ -3,15 +3,64 @@
// found in the LICENSE file. // found in the LICENSE file.
#include <string> #include <string>
#include <utility>
#include "atom/common/api/atom_api_key_weak_map.h" #include "atom/common/api/atom_api_key_weak_map.h"
#include "atom/common/api/remote_callback_freer.h" #include "atom/common/api/remote_callback_freer.h"
#include "atom/common/api/remote_object_freer.h" #include "atom/common/api/remote_object_freer.h"
#include "atom/common/native_mate_converters/content_converter.h" #include "atom/common/native_mate_converters/content_converter.h"
#include "atom/common/node_includes.h" #include "atom/common/node_includes.h"
#include "base/hash.h"
#include "native_mate/dictionary.h" #include "native_mate/dictionary.h"
#include "v8/include/v8-profiler.h" #include "v8/include/v8-profiler.h"
namespace base {
// Following code should be removed after we upgraded to Chrome 50.
template <typename T1, typename T2>
inline size_t HashInts(T1 value1, T2 value2) {
// This condition is expected to be compile-time evaluated and optimised away
// in release builds.
if (sizeof(T1) > sizeof(uint32_t) || (sizeof(T2) > sizeof(uint32_t)))
return HashInts64(value1, value2);
return HashInts32(value1, value2);
}
} // namespace base
namespace std {
// The hash function used by DoubleIDWeakMap.
template <typename Type1, typename Type2>
struct hash<std::pair<Type1, Type2>> {
std::size_t operator()(std::pair<Type1, Type2> value) const {
return base::HashInts<Type1, Type2>(value.first, value.second);
}
};
} // namespace std
namespace mate {
template<typename Type1, typename Type2>
struct Converter<std::pair<Type1, Type2>> {
static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val,
std::pair<Type1, Type2>* out) {
if (!val->IsArray())
return false;
v8::Local<v8::Array> array(v8::Local<v8::Array>::Cast(val));
if (array->Length() != 2)
return false;
return Converter<Type1>::FromV8(isolate, array->Get(0), &out->first) &&
Converter<Type2>::FromV8(isolate, array->Get(1), &out->second);
}
};
} // namespace mate
namespace { namespace {
v8::Local<v8::Value> GetHiddenValue(v8::Isolate* isolate, v8::Local<v8::Value> GetHiddenValue(v8::Isolate* isolate,
@ -69,6 +118,8 @@ void Initialize(v8::Local<v8::Object> exports, v8::Local<v8::Value> unused,
dict.SetMethod("setRemoteCallbackFreer", &atom::RemoteCallbackFreer::BindTo); dict.SetMethod("setRemoteCallbackFreer", &atom::RemoteCallbackFreer::BindTo);
dict.SetMethod("setRemoteObjectFreer", &atom::RemoteObjectFreer::BindTo); dict.SetMethod("setRemoteObjectFreer", &atom::RemoteObjectFreer::BindTo);
dict.SetMethod("createIDWeakMap", &atom::api::KeyWeakMap<int32_t>::Create); dict.SetMethod("createIDWeakMap", &atom::api::KeyWeakMap<int32_t>::Create);
dict.SetMethod("createDoubleIDWeakMap",
&atom::api::KeyWeakMap<std::pair<int32_t, int32_t>>::Create);
} }
} // namespace } // namespace

View file

@ -70,10 +70,8 @@ class KeyWeakMap {
// Remove object with |key| in the WeakMap. // Remove object with |key| in the WeakMap.
void Remove(const K& key) { void Remove(const K& key) {
auto iter = map_.find(key); auto iter = map_.find(key);
if (iter == map_.end()) { if (iter == map_.end())
LOG(WARNING) << "Removing unexist object with ID " << key;
return; return;
}
iter->second.second->ClearWeak(); iter->second.second->ClearWeak();
map_.erase(iter); map_.erase(iter);

View file

@ -13,18 +13,7 @@ const FUNCTION_PROPERTIES = [
// The remote functions in renderer processes. // The remote functions in renderer processes.
// id => Function // id => Function
let rendererFunctions = v8Util.createIDWeakMap() let rendererFunctions = v8Util.createDoubleIDWeakMap()
// Merge two IDs together.
let mergeIds = function (webContentsId, metaId) {
const PADDING_BITS = 20
if ((webContentsId << PADDING_BITS) < 0) {
throw new Error(`webContents ID is too large: ${webContentsId}`)
} else if (metaId > (1 << PADDING_BITS)) {
throw new Error(`Object ID is too large: ${metaId}`)
}
return (webContentsId << PADDING_BITS) + metaId
}
// Return the description of object's members: // Return the description of object's members:
let getObjectMembers = function (object) { let getObjectMembers = function (object) {
@ -179,7 +168,7 @@ var unwrapArgs = function (sender, args) {
// Merge webContentsId and meta.id, since meta.id can be the same in // Merge webContentsId and meta.id, since meta.id can be the same in
// different webContents. // different webContents.
const webContentsId = sender.getId() const webContentsId = sender.getId()
const objectId = mergeIds(webContentsId, meta.id) const objectId = [webContentsId, meta.id]
// Cache the callbacks in renderer. // Cache the callbacks in renderer.
if (rendererFunctions.has(objectId)) { if (rendererFunctions.has(objectId)) {