John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 1 | // Copyright 2014 PDFium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
Lei Zhang | a6d9f0e | 2015-06-13 00:48:38 -0700 | [diff] [blame] | 4 | |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 5 | // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| 6 | |
dsinclair | 4355468 | 2016-09-29 17:29:48 -0700 | [diff] [blame] | 7 | #include "fxjs/fxjs_v8.h" |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 8 | |
Dan Sinclair | 3ebd121 | 2016-03-09 09:59:23 -0500 | [diff] [blame] | 9 | #include <vector> |
| 10 | |
Chris Palmer | e4b035b | 2017-03-26 15:48:34 -0700 | [diff] [blame] | 11 | #include "third_party/base/allocator/partition_allocator/partition_alloc.h" |
Lei Zhang | 4d4a442 | 2015-10-08 12:00:14 -0700 | [diff] [blame] | 12 | |
Tom Sepez | bd7fabf | 2015-09-28 10:31:27 -0700 | [diff] [blame] | 13 | // Keep this consistent with the values defined in gin/public/context_holder.h |
| 14 | // (without actually requiring a dependency on gin itself for the standalone |
| 15 | // embedders of PDFIum). The value we want to use is: |
| 16 | // kPerContextDataStartIndex + kEmbedderPDFium, which is 3. |
| 17 | static const unsigned int kPerContextDataIndex = 3u; |
Tom Sepez | a72e8e2 | 2015-10-07 10:17:53 -0700 | [diff] [blame] | 18 | static unsigned int g_embedderDataSlot = 1u; |
| 19 | static v8::Isolate* g_isolate = nullptr; |
Lei Zhang | 4d4a442 | 2015-10-08 12:00:14 -0700 | [diff] [blame] | 20 | static size_t g_isolate_ref_count = 0; |
Tom Sepez | a72e8e2 | 2015-10-07 10:17:53 -0700 | [diff] [blame] | 21 | static FXJS_ArrayBufferAllocator* g_arrayBufferAllocator = nullptr; |
Tom Sepez | 8b315b6 | 2015-10-02 08:59:57 -0700 | [diff] [blame] | 22 | static v8::Global<v8::ObjectTemplate>* g_DefaultGlobalObjectTemplate = nullptr; |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 23 | static wchar_t kPerObjectDataTag[] = L"CFXJS_PerObjectData"; |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 24 | |
Tom Sepez | 6cf117c | 2015-11-06 14:52:10 -0800 | [diff] [blame] | 25 | class CFXJS_PerObjectData { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 26 | public: |
Lei Zhang | d88a364 | 2015-11-10 09:38:57 -0800 | [diff] [blame] | 27 | explicit CFXJS_PerObjectData(int nObjDefID) |
Tom Sepez | 6cf117c | 2015-11-06 14:52:10 -0800 | [diff] [blame] | 28 | : m_ObjDefID(nObjDefID), m_pPrivate(nullptr) {} |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 29 | |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 30 | static void SetInObject(CFXJS_PerObjectData* pData, |
| 31 | v8::Local<v8::Object> pObj) { |
| 32 | if (pObj->InternalFieldCount() == 2) { |
| 33 | pObj->SetAlignedPointerInInternalField(0, pData); |
| 34 | pObj->SetAlignedPointerInInternalField( |
| 35 | 1, static_cast<void*>(kPerObjectDataTag)); |
| 36 | } |
| 37 | } |
| 38 | |
| 39 | static CFXJS_PerObjectData* GetFromObject(v8::Local<v8::Object> pObj) { |
| 40 | if (pObj.IsEmpty() || pObj->InternalFieldCount() != 2 || |
| 41 | pObj->GetAlignedPointerFromInternalField(1) != |
| 42 | static_cast<void*>(kPerObjectDataTag)) { |
| 43 | return nullptr; |
| 44 | } |
| 45 | return static_cast<CFXJS_PerObjectData*>( |
| 46 | pObj->GetAlignedPointerFromInternalField(0)); |
| 47 | } |
| 48 | |
Lei Zhang | d88a364 | 2015-11-10 09:38:57 -0800 | [diff] [blame] | 49 | const int m_ObjDefID; |
Tom Sepez | 6cf117c | 2015-11-06 14:52:10 -0800 | [diff] [blame] | 50 | void* m_pPrivate; |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 51 | }; |
| 52 | |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 53 | class CFXJS_ObjDefinition { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 54 | public: |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 55 | static int MaxID(v8::Isolate* pIsolate) { |
Lei Zhang | d88a364 | 2015-11-10 09:38:57 -0800 | [diff] [blame] | 56 | return FXJS_PerIsolateData::Get(pIsolate)->m_ObjectDefnArray.size(); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 57 | } |
Lei Zhang | d88a364 | 2015-11-10 09:38:57 -0800 | [diff] [blame] | 58 | |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 59 | static CFXJS_ObjDefinition* ForID(v8::Isolate* pIsolate, int id) { |
| 60 | // Note: GetAt() halts if out-of-range even in release builds. |
weili | a4ad595 | 2016-09-22 10:38:53 -0700 | [diff] [blame] | 61 | return FXJS_PerIsolateData::Get(pIsolate)->m_ObjectDefnArray[id].get(); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 62 | } |
Lei Zhang | d88a364 | 2015-11-10 09:38:57 -0800 | [diff] [blame] | 63 | |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 64 | CFXJS_ObjDefinition(v8::Isolate* isolate, |
Tom Sepez | 892d751 | 2017-02-21 13:57:13 -0800 | [diff] [blame] | 65 | const char* sObjName, |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 66 | FXJSOBJTYPE eObjType, |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 67 | CFXJS_Engine::Constructor pConstructor, |
| 68 | CFXJS_Engine::Destructor pDestructor) |
Tom Sepez | cd56a7d | 2015-10-06 11:45:28 -0700 | [diff] [blame] | 69 | : m_ObjName(sObjName), |
| 70 | m_ObjType(eObjType), |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 71 | m_pConstructor(pConstructor), |
| 72 | m_pDestructor(pDestructor), |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 73 | m_pIsolate(isolate) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 74 | v8::Isolate::Scope isolate_scope(isolate); |
| 75 | v8::HandleScope handle_scope(isolate); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 76 | |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 77 | v8::Local<v8::FunctionTemplate> fun = v8::FunctionTemplate::New(isolate); |
| 78 | fun->InstanceTemplate()->SetInternalFieldCount(2); |
Tom Sepez | c5a1472 | 2017-02-24 15:31:12 -0800 | [diff] [blame] | 79 | fun->SetCallHandler([](const v8::FunctionCallbackInfo<v8::Value>& info) { |
| 80 | v8::Local<v8::Object> holder = info.Holder(); |
| 81 | ASSERT(holder->InternalFieldCount() == 2); |
| 82 | holder->SetAlignedPointerInInternalField(0, nullptr); |
| 83 | holder->SetAlignedPointerInInternalField(1, nullptr); |
| 84 | }); |
jochen | 7e6a848 | 2016-07-06 11:02:27 -0700 | [diff] [blame] | 85 | if (eObjType == FXJSOBJTYPE_GLOBAL) { |
| 86 | fun->InstanceTemplate()->Set( |
| 87 | v8::Symbol::GetToStringTag(isolate), |
| 88 | v8::String::NewFromUtf8(isolate, "global", v8::NewStringType::kNormal) |
| 89 | .ToLocalChecked()); |
| 90 | } |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 91 | m_FunctionTemplate.Reset(isolate, fun); |
| 92 | |
| 93 | v8::Local<v8::Signature> sig = v8::Signature::New(isolate, fun); |
| 94 | m_Signature.Reset(isolate, sig); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 95 | } |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 96 | |
| 97 | int AssignID() { |
| 98 | FXJS_PerIsolateData* pData = FXJS_PerIsolateData::Get(m_pIsolate); |
weili | a4ad595 | 2016-09-22 10:38:53 -0700 | [diff] [blame] | 99 | pData->m_ObjectDefnArray.emplace_back(this); |
Lei Zhang | d88a364 | 2015-11-10 09:38:57 -0800 | [diff] [blame] | 100 | return pData->m_ObjectDefnArray.size() - 1; |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 101 | } |
| 102 | |
| 103 | v8::Local<v8::ObjectTemplate> GetInstanceTemplate() { |
| 104 | v8::EscapableHandleScope scope(m_pIsolate); |
| 105 | v8::Local<v8::FunctionTemplate> function = |
| 106 | m_FunctionTemplate.Get(m_pIsolate); |
| 107 | return scope.Escape(function->InstanceTemplate()); |
| 108 | } |
| 109 | |
| 110 | v8::Local<v8::Signature> GetSignature() { |
| 111 | v8::EscapableHandleScope scope(m_pIsolate); |
| 112 | return scope.Escape(m_Signature.Get(m_pIsolate)); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 113 | } |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 114 | |
Tom Sepez | 892d751 | 2017-02-21 13:57:13 -0800 | [diff] [blame] | 115 | const char* const m_ObjName; |
Tom Sepez | cd56a7d | 2015-10-06 11:45:28 -0700 | [diff] [blame] | 116 | const FXJSOBJTYPE m_ObjType; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 117 | const CFXJS_Engine::Constructor m_pConstructor; |
| 118 | const CFXJS_Engine::Destructor m_pDestructor; |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 119 | |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 120 | v8::Isolate* m_pIsolate; |
| 121 | v8::Global<v8::FunctionTemplate> m_FunctionTemplate; |
| 122 | v8::Global<v8::Signature> m_Signature; |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 123 | }; |
| 124 | |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 125 | static v8::Local<v8::ObjectTemplate> GetGlobalObjectTemplate( |
| 126 | v8::Isolate* pIsolate) { |
| 127 | int maxID = CFXJS_ObjDefinition::MaxID(pIsolate); |
| 128 | for (int i = 0; i < maxID; ++i) { |
| 129 | CFXJS_ObjDefinition* pObjDef = CFXJS_ObjDefinition::ForID(pIsolate, i); |
Tom Sepez | cd56a7d | 2015-10-06 11:45:28 -0700 | [diff] [blame] | 130 | if (pObjDef->m_ObjType == FXJSOBJTYPE_GLOBAL) |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 131 | return pObjDef->GetInstanceTemplate(); |
| 132 | } |
Tom Sepez | 8b315b6 | 2015-10-02 08:59:57 -0700 | [diff] [blame] | 133 | if (!g_DefaultGlobalObjectTemplate) { |
jochen | 7e6a848 | 2016-07-06 11:02:27 -0700 | [diff] [blame] | 134 | v8::Local<v8::ObjectTemplate> hGlobalTemplate = |
| 135 | v8::ObjectTemplate::New(pIsolate); |
| 136 | hGlobalTemplate->Set( |
| 137 | v8::Symbol::GetToStringTag(pIsolate), |
| 138 | v8::String::NewFromUtf8(pIsolate, "global", v8::NewStringType::kNormal) |
| 139 | .ToLocalChecked()); |
| 140 | g_DefaultGlobalObjectTemplate = |
| 141 | new v8::Global<v8::ObjectTemplate>(pIsolate, hGlobalTemplate); |
Tom Sepez | 8b315b6 | 2015-10-02 08:59:57 -0700 | [diff] [blame] | 142 | } |
| 143 | return g_DefaultGlobalObjectTemplate->Get(pIsolate); |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 144 | } |
| 145 | |
Tom Sepez | 39bfe12 | 2015-09-17 15:25:23 -0700 | [diff] [blame] | 146 | void* FXJS_ArrayBufferAllocator::Allocate(size_t length) { |
Chris Palmer | e4b035b | 2017-03-26 15:48:34 -0700 | [diff] [blame] | 147 | if (length > kMaxAllowedBytes) |
| 148 | return nullptr; |
| 149 | void* p = AllocateUninitialized(length); |
| 150 | if (p) |
| 151 | memset(p, 0, length); |
| 152 | return p; |
Tom Sepez | 7d0fcbf | 2015-09-15 15:30:34 -0700 | [diff] [blame] | 153 | } |
| 154 | |
Tom Sepez | 39bfe12 | 2015-09-17 15:25:23 -0700 | [diff] [blame] | 155 | void* FXJS_ArrayBufferAllocator::AllocateUninitialized(size_t length) { |
Chris Palmer | e4b035b | 2017-03-26 15:48:34 -0700 | [diff] [blame] | 156 | if (length > kMaxAllowedBytes) |
| 157 | return nullptr; |
| 158 | return pdfium::base::PartitionAllocGeneric( |
| 159 | gArrayBufferPartitionAllocator.root(), length, "FXJS_ArrayBuffer"); |
Tom Sepez | 7d0fcbf | 2015-09-15 15:30:34 -0700 | [diff] [blame] | 160 | } |
| 161 | |
Tom Sepez | 39bfe12 | 2015-09-17 15:25:23 -0700 | [diff] [blame] | 162 | void FXJS_ArrayBufferAllocator::Free(void* data, size_t length) { |
Chris Palmer | 5ff8c88 | 2017-03-31 16:09:54 -0700 | [diff] [blame] | 163 | pdfium::base::PartitionFreeGeneric(gArrayBufferPartitionAllocator.root(), |
| 164 | data); |
Tom Sepez | 7d0fcbf | 2015-09-15 15:30:34 -0700 | [diff] [blame] | 165 | } |
| 166 | |
jinming_wang | 61dc96f | 2016-01-28 14:39:56 +0800 | [diff] [blame] | 167 | void V8TemplateMapTraits::Dispose(v8::Isolate* isolate, |
| 168 | v8::Global<v8::Object> value, |
| 169 | void* key) { |
| 170 | v8::Local<v8::Object> obj = value.Get(isolate); |
| 171 | if (obj.IsEmpty()) |
| 172 | return; |
Tom Sepez | ce84f71 | 2017-05-30 17:17:33 -0700 | [diff] [blame] | 173 | int id = CFXJS_Engine::GetObjDefnID(obj); |
jinming_wang | 61dc96f | 2016-01-28 14:39:56 +0800 | [diff] [blame] | 174 | if (id == -1) |
| 175 | return; |
jinming_wang | 61dc96f | 2016-01-28 14:39:56 +0800 | [diff] [blame] | 176 | CFXJS_ObjDefinition* pObjDef = CFXJS_ObjDefinition::ForID(isolate, id); |
| 177 | if (!pObjDef) |
| 178 | return; |
Tom Sepez | ce84f71 | 2017-05-30 17:17:33 -0700 | [diff] [blame] | 179 | if (pObjDef->m_pDestructor) { |
| 180 | pObjDef->m_pDestructor(CFXJS_Engine::CurrentEngineFromIsolate(isolate), |
| 181 | obj); |
| 182 | } |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 183 | CFXJS_Engine::FreeObjectPrivate(obj); |
jinming_wang | 61dc96f | 2016-01-28 14:39:56 +0800 | [diff] [blame] | 184 | } |
| 185 | |
| 186 | V8TemplateMapTraits::MapType* V8TemplateMapTraits::MapFromWeakCallbackInfo( |
| 187 | const v8::WeakCallbackInfo<WeakCallbackDataType>& data) { |
| 188 | V8TemplateMap* pMap = |
weili | a4ad595 | 2016-09-22 10:38:53 -0700 | [diff] [blame] | 189 | (FXJS_PerIsolateData::Get(data.GetIsolate()))->m_pDynamicObjsMap.get(); |
jinming_wang | 61dc96f | 2016-01-28 14:39:56 +0800 | [diff] [blame] | 190 | return pMap ? &pMap->m_map : nullptr; |
| 191 | } |
| 192 | |
Tom Sepez | a72e8e2 | 2015-10-07 10:17:53 -0700 | [diff] [blame] | 193 | void FXJS_Initialize(unsigned int embedderDataSlot, v8::Isolate* pIsolate) { |
Lei Zhang | 4d4a442 | 2015-10-08 12:00:14 -0700 | [diff] [blame] | 194 | if (g_isolate) { |
| 195 | ASSERT(g_embedderDataSlot == embedderDataSlot); |
| 196 | ASSERT(g_isolate == pIsolate); |
| 197 | return; |
| 198 | } |
Tom Sepez | a72e8e2 | 2015-10-07 10:17:53 -0700 | [diff] [blame] | 199 | g_embedderDataSlot = embedderDataSlot; |
| 200 | g_isolate = pIsolate; |
| 201 | } |
| 202 | |
| 203 | void FXJS_Release() { |
Lei Zhang | 4d4a442 | 2015-10-08 12:00:14 -0700 | [diff] [blame] | 204 | ASSERT(!g_isolate || g_isolate_ref_count == 0); |
Lei Zhang | b391493 | 2015-10-08 12:08:47 -0700 | [diff] [blame] | 205 | delete g_DefaultGlobalObjectTemplate; |
Tom Sepez | a72e8e2 | 2015-10-07 10:17:53 -0700 | [diff] [blame] | 206 | g_DefaultGlobalObjectTemplate = nullptr; |
| 207 | g_isolate = nullptr; |
| 208 | |
| 209 | delete g_arrayBufferAllocator; |
| 210 | g_arrayBufferAllocator = nullptr; |
| 211 | } |
| 212 | |
| 213 | bool FXJS_GetIsolate(v8::Isolate** pResultIsolate) { |
| 214 | if (g_isolate) { |
| 215 | *pResultIsolate = g_isolate; |
| 216 | return false; |
| 217 | } |
| 218 | // Provide backwards compatibility when no external isolate. |
| 219 | if (!g_arrayBufferAllocator) |
| 220 | g_arrayBufferAllocator = new FXJS_ArrayBufferAllocator(); |
| 221 | v8::Isolate::CreateParams params; |
| 222 | params.array_buffer_allocator = g_arrayBufferAllocator; |
| 223 | *pResultIsolate = v8::Isolate::New(params); |
| 224 | return true; |
| 225 | } |
| 226 | |
Lei Zhang | 3fa115b | 2015-10-08 12:04:47 -0700 | [diff] [blame] | 227 | size_t FXJS_GlobalIsolateRefCount() { |
| 228 | return g_isolate_ref_count; |
| 229 | } |
| 230 | |
weili | 625ad66 | 2016-06-15 11:21:33 -0700 | [diff] [blame] | 231 | V8TemplateMap::V8TemplateMap(v8::Isolate* isolate) : m_map(isolate) {} |
| 232 | |
| 233 | V8TemplateMap::~V8TemplateMap() {} |
| 234 | |
| 235 | void V8TemplateMap::set(void* key, v8::Local<v8::Object> handle) { |
| 236 | ASSERT(!m_map.Contains(key)); |
| 237 | m_map.Set(key, handle); |
| 238 | } |
| 239 | |
| 240 | FXJS_PerIsolateData::~FXJS_PerIsolateData() {} |
| 241 | |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 242 | // static |
| 243 | void FXJS_PerIsolateData::SetUp(v8::Isolate* pIsolate) { |
Tom Sepez | 7d0fcbf | 2015-09-15 15:30:34 -0700 | [diff] [blame] | 244 | if (!pIsolate->GetData(g_embedderDataSlot)) |
weili | a4ad595 | 2016-09-22 10:38:53 -0700 | [diff] [blame] | 245 | pIsolate->SetData(g_embedderDataSlot, new FXJS_PerIsolateData(pIsolate)); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 246 | } |
| 247 | |
| 248 | // static |
| 249 | FXJS_PerIsolateData* FXJS_PerIsolateData::Get(v8::Isolate* pIsolate) { |
| 250 | return static_cast<FXJS_PerIsolateData*>( |
| 251 | pIsolate->GetData(g_embedderDataSlot)); |
Tom Sepez | 7d0fcbf | 2015-09-15 15:30:34 -0700 | [diff] [blame] | 252 | } |
| 253 | |
weili | a4ad595 | 2016-09-22 10:38:53 -0700 | [diff] [blame] | 254 | FXJS_PerIsolateData::FXJS_PerIsolateData(v8::Isolate* pIsolate) |
| 255 | : m_pDynamicObjsMap(new V8TemplateMap(pIsolate)) {} |
weili | 625ad66 | 2016-06-15 11:21:33 -0700 | [diff] [blame] | 256 | |
tsepez | a494191 | 2016-08-15 11:40:12 -0700 | [diff] [blame] | 257 | CFXJS_Engine::CFXJS_Engine() : m_isolate(nullptr) {} |
tsepez | a494191 | 2016-08-15 11:40:12 -0700 | [diff] [blame] | 258 | |
weili | 0b2a987 | 2016-09-21 11:50:43 -0700 | [diff] [blame] | 259 | CFXJS_Engine::CFXJS_Engine(v8::Isolate* pIsolate) : m_isolate(pIsolate) {} |
| 260 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 261 | CFXJS_Engine::~CFXJS_Engine() { |
| 262 | m_V8PersistentContext.Reset(); |
| 263 | } |
Tom Sepez | 7d0fcbf | 2015-09-15 15:30:34 -0700 | [diff] [blame] | 264 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 265 | // static |
| 266 | CFXJS_Engine* CFXJS_Engine::CurrentEngineFromIsolate(v8::Isolate* pIsolate) { |
| 267 | return static_cast<CFXJS_Engine*>( |
| 268 | pIsolate->GetCurrentContext()->GetAlignedPointerFromEmbedderData( |
| 269 | kPerContextDataIndex)); |
| 270 | } |
| 271 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 272 | // static |
| 273 | int CFXJS_Engine::GetObjDefnID(v8::Local<v8::Object> pObj) { |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 274 | CFXJS_PerObjectData* pData = CFXJS_PerObjectData::GetFromObject(pObj); |
| 275 | return pData ? pData->m_ObjDefID : -1; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 276 | } |
| 277 | |
| 278 | // static |
| 279 | void CFXJS_Engine::FreeObjectPrivate(void* pPerObjectData) { |
| 280 | delete static_cast<CFXJS_PerObjectData*>(pPerObjectData); |
| 281 | } |
| 282 | |
| 283 | // static |
| 284 | void CFXJS_Engine::FreeObjectPrivate(v8::Local<v8::Object> pObj) { |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 285 | CFXJS_PerObjectData* pData = CFXJS_PerObjectData::GetFromObject(pObj); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 286 | pObj->SetAlignedPointerInInternalField(0, nullptr); |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 287 | pObj->SetAlignedPointerInInternalField(1, nullptr); |
| 288 | delete pData; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 289 | } |
| 290 | |
Tom Sepez | 892d751 | 2017-02-21 13:57:13 -0800 | [diff] [blame] | 291 | int CFXJS_Engine::DefineObj(const char* sObjName, |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 292 | FXJSOBJTYPE eObjType, |
| 293 | CFXJS_Engine::Constructor pConstructor, |
| 294 | CFXJS_Engine::Destructor pDestructor) { |
| 295 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 296 | v8::HandleScope handle_scope(m_isolate); |
| 297 | FXJS_PerIsolateData::SetUp(m_isolate); |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 298 | CFXJS_ObjDefinition* pObjDef = new CFXJS_ObjDefinition( |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 299 | m_isolate, sObjName, eObjType, pConstructor, pDestructor); |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 300 | return pObjDef->AssignID(); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 301 | } |
| 302 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 303 | void CFXJS_Engine::DefineObjMethod(int nObjDefnID, |
Tom Sepez | 9b99b63 | 2017-02-21 15:05:57 -0800 | [diff] [blame] | 304 | const char* sMethodName, |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 305 | v8::FunctionCallback pMethodCall) { |
| 306 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 307 | v8::HandleScope handle_scope(m_isolate); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 308 | CFXJS_ObjDefinition* pObjDef = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 309 | CFXJS_ObjDefinition::ForID(m_isolate, nObjDefnID); |
jochen | c4dedf3 | 2016-07-06 05:26:23 -0700 | [diff] [blame] | 310 | v8::Local<v8::FunctionTemplate> fun = v8::FunctionTemplate::New( |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 311 | m_isolate, pMethodCall, v8::Local<v8::Value>(), pObjDef->GetSignature()); |
jochen | c4dedf3 | 2016-07-06 05:26:23 -0700 | [diff] [blame] | 312 | fun->RemovePrototype(); |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 313 | pObjDef->GetInstanceTemplate()->Set(NewString(sMethodName), fun, |
| 314 | v8::ReadOnly); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 315 | } |
| 316 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 317 | void CFXJS_Engine::DefineObjProperty(int nObjDefnID, |
Tom Sepez | 4d5b8c5 | 2017-02-21 15:17:07 -0800 | [diff] [blame] | 318 | const char* sPropName, |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 319 | v8::AccessorGetterCallback pPropGet, |
| 320 | v8::AccessorSetterCallback pPropPut) { |
| 321 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 322 | v8::HandleScope handle_scope(m_isolate); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 323 | CFXJS_ObjDefinition* pObjDef = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 324 | CFXJS_ObjDefinition::ForID(m_isolate, nObjDefnID); |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 325 | pObjDef->GetInstanceTemplate()->SetAccessor(NewString(sPropName), pPropGet, |
| 326 | pPropPut); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 327 | } |
| 328 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 329 | void CFXJS_Engine::DefineObjAllProperties( |
| 330 | int nObjDefnID, |
| 331 | v8::NamedPropertyQueryCallback pPropQurey, |
| 332 | v8::NamedPropertyGetterCallback pPropGet, |
| 333 | v8::NamedPropertySetterCallback pPropPut, |
| 334 | v8::NamedPropertyDeleterCallback pPropDel) { |
| 335 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 336 | v8::HandleScope handle_scope(m_isolate); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 337 | CFXJS_ObjDefinition* pObjDef = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 338 | CFXJS_ObjDefinition::ForID(m_isolate, nObjDefnID); |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 339 | pObjDef->GetInstanceTemplate()->SetNamedPropertyHandler(pPropGet, pPropPut, |
| 340 | pPropQurey, pPropDel); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 341 | } |
| 342 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 343 | void CFXJS_Engine::DefineObjConst(int nObjDefnID, |
Tom Sepez | 55db091 | 2017-02-21 15:26:52 -0800 | [diff] [blame] | 344 | const char* sConstName, |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 345 | v8::Local<v8::Value> pDefault) { |
| 346 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 347 | v8::HandleScope handle_scope(m_isolate); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 348 | CFXJS_ObjDefinition* pObjDef = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 349 | CFXJS_ObjDefinition::ForID(m_isolate, nObjDefnID); |
Tom Sepez | 55db091 | 2017-02-21 15:26:52 -0800 | [diff] [blame] | 350 | pObjDef->GetInstanceTemplate()->Set(m_isolate, sConstName, pDefault); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 351 | } |
| 352 | |
Tom Sepez | 9b99b63 | 2017-02-21 15:05:57 -0800 | [diff] [blame] | 353 | void CFXJS_Engine::DefineGlobalMethod(const char* sMethodName, |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 354 | v8::FunctionCallback pMethodCall) { |
| 355 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 356 | v8::HandleScope handle_scope(m_isolate); |
jochen | c4dedf3 | 2016-07-06 05:26:23 -0700 | [diff] [blame] | 357 | v8::Local<v8::FunctionTemplate> fun = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 358 | v8::FunctionTemplate::New(m_isolate, pMethodCall); |
jochen | c4dedf3 | 2016-07-06 05:26:23 -0700 | [diff] [blame] | 359 | fun->RemovePrototype(); |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 360 | GetGlobalObjectTemplate(m_isolate)->Set(NewString(sMethodName), fun, |
| 361 | v8::ReadOnly); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 362 | } |
| 363 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 364 | void CFXJS_Engine::DefineGlobalConst(const wchar_t* sConstName, |
| 365 | v8::FunctionCallback pConstGetter) { |
| 366 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 367 | v8::HandleScope handle_scope(m_isolate); |
jochen | c4dedf3 | 2016-07-06 05:26:23 -0700 | [diff] [blame] | 368 | v8::Local<v8::FunctionTemplate> fun = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 369 | v8::FunctionTemplate::New(m_isolate, pConstGetter); |
jochen | c4dedf3 | 2016-07-06 05:26:23 -0700 | [diff] [blame] | 370 | fun->RemovePrototype(); |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 371 | GetGlobalObjectTemplate(m_isolate)->SetAccessorProperty(NewString(sConstName), |
| 372 | fun); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 373 | } |
| 374 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 375 | void CFXJS_Engine::InitializeEngine() { |
| 376 | if (m_isolate == g_isolate) |
Lei Zhang | 4d4a442 | 2015-10-08 12:00:14 -0700 | [diff] [blame] | 377 | ++g_isolate_ref_count; |
| 378 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 379 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 380 | v8::HandleScope handle_scope(m_isolate); |
dsinclair | 7d554c9 | 2016-06-20 06:06:31 -0700 | [diff] [blame] | 381 | |
| 382 | // This has to happen before we call GetGlobalObjectTemplate because that |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 383 | // method gets the PerIsolateData from m_isolate. |
| 384 | FXJS_PerIsolateData::SetUp(m_isolate); |
dsinclair | 7d554c9 | 2016-06-20 06:06:31 -0700 | [diff] [blame] | 385 | |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 386 | v8::Local<v8::Context> v8Context = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 387 | v8::Context::New(m_isolate, nullptr, GetGlobalObjectTemplate(m_isolate)); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 388 | v8::Context::Scope context_scope(v8Context); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 389 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 390 | v8Context->SetAlignedPointerInEmbedderData(kPerContextDataIndex, this); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 391 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 392 | int maxID = CFXJS_ObjDefinition::MaxID(m_isolate); |
| 393 | m_StaticObjects.resize(maxID + 1); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 394 | for (int i = 0; i < maxID; ++i) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 395 | CFXJS_ObjDefinition* pObjDef = CFXJS_ObjDefinition::ForID(m_isolate, i); |
Tom Sepez | cd56a7d | 2015-10-06 11:45:28 -0700 | [diff] [blame] | 396 | if (pObjDef->m_ObjType == FXJSOBJTYPE_GLOBAL) { |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 397 | CFXJS_PerObjectData::SetInObject(new CFXJS_PerObjectData(i), |
| 398 | v8Context->Global() |
| 399 | ->GetPrototype() |
| 400 | ->ToObject(v8Context) |
| 401 | .ToLocalChecked()); |
| 402 | if (pObjDef->m_pConstructor) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 403 | pObjDef->m_pConstructor(this, v8Context->Global() |
| 404 | ->GetPrototype() |
| 405 | ->ToObject(v8Context) |
| 406 | .ToLocalChecked()); |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 407 | } |
Tom Sepez | cd56a7d | 2015-10-06 11:45:28 -0700 | [diff] [blame] | 408 | } else if (pObjDef->m_ObjType == FXJSOBJTYPE_STATIC) { |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 409 | v8::Local<v8::String> pObjName = NewString(pObjDef->m_ObjName); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 410 | v8::Local<v8::Object> obj = NewFxDynamicObj(i, true); |
Tom Sepez | 63b2fc7 | 2017-08-14 16:24:29 -0700 | [diff] [blame] | 411 | if (!obj.IsEmpty()) { |
| 412 | v8Context->Global()->Set(v8Context, pObjName, obj).FromJust(); |
| 413 | m_StaticObjects[i] = new v8::Global<v8::Object>(m_isolate, obj); |
| 414 | } else { |
| 415 | m_StaticObjects[i] = nullptr; |
| 416 | } |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 417 | } |
| 418 | } |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 419 | m_V8PersistentContext.Reset(m_isolate, v8Context); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 420 | } |
| 421 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 422 | void CFXJS_Engine::ReleaseEngine() { |
| 423 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 424 | v8::HandleScope handle_scope(m_isolate); |
Tom Sepez | 808a99e | 2015-09-10 12:28:37 -0700 | [diff] [blame] | 425 | v8::Local<v8::Context> context = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 426 | v8::Local<v8::Context>::New(m_isolate, m_V8PersistentContext); |
Tom Sepez | 808a99e | 2015-09-10 12:28:37 -0700 | [diff] [blame] | 427 | v8::Context::Scope context_scope(context); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 428 | FXJS_PerIsolateData* pData = FXJS_PerIsolateData::Get(m_isolate); |
Tom Sepez | 129a60e | 2016-03-24 09:19:20 -0700 | [diff] [blame] | 429 | if (!pData) |
| 430 | return; |
Tom Sepez | 129a60e | 2016-03-24 09:19:20 -0700 | [diff] [blame] | 431 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 432 | m_ConstArrays.clear(); |
| 433 | |
| 434 | int maxID = CFXJS_ObjDefinition::MaxID(m_isolate); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 435 | for (int i = 0; i < maxID; ++i) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 436 | CFXJS_ObjDefinition* pObjDef = CFXJS_ObjDefinition::ForID(m_isolate, i); |
Oliver Chang | 8f16b69 | 2015-10-27 09:34:49 -0700 | [diff] [blame] | 437 | v8::Local<v8::Object> pObj; |
| 438 | if (pObjDef->m_ObjType == FXJSOBJTYPE_GLOBAL) { |
| 439 | pObj = |
| 440 | context->Global()->GetPrototype()->ToObject(context).ToLocalChecked(); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 441 | } else if (m_StaticObjects[i] && !m_StaticObjects[i]->IsEmpty()) { |
| 442 | pObj = v8::Local<v8::Object>::New(m_isolate, *m_StaticObjects[i]); |
| 443 | delete m_StaticObjects[i]; |
| 444 | m_StaticObjects[i] = nullptr; |
Oliver Chang | 8f16b69 | 2015-10-27 09:34:49 -0700 | [diff] [blame] | 445 | } |
| 446 | |
| 447 | if (!pObj.IsEmpty()) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 448 | if (pObjDef->m_pDestructor) |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 449 | pObjDef->m_pDestructor(this, pObj); |
| 450 | FreeObjectPrivate(pObj); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 451 | } |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 452 | } |
Tom Sepez | 808a99e | 2015-09-10 12:28:37 -0700 | [diff] [blame] | 453 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 454 | m_V8PersistentContext.Reset(); |
| 455 | |
| 456 | if (m_isolate == g_isolate && --g_isolate_ref_count > 0) |
Tom Sepez | 4237aed | 2015-11-10 15:19:17 -0800 | [diff] [blame] | 457 | return; |
| 458 | |
Tom Sepez | 129a60e | 2016-03-24 09:19:20 -0700 | [diff] [blame] | 459 | delete pData; |
weili | a4ad595 | 2016-09-22 10:38:53 -0700 | [diff] [blame] | 460 | m_isolate->SetData(g_embedderDataSlot, nullptr); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 461 | } |
| 462 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 463 | int CFXJS_Engine::Execute(const WideString& script, FXJSErr* pError) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 464 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 465 | v8::TryCatch try_catch(m_isolate); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 466 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 467 | v8::Local<v8::Script> compiled_script; |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 468 | if (!v8::Script::Compile(context, NewString(script.AsStringView())) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 469 | .ToLocal(&compiled_script)) { |
| 470 | v8::String::Utf8Value error(try_catch.Exception()); |
Tom Sepez | 39bfe12 | 2015-09-17 15:25:23 -0700 | [diff] [blame] | 471 | // TODO(tsepez): return error via pError->message. |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 472 | return -1; |
| 473 | } |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 474 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 475 | v8::Local<v8::Value> result; |
| 476 | if (!compiled_script->Run(context).ToLocal(&result)) { |
| 477 | v8::String::Utf8Value error(try_catch.Exception()); |
Tom Sepez | 39bfe12 | 2015-09-17 15:25:23 -0700 | [diff] [blame] | 478 | // TODO(tsepez): return error via pError->message. |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 479 | return -1; |
| 480 | } |
| 481 | return 0; |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 482 | } |
| 483 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 484 | v8::Local<v8::Object> CFXJS_Engine::NewFxDynamicObj(int nObjDefnID, |
| 485 | bool bStatic) { |
| 486 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 487 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 488 | if (nObjDefnID == -1) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 489 | v8::Local<v8::ObjectTemplate> objTempl = v8::ObjectTemplate::New(m_isolate); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 490 | v8::Local<v8::Object> obj; |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 491 | if (!objTempl->NewInstance(context).ToLocal(&obj)) |
| 492 | return v8::Local<v8::Object>(); |
| 493 | return obj; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 494 | } |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 495 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 496 | FXJS_PerIsolateData* pData = FXJS_PerIsolateData::Get(m_isolate); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 497 | if (!pData) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 498 | return v8::Local<v8::Object>(); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 499 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 500 | if (nObjDefnID < 0 || nObjDefnID >= CFXJS_ObjDefinition::MaxID(m_isolate)) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 501 | return v8::Local<v8::Object>(); |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 502 | |
| 503 | CFXJS_ObjDefinition* pObjDef = |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 504 | CFXJS_ObjDefinition::ForID(m_isolate, nObjDefnID); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 505 | v8::Local<v8::Object> obj; |
Tom Sepez | 016a347 | 2015-09-30 15:50:57 -0700 | [diff] [blame] | 506 | if (!pObjDef->GetInstanceTemplate()->NewInstance(context).ToLocal(&obj)) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 507 | return v8::Local<v8::Object>(); |
Tom Sepez | 39bfe12 | 2015-09-17 15:25:23 -0700 | [diff] [blame] | 508 | |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 509 | CFXJS_PerObjectData* pObjData = new CFXJS_PerObjectData(nObjDefnID); |
| 510 | CFXJS_PerObjectData::SetInObject(pObjData, obj); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 511 | if (pObjDef->m_pConstructor) |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 512 | pObjDef->m_pConstructor(this, obj); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 513 | |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 514 | if (!bStatic && FXJS_PerIsolateData::Get(m_isolate)->m_pDynamicObjsMap) |
| 515 | FXJS_PerIsolateData::Get(m_isolate)->m_pDynamicObjsMap->set(pObjData, obj); |
| 516 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 517 | return obj; |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 518 | } |
| 519 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 520 | v8::Local<v8::Object> CFXJS_Engine::GetThisObj() { |
| 521 | v8::Isolate::Scope isolate_scope(m_isolate); |
| 522 | if (!FXJS_PerIsolateData::Get(m_isolate)) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 523 | return v8::Local<v8::Object>(); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 524 | |
Tom Sepez | ed7b2b5 | 2015-09-22 08:36:17 -0700 | [diff] [blame] | 525 | // Return the global object. |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 526 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 527 | return context->Global()->GetPrototype()->ToObject(context).ToLocalChecked(); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 528 | } |
| 529 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 530 | void CFXJS_Engine::Error(const WideString& message) { |
| 531 | m_isolate->ThrowException(NewString(message.AsStringView())); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 532 | } |
| 533 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 534 | void CFXJS_Engine::SetObjectPrivate(v8::Local<v8::Object> pObj, void* p) { |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 535 | CFXJS_PerObjectData* pPerObjectData = |
| 536 | CFXJS_PerObjectData::GetFromObject(pObj); |
Tom Sepez | 6cf117c | 2015-11-06 14:52:10 -0800 | [diff] [blame] | 537 | if (!pPerObjectData) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 538 | return; |
Tom Sepez | 6cf117c | 2015-11-06 14:52:10 -0800 | [diff] [blame] | 539 | pPerObjectData->m_pPrivate = p; |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 540 | } |
| 541 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 542 | void* CFXJS_Engine::GetObjectPrivate(v8::Local<v8::Object> pObj) { |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 543 | CFXJS_PerObjectData* pData = CFXJS_PerObjectData::GetFromObject(pObj); |
| 544 | if (!pData && !pObj.IsEmpty()) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 545 | // It could be a global proxy object. |
| 546 | v8::Local<v8::Value> v = pObj->GetPrototype(); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 547 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Lei Zhang | 4d4a442 | 2015-10-08 12:00:14 -0700 | [diff] [blame] | 548 | if (v->IsObject()) { |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 549 | pData = CFXJS_PerObjectData::GetFromObject( |
| 550 | v->ToObject(context).ToLocalChecked()); |
Lei Zhang | 4d4a442 | 2015-10-08 12:00:14 -0700 | [diff] [blame] | 551 | } |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 552 | } |
Tom Sepez | 3f72fb4 | 2017-02-27 11:43:55 -0800 | [diff] [blame] | 553 | return pData ? pData->m_pPrivate : nullptr; |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 554 | } |
| 555 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 556 | v8::Local<v8::Value> CFXJS_Engine::GetObjectProperty( |
tsepez | 018935c | 2016-04-15 13:15:12 -0700 | [diff] [blame] | 557 | v8::Local<v8::Object> pObj, |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 558 | const WideString& wsPropertyName) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 559 | if (pObj.IsEmpty()) |
| 560 | return v8::Local<v8::Value>(); |
| 561 | v8::Local<v8::Value> val; |
Tom Sepez | c6dc69f | 2017-02-23 09:53:09 -0800 | [diff] [blame] | 562 | if (!pObj->Get(m_isolate->GetCurrentContext(), |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 563 | NewString(wsPropertyName.AsStringView())) |
Dan Sinclair | f766ad2 | 2016-03-14 13:51:24 -0400 | [diff] [blame] | 564 | .ToLocal(&val)) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 565 | return v8::Local<v8::Value>(); |
| 566 | return val; |
| 567 | } |
| 568 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 569 | std::vector<WideString> CFXJS_Engine::GetObjectPropertyNames( |
tsepez | d0b6ed1 | 2016-08-11 19:50:57 -0700 | [diff] [blame] | 570 | v8::Local<v8::Object> pObj) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 571 | if (pObj.IsEmpty()) |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 572 | return std::vector<WideString>(); |
tsepez | d0b6ed1 | 2016-08-11 19:50:57 -0700 | [diff] [blame] | 573 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 574 | v8::Local<v8::Array> val; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 575 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
| 576 | if (!pObj->GetPropertyNames(context).ToLocal(&val)) |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 577 | return std::vector<WideString>(); |
tsepez | d0b6ed1 | 2016-08-11 19:50:57 -0700 | [diff] [blame] | 578 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 579 | std::vector<WideString> result; |
tsepez | d0b6ed1 | 2016-08-11 19:50:57 -0700 | [diff] [blame] | 580 | for (uint32_t i = 0; i < val->Length(); ++i) { |
tsepez | e6cf013 | 2017-01-18 14:38:18 -0800 | [diff] [blame] | 581 | result.push_back(ToWideString(val->Get(context, i).ToLocalChecked())); |
tsepez | d0b6ed1 | 2016-08-11 19:50:57 -0700 | [diff] [blame] | 582 | } |
| 583 | |
| 584 | return result; |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 585 | } |
| 586 | |
tsepez | e6cf013 | 2017-01-18 14:38:18 -0800 | [diff] [blame] | 587 | void CFXJS_Engine::PutObjectProperty(v8::Local<v8::Object> pObj, |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 588 | const WideString& wsPropertyName, |
tsepez | e6cf013 | 2017-01-18 14:38:18 -0800 | [diff] [blame] | 589 | v8::Local<v8::Value> pPut) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 590 | if (pObj.IsEmpty()) |
| 591 | return; |
Tom Sepez | c6dc69f | 2017-02-23 09:53:09 -0800 | [diff] [blame] | 592 | pObj->Set(m_isolate->GetCurrentContext(), |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 593 | NewString(wsPropertyName.AsStringView()), pPut) |
Dan Sinclair | f766ad2 | 2016-03-14 13:51:24 -0400 | [diff] [blame] | 594 | .FromJust(); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 595 | } |
| 596 | |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 597 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 598 | v8::Local<v8::Array> CFXJS_Engine::NewArray() { |
| 599 | return v8::Array::New(m_isolate); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 600 | } |
| 601 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 602 | unsigned CFXJS_Engine::PutArrayElement(v8::Local<v8::Array> pArray, |
| 603 | unsigned index, |
| 604 | v8::Local<v8::Value> pValue) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 605 | if (pArray.IsEmpty()) |
| 606 | return 0; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 607 | if (pArray->Set(m_isolate->GetCurrentContext(), index, pValue).IsNothing()) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 608 | return 0; |
| 609 | return 1; |
| 610 | } |
| 611 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 612 | v8::Local<v8::Value> CFXJS_Engine::GetArrayElement(v8::Local<v8::Array> pArray, |
| 613 | unsigned index) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 614 | if (pArray.IsEmpty()) |
| 615 | return v8::Local<v8::Value>(); |
| 616 | v8::Local<v8::Value> val; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 617 | if (!pArray->Get(m_isolate->GetCurrentContext(), index).ToLocal(&val)) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 618 | return v8::Local<v8::Value>(); |
| 619 | return val; |
| 620 | } |
| 621 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 622 | unsigned CFXJS_Engine::GetArrayLength(v8::Local<v8::Array> pArray) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 623 | if (pArray.IsEmpty()) |
| 624 | return 0; |
| 625 | return pArray->Length(); |
| 626 | } |
| 627 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 628 | v8::Local<v8::Context> CFXJS_Engine::NewLocalContext() { |
| 629 | return v8::Local<v8::Context>::New(m_isolate, m_V8PersistentContext); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 630 | } |
| 631 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 632 | v8::Local<v8::Context> CFXJS_Engine::GetPersistentContext() { |
| 633 | return m_V8PersistentContext.Get(m_isolate); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 634 | } |
| 635 | |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 636 | v8::Local<v8::Number> CFXJS_Engine::NewNumber(int number) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 637 | return v8::Int32::New(m_isolate, number); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 638 | } |
| 639 | |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 640 | v8::Local<v8::Number> CFXJS_Engine::NewNumber(double number) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 641 | return v8::Number::New(m_isolate, number); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 642 | } |
| 643 | |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 644 | v8::Local<v8::Number> CFXJS_Engine::NewNumber(float number) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 645 | return v8::Number::New(m_isolate, (float)number); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 646 | } |
| 647 | |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 648 | v8::Local<v8::Boolean> CFXJS_Engine::NewBoolean(bool b) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 649 | return v8::Boolean::New(m_isolate, b); |
| 650 | } |
| 651 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 652 | v8::Local<v8::String> CFXJS_Engine::NewString(const ByteStringView& str) { |
Tom Sepez | c6dc69f | 2017-02-23 09:53:09 -0800 | [diff] [blame] | 653 | v8::Isolate* pIsolate = m_isolate ? m_isolate : v8::Isolate::GetCurrent(); |
Tom Sepez | 33b42e4 | 2017-07-19 13:19:12 -0700 | [diff] [blame] | 654 | return v8::String::NewFromUtf8(pIsolate, str.unterminated_c_str(), |
Tom Sepez | c6dc69f | 2017-02-23 09:53:09 -0800 | [diff] [blame] | 655 | v8::NewStringType::kNormal, str.GetLength()) |
| 656 | .ToLocalChecked(); |
| 657 | } |
| 658 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 659 | v8::Local<v8::String> CFXJS_Engine::NewString(const WideStringView& str) { |
Tom Sepez | f6ca07b | 2017-06-01 09:17:18 -0700 | [diff] [blame] | 660 | // Conversion from pdfium's wchar_t wide-strings to v8's uint16_t |
| 661 | // wide-strings isn't handled by v8, so use UTF8 as a common |
| 662 | // intermediate format. |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 663 | return NewString(FX_UTF8Encode(str).AsStringView()); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 664 | } |
| 665 | |
| 666 | v8::Local<v8::Value> CFXJS_Engine::NewNull() { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 667 | return v8::Local<v8::Value>(); |
| 668 | } |
| 669 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 670 | v8::Local<v8::Date> CFXJS_Engine::NewDate(double d) { |
| 671 | return v8::Date::New(m_isolate->GetCurrentContext(), d) |
tsepez | 135b998 | 2016-08-05 09:32:50 -0700 | [diff] [blame] | 672 | .ToLocalChecked() |
| 673 | .As<v8::Date>(); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 674 | } |
| 675 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 676 | int CFXJS_Engine::ToInt32(v8::Local<v8::Value> pValue) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 677 | if (pValue.IsEmpty()) |
| 678 | return 0; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 679 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Tom Sepez | fb7021c | 2017-05-31 10:29:25 -0700 | [diff] [blame] | 680 | v8::MaybeLocal<v8::Int32> maybe_int32 = pValue->ToInt32(context); |
| 681 | if (maybe_int32.IsEmpty()) |
| 682 | return 0; |
| 683 | return maybe_int32.ToLocalChecked()->Value(); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 684 | } |
| 685 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 686 | bool CFXJS_Engine::ToBoolean(v8::Local<v8::Value> pValue) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 687 | if (pValue.IsEmpty()) |
| 688 | return false; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 689 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Tom Sepez | fb7021c | 2017-05-31 10:29:25 -0700 | [diff] [blame] | 690 | v8::MaybeLocal<v8::Boolean> maybe_boolean = pValue->ToBoolean(context); |
| 691 | if (maybe_boolean.IsEmpty()) |
| 692 | return false; |
| 693 | return maybe_boolean.ToLocalChecked()->Value(); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 694 | } |
| 695 | |
tsepez | e6cf013 | 2017-01-18 14:38:18 -0800 | [diff] [blame] | 696 | double CFXJS_Engine::ToDouble(v8::Local<v8::Value> pValue) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 697 | if (pValue.IsEmpty()) |
| 698 | return 0.0; |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 699 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Tom Sepez | fb7021c | 2017-05-31 10:29:25 -0700 | [diff] [blame] | 700 | v8::MaybeLocal<v8::Number> maybe_number = pValue->ToNumber(context); |
| 701 | if (maybe_number.IsEmpty()) |
| 702 | return 0.0; |
| 703 | return maybe_number.ToLocalChecked()->Value(); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 704 | } |
| 705 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 706 | WideString CFXJS_Engine::ToWideString(v8::Local<v8::Value> pValue) { |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 707 | if (pValue.IsEmpty()) |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 708 | return WideString(); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 709 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Tom Sepez | fb7021c | 2017-05-31 10:29:25 -0700 | [diff] [blame] | 710 | v8::MaybeLocal<v8::String> maybe_string = pValue->ToString(context); |
| 711 | if (maybe_string.IsEmpty()) |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 712 | return WideString(); |
Tom Sepez | fb7021c | 2017-05-31 10:29:25 -0700 | [diff] [blame] | 713 | v8::String::Utf8Value s(maybe_string.ToLocalChecked()); |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 714 | return WideString::FromUTF8(ByteStringView(*s, s.length())); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 715 | } |
| 716 | |
tsepez | e6cf013 | 2017-01-18 14:38:18 -0800 | [diff] [blame] | 717 | v8::Local<v8::Object> CFXJS_Engine::ToObject(v8::Local<v8::Value> pValue) { |
| 718 | if (pValue.IsEmpty() || !pValue->IsObject()) |
| 719 | return v8::Local<v8::Object>(); |
| 720 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
| 721 | return pValue->ToObject(context).ToLocalChecked(); |
| 722 | } |
| 723 | |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 724 | v8::Local<v8::Array> CFXJS_Engine::ToArray(v8::Local<v8::Value> pValue) { |
tsepez | e6cf013 | 2017-01-18 14:38:18 -0800 | [diff] [blame] | 725 | if (pValue.IsEmpty() || !pValue->IsArray()) |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 726 | return v8::Local<v8::Array>(); |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 727 | v8::Local<v8::Context> context = m_isolate->GetCurrentContext(); |
Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 728 | return v8::Local<v8::Array>::Cast(pValue->ToObject(context).ToLocalChecked()); |
John Abd-El-Malek | 5110c47 | 2014-05-17 22:33:34 -0700 | [diff] [blame] | 729 | } |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 730 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 731 | void CFXJS_Engine::SetConstArray(const WideString& name, |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 732 | v8::Local<v8::Array> array) { |
| 733 | m_ConstArrays[name] = v8::Global<v8::Array>(GetIsolate(), array); |
| 734 | } |
| 735 | |
Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 736 | v8::Local<v8::Array> CFXJS_Engine::GetConstArray(const WideString& name) { |
tsepez | b469424 | 2016-08-15 16:44:55 -0700 | [diff] [blame] | 737 | return v8::Local<v8::Array>::New(GetIsolate(), m_ConstArrays[name]); |
| 738 | } |