[llvm] [InstCombine] Generalise optimisation of redundant floating point comparisons with `ConstantFPRange` (PR #159315)

Yingwei Zheng via llvm-commits llvm-commits at lists.llvm.org
Fri Sep 19 22:26:01 PDT 2025


================
@@ -1812,6 +1813,61 @@ static Value *simplifyOrOfICmps(ICmpInst *Op0, ICmpInst *Op1,
   return nullptr;
 }
 
+/// Test if a pair of compares with a shared operand and 2 constants has an
+/// empty set intersection, full set union, or if one compare is a superset of
+/// the other.
+static Value *simplifyAndOrOfFCmpsWithConstants(FCmpInst *Cmp0, FCmpInst *Cmp1,
+                                                bool IsAnd) {
+  // Look for this pattern: {and/or} (fcmp X, C0), (fcmp X, C1)).
+  if (Cmp0->getOperand(0) != Cmp1->getOperand(0))
+    return nullptr;
+
+  const APFloat *C0, *C1;
+  if (!match(Cmp0->getOperand(1), m_APFloat(C0)) ||
+      !match(Cmp1->getOperand(1), m_APFloat(C1)))
+    return nullptr;
+
+  auto Range0 = ConstantFPRange::makeExactFCmpRegion(Cmp0->getPredicate(), *C0);
+  auto Range1 = ConstantFPRange::makeExactFCmpRegion(Cmp1->getPredicate(), *C1);
+
+  if (!Range0 || !Range1)
+    return nullptr;
+
+  // For and-of-compares, check if the intersection is empty:
+  // (fcmp X, C0) && (fcmp X, C1) --> empty set --> false
+  if (IsAnd && (*Range0).intersectWith(*Range1).isEmptySet())
+    return getFalse(Cmp0->getType());
+
+  // For or-of-compares, check if the union is full:
+  // (fcmp X, C0) || (fcmp X, C1) --> full set --> true
+  //
+  // TODO: `unionWith` is not precise at the moment, so
----------------
dtcxzyw wrote:

See also our discussions in the original PR: https://github.com/llvm/llvm-project/pull/86483
Currently we cannot represent the union of two continuous ranges like `X < -1 || X > 1`. Supporting such ranges makes the implementation more complex. So we decided not to handle them unless it is useful in practice. Feel free to extend ConstantFPRange if you have a better idea :)

||ConstantRange|ConstantFPRange|
|--|--|--|
|unionWith|superset|superset|
|intersectWith|superset|exact|
|inverse|exact|not supported|

Another workaround is adding something like `exactUnionWith`. Actually, we should also use exact union/intersect in `simplifyAndOrOfICmpsWithConstants`. But the current implementation happens to work for integers (`ConstantRange::union` returns the smallest result).

Anyway, I still think my solution (i.e., applying De Morgan's Laws) is simple and fast. At least we don't need to touch ConstantFPRange's stuff.



https://github.com/llvm/llvm-project/pull/159315


More information about the llvm-commits mailing list