338 lines
11 KiB
C++
338 lines
11 KiB
C++
// Copyright 2020 The PDFium Authors
|
|
// 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/fxv8.h"
|
|
|
|
#include "third_party/base/numerics/safe_conversions.h"
|
|
#include "v8/include/v8-container.h"
|
|
#include "v8/include/v8-date.h"
|
|
#include "v8/include/v8-exception.h"
|
|
#include "v8/include/v8-isolate.h"
|
|
#include "v8/include/v8-primitive.h"
|
|
#include "v8/include/v8-value.h"
|
|
|
|
namespace fxv8 {
|
|
|
|
bool IsUndefined(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsUndefined();
|
|
}
|
|
|
|
bool IsNull(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsNull();
|
|
}
|
|
|
|
bool IsBoolean(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsBoolean();
|
|
}
|
|
|
|
bool IsString(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsString();
|
|
}
|
|
|
|
bool IsNumber(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsNumber();
|
|
}
|
|
|
|
bool IsInteger(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsInt32();
|
|
}
|
|
|
|
bool IsObject(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsObject();
|
|
}
|
|
|
|
bool IsArray(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsArray();
|
|
}
|
|
|
|
bool IsDate(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsDate();
|
|
}
|
|
|
|
bool IsFunction(v8::Local<v8::Value> value) {
|
|
return !value.IsEmpty() && value->IsFunction();
|
|
}
|
|
|
|
v8::Local<v8::Value> NewNullHelper(v8::Isolate* pIsolate) {
|
|
return v8::Null(pIsolate);
|
|
}
|
|
|
|
v8::Local<v8::Value> NewUndefinedHelper(v8::Isolate* pIsolate) {
|
|
return v8::Undefined(pIsolate);
|
|
}
|
|
|
|
v8::Local<v8::Number> NewNumberHelper(v8::Isolate* pIsolate, int number) {
|
|
return v8::Int32::New(pIsolate, number);
|
|
}
|
|
|
|
v8::Local<v8::Number> NewNumberHelper(v8::Isolate* pIsolate, double number) {
|
|
return v8::Number::New(pIsolate, number);
|
|
}
|
|
|
|
v8::Local<v8::Number> NewNumberHelper(v8::Isolate* pIsolate, float number) {
|
|
return v8::Number::New(pIsolate, number);
|
|
}
|
|
|
|
v8::Local<v8::Boolean> NewBooleanHelper(v8::Isolate* pIsolate, bool b) {
|
|
return v8::Boolean::New(pIsolate, b);
|
|
}
|
|
|
|
v8::Local<v8::String> NewStringHelper(v8::Isolate* pIsolate,
|
|
ByteStringView str) {
|
|
return v8::String::NewFromUtf8(
|
|
pIsolate, str.unterminated_c_str(), v8::NewStringType::kNormal,
|
|
pdfium::base::checked_cast<int>(str.GetLength()))
|
|
.ToLocalChecked();
|
|
}
|
|
|
|
v8::Local<v8::String> NewStringHelper(v8::Isolate* pIsolate,
|
|
WideStringView str) {
|
|
return NewStringHelper(pIsolate, FX_UTF8Encode(str).AsStringView());
|
|
}
|
|
|
|
v8::Local<v8::Array> NewArrayHelper(v8::Isolate* pIsolate) {
|
|
return v8::Array::New(pIsolate);
|
|
}
|
|
|
|
v8::Local<v8::Array> NewArrayHelper(v8::Isolate* pIsolate,
|
|
pdfium::span<v8::Local<v8::Value>> values) {
|
|
v8::Local<v8::Array> result = NewArrayHelper(pIsolate);
|
|
for (size_t i = 0; i < values.size(); ++i) {
|
|
fxv8::ReentrantPutArrayElementHelper(
|
|
pIsolate, result, i,
|
|
values[i].IsEmpty() ? fxv8::NewUndefinedHelper(pIsolate) : values[i]);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
v8::Local<v8::Object> NewObjectHelper(v8::Isolate* pIsolate) {
|
|
return v8::Object::New(pIsolate);
|
|
}
|
|
|
|
v8::Local<v8::Date> NewDateHelper(v8::Isolate* pIsolate, double d) {
|
|
return v8::Date::New(pIsolate->GetCurrentContext(), d)
|
|
.ToLocalChecked()
|
|
.As<v8::Date>();
|
|
}
|
|
|
|
WideString ToWideString(v8::Isolate* pIsolate, v8::Local<v8::String> pValue) {
|
|
v8::String::Utf8Value s(pIsolate, pValue);
|
|
return WideString::FromUTF8(ByteStringView(*s, s.length()));
|
|
}
|
|
|
|
ByteString ToByteString(v8::Isolate* pIsolate, v8::Local<v8::String> pValue) {
|
|
v8::String::Utf8Value s(pIsolate, pValue);
|
|
return ByteString(*s, s.length());
|
|
}
|
|
|
|
int ReentrantToInt32Helper(v8::Isolate* pIsolate, v8::Local<v8::Value> pValue) {
|
|
if (pValue.IsEmpty())
|
|
return 0;
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
return pValue->Int32Value(pIsolate->GetCurrentContext()).FromMaybe(0);
|
|
}
|
|
|
|
bool ReentrantToBooleanHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (pValue.IsEmpty())
|
|
return false;
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
return pValue->BooleanValue(pIsolate);
|
|
}
|
|
|
|
float ReentrantToFloatHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Value> pValue) {
|
|
return static_cast<float>(ReentrantToDoubleHelper(pIsolate, pValue));
|
|
}
|
|
|
|
double ReentrantToDoubleHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (pValue.IsEmpty())
|
|
return 0.0;
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
return pValue->NumberValue(pIsolate->GetCurrentContext()).FromMaybe(0.0);
|
|
}
|
|
|
|
WideString ReentrantToWideStringHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (pValue.IsEmpty())
|
|
return WideString();
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::MaybeLocal<v8::String> maybe_string =
|
|
pValue->ToString(pIsolate->GetCurrentContext());
|
|
if (maybe_string.IsEmpty())
|
|
return WideString();
|
|
|
|
return ToWideString(pIsolate, maybe_string.ToLocalChecked());
|
|
}
|
|
|
|
ByteString ReentrantToByteStringHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (pValue.IsEmpty())
|
|
return ByteString();
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::MaybeLocal<v8::String> maybe_string =
|
|
pValue->ToString(pIsolate->GetCurrentContext());
|
|
if (maybe_string.IsEmpty())
|
|
return ByteString();
|
|
|
|
return ToByteString(pIsolate, maybe_string.ToLocalChecked());
|
|
}
|
|
|
|
v8::Local<v8::Object> ReentrantToObjectHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (!fxv8::IsObject(pValue))
|
|
return v8::Local<v8::Object>();
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::Context> context = pIsolate->GetCurrentContext();
|
|
return pValue->ToObject(context).ToLocalChecked();
|
|
}
|
|
|
|
v8::Local<v8::Array> ReentrantToArrayHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (!fxv8::IsArray(pValue))
|
|
return v8::Local<v8::Array>();
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::Context> context = pIsolate->GetCurrentContext();
|
|
return v8::Local<v8::Array>::Cast(pValue->ToObject(context).ToLocalChecked());
|
|
}
|
|
|
|
v8::Local<v8::Value> ReentrantGetObjectPropertyHelper(
|
|
v8::Isolate* pIsolate,
|
|
v8::Local<v8::Object> pObj,
|
|
ByteStringView bsUTF8PropertyName) {
|
|
if (pObj.IsEmpty())
|
|
return v8::Local<v8::Value>();
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::Value> val;
|
|
if (!pObj->Get(pIsolate->GetCurrentContext(),
|
|
NewStringHelper(pIsolate, bsUTF8PropertyName))
|
|
.ToLocal(&val)) {
|
|
return v8::Local<v8::Value>();
|
|
}
|
|
return val;
|
|
}
|
|
|
|
std::vector<WideString> ReentrantGetObjectPropertyNamesHelper(
|
|
v8::Isolate* pIsolate,
|
|
v8::Local<v8::Object> pObj) {
|
|
if (pObj.IsEmpty())
|
|
return std::vector<WideString>();
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::Array> val;
|
|
v8::Local<v8::Context> context = pIsolate->GetCurrentContext();
|
|
if (!pObj->GetPropertyNames(context).ToLocal(&val))
|
|
return std::vector<WideString>();
|
|
|
|
std::vector<WideString> result;
|
|
for (uint32_t i = 0; i < val->Length(); ++i) {
|
|
result.push_back(ReentrantToWideStringHelper(
|
|
pIsolate, val->Get(context, i).ToLocalChecked()));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
bool ReentrantHasObjectOwnPropertyHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Object> pObj,
|
|
ByteStringView bsUTF8PropertyName) {
|
|
if (pObj.IsEmpty())
|
|
return false;
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::Context> pContext = pIsolate->GetCurrentContext();
|
|
v8::Local<v8::String> hKey =
|
|
fxv8::NewStringHelper(pIsolate, bsUTF8PropertyName);
|
|
return pObj->HasRealNamedProperty(pContext, hKey).FromJust();
|
|
}
|
|
|
|
bool ReentrantSetObjectOwnPropertyHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Object> pObj,
|
|
ByteStringView bsUTF8PropertyName,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (pObj.IsEmpty() || pValue.IsEmpty())
|
|
return false;
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::String> name = NewStringHelper(pIsolate, bsUTF8PropertyName);
|
|
return pObj->DefineOwnProperty(pIsolate->GetCurrentContext(), name, pValue)
|
|
.FromMaybe(false);
|
|
}
|
|
|
|
bool ReentrantPutObjectPropertyHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Object> pObj,
|
|
ByteStringView bsUTF8PropertyName,
|
|
v8::Local<v8::Value> pPut) {
|
|
if (pObj.IsEmpty() || pPut.IsEmpty())
|
|
return false;
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::String> name = NewStringHelper(pIsolate, bsUTF8PropertyName);
|
|
v8::Maybe<bool> result = pObj->Set(pIsolate->GetCurrentContext(), name, pPut);
|
|
return result.IsJust() && result.FromJust();
|
|
}
|
|
|
|
void ReentrantDeleteObjectPropertyHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Object> pObj,
|
|
ByteStringView bsUTF8PropertyName) {
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
pObj->Delete(pIsolate->GetCurrentContext(),
|
|
fxv8::NewStringHelper(pIsolate, bsUTF8PropertyName))
|
|
.FromJust();
|
|
}
|
|
|
|
bool ReentrantPutArrayElementHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Array> pArray,
|
|
size_t index,
|
|
v8::Local<v8::Value> pValue) {
|
|
if (pArray.IsEmpty())
|
|
return false;
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Maybe<bool> result =
|
|
pArray->Set(pIsolate->GetCurrentContext(),
|
|
pdfium::base::checked_cast<uint32_t>(index), pValue);
|
|
return result.IsJust() && result.FromJust();
|
|
}
|
|
|
|
v8::Local<v8::Value> ReentrantGetArrayElementHelper(v8::Isolate* pIsolate,
|
|
v8::Local<v8::Array> pArray,
|
|
size_t index) {
|
|
if (pArray.IsEmpty())
|
|
return v8::Local<v8::Value>();
|
|
|
|
v8::TryCatch squash_exceptions(pIsolate);
|
|
v8::Local<v8::Value> val;
|
|
if (!pArray
|
|
->Get(pIsolate->GetCurrentContext(),
|
|
pdfium::base::checked_cast<uint32_t>(index))
|
|
.ToLocal(&val)) {
|
|
return v8::Local<v8::Value>();
|
|
}
|
|
return val;
|
|
}
|
|
|
|
size_t GetArrayLengthHelper(v8::Local<v8::Array> pArray) {
|
|
if (pArray.IsEmpty())
|
|
return 0;
|
|
return pArray->Length();
|
|
}
|
|
|
|
void ThrowExceptionHelper(v8::Isolate* pIsolate, ByteStringView str) {
|
|
pIsolate->ThrowException(NewStringHelper(pIsolate, str));
|
|
}
|
|
|
|
void ThrowExceptionHelper(v8::Isolate* pIsolate, WideStringView str) {
|
|
pIsolate->ThrowException(NewStringHelper(pIsolate, str));
|
|
}
|
|
|
|
} // namespace fxv8
|