mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
The block info block does not require an explicit abbreviation width as the local abbreviations are internal to the block. Remove the parameter similar to the clang change.
521 lines
18 KiB
C++
521 lines
18 KiB
C++
//===--- SerializedDiagnosticConsumer.cpp - Serialize Diagnostics ---------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2016 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See http://swift.org/LICENSE.txt for license information
|
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements the SerializedDiagnosticConsumer class.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/Frontend/SerializedDiagnosticConsumer.h"
|
|
#include "swift/Basic/DiagnosticConsumer.h"
|
|
#include "swift/Basic/LLVM.h"
|
|
#include "swift/Basic/SourceManager.h"
|
|
#include "swift/Parse/Lexer.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/ADT/Twine.h"
|
|
#include "llvm/ADT/DenseMap.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include "llvm/ADT/IntrusiveRefCntPtr.h"
|
|
#include "llvm/ADT/SmallString.h"
|
|
#include "llvm/Bitcode/BitstreamWriter.h"
|
|
|
|
// For constant values only.
|
|
#include "clang/Frontend/SerializedDiagnosticPrinter.h"
|
|
|
|
using namespace swift;
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// These must match Clang's diagnostic IDs. We can consider sharing the
|
|
// header files to avoid this copy-paste.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
enum BlockIDs {
|
|
/// \brief A top-level block which represents any meta data associated
|
|
/// with the diagnostics, including versioning of the format.
|
|
BLOCK_META = llvm::bitc::FIRST_APPLICATION_BLOCKID,
|
|
|
|
/// \brief The this block acts as a container for all the information
|
|
/// for a specific diagnostic.
|
|
BLOCK_DIAG
|
|
};
|
|
|
|
enum RecordIDs {
|
|
RECORD_VERSION = 1,
|
|
RECORD_DIAG,
|
|
RECORD_SOURCE_RANGE,
|
|
RECORD_DIAG_FLAG,
|
|
RECORD_CATEGORY,
|
|
RECORD_FILENAME,
|
|
RECORD_FIXIT,
|
|
RECORD_FIRST = RECORD_VERSION,
|
|
RECORD_LAST = RECORD_FIXIT
|
|
};
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
namespace {
|
|
class AbbreviationMap {
|
|
llvm::DenseMap<unsigned, unsigned> Abbrevs;
|
|
public:
|
|
AbbreviationMap() {}
|
|
|
|
void set(unsigned recordID, unsigned abbrevID) {
|
|
assert(Abbrevs.find(recordID) == Abbrevs.end()
|
|
&& "Abbreviation already set.");
|
|
Abbrevs[recordID] = abbrevID;
|
|
}
|
|
|
|
unsigned get(unsigned recordID) {
|
|
assert(Abbrevs.find(recordID) != Abbrevs.end() &&
|
|
"Abbreviation not set.");
|
|
return Abbrevs[recordID];
|
|
}
|
|
};
|
|
|
|
typedef SmallVector<uint64_t, 64> RecordData;
|
|
typedef SmallVectorImpl<uint64_t> RecordDataImpl;
|
|
|
|
struct SharedState : llvm::RefCountedBase<SharedState> {
|
|
SharedState(std::unique_ptr<raw_ostream> OS)
|
|
: Stream(Buffer), OS(std::move(OS)), EmittedAnyDiagBlocks(false) { }
|
|
|
|
/// \brief The byte buffer for the serialized content.
|
|
llvm::SmallString<1024> Buffer;
|
|
|
|
/// \brief The BitStreamWriter for the serialized diagnostics.
|
|
llvm::BitstreamWriter Stream;
|
|
|
|
/// \brief The name of the diagnostics file.
|
|
std::unique_ptr<raw_ostream> OS;
|
|
|
|
/// \brief The set of constructed record abbreviations.
|
|
AbbreviationMap Abbrevs;
|
|
|
|
/// \brief A utility buffer for constructing record content.
|
|
RecordData Record;
|
|
|
|
/// \brief A text buffer for rendering diagnostic text.
|
|
llvm::SmallString<256> diagBuf;
|
|
|
|
/// \brief The collection of files used.
|
|
llvm::DenseMap<const char *, unsigned> Files;
|
|
|
|
typedef llvm::DenseMap<const void *, std::pair<unsigned, StringRef> >
|
|
DiagFlagsTy;
|
|
|
|
/// \brief Map for uniquing strings.
|
|
DiagFlagsTy DiagFlags;
|
|
|
|
/// \brief Whether we have already started emission of any DIAG blocks. Once
|
|
/// this becomes \c true, we never close a DIAG block until we know that we're
|
|
/// starting another one or we're done.
|
|
bool EmittedAnyDiagBlocks;
|
|
};
|
|
|
|
/// \brief Diagnostic consumer that serializes diagnostics to a stream.
|
|
class SerializedDiagnosticConsumer : public DiagnosticConsumer {
|
|
/// \brief State shared among the various clones of this diagnostic consumer.
|
|
llvm::IntrusiveRefCntPtr<SharedState> State;
|
|
public:
|
|
SerializedDiagnosticConsumer(std::unique_ptr<raw_ostream> OS)
|
|
: State(new SharedState(std::move(OS))) {
|
|
emitPreamble();
|
|
}
|
|
|
|
~SerializedDiagnosticConsumer() {
|
|
// FIXME: we may not wish to put this in a destructor.
|
|
// That's not what clang does.
|
|
|
|
// NOTE: clang also does check for shared instances. We don't
|
|
// have these yet in Swift, but if we do we need to add an extra
|
|
// check here.
|
|
|
|
// Finish off any diagnostic we were in the process of emitting.
|
|
if (State->EmittedAnyDiagBlocks)
|
|
exitDiagBlock();
|
|
|
|
// Write the generated bitstream to "Out".
|
|
State->OS->write((char *)&State->Buffer.front(), State->Buffer.size());
|
|
State->OS->flush();
|
|
State->OS.reset(0);
|
|
}
|
|
|
|
virtual void handleDiagnostic(SourceManager &SM, SourceLoc Loc,
|
|
DiagnosticKind Kind, llvm::StringRef Text,
|
|
const DiagnosticInfo &Info) override;
|
|
|
|
/// \brief The version of the diagnostics file.
|
|
enum { Version = 1 };
|
|
|
|
private:
|
|
/// \brief Emit bitcode for the preamble.
|
|
void emitPreamble();
|
|
|
|
/// \brief Emit bitcode for the BlockInfoBlock (part of the preamble).
|
|
void emitBlockInfoBlock();
|
|
|
|
/// \brief Emit bitcode for metadata block (part of preamble).
|
|
void emitMetaBlock();
|
|
|
|
/// \brief Emit bitcode to enter a block for a diagnostic.
|
|
void enterDiagBlock() {
|
|
State->Stream.EnterSubblock(BLOCK_DIAG, 4);
|
|
}
|
|
|
|
/// \brief Emit bitcode to exit a block for a diagnostic.
|
|
void exitDiagBlock() {
|
|
State->Stream.ExitBlock();
|
|
}
|
|
|
|
// Record identifier for the file.
|
|
unsigned getEmitFile(StringRef Filename);
|
|
|
|
/// \brief Add a source location to a record.
|
|
void addLocToRecord(SourceLoc Loc,
|
|
SourceManager &SM,
|
|
StringRef Filename,
|
|
RecordDataImpl &Record);
|
|
|
|
void addRangeToRecord(CharSourceRange Range, SourceManager &SM,
|
|
StringRef Filename, RecordDataImpl &Record);
|
|
|
|
/// \brief Emit the message payload of a diagnostic to bitcode.
|
|
void emitDiagnosticMessage(SourceManager &SM, SourceLoc Loc,
|
|
DiagnosticKind Kind,
|
|
StringRef Text, const DiagnosticInfo &Info);
|
|
};
|
|
}
|
|
|
|
namespace swift { namespace serialized_diagnostics {
|
|
DiagnosticConsumer *createConsumer(std::unique_ptr<llvm::raw_ostream> OS) {
|
|
return new SerializedDiagnosticConsumer(std::move(OS));
|
|
}
|
|
}}
|
|
|
|
unsigned SerializedDiagnosticConsumer::getEmitFile(StringRef Filename) {
|
|
// NOTE: Using Filename.data() here relies on SourceMgr using
|
|
// const char* as buffer identifiers. This is fast, but may
|
|
// be brittle. We can always switch over to using a StringMap.
|
|
unsigned &entry = State->Files[Filename.data()];
|
|
if (entry)
|
|
return entry;
|
|
|
|
// Lazily generate the record for the file. Note that in
|
|
// practice we only expect there to be one file, but this is
|
|
// general and is what the diagnostic file expects.
|
|
entry = State->Files.size();
|
|
RecordData Record;
|
|
Record.push_back(RECORD_FILENAME);
|
|
Record.push_back(entry);
|
|
Record.push_back(0); // For legacy.
|
|
Record.push_back(0); // For legacy.
|
|
Record.push_back(Filename.size());
|
|
State->Stream.EmitRecordWithBlob(State->Abbrevs.get(RECORD_FILENAME),
|
|
Record, Filename.data());
|
|
|
|
return entry;
|
|
}
|
|
|
|
void SerializedDiagnosticConsumer::addLocToRecord(SourceLoc Loc,
|
|
SourceManager &SM,
|
|
StringRef Filename,
|
|
RecordDataImpl &Record) {
|
|
if (!Loc.isValid()) {
|
|
// Emit a "sentinel" location.
|
|
Record.push_back((unsigned)0); // File.
|
|
Record.push_back((unsigned)0); // Line.
|
|
Record.push_back((unsigned)0); // Column.
|
|
Record.push_back((unsigned)0); // Offset.
|
|
return;
|
|
}
|
|
|
|
unsigned line, col;
|
|
std::tie(line, col) = SM.getLineAndColumn(Loc);
|
|
|
|
Record.push_back(getEmitFile(Filename));
|
|
Record.push_back(line);
|
|
Record.push_back(col);
|
|
Record.push_back(0);
|
|
}
|
|
|
|
void SerializedDiagnosticConsumer::addRangeToRecord(CharSourceRange Range,
|
|
SourceManager &SM,
|
|
StringRef Filename,
|
|
RecordDataImpl &Record) {
|
|
assert(Range.isValid());
|
|
addLocToRecord(Range.getStart(), SM, Filename, Record);
|
|
addLocToRecord(Range.getEnd(), SM, Filename, Record);
|
|
}
|
|
|
|
/// \brief Map a Swift DiagosticKind to the diagnostic level expected
|
|
/// for serialized diagnostics.
|
|
static clang::serialized_diags::Level getDiagnosticLevel(DiagnosticKind Kind) {
|
|
switch (Kind) {
|
|
case DiagnosticKind::Error:
|
|
return clang::serialized_diags::Error;
|
|
case DiagnosticKind::Note:
|
|
return clang::serialized_diags::Note;
|
|
case DiagnosticKind::Warning:
|
|
return clang::serialized_diags::Warning;
|
|
}
|
|
}
|
|
|
|
void SerializedDiagnosticConsumer::emitPreamble() {
|
|
State->Stream.Emit((unsigned)'D', 8);
|
|
State->Stream.Emit((unsigned)'I', 8);
|
|
State->Stream.Emit((unsigned)'A', 8);
|
|
State->Stream.Emit((unsigned)'G', 8);
|
|
emitBlockInfoBlock();
|
|
emitMetaBlock();
|
|
}
|
|
|
|
|
|
void SerializedDiagnosticConsumer::emitMetaBlock() {
|
|
llvm::BitstreamWriter &Stream = State->Stream;
|
|
RecordData &Record = State->Record;
|
|
AbbreviationMap &Abbrevs = State->Abbrevs;
|
|
|
|
Stream.EnterSubblock(BLOCK_META, 3);
|
|
Record.clear();
|
|
Record.push_back(RECORD_VERSION);
|
|
Record.push_back(Version);
|
|
Stream.EmitRecordWithAbbrev(Abbrevs.get(RECORD_VERSION), Record);
|
|
Stream.ExitBlock();
|
|
}
|
|
|
|
|
|
/// \brief Emits a block ID in the BLOCKINFO block.
|
|
static void emitBlockID(unsigned ID, const char *Name,
|
|
llvm::BitstreamWriter &Stream,
|
|
RecordDataImpl &Record) {
|
|
Record.clear();
|
|
Record.push_back(ID);
|
|
Stream.EmitRecord(llvm::bitc::BLOCKINFO_CODE_SETBID, Record);
|
|
|
|
// Emit the block name if present.
|
|
if (Name == 0 || Name[0] == 0)
|
|
return;
|
|
|
|
Record.clear();
|
|
|
|
while (*Name)
|
|
Record.push_back(*Name++);
|
|
|
|
Stream.EmitRecord(llvm::bitc::BLOCKINFO_CODE_BLOCKNAME, Record);
|
|
}
|
|
|
|
/// \brief Emits a record ID in the BLOCKINFO block.
|
|
static void emitRecordID(unsigned ID, const char *Name,
|
|
llvm::BitstreamWriter &Stream,
|
|
RecordDataImpl &Record) {
|
|
Record.clear();
|
|
Record.push_back(ID);
|
|
|
|
while (*Name)
|
|
Record.push_back(*Name++);
|
|
|
|
Stream.EmitRecord(llvm::bitc::BLOCKINFO_CODE_SETRECORDNAME, Record);
|
|
}
|
|
|
|
/// \brief Emit bitcode for abbreviation for source locations.
|
|
static void addSourceLocationAbbrev(llvm::BitCodeAbbrev *Abbrev) {
|
|
using namespace llvm;
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 10)); // File ID.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 32)); // Line.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 32)); // Column.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 32)); // Offset;
|
|
}
|
|
|
|
/// \brief Emit bitcode for abbreviation for source ranges.
|
|
static void addRangeLocationAbbrev(llvm::BitCodeAbbrev *Abbrev) {
|
|
addSourceLocationAbbrev(Abbrev);
|
|
addSourceLocationAbbrev(Abbrev);
|
|
}
|
|
|
|
void SerializedDiagnosticConsumer::emitBlockInfoBlock() {
|
|
State->Stream.EnterBlockInfoBlock();
|
|
|
|
using namespace llvm;
|
|
llvm::BitstreamWriter &Stream = State->Stream;
|
|
RecordData &Record = State->Record;
|
|
AbbreviationMap &Abbrevs = State->Abbrevs;
|
|
|
|
// ==---------------------------------------------------------------------==//
|
|
// The subsequent records and Abbrevs are for the "Meta" block.
|
|
// ==---------------------------------------------------------------------==//
|
|
|
|
emitBlockID(BLOCK_META, "Meta", Stream, Record);
|
|
emitRecordID(RECORD_VERSION, "Version", Stream, Record);
|
|
BitCodeAbbrev *Abbrev = new BitCodeAbbrev();
|
|
Abbrev->Add(BitCodeAbbrevOp(RECORD_VERSION));
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 32));
|
|
Abbrevs.set(RECORD_VERSION, Stream.EmitBlockInfoAbbrev(BLOCK_META, Abbrev));
|
|
|
|
// ==---------------------------------------------------------------------==//
|
|
// The subsequent records and Abbrevs are for the "Diagnostic" block.
|
|
// ==---------------------------------------------------------------------==//
|
|
|
|
emitBlockID(BLOCK_DIAG, "Diag", Stream, Record);
|
|
emitRecordID(RECORD_DIAG, "DiagInfo", Stream, Record);
|
|
emitRecordID(RECORD_SOURCE_RANGE, "SrcRange", Stream, Record);
|
|
emitRecordID(RECORD_CATEGORY, "CatName", Stream, Record);
|
|
emitRecordID(RECORD_DIAG_FLAG, "DiagFlag", Stream, Record);
|
|
emitRecordID(RECORD_FILENAME, "FileName", Stream, Record);
|
|
emitRecordID(RECORD_FIXIT, "FixIt", Stream, Record);
|
|
|
|
// Emit abbreviation for RECORD_DIAG.
|
|
Abbrev = new BitCodeAbbrev();
|
|
Abbrev->Add(BitCodeAbbrevOp(RECORD_DIAG));
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 3)); // Diag level.
|
|
addSourceLocationAbbrev(Abbrev);
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 10)); // Category.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 10)); // Mapped Diag ID.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 16)); // Text size.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Blob)); // Diagnostic text.
|
|
Abbrevs.set(RECORD_DIAG, Stream.EmitBlockInfoAbbrev(BLOCK_DIAG, Abbrev));
|
|
|
|
// Emit abbreviation for RECORD_CATEGORY.
|
|
Abbrev = new BitCodeAbbrev();
|
|
Abbrev->Add(BitCodeAbbrevOp(RECORD_CATEGORY));
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 16)); // Category ID.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 8)); // Text size.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Blob)); // Category text.
|
|
Abbrevs.set(RECORD_CATEGORY, Stream.EmitBlockInfoAbbrev(BLOCK_DIAG, Abbrev));
|
|
|
|
// Emit abbreviation for RECORD_SOURCE_RANGE.
|
|
Abbrev = new BitCodeAbbrev();
|
|
Abbrev->Add(BitCodeAbbrevOp(RECORD_SOURCE_RANGE));
|
|
addRangeLocationAbbrev(Abbrev);
|
|
Abbrevs.set(RECORD_SOURCE_RANGE,
|
|
Stream.EmitBlockInfoAbbrev(BLOCK_DIAG, Abbrev));
|
|
|
|
// Emit the abbreviation for RECORD_DIAG_FLAG.
|
|
Abbrev = new BitCodeAbbrev();
|
|
Abbrev->Add(BitCodeAbbrevOp(RECORD_DIAG_FLAG));
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 10)); // Mapped Diag ID.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 16)); // Text size.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Blob)); // Flag name text.
|
|
Abbrevs.set(RECORD_DIAG_FLAG, Stream.EmitBlockInfoAbbrev(BLOCK_DIAG,
|
|
Abbrev));
|
|
|
|
// Emit the abbreviation for RECORD_FILENAME.
|
|
Abbrev = new BitCodeAbbrev();
|
|
Abbrev->Add(BitCodeAbbrevOp(RECORD_FILENAME));
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 10)); // Mapped file ID.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 32)); // Size.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 32)); // Modification time.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 16)); // Text size.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Blob)); // File name text.
|
|
Abbrevs.set(RECORD_FILENAME, Stream.EmitBlockInfoAbbrev(BLOCK_DIAG,
|
|
Abbrev));
|
|
|
|
// Emit the abbreviation for RECORD_FIXIT.
|
|
Abbrev = new BitCodeAbbrev();
|
|
Abbrev->Add(BitCodeAbbrevOp(RECORD_FIXIT));
|
|
addRangeLocationAbbrev(Abbrev);
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 16)); // Text size.
|
|
Abbrev->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Blob)); // FixIt text.
|
|
Abbrevs.set(RECORD_FIXIT, Stream.EmitBlockInfoAbbrev(BLOCK_DIAG,
|
|
Abbrev));
|
|
|
|
Stream.ExitBlock();
|
|
}
|
|
|
|
void SerializedDiagnosticConsumer::
|
|
emitDiagnosticMessage(SourceManager &SM,
|
|
SourceLoc Loc,
|
|
DiagnosticKind Kind,
|
|
StringRef Text,
|
|
const DiagnosticInfo &Info) {
|
|
|
|
// Emit the diagnostic to bitcode.
|
|
llvm::BitstreamWriter &Stream = State->Stream;
|
|
RecordData &Record = State->Record;
|
|
AbbreviationMap &Abbrevs = State->Abbrevs;
|
|
|
|
StringRef filename = "";
|
|
if (Loc.isValid())
|
|
filename = SM.getIdentifierForBuffer(SM.findBufferContainingLoc(Loc));
|
|
|
|
// Emit the RECORD_DIAG record.
|
|
Record.clear();
|
|
Record.push_back(RECORD_DIAG);
|
|
Record.push_back(getDiagnosticLevel(Kind));
|
|
addLocToRecord(Loc, SM, filename, Record);
|
|
|
|
// FIXME: Swift diagnostics currently have no category.
|
|
Record.push_back(0);
|
|
// FIXME: Swift diagnostics currently have no flags.
|
|
Record.push_back(0);
|
|
|
|
// Emit the message.
|
|
Record.push_back(Text.size());
|
|
Stream.EmitRecordWithBlob(Abbrevs.get(RECORD_DIAG), Record, Text);
|
|
|
|
// If the location is invalid, do not emit source ranges or fixits.
|
|
if (Loc.isInvalid())
|
|
return;
|
|
|
|
// Emit source ranges.
|
|
auto RangeAbbrev = State->Abbrevs.get(RECORD_SOURCE_RANGE);
|
|
for (const auto &R : Info.Ranges) {
|
|
if (R.isInvalid())
|
|
continue;
|
|
State->Record.clear();
|
|
State->Record.push_back(RECORD_SOURCE_RANGE);
|
|
addRangeToRecord(R, SM, filename, State->Record);
|
|
State->Stream.EmitRecordWithAbbrev(RangeAbbrev, State->Record);
|
|
}
|
|
|
|
// Emit FixIts.
|
|
auto FixItAbbrev = State->Abbrevs.get(RECORD_FIXIT);
|
|
for (const auto &F : Info.FixIts) {
|
|
if (F.getRange().isValid()) {
|
|
State->Record.clear();
|
|
State->Record.push_back(RECORD_FIXIT);
|
|
addRangeToRecord(F.getRange(), SM, filename, State->Record);
|
|
State->Record.push_back(F.getText().size());
|
|
Stream.EmitRecordWithBlob(FixItAbbrev, Record, F.getText());
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
SerializedDiagnosticConsumer::handleDiagnostic(SourceManager &SM,
|
|
SourceLoc Loc,
|
|
DiagnosticKind Kind,
|
|
StringRef Text,
|
|
const DiagnosticInfo &Info) {
|
|
// Enter the block for a non-note diagnostic immediately, rather
|
|
// than waiting for beginDiagnostic, in case associated notes
|
|
// are emitted before we get there.
|
|
if (Kind != DiagnosticKind::Note) {
|
|
if (State->EmittedAnyDiagBlocks)
|
|
exitDiagBlock();
|
|
|
|
enterDiagBlock();
|
|
State->EmittedAnyDiagBlocks = true;
|
|
}
|
|
|
|
// Special-case diagnostics with no location.
|
|
// Make sure we bracket all notes as "sub-diagnostics".
|
|
bool bracketDiagnostic = (Kind == DiagnosticKind::Note);
|
|
|
|
if (bracketDiagnostic)
|
|
enterDiagBlock();
|
|
|
|
emitDiagnosticMessage(SM, Loc, Kind, Text, Info);
|
|
|
|
if (bracketDiagnostic)
|
|
exitDiagBlock();
|
|
}
|
|
|