forked from OSchip/llvm-project
[Driver] Fix driver support for color diagnostics
Diagnostics that happen during driver time do not have color output support unless -fcolor-diagonostic is explicitly passed into the driver. This is not a problem for cc1 since dianostic arguments are properly handled and color is enabled by default if the terminal supports it. Make the driver behave like CC1. There are tests that already check for these flags, but for the color itself there's no sensible way to test it. Differential Revision: http://reviews.llvm.org/D20404 rdar://problem/26290980 llvm-svn: 271042
This commit is contained in:
parent
88a7892a07
commit
681d717219
|
@ -47,7 +47,8 @@ class DiagnosticsEngine;
|
||||||
/// When errors are encountered, return false and, if Diags is non-null,
|
/// When errors are encountered, return false and, if Diags is non-null,
|
||||||
/// report the error(s).
|
/// report the error(s).
|
||||||
bool ParseDiagnosticArgs(DiagnosticOptions &Opts, llvm::opt::ArgList &Args,
|
bool ParseDiagnosticArgs(DiagnosticOptions &Opts, llvm::opt::ArgList &Args,
|
||||||
DiagnosticsEngine *Diags = nullptr);
|
DiagnosticsEngine *Diags = nullptr,
|
||||||
|
bool DefaultDiagColor = true);
|
||||||
|
|
||||||
class CompilerInvocationBase : public RefCountedBase<CompilerInvocation> {
|
class CompilerInvocationBase : public RefCountedBase<CompilerInvocation> {
|
||||||
void operator=(const CompilerInvocationBase &) = delete;
|
void operator=(const CompilerInvocationBase &) = delete;
|
||||||
|
|
|
@ -5589,43 +5589,27 @@ void Clang::ConstructJob(Compilation &C, const JobAction &JA,
|
||||||
CmdArgs.push_back("-fno-diagnostics-show-note-include-stack");
|
CmdArgs.push_back("-fno-diagnostics-show-note-include-stack");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Color diagnostics are the default, unless the terminal doesn't support
|
// Color diagnostics are parsed by the driver directly from argv
|
||||||
// them.
|
// and later re-parsed to construct this job; claim any possible
|
||||||
// Support both clang's -f[no-]color-diagnostics and gcc's
|
// color diagnostic here to avoid warn_drv_unused_argument and
|
||||||
// -f[no-]diagnostics-colors[=never|always|auto].
|
// diagnose bad OPT_fdiagnostics_color_EQ values.
|
||||||
enum { Colors_On, Colors_Off, Colors_Auto } ShowColors = Colors_Auto;
|
for (Arg *A : Args) {
|
||||||
for (const auto &Arg : Args) {
|
const Option &O = A->getOption();
|
||||||
const Option &O = Arg->getOption();
|
|
||||||
if (!O.matches(options::OPT_fcolor_diagnostics) &&
|
if (!O.matches(options::OPT_fcolor_diagnostics) &&
|
||||||
!O.matches(options::OPT_fdiagnostics_color) &&
|
!O.matches(options::OPT_fdiagnostics_color) &&
|
||||||
!O.matches(options::OPT_fno_color_diagnostics) &&
|
!O.matches(options::OPT_fno_color_diagnostics) &&
|
||||||
!O.matches(options::OPT_fno_diagnostics_color) &&
|
!O.matches(options::OPT_fno_diagnostics_color) &&
|
||||||
!O.matches(options::OPT_fdiagnostics_color_EQ))
|
!O.matches(options::OPT_fdiagnostics_color_EQ))
|
||||||
continue;
|
continue;
|
||||||
|
if (O.matches(options::OPT_fdiagnostics_color_EQ)) {
|
||||||
Arg->claim();
|
StringRef Value(A->getValue());
|
||||||
if (O.matches(options::OPT_fcolor_diagnostics) ||
|
if (Value != "always" && Value != "never" && Value != "auto")
|
||||||
O.matches(options::OPT_fdiagnostics_color)) {
|
|
||||||
ShowColors = Colors_On;
|
|
||||||
} else if (O.matches(options::OPT_fno_color_diagnostics) ||
|
|
||||||
O.matches(options::OPT_fno_diagnostics_color)) {
|
|
||||||
ShowColors = Colors_Off;
|
|
||||||
} else {
|
|
||||||
assert(O.matches(options::OPT_fdiagnostics_color_EQ));
|
|
||||||
StringRef value(Arg->getValue());
|
|
||||||
if (value == "always")
|
|
||||||
ShowColors = Colors_On;
|
|
||||||
else if (value == "never")
|
|
||||||
ShowColors = Colors_Off;
|
|
||||||
else if (value == "auto")
|
|
||||||
ShowColors = Colors_Auto;
|
|
||||||
else
|
|
||||||
getToolChain().getDriver().Diag(diag::err_drv_clang_unsupported)
|
getToolChain().getDriver().Diag(diag::err_drv_clang_unsupported)
|
||||||
<< ("-fdiagnostics-color=" + value).str();
|
<< ("-fdiagnostics-color=" + Value).str();
|
||||||
}
|
}
|
||||||
|
A->claim();
|
||||||
}
|
}
|
||||||
if (ShowColors == Colors_On ||
|
if (D.getDiags().getDiagnosticOptions().ShowColors)
|
||||||
(ShowColors == Colors_Auto && llvm::sys::Process::StandardErrHasColors()))
|
|
||||||
CmdArgs.push_back("-fcolor-diagnostics");
|
CmdArgs.push_back("-fcolor-diagnostics");
|
||||||
|
|
||||||
if (Args.hasArg(options::OPT_fansi_escape_codes))
|
if (Args.hasArg(options::OPT_fansi_escape_codes))
|
||||||
|
|
|
@ -860,8 +860,51 @@ static void ParseDependencyOutputArgs(DependencyOutputOptions &Opts,
|
||||||
ModuleFiles.end());
|
ModuleFiles.end());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool parseShowColorsArgs(const ArgList &Args, bool DefaultColor) {
|
||||||
|
// Color diagnostics default to auto ("on" if terminal supports) in the driver
|
||||||
|
// but default to off in cc1, needing an explicit OPT_fdiagnostics_color.
|
||||||
|
// Support both clang's -f[no-]color-diagnostics and gcc's
|
||||||
|
// -f[no-]diagnostics-colors[=never|always|auto].
|
||||||
|
enum {
|
||||||
|
Colors_On,
|
||||||
|
Colors_Off,
|
||||||
|
Colors_Auto
|
||||||
|
} ShowColors = DefaultColor ? Colors_Auto : Colors_Off;
|
||||||
|
for (Arg *A : Args) {
|
||||||
|
const Option &O = A->getOption();
|
||||||
|
if (!O.matches(options::OPT_fcolor_diagnostics) &&
|
||||||
|
!O.matches(options::OPT_fdiagnostics_color) &&
|
||||||
|
!O.matches(options::OPT_fno_color_diagnostics) &&
|
||||||
|
!O.matches(options::OPT_fno_diagnostics_color) &&
|
||||||
|
!O.matches(options::OPT_fdiagnostics_color_EQ))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (O.matches(options::OPT_fcolor_diagnostics) ||
|
||||||
|
O.matches(options::OPT_fdiagnostics_color)) {
|
||||||
|
ShowColors = Colors_On;
|
||||||
|
} else if (O.matches(options::OPT_fno_color_diagnostics) ||
|
||||||
|
O.matches(options::OPT_fno_diagnostics_color)) {
|
||||||
|
ShowColors = Colors_Off;
|
||||||
|
} else {
|
||||||
|
assert(O.matches(options::OPT_fdiagnostics_color_EQ));
|
||||||
|
StringRef Value(A->getValue());
|
||||||
|
if (Value == "always")
|
||||||
|
ShowColors = Colors_On;
|
||||||
|
else if (Value == "never")
|
||||||
|
ShowColors = Colors_Off;
|
||||||
|
else if (Value == "auto")
|
||||||
|
ShowColors = Colors_Auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (ShowColors == Colors_On ||
|
||||||
|
(ShowColors == Colors_Auto && llvm::sys::Process::StandardErrHasColors()))
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool clang::ParseDiagnosticArgs(DiagnosticOptions &Opts, ArgList &Args,
|
bool clang::ParseDiagnosticArgs(DiagnosticOptions &Opts, ArgList &Args,
|
||||||
DiagnosticsEngine *Diags) {
|
DiagnosticsEngine *Diags,
|
||||||
|
bool DefaultDiagColor) {
|
||||||
using namespace options;
|
using namespace options;
|
||||||
bool Success = true;
|
bool Success = true;
|
||||||
|
|
||||||
|
@ -874,7 +917,7 @@ bool clang::ParseDiagnosticArgs(DiagnosticOptions &Opts, ArgList &Args,
|
||||||
Opts.Pedantic = Args.hasArg(OPT_pedantic);
|
Opts.Pedantic = Args.hasArg(OPT_pedantic);
|
||||||
Opts.PedanticErrors = Args.hasArg(OPT_pedantic_errors);
|
Opts.PedanticErrors = Args.hasArg(OPT_pedantic_errors);
|
||||||
Opts.ShowCarets = !Args.hasArg(OPT_fno_caret_diagnostics);
|
Opts.ShowCarets = !Args.hasArg(OPT_fno_caret_diagnostics);
|
||||||
Opts.ShowColors = Args.hasArg(OPT_fcolor_diagnostics);
|
Opts.ShowColors = parseShowColorsArgs(Args, DefaultDiagColor);
|
||||||
Opts.ShowColumn = Args.hasFlag(OPT_fshow_column,
|
Opts.ShowColumn = Args.hasFlag(OPT_fshow_column,
|
||||||
OPT_fno_show_column,
|
OPT_fno_show_column,
|
||||||
/*Default=*/true);
|
/*Default=*/true);
|
||||||
|
@ -2240,7 +2283,8 @@ bool CompilerInvocation::CreateFromArgs(CompilerInvocation &Res,
|
||||||
Success &= ParseAnalyzerArgs(*Res.getAnalyzerOpts(), Args, Diags);
|
Success &= ParseAnalyzerArgs(*Res.getAnalyzerOpts(), Args, Diags);
|
||||||
Success &= ParseMigratorArgs(Res.getMigratorOpts(), Args);
|
Success &= ParseMigratorArgs(Res.getMigratorOpts(), Args);
|
||||||
ParseDependencyOutputArgs(Res.getDependencyOutputOpts(), Args);
|
ParseDependencyOutputArgs(Res.getDependencyOutputOpts(), Args);
|
||||||
Success &= ParseDiagnosticArgs(Res.getDiagnosticOpts(), Args, &Diags);
|
Success &= ParseDiagnosticArgs(Res.getDiagnosticOpts(), Args, &Diags,
|
||||||
|
false /*DefaultDiagColor*/);
|
||||||
ParseCommentArgs(LangOpts.CommentOpts, Args);
|
ParseCommentArgs(LangOpts.CommentOpts, Args);
|
||||||
ParseFileSystemArgs(Res.getFileSystemOpts(), Args);
|
ParseFileSystemArgs(Res.getFileSystemOpts(), Args);
|
||||||
// FIXME: We shouldn't have to pass the DashX option around here
|
// FIXME: We shouldn't have to pass the DashX option around here
|
||||||
|
|
Loading…
Reference in New Issue