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 fadd
230 fadd(const KnownFPClass &LHS, const KnownFPClass &RHS,
232
233 /// Report known values for fadd x, x
235 fadd_self(const KnownFPClass &Src,
237
238 /// Report known values for fsub
240 fsub(const KnownFPClass &LHS, const KnownFPClass &RHS,
242
243 /// Report known values for fmul
245 fmul(const KnownFPClass &LHS, const KnownFPClass &RHS,
247
248 // Special case of fmul x, x.
251 KnownFPClass Known = fmul(Src, Src, Mode);
252
253 // X * X is always non-negative or a NaN.
254 Known.knownNot(fcNegative);
255 Known.propagateNaN(Src);
256 return Known;
257 }
258
260 fmul(const KnownFPClass &LHS, const APFloat &RHS,
262
263 /// Report known values for fdiv
265 fdiv(const KnownFPClass &LHS, const KnownFPClass &RHS,
267
268 /// Report known values for fdiv x, x
270 fdiv_self(const KnownFPClass &Src,
272
273 /// Report known values for frem
275 frem_self(const KnownFPClass &Src,
277
278 /// Report known values for fma
280 fma(const KnownFPClass &LHS, const KnownFPClass &RHS,
281 const KnownFPClass &Addend,
283
284 /// Report known values for fma squared, squared, addend
286 fma_square(const KnownFPClass &Squared, const KnownFPClass &Addend,
288
289 /// Report known values for exp, exp2 and exp10.
290 LLVM_ABI static KnownFPClass exp(const KnownFPClass &Src);
291
292 /// Report known values for sin
293 LLVM_ABI static KnownFPClass sin(const KnownFPClass &Src);
294
295 /// Report known values for cos
296 LLVM_ABI static KnownFPClass cos(const KnownFPClass &Src);
297
298 /// Return true if the sign bit must be 0, ignoring the sign of nans.
300
301 /// Assume the sign bit is zero.
304 SignBit = false;
305 }
306
307 /// Assume the sign bit is one.
310 SignBit = true;
311 }
312
313 void copysign(const KnownFPClass &Sign) {
314 // Don't know anything about the sign of the source. Expand the possible set
315 // to its opposite sign pair.
322 if (KnownFPClasses & fcInf)
324
325 // Sign bit is exactly preserved even for nans.
326 SignBit = Sign.SignBit;
327
328 // Clear sign bits based on the input sign mask.
329 if (Sign.isKnownNever(fcPositive | fcNan) || (SignBit && *SignBit))
331 if (Sign.isKnownNever(fcNegative | fcNan) || (SignBit && !*SignBit))
333 }
334
335 static KnownFPClass copysign(const KnownFPClass &KnownMag,
336 const KnownFPClass &KnownSign) {
337 KnownFPClass Known = KnownMag;
338 Known.copysign(KnownSign);
339 return Known;
340 }
341
342 // Propagate knowledge that a non-NaN source implies the result can also not
343 // be a NaN. For unconstrained operations, signaling nans are not guaranteed
344 // to be quieted but cannot be introduced.
345 void propagateNaN(const KnownFPClass &Src, bool PreserveSign = false) {
346 if (Src.isKnownNever(fcNan)) {
348 if (PreserveSign)
349 SignBit = Src.SignBit;
350 } else if (Src.isKnownNever(fcSNan))
352 }
353
354 /// Propagate knowledge from a source value that could be a denormal or
355 /// zero. We have to be conservative since output flushing is not guaranteed,
356 /// so known-never-zero may not hold.
357 ///
358 /// This assumes a copy-like operation and will replace any currently known
359 /// information.
361
362 /// Report known classes if \p Src is evaluated through a potentially
363 /// canonicalizing operation. We can assume signaling nans will not be
364 /// introduced, but cannot assume a denormal will be flushed under FTZ/DAZ.
365 ///
366 /// This assumes a copy-like operation and will replace any currently known
367 /// information.
370
371 /// Propagate known class for log/log2/log10
374
375 /// Propagate known class for sqrt
378
379 /// Propagate known class for fpext.
380 static LLVM_ABI KnownFPClass fpext(const KnownFPClass &KnownSrc,
381 const fltSemantics &DstTy,
382 const fltSemantics &SrcTy);
383
384 /// Propagate known class for fptrunc.
385 static LLVM_ABI KnownFPClass fptrunc(const KnownFPClass &KnownSrc);
386
387 /// Propagate known class for rounding intrinsics (trunc, floor, ceil, rint,
388 /// nearbyint, round, roundeven). This is trunc if \p IsTrunc. \p
389 /// IsMultiUnitFPType if this is for a multi-unit floating-point type.
391 bool IsTrunc,
392 bool IsMultiUnitFPType);
393
394 /// Propagate known class for mantissa component of frexp
397
398 /// Propagate known class for ldexp
400 ldexp(const KnownFPClass &Src, const KnownBits &N, const fltSemantics &Flt,
402
403 /// Propagate known class for powi
404 static LLVM_ABI KnownFPClass powi(const KnownFPClass &Src,
405 const KnownBits &N);
406
407 void resetAll() { *this = KnownFPClass(); }
408};
409
411 LHS |= RHS;
412 return LHS;
413}
414
416 RHS |= LHS;
417 return std::move(RHS);
418}
419
420} // namespace llvm
421
422#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.
Definition Types.h:26
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:2158
#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 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.