You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			159 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			159 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===--- TextDiagnosticPrinter.cpp - Diagnostic Printer -------------------===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | //
 | ||
|  | // This diagnostic client prints out their diagnostic messages.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "clang/Frontend/TextDiagnosticPrinter.h"
 | ||
|  | #include "clang/Basic/DiagnosticOptions.h"
 | ||
|  | #include "clang/Basic/SourceManager.h"
 | ||
|  | #include "clang/Frontend/TextDiagnostic.h"
 | ||
|  | #include "clang/Lex/Lexer.h"
 | ||
|  | #include "llvm/ADT/SmallString.h"
 | ||
|  | #include "llvm/Support/ErrorHandling.h"
 | ||
|  | #include "llvm/Support/raw_ostream.h"
 | ||
|  | #include <algorithm>
 | ||
|  | using namespace clang; | ||
|  | 
 | ||
|  | TextDiagnosticPrinter::TextDiagnosticPrinter(raw_ostream &os, | ||
|  |                                              DiagnosticOptions *diags, | ||
|  |                                              bool _OwnsOutputStream) | ||
|  |   : OS(os), DiagOpts(diags), | ||
|  |     OwnsOutputStream(_OwnsOutputStream) { | ||
|  | } | ||
|  | 
 | ||
|  | TextDiagnosticPrinter::~TextDiagnosticPrinter() { | ||
|  |   if (OwnsOutputStream) | ||
|  |     delete &OS; | ||
|  | } | ||
|  | 
 | ||
|  | void TextDiagnosticPrinter::BeginSourceFile(const LangOptions &LO, | ||
|  |                                             const Preprocessor *PP) { | ||
|  |   // Build the TextDiagnostic utility.
 | ||
|  |   TextDiag.reset(new TextDiagnostic(OS, LO, &*DiagOpts)); | ||
|  | } | ||
|  | 
 | ||
|  | void TextDiagnosticPrinter::EndSourceFile() { | ||
|  |   TextDiag.reset(); | ||
|  | } | ||
|  | 
 | ||
|  | /// \brief Print any diagnostic option information to a raw_ostream.
 | ||
|  | ///
 | ||
|  | /// This implements all of the logic for adding diagnostic options to a message
 | ||
|  | /// (via OS). Each relevant option is comma separated and all are enclosed in
 | ||
|  | /// the standard bracketing: " [...]".
 | ||
|  | static void printDiagnosticOptions(raw_ostream &OS, | ||
|  |                                    DiagnosticsEngine::Level Level, | ||
|  |                                    const Diagnostic &Info, | ||
|  |                                    const DiagnosticOptions &DiagOpts) { | ||
|  |   bool Started = false; | ||
|  |   if (DiagOpts.ShowOptionNames) { | ||
|  |     // Handle special cases for non-warnings early.
 | ||
|  |     if (Info.getID() == diag::fatal_too_many_errors) { | ||
|  |       OS << " [-ferror-limit=]"; | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     // The code below is somewhat fragile because we are essentially trying to
 | ||
|  |     // report to the user what happened by inferring what the diagnostic engine
 | ||
|  |     // did. Eventually it might make more sense to have the diagnostic engine
 | ||
|  |     // include some "why" information in the diagnostic.
 | ||
|  | 
 | ||
|  |     // If this is a warning which has been mapped to an error by the user (as
 | ||
|  |     // inferred by checking whether the default mapping is to an error) then
 | ||
|  |     // flag it as such. Note that diagnostics could also have been mapped by a
 | ||
|  |     // pragma, but we don't currently have a way to distinguish this.
 | ||
|  |     if (Level == DiagnosticsEngine::Error && | ||
|  |         DiagnosticIDs::isBuiltinWarningOrExtension(Info.getID()) && | ||
|  |         !DiagnosticIDs::isDefaultMappingAsError(Info.getID())) { | ||
|  |       OS << " [-Werror"; | ||
|  |       Started = true; | ||
|  |     } | ||
|  | 
 | ||
|  |     StringRef Opt = DiagnosticIDs::getWarningOptionForDiag(Info.getID()); | ||
|  |     if (!Opt.empty()) { | ||
|  |       OS << (Started ? "," : " [") | ||
|  |          << (Level == DiagnosticsEngine::Remark ? "-R" : "-W") << Opt; | ||
|  |       StringRef OptValue = Info.getDiags()->getFlagValue(); | ||
|  |       if (!OptValue.empty()) | ||
|  |         OS << "=" << OptValue; | ||
|  |       Started = true; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   // If the user wants to see category information, include it too.
 | ||
|  |   if (DiagOpts.ShowCategories) { | ||
|  |     unsigned DiagCategory = | ||
|  |       DiagnosticIDs::getCategoryNumberForDiag(Info.getID()); | ||
|  |     if (DiagCategory) { | ||
|  |       OS << (Started ? "," : " ["); | ||
|  |       Started = true; | ||
|  |       if (DiagOpts.ShowCategories == 1) | ||
|  |         OS << DiagCategory; | ||
|  |       else { | ||
|  |         assert(DiagOpts.ShowCategories == 2 && "Invalid ShowCategories value"); | ||
|  |         OS << DiagnosticIDs::getCategoryNameFromID(DiagCategory); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  |   if (Started) | ||
|  |     OS << ']'; | ||
|  | } | ||
|  | 
 | ||
|  | void TextDiagnosticPrinter::HandleDiagnostic(DiagnosticsEngine::Level Level, | ||
|  |                                              const Diagnostic &Info) { | ||
|  |   // Default implementation (Warnings/errors count).
 | ||
|  |   DiagnosticConsumer::HandleDiagnostic(Level, Info); | ||
|  | 
 | ||
|  |   // Render the diagnostic message into a temporary buffer eagerly. We'll use
 | ||
|  |   // this later as we print out the diagnostic to the terminal.
 | ||
|  |   SmallString<100> OutStr; | ||
|  |   Info.FormatDiagnostic(OutStr); | ||
|  | 
 | ||
|  |   llvm::raw_svector_ostream DiagMessageStream(OutStr); | ||
|  |   printDiagnosticOptions(DiagMessageStream, Level, Info, *DiagOpts); | ||
|  | 
 | ||
|  |   // Keeps track of the starting position of the location
 | ||
|  |   // information (e.g., "foo.c:10:4:") that precedes the error
 | ||
|  |   // message. We use this information to determine how long the
 | ||
|  |   // file+line+column number prefix is.
 | ||
|  |   uint64_t StartOfLocationInfo = OS.tell(); | ||
|  | 
 | ||
|  |   if (!Prefix.empty()) | ||
|  |     OS << Prefix << ": "; | ||
|  | 
 | ||
|  |   // Use a dedicated, simpler path for diagnostics without a valid location.
 | ||
|  |   // This is important as if the location is missing, we may be emitting
 | ||
|  |   // diagnostics in a context that lacks language options, a source manager, or
 | ||
|  |   // other infrastructure necessary when emitting more rich diagnostics.
 | ||
|  |   if (!Info.getLocation().isValid()) { | ||
|  |     TextDiagnostic::printDiagnosticLevel(OS, Level, DiagOpts->ShowColors, | ||
|  |                                          DiagOpts->CLFallbackMode); | ||
|  |     TextDiagnostic::printDiagnosticMessage(OS, Level, DiagMessageStream.str(), | ||
|  |                                            OS.tell() - StartOfLocationInfo, | ||
|  |                                            DiagOpts->MessageLength, | ||
|  |                                            DiagOpts->ShowColors); | ||
|  |     OS.flush(); | ||
|  |     return; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Assert that the rest of our infrastructure is setup properly.
 | ||
|  |   assert(DiagOpts && "Unexpected diagnostic without options set"); | ||
|  |   assert(Info.hasSourceManager() && | ||
|  |          "Unexpected diagnostic with no source manager"); | ||
|  |   assert(TextDiag && "Unexpected diagnostic outside source file processing"); | ||
|  | 
 | ||
|  |   TextDiag->emitDiagnostic( | ||
|  |       FullSourceLoc(Info.getLocation(), Info.getSourceManager()), Level, | ||
|  |       DiagMessageStream.str(), Info.getRanges(), Info.getFixItHints()); | ||
|  | 
 | ||
|  |   OS.flush(); | ||
|  | } |