// Copyright (c) 2013 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include #include #include "atom/common/api/atom_api_key_weak_map.h" #include "atom/common/api/remote_callback_freer.h" #include "atom/common/api/remote_object_freer.h" #include "atom/common/native_mate_converters/content_converter.h" #include "atom/common/node_includes.h" #include "base/hash.h" #include "native_mate/dictionary.h" #include "v8/include/v8-profiler.h" namespace base { // Following code should be removed after we upgraded to Chrome 50. template 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 struct hash> { std::size_t operator()(std::pair value) const { return base::HashInts(value.first, value.second); } }; } // namespace std namespace mate { template struct Converter> { static bool FromV8(v8::Isolate* isolate, v8::Local val, std::pair* out) { if (!val->IsArray()) return false; v8::Local array(v8::Local::Cast(val)); if (array->Length() != 2) return false; return Converter::FromV8(isolate, array->Get(0), &out->first) && Converter::FromV8(isolate, array->Get(1), &out->second); } }; } // namespace mate namespace { v8::Local GetHiddenValue(v8::Isolate* isolate, v8::Local object, v8::Local key) { v8::Local context = isolate->GetCurrentContext(); v8::Local privateKey = v8::Private::ForApi(isolate, key); v8::Local value; v8::Maybe result = object->HasPrivate(context, privateKey); if (!(result.IsJust() && result.FromJust())) return v8::Local(); if (object->GetPrivate(context, privateKey).ToLocal(&value)) return value; return v8::Local(); } void SetHiddenValue(v8::Isolate* isolate, v8::Local object, v8::Local key, v8::Local value) { if (value.IsEmpty()) return; v8::Local context = isolate->GetCurrentContext(); v8::Local privateKey = v8::Private::ForApi(isolate, key); object->SetPrivate(context, privateKey, value); } void DeleteHiddenValue(v8::Isolate* isolate, v8::Local object, v8::Local key) { v8::Local context = isolate->GetCurrentContext(); v8::Local privateKey = v8::Private::ForApi(isolate, key); // Actually deleting the value would make force the object into // dictionary mode which is unnecessarily slow. Instead, we replace // the hidden value with "undefined". object->SetPrivate(context, privateKey, v8::Undefined(isolate)); } int32_t GetObjectHash(v8::Local object) { return object->GetIdentityHash(); } void TakeHeapSnapshot(v8::Isolate* isolate) { isolate->GetHeapProfiler()->TakeHeapSnapshot(); } void Initialize(v8::Local exports, v8::Local unused, v8::Local context, void* priv) { mate::Dictionary dict(context->GetIsolate(), exports); dict.SetMethod("getHiddenValue", &GetHiddenValue); dict.SetMethod("setHiddenValue", &SetHiddenValue); dict.SetMethod("deleteHiddenValue", &DeleteHiddenValue); dict.SetMethod("getObjectHash", &GetObjectHash); dict.SetMethod("takeHeapSnapshot", &TakeHeapSnapshot); dict.SetMethod("setRemoteCallbackFreer", &atom::RemoteCallbackFreer::BindTo); dict.SetMethod("setRemoteObjectFreer", &atom::RemoteObjectFreer::BindTo); dict.SetMethod("createIDWeakMap", &atom::api::KeyWeakMap::Create); dict.SetMethod("createDoubleIDWeakMap", &atom::api::KeyWeakMap>::Create); } } // namespace NODE_MODULE_CONTEXT_AWARE_BUILTIN(atom_common_v8_util, Initialize)