forked from OSchip/llvm-project
Add -ffp-contract = { fast | on | off } command line option support.
This flag sets the 'fp-contract' mode, which controls the formation of fused floating point operations. Available modes are: - Fast: Form fused operations anywhere. - On: Form fused operations where allowed by FP_CONTRACT. This is the default mode. - Off: Don't form fused operations (in future this may be relaxed to forming fused operations where it can be proved that the result won't be affected). Currently clang doesn't support the FP_CONTRACT pragma, so the 'On' and 'Off' modes are equivalent. llvm-svn: 159794
This commit is contained in:
parent
c7ac1bb94c
commit
aa53b936ec
|
@ -148,6 +148,7 @@ ENUM_LANGOPT(StackProtector, StackProtectorMode, 2, SSPOff,
|
||||||
"stack protector mode")
|
"stack protector mode")
|
||||||
ENUM_LANGOPT(SignedOverflowBehavior, SignedOverflowBehaviorTy, 2, SOB_Undefined,
|
ENUM_LANGOPT(SignedOverflowBehavior, SignedOverflowBehaviorTy, 2, SOB_Undefined,
|
||||||
"signed integer overflow handling")
|
"signed integer overflow handling")
|
||||||
|
ENUM_LANGOPT(FPContractMode, FPContractModeTy, 2, FPC_On, "FP_CONTRACT mode")
|
||||||
|
|
||||||
BENIGN_LANGOPT(InstantiationDepth, 32, 512,
|
BENIGN_LANGOPT(InstantiationDepth, 32, 512,
|
||||||
"maximum template instantiation depth")
|
"maximum template instantiation depth")
|
||||||
|
|
|
@ -56,6 +56,12 @@ public:
|
||||||
SOB_Trapping // -ftrapv
|
SOB_Trapping // -ftrapv
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum FPContractModeKind {
|
||||||
|
FPC_Off, // Form fused FP ops only where result will not be affected.
|
||||||
|
FPC_On, // Form fused FP ops according to FP_CONTRACT rules.
|
||||||
|
FPC_Fast // Aggressively fuse FP ops (E.g. FMA).
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
clang::ObjCRuntime ObjCRuntime;
|
clang::ObjCRuntime ObjCRuntime;
|
||||||
|
|
||||||
|
|
|
@ -410,6 +410,9 @@ def fhonor_infinites : Flag<"-fhonor-infinites">, Alias<fhonor_infinities>;
|
||||||
def fno_honor_infinites : Flag<"-fno-honor-infinites">, Alias<fno_honor_infinities>;
|
def fno_honor_infinites : Flag<"-fno-honor-infinites">, Alias<fno_honor_infinities>;
|
||||||
def ftrapping_math : Flag<"-ftrapping-math">, Group<f_Group>;
|
def ftrapping_math : Flag<"-ftrapping-math">, Group<f_Group>;
|
||||||
def fno_trapping_math : Flag<"-fno-trapping-math">, Group<f_Group>;
|
def fno_trapping_math : Flag<"-fno-trapping-math">, Group<f_Group>;
|
||||||
|
def ffp_contract : Joined<"-ffp-contract=">, Group<f_Group>,
|
||||||
|
Flags<[CC1Option]>, HelpText<"Form fused FP ops (e.g. FMAs): fast (everywhere)"
|
||||||
|
" | on (according to FP_CONTRACT pragma, default) | off (never fuse)">;
|
||||||
|
|
||||||
def ffor_scope : Flag<"-ffor-scope">, Group<f_Group>;
|
def ffor_scope : Flag<"-ffor-scope">, Group<f_Group>;
|
||||||
def fno_for_scope : Flag<"-fno-for-scope">, Group<f_Group>;
|
def fno_for_scope : Flag<"-fno-for-scope">, Group<f_Group>;
|
||||||
|
|
|
@ -351,6 +351,19 @@ bool EmitAssemblyHelper::AddEmitPasses(BackendAction Action,
|
||||||
Options.FloatABIType = llvm::FloatABI::Default;
|
Options.FloatABIType = llvm::FloatABI::Default;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Set FP fusion mode.
|
||||||
|
switch (LangOpts.getFPContractMode()) {
|
||||||
|
case LangOptions::FPC_Off:
|
||||||
|
Options.AllowFPOpFusion = llvm::FPOpFusion::Strict;
|
||||||
|
break;
|
||||||
|
case LangOptions::FPC_On:
|
||||||
|
Options.AllowFPOpFusion = llvm::FPOpFusion::Standard;
|
||||||
|
break;
|
||||||
|
case LangOptions::FPC_Fast:
|
||||||
|
Options.AllowFPOpFusion = llvm::FPOpFusion::Fast;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
Options.LessPreciseFPMADOption = CodeGenOpts.LessPreciseFPMAD;
|
Options.LessPreciseFPMADOption = CodeGenOpts.LessPreciseFPMAD;
|
||||||
Options.NoInfsFPMath = CodeGenOpts.NoInfsFPMath;
|
Options.NoInfsFPMath = CodeGenOpts.NoInfsFPMath;
|
||||||
Options.NoNaNsFPMath = CodeGenOpts.NoNaNsFPMath;
|
Options.NoNaNsFPMath = CodeGenOpts.NoNaNsFPMath;
|
||||||
|
|
|
@ -1805,6 +1805,24 @@ void Clang::ConstructJob(Compilation &C, const JobAction &JA,
|
||||||
!TrappingMath)
|
!TrappingMath)
|
||||||
CmdArgs.push_back("-menable-unsafe-fp-math");
|
CmdArgs.push_back("-menable-unsafe-fp-math");
|
||||||
|
|
||||||
|
|
||||||
|
// Validate and pass through -fp-contract option.
|
||||||
|
if (Arg *A = Args.getLastArg(options::OPT_ffast_math,
|
||||||
|
options::OPT_ffp_contract)) {
|
||||||
|
if (A->getOption().getID() == options::OPT_ffp_contract) {
|
||||||
|
StringRef Val = A->getValue(Args);
|
||||||
|
if (Val == "fast" || Val == "on" || Val == "off") {
|
||||||
|
CmdArgs.push_back(Args.MakeArgString("-ffp-contract=" + Val));
|
||||||
|
} else {
|
||||||
|
D.Diag(diag::err_drv_unsupported_option_argument)
|
||||||
|
<< A->getOption().getName() << Val;
|
||||||
|
}
|
||||||
|
} else { // A is OPT_ffast_math
|
||||||
|
// If fast-math is set then set the fp-contract mode to fast.
|
||||||
|
CmdArgs.push_back(Args.MakeArgString("-ffp-contract=fast"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// We separately look for the '-ffast-math' flag, and if we find it, tell the
|
// We separately look for the '-ffast-math' flag, and if we find it, tell the
|
||||||
// frontend to provide the appropriate preprocessor macros. This is distinct
|
// frontend to provide the appropriate preprocessor macros. This is distinct
|
||||||
// from enabling any optimizations as it induces a language change which must
|
// from enabling any optimizations as it induces a language change which must
|
||||||
|
|
|
@ -759,6 +759,11 @@ static void LangOptsToArgs(const LangOptions &Opts, ToArgsList &Res) {
|
||||||
Res.push_back("-ftrapv-handler", Opts.OverflowHandler);
|
Res.push_back("-ftrapv-handler", Opts.OverflowHandler);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
switch (Opts.getFPContractMode()) {
|
||||||
|
case LangOptions::FPC_Off: Res.push_back("-ffp-contract=off"); break;
|
||||||
|
case LangOptions::FPC_On: Res.push_back("-ffp-contract=on"); break;
|
||||||
|
case LangOptions::FPC_Fast: Res.push_back("-ffp-contract=fast"); break;
|
||||||
|
}
|
||||||
if (Opts.HeinousExtensions)
|
if (Opts.HeinousExtensions)
|
||||||
Res.push_back("-fheinous-gnu-extensions");
|
Res.push_back("-fheinous-gnu-extensions");
|
||||||
// Optimize is implicit.
|
// Optimize is implicit.
|
||||||
|
@ -1975,6 +1980,18 @@ static void ParseLangArgs(LangOptions &Opts, ArgList &Args, InputKind IK,
|
||||||
Diags.Report(diag::err_drv_invalid_value)
|
Diags.Report(diag::err_drv_invalid_value)
|
||||||
<< Args.getLastArg(OPT_fvisibility)->getAsString(Args) << Vis;
|
<< Args.getLastArg(OPT_fvisibility)->getAsString(Args) << Vis;
|
||||||
|
|
||||||
|
if (Arg *A = Args.getLastArg(OPT_ffp_contract)) {
|
||||||
|
StringRef Val = A->getValue(Args);
|
||||||
|
if (Val == "fast")
|
||||||
|
Opts.setFPContractMode(LangOptions::FPC_Fast);
|
||||||
|
else if (Val == "on")
|
||||||
|
Opts.setFPContractMode(LangOptions::FPC_On);
|
||||||
|
else if (Val == "off")
|
||||||
|
Opts.setFPContractMode(LangOptions::FPC_Off);
|
||||||
|
else
|
||||||
|
Diags.Report(diag::err_drv_invalid_value) << A->getAsString(Args) << Val;
|
||||||
|
}
|
||||||
|
|
||||||
if (Args.hasArg(OPT_fvisibility_inlines_hidden))
|
if (Args.hasArg(OPT_fvisibility_inlines_hidden))
|
||||||
Opts.InlineVisibilityHidden = 1;
|
Opts.InlineVisibilityHidden = 1;
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
// RUN: %clang_cc1 -O3 -ffp-contract=fast -triple=powerpc-apple-darwin10 -S -o - %s | FileCheck %s
|
||||||
|
|
||||||
|
float fma_test1(float a, float b, float c) {
|
||||||
|
// CHECK: fmadds
|
||||||
|
float x = a * b;
|
||||||
|
float y = x + c;
|
||||||
|
return y;
|
||||||
|
}
|
|
@ -29,3 +29,9 @@
|
||||||
// RUN: %clang -### -c -Wdeprecated %s 2>&1 | FileCheck -check-prefix=DEPRECATED-OFF-CHECK %s
|
// RUN: %clang -### -c -Wdeprecated %s 2>&1 | FileCheck -check-prefix=DEPRECATED-OFF-CHECK %s
|
||||||
// DEPRECATED-ON-CHECK: -fdeprecated-macro
|
// DEPRECATED-ON-CHECK: -fdeprecated-macro
|
||||||
// DEPRECATED-OFF-CHECK-NOT: -fdeprecated-macro
|
// DEPRECATED-OFF-CHECK-NOT: -fdeprecated-macro
|
||||||
|
|
||||||
|
// RUN: %clang -### -S -ffp-contract=fast %s 2>&1 | FileCheck -check-prefix=FP-CONTRACT-FAST-CHECK %s
|
||||||
|
// RUN: %clang -### -S -ffast-math %s 2>&1 | FileCheck -check-prefix=FP-CONTRACT-FAST-CHECK %s
|
||||||
|
// RUN: %clang -### -S -ffp-contract=off %s 2>&1 | FileCheck -check-prefix=FP-CONTRACT-OFF-CHECK %s
|
||||||
|
// FP-CONTRACT-FAST-CHECK: -ffp-contract=fast
|
||||||
|
// FP-CONTRACT-OFF-CHECK: -ffp-contract=off
|
||||||
|
|
Loading…
Reference in New Issue