[llvm] [DAG] Support saturated truncate (PR #99418)
Simon Pilgrim via llvm-commits
llvm-commits at lists.llvm.org
Thu Aug 8 04:16:13 PDT 2024
================
@@ -14915,6 +14920,153 @@ SDValue DAGCombiner::visitEXTEND_VECTOR_INREG(SDNode *N) {
return SDValue();
}
+SDValue DAGCombiner::visitTRUNCATE_USAT(SDNode *N) {
+ EVT VT = N->getValueType(0);
+ SDValue N0 = N->getOperand(0);
+
+ std::function<SDValue(SDValue)> MatchFPTOINT = [&](SDValue Val) -> SDValue {
+ if (Val.getOpcode() == ISD::FP_TO_UINT)
+ return Val;
+ return SDValue();
+ };
+
+ SDValue FPInstr = MatchFPTOINT(N0);
+ if (!FPInstr)
+ return SDValue();
+
+ EVT FPVT = FPInstr.getOperand(0).getValueType();
+ if (!DAG.getTargetLoweringInfo().shouldConvertFpToSat(ISD::FP_TO_UINT_SAT,
+ FPVT, VT))
+ return SDValue();
+ return DAG.getNode(ISD::FP_TO_UINT_SAT, SDLoc(FPInstr), VT,
+ FPInstr.getOperand(0),
+ DAG.getValueType(VT.getScalarType()));
+}
+
+// Match min and return limit value as a parameter.
+static SDValue matchMinMax(SDValue V, unsigned Opcode, APInt &Limit) {
+ if (V.getOpcode() == Opcode) {
+ APInt C;
+ if (ISD::isConstantSplatVector(V.getOperand(1).getNode(), C)) {
+ if (C == Limit)
+ return V.getOperand(0);
+ }
+ }
+ return SDValue();
+}
+
+/// Detect patterns of truncation with unsigned saturation:
+///
+/// (truncate (umin (x, unsigned_max_of_dest_type)) to dest_type).
+/// Return the source value x to be truncated or SDValue() if the pattern was
+/// not matched.
+///
+static SDValue detectUSatUPattern(SDValue In, EVT VT) {
+ unsigned NumDstBits = VT.getScalarSizeInBits();
+ unsigned NumSrcBits = In.getScalarValueSizeInBits();
+ // Saturation with truncation. We truncate from InVT to VT.
+ assert(NumSrcBits > NumDstBits && "Unexpected types for truncate operation");
+
+ APInt UnsignedMax = APInt::getMaxValue(NumDstBits).zext(NumSrcBits);
+ if (SDValue Min = matchMinMax(In, ISD::UMIN, UnsignedMax)) {
----------------
RKSimon wrote:
These could all be done more cleanly with sd_match:
```cpp
SDValue Min;
if (sd_match(In, m_UMin(m_Value(Min), m_SpecificInt(UnsignedMask))))
return Min;
```
https://github.com/llvm/llvm-project/pull/99418
More information about the llvm-commits
mailing list