// Copyright 2017 PDFium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com #include "fxjs/cfx_v8.h" #include "core/fxcrt/fx_memory.h" #include "third_party/base/allocator/partition_allocator/partition_alloc.h" CFX_V8::CFX_V8(v8::Isolate* isolate) : m_pIsolate(isolate) {} CFX_V8::~CFX_V8() = default; v8::Local CFX_V8::GetObjectProperty( v8::Local pObj, const WideString& wsPropertyName) { if (pObj.IsEmpty()) return v8::Local(); v8::Local val; if (!pObj->Get(m_pIsolate->GetCurrentContext(), NewString(wsPropertyName.AsStringView())) .ToLocal(&val)) return v8::Local(); return val; } std::vector CFX_V8::GetObjectPropertyNames( v8::Local pObj) { if (pObj.IsEmpty()) return std::vector(); v8::Local val; v8::Local context = m_pIsolate->GetCurrentContext(); if (!pObj->GetPropertyNames(context).ToLocal(&val)) return std::vector(); std::vector result; for (uint32_t i = 0; i < val->Length(); ++i) { result.push_back(ToWideString(val->Get(context, i).ToLocalChecked())); } return result; } void CFX_V8::PutObjectProperty(v8::Local pObj, const WideString& wsPropertyName, v8::Local pPut) { if (pObj.IsEmpty()) return; pObj->Set(m_pIsolate->GetCurrentContext(), NewString(wsPropertyName.AsStringView()), pPut) .FromJust(); } void CFX_V8::DisposeIsolate() { if (m_pIsolate) m_pIsolate.Release()->Dispose(); } v8::Local CFX_V8::NewArray() { return v8::Array::New(GetIsolate()); } v8::Local CFX_V8::NewObject() { return v8::Object::New(GetIsolate()); } unsigned CFX_V8::PutArrayElement(v8::Local pArray, unsigned index, v8::Local pValue) { if (pArray.IsEmpty()) return 0; if (pArray->Set(m_pIsolate->GetCurrentContext(), index, pValue).IsNothing()) return 0; return 1; } v8::Local CFX_V8::GetArrayElement(v8::Local pArray, unsigned index) { if (pArray.IsEmpty()) return v8::Local(); v8::Local val; if (!pArray->Get(m_pIsolate->GetCurrentContext(), index).ToLocal(&val)) return v8::Local(); return val; } unsigned CFX_V8::GetArrayLength(v8::Local pArray) { if (pArray.IsEmpty()) return 0; return pArray->Length(); } v8::Local CFX_V8::NewNumber(int number) { return v8::Int32::New(GetIsolate(), number); } v8::Local CFX_V8::NewNumber(double number) { return v8::Number::New(GetIsolate(), number); } v8::Local CFX_V8::NewNumber(float number) { return v8::Number::New(GetIsolate(), (float)number); } v8::Local CFX_V8::NewBoolean(bool b) { return v8::Boolean::New(GetIsolate(), b); } v8::Local CFX_V8::NewString(const ByteStringView& str) { v8::Isolate* pIsolate = m_pIsolate ? GetIsolate() : v8::Isolate::GetCurrent(); return v8::String::NewFromUtf8(pIsolate, str.unterminated_c_str(), v8::NewStringType::kNormal, str.GetLength()) .ToLocalChecked(); } v8::Local CFX_V8::NewString(const WideStringView& str) { // Conversion from pdfium's wchar_t wide-strings to v8's uint16_t // wide-strings isn't handled by v8, so use UTF8 as a common // intermediate format. return NewString(FX_UTF8Encode(str).AsStringView()); } v8::Local CFX_V8::NewNull() { return v8::Null(GetIsolate()); } v8::Local CFX_V8::NewUndefined() { return v8::Undefined(GetIsolate()); } v8::Local CFX_V8::NewDate(double d) { return v8::Date::New(m_pIsolate->GetCurrentContext(), d) .ToLocalChecked() .As(); } int CFX_V8::ToInt32(v8::Local pValue) { if (pValue.IsEmpty()) return 0; v8::Local context = m_pIsolate->GetCurrentContext(); v8::MaybeLocal maybe_int32 = pValue->ToInt32(context); if (maybe_int32.IsEmpty()) return 0; return maybe_int32.ToLocalChecked()->Value(); } bool CFX_V8::ToBoolean(v8::Local pValue) { if (pValue.IsEmpty()) return false; return pValue->BooleanValue(m_pIsolate.Get()); } double CFX_V8::ToDouble(v8::Local pValue) { if (pValue.IsEmpty()) return 0.0; v8::Local context = m_pIsolate->GetCurrentContext(); v8::MaybeLocal maybe_number = pValue->ToNumber(context); if (maybe_number.IsEmpty()) return 0.0; return maybe_number.ToLocalChecked()->Value(); } WideString CFX_V8::ToWideString(v8::Local pValue) { if (pValue.IsEmpty()) return WideString(); v8::Local context = m_pIsolate->GetCurrentContext(); v8::MaybeLocal maybe_string = pValue->ToString(context); if (maybe_string.IsEmpty()) return WideString(); v8::String::Utf8Value s(GetIsolate(), maybe_string.ToLocalChecked()); return WideString::FromUTF8(ByteStringView(*s, s.length())); } ByteString CFX_V8::ToByteString(v8::Local pValue) { if (pValue.IsEmpty()) return ByteString(); v8::Local context = m_pIsolate->GetCurrentContext(); v8::MaybeLocal maybe_string = pValue->ToString(context); if (maybe_string.IsEmpty()) return ByteString(); v8::String::Utf8Value s(GetIsolate(), maybe_string.ToLocalChecked()); return ByteString(*s); } v8::Local CFX_V8::ToObject(v8::Local pValue) { if (pValue.IsEmpty() || !pValue->IsObject()) return v8::Local(); v8::Local context = m_pIsolate->GetCurrentContext(); return pValue->ToObject(context).ToLocalChecked(); } v8::Local CFX_V8::ToArray(v8::Local pValue) { if (pValue.IsEmpty() || !pValue->IsArray()) return v8::Local(); v8::Local context = m_pIsolate->GetCurrentContext(); return v8::Local::Cast(pValue->ToObject(context).ToLocalChecked()); } void* CFX_V8ArrayBufferAllocator::Allocate(size_t length) { if (length > kMaxAllowedBytes) return nullptr; return GetArrayBufferPartitionAllocator().root()->AllocFlags( pdfium::base::PartitionAllocZeroFill, length, "CFX_V8ArrayBuffer"); } void* CFX_V8ArrayBufferAllocator::AllocateUninitialized(size_t length) { if (length > kMaxAllowedBytes) return nullptr; return GetArrayBufferPartitionAllocator().root()->Alloc(length, "CFX_V8ArrayBuffer"); } void CFX_V8ArrayBufferAllocator::Free(void* data, size_t length) { GetArrayBufferPartitionAllocator().root()->Free(data); }