//===-- VE.h - Top-level interface for VE representation --------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // This file contains the entry points for global functions defined in the LLVM // VE back-end. // //===----------------------------------------------------------------------===// #ifndef LLVM_LIB_TARGET_VE_VE_H #define LLVM_LIB_TARGET_VE_VE_H #include "MCTargetDesc/VEMCTargetDesc.h" #include "llvm/ADT/StringSwitch.h" #include "llvm/CodeGen/ISDOpcodes.h" #include "llvm/CodeGen/SelectionDAGNodes.h" #include "llvm/Support/ErrorHandling.h" #include "llvm/Target/TargetMachine.h" namespace llvm { class AsmPrinter; class FunctionPass; class MCInst; class MachineInstr; class PassRegistry; class VETargetMachine; FunctionPass *createVEISelDag(VETargetMachine &TM); FunctionPass *createLVLGenPass(); void initializeVEDAGToDAGISelPass(PassRegistry &); void LowerVEMachineInstrToMCInst(const MachineInstr *MI, MCInst &OutMI, AsmPrinter &AP); } // namespace llvm namespace llvm { // Enums corresponding to VE condition codes, both icc's and fcc's. These // values must be kept in sync with the ones in the .td file. namespace VECC { enum CondCode { // Integer comparison CC_IG = 0, // Greater CC_IL = 1, // Less CC_INE = 2, // Not Equal CC_IEQ = 3, // Equal CC_IGE = 4, // Greater or Equal CC_ILE = 5, // Less or Equal // Floating point comparison CC_AF = 0 + 6, // Never CC_G = 1 + 6, // Greater CC_L = 2 + 6, // Less CC_NE = 3 + 6, // Not Equal CC_EQ = 4 + 6, // Equal CC_GE = 5 + 6, // Greater or Equal CC_LE = 6 + 6, // Less or Equal CC_NUM = 7 + 6, // Number CC_NAN = 8 + 6, // NaN CC_GNAN = 9 + 6, // Greater or NaN CC_LNAN = 10 + 6, // Less or NaN CC_NENAN = 11 + 6, // Not Equal or NaN CC_EQNAN = 12 + 6, // Equal or NaN CC_GENAN = 13 + 6, // Greater or Equal or NaN CC_LENAN = 14 + 6, // Less or Equal or NaN CC_AT = 15 + 6, // Always UNKNOWN }; } // Enums corresponding to VE Rounding Mode. These values must be kept in // sync with the ones in the .td file. namespace VERD { enum RoundingMode { RD_NONE = 0, // According to PSW RD_RZ = 8, // Round toward Zero RD_RP = 9, // Round toward Plus infinity RD_RM = 10, // Round toward Minus infinity RD_RN = 11, // Round to Nearest (ties to Even) RD_RA = 12, // Round to Nearest (ties to Away) UNKNOWN }; } inline static const char *VECondCodeToString(VECC::CondCode CC) { switch (CC) { case VECC::CC_IG: return "gt"; case VECC::CC_IL: return "lt"; case VECC::CC_INE: return "ne"; case VECC::CC_IEQ: return "eq"; case VECC::CC_IGE: return "ge"; case VECC::CC_ILE: return "le"; case VECC::CC_AF: return "af"; case VECC::CC_G: return "gt"; case VECC::CC_L: return "lt"; case VECC::CC_NE: return "ne"; case VECC::CC_EQ: return "eq"; case VECC::CC_GE: return "ge"; case VECC::CC_LE: return "le"; case VECC::CC_NUM: return "num"; case VECC::CC_NAN: return "nan"; case VECC::CC_GNAN: return "gtnan"; case VECC::CC_LNAN: return "ltnan"; case VECC::CC_NENAN: return "nenan"; case VECC::CC_EQNAN: return "eqnan"; case VECC::CC_GENAN: return "genan"; case VECC::CC_LENAN: return "lenan"; case VECC::CC_AT: return "at"; default: llvm_unreachable("Invalid cond code"); } } inline static VECC::CondCode stringToVEICondCode(StringRef S) { return StringSwitch(S) .Case("gt", VECC::CC_IG) .Case("lt", VECC::CC_IL) .Case("ne", VECC::CC_INE) .Case("eq", VECC::CC_IEQ) .Case("ge", VECC::CC_IGE) .Case("le", VECC::CC_ILE) .Case("af", VECC::CC_AF) .Case("at", VECC::CC_AT) .Case("", VECC::CC_AT) .Default(VECC::UNKNOWN); } inline static VECC::CondCode stringToVEFCondCode(StringRef S) { return StringSwitch(S) .Case("gt", VECC::CC_G) .Case("lt", VECC::CC_L) .Case("ne", VECC::CC_NE) .Case("eq", VECC::CC_EQ) .Case("ge", VECC::CC_GE) .Case("le", VECC::CC_LE) .Case("num", VECC::CC_NUM) .Case("nan", VECC::CC_NAN) .Case("gtnan", VECC::CC_GNAN) .Case("ltnan", VECC::CC_LNAN) .Case("nenan", VECC::CC_NENAN) .Case("eqnan", VECC::CC_EQNAN) .Case("genan", VECC::CC_GENAN) .Case("lenan", VECC::CC_LENAN) .Case("af", VECC::CC_AF) .Case("at", VECC::CC_AT) .Case("", VECC::CC_AT) .Default(VECC::UNKNOWN); } inline static bool isIntVECondCode(VECC::CondCode CC) { return CC < VECC::CC_AF; } inline static unsigned VECondCodeToVal(VECC::CondCode CC) { switch (CC) { case VECC::CC_IG: return 1; case VECC::CC_IL: return 2; case VECC::CC_INE: return 3; case VECC::CC_IEQ: return 4; case VECC::CC_IGE: return 5; case VECC::CC_ILE: return 6; case VECC::CC_AF: return 0; case VECC::CC_G: return 1; case VECC::CC_L: return 2; case VECC::CC_NE: return 3; case VECC::CC_EQ: return 4; case VECC::CC_GE: return 5; case VECC::CC_LE: return 6; case VECC::CC_NUM: return 7; case VECC::CC_NAN: return 8; case VECC::CC_GNAN: return 9; case VECC::CC_LNAN: return 10; case VECC::CC_NENAN: return 11; case VECC::CC_EQNAN: return 12; case VECC::CC_GENAN: return 13; case VECC::CC_LENAN: return 14; case VECC::CC_AT: return 15; default: llvm_unreachable("Invalid cond code"); } } /// Convert a DAG integer condition code to a VE ICC condition. inline static VECC::CondCode intCondCode2Icc(ISD::CondCode CC) { switch (CC) { default: llvm_unreachable("Unknown integer condition code!"); case ISD::SETEQ: return VECC::CC_IEQ; case ISD::SETNE: return VECC::CC_INE; case ISD::SETLT: return VECC::CC_IL; case ISD::SETGT: return VECC::CC_IG; case ISD::SETLE: return VECC::CC_ILE; case ISD::SETGE: return VECC::CC_IGE; case ISD::SETULT: return VECC::CC_IL; case ISD::SETULE: return VECC::CC_ILE; case ISD::SETUGT: return VECC::CC_IG; case ISD::SETUGE: return VECC::CC_IGE; } } /// Convert a DAG floating point condition code to a VE FCC condition. inline static VECC::CondCode fpCondCode2Fcc(ISD::CondCode CC) { switch (CC) { default: llvm_unreachable("Unknown fp condition code!"); case ISD::SETFALSE: return VECC::CC_AF; case ISD::SETEQ: case ISD::SETOEQ: return VECC::CC_EQ; case ISD::SETNE: case ISD::SETONE: return VECC::CC_NE; case ISD::SETLT: case ISD::SETOLT: return VECC::CC_L; case ISD::SETGT: case ISD::SETOGT: return VECC::CC_G; case ISD::SETLE: case ISD::SETOLE: return VECC::CC_LE; case ISD::SETGE: case ISD::SETOGE: return VECC::CC_GE; case ISD::SETO: return VECC::CC_NUM; case ISD::SETUO: return VECC::CC_NAN; case ISD::SETUEQ: return VECC::CC_EQNAN; case ISD::SETUNE: return VECC::CC_NENAN; case ISD::SETULT: return VECC::CC_LNAN; case ISD::SETUGT: return VECC::CC_GNAN; case ISD::SETULE: return VECC::CC_LENAN; case ISD::SETUGE: return VECC::CC_GENAN; case ISD::SETTRUE: return VECC::CC_AT; } } inline static VECC::CondCode VEValToCondCode(unsigned Val, bool IsInteger) { if (IsInteger) { switch (Val) { case 0: return VECC::CC_AF; case 1: return VECC::CC_IG; case 2: return VECC::CC_IL; case 3: return VECC::CC_INE; case 4: return VECC::CC_IEQ; case 5: return VECC::CC_IGE; case 6: return VECC::CC_ILE; case 15: return VECC::CC_AT; } } else { switch (Val) { case 0: return VECC::CC_AF; case 1: return VECC::CC_G; case 2: return VECC::CC_L; case 3: return VECC::CC_NE; case 4: return VECC::CC_EQ; case 5: return VECC::CC_GE; case 6: return VECC::CC_LE; case 7: return VECC::CC_NUM; case 8: return VECC::CC_NAN; case 9: return VECC::CC_GNAN; case 10: return VECC::CC_LNAN; case 11: return VECC::CC_NENAN; case 12: return VECC::CC_EQNAN; case 13: return VECC::CC_GENAN; case 14: return VECC::CC_LENAN; case 15: return VECC::CC_AT; } } llvm_unreachable("Invalid cond code"); } inline static const char *VERDToString(VERD::RoundingMode R) { switch (R) { case VERD::RD_NONE: return ""; case VERD::RD_RZ: return ".rz"; case VERD::RD_RP: return ".rp"; case VERD::RD_RM: return ".rm"; case VERD::RD_RN: return ".rn"; case VERD::RD_RA: return ".ra"; default: llvm_unreachable("Invalid branch predicate"); } } inline static VERD::RoundingMode stringToVERD(StringRef S) { return StringSwitch(S) .Case("", VERD::RD_NONE) .Case(".rz", VERD::RD_RZ) .Case(".rp", VERD::RD_RP) .Case(".rm", VERD::RD_RM) .Case(".rn", VERD::RD_RN) .Case(".ra", VERD::RD_RA) .Default(VERD::UNKNOWN); } inline static unsigned VERDToVal(VERD::RoundingMode R) { switch (R) { case VERD::RD_NONE: case VERD::RD_RZ: case VERD::RD_RP: case VERD::RD_RM: case VERD::RD_RN: case VERD::RD_RA: return static_cast(R); default: break; } llvm_unreachable("Invalid branch predicates"); } inline static VERD::RoundingMode VEValToRD(unsigned Val) { switch (Val) { case static_cast(VERD::RD_NONE): return VERD::RD_NONE; case static_cast(VERD::RD_RZ): return VERD::RD_RZ; case static_cast(VERD::RD_RP): return VERD::RD_RP; case static_cast(VERD::RD_RM): return VERD::RD_RM; case static_cast(VERD::RD_RN): return VERD::RD_RN; case static_cast(VERD::RD_RA): return VERD::RD_RA; default: break; } llvm_unreachable("Invalid branch predicates"); } /// getImmVal - get immediate representation of integer value inline static uint64_t getImmVal(const ConstantSDNode *N) { return N->getSExtValue(); } /// getFpImmVal - get immediate representation of floating point value inline static uint64_t getFpImmVal(const ConstantFPSDNode *N) { const APInt &Imm = N->getValueAPF().bitcastToAPInt(); uint64_t Val = Imm.getZExtValue(); if (Imm.getBitWidth() == 32) { // Immediate value of float place places at higher bits on VE. Val <<= 32; } return Val; } // MImm - Special immediate value of sequential bit stream of 0 or 1. // See VEInstrInfo.td for details. inline static bool isMImmVal(uint64_t Val) { if (Val == 0) { // (0)1 is 0 return true; } if (isMask_64(Val)) { // (m)0 patterns return true; } // (m)1 patterns return (Val & (UINT64_C(1) << 63)) && isShiftedMask_64(Val); } inline static bool isMImm32Val(uint32_t Val) { if (Val == 0) { // (0)1 is 0 return true; } if (isMask_32(Val)) { // (m)0 patterns return true; } // (m)1 patterns return (Val & (UINT32_C(1) << 31)) && isShiftedMask_32(Val); } /// val2MImm - Convert an integer immediate value to target MImm immediate. inline static uint64_t val2MImm(uint64_t Val) { if (Val == 0) return 0; // (0)1 if (Val & (UINT64_C(1) << 63)) return countLeadingOnes(Val); // (m)1 return countLeadingZeros(Val) | 0x40; // (m)0 } /// mimm2Val - Convert a target MImm immediate to an integer immediate value. inline static uint64_t mimm2Val(uint64_t Val) { if (Val == 0) return 0; // (0)1 if ((Val & 0x40) == 0) return (uint64_t)((INT64_C(1) << 63) >> (Val & 0x3f)); // (m)1 return ((uint64_t)INT64_C(-1) >> (Val & 0x3f)); // (m)0 } inline unsigned M0(unsigned Val) { return Val + 64; } inline unsigned M1(unsigned Val) { return Val; } static const unsigned StandardVectorWidth = 256; static const unsigned PackedVectorWidth = 512; } // namespace llvm #endif