diff --git a/clang/include/clang/Driver/Options.td b/clang/include/clang/Driver/Options.td --- a/clang/include/clang/Driver/Options.td +++ b/clang/include/clang/Driver/Options.td @@ -262,7 +262,7 @@ class FileSystemOpts : KeyPathAndMacro<"FileSystemOpts.", base> {} class AnalyzerOpts - : KeyPathAndMacro<"AnalyzerOpts->", base> {} + : KeyPathAndMacro<"AnalyzerOpts->", base, "ANALYZER_"> {} class MigratorOpts : KeyPathAndMacro<"MigratorOpts.", base> {} diff --git a/clang/lib/Frontend/CompilerInvocation.cpp b/clang/lib/Frontend/CompilerInvocation.cpp --- a/clang/lib/Frontend/CompilerInvocation.cpp +++ b/clang/lib/Frontend/CompilerInvocation.cpp @@ -727,9 +727,108 @@ Funcs.insert(Funcs.end(), Values.begin(), Values.end()); } +static void GenerateAnalyzerArgs(AnalyzerOptions &Opts, + SmallVectorImpl &Args, + CompilerInvocation::StringAllocator SA) { + const AnalyzerOptions *AnalyzerOpts = &Opts; + +#define ANALYZER_OPTION_WITH_MARSHALLING(...) \ + GENERATE_OPTION_WITH_MARSHALLING(Args, SA, NO_PREFIX, __VA_ARGS__) +#include "clang/Driver/Options.inc" +#undef ANALYZER_OPTION_WITH_MARSHALLING + + if (Opts.AnalysisStoreOpt != RegionStoreModel) { + switch (Opts.AnalysisStoreOpt) { +#define ANALYSIS_STORE(NAME, CMDFLAG, DESC, CREATFN) \ + case NAME##Model: \ + GenerateArg(Args, OPT_analyzer_store, CMDFLAG, SA); \ + break; +#include "clang/StaticAnalyzer/Core/Analyses.def" + default: + llvm_unreachable("Tried to generate unknown analysis store."); + } + } + + if (Opts.AnalysisConstraintsOpt != RangeConstraintsModel) { + switch (Opts.AnalysisConstraintsOpt) { +#define ANALYSIS_CONSTRAINTS(NAME, CMDFLAG, DESC, CREATFN) \ + case NAME##Model: \ + GenerateArg(Args, OPT_analyzer_constraints, CMDFLAG, SA); \ + break; +#include "clang/StaticAnalyzer/Core/Analyses.def" + default: + llvm_unreachable("Tried to generate unknown analysis constraint."); + } + } + + if (Opts.AnalysisDiagOpt != PD_HTML) { + switch (Opts.AnalysisDiagOpt) { +#define ANALYSIS_DIAGNOSTICS(NAME, CMDFLAG, DESC, CREATFN) \ + case PD_##NAME: \ + GenerateArg(Args, OPT_analyzer_output, CMDFLAG, SA); \ + break; +#include "clang/StaticAnalyzer/Core/Analyses.def" + default: + llvm_unreachable("Tried to generate unknown analysis diagnostic client."); + } + } + + if (Opts.AnalysisPurgeOpt != PurgeStmt) { + switch (Opts.AnalysisPurgeOpt) { +#define ANALYSIS_PURGE(NAME, CMDFLAG, DESC) \ + case NAME: \ + GenerateArg(Args, OPT_analyzer_purge, CMDFLAG, SA); \ + break; +#include "clang/StaticAnalyzer/Core/Analyses.def" + default: + llvm_unreachable("Tried to generate unknown analysis purge mode."); + } + } + + if (Opts.InliningMode != NoRedundancy) { + switch (Opts.InliningMode) { +#define ANALYSIS_INLINING_MODE(NAME, CMDFLAG, DESC) \ + case NAME: \ + GenerateArg(Args, OPT_analyzer_inlining_mode, CMDFLAG, SA); \ + break; +#include "clang/StaticAnalyzer/Core/Analyses.def" + default: + llvm_unreachable("Tried to generate unknown analysis inlining mode."); + } + } + + for (const auto &CP : Opts.CheckersAndPackages) { + OptSpecifier Opt = + CP.second ? OPT_analyzer_checker : OPT_analyzer_disable_checker; + GenerateArg(Args, Opt, CP.first, SA); + } + + AnalyzerOptions ConfigOpts; + parseAnalyzerConfigs(ConfigOpts, nullptr); + + for (const auto &C : Opts.Config) { + // Don't generate anything that came from parseAnalyzerConfigs. It would be + // redundant and may not be valid on the command line. + auto Entry = ConfigOpts.Config.find(C.first); + if (Entry != ConfigOpts.Config.end() && Entry->second == C.second) + continue; + + GenerateArg(Args, OPT_analyzer_config, C.first + "=" + C.second, SA); + } + + // Nothing to generate for FullCompilerInvocation. +} + static bool ParseAnalyzerArgs(AnalyzerOptions &Opts, ArgList &Args, DiagnosticsEngine &Diags) { + AnalyzerOptions *AnalyzerOpts = &Opts; bool Success = true; + +#define ANALYZER_OPTION_WITH_MARSHALLING(...) \ + PARSE_OPTION_WITH_MARSHALLING(Args, Diags, Success, NO_PREFIX, __VA_ARGS__) +#include "clang/Driver/Options.inc" +#undef ANALYZER_OPTION_WITH_MARSHALLING + if (Arg *A = Args.getLastArg(OPT_analyzer_store)) { StringRef Name = A->getValue(); AnalysisStores Value = llvm::StringSwitch(Name) @@ -853,8 +952,10 @@ // TODO: Check checker options too, possibly in CheckerRegistry. // Leave unknown non-checker configs unclaimed. if (!key.contains(":") && Opts.isUnknownAnalyzerConfig(key)) { - if (Opts.ShouldEmitErrorsOnInvalidConfigValue) + if (Opts.ShouldEmitErrorsOnInvalidConfigValue) { Diags.Report(diag::err_analyzer_config_unknown) << key; + Success = false; + } continue; } @@ -879,6 +980,28 @@ return Success; } +static bool ParseAnalyzerArgs(CompilerInvocation &Res, AnalyzerOptions &Opts, + ArgList &Args, DiagnosticsEngine &Diags) { + auto Temp = IntrusiveRefCntPtr(new AnalyzerOptions()); + + auto Parse = [](CompilerInvocation &Res, ArgList &Args, + DiagnosticsEngine &Diags) { + return ::ParseAnalyzerArgs(*Res.getAnalyzerOpts(), Args, Diags); + }; + + auto Generate = [](CompilerInvocation &Res, + SmallVectorImpl &Args, + CompilerInvocation::StringAllocator SA) { + GenerateAnalyzerArgs(*Res.getAnalyzerOpts(), Args, SA); + }; + + auto Swap = [&Temp](CompilerInvocation &Res) { + Res.getAnalyzerOpts().swap(Temp); + }; + + return RoundTrip(Parse, Generate, Swap, Res, Args, Diags, "AnalyzerOptions"); +} + static StringRef getStringOption(AnalyzerOptions::ConfigTable &Config, StringRef OptionName, StringRef DefaultVal) { return Config.insert({std::string(OptionName), std::string(DefaultVal)}) @@ -3035,7 +3158,7 @@ Success &= Res.parseSimpleArgs(Args, Diags); - Success &= ParseAnalyzerArgs(*Res.getAnalyzerOpts(), Args, Diags); + Success &= ParseAnalyzerArgs(Res, *Res.getAnalyzerOpts(), Args, Diags); ParseDependencyOutputArgs(Res.getDependencyOutputOpts(), Args); if (!Res.getDependencyOutputOpts().OutputFile.empty() && Res.getDependencyOutputOpts().Targets.empty()) { @@ -3255,6 +3378,7 @@ #undef DIAG_OPTION_WITH_MARSHALLING #undef OPTION_WITH_MARSHALLING + GenerateAnalyzerArgs(*AnalyzerOpts, Args, SA); GenerateHeaderSearchArgs(*HeaderSearchOpts, Args, SA); GenerateLangArgs(*LangOpts, Args, SA); }