arkcompiler_ets_runtime/ecmascript/jspandafile/method_literal.h

320 lines
10 KiB
C
Raw Normal View History

/*
* 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.
*/
#ifndef ECMASCRIPT_METHOD_LITERAL_H
#define ECMASCRIPT_METHOD_LITERAL_H
#include "ecmascript/base/aligned_struct.h"
#include "ecmascript/js_function_kind.h"
#include "ecmascript/js_tagged_value.h"
#include "ecmascript/mem/c_string.h"
#include "libpandafile/file.h"
static constexpr uint32_t CALL_TYPE_MASK = 0xF; // 0xF: the last 4 bits are used as callType
namespace panda::ecmascript {
class JSPandaFile;
using EntityId = panda_file::File::EntityId;
struct PUBLIC_API MethodLiteral : public base::AlignedStruct<sizeof(uint64_t),
base::AlignedUint64,
base::AlignedPointer,
base::AlignedUint64,
base::AlignedUint64> {
enum class Index : size_t {
CALL_FIELD_INDEX = 0,
NATIVE_POINTER_OR_BYTECODE_ARRAY_INDEX,
LITERAL_INFO_INDEX,
EXTRA_LITERAL_INFO_INDEX,
NUM_OF_MEMBERS
};
static_assert(static_cast<size_t>(Index::NUM_OF_MEMBERS) == NumOfTypes);
static constexpr uint8_t INVALID_IC_SLOT = 0xFFU;
static constexpr uint16_t MAX_SLOT_SIZE = 0xFFFFU;
MethodLiteral(const JSPandaFile *jsPandaFile, EntityId fileId);
MethodLiteral() = delete;
~MethodLiteral() = default;
MethodLiteral(const MethodLiteral &) = delete;
MethodLiteral(MethodLiteral &&) = delete;
MethodLiteral &operator=(const MethodLiteral &) = delete;
MethodLiteral &operator=(MethodLiteral &&) = delete;
static constexpr size_t VREGS_ARGS_NUM_BITS = 28; // 28: maximum 268,435,455
using HaveThisBit = BitField<bool, 0, 1>; // offset 0
using HaveNewTargetBit = HaveThisBit::NextFlag; // offset 1
using HaveExtraBit = HaveNewTargetBit::NextFlag; // offset 2
using HaveFuncBit = HaveExtraBit::NextFlag; // offset 3
using NumVregsBits = HaveFuncBit::NextField<uint32_t, VREGS_ARGS_NUM_BITS>; // offset 4-31
using NumArgsBits = NumVregsBits::NextField<uint32_t, VREGS_ARGS_NUM_BITS>; // offset 32-59
using IsNativeBit = NumArgsBits::NextFlag; // offset 60
using IsAotCodeBit = IsNativeBit::NextFlag; // offset 61
using IsFastBuiltinBit = IsAotCodeBit::NextFlag; // offset 62
uint64_t GetCallField() const
{
return callField_;
}
void SetNativeBit(bool isNative)
{
callField_ = IsNativeBit::Update(callField_, isNative);
}
void SetAotCodeBit(bool isCompiled)
{
callField_ = IsAotCodeBit::Update(callField_, isCompiled);
}
void InitializeCallField(const JSPandaFile *jsPandaFile, uint32_t numVregs, uint32_t numArgs);
bool HaveThisWithCallField() const
{
return HaveThisBit::Decode(callField_);
}
bool HaveNewTargetWithCallField() const
{
return HaveNewTargetBit::Decode(callField_);
}
bool HaveExtraWithCallField() const
{
return HaveExtraBit::Decode(callField_);
}
bool HaveFuncWithCallField() const
{
return HaveFuncBit::Decode(callField_);
}
bool IsNativeWithCallField() const
{
return IsNativeBit::Decode(callField_);
}
bool IsAotWithCallField() const
{
return IsAotCodeBit::Decode(callField_);
}
uint32_t GetNumArgsWithCallField() const
{
return NumArgsBits::Decode(callField_);
}
uint32_t GetNumArgs() const
{
return GetNumArgsWithCallField() + HaveFuncWithCallField() +
HaveNewTargetWithCallField() + HaveThisWithCallField();
}
static uint64_t SetNumArgsWithCallField(uint64_t callField, uint32_t numargs)
{
return NumArgsBits::Update(callField, numargs);
}
static uint64_t SetNativeBit(uint64_t callField, bool isNative)
{
return IsNativeBit::Update(callField, isNative);
}
static uint64_t SetAotCodeBit(uint64_t callField, bool isCompiled)
{
return IsAotCodeBit::Update(callField, isCompiled);
}
static uint64_t SetFastBuiltinBit(uint64_t callField, bool isFastBuiltin)
{
return IsFastBuiltinBit::Update(callField, isFastBuiltin);
}
static bool HaveThisWithCallField(uint64_t callField)
{
return HaveThisBit::Decode(callField);
}
static bool HaveNewTargetWithCallField(uint64_t callField)
{
return HaveNewTargetBit::Decode(callField);
}
static bool HaveExtraWithCallField(uint64_t callField)
{
return HaveExtraBit::Decode(callField);
}
static bool HaveFuncWithCallField(uint64_t callField)
{
return HaveFuncBit::Decode(callField);
}
static bool IsNativeWithCallField(uint64_t callField)
{
return IsNativeBit::Decode(callField);
}
static bool IsAotWithCallField(uint64_t callField)
{
return IsAotCodeBit::Decode(callField);
}
static bool OnlyHaveThisWithCallField(uint64_t callField)
{
return (callField & CALL_TYPE_MASK) == 1; // 1: the first bit of callFiled is HaveThisBit
}
static bool OnlyHaveNewTagetAndThisWithCallField(uint64_t callField)
{
return (callField & CALL_TYPE_MASK) == 0b11; // the first two bit of callFiled is `This` and `NewTarget`
}
static uint32_t GetNumVregsWithCallField(uint64_t callField)
{
return NumVregsBits::Decode(callField);
}
static uint32_t GetNumArgsWithCallField(uint64_t callField)
{
return NumArgsBits::Decode(callField);
}
static constexpr size_t METHOD_ARGS_NUM_BITS = 16;
static constexpr size_t METHOD_ARGS_METHODID_BITS = 32;
static constexpr size_t METHOD_SLOT_SIZE_BITS = 16;
using HotnessCounterBits = BitField<int16_t, 0, METHOD_ARGS_NUM_BITS>; // offset 0-15
using MethodIdBits = HotnessCounterBits::NextField<uint32_t, METHOD_ARGS_METHODID_BITS>; // offset 16-47
using SlotSizeBits = MethodIdBits::NextField<uint16_t, METHOD_SLOT_SIZE_BITS>; // offset 48-63
static constexpr size_t BUILTINID_NUM_BITS = 8;
static constexpr size_t FUNCTION_KIND_NUM_BITS = 4;
using BuiltinIdBits = BitField<uint8_t, 0, BUILTINID_NUM_BITS>; // offset 0-7
using FunctionKindBits = BuiltinIdBits::NextField<FunctionKind, FUNCTION_KIND_NUM_BITS>; // offset 8-11
inline NO_THREAD_SANITIZE void SetHotnessCounter(int16_t counter)
{
literalInfo_ = HotnessCounterBits::Update(literalInfo_, counter);
}
EntityId GetMethodId() const
{
return EntityId(MethodIdBits::Decode(literalInfo_));
}
void SetMethodId(EntityId methodId)
{
literalInfo_ = MethodIdBits::Update(literalInfo_, methodId.GetOffset());
}
uint16_t GetSlotSize() const
{
return SlotSizeBits::Decode(literalInfo_);
}
uint8_t UpdateSlotSizeWith8Bit(uint16_t size)
{
uint16_t start = GetSlotSize();
uint32_t end = start + size;
// ic overflow
if (end >= INVALID_IC_SLOT) {
if (GetSlotSize() < INVALID_IC_SLOT + 1) {
literalInfo_ = SlotSizeBits::Update(literalInfo_, INVALID_IC_SLOT + 1);
}
return INVALID_IC_SLOT;
}
literalInfo_ = SlotSizeBits::Update(literalInfo_, static_cast<uint8_t>(end));
return start;
}
void SetFunctionKind(FunctionKind kind)
{
extraLiteralInfo_ = FunctionKindBits::Update(extraLiteralInfo_, kind);
}
FunctionKind GetFunctionKind() const
{
return static_cast<FunctionKind>(FunctionKindBits::Decode(extraLiteralInfo_));
}
static inline int16_t GetHotnessCounter(uint64_t literalInfo)
{
return HotnessCounterBits::Decode(literalInfo);
}
static uint64_t SetHotnessCounter(uint64_t literalInfo, int16_t counter)
{
return HotnessCounterBits::Update(literalInfo, counter);
}
static EntityId GetMethodId(uint64_t literalInfo)
{
return EntityId(MethodIdBits::Decode(literalInfo));
}
static uint8_t GetSlotSize(uint64_t literalInfo)
{
return SlotSizeBits::Decode(literalInfo);
}
static uint8_t GetBuiltinId(uint64_t literalInfo)
{
return BuiltinIdBits::Decode(literalInfo);
}
static uint64_t SetBuiltinId(uint64_t literalInfo, uint8_t id)
{
return BuiltinIdBits::Update(literalInfo, id);
}
static uint32_t PUBLIC_API GetNumVregs(const JSPandaFile *jsPandaFile, const MethodLiteral *methodLiteral);
static panda_file::File::StringData GetName(const JSPandaFile *jsPandaFile, EntityId methodId);
static const char * PUBLIC_API GetMethodName(const JSPandaFile *jsPandaFile, EntityId methodId);
static std::string PUBLIC_API ParseFunctionName(const JSPandaFile *jsPandaFile, EntityId methodId);
static uint32_t GetCodeSize(const JSPandaFile *jsPandaFile, EntityId methodId);
const uint8_t *GetBytecodeArray() const
{
return reinterpret_cast<const uint8_t *>(nativePointerOrBytecodeArray_);
}
const void* GetNativePointer() const
{
return nativePointerOrBytecodeArray_;
}
uint64_t GetLiteralInfo() const
{
return literalInfo_;
}
uint64_t GetExtraLiteralInfo() const
{
return extraLiteralInfo_;
}
alignas(EAS) uint64_t callField_ {0ULL};
// Native method decides this filed is NativePointer or BytecodeArray pointer.
alignas(EAS) const void *nativePointerOrBytecodeArray_ {nullptr};
// hotnessCounter, methodId and slotSize are encoded in literalInfo_.
alignas(EAS) uint64_t literalInfo_ {0ULL};
// BuiltinId, FunctionKind are encoded in extraLiteralInfo_.
alignas(EAS) uint64_t extraLiteralInfo_ {0ULL};
};
STATIC_ASSERT_EQ_ARCH(sizeof(MethodLiteral), MethodLiteral::SizeArch32, MethodLiteral::SizeArch64);
} // namespace panda::ecmascript
#endif // ECMASCRIPT_METHOD_LITERAL_H