jansvoboda11 created this revision. jansvoboda11 added reviewers: dexonsmith, Bigcheese. Herald added a subscriber: dang. jansvoboda11 requested review of this revision. Herald added projects: clang, LLVM. Herald added subscribers: llvm-commits, cfe-commits.
We cannot be sure whether a flag is CC1Option inside the definition of BoolOption. This occurs when CC1Option is added like so: let Flags = [CC1Option] in { defm xxx : BoolOption<...>; } because the `let yyy in {...}` is applied after all the xxx record is fully declared. Repository: rG LLVM Github Monorepo https://reviews.llvm.org/D93008 Files: clang/include/clang/Driver/Options.td clang/lib/Frontend/CompilerInvocation.cpp clang/unittests/Frontend/CompilerInvocationTest.cpp llvm/include/llvm/Option/OptParser.td
Index: llvm/include/llvm/Option/OptParser.td =================================================================== --- llvm/include/llvm/Option/OptParser.td +++ llvm/include/llvm/Option/OptParser.td @@ -173,10 +173,10 @@ // Marshalling info for booleans. Applied to the flag setting keypath to false. class MarshallingInfoBooleanFlag<code keypath, code defaultvalue, code value, code name, - code other_value, code other_name, string other_spelling> + code other_value, code other_name> : MarshallingInfoFlag<keypath, defaultvalue> { code Normalizer = "makeBooleanOptionNormalizer("#value#", "#other_value#", OPT_"#other_name#")"; - code Denormalizer = "makeBooleanOptionDenormalizer("#value#", \""#other_spelling#"\")"; + code Denormalizer = "makeBooleanOptionDenormalizer("#value#")"; } // Mixins for additional marshalling attributes. Index: clang/unittests/Frontend/CompilerInvocationTest.cpp =================================================================== --- clang/unittests/Frontend/CompilerInvocationTest.cpp +++ clang/unittests/Frontend/CompilerInvocationTest.cpp @@ -233,6 +233,55 @@ Not(Contains(StrEq("-fexperimental-new-pass-manager")))); } +// Boolean option that gets the CC1Option flag from a let statement (which +// happens applied **after** the record is defined): +// +// let Flags = [CC1Option] in { +// defm option : BoolOption<...>; +// } + +TEST_F(CommandLineTest, BoolOptionCC1ViaLetPresentNone) { + const char *Args[] = {""}; + + CompilerInvocation::CreateFromArgs(Invocation, Args, *Diags); + + ASSERT_FALSE(Diags->hasErrorOccurred()); + ASSERT_FALSE(Invocation.getCodeGenOpts().DebugPassManager); + + Invocation.generateCC1CommandLine(GeneratedArgs, *this); + + ASSERT_THAT(GeneratedArgs, Not(Contains(StrEq("-fdebug-pass-manager")))); + ASSERT_THAT(GeneratedArgs, Not(Contains(StrEq("-fno-debug-pass-manager")))); +} + +TEST_F(CommandLineTest, BoolOptionCC1ViaLetPresentPos) { + const char *Args[] = {"-fdebug-pass-manager"}; + + CompilerInvocation::CreateFromArgs(Invocation, Args, *Diags); + + ASSERT_FALSE(Diags->hasErrorOccurred()); + ASSERT_TRUE(Invocation.getCodeGenOpts().DebugPassManager); + + Invocation.generateCC1CommandLine(GeneratedArgs, *this); + + ASSERT_THAT(GeneratedArgs, Contains(StrEq("-fdebug-pass-manager"))); + ASSERT_THAT(GeneratedArgs, Not(Contains(StrEq("-fno-debug-pass-manager")))); +} + +TEST_F(CommandLineTest, BoolOptionCC1ViaLetPresentNeg) { + const char *Args[] = {"-fno-debug-pass-manager"}; + + CompilerInvocation::CreateFromArgs(Invocation, Args, *Diags); + + ASSERT_FALSE(Diags->hasErrorOccurred()); + ASSERT_FALSE(Invocation.getCodeGenOpts().DebugPassManager); + + Invocation.generateCC1CommandLine(GeneratedArgs, *this); + + ASSERT_THAT(GeneratedArgs, Not(Contains(StrEq("-fno-debug-pass-manager")))); + ASSERT_THAT(GeneratedArgs, Not(Contains(StrEq("-fdebug-pass-manager")))); +} + TEST_F(CommandLineTest, CanGenerateCC1CommandLineFlag) { const char *Args[] = {"-fmodules-strict-context-hash"}; Index: clang/lib/Frontend/CompilerInvocation.cpp =================================================================== --- clang/lib/Frontend/CompilerInvocation.cpp +++ clang/lib/Frontend/CompilerInvocation.cpp @@ -197,12 +197,11 @@ }; } -static auto makeBooleanOptionDenormalizer(bool Value, - const char *OtherSpelling) { - return [Value, OtherSpelling]( - SmallVectorImpl<const char *> &Args, const char *Spelling, - CompilerInvocation::StringAllocator, unsigned, bool KeyPath) { - Args.push_back(KeyPath == Value ? Spelling : OtherSpelling); +static auto makeBooleanOptionDenormalizer(bool Value) { + return [Value](SmallVectorImpl<const char *> &Args, const char *Spelling, + CompilerInvocation::StringAllocator, unsigned, bool KeyPath) { + if (KeyPath == Value) + Args.push_back(Spelling); }; } @@ -853,10 +852,6 @@ } } - Opts.DebugPassManager = - Args.hasFlag(OPT_fdebug_pass_manager, OPT_fno_debug_pass_manager, - /* Default */ false); - if (Arg *A = Args.getLastArg(OPT_fveclib)) { StringRef Name = A->getValue(); if (Name == "Accelerate") Index: clang/include/clang/Driver/Options.td =================================================================== --- clang/include/clang/Driver/Options.td +++ clang/include/clang/Driver/Options.td @@ -325,10 +325,6 @@ string Spelling = prefix#!cond(flag.Polarity.Value : "", true : "no-")#spelling; - // Does the flag have CC1Option? - bit IsCC1 = !not(!empty(!filter(opt_flag, flag.OptionFlags, - !eq(opt_flag, CC1Option)))); - // Can the flag be implied by another flag? bit CanBeImplied = !not(!empty(flag.ImpliedBy)); @@ -336,18 +332,14 @@ code ValueAsCode = !cond(flag.Value : "true", true: "false"); } -// Creates simple flag record. -class BoolOptionFlag<FlagDefExpanded flag, code keypath, Default default> - : Flag<["-"], flag.Spelling>, Flags<flag.OptionFlags>, HelpText<flag.Help> {} - -// Creates marshalled flag record. -class CC1BoolOptionFlag<FlagDefExpanded flag, FlagDefExpanded other, - code keypath, Default default> +// Creates record with a marshalled flag. +class BoolOptionFlag<FlagDefExpanded flag, FlagDefExpanded other, + FlagDefExpanded implied, code keypath, Default default> : Flag<["-"], flag.Spelling>, Flags<flag.OptionFlags>, HelpText<flag.Help>, MarshallingInfoBooleanFlag<keypath, default.Value, flag.ValueAsCode, flag.RecordName, other.ValueAsCode, - other.RecordName, other.Spelling>, - ImpliedByAnyOf<flag.ImpliedBy, flag.ValueAsCode> {} + other.RecordName>, + ImpliedByAnyOf<implied.ImpliedBy, implied.ValueAsCode> {} // Generates TableGen records for two command line flags that control the same // keypath via the marshalling infrastructure. @@ -370,17 +362,10 @@ // TODO: Assert that the flags have different value. // TODO: Assert that only one of the flags can be implied. - if flag1.IsCC1 then { - def flag1.RecordName : CC1BoolOptionFlag<flag1, flag2, keypath, default>; - } else { - def flag1.RecordName : BoolOptionFlag<flag1, keypath, default>; - } - - if flag2.IsCC1 then { - def flag2.RecordName : CC1BoolOptionFlag<flag2, flag1, keypath, default>; - } else { - def flag2.RecordName : BoolOptionFlag<flag2, keypath, default>; - } + defvar implied = !cond(flag1.CanBeImplied: flag1, true: flag2); + + def flag1.RecordName : BoolOptionFlag<flag1, flag2, implied, keypath, default>; + def flag2.RecordName : BoolOptionFlag<flag2, flag1, implied, keypath, default>; } //===----------------------------------------------------------------------===// @@ -4296,10 +4281,11 @@ def flto_unit: Flag<["-"], "flto-unit">, HelpText<"Emit IR to support LTO unit features (CFI, whole program vtable opt)">; def fno_lto_unit: Flag<["-"], "fno-lto-unit">; -def fdebug_pass_manager : Flag<["-"], "fdebug-pass-manager">, - HelpText<"Prints debug information for the new pass manager">; -def fno_debug_pass_manager : Flag<["-"], "fno-debug-pass-manager">, - HelpText<"Disables debug printing for the new pass manager">; +defm debug_pass_manager : BoolOption<"debug-pass-manager", + "CodeGenOpts.DebugPassManager", DefaultsToFalse, + ChangedBy<PosFlag, [], "Prints debug information for the new pass manager">, + ResetBy<NegFlag, [], "Disables debug printing for the new pass manager">, + BothFlags<[]>, "f">; def fexperimental_debug_variable_locations : Flag<["-"], "fexperimental-debug-variable-locations">, HelpText<"Use experimental new value-tracking variable locations">;
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits