//===-- 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/IR/FPEnv.h" #include "llvm/ADT/StringSwitch.h" #include "llvm/IR/Instruction.h" #include "llvm/IR/IntrinsicInst.h" #include "llvm/IR/Intrinsics.h" namespace llvm { Optional convertStrToRoundingMode(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", RoundingMode::Dynamic) .Case("round.tonearest", RoundingMode::NearestTiesToEven) .Case("round.tonearestaway", RoundingMode::NearestTiesToAway) .Case("round.downward", RoundingMode::TowardNegative) .Case("round.upward", RoundingMode::TowardPositive) .Case("round.towardzero", RoundingMode::TowardZero) .Default(None); } Optional convertRoundingModeToStr(RoundingMode UseRounding) { Optional RoundingStr = None; switch (UseRounding) { case RoundingMode::Dynamic: RoundingStr = "round.dynamic"; break; case RoundingMode::NearestTiesToEven: RoundingStr = "round.tonearest"; break; case RoundingMode::NearestTiesToAway: RoundingStr = "round.tonearestaway"; break; case RoundingMode::TowardNegative: RoundingStr = "round.downward"; break; case RoundingMode::TowardPositive: RoundingStr = "round.upward"; break; case RoundingMode::TowardZero: RoundingStr = "round.towardzero"; break; default: break; } return RoundingStr; } Optional convertStrToExceptionBehavior(StringRef ExceptionArg) { return StringSwitch>(ExceptionArg) .Case("fpexcept.ignore", fp::ebIgnore) .Case("fpexcept.maytrap", fp::ebMayTrap) .Case("fpexcept.strict", fp::ebStrict) .Default(None); } Optional convertExceptionBehaviorToStr(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; } Intrinsic::ID getConstrainedIntrinsicID(const Instruction &Instr) { Intrinsic::ID IID = Intrinsic::not_intrinsic; switch (Instr.getOpcode()) { case Instruction::FCmp: // Unlike other instructions FCmp can be mapped to one of two intrinsic // functions. We choose the non-signaling variant. IID = Intrinsic::experimental_constrained_fcmp; break; // Instructions #define INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC) \ case Instruction::NAME: \ IID = Intrinsic::INTRINSIC; \ break; #define FUNCTION(NAME, NARG, ROUND_MODE, INTRINSIC) #define CMP_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) #include "llvm/IR/ConstrainedOps.def" // Intrinsic calls. case Instruction::Call: if (auto *IntrinCall = dyn_cast(&Instr)) { switch (IntrinCall->getIntrinsicID()) { #define FUNCTION(NAME, NARG, ROUND_MODE, INTRINSIC) \ case Intrinsic::NAME: \ IID = Intrinsic::INTRINSIC; \ break; #define INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC) #define CMP_INSTRUCTION(NAME, NARG, ROUND_MODE, INTRINSIC, DAGN) #include "llvm/IR/ConstrainedOps.def" default: break; } } break; default: break; } return IID; } } // namespace llvm