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