14514f5e3Sopenharmony_ci/* 24514f5e3Sopenharmony_ci * Copyright (c) 2023 Huawei Device Co., Ltd. 34514f5e3Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License"); 44514f5e3Sopenharmony_ci * you may not use this file except in compliance with the License. 54514f5e3Sopenharmony_ci * You may obtain a copy of the License at 64514f5e3Sopenharmony_ci * 74514f5e3Sopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0 84514f5e3Sopenharmony_ci * 94514f5e3Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software 104514f5e3Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS, 114514f5e3Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 124514f5e3Sopenharmony_ci * See the License for the specific language governing permissions and 134514f5e3Sopenharmony_ci * limitations under the License. 144514f5e3Sopenharmony_ci */ 154514f5e3Sopenharmony_ci 164514f5e3Sopenharmony_ci#ifndef ECMASCRIPT_ELEMENT_ACCESSOR_INL_H 174514f5e3Sopenharmony_ci#define ECMASCRIPT_ELEMENT_ACCESSOR_INL_H 184514f5e3Sopenharmony_ci 194514f5e3Sopenharmony_ci#include "ecmascript/element_accessor.h" 204514f5e3Sopenharmony_ci 214514f5e3Sopenharmony_ci#include "ecmascript/mem/barriers-inl.h" 224514f5e3Sopenharmony_ci#include "ecmascript/js_tagged_value-inl.h" 234514f5e3Sopenharmony_ci#include "ecmascript/js_thread.h" 244514f5e3Sopenharmony_ci#include "ecmascript/tagged_array.h" 254514f5e3Sopenharmony_ci 264514f5e3Sopenharmony_cinamespace panda::ecmascript { 274514f5e3Sopenharmony_ci 284514f5e3Sopenharmony_citemplate<typename T> 294514f5e3Sopenharmony_ciinline void ElementAccessor::Set(const JSThread *thread, JSHandle<JSObject> receiver, uint32_t idx, 304514f5e3Sopenharmony_ci const JSHandle<T> &value, bool needTransition, ElementsKind extraKind) 314514f5e3Sopenharmony_ci{ 324514f5e3Sopenharmony_ci // Change elementsKind 334514f5e3Sopenharmony_ci ElementsKind oldKind = receiver->GetClass()->GetElementsKind(); 344514f5e3Sopenharmony_ci if (needTransition && JSHClass::TransitToElementsKind(thread, receiver, 354514f5e3Sopenharmony_ci JSHandle<JSTaggedValue>(value), extraKind)) { 364514f5e3Sopenharmony_ci ElementsKind newKind = receiver->GetClass()->GetElementsKind(); 374514f5e3Sopenharmony_ci // GC might happen when migrating Array 384514f5e3Sopenharmony_ci Elements::MigrateArrayWithKind(thread, receiver, oldKind, newKind); 394514f5e3Sopenharmony_ci } 404514f5e3Sopenharmony_ci 414514f5e3Sopenharmony_ci TaggedArray *elements = TaggedArray::Cast(receiver->GetElements()); 424514f5e3Sopenharmony_ci ASSERT(idx < elements->GetLength()); 434514f5e3Sopenharmony_ci size_t offset = JSTaggedValue::TaggedTypeSize() * idx; 444514f5e3Sopenharmony_ci 454514f5e3Sopenharmony_ci ElementsKind kind = receiver->GetClass()->GetElementsKind(); 464514f5e3Sopenharmony_ci if (!elements->GetClass()->IsMutantTaggedArray()) { 474514f5e3Sopenharmony_ci kind = ElementsKind::GENERIC; 484514f5e3Sopenharmony_ci } 494514f5e3Sopenharmony_ci 504514f5e3Sopenharmony_ci JSTaggedType convertedValue = ConvertTaggedValueWithElementsKind(value.GetTaggedValue(), kind); 514514f5e3Sopenharmony_ci // NOLINTNEXTLINE(readability-braces-around-statements, bugprone-suspicious-semicolon) 524514f5e3Sopenharmony_ci if (value.GetTaggedValue().IsHeapObject()) { 534514f5e3Sopenharmony_ci Barriers::SetObject<true>(thread, elements->GetData(), offset, convertedValue); 544514f5e3Sopenharmony_ci } else { // NOLINTNEXTLINE(readability-misleading-indentation) 554514f5e3Sopenharmony_ci Barriers::SetPrimitive<JSTaggedType>(elements->GetData(), offset, convertedValue); 564514f5e3Sopenharmony_ci } 574514f5e3Sopenharmony_ci} 584514f5e3Sopenharmony_ci 594514f5e3Sopenharmony_citemplate<typename T> 604514f5e3Sopenharmony_civoid ElementAccessor::FastSet(const JSThread *thread, JSHandle<TaggedArray> elements, uint32_t idx, 614514f5e3Sopenharmony_ci const JSHandle<T> &value, ElementsKind kind) 624514f5e3Sopenharmony_ci{ 634514f5e3Sopenharmony_ci ASSERT(idx < elements->GetLength()); 644514f5e3Sopenharmony_ci size_t offset = JSTaggedValue::TaggedTypeSize() * idx; 654514f5e3Sopenharmony_ci JSTaggedValue rawValue = value.GetTaggedValue(); 664514f5e3Sopenharmony_ci switch (kind) { 674514f5e3Sopenharmony_ci case ElementsKind::INT: 684514f5e3Sopenharmony_ci Barriers::SetPrimitive<JSTaggedType>(elements->GetData(), offset, 694514f5e3Sopenharmony_ci static_cast<JSTaggedType>(rawValue.GetInt())); 704514f5e3Sopenharmony_ci break; 714514f5e3Sopenharmony_ci case ElementsKind::NUMBER: 724514f5e3Sopenharmony_ci if (rawValue.IsInt()) { 734514f5e3Sopenharmony_ci int intValue = rawValue.GetInt(); 744514f5e3Sopenharmony_ci Barriers::SetPrimitive<JSTaggedType>(elements->GetData(), offset, 754514f5e3Sopenharmony_ci base::bit_cast<JSTaggedType>(static_cast<double>(intValue))); 764514f5e3Sopenharmony_ci } else { 774514f5e3Sopenharmony_ci Barriers::SetPrimitive<JSTaggedType>(elements->GetData(), offset, 784514f5e3Sopenharmony_ci base::bit_cast<JSTaggedType>(rawValue.GetDouble())); 794514f5e3Sopenharmony_ci } 804514f5e3Sopenharmony_ci break; 814514f5e3Sopenharmony_ci case ElementsKind::TAGGED: 824514f5e3Sopenharmony_ci if (value.GetTaggedValue().IsHeapObject()) { 834514f5e3Sopenharmony_ci Barriers::SetObject<true>(thread, elements->GetData(), offset, rawValue.GetRawData()); 844514f5e3Sopenharmony_ci } else { // NOLINTNEXTLINE(readability-misleading-indentation) 854514f5e3Sopenharmony_ci Barriers::SetPrimitive<JSTaggedType>(elements->GetData(), offset, rawValue.GetRawData()); 864514f5e3Sopenharmony_ci } 874514f5e3Sopenharmony_ci break; 884514f5e3Sopenharmony_ci default: 894514f5e3Sopenharmony_ci LOG_ECMA(FATAL) << "Trying to Convert TaggedValue With Unknown ElementsKind"; 904514f5e3Sopenharmony_ci UNREACHABLE(); 914514f5e3Sopenharmony_ci break; 924514f5e3Sopenharmony_ci } 934514f5e3Sopenharmony_ci} 944514f5e3Sopenharmony_ci} // namespace panda::ecmascript 954514f5e3Sopenharmony_ci#endif // ECMASCRIPT_ELEMENT_ACCESSOR_INL_H 96