/* * Copyright (c) 2022 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "ecmascript/js_api/js_api_stack.h" #include "ecmascript/containers/containers_errors.h" #include "ecmascript/object_factory-inl.h" namespace panda::ecmascript { using ContainerError = containers::ContainerError; using ErrorFlag = containers::ErrorFlag; bool JSAPIStack::Empty() { if (this->GetTop() == -1) { return true; } return false; } JSTaggedValue JSAPIStack::Push(JSThread *thread, const JSHandle &stack, const JSHandle &value) { int top = static_cast(stack->GetTop()); JSHandle elements = GrowCapacity(thread, stack, top + 1); ASSERT(!elements->IsDictionaryMode()); elements->Set(thread, top + 1, value); stack->SetTop(++top); return value.GetTaggedValue(); } JSTaggedValue JSAPIStack::Peek(JSThread *thread) { int top = this->GetTop(); if (top == -1) { return JSTaggedValue::Undefined(); } TaggedArray *elements = TaggedArray::Cast(this->GetElements(thread).GetTaggedObject()); ASSERT(!elements->IsDictionaryMode()); return elements->Get(thread, top); } JSTaggedValue JSAPIStack::Pop(JSThread *thread) { int top = this->GetTop(); if (top == -1) { return JSTaggedValue::Undefined(); } TaggedArray *elements = TaggedArray::Cast(this->GetElements(thread).GetTaggedObject()); ASSERT(!elements->IsDictionaryMode()); this->SetTop(--top); return elements->Get(thread, top + 1); } int JSAPIStack::Search(JSThread *thread, const JSHandle &value) { int top = this->GetTop(); TaggedArray *elements = TaggedArray::Cast(this->GetElements(thread).GetTaggedObject()); ASSERT(!elements->IsDictionaryMode()); for (int i = 0; i <= top; i++) { if (value.GetTaggedValue() == elements->Get(thread, i)) { return i; } } return -1; } JSHandle JSAPIStack::GrowCapacity(const JSThread *thread, const JSHandle &obj, uint32_t capacity) { JSHandle oldElements(thread, obj->GetElements(thread)); uint32_t oldLength = oldElements->GetLength(); if (capacity < oldLength) { return oldElements; } uint32_t newCapacity = ComputeCapacity(capacity); JSHandle newElements = thread->GetEcmaVM()->GetFactory()->CopyArray(oldElements, oldLength, newCapacity); obj->SetElements(thread, newElements); return newElements; } JSTaggedValue JSAPIStack::Get(JSThread *thread, const uint32_t index) { ASSERT(static_cast(index) <= GetTop()); TaggedArray *elements = TaggedArray::Cast(GetElements(thread).GetTaggedObject()); return elements->Get(thread, index); } JSTaggedValue JSAPIStack::Set(JSThread *thread, const uint32_t index, JSTaggedValue value) { int32_t length = GetSize() + 1; if (index >= static_cast(length)) { return JSTaggedValue::Undefined(); } TaggedArray *elements = TaggedArray::Cast(GetElements(thread).GetTaggedObject()); elements->Set(thread, index, value); return JSTaggedValue::Undefined(); } bool JSAPIStack::Has(JSThread *thread, JSTaggedValue value) const { TaggedArray *elements = TaggedArray::Cast(GetElements(thread).GetTaggedObject()); int top = static_cast(GetTop()); if (top == -1) { return false; } for (int i = 0; i < top + 1; i++) { if (JSTaggedValue::SameValue(thread, elements->Get(thread, i), value)) { return true; } } return false; } JSHandle JSAPIStack::OwnKeys(JSThread *thread, const JSHandle &obj) { return JSObject::GetOwnPropertyKeys(thread, JSHandle::Cast(obj)); } JSHandle JSAPIStack::OwnEnumKeys(JSThread *thread, const JSHandle &obj) { return JSObject::GetOwnEnumPropertyKeys(thread, JSHandle::Cast(obj)); } bool JSAPIStack::GetOwnProperty(JSThread *thread, const JSHandle &obj, const JSHandle &key) { uint32_t index = 0; if (UNLIKELY(!JSTaggedValue::ToElementIndex(thread, key.GetTaggedValue(), &index))) { JSHandle result = JSTaggedValue::ToString(thread, key.GetTaggedValue()); RETURN_VALUE_IF_ABRUPT_COMPLETION(thread, false); CString errorMsg = "The type of \"index\" can not obtain attributes of no-number type. Received value is: " + ConvertToString(thread, *result); JSTaggedValue error = ContainerError::BusinessError(thread, ErrorFlag::TYPE_ERROR, errorMsg.c_str()); THROW_NEW_ERROR_AND_RETURN_VALUE(thread, error, false); } uint32_t length = static_cast(obj->GetTop() + 1); if (length == 0) { JSTaggedValue error = ContainerError::BusinessError(thread, ErrorFlag::RANGE_ERROR, "Container is empty"); THROW_NEW_ERROR_AND_RETURN_VALUE(thread, error, false); } if (index >= length) { std::ostringstream oss; ASSERT(length > 0); oss << "The value of \"index\" is out of range. It must be > " << (length - 1) << ". Received value is: " << index; JSTaggedValue error = ContainerError::BusinessError(thread, ErrorFlag::RANGE_ERROR, oss.str().c_str()); THROW_NEW_ERROR_AND_RETURN_VALUE(thread, error, false); } obj->Get(thread, index); return true; } OperationResult JSAPIStack::GetProperty(JSThread *thread, const JSHandle &obj, const JSHandle &key) { int length = obj->GetTop() + 1; if (length == 0) { JSTaggedValue error = ContainerError::BusinessError(thread, ErrorFlag::RANGE_ERROR, "Container is empty"); THROW_NEW_ERROR_AND_RETURN_VALUE(thread, error, OperationResult(thread, JSTaggedValue::Exception(), PropertyMetaData(false))); } JSHandle indexKey = key; if (indexKey->IsDouble()) { // Math.floor(1) will produce TaggedDouble, we need to cast into TaggedInt // For integer which is greater than INT32_MAX, it will remain TaggedDouble indexKey = JSHandle(thread, JSTaggedValue::TryCastDoubleToInt32(indexKey->GetDouble())); } if (!indexKey->IsInt()) { CString errorMsg = "The type of \"index\" must be small integer."; JSTaggedValue error = ContainerError::BusinessError(thread, ErrorFlag::TYPE_ERROR, errorMsg.c_str()); THROW_NEW_ERROR_AND_RETURN_VALUE(thread, error, OperationResult(thread, JSTaggedValue::Exception(), PropertyMetaData(false))); } int index = indexKey->GetInt(); if (index < 0 || index >= length) { std::ostringstream oss; oss << "The value of \"index\" is out of range. It must be >= 0 && <= " << (length - 1) << ". Received value is: " << index; JSTaggedValue error = ContainerError::BusinessError(thread, ErrorFlag::RANGE_ERROR, oss.str().c_str()); THROW_NEW_ERROR_AND_RETURN_VALUE(thread, error, OperationResult(thread, JSTaggedValue::Exception(), PropertyMetaData(false))); } return OperationResult(thread, obj->Get(thread, index), PropertyMetaData(false)); } bool JSAPIStack::SetProperty(JSThread *thread, const JSHandle &obj, const JSHandle &key, const JSHandle &value) { int length = obj->GetTop() + 1; JSHandle indexKey = key; if (indexKey->IsDouble()) { // Math.floor(1) will produce TaggedDouble, we need to cast into TaggedInt // For integer which is greater than INT32_MAX, it will remain TaggedDouble indexKey = JSHandle(thread, JSTaggedValue::TryCastDoubleToInt32(indexKey->GetDouble())); } if (!indexKey->IsInt()) { return false; } int index = indexKey->GetInt(); if (index < 0 || index >= length) { return false; } obj->Set(thread, index, value.GetTaggedValue()); return true; } } // namespace panda::ecmascript