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