[llvm-cov] Separate presentation logic from formatting logic, NFC

This makes it easier to add renderers for new kinds of output formats.

- Define and document a pure-virtual coverage rendering interface.
- Move the text-based rendering logic into its a new file.
- Re-work the API to better reflect the presentation/formatting split.

llvm-svn: 273767
This commit is contained in:
Vedant Kumar 2016-06-25 02:58:30 +00:00
parent cc676c47a3
commit f9151b9372
6 changed files with 364 additions and 202 deletions

View File

@ -8,5 +8,6 @@ add_llvm_tool(llvm-cov
CoverageReport.cpp CoverageReport.cpp
CoverageSummaryInfo.cpp CoverageSummaryInfo.cpp
SourceCoverageView.cpp SourceCoverageView.cpp
SourceCoverageViewText.cpp
TestingSupport.cpp TestingSupport.cpp
) )

View File

@ -132,9 +132,9 @@ CodeCoverageTool::attachExpansionSubViews(SourceCoverageView &View,
continue; continue;
auto SubViewExpansions = ExpansionCoverage.getExpansions(); auto SubViewExpansions = ExpansionCoverage.getExpansions();
auto SubView = llvm::make_unique<SourceCoverageView>( auto SubView =
Expansion.Function.Name, SourceBuffer.get(), ViewOpts, SourceCoverageView::create(Expansion.Function.Name, SourceBuffer.get(),
std::move(ExpansionCoverage)); ViewOpts, std::move(ExpansionCoverage));
attachExpansionSubViews(*SubView, SubViewExpansions, Coverage); attachExpansionSubViews(*SubView, SubViewExpansions, Coverage);
View.addExpansion(Expansion.Region, std::move(SubView)); View.addExpansion(Expansion.Region, std::move(SubView));
} }
@ -151,8 +151,8 @@ CodeCoverageTool::createFunctionView(const FunctionRecord &Function,
return nullptr; return nullptr;
auto Expansions = FunctionCoverage.getExpansions(); auto Expansions = FunctionCoverage.getExpansions();
auto View = llvm::make_unique<SourceCoverageView>( auto View = SourceCoverageView::create(Function.Name, SourceBuffer.get(),
Function.Name, SourceBuffer.get(), ViewOpts, std::move(FunctionCoverage)); ViewOpts, std::move(FunctionCoverage));
attachExpansionSubViews(*View, Expansions, Coverage); attachExpansionSubViews(*View, Expansions, Coverage);
return View; return View;
@ -169,16 +169,16 @@ CodeCoverageTool::createSourceFileView(StringRef SourceFile,
return nullptr; return nullptr;
auto Expansions = FileCoverage.getExpansions(); auto Expansions = FileCoverage.getExpansions();
auto View = llvm::make_unique<SourceCoverageView>( auto View = SourceCoverageView::create(SourceFile, SourceBuffer.get(),
SourceFile, SourceBuffer.get(), ViewOpts, std::move(FileCoverage)); ViewOpts, std::move(FileCoverage));
attachExpansionSubViews(*View, Expansions, Coverage); attachExpansionSubViews(*View, Expansions, Coverage);
for (auto Function : Coverage.getInstantiations(SourceFile)) { for (auto Function : Coverage.getInstantiations(SourceFile)) {
auto SubViewCoverage = Coverage.getCoverageForFunction(*Function); auto SubViewCoverage = Coverage.getCoverageForFunction(*Function);
auto SubViewExpansions = SubViewCoverage.getExpansions(); auto SubViewExpansions = SubViewCoverage.getExpansions();
auto SubView = llvm::make_unique<SourceCoverageView>( auto SubView =
Function->Name, SourceBuffer.get(), ViewOpts, SourceCoverageView::create(Function->Name, SourceBuffer.get(), ViewOpts,
std::move(SubViewCoverage)); std::move(SubViewCoverage));
attachExpansionSubViews(*SubView, SubViewExpansions, Coverage); attachExpansionSubViews(*SubView, SubViewExpansions, Coverage);
if (SubView) { if (SubView) {
@ -428,8 +428,7 @@ int CodeCoverageTool::show(int argc, const char **argv,
<< "\n"; << "\n";
continue; continue;
} }
mainView->renderSourceName(outs()); mainView->print(outs(), /*WholeFile=*/false, /*ShowSourceName=*/true);
mainView->render(outs(), /*WholeFile=*/false);
outs() << "\n"; outs() << "\n";
} }
return 0; return 0;
@ -452,10 +451,7 @@ int CodeCoverageTool::show(int argc, const char **argv,
continue; continue;
} }
if (ShowFilenames) mainView->print(outs(), /*Wholefile=*/true, /*ShowSourceName=*/ShowFilenames);
mainView->renderSourceName(outs());
mainView->render(outs(), /*Wholefile=*/true);
if (SourceFiles.size() > 1) if (SourceFiles.size() > 1)
outs() << "\n"; outs() << "\n";
} }

View File

@ -12,75 +12,14 @@
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
#include "SourceCoverageView.h" #include "SourceCoverageView.h"
#include "llvm/ADT/Optional.h" #include "SourceCoverageViewText.h"
#include "llvm/ADT/SmallString.h" #include "llvm/ADT/SmallString.h"
#include "llvm/ADT/StringExtras.h" #include "llvm/ADT/StringExtras.h"
#include "llvm/Support/LineIterator.h" #include "llvm/Support/LineIterator.h"
using namespace llvm; using namespace llvm;
void SourceCoverageView::renderLine( std::string SourceCoverageView::formatCount(uint64_t N) {
raw_ostream &OS, StringRef Line, int64_t LineNumber,
const coverage::CoverageSegment *WrappedSegment,
ArrayRef<const coverage::CoverageSegment *> Segments,
unsigned ExpansionCol) {
Optional<raw_ostream::Colors> Highlight;
SmallVector<std::pair<unsigned, unsigned>, 2> HighlightedRanges;
// The first segment overlaps from a previous line, so we treat it specially.
if (WrappedSegment && WrappedSegment->HasCount && WrappedSegment->Count == 0)
Highlight = raw_ostream::RED;
// Output each segment of the line, possibly highlighted.
unsigned Col = 1;
for (const auto *S : Segments) {
unsigned End = std::min(S->Col, static_cast<unsigned>(Line.size()) + 1);
colored_ostream(OS, Highlight ? *Highlight : raw_ostream::SAVEDCOLOR,
getOptions().Colors && Highlight, /*Bold=*/false,
/*BG=*/true)
<< Line.substr(Col - 1, End - Col);
if (getOptions().Debug && Highlight)
HighlightedRanges.push_back(std::make_pair(Col, End));
Col = End;
if (Col == ExpansionCol)
Highlight = raw_ostream::CYAN;
else if (S->HasCount && S->Count == 0)
Highlight = raw_ostream::RED;
else
Highlight = None;
}
// Show the rest of the line
colored_ostream(OS, Highlight ? *Highlight : raw_ostream::SAVEDCOLOR,
getOptions().Colors && Highlight, /*Bold=*/false, /*BG=*/true)
<< Line.substr(Col - 1, Line.size() - Col + 1);
OS << "\n";
if (getOptions().Debug) {
for (const auto &Range : HighlightedRanges)
errs() << "Highlighted line " << LineNumber << ", " << Range.first
<< " -> " << Range.second << "\n";
if (Highlight)
errs() << "Highlighted line " << LineNumber << ", " << Col << " -> ?\n";
}
}
void SourceCoverageView::renderIndent(raw_ostream &OS, unsigned Level) {
for (unsigned I = 0; I < Level; ++I)
OS << " |";
}
void SourceCoverageView::renderViewDivider(unsigned Level, unsigned Length,
raw_ostream &OS) {
assert(Level != 0 && "Cannot render divider at top level");
renderIndent(OS, Level - 1);
OS.indent(2);
for (unsigned I = 0; I < Length; ++I)
OS << "-";
}
/// Format a count using engineering notation with 3 significant digits.
static std::string formatCount(uint64_t N) {
std::string Number = utostr(N); std::string Number = utostr(N);
int Len = Number.size(); int Len = Number.size();
if (Len <= 3) if (Len <= 3)
@ -95,63 +34,30 @@ static std::string formatCount(uint64_t N) {
return Result; return Result;
} }
void void SourceCoverageView::addExpansion(
SourceCoverageView::renderLineCoverageColumn(raw_ostream &OS, const coverage::CounterMappingRegion &Region,
const LineCoverageStats &Line) { std::unique_ptr<SourceCoverageView> View) {
if (!Line.isMapped()) { ExpansionSubViews.emplace_back(Region, std::move(View));
OS.indent(LineCoverageColumnWidth) << '|';
return;
}
std::string C = formatCount(Line.ExecutionCount);
OS.indent(LineCoverageColumnWidth - C.size());
colored_ostream(OS, raw_ostream::MAGENTA,
Line.hasMultipleRegions() && getOptions().Colors)
<< C;
OS << '|';
} }
void SourceCoverageView::renderLineNumberColumn(raw_ostream &OS, void SourceCoverageView::addInstantiation(
unsigned LineNo) { StringRef FunctionName, unsigned Line,
SmallString<32> Buffer; std::unique_ptr<SourceCoverageView> View) {
raw_svector_ostream BufferOS(Buffer); InstantiationSubViews.emplace_back(FunctionName, Line, std::move(View));
BufferOS << LineNo;
auto Str = BufferOS.str();
// Trim and align to the right
Str = Str.substr(0, std::min(Str.size(), (size_t)LineNumberColumnWidth));
OS.indent(LineNumberColumnWidth - Str.size()) << Str << '|';
} }
void SourceCoverageView::renderRegionMarkers( std::unique_ptr<SourceCoverageView>
raw_ostream &OS, ArrayRef<const coverage::CoverageSegment *> Segments) { SourceCoverageView::create(StringRef SourceName, const MemoryBuffer &File,
unsigned PrevColumn = 1; const CoverageViewOptions &Options,
for (const auto *S : Segments) { coverage::CoverageData &&CoverageInfo) {
if (!S->IsRegionEntry) return llvm::make_unique<SourceCoverageViewText>(SourceName, File, Options,
continue; std::move(CoverageInfo));
// Skip to the new region
if (S->Col > PrevColumn)
OS.indent(S->Col - PrevColumn);
PrevColumn = S->Col + 1;
std::string C = formatCount(S->Count);
PrevColumn += C.size();
OS << '^' << C;
}
OS << "\n";
if (getOptions().Debug)
for (const auto *S : Segments)
errs() << "Marker at " << S->Line << ":" << S->Col << " = "
<< formatCount(S->Count) << (S->IsRegionEntry ? "\n" : " (pop)\n");
} }
void SourceCoverageView::render(raw_ostream &OS, bool WholeFile, void SourceCoverageView::print(raw_ostream &OS, bool WholeFile,
unsigned IndentLevel) { bool ShowSourceName, unsigned ViewDepth) {
// The width of the leading columns if (ShowSourceName)
unsigned CombinedColumnWidth = renderSourceName(OS);
(getOptions().ShowLineStats ? LineCoverageColumnWidth + 1 : 0) +
(getOptions().ShowLineNumbers ? LineNumberColumnWidth + 1 : 0);
// The width of the line that is used to divide between the view and the
// subviews.
unsigned DividerWidth = CombinedColumnWidth + 4;
// We need the expansions and instantiations sorted so we can go through them // We need the expansions and instantiations sorted so we can go through them
// while we iterate lines. // while we iterate lines.
@ -192,10 +98,9 @@ void SourceCoverageView::render(raw_ostream &OS, bool WholeFile,
LineCount.addRegionCount(WrappedSegment->Count); LineCount.addRegionCount(WrappedSegment->Count);
for (const auto *S : LineSegments) for (const auto *S : LineSegments)
if (S->HasCount && S->IsRegionEntry) if (S->HasCount && S->IsRegionEntry)
LineCount.addRegionStartCount(S->Count); LineCount.addRegionStartCount(S->Count);
// Render the line prefix. renderLinePrefix(OS, ViewDepth);
renderIndent(OS, IndentLevel);
if (getOptions().ShowLineStats) if (getOptions().ShowLineStats)
renderLineCoverageColumn(OS, LineCount); renderLineCoverageColumn(OS, LineCount);
if (getOptions().ShowLineNumbers) if (getOptions().ShowLineNumbers)
@ -208,62 +113,34 @@ void SourceCoverageView::render(raw_ostream &OS, bool WholeFile,
ExpansionColumn = NextESV->getStartCol(); ExpansionColumn = NextESV->getStartCol();
// Display the source code for the current line. // Display the source code for the current line.
renderLine(OS, *LI, LI.line_number(), WrappedSegment, LineSegments, renderLine(OS, {*LI, LI.line_number()}, WrappedSegment, LineSegments,
ExpansionColumn); ExpansionColumn, ViewDepth);
// Show the region markers. // Show the region markers.
if (getOptions().ShowRegionMarkers && if (getOptions().ShowRegionMarkers &&
(!getOptions().ShowLineStatsOrRegionMarkers || (!getOptions().ShowLineStatsOrRegionMarkers ||
LineCount.hasMultipleRegions()) && LineCount.hasMultipleRegions()) &&
!LineSegments.empty()) { !LineSegments.empty()) {
renderIndent(OS, IndentLevel); renderRegionMarkers(OS, LineSegments, ViewDepth);
OS.indent(CombinedColumnWidth);
renderRegionMarkers(OS, LineSegments);
} }
// Show the expansions and instantiations for this line. // Show the expansions and instantiations for this line.
unsigned NestedIndent = IndentLevel + 1;
bool RenderedSubView = false; bool RenderedSubView = false;
for (; NextESV != EndESV && NextESV->getLine() == LI.line_number(); for (; NextESV != EndESV && NextESV->getLine() == LI.line_number();
++NextESV) { ++NextESV) {
renderViewDivider(NestedIndent, DividerWidth, OS); renderViewDivider(OS, ViewDepth + 1);
OS << "\n"; ExpansionColumn = renderExpansionView(
if (RenderedSubView) { OS, *NextESV,
// Re-render the current line and highlight the expansion range for RenderedSubView ? Optional<LineRef>({*LI, LI.line_number()})
// this subview. : Optional<LineRef>(),
ExpansionColumn = NextESV->getStartCol(); ExpansionColumn, WrappedSegment, LineSegments, ViewDepth);
renderIndent(OS, IndentLevel);
OS.indent(CombinedColumnWidth + (IndentLevel == 0 ? 0 : 1));
renderLine(OS, *LI, LI.line_number(), WrappedSegment, LineSegments,
ExpansionColumn);
renderViewDivider(NestedIndent, DividerWidth, OS);
OS << "\n";
}
// Render the child subview
if (getOptions().Debug)
errs() << "Expansion at line " << NextESV->getLine() << ", "
<< NextESV->getStartCol() << " -> " << NextESV->getEndCol()
<< "\n";
NextESV->View->render(OS, false, NestedIndent);
RenderedSubView = true; RenderedSubView = true;
} }
for (; NextISV != EndISV && NextISV->Line == LI.line_number(); ++NextISV) { for (; NextISV != EndISV && NextISV->Line == LI.line_number(); ++NextISV) {
renderViewDivider(NestedIndent, DividerWidth, OS); renderInstantiationView(OS, *NextISV, ViewDepth + 1);
OS << "\n";
renderIndent(OS, NestedIndent);
OS << ' ';
NextISV->View->renderSourceName(OS);
NextISV->View->render(OS, false, NestedIndent);
RenderedSubView = true; RenderedSubView = true;
} }
if (RenderedSubView) { if (RenderedSubView)
renderViewDivider(NestedIndent, DividerWidth, OS); renderViewDivider(OS, ViewDepth + 1);
OS << "\n";
}
} }
} }
void SourceCoverageView::renderSourceName(raw_ostream &OS) {
getOptions().colored_ostream(OS, raw_ostream::CYAN) << getSourceName()
<< ":\n";
}

View File

@ -16,6 +16,7 @@
#include "CoverageViewOptions.h" #include "CoverageViewOptions.h"
#include "llvm/ProfileData/Coverage/CoverageMapping.h" #include "llvm/ProfileData/Coverage/CoverageMapping.h"
#include "llvm/ADT/Optional.h"
#include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/MemoryBuffer.h"
#include <vector> #include <vector>
@ -100,7 +101,6 @@ struct LineCoverageStats {
/// \brief A code coverage view of a specific source file. /// \brief A code coverage view of a specific source file.
/// It can have embedded coverage views. /// It can have embedded coverage views.
class SourceCoverageView { class SourceCoverageView {
private:
/// A function or file name. /// A function or file name.
StringRef SourceName; StringRef SourceName;
@ -120,59 +120,93 @@ private:
/// on display. /// on display.
std::vector<InstantiationView> InstantiationSubViews; std::vector<InstantiationView> InstantiationSubViews;
protected:
struct LineRef {
StringRef Line;
int64_t LineNo;
LineRef(StringRef Line, int64_t LineNo) : Line(Line), LineNo(LineNo) {}
};
using CoverageSegmentArray = ArrayRef<const coverage::CoverageSegment *>;
/// @name Rendering Interface
/// @{
/// \brief Render the source name for the view.
virtual void renderSourceName(raw_ostream &OS) = 0;
/// \brief Render the line prefix at the given \p ViewDepth.
virtual void renderLinePrefix(raw_ostream &OS, unsigned ViewDepth) = 0;
/// \brief Render a view divider at the given \p ViewDepth.
virtual void renderViewDivider(raw_ostream &OS, unsigned ViewDepth) = 0;
/// \brief Render a source line with highlighting. /// \brief Render a source line with highlighting.
void renderLine(raw_ostream &OS, StringRef Line, int64_t LineNumber, virtual void renderLine(raw_ostream &OS, LineRef L,
const coverage::CoverageSegment *WrappedSegment, const coverage::CoverageSegment *WrappedSegment,
ArrayRef<const coverage::CoverageSegment *> Segments, CoverageSegmentArray Segments, unsigned ExpansionCol,
unsigned ExpansionCol); unsigned ViewDepth) = 0;
void renderIndent(raw_ostream &OS, unsigned Level);
void renderViewDivider(unsigned Offset, unsigned Length, raw_ostream &OS);
/// \brief Render the line's execution count column. /// \brief Render the line's execution count column.
void renderLineCoverageColumn(raw_ostream &OS, const LineCoverageStats &Line); virtual void renderLineCoverageColumn(raw_ostream &OS,
const LineCoverageStats &Line) = 0;
/// \brief Render the line number column. /// \brief Render the line number column.
void renderLineNumberColumn(raw_ostream &OS, unsigned LineNo); virtual void renderLineNumberColumn(raw_ostream &OS, unsigned LineNo) = 0;
/// \brief Render all the region's execution counts on a line. /// \brief Render all the region's execution counts on a line.
void virtual void renderRegionMarkers(raw_ostream &OS,
renderRegionMarkers(raw_ostream &OS, CoverageSegmentArray Segments,
ArrayRef<const coverage::CoverageSegment *> Segments); unsigned ViewDepth) = 0;
static const unsigned LineCoverageColumnWidth = 7; /// \brief Render an expansion view. If \p FirstLine is provided, it points
static const unsigned LineNumberColumnWidth = 5; /// to the expansion site, which must be re-rendered for clarity.
virtual unsigned renderExpansionView(
raw_ostream &OS, ExpansionView &ESV, Optional<LineRef> FirstLine,
unsigned ExpansionCol, const coverage::CoverageSegment *WrappedSegment,
CoverageSegmentArray LineSegments, unsigned ViewDepth) = 0;
/// \brief Render an instantiation view.
virtual void renderInstantiationView(raw_ostream &OS, InstantiationView &ISV,
unsigned ViewDepth) = 0;
/// @}
/// \brief Format a count using engineering notation with 3 significant
/// digits.
static std::string formatCount(uint64_t N);
public:
SourceCoverageView(StringRef SourceName, const MemoryBuffer &File, SourceCoverageView(StringRef SourceName, const MemoryBuffer &File,
const CoverageViewOptions &Options, const CoverageViewOptions &Options,
coverage::CoverageData &&CoverageInfo) coverage::CoverageData &&CoverageInfo)
: SourceName(SourceName), File(File), Options(Options), : SourceName(SourceName), File(File), Options(Options),
CoverageInfo(std::move(CoverageInfo)) {} CoverageInfo(std::move(CoverageInfo)) {}
public:
static std::unique_ptr<SourceCoverageView>
create(StringRef SourceName, const MemoryBuffer &File,
const CoverageViewOptions &Options,
coverage::CoverageData &&CoverageInfo);
virtual ~SourceCoverageView() {}
StringRef getSourceName() const { return SourceName; } StringRef getSourceName() const { return SourceName; }
const CoverageViewOptions &getOptions() const { return Options; } const CoverageViewOptions &getOptions() const { return Options; }
/// \brief Add an expansion subview to this view. /// \brief Add an expansion subview to this view.
void addExpansion(const coverage::CounterMappingRegion &Region, void addExpansion(const coverage::CounterMappingRegion &Region,
std::unique_ptr<SourceCoverageView> View) { std::unique_ptr<SourceCoverageView> View);
ExpansionSubViews.emplace_back(Region, std::move(View));
}
/// \brief Add a function instantiation subview to this view. /// \brief Add a function instantiation subview to this view.
void addInstantiation(StringRef FunctionName, unsigned Line, void addInstantiation(StringRef FunctionName, unsigned Line,
std::unique_ptr<SourceCoverageView> View) { std::unique_ptr<SourceCoverageView> View);
InstantiationSubViews.emplace_back(FunctionName, Line, std::move(View));
}
/// \brief Print the code coverage information for a specific /// \brief Print the code coverage information for a specific portion of a
/// portion of a source file to the output stream. /// source file to the output stream.
void render(raw_ostream &OS, bool WholeFile, unsigned IndentLevel = 0); void print(raw_ostream &OS, bool WholeFile, bool ShowSourceName,
unsigned ViewDepth = 0);
/// \brief Print the source name corresponding to this view.
void renderSourceName(raw_ostream &OS);
}; };
} // namespace llvm } // namespace llvm

View File

@ -0,0 +1,193 @@
//===- SourceCoverageViewText.cpp - A text-based code coverage view -------===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This file implements the text-based coverage renderer.
//
//===----------------------------------------------------------------------===//
#include "SourceCoverageViewText.h"
#include "llvm/ADT/SmallString.h"
#include "llvm/ADT/StringExtras.h"
using namespace llvm;
namespace {
constexpr unsigned LineCoverageColumnWidth = 7;
constexpr unsigned LineNumberColumnWidth = 5;
/// \brief Get the width of the leading columns.
unsigned getCombinedColumnWidth(const CoverageViewOptions &Opts) {
return (Opts.ShowLineStats ? LineCoverageColumnWidth + 1 : 0) +
(Opts.ShowLineNumbers ? LineNumberColumnWidth + 1 : 0);
}
/// \brief The width of the line that is used to divide between the view and
/// the subviews.
unsigned getDividerWidth(const CoverageViewOptions &Opts) {
return getCombinedColumnWidth(Opts) + 4;
}
} // anonymous namespace
void SourceCoverageViewText::renderSourceName(raw_ostream &OS) {
getOptions().colored_ostream(OS, raw_ostream::CYAN) << getSourceName()
<< ":\n";
}
void SourceCoverageViewText::renderLinePrefix(raw_ostream &OS,
unsigned ViewDepth) {
for (unsigned I = 0; I < ViewDepth; ++I)
OS << " |";
}
void SourceCoverageViewText::renderViewDivider(raw_ostream &OS,
unsigned ViewDepth) {
assert(ViewDepth != 0 && "Cannot render divider at top level");
renderLinePrefix(OS, ViewDepth - 1);
OS.indent(2);
unsigned Length = getDividerWidth(getOptions());
for (unsigned I = 0; I < Length; ++I)
OS << '-';
OS << '\n';
}
void SourceCoverageViewText::renderLine(
raw_ostream &OS, LineRef L,
const coverage::CoverageSegment *WrappedSegment,
CoverageSegmentArray Segments, unsigned ExpansionCol, unsigned ViewDepth) {
StringRef Line = L.Line;
unsigned LineNumber = L.LineNo;
Optional<raw_ostream::Colors> Highlight;
SmallVector<std::pair<unsigned, unsigned>, 2> HighlightedRanges;
// The first segment overlaps from a previous line, so we treat it specially.
if (WrappedSegment && WrappedSegment->HasCount && WrappedSegment->Count == 0)
Highlight = raw_ostream::RED;
// Output each segment of the line, possibly highlighted.
unsigned Col = 1;
for (const auto *S : Segments) {
unsigned End = std::min(S->Col, static_cast<unsigned>(Line.size()) + 1);
colored_ostream(OS, Highlight ? *Highlight : raw_ostream::SAVEDCOLOR,
getOptions().Colors && Highlight, /*Bold=*/false,
/*BG=*/true)
<< Line.substr(Col - 1, End - Col);
if (getOptions().Debug && Highlight)
HighlightedRanges.push_back(std::make_pair(Col, End));
Col = End;
if (Col == ExpansionCol)
Highlight = raw_ostream::CYAN;
else if (S->HasCount && S->Count == 0)
Highlight = raw_ostream::RED;
else
Highlight = None;
}
// Show the rest of the line.
colored_ostream(OS, Highlight ? *Highlight : raw_ostream::SAVEDCOLOR,
getOptions().Colors && Highlight, /*Bold=*/false, /*BG=*/true)
<< Line.substr(Col - 1, Line.size() - Col + 1);
OS << '\n';
if (getOptions().Debug) {
for (const auto &Range : HighlightedRanges)
errs() << "Highlighted line " << LineNumber << ", " << Range.first
<< " -> " << Range.second << '\n';
if (Highlight)
errs() << "Highlighted line " << LineNumber << ", " << Col << " -> ?\n";
}
}
void SourceCoverageViewText::renderLineCoverageColumn(
raw_ostream &OS, const LineCoverageStats &Line) {
if (!Line.isMapped()) {
OS.indent(LineCoverageColumnWidth) << '|';
return;
}
std::string C = formatCount(Line.ExecutionCount);
OS.indent(LineCoverageColumnWidth - C.size());
colored_ostream(OS, raw_ostream::MAGENTA,
Line.hasMultipleRegions() && getOptions().Colors)
<< C;
OS << '|';
}
void SourceCoverageViewText::renderLineNumberColumn(raw_ostream &OS,
unsigned LineNo) {
SmallString<32> Buffer;
raw_svector_ostream BufferOS(Buffer);
BufferOS << LineNo;
auto Str = BufferOS.str();
// Trim and align to the right.
Str = Str.substr(0, std::min(Str.size(), (size_t)LineNumberColumnWidth));
OS.indent(LineNumberColumnWidth - Str.size()) << Str << '|';
}
void SourceCoverageViewText::renderRegionMarkers(
raw_ostream &OS, CoverageSegmentArray Segments, unsigned ViewDepth) {
renderLinePrefix(OS, ViewDepth);
OS.indent(getCombinedColumnWidth(getOptions()));
unsigned PrevColumn = 1;
for (const auto *S : Segments) {
if (!S->IsRegionEntry)
continue;
// Skip to the new region.
if (S->Col > PrevColumn)
OS.indent(S->Col - PrevColumn);
PrevColumn = S->Col + 1;
std::string C = formatCount(S->Count);
PrevColumn += C.size();
OS << '^' << C;
}
OS << '\n';
if (getOptions().Debug)
for (const auto *S : Segments)
errs() << "Marker at " << S->Line << ":" << S->Col << " = "
<< formatCount(S->Count) << (S->IsRegionEntry ? "\n" : " (pop)\n");
}
unsigned SourceCoverageViewText::renderExpansionView(
raw_ostream &OS, ExpansionView &ESV, Optional<LineRef> FirstLine,
unsigned ExpansionCol, const coverage::CoverageSegment *WrappedSegment,
CoverageSegmentArray LineSegments, unsigned ViewDepth) {
unsigned NextExpansionCol = ExpansionCol;
if (FirstLine.hasValue()) {
// Re-render the current line and highlight the expansion range for
// this subview.
NextExpansionCol = ESV.getStartCol();
renderLinePrefix(OS, ViewDepth);
OS.indent(getCombinedColumnWidth(getOptions()) + (ViewDepth == 0 ? 0 : 1));
renderLine(OS, *FirstLine, WrappedSegment, LineSegments, ExpansionCol,
ViewDepth);
renderViewDivider(OS, ViewDepth + 1);
}
// Render the child subview.
if (getOptions().Debug)
errs() << "Expansion at line " << ESV.getLine() << ", " << ESV.getStartCol()
<< " -> " << ESV.getEndCol() << '\n';
ESV.View->print(OS, /*WholeFile=*/false, /*ShowSourceName=*/false,
ViewDepth + 1);
return NextExpansionCol;
}
void SourceCoverageViewText::renderInstantiationView(raw_ostream &OS,
InstantiationView &ISV,
unsigned ViewDepth) {
renderViewDivider(OS, ViewDepth);
renderLinePrefix(OS, ViewDepth);
OS << ' ';
ISV.View->print(OS, /*WholeFile=*/false, /*ShowSourceName=*/true, ViewDepth);
}

View File

@ -0,0 +1,61 @@
//===- SourceCoverageViewText.h - A text-based code coverage view ---------===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This file defines the interface to the text-based coverage renderer.
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_COV_SOURCECOVERAGEVIEWTEXT_H
#define LLVM_COV_SOURCECOVERAGEVIEWTEXT_H
#include "SourceCoverageView.h"
namespace llvm {
class SourceCoverageViewText : public SourceCoverageView {
void renderSourceName(raw_ostream &OS) override;
void renderLinePrefix(raw_ostream &OS, unsigned ViewDepth) override;
void renderViewDivider(raw_ostream &OS, unsigned ViewDepth) override;
void renderLine(raw_ostream &OS, LineRef L,
const coverage::CoverageSegment *WrappedSegment,
CoverageSegmentArray Segments, unsigned ExpansionCol,
unsigned ViewDepth) override;
unsigned renderExpansionView(raw_ostream &OS, ExpansionView &ESV,
Optional<LineRef> FirstLine,
unsigned ExpansionCol,
const coverage::CoverageSegment *WrappedSegment,
CoverageSegmentArray LineSegments,
unsigned ViewDepth) override;
void renderInstantiationView(raw_ostream &OS, InstantiationView &ISV,
unsigned ViewDepth) override;
void renderLineCoverageColumn(raw_ostream &OS,
const LineCoverageStats &Line) override;
void renderLineNumberColumn(raw_ostream &OS, unsigned LineNo) override;
void renderRegionMarkers(raw_ostream &OS, CoverageSegmentArray Segments,
unsigned ViewDepth) override;
public:
SourceCoverageViewText(StringRef SourceName, const MemoryBuffer &File,
const CoverageViewOptions &Options,
coverage::CoverageData &&CoverageInfo)
: SourceCoverageView(SourceName, File, Options, std::move(CoverageInfo)) {
}
};
} // namespace llvm
#endif // LLVM_COV_SOURCECOVERAGEVIEWTEXT_H