arkcompiler_ets_runtime/ecmascript/js_array_iterator.cpp
xliu 3669288c00 Js TypeArray optimization
Description
  1. Add number check in CanonicalNumericIndexString;
  2. modif GetProperty, SetProperty, DeleteProperty,
     DefineOwnProperty,GetOwnProperty,HasProperty
  3. Add FastPath in TypeArray.
  4. Refactor TypeArray ArrayLength, ByteOffset, ByteLength
  5. Add test for TypeArray.
Issue:
  #I56S75:Js TypeArray optimization

Signed-off-by: xliu <liuxin259@huawei.com>
Change-Id: I68dd973269eb0bab284e7b97e0b8bf93e5aee1a2
2022-05-21 18:15:02 +08:00

97 lines
4.4 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright (c) 2021 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 "js_array_iterator.h"
#include "builtins/builtins_errors.h"
#include "ecmascript/base/typed_array_helper-inl.h"
#include "ecmascript/base/typed_array_helper.h"
#include "global_env.h"
#include "js_array.h"
#include "object_factory.h"
namespace panda::ecmascript {
using BuiltinsBase = base::BuiltinsBase;
// 22.1.5.2.1 %ArrayIteratorPrototype%.next ( )
JSTaggedValue JSArrayIterator::Next(EcmaRuntimeCallInfo *argv)
{
ASSERT(argv);
JSThread *thread = argv->GetThread();
[[maybe_unused]] EcmaHandleScope handleScope(thread);
// 1.Let O be the this value.
JSHandle<JSTaggedValue> input(BuiltinsBase::GetThis(argv));
// 2.If Type(O) is not Object, throw a TypeError exception.
// 3.If O does not have all of the internal slots of an TaggedArray Iterator Instance (22.1.5.3), throw a TypeError
// exception.
if (!input->IsJSArrayIterator()) {
THROW_TYPE_ERROR_AND_RETURN(thread, "this value is not an array iterator", JSTaggedValue::Exception());
}
JSHandle<JSArrayIterator> iter(input);
// 4.Let a be O.[[IteratedArrayLike]].
JSHandle<JSTaggedValue> array(thread, iter->GetIteratedArray());
JSHandle<JSTaggedValue> undefinedHandle(thread, JSTaggedValue::Undefined());
// 5.If a is undefined, return CreateIterResultObject(undefined, true).
if (array->IsUndefined()) {
return JSIterator::CreateIterResultObject(thread, undefinedHandle, true).GetTaggedValue();
}
// 6.Let index be O.[[ArrayLikeNextIndex]].
uint32_t index = iter->GetNextIndex();
// 7.Let itemKind be O.[[ArrayLikeIterationKind]].
IterationKind itemKind = iter->GetIterationKind();
uint32_t length;
// 8. If a has a [[TypedArrayName]] internal slot, then
// a. Let len be the value of Os [[ArrayLength]] internal slot.
if (array->IsTypedArray()) {
length = JSHandle<JSTypedArray>::Cast(array)->GetArrayLength();
} else {
// 9.Else
JSHandle<JSTaggedValue> lengthKey = thread->GlobalConstants()->GetHandledLengthString();
length = JSTaggedValue::GetProperty(thread, array, lengthKey).GetValue()->GetArrayLength();
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
}
// 10.If index ≥ len, then
if (index >= length) {
// Set O.[[IteratedArrayLike]] to undefined.
// Return CreateIterResultObject(undefined, true).
iter->SetIteratedArray(thread, undefinedHandle);
return JSIterator::CreateIterResultObject(thread, undefinedHandle, true).GetTaggedValue();
}
// 11.Set O.[[ArrayLikeNextIndex]] to index + 1.
iter->SetNextIndex(index + 1);
// 12.If itemKind is key, return CreateIterResultObject(index, false).
JSHandle<JSTaggedValue> key(thread, JSTaggedValue(index));
if (itemKind == IterationKind::KEY) {
return JSIterator::CreateIterResultObject(thread, key, false).GetTaggedValue();
}
JSHandle<JSTaggedValue> sKey(JSTaggedValue::ToString(thread, key));
JSHandle<JSTaggedValue> value = JSTaggedValue::GetProperty(thread, array, sKey).GetValue();
RETURN_EXCEPTION_IF_ABRUPT_COMPLETION(thread);
// 15.If itemKind is value, let result be elementValue.
if (itemKind == IterationKind::VALUE) {
return JSIterator::CreateIterResultObject(thread, value, false).GetTaggedValue();
}
// 16. Else
ASSERT_PRINT(itemKind == IterationKind::KEY_AND_VALUE, "itemKind is invalid");
ObjectFactory *factory = thread->GetEcmaVM()->GetFactory();
JSHandle<TaggedArray> resultArray(factory->NewTaggedArray(2)); // 2 means the length of array
resultArray->Set(thread, 0, key);
resultArray->Set(thread, 1, value);
JSHandle<JSTaggedValue> keyAndValue(JSArray::CreateArrayFromList(thread, resultArray));
return JSIterator::CreateIterResultObject(thread, keyAndValue, false).GetTaggedValue();
}
} // namespace panda::ecmascript