13af6ab5fSopenharmony_ci/**
23af6ab5fSopenharmony_ci * Copyright (c) 2021-2024 Huawei Device Co., Ltd.
33af6ab5fSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
43af6ab5fSopenharmony_ci * you may not use this file except in compliance with the License.
53af6ab5fSopenharmony_ci * You may obtain a copy of the License at
63af6ab5fSopenharmony_ci *
73af6ab5fSopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0
83af6ab5fSopenharmony_ci *
93af6ab5fSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
103af6ab5fSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
113af6ab5fSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
123af6ab5fSopenharmony_ci * See the License for the specific language governing permissions and
133af6ab5fSopenharmony_ci * limitations under the License.
143af6ab5fSopenharmony_ci */
153af6ab5fSopenharmony_ci
163af6ab5fSopenharmony_ci#ifndef ES2PANDA_COMPILER_IR_IRNODE_H
173af6ab5fSopenharmony_ci#define ES2PANDA_COMPILER_IR_IRNODE_H
183af6ab5fSopenharmony_ci
193af6ab5fSopenharmony_ci#include "compiler/base/literals.h"
203af6ab5fSopenharmony_ci#include "compiler/core/vReg.h"
213af6ab5fSopenharmony_ci#include "compiler/core/programElement.h"
223af6ab5fSopenharmony_ci#include "lexer/token/sourceLocation.h"
233af6ab5fSopenharmony_ci#include "macros.h"
243af6ab5fSopenharmony_ci#include "util/ustring.h"
253af6ab5fSopenharmony_ci#include "utils/span.h"
263af6ab5fSopenharmony_ci
273af6ab5fSopenharmony_ci#include <cstdint>
283af6ab5fSopenharmony_ci#include <list>
293af6ab5fSopenharmony_ci#include <limits>
303af6ab5fSopenharmony_ci#include <sstream>
313af6ab5fSopenharmony_ci#include <utility>
323af6ab5fSopenharmony_ci#include <variant>
333af6ab5fSopenharmony_ci#include <vector>
343af6ab5fSopenharmony_ci
353af6ab5fSopenharmony_cinamespace ark::es2panda::ir {
363af6ab5fSopenharmony_ciclass AstNode;
373af6ab5fSopenharmony_ci}  // namespace ark::es2panda::ir
383af6ab5fSopenharmony_ci
393af6ab5fSopenharmony_cinamespace ark::es2panda::compiler {
403af6ab5fSopenharmony_cienum class OperandKind {
413af6ab5fSopenharmony_ci    // the least significant bit indicates vreg
423af6ab5fSopenharmony_ci    // the second bit indicates src or dst
433af6ab5fSopenharmony_ci    SRC_VREG,
443af6ab5fSopenharmony_ci    DST_VREG,
453af6ab5fSopenharmony_ci    SRC_DST_VREG,
463af6ab5fSopenharmony_ci    IMM,
473af6ab5fSopenharmony_ci    ID,
483af6ab5fSopenharmony_ci    STRING_ID,
493af6ab5fSopenharmony_ci    LABEL
503af6ab5fSopenharmony_ci};
513af6ab5fSopenharmony_ci
523af6ab5fSopenharmony_cienum class OperandType {
533af6ab5fSopenharmony_ci    REF,  // ref
543af6ab5fSopenharmony_ci    B32,  // u1 u2 i8 u8 i16 u16 i32 u32 b32 f32
553af6ab5fSopenharmony_ci    B64,  // i64, f64, b64
563af6ab5fSopenharmony_ci    ANY,  // any
573af6ab5fSopenharmony_ci    NONE
583af6ab5fSopenharmony_ci};
593af6ab5fSopenharmony_ci
603af6ab5fSopenharmony_cistruct OutVReg {
613af6ab5fSopenharmony_ci    const VReg *reg;
623af6ab5fSopenharmony_ci    OperandType type;
633af6ab5fSopenharmony_ci};
643af6ab5fSopenharmony_ci
653af6ab5fSopenharmony_ciclass FormatItem {
663af6ab5fSopenharmony_cipublic:
673af6ab5fSopenharmony_ci    constexpr FormatItem(OperandKind kind, uint32_t bitWidth) : kind_(kind), bitWidth_(bitWidth) {}
683af6ab5fSopenharmony_ci
693af6ab5fSopenharmony_ci    OperandKind Kind() const
703af6ab5fSopenharmony_ci    {
713af6ab5fSopenharmony_ci        return kind_;
723af6ab5fSopenharmony_ci    };
733af6ab5fSopenharmony_ci
743af6ab5fSopenharmony_ci    bool constexpr IsVReg() const
753af6ab5fSopenharmony_ci    {
763af6ab5fSopenharmony_ci        return kind_ == OperandKind::SRC_VREG || kind_ == OperandKind::DST_VREG || kind_ == OperandKind::SRC_DST_VREG;
773af6ab5fSopenharmony_ci    }
783af6ab5fSopenharmony_ci
793af6ab5fSopenharmony_ci    uint32_t BitWidth() const
803af6ab5fSopenharmony_ci    {
813af6ab5fSopenharmony_ci        return bitWidth_;
823af6ab5fSopenharmony_ci    };
833af6ab5fSopenharmony_ci
843af6ab5fSopenharmony_ciprivate:
853af6ab5fSopenharmony_ci    OperandKind kind_;
863af6ab5fSopenharmony_ci    uint32_t bitWidth_;
873af6ab5fSopenharmony_ci};
883af6ab5fSopenharmony_ci
893af6ab5fSopenharmony_ciclass Format {
903af6ab5fSopenharmony_cipublic:
913af6ab5fSopenharmony_ci    constexpr Format(const FormatItem *item, size_t size) : item_(item), size_(size) {}
923af6ab5fSopenharmony_ci
933af6ab5fSopenharmony_ci    ark::Span<const FormatItem> GetFormatItem() const
943af6ab5fSopenharmony_ci    {
953af6ab5fSopenharmony_ci        return ark::Span<const FormatItem>(item_, size_);
963af6ab5fSopenharmony_ci    }
973af6ab5fSopenharmony_ci
983af6ab5fSopenharmony_ciprivate:
993af6ab5fSopenharmony_ci    const FormatItem *item_;
1003af6ab5fSopenharmony_ci    size_t size_;
1013af6ab5fSopenharmony_ci};
1023af6ab5fSopenharmony_ci
1033af6ab5fSopenharmony_ciusing Formats = ark::Span<const Format>;
1043af6ab5fSopenharmony_ci
1053af6ab5fSopenharmony_ciclass Label;
1063af6ab5fSopenharmony_ciclass IRNode;
1073af6ab5fSopenharmony_ci
1083af6ab5fSopenharmony_ciusing Operand = std::variant<compiler::VReg, double, int64_t, util::StringView, Label *>;
1093af6ab5fSopenharmony_ci
1103af6ab5fSopenharmony_ci// NOLINTNEXTLINE(cppcoreguidelines-macro-usage)
1113af6ab5fSopenharmony_ci#define FIRST_NODE_OF_FUNCTION (reinterpret_cast<ir::AstNode *>(0x1))
1123af6ab5fSopenharmony_ci
1133af6ab5fSopenharmony_ciclass IRNode {
1143af6ab5fSopenharmony_cipublic:
1153af6ab5fSopenharmony_ci    explicit IRNode(const ir::AstNode *node) : node_(node) {};
1163af6ab5fSopenharmony_ci    virtual ~IRNode() = default;
1173af6ab5fSopenharmony_ci
1183af6ab5fSopenharmony_ci    NO_COPY_SEMANTIC(IRNode);
1193af6ab5fSopenharmony_ci    NO_MOVE_SEMANTIC(IRNode);
1203af6ab5fSopenharmony_ci
1213af6ab5fSopenharmony_ci    const ir::AstNode *Node() const
1223af6ab5fSopenharmony_ci    {
1233af6ab5fSopenharmony_ci        return node_;
1243af6ab5fSopenharmony_ci    }
1253af6ab5fSopenharmony_ci
1263af6ab5fSopenharmony_ci    static uint16_t MapRegister(uint32_t reg, uint32_t totalRegs)
1273af6ab5fSopenharmony_ci    {
1283af6ab5fSopenharmony_ci        ASSERT(reg != VReg::Invalid().GetIndex());
1293af6ab5fSopenharmony_ci
1303af6ab5fSopenharmony_ci        uint32_t regCount = VReg::REG_START - totalRegs;
1313af6ab5fSopenharmony_ci        uint16_t newReg = 0;
1323af6ab5fSopenharmony_ci
1333af6ab5fSopenharmony_ci        if (reg >= VReg::PARAM_START) {
1343af6ab5fSopenharmony_ci            newReg = reg - VReg::PARAM_START + regCount;
1353af6ab5fSopenharmony_ci            // NOTE: dbatiz. Remove this else if, and fix the regIndexes
1363af6ab5fSopenharmony_ci        } else if (reg <= regCount + VReg::MANDATORY_PARAM_NUM) {
1373af6ab5fSopenharmony_ci            newReg = VReg::REG_START - totalRegs + VReg::MANDATORY_PARAM_NUM + reg;
1383af6ab5fSopenharmony_ci        } else {
1393af6ab5fSopenharmony_ci            uint32_t regOffset = reg - totalRegs;
1403af6ab5fSopenharmony_ci            newReg = std::abs(static_cast<int32_t>(regOffset - regCount));
1413af6ab5fSopenharmony_ci        }
1423af6ab5fSopenharmony_ci
1433af6ab5fSopenharmony_ci        return newReg;
1443af6ab5fSopenharmony_ci    }
1453af6ab5fSopenharmony_ci
1463af6ab5fSopenharmony_ci    static constexpr auto MAX_REG_OPERAND = 5;
1473af6ab5fSopenharmony_ci
1483af6ab5fSopenharmony_ci    virtual Formats GetFormats() const = 0;
1493af6ab5fSopenharmony_ci    virtual size_t Registers([[maybe_unused]] std::array<VReg *, MAX_REG_OPERAND> *regs) = 0;
1503af6ab5fSopenharmony_ci    virtual size_t Registers([[maybe_unused]] std::array<const VReg *, MAX_REG_OPERAND> *regs) const = 0;
1513af6ab5fSopenharmony_ci    virtual size_t OutRegisters([[maybe_unused]] std::array<OutVReg, MAX_REG_OPERAND> *regs) const = 0;
1523af6ab5fSopenharmony_ci    virtual void Transform(ark::pandasm::Ins *ins, [[maybe_unused]] ProgramElement *programElement,
1533af6ab5fSopenharmony_ci                           [[maybe_unused]] uint32_t totalRegs) const = 0;
1543af6ab5fSopenharmony_ci
1553af6ab5fSopenharmony_ciprivate:
1563af6ab5fSopenharmony_ci    const ir::AstNode *node_;
1573af6ab5fSopenharmony_ci};
1583af6ab5fSopenharmony_ci}  // namespace ark::es2panda::compiler
1593af6ab5fSopenharmony_ci
1603af6ab5fSopenharmony_ci#endif
161