diff --git a/llvm/include/llvm/IR/FPEnv.h b/llvm/include/llvm/IR/FPEnv.h new file mode 100644 --- /dev/null +++ b/llvm/include/llvm/IR/FPEnv.h @@ -0,0 +1,70 @@ +//===- FPEnv.h ---- FP Environment ------------------------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +/// @file +/// This file contains the declarations of entities that describe floating +/// point environment and related functions. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_IR_FLOATINGPOINT_H +#define LLVM_IR_FLOATINGPOINT_H + +#include "llvm/ADT/Optional.h" +#include "llvm/ADT/StringRef.h" +#include + +namespace llvm { + +namespace fp { + +/// Rounding mode used for floating point operations. +/// +/// Each of these values correspond to some metadata argument value of a +/// constrained floating point intrinsic. See the LLVM Language Reference Manual +/// for details. +enum RoundingMode : uint8_t { + rmDynamic, ///< This corresponds to "fpround.dynamic". + rmToNearest, ///< This corresponds to "fpround.tonearest". + rmDownward, ///< This corresponds to "fpround.downward". + rmUpward, ///< This corresponds to "fpround.upward". + rmTowardZero ///< This corresponds to "fpround.tozero". +}; + +/// Exception behavior used for floating point operations. +/// +/// Each of these values correspond to some metadata argument value of a +/// constrained floating point intrinsic. See the LLVM Language Reference Manual +/// for details. +enum ExceptionBehavior : uint8_t { + ebIgnore, ///< This corresponds to "fpexcept.ignore". + ebMayTrap, ///< This corresponds to "fpexcept.maytrap". + ebStrict ///< This corresponds to "fpexcept.strict". +}; + +} + +/// Returns a valid RoundingMode enumerator when given a string +/// that is valid as input in constrained intrinsic rounding mode +/// metadata. +Optional StrToRoundingMode(StringRef); + +/// For any RoundingMode enumerator, returns a string valid as input in +/// constrained intrinsic rounding mode metadata. +Optional RoundingModeToStr(fp::RoundingMode); + +/// Returns a valid ExceptionBehavior enumerator when given a string +/// valid as input in constrained intrinsic exception behavior metadata. +Optional StrToExceptionBehavior(StringRef); + +/// For any ExceptionBehavior enumerator, returns a string valid as +/// input in constrained intrinsic exception behavior metadata. +Optional ExceptionBehaviorToStr(fp::ExceptionBehavior); + +} +#endif diff --git a/llvm/include/llvm/IR/IRBuilder.h b/llvm/include/llvm/IR/IRBuilder.h --- a/llvm/include/llvm/IR/IRBuilder.h +++ b/llvm/include/llvm/IR/IRBuilder.h @@ -97,8 +97,8 @@ FastMathFlags FMF; bool IsFPConstrained; - ConstrainedFPIntrinsic::ExceptionBehavior DefaultConstrainedExcept; - ConstrainedFPIntrinsic::RoundingMode DefaultConstrainedRounding; + fp::ExceptionBehavior DefaultConstrainedExcept; + fp::RoundingMode DefaultConstrainedRounding; ArrayRef DefaultOperandBundles; @@ -106,8 +106,8 @@ IRBuilderBase(LLVMContext &context, MDNode *FPMathTag = nullptr, ArrayRef OpBundles = None) : Context(context), DefaultFPMathTag(FPMathTag), IsFPConstrained(false), - DefaultConstrainedExcept(ConstrainedFPIntrinsic::ebStrict), - DefaultConstrainedRounding(ConstrainedFPIntrinsic::rmDynamic), + DefaultConstrainedExcept(fp::ebStrict), + DefaultConstrainedRounding(fp::rmDynamic), DefaultOperandBundles(OpBundles) { ClearInsertionPoint(); } @@ -234,24 +234,22 @@ bool getIsFPConstrained() { return IsFPConstrained; } /// Set the exception handling to be used with constrained floating point - void setDefaultConstrainedExcept( - ConstrainedFPIntrinsic::ExceptionBehavior NewExcept) { + void setDefaultConstrainedExcept(fp::ExceptionBehavior NewExcept) { DefaultConstrainedExcept = NewExcept; } /// Set the rounding mode handling to be used with constrained floating point - void setDefaultConstrainedRounding( - ConstrainedFPIntrinsic::RoundingMode NewRounding) { + void setDefaultConstrainedRounding(fp::RoundingMode NewRounding) { DefaultConstrainedRounding = NewRounding; } /// Get the exception handling used with constrained floating point - ConstrainedFPIntrinsic::ExceptionBehavior getDefaultConstrainedExcept() { + fp::ExceptionBehavior getDefaultConstrainedExcept() { return DefaultConstrainedExcept; } /// Get the rounding mode handling used with constrained floating point - ConstrainedFPIntrinsic::RoundingMode getDefaultConstrainedRounding() { + fp::RoundingMode getDefaultConstrainedRounding() { return DefaultConstrainedRounding; } @@ -1097,32 +1095,26 @@ return (LC && RC) ? Insert(Folder.CreateBinOp(Opc, LC, RC), Name) : nullptr; } - Value *getConstrainedFPRounding( - Optional Rounding) { - ConstrainedFPIntrinsic::RoundingMode UseRounding = - DefaultConstrainedRounding; + Value *getConstrainedFPRounding(Optional Rounding) { + fp::RoundingMode UseRounding = DefaultConstrainedRounding; if (Rounding.hasValue()) UseRounding = Rounding.getValue(); - Optional RoundingStr = - ConstrainedFPIntrinsic::RoundingModeToStr(UseRounding); + Optional RoundingStr = RoundingModeToStr(UseRounding); assert(RoundingStr.hasValue() && "Garbage strict rounding mode!"); auto *RoundingMDS = MDString::get(Context, RoundingStr.getValue()); return MetadataAsValue::get(Context, RoundingMDS); } - Value *getConstrainedFPExcept( - Optional Except) { - ConstrainedFPIntrinsic::ExceptionBehavior UseExcept = - DefaultConstrainedExcept; + Value *getConstrainedFPExcept(Optional Except) { + fp::ExceptionBehavior UseExcept = DefaultConstrainedExcept; if (Except.hasValue()) UseExcept = Except.getValue(); - Optional ExceptStr = - ConstrainedFPIntrinsic::ExceptionBehaviorToStr(UseExcept); + Optional ExceptStr = ExceptionBehaviorToStr(UseExcept); assert(ExceptStr.hasValue() && "Garbage strict exception behavior!"); auto *ExceptMDS = MDString::get(Context, ExceptStr.getValue()); @@ -1483,8 +1475,8 @@ CallInst *CreateConstrainedFPBinOp( Intrinsic::ID ID, Value *L, Value *R, Instruction *FMFSource = nullptr, const Twine &Name = "", MDNode *FPMathTag = nullptr, - Optional Rounding = None, - Optional Except = None) { + Optional Rounding = None, + Optional Except = None) { Value *RoundingV = getConstrainedFPRounding(Rounding); Value *ExceptV = getConstrainedFPExcept(Except); @@ -2078,8 +2070,8 @@ Intrinsic::ID ID, Value *V, Type *DestTy, Instruction *FMFSource = nullptr, const Twine &Name = "", MDNode *FPMathTag = nullptr, - Optional Rounding = None, - Optional Except = None) { + Optional Rounding = None, + Optional Except = None) { Value *ExceptV = getConstrainedFPExcept(Except); FastMathFlags UseFMF = FMF; diff --git a/llvm/include/llvm/IR/IntrinsicInst.h b/llvm/include/llvm/IR/IntrinsicInst.h --- a/llvm/include/llvm/IR/IntrinsicInst.h +++ b/llvm/include/llvm/IR/IntrinsicInst.h @@ -25,6 +25,7 @@ #include "llvm/IR/Constants.h" #include "llvm/IR/DerivedTypes.h" +#include "llvm/IR/FPEnv.h" #include "llvm/IR/Function.h" #include "llvm/IR/GlobalVariable.h" #include "llvm/IR/Instructions.h" @@ -208,47 +209,10 @@ /// This is the common base class for constrained floating point intrinsics. class ConstrainedFPIntrinsic : public IntrinsicInst { public: - /// Specifies the rounding mode to be assumed. This is only used when - /// when constrained floating point is enabled. See the LLVM Language - /// Reference Manual for details. - enum RoundingMode : uint8_t { - rmDynamic, ///< This corresponds to "fpround.dynamic". - rmToNearest, ///< This corresponds to "fpround.tonearest". - rmDownward, ///< This corresponds to "fpround.downward". - rmUpward, ///< This corresponds to "fpround.upward". - rmTowardZero ///< This corresponds to "fpround.tozero". - }; - - /// Specifies the required exception behavior. This is only used when - /// when constrained floating point is used. See the LLVM Language - /// Reference Manual for details. - enum ExceptionBehavior : uint8_t { - ebIgnore, ///< This corresponds to "fpexcept.ignore". - ebMayTrap, ///< This corresponds to "fpexcept.maytrap". - ebStrict ///< This corresponds to "fpexcept.strict". - }; - bool isUnaryOp() const; bool isTernaryOp() const; - Optional getRoundingMode() const; - Optional getExceptionBehavior() const; - - /// Returns a valid RoundingMode enumerator when given a string - /// that is valid as input in constrained intrinsic rounding mode - /// metadata. - static Optional StrToRoundingMode(StringRef); - - /// For any RoundingMode enumerator, returns a string valid as input in - /// constrained intrinsic rounding mode metadata. - static Optional RoundingModeToStr(RoundingMode); - - /// Returns a valid ExceptionBehavior enumerator when given a string - /// valid as input in constrained intrinsic exception behavior metadata. - static Optional StrToExceptionBehavior(StringRef); - - /// For any ExceptionBehavior enumerator, returns a string valid as - /// input in constrained intrinsic exception behavior metadata. - static Optional ExceptionBehaviorToStr(ExceptionBehavior); + Optional getRoundingMode() const; + Optional getExceptionBehavior() const; // Methods for support type inquiry through isa, cast, and dyn_cast: static bool classof(const IntrinsicInst *I) { diff --git a/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp b/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp --- a/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp +++ b/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp @@ -7020,8 +7020,7 @@ SDVTList VTs = DAG.getVTList(ValueVTs); SDValue Result = DAG.getNode(Opcode, sdl, VTs, Opers); - if (FPI.getExceptionBehavior() != - ConstrainedFPIntrinsic::ExceptionBehavior::ebIgnore) { + if (FPI.getExceptionBehavior() != fp::ExceptionBehavior::ebIgnore) { SDNodeFlags Flags; Flags.setFPExcept(true); Result->setFlags(Flags); diff --git a/llvm/lib/IR/CMakeLists.txt b/llvm/lib/IR/CMakeLists.txt --- a/llvm/lib/IR/CMakeLists.txt +++ b/llvm/lib/IR/CMakeLists.txt @@ -22,6 +22,7 @@ DiagnosticInfo.cpp DiagnosticPrinter.cpp Dominators.cpp + FPEnv.cpp Function.cpp GVMaterializer.cpp Globals.cpp diff --git a/llvm/lib/IR/FPEnv.cpp b/llvm/lib/IR/FPEnv.cpp new file mode 100644 --- /dev/null +++ b/llvm/lib/IR/FPEnv.cpp @@ -0,0 +1,78 @@ +//===-- FPEnv.cpp ---- FP Environment -------------------------------------===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +/// @file +/// This file contains the implementations of entities that describe floating +/// point environment. +// +//===----------------------------------------------------------------------===// + +#include "llvm/ADT/StringSwitch.h" +#include "llvm/IR/FPEnv.h" + +namespace llvm { + +Optional StrToRoundingMode(StringRef RoundingArg) { + // For dynamic rounding mode, we use round to nearest but we will set the + // 'exact' SDNodeFlag so that the value will not be rounded. + return StringSwitch>(RoundingArg) + .Case("round.dynamic", fp::rmDynamic) + .Case("round.tonearest", fp::rmToNearest) + .Case("round.downward", fp::rmDownward) + .Case("round.upward", fp::rmUpward) + .Case("round.towardzero", fp::rmTowardZero) + .Default(None); +} + +Optional RoundingModeToStr(fp::RoundingMode UseRounding) { + Optional RoundingStr = None; + switch (UseRounding) { + case fp::rmDynamic: + RoundingStr = "round.dynamic"; + break; + case fp::rmToNearest: + RoundingStr = "round.tonearest"; + break; + case fp::rmDownward: + RoundingStr = "round.downward"; + break; + case fp::rmUpward: + RoundingStr = "round.upward"; + break; + case fp::rmTowardZero: + RoundingStr = "round.towardzero"; + break; + } + return RoundingStr; +} + +Optional StrToExceptionBehavior(StringRef ExceptionArg) { + return StringSwitch>(ExceptionArg) + .Case("fpexcept.ignore", fp::ebIgnore) + .Case("fpexcept.maytrap", fp::ebMayTrap) + .Case("fpexcept.strict", fp::ebStrict) + .Default(None); +} + +Optional ExceptionBehaviorToStr(fp::ExceptionBehavior UseExcept) { + Optional ExceptStr = None; + switch (UseExcept) { + case fp::ebStrict: + ExceptStr = "fpexcept.strict"; + break; + case fp::ebIgnore: + ExceptStr = "fpexcept.ignore"; + break; + case fp::ebMayTrap: + ExceptStr = "fpexcept.maytrap"; + break; + } + return ExceptStr; +} + +} \ No newline at end of file diff --git a/llvm/lib/IR/IntrinsicInst.cpp b/llvm/lib/IR/IntrinsicInst.cpp --- a/llvm/lib/IR/IntrinsicInst.cpp +++ b/llvm/lib/IR/IntrinsicInst.cpp @@ -102,8 +102,7 @@ return ConstantInt::get(Type::getInt64Ty(Context), 1); } -Optional -ConstrainedFPIntrinsic::getRoundingMode() const { +Optional ConstrainedFPIntrinsic::getRoundingMode() const { unsigned NumOperands = getNumArgOperands(); Metadata *MD = cast(getArgOperand(NumOperands - 2))->getMetadata(); @@ -112,43 +111,7 @@ return StrToRoundingMode(cast(MD)->getString()); } -Optional -ConstrainedFPIntrinsic::StrToRoundingMode(StringRef RoundingArg) { - // For dynamic rounding mode, we use round to nearest but we will set the - // 'exact' SDNodeFlag so that the value will not be rounded. - return StringSwitch>(RoundingArg) - .Case("round.dynamic", rmDynamic) - .Case("round.tonearest", rmToNearest) - .Case("round.downward", rmDownward) - .Case("round.upward", rmUpward) - .Case("round.towardzero", rmTowardZero) - .Default(None); -} - -Optional -ConstrainedFPIntrinsic::RoundingModeToStr(RoundingMode UseRounding) { - Optional RoundingStr = None; - switch (UseRounding) { - case ConstrainedFPIntrinsic::rmDynamic: - RoundingStr = "round.dynamic"; - break; - case ConstrainedFPIntrinsic::rmToNearest: - RoundingStr = "round.tonearest"; - break; - case ConstrainedFPIntrinsic::rmDownward: - RoundingStr = "round.downward"; - break; - case ConstrainedFPIntrinsic::rmUpward: - RoundingStr = "round.upward"; - break; - case ConstrainedFPIntrinsic::rmTowardZero: - RoundingStr = "round.towardzero"; - break; - } - return RoundingStr; -} - -Optional +Optional ConstrainedFPIntrinsic::getExceptionBehavior() const { unsigned NumOperands = getNumArgOperands(); Metadata *MD = @@ -158,32 +121,6 @@ return StrToExceptionBehavior(cast(MD)->getString()); } -Optional -ConstrainedFPIntrinsic::StrToExceptionBehavior(StringRef ExceptionArg) { - return StringSwitch>(ExceptionArg) - .Case("fpexcept.ignore", ebIgnore) - .Case("fpexcept.maytrap", ebMayTrap) - .Case("fpexcept.strict", ebStrict) - .Default(None); -} - -Optional -ConstrainedFPIntrinsic::ExceptionBehaviorToStr(ExceptionBehavior UseExcept) { - Optional ExceptStr = None; - switch (UseExcept) { - case ConstrainedFPIntrinsic::ebStrict: - ExceptStr = "fpexcept.strict"; - break; - case ConstrainedFPIntrinsic::ebIgnore: - ExceptStr = "fpexcept.ignore"; - break; - case ConstrainedFPIntrinsic::ebMayTrap: - ExceptStr = "fpexcept.maytrap"; - break; - } - return ExceptStr; -} - bool ConstrainedFPIntrinsic::isUnaryOp() const { switch (getIntrinsicID()) { default: diff --git a/llvm/unittests/IR/IRBuilderTest.cpp b/llvm/unittests/IR/IRBuilderTest.cpp --- a/llvm/unittests/IR/IRBuilderTest.cpp +++ b/llvm/unittests/IR/IRBuilderTest.cpp @@ -242,52 +242,52 @@ V = Builder.CreateFAdd(V, V); ASSERT_TRUE(isa(V)); auto *CII = cast(V); - ASSERT_TRUE(CII->getExceptionBehavior() == ConstrainedFPIntrinsic::ebStrict); - ASSERT_TRUE(CII->getRoundingMode() == ConstrainedFPIntrinsic::rmDynamic); + ASSERT_TRUE(CII->getExceptionBehavior() == fp::ebStrict); + ASSERT_TRUE(CII->getRoundingMode() == fp::rmDynamic); - Builder.setDefaultConstrainedExcept(ConstrainedFPIntrinsic::ebIgnore); - Builder.setDefaultConstrainedRounding(ConstrainedFPIntrinsic::rmUpward); + Builder.setDefaultConstrainedExcept(fp::ebIgnore); + Builder.setDefaultConstrainedRounding(fp::rmUpward); V = Builder.CreateFAdd(V, V); CII = cast(V); - ASSERT_TRUE(CII->getExceptionBehavior() == ConstrainedFPIntrinsic::ebIgnore); - ASSERT_TRUE(CII->getRoundingMode() == ConstrainedFPIntrinsic::rmUpward); + ASSERT_TRUE(CII->getExceptionBehavior() == fp::ebIgnore); + ASSERT_TRUE(CII->getRoundingMode() == fp::rmUpward); - Builder.setDefaultConstrainedExcept(ConstrainedFPIntrinsic::ebIgnore); - Builder.setDefaultConstrainedRounding(ConstrainedFPIntrinsic::rmToNearest); + Builder.setDefaultConstrainedExcept(fp::ebIgnore); + Builder.setDefaultConstrainedRounding(fp::rmToNearest); V = Builder.CreateFAdd(V, V); CII = cast(V); - ASSERT_TRUE(CII->getExceptionBehavior() == ConstrainedFPIntrinsic::ebIgnore); - ASSERT_TRUE(CII->getRoundingMode() == ConstrainedFPIntrinsic::rmToNearest); + ASSERT_TRUE(CII->getExceptionBehavior() == fp::ebIgnore); + ASSERT_TRUE(CII->getRoundingMode() == fp::rmToNearest); - Builder.setDefaultConstrainedExcept(ConstrainedFPIntrinsic::ebMayTrap); - Builder.setDefaultConstrainedRounding(ConstrainedFPIntrinsic::rmDownward); + Builder.setDefaultConstrainedExcept(fp::ebMayTrap); + Builder.setDefaultConstrainedRounding(fp::rmDownward); V = Builder.CreateFAdd(V, V); CII = cast(V); - ASSERT_TRUE(CII->getExceptionBehavior() == ConstrainedFPIntrinsic::ebMayTrap); - ASSERT_TRUE(CII->getRoundingMode() == ConstrainedFPIntrinsic::rmDownward); + ASSERT_TRUE(CII->getExceptionBehavior() == fp::ebMayTrap); + ASSERT_TRUE(CII->getRoundingMode() == fp::rmDownward); - Builder.setDefaultConstrainedExcept(ConstrainedFPIntrinsic::ebStrict); - Builder.setDefaultConstrainedRounding(ConstrainedFPIntrinsic::rmTowardZero); + Builder.setDefaultConstrainedExcept(fp::ebStrict); + Builder.setDefaultConstrainedRounding(fp::rmTowardZero); V = Builder.CreateFAdd(V, V); CII = cast(V); - ASSERT_TRUE(CII->getExceptionBehavior() == ConstrainedFPIntrinsic::ebStrict); - ASSERT_TRUE(CII->getRoundingMode() == ConstrainedFPIntrinsic::rmTowardZero); + ASSERT_TRUE(CII->getExceptionBehavior() == fp::ebStrict); + ASSERT_TRUE(CII->getRoundingMode() == fp::rmTowardZero); - Builder.setDefaultConstrainedExcept(ConstrainedFPIntrinsic::ebIgnore); - Builder.setDefaultConstrainedRounding(ConstrainedFPIntrinsic::rmDynamic); + Builder.setDefaultConstrainedExcept(fp::ebIgnore); + Builder.setDefaultConstrainedRounding(fp::rmDynamic); V = Builder.CreateFAdd(V, V); CII = cast(V); - ASSERT_TRUE(CII->getExceptionBehavior() == ConstrainedFPIntrinsic::ebIgnore); - ASSERT_TRUE(CII->getRoundingMode() == ConstrainedFPIntrinsic::rmDynamic); + ASSERT_TRUE(CII->getExceptionBehavior() == fp::ebIgnore); + ASSERT_TRUE(CII->getRoundingMode() == fp::rmDynamic); // Now override the defaults. Call = Builder.CreateConstrainedFPBinOp( Intrinsic::experimental_constrained_fadd, V, V, nullptr, "", nullptr, - ConstrainedFPIntrinsic::rmDownward, ConstrainedFPIntrinsic::ebMayTrap); + fp::rmDownward, fp::ebMayTrap); CII = cast(Call); EXPECT_EQ(CII->getIntrinsicID(), Intrinsic::experimental_constrained_fadd); - ASSERT_TRUE(CII->getExceptionBehavior() == ConstrainedFPIntrinsic::ebMayTrap); - ASSERT_TRUE(CII->getRoundingMode() == ConstrainedFPIntrinsic::rmDownward); + ASSERT_TRUE(CII->getExceptionBehavior() == fp::ebMayTrap); + ASSERT_TRUE(CII->getRoundingMode() == fp::rmDownward); Builder.CreateRetVoid(); EXPECT_FALSE(verifyModule(*M));