LLVM 23.0.0git
KnownFPClass.h
Go to the documentation of this file.
1//===- llvm/Support/KnownFPClass.h - Stores known fpclass -------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This file contains a class for representing known fpclasses used by
10// computeKnownFPClass.
11//
12//===----------------------------------------------------------------------===//
13
14#ifndef LLVM_SUPPORT_KNOWNFPCLASS_H
15#define LLVM_SUPPORT_KNOWNFPCLASS_H
16
19#include <optional>
20
21namespace llvm {
22class APFloat;
23struct fltSemantics;
24struct KnownBits;
25
27 /// Floating-point classes the value could be one of.
29
30 /// std::nullopt if the sign bit is unknown, true if the sign bit is
31 /// definitely set or false if the sign bit is definitely unset.
32 std::optional<bool> SignBit;
33
34 KnownFPClass(FPClassTest Known = fcAllFlags, std::optional<bool> Sign = {})
35 : KnownFPClasses(Known), SignBit(Sign) {}
36 LLVM_ABI KnownFPClass(const APFloat &C);
37
39 return KnownFPClasses == Other.KnownFPClasses && SignBit == Other.SignBit;
40 }
41
42 /// Return true if it's known this can never be one of the mask entries.
43 bool isKnownNever(FPClassTest Mask) const {
44 return (KnownFPClasses & Mask) == fcNone;
45 }
46
47 bool isKnownAlways(FPClassTest Mask) const { return isKnownNever(~Mask); }
48
49 bool isUnknown() const { return KnownFPClasses == fcAllFlags && !SignBit; }
50
51 /// Return true if it's known this can never be a nan.
52 bool isKnownNeverNaN() const { return isKnownNever(fcNan); }
53
54 /// Return true if it's known this must always be a nan.
55 bool isKnownAlwaysNaN() const { return isKnownAlways(fcNan); }
56
57 /// Return true if it's known this can never be an infinity.
58 bool isKnownNeverInfinity() const { return isKnownNever(fcInf); }
59
60 /// Return true if it's known this can never be an infinity or nan
61 bool isKnownNeverInfOrNaN() const { return isKnownNever(fcInf | fcNan); }
62
63 /// Return true if it's known this can never be +infinity.
65
66 /// Return true if it's known this can never be -infinity.
68
69 /// Return true if it's known this can never be a subnormal
71
72 /// Return true if it's known this can never be a positive subnormal
74
75 /// Return true if it's known this can never be a negative subnormal
77
78 /// Return true if it's known this can never be a zero. This means a literal
79 /// [+-]0, and does not include denormal inputs implicitly treated as [+-]0.
80 bool isKnownNeverZero() const { return isKnownNever(fcZero); }
81
82 /// Return true if it's known this can never be a literal positive zero.
83 bool isKnownNeverPosZero() const { return isKnownNever(fcPosZero); }
84
85 /// Return true if it's known this can never be a negative zero. This means a
86 /// literal -0 and does not include denormal inputs implicitly treated as -0.
87 bool isKnownNeverNegZero() const { return isKnownNever(fcNegZero); }
88
89 /// Return true if it's known this can never be interpreted as a zero. This
90 /// extends isKnownNeverZero to cover the case where the assumed
91 /// floating-point mode for the function interprets denormals as zero.
93
94 /// Return true if it's known this can never be interpreted as a negative
95 /// zero.
97
98 /// Return true if it's known this can never be interpreted as a positive
99 /// zero.
101
106
107 /// Return true if we can prove that the analyzed floating-point value is
108 /// either NaN or never less than -0.0.
109 ///
110 /// NaN --> true
111 /// +0 --> true
112 /// -0 --> true
113 /// x > +0 --> true
114 /// x < -0 --> false
118
119 /// Return true if we can prove that the analyzed floating-point value is
120 /// either NaN or never greater than -0.0.
121 /// NaN --> true
122 /// +0 --> true
123 /// -0 --> true
124 /// x > +0 --> false
125 /// x < -0 --> true
129
130 /// Return true if it's know this can never be a negative value or a logical
131 /// 0.
132 ///
133 /// NaN --> true
134 /// x >= -0 --> false
135 /// nsub --> true if mode is ieee, false otherwise.
136 /// x < -0 --> true
140
142 return KnownFPClass(KnownFPClasses | RHS.KnownFPClasses,
143 SignBit == RHS.SignBit ? SignBit : std::nullopt);
144 }
145
147 std::optional<bool> MergedSignBit;
148 if (SignBit && !RHS.SignBit)
149 MergedSignBit = SignBit;
150 else if (!SignBit && RHS.SignBit)
151 MergedSignBit = RHS.SignBit;
152
153 return KnownFPClass(KnownFPClasses & RHS.KnownFPClasses, MergedSignBit);
154 }
155
157 KnownFPClasses = KnownFPClasses | RHS.KnownFPClasses;
158
159 if (SignBit != RHS.SignBit)
160 SignBit = std::nullopt;
161 return *this;
162 }
163
164 void knownNot(FPClassTest RuleOut) {
165 KnownFPClasses = KnownFPClasses & ~RuleOut;
166 if (isKnownNever(fcNan) && !SignBit) {
168 SignBit = false;
169 else if (isKnownNever(fcPositive))
170 SignBit = true;
171 }
172 }
173
174 void fneg() {
176 if (SignBit)
177 SignBit = !*SignBit;
178 }
179
180 static KnownFPClass fneg(const KnownFPClass &Src) {
181 KnownFPClass Known = Src;
182 Known.fneg();
183 return Known;
184 }
185
201
202 static KnownFPClass fabs(const KnownFPClass &Src) {
203 KnownFPClass Known = Src;
204 Known.fabs();
205 return Known;
206 }
207
208 // Enum of min/max intrinsics to avoid dependency on IR.
217
221
222 /// Apply the canonicalize intrinsic to this value. This is essentially a
223 /// stronger form of propagateCanonicalizingSrc.
225 canonicalize(const KnownFPClass &Src,
227
228 /// Report known values for a bitcast into a float with provided semantics.
229 LLVM_ABI static KnownFPClass bitcast(const fltSemantics &FltSemantics,
230 const KnownBits &Bits);
231
232 /// Report known values for fadd
234 fadd(const KnownFPClass &LHS, const KnownFPClass &RHS,
236
237 /// Report known values for fadd x, x
239 fadd_self(const KnownFPClass &Src,
241
242 /// Report known values for fsub
244 fsub(const KnownFPClass &LHS, const KnownFPClass &RHS,
246
247 /// Report known values for fmul
249 fmul(const KnownFPClass &LHS, const KnownFPClass &RHS,
251
252 // Special case of fmul x, x.
255 KnownFPClass Known = fmul(Src, Src, Mode);
256
257 // X * X is always non-negative or a NaN.
258 Known.knownNot(fcNegative);
259 Known.propagateNaN(Src);
260 return Known;
261 }
262
264 fmul(const KnownFPClass &LHS, const APFloat &RHS,
266
267 /// Report known values for fdiv
269 fdiv(const KnownFPClass &LHS, const KnownFPClass &RHS,
271
272 /// Report known values for fdiv x, x
274 fdiv_self(const KnownFPClass &Src,
276
277 /// Report known values for frem
279 frem_self(const KnownFPClass &Src,
281
282 /// Report known values for fma
284 fma(const KnownFPClass &LHS, const KnownFPClass &RHS,
285 const KnownFPClass &Addend,
287
288 /// Report known values for fma squared, squared, addend
290 fma_square(const KnownFPClass &Squared, const KnownFPClass &Addend,
292
293 /// Report known values for exp, exp2 and exp10.
294 LLVM_ABI static KnownFPClass exp(const KnownFPClass &Src);
295
296 /// Report known values for sin
297 LLVM_ABI static KnownFPClass sin(const KnownFPClass &Src);
298
299 /// Report known values for cos
300 LLVM_ABI static KnownFPClass cos(const KnownFPClass &Src);
301
302 /// Return true if the sign bit must be 0, ignoring the sign of nans.
304
305 /// Assume the sign bit is zero.
308 SignBit = false;
309 }
310
311 /// Assume the sign bit is one.
314 SignBit = true;
315 }
316
317 void copysign(const KnownFPClass &Sign) {
318 // Don't know anything about the sign of the source. Expand the possible set
319 // to its opposite sign pair.
326 if (KnownFPClasses & fcInf)
328
329 // Sign bit is exactly preserved even for nans.
330 SignBit = Sign.SignBit;
331
332 // Clear sign bits based on the input sign mask.
333 if (Sign.isKnownNever(fcPositive | fcNan) || (SignBit && *SignBit))
335 if (Sign.isKnownNever(fcNegative | fcNan) || (SignBit && !*SignBit))
337 }
338
339 static KnownFPClass copysign(const KnownFPClass &KnownMag,
340 const KnownFPClass &KnownSign) {
341 KnownFPClass Known = KnownMag;
342 Known.copysign(KnownSign);
343 return Known;
344 }
345
346 // Propagate knowledge that a non-NaN source implies the result can also not
347 // be a NaN. For unconstrained operations, signaling nans are not guaranteed
348 // to be quieted but cannot be introduced.
349 void propagateNaN(const KnownFPClass &Src, bool PreserveSign = false) {
350 if (Src.isKnownNever(fcNan)) {
352 if (PreserveSign)
353 SignBit = Src.SignBit;
354 } else if (Src.isKnownNever(fcSNan))
356 }
357
358 /// Propagate knowledge from a source value that could be a denormal or
359 /// zero. We have to be conservative since output flushing is not guaranteed,
360 /// so known-never-zero may not hold.
361 ///
362 /// This assumes a copy-like operation and will replace any currently known
363 /// information.
365
366 /// Report known classes if \p Src is evaluated through a potentially
367 /// canonicalizing operation. We can assume signaling nans will not be
368 /// introduced, but cannot assume a denormal will be flushed under FTZ/DAZ.
369 ///
370 /// This assumes a copy-like operation and will replace any currently known
371 /// information.
374
375 /// Propagate known class for log/log2/log10
378
379 /// Propagate known class for sqrt
382
383 /// Propagate known class for fpext.
384 static LLVM_ABI KnownFPClass fpext(const KnownFPClass &KnownSrc,
385 const fltSemantics &DstTy,
386 const fltSemantics &SrcTy);
387
388 /// Propagate known class for fptrunc.
389 static LLVM_ABI KnownFPClass fptrunc(const KnownFPClass &KnownSrc);
390
391 /// Propagate known class for rounding intrinsics (trunc, floor, ceil, rint,
392 /// nearbyint, round, roundeven). This is trunc if \p IsTrunc. \p
393 /// IsMultiUnitFPType if this is for a multi-unit floating-point type.
395 bool IsTrunc,
396 bool IsMultiUnitFPType);
397
398 /// Propagate known class for mantissa component of frexp
401
402 /// Propagate known class for ldexp
404 ldexp(const KnownFPClass &Src, const KnownBits &N, const fltSemantics &Flt,
406
407 /// Propagate known class for powi
408 static LLVM_ABI KnownFPClass powi(const KnownFPClass &Src,
409 const KnownBits &N);
410
411 void resetAll() { *this = KnownFPClass(); }
412};
413
415 LHS |= RHS;
416 return LHS;
417}
418
420 RHS |= LHS;
421 return std::move(RHS);
422}
423
424} // namespace llvm
425
426#endif
static GCRegistry::Add< ShadowStackGC > C("shadow-stack", "Very portable GC for uncooperative code generators")
#define LLVM_ABI
Definition Compiler.h:213
Utilities for dealing with flags related to floating point properties and mode controls.
static cl::opt< RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode > Mode("regalloc-enable-advisor", cl::Hidden, cl::init(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Default), cl::desc("Enable regalloc advisor mode"), cl::values(clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Default, "default", "Default"), clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Release, "release", "precompiled"), clEnumValN(RegAllocEvictionAdvisorAnalysisLegacy::AdvisorMode::Development, "development", "for training")))
Value * RHS
Value * LHS
This is an optimization pass for GlobalISel generic memory operations.
LLVM_ABI FPClassTest fneg(FPClassTest Mask)
Return the test mask which returns true if the value's sign bit is flipped.
FPClassTest
Floating-point class tests, supported by 'is_fpclass' intrinsic.
@ Other
Any other memory.
Definition ModRef.h:68
APInt operator|(APInt a, const APInt &b)
Definition APInt.h:2161
#define N
Represent subnormal handling kind for floating point instruction inputs and outputs.
static constexpr DenormalMode getDynamic()
bool isKnownNeverInfOrNaN() const
Return true if it's known this can never be an infinity or nan.
bool isKnownAlwaysNaN() const
Return true if it's known this must always be a nan.
FPClassTest KnownFPClasses
Floating-point classes the value could be one of.
bool isKnownNeverInfinity() const
Return true if it's known this can never be an infinity.
KnownFPClass(FPClassTest Known=fcAllFlags, std::optional< bool > Sign={})
bool cannotBeOrderedGreaterThanZero() const
Return true if we can prove that the analyzed floating-point value is either NaN or never greater tha...
static LLVM_ABI KnownFPClass sin(const KnownFPClass &Src)
Report known values for sin.
static LLVM_ABI KnownFPClass fdiv_self(const KnownFPClass &Src, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fdiv x, x.
static constexpr FPClassTest OrderedGreaterThanZeroMask
static constexpr FPClassTest OrderedLessThanZeroMask
void knownNot(FPClassTest RuleOut)
static LLVM_ABI KnownFPClass fmul(const KnownFPClass &LHS, const KnownFPClass &RHS, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fmul.
static LLVM_ABI KnownFPClass fadd_self(const KnownFPClass &Src, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fadd x, x.
bool isKnownNeverZero() const
Return true if it's known this can never be a zero.
void copysign(const KnownFPClass &Sign)
static KnownFPClass square(const KnownFPClass &Src, DenormalMode Mode=DenormalMode::getDynamic())
static LLVM_ABI KnownFPClass fsub(const KnownFPClass &LHS, const KnownFPClass &RHS, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fsub.
bool isKnownNeverSubnormal() const
Return true if it's known this can never be a subnormal.
bool isKnownAlways(FPClassTest Mask) const
KnownFPClass unionWith(const KnownFPClass &RHS) const
static LLVM_ABI KnownFPClass canonicalize(const KnownFPClass &Src, DenormalMode DenormMode=DenormalMode::getDynamic())
Apply the canonicalize intrinsic to this value.
LLVM_ABI bool isKnownNeverLogicalZero(DenormalMode Mode) const
Return true if it's known this can never be interpreted as a zero.
static LLVM_ABI KnownFPClass log(const KnownFPClass &Src, DenormalMode Mode=DenormalMode::getDynamic())
Propagate known class for log/log2/log10.
LLVM_ABI void propagateDenormal(const KnownFPClass &Src, DenormalMode Mode)
Propagate knowledge from a source value that could be a denormal or zero.
static LLVM_ABI KnownFPClass fdiv(const KnownFPClass &LHS, const KnownFPClass &RHS, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fdiv.
KnownFPClass & operator|=(const KnownFPClass &RHS)
static KnownFPClass fneg(const KnownFPClass &Src)
static LLVM_ABI KnownFPClass roundToIntegral(const KnownFPClass &Src, bool IsTrunc, bool IsMultiUnitFPType)
Propagate known class for rounding intrinsics (trunc, floor, ceil, rint, nearbyint,...
static LLVM_ABI KnownFPClass cos(const KnownFPClass &Src)
Report known values for cos.
static LLVM_ABI KnownFPClass ldexp(const KnownFPClass &Src, const KnownBits &N, const fltSemantics &Flt, DenormalMode Mode=DenormalMode::getDynamic())
Propagate known class for ldexp.
static LLVM_ABI KnownFPClass minMaxLike(const KnownFPClass &LHS, const KnownFPClass &RHS, MinMaxKind Kind, DenormalMode DenormMode=DenormalMode::getDynamic())
bool isUnknown() const
KnownFPClass intersectWith(const KnownFPClass &RHS) const
bool isKnownNeverNegInfinity() const
Return true if it's known this can never be -infinity.
bool isKnownNeverNegSubnormal() const
Return true if it's known this can never be a negative subnormal.
bool isKnownNeverPosZero() const
Return true if it's known this can never be a literal positive zero.
static LLVM_ABI KnownFPClass exp(const KnownFPClass &Src)
Report known values for exp, exp2 and exp10.
static LLVM_ABI KnownFPClass frexp_mant(const KnownFPClass &Src, DenormalMode Mode=DenormalMode::getDynamic())
Propagate known class for mantissa component of frexp.
std::optional< bool > SignBit
std::nullopt if the sign bit is unknown, true if the sign bit is definitely set or false if the sign ...
bool isKnownNeverNaN() const
Return true if it's known this can never be a nan.
bool isKnownNever(FPClassTest Mask) const
Return true if it's known this can never be one of the mask entries.
static LLVM_ABI KnownFPClass fpext(const KnownFPClass &KnownSrc, const fltSemantics &DstTy, const fltSemantics &SrcTy)
Propagate known class for fpext.
static KnownFPClass copysign(const KnownFPClass &KnownMag, const KnownFPClass &KnownSign)
bool isKnownNeverNegZero() const
Return true if it's known this can never be a negative zero.
static LLVM_ABI KnownFPClass fma(const KnownFPClass &LHS, const KnownFPClass &RHS, const KnownFPClass &Addend, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fma.
void propagateNaN(const KnownFPClass &Src, bool PreserveSign=false)
static LLVM_ABI KnownFPClass fptrunc(const KnownFPClass &KnownSrc)
Propagate known class for fptrunc.
bool cannotBeOrderedLessThanZero() const
Return true if we can prove that the analyzed floating-point value is either NaN or never less than -...
void signBitMustBeOne()
Assume the sign bit is one.
LLVM_ABI void propagateCanonicalizingSrc(const KnownFPClass &Src, DenormalMode Mode)
Report known classes if Src is evaluated through a potentially canonicalizing operation.
void signBitMustBeZero()
Assume the sign bit is zero.
static LLVM_ABI KnownFPClass sqrt(const KnownFPClass &Src, DenormalMode Mode=DenormalMode::getDynamic())
Propagate known class for sqrt.
LLVM_ABI bool isKnownNeverLogicalPosZero(DenormalMode Mode) const
Return true if it's known this can never be interpreted as a positive zero.
bool isKnownNeverPosInfinity() const
Return true if it's known this can never be +infinity.
bool cannotBeOrderedGreaterEqZero(DenormalMode Mode) const
Return true if it's know this can never be a negative value or a logical 0.
static LLVM_ABI KnownFPClass fadd(const KnownFPClass &LHS, const KnownFPClass &RHS, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fadd.
static KnownFPClass fabs(const KnownFPClass &Src)
LLVM_ABI bool isKnownNeverLogicalNegZero(DenormalMode Mode) const
Return true if it's known this can never be interpreted as a negative zero.
static LLVM_ABI KnownFPClass bitcast(const fltSemantics &FltSemantics, const KnownBits &Bits)
Report known values for a bitcast into a float with provided semantics.
static LLVM_ABI KnownFPClass fma_square(const KnownFPClass &Squared, const KnownFPClass &Addend, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for fma squared, squared, addend.
bool operator==(KnownFPClass Other) const
static LLVM_ABI KnownFPClass frem_self(const KnownFPClass &Src, DenormalMode Mode=DenormalMode::getDynamic())
Report known values for frem.
static LLVM_ABI KnownFPClass powi(const KnownFPClass &Src, const KnownBits &N)
Propagate known class for powi.
bool signBitIsZeroOrNaN() const
Return true if the sign bit must be 0, ignoring the sign of nans.
bool isKnownNeverPosSubnormal() const
Return true if it's known this can never be a positive subnormal.