/* * 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. */ #ifndef ECMASCRIPT_ACCESSOR_DATA_H #define ECMASCRIPT_ACCESSOR_DATA_H #include "ecmascript/ecma_macros.h" #include "ecmascript/js_hclass.h" #include "ecmascript/js_native_pointer.h" #include "ecmascript/js_object.h" #include "ecmascript/js_tagged_value.h" #include "ecmascript/mem/slots.h" #include "ecmascript/mem/visitor.h" #include "ecmascript/record.h" namespace panda::ecmascript { class InternalAccessor final : public Record { public: using InternalGetFunc = JSTaggedValue (*)(JSThread *, const JSHandle &); using InternalSetFunc = bool (*)(JSThread *, const JSHandle &, const JSHandle &, bool); CAST_CHECK(InternalAccessor, IsInternalAccessor); inline bool HasSetter() const { return GetSetter() != nullptr; } static constexpr size_t GETTER_OFFSET = Record::SIZE; ACCESSORS_FIXED_SIZE_FIELD(Getter, InternalGetFunc, JSTaggedType, GETTER_OFFSET, SETTER_OFFSET) ACCESSORS_FIXED_SIZE_FIELD(Setter, InternalSetFunc, JSTaggedType, SETTER_OFFSET, SIZE) DECL_VISIT_NATIVE_FIELD(GETTER_OFFSET, SIZE) }; class AccessorData final : public Record { public: static AccessorData *Cast(TaggedObject *object) { ASSERT(JSTaggedValue(object).IsAccessorData() || JSTaggedValue(object).IsInternalAccessor()); return static_cast(object); } inline bool IsInternal() const { return GetClass()->IsInternalAccessor(); } inline bool HasSetter(const JSThread *thread) const { auto setter = GetSetter(thread); // When the raw data is 0, means the InternalAccessor's setter is nullptr. return !(setter.IsUndefined() || setter.GetRawData() == 0U); } JSTaggedValue CallInternalGet(JSThread *thread, const JSHandle &obj) const { ASSERT(IsInternal()); auto getFunc = InternalAccessor::ConstCast(this)->GetGetter(); return getFunc(thread, obj); } bool CallInternalSet(JSThread *thread, const JSHandle &obj, const JSHandle &value, bool mayThrow = false) const { ASSERT(IsInternal()); auto setFunc = InternalAccessor::ConstCast(this)->GetSetter(); return setFunc(thread, obj, value, mayThrow); } static constexpr size_t GETTER_OFFSET = Record::SIZE; ACCESSORS(Getter, GETTER_OFFSET, SETTER_OFFSET); ACCESSORS(Setter, SETTER_OFFSET, SIZE); DECL_DUMP() DECL_VISIT_OBJECT(GETTER_OFFSET, SIZE) }; static_assert(AccessorData::SIZE == InternalAccessor::SIZE && AccessorData::GETTER_OFFSET == InternalAccessor::GETTER_OFFSET && AccessorData::SETTER_OFFSET == InternalAccessor::SETTER_OFFSET); enum class CompletionRecordType : uint8_t { NORMAL = 0U, BREAK, CONTINUE, RETURN, THROW }; class CompletionRecord final : public Record { public: static CompletionRecord *Cast(TaggedObject *object) { ASSERT(JSTaggedValue(object).IsCompletionRecord()); return static_cast(object); } bool IsThrow() const { return GetType() == CompletionRecordType::THROW; } static constexpr size_t VALUE_OFFSET = Record::SIZE; ACCESSORS(Value, VALUE_OFFSET, BIT_FIELD_OFFSET) ACCESSORS_BIT_FIELD(BitField, BIT_FIELD_OFFSET, LAST_OFFSET) DEFINE_ALIGN_SIZE(LAST_OFFSET); // define BitField static constexpr size_t TYPE_BITS = 3; FIRST_BIT_FIELD(BitField, Type, CompletionRecordType, TYPE_BITS) DECL_DUMP() DECL_VISIT_OBJECT(VALUE_OFFSET, BIT_FIELD_OFFSET) }; } // namespace panda::ecmascript #endif // ECMASCRIPT_ACCESSOR_DATA_H