/* * 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_PROPERTY_ATTRIBUTES_H #define ECMASCRIPT_PROPERTY_ATTRIBUTES_H #include "ecmascript/js_tagged_value.h" #include "utils/bit_field.h" namespace panda::ecmascript { class PropertyDescriptor; enum class Representation { NONE, INT, DOUBLE, NUMBER, OBJECT, MIXED, }; enum class PropertyBoxType { // Meaningful when a property cell does not contain the hole. UNDEFINED, // The PREMONOMORPHIC of property cells. CONSTANT, // Cell has been assigned only once. CONSTANTTYPE, // Cell has been assigned only one type. MUTABLE, // Cell will no longer be tracked as constant. // Meaningful when a property cell contains the hole. UNINITIALIZED = UNDEFINED, // Cell has never been initialized. INVALIDATED = CONSTANT, // Cell has been deleted, invalidated or never existed. }; class PropertyAttributes { public: explicit PropertyAttributes() = default; ~PropertyAttributes() = default; DEFAULT_NOEXCEPT_MOVE_SEMANTIC(PropertyAttributes); DEFAULT_COPY_SEMANTIC(PropertyAttributes); explicit PropertyAttributes(uint32_t v) : value_(v) {} explicit PropertyAttributes(int32_t v) : value_(static_cast(v)) {} explicit PropertyAttributes(JSTaggedValue v) : value_(v.GetInt()) {} explicit PropertyAttributes(const PropertyDescriptor &desc); static constexpr uint32_t DICTIONARY_ORDER_NUM = 20; static constexpr uint32_t OFFSET_BITFIELD_NUM = 10; static constexpr uint32_t MAX_CAPACITY_OF_PROPERTIES = (1U << OFFSET_BITFIELD_NUM) - 1; using PropertyMetaDataField = BitField; // 4: property metaData field occupies 4 bits using AttributesField = BitField; // 4: attributes field occupies 4 bits using DefaultAttributesField = BitField; // 3: default attributes field occupies 3 bits using WritableField = BitField; // 1: writable field occupies 1 bits using EnumerableField = WritableField::NextFlag; using ConfigurableField = EnumerableField::NextFlag; using IsAccessorField = ConfigurableField::NextFlag; // 4 // fast mode using IsInlinedPropsField = PropertyMetaDataField::NextFlag; // 5 using RepresentationField = IsInlinedPropsField::NextField; // 3: 3 bits, 6-8 using OffsetField = RepresentationField::NextField; // 18 static constexpr uint32_t NORMAL_ATTR_BITS = 18; using NormalAttrField = BitField; using SortedIndexField = OffsetField::NextField; // 28 using IsConstPropsField = SortedIndexField::NextFlag; // 29 // dictionary mode, include global using PropertyBoxTypeField = PropertyMetaDataField::NextField; // 2: 2 bits, 5-6 using DictionaryOrderField = PropertyBoxTypeField::NextField; // 26 static constexpr uint32_t BIT_SIZE = 28; static constexpr int INTIAL_PROPERTY_INDEX = 0; inline int GetPropertyMetaData() const { return PropertyMetaDataField::Get(value_); } static PropertyAttributes Default() { return PropertyAttributes(GetDefaultAttributes()); } static PropertyAttributes Default(bool w, bool e, bool c, bool isAccessor = false) { uint32_t value = WritableField::Encode(w) | EnumerableField::Encode(e) | ConfigurableField::Encode(c) | IsAccessorField::Encode(isAccessor); return PropertyAttributes(value); } static PropertyAttributes DefaultAccessor(bool w, bool e, bool c) { uint32_t value = WritableField::Encode(w) | EnumerableField::Encode(e) | ConfigurableField::Encode(c) | IsAccessorField::Encode(true); return PropertyAttributes(value); } inline void SetDefaultAttributes() { AttributesField::Set(DefaultAttributesField::Mask(), &value_); } static inline int GetDefaultAttributes() { return DefaultAttributesField::Mask(); } static inline Representation TaggedToRepresentation(JSTaggedValue value) { if (value.IsInt()) { return Representation::INT; } if (value.IsDouble()) { return Representation::DOUBLE; } return Representation::OBJECT; } static Representation UpdateRepresentation(Representation oldRep, JSTaggedValue value) { if (oldRep == Representation::MIXED) { return oldRep; } Representation newRep = TaggedToRepresentation(value); if (oldRep == Representation::NONE) { return newRep; } if (oldRep == newRep) { return oldRep; } switch (oldRep) { case Representation::INT: case Representation::DOUBLE: case Representation::NUMBER: if (newRep != Representation::OBJECT) { return Representation::NUMBER; } return Representation::MIXED; case Representation::OBJECT: return Representation::MIXED; default: UNREACHABLE(); } } inline bool IsDefaultAttributes() const { return AttributesField::Get(value_) == static_cast(DefaultAttributesField::Mask()); } inline void SetNoneAttributes() { AttributesField::Set(0U, &value_); } inline bool IsNoneAttributes() const { return AttributesField::Get(value_) == 0; } inline void SetWritable(bool flag) { WritableField::Set(flag, &value_); } inline bool IsWritable() const { return WritableField::Get(value_); } inline void SetEnumerable(bool flag) { EnumerableField::Set(flag, &value_); } inline bool IsEnumerable() const { return EnumerableField::Get(value_); } inline void SetConfigurable(bool flag) { ConfigurableField::Set(flag, &value_); } inline bool IsConfigurable() const { return ConfigurableField::Get(value_); } inline void SetIsAccessor(bool flag) { IsAccessorField::Set(flag, &value_); } inline bool IsAccessor() const { return IsAccessorField::Get(value_); } inline void SetIsInlinedProps(bool flag) { IsInlinedPropsField::Set(flag, &value_); } inline bool IsInlinedProps() const { return IsInlinedPropsField::Get(value_); } inline void SetIsConstProps(bool flag) { IsConstPropsField::Set(flag, &value_); } inline bool IsConstProps() const { return IsConstPropsField::Get(value_); } inline void SetRepresentation(Representation representation) { RepresentationField::Set(representation, &value_); } inline Representation GetRepresentation() const { return RepresentationField::Get(value_); } inline void SetDictionaryOrder(uint32_t order) { DictionaryOrderField::Set(order, &value_); } inline uint32_t GetDictionaryOrder() const { return DictionaryOrderField::Get(value_); } inline void SetOffset(uint32_t offset) { OffsetField::Set(offset, &value_); } inline uint32_t GetOffset() const { return OffsetField::Get(value_); } inline void SetSortedIndex(uint32_t sortedIndex) { SortedIndexField::Set(sortedIndex, &value_); } inline uint32_t GetSortedIndex() const { return SortedIndexField::Get(value_); } inline void SetNormalAttr(uint32_t normalAttr) { NormalAttrField::Set(normalAttr, &value_); } inline uint32_t GetNormalAttr() const { return NormalAttrField::Get(value_); } inline JSTaggedValue GetNormalTagged() const { return JSTaggedValue(static_cast(GetNormalAttr())); } inline uint32_t GetValue() const { return value_; } inline void SetBoxType(PropertyBoxType cellType) { PropertyBoxTypeField::Set(cellType, &value_); } inline PropertyBoxType GetBoxType() const { return PropertyBoxTypeField::Get(value_); } inline static bool IsValidIndex(int index) { return DictionaryOrderField::IsValid(index); } inline JSTaggedValue GetTaggedValue() const { return JSTaggedValue(static_cast(value_)); } private: uint32_t value_{0}; }; } // namespace panda::ecmascript #endif // ECMASCRIPT_PROPERTY_ATTRIBUTES_H