This intrinsic implements IEEE-754 operation roundToIntegralTiesToEven,
and performs rounding to the nearest integer value, rounding halfway
cases to even. This intrinsic represents the missed case of IEEE-754
rounding operations and now llvm provides full support of the rounding
operations defined by the standard.
Details
Diff Detail
- Repository
- rG LLVM Github Monorepo
Event Timeline
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 | Isn't this the same as rint? Since the default rounding mode is assumed | |
13028 | I've never heard of this C standard function? |
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 | The name is a hint that we should prefer to lower to a library function named "roundeven", if we can't generate an inline sequence. Other than that, yes, it's identical. I vaguely recall that I tried to merge llvm.rint and llvm.nearbyint at some point, but someone ran into issues with missing symbols. I can try to dig it up if you think it's important. | |
13028 | It's part of TS 18661-1, which I think is proposed to be part of the next version of the C standard. |
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 | I don't think the intrinsic name should ever influence codegen that way. We should not have redundant intrinsics, especially for some weird lowering hint |
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 | See https://reviews.llvm.org/rL299247 for recent history of this. |
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 |
roundeven does not assume any mode. It always rounds to nearest, ties to even.
There is a library function roundeven, which of course differs from any other rounding function.
This is a function defined in the recent C standard draft http://www.open-std.org/jtc1/sc22/wg14/www/docs/n2454.pdf, 7.12.9.8.
It is semantically necessary change as rint and nearbyint are different functions if we are interested with FP exception status. |
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 |
This is already very much the case. The regular FP operations and non constrained intrinsics assume the default FP environment |
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 |
That's true. We need this intrinsic to implement nearbyint and rint in default mode. |
Although the patch appears large, most of it is made by copy-paste of llvm.round pieces.
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 |
In terms of LLVM IR, the unconstrained intrinsics llvm.rint, llvm.nearbyint, and llvm.roundeven are semantially identical. Backends should be able to lower them all the same way. If not, that's a bug. For this reason, I'm not sure we need all three. |
llvm/docs/LangRef.rst | ||
---|---|---|
12994–12997 |
Yes, unconstrained llvm.rint and llvm.nearbyint implement roundToIntegralTiesToEven in default mode, so they are not needed. I asked some time ago, if we need these intrinsics (http://lists.llvm.org/pipermail/cfe-dev/2020-March/064803.html). It looks like there is no reason to keep them, so they can be removed, only constrained versions should remain. Of unconstrained intrinsic only llvm.roundeven should survive as it is proposed exactly for this kind of rounding. |
Replacement of llvm.nearbyint and llvm.rint with llvm.roundeven in the case of default FP environment is an optimization. If it is not done, these intrinsics will be converted into runtime library calls and the semantics expected by user will not be broken. While pragma STDC FENV_ACCESS is not fully supported, compiler should not do such optimization, because it would made functions nearbyint* and rint* unavailable.
This patch only introduces new intrinsic function that will be used to implement functions roundeven* defined in new C standard.
I think the experimental_constrained_roundeven part is uncontroversial.
For the rest, there are two competing interests here:
- We don't want multiple intrinsics that represent the same thing.
- Transforming calls to roundeven/rint/nearbyint to one of the other functions is surprising, even if it's legal in some sense, and will likely break some practical use because we don't control the C library.
Violating (1) seems annoying, but not much of a practical problem; the few bits of code that actually care can easily check for three names instead of one. Violating (2) seems like fodder for weird bug reports I don't want to deal with. Given that, I think we should accept this patch as-is.
Isn't this the same as rint? Since the default rounding mode is assumed