
This carries a bitmask indicating forbidden floating-point value kinds in the argument or return value. This will enable interprocedural -ffinite-math-only optimizations. This is primarily to cover the no-nans and no-infinities cases, but also covers the other floating point classes for free. Textually, this provides a number of names corresponding to bits in FPClassTest, e.g. call nofpclass(nan inf) @must_be_finite() call nofpclass(snan) @cannot_be_snan() This is more expressive than the existing nnan and ninf fast math flags. As an added bonus, you can represent fun things like nanf: declare nofpclass(inf zero sub norm) float @only_nans() Compared to nnan/ninf: - Can be applied to individual call operands as well as the return value - Can distinguish signaling and quiet nans - Distinguishes the sign of infinities - Can be safely propagated since it doesn't imply anything about other operands. - Does not apply to FP instructions; it's not a flag This is one step closer to being able to retire "no-nans-fp-math" and "no-infs-fp-math". The one remaining situation where we have no way to represent no-nans/infs is for loads (if we wanted to solve this we could introduce !nofpclass metadata, following along with noundef/!noundef). This is to help simplify the GPU builtin math library distribution. Currently the library code has explicit finite math only checks, read from global constants the compiler driver needs to set based on the compiler flags during linking. We end up having to internalize the library into each translation unit in case different linked modules have different math flags. By propagating known-not-nan and known-not-infinity information, we can automatically prune the edge case handling in most functions if the function is only reached from fast math uses.
53 lines
1.9 KiB
LLVM
53 lines
1.9 KiB
LLVM
; RUN: opt -S -passes=pgo-icall-prom -icp-total-percent-threshold=0 < %s 2>&1 | FileCheck %s
|
|
|
|
; Test that CallPromotionUtils will promote calls which require pointer cast
|
|
; safely, i.e. drop incompatible attributes.
|
|
|
|
@foo = common global ptr null, align 8
|
|
|
|
; correct type, preserve attributes
|
|
define double @func_double(double %a) {
|
|
ret double poison
|
|
}
|
|
|
|
; drop nofpclass attributes
|
|
define i64 @func_i64(i64 %a) {
|
|
ret i64 poison
|
|
}
|
|
|
|
define double @cast_scalar_fp(double %arg) {
|
|
%tmp = load ptr, ptr @foo, align 8
|
|
|
|
; Make sure callsite attributes are dropped on arguments and retval.
|
|
; CHECK: [[ARG:%[0-9]+]] = bitcast double %arg to i64
|
|
; CHECK-NEXT: call i64 @func_i64(i64 [[ARG]])
|
|
|
|
; Make sure callsite attributes are preserved on arguments and retval.
|
|
; CHECK: call nofpclass(inf) double @func_double(double nofpclass(nan)
|
|
|
|
; CHECK: call nofpclass(inf) double %tmp(double nofpclass(nan) %arg)
|
|
%call = call nofpclass(inf) double %tmp(double nofpclass(nan) %arg), !prof !0
|
|
ret double %call
|
|
}
|
|
|
|
; ; correct type, preserve attributes
|
|
define [2 x [2 x <2 x double>]] @func_array_vector_f64([2 x [2 x <2 x double>]] %a) {
|
|
ret [2 x [2 x <2 x double>]] poison
|
|
}
|
|
|
|
; drop nofpclass attributes
|
|
define [2 x [2 x <2 x i64>]] @func_array_vector_i64([2 x [2 x <2 x i64>]] %a) {
|
|
ret [2 x [2 x <2 x i64>]] poison
|
|
}
|
|
|
|
; FIXME: This is not promoted
|
|
; CHECK: %call = call nofpclass(inf) [2 x [2 x <2 x double>]] %tmp([2 x [2 x <2 x double>]] nofpclass(nan) %arg)
|
|
define [2 x [2 x <2 x double>]] @cast_array_vector([2 x [2 x <2 x double>]] %arg) {
|
|
%tmp = load ptr, ptr @foo, align 8
|
|
%call = call nofpclass(inf) [2 x [2 x <2 x double>]] %tmp([2 x [2 x <2 x double>]] nofpclass(nan) %arg), !prof !1
|
|
ret [2 x [2 x <2 x double>]] %call
|
|
}
|
|
|
|
!0 = !{!"VP", i32 0, i64 1440, i64 15573779287943805696, i64 1030, i64 16900752280434761561, i64 410}
|
|
!1 = !{!"VP", i32 0, i64 1440, i64 1124945363680759394, i64 1030, i64 16341336592352938424, i64 410}
|