2022-01-18 18:32:19 +08:00
|
|
|
//===-- VECustomDAG.h - VE Custom DAG Nodes ------------*- 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 defines the interfaces that VE uses to lower LLVM code into a
|
|
|
|
// selection DAG.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "VECustomDAG.h"
|
|
|
|
|
|
|
|
#ifndef DEBUG_TYPE
|
|
|
|
#define DEBUG_TYPE "vecustomdag"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2022-01-26 17:32:26 +08:00
|
|
|
static const int StandardVectorWidth = 256;
|
|
|
|
|
|
|
|
bool isPackedVectorType(EVT SomeVT) {
|
|
|
|
if (!SomeVT.isVector())
|
|
|
|
return false;
|
|
|
|
return SomeVT.getVectorNumElements() > StandardVectorWidth;
|
|
|
|
}
|
|
|
|
|
2022-01-21 16:15:50 +08:00
|
|
|
/// \returns the VVP_* SDNode opcode corresponsing to \p OC.
|
|
|
|
Optional<unsigned> getVVPOpcode(unsigned Opcode) {
|
|
|
|
switch (Opcode) {
|
|
|
|
#define HANDLE_VP_TO_VVP(VPOPC, VVPNAME) \
|
|
|
|
case ISD::VPOPC: \
|
|
|
|
return VEISD::VVPNAME;
|
|
|
|
#define ADD_VVP_OP(VVPNAME, SDNAME) \
|
|
|
|
case VEISD::VVPNAME: \
|
|
|
|
case ISD::SDNAME: \
|
|
|
|
return VEISD::VVPNAME;
|
|
|
|
#include "VVPNodes.def"
|
|
|
|
}
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2022-02-02 16:11:33 +08:00
|
|
|
bool maySafelyIgnoreMask(SDValue Op) {
|
|
|
|
auto VVPOpc = getVVPOpcode(Op->getOpcode());
|
|
|
|
auto Opc = VVPOpc.getValueOr(Op->getOpcode());
|
|
|
|
|
|
|
|
switch (Opc) {
|
|
|
|
case VEISD::VVP_SDIV:
|
|
|
|
case VEISD::VVP_UDIV:
|
|
|
|
case VEISD::VVP_FDIV:
|
|
|
|
case VEISD::VVP_SELECT:
|
|
|
|
return false;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isVVPOrVEC(unsigned Opcode) {
|
|
|
|
switch (Opcode) {
|
|
|
|
case VEISD::VEC_BROADCAST:
|
|
|
|
#define ADD_VVP_OP(VVPNAME, ...) case VEISD::VVPNAME:
|
|
|
|
#include "VVPNodes.def"
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-01-21 16:15:50 +08:00
|
|
|
bool isVVPBinaryOp(unsigned VVPOpcode) {
|
|
|
|
switch (VVPOpcode) {
|
|
|
|
#define ADD_BINARY_VVP_OP(VVPNAME, ...) \
|
|
|
|
case VEISD::VVPNAME: \
|
|
|
|
return true;
|
|
|
|
#include "VVPNodes.def"
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-02-02 16:11:33 +08:00
|
|
|
// Return the AVL operand position for this VVP or VEC Op.
|
|
|
|
Optional<int> getAVLPos(unsigned Opc) {
|
|
|
|
// This is only available for VP SDNodes
|
|
|
|
auto PosOpt = ISD::getVPExplicitVectorLengthIdx(Opc);
|
|
|
|
if (PosOpt)
|
|
|
|
return *PosOpt;
|
|
|
|
|
|
|
|
// VVP Opcodes.
|
|
|
|
if (isVVPBinaryOp(Opc))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
// VM Opcodes.
|
|
|
|
switch (Opc) {
|
|
|
|
case VEISD::VEC_BROADCAST:
|
|
|
|
return 1;
|
|
|
|
case VEISD::VVP_SELECT:
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isLegalAVL(SDValue AVL) { return AVL->getOpcode() == VEISD::LEGALAVL; }
|
|
|
|
|
|
|
|
SDValue getNodeAVL(SDValue Op) {
|
|
|
|
auto PosOpt = getAVLPos(Op->getOpcode());
|
|
|
|
return PosOpt ? Op->getOperand(*PosOpt) : SDValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<SDValue, bool> getAnnotatedNodeAVL(SDValue Op) {
|
|
|
|
SDValue AVL = getNodeAVL(Op);
|
|
|
|
if (!AVL)
|
|
|
|
return {SDValue(), true};
|
|
|
|
if (isLegalAVL(AVL))
|
|
|
|
return {AVL->getOperand(0), true};
|
|
|
|
return {AVL, false};
|
|
|
|
}
|
|
|
|
|
2022-01-18 18:32:19 +08:00
|
|
|
SDValue VECustomDAG::getConstant(uint64_t Val, EVT VT, bool IsTarget,
|
|
|
|
bool IsOpaque) const {
|
|
|
|
return DAG.getConstant(Val, DL, VT, IsTarget, IsOpaque);
|
|
|
|
}
|
|
|
|
|
2022-01-21 16:15:50 +08:00
|
|
|
SDValue VECustomDAG::getBroadcast(EVT ResultVT, SDValue Scalar,
|
|
|
|
SDValue AVL) const {
|
2022-01-26 17:32:26 +08:00
|
|
|
assert(ResultVT.isVector());
|
|
|
|
auto ScaVT = Scalar.getValueType();
|
|
|
|
assert(ScaVT != MVT::i1 && "TODO: Mask broadcasts");
|
|
|
|
|
|
|
|
if (isPackedVectorType(ResultVT)) {
|
|
|
|
// v512x packed mode broadcast
|
|
|
|
// Replicate the scalar reg (f32 or i32) onto the opposing half of the full
|
|
|
|
// scalar register. If it's an I64 type, assume that this has already
|
|
|
|
// happened.
|
|
|
|
if (ScaVT == MVT::f32) {
|
|
|
|
Scalar = getNode(VEISD::REPL_F32, MVT::i64, Scalar);
|
|
|
|
} else if (ScaVT == MVT::i32) {
|
|
|
|
Scalar = getNode(VEISD::REPL_I32, MVT::i64, Scalar);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-21 16:15:50 +08:00
|
|
|
return getNode(VEISD::VEC_BROADCAST, ResultVT, {Scalar, AVL});
|
|
|
|
}
|
|
|
|
|
2022-02-02 16:11:33 +08:00
|
|
|
SDValue VECustomDAG::annotateLegalAVL(SDValue AVL) const {
|
|
|
|
if (isLegalAVL(AVL))
|
|
|
|
return AVL;
|
|
|
|
return getNode(VEISD::LEGALAVL, AVL.getValueType(), AVL);
|
|
|
|
}
|
|
|
|
|
2022-01-18 18:32:19 +08:00
|
|
|
} // namespace llvm
|