Skip to content

Commit d36b47c

Browse files
committedSep 6, 2016
[llvm-cov] Add the "Go to first unexecuted line" feature.
This patch provides easy navigation to find the zero count lines, especially useful when the source file is very large. Differential Revision: https://reviews.llvm.org/D23277 llvm-svn: 280739
1 parent 19eeb37 commit d36b47c

8 files changed

+56
-10
lines changed
 

‎llvm/test/tools/llvm-cov/native_separators.c

+6
Original file line numberDiff line numberDiff line change
@@ -19,3 +19,9 @@
1919
// HTML: tools\llvm-cov\Inputs\native_separators.covmapping</pre>
2020

2121
int main() {}
22+
23+
// Re-purpose this file to test that "Go to first unexecuted line" feature.
24+
25+
// RUN: llvm-cov show %S/Inputs/native_separators.covmapping -instr-profile %t.profdata -filename-equivalence -format html -o %t.dir %s
26+
// RUN: FileCheck -input-file %t.dir/coverage/tmp/native_separators.c.html %s
27+
// CHECK-NOT: >Go to first unexecuted line<

‎llvm/test/tools/llvm-cov/showProjectSummary.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -24,9 +24,9 @@ int main(int argc, char ** argv) {
2424

2525
// Test html output.
2626
// RUN: llvm-cov show %S/Inputs/showProjectSummary.covmapping -format=html -o %t.dir -instr-profile %t.profdata -filename-equivalence %s
27-
// RUN: FileCheck -check-prefixes=HTML,HTML-FILE,HTML-HEADER -input-file %t.dir/coverage/tmp/showProjectSummary.cpp.html %s
27+
// RUN: FileCheck -check-prefixes=HTML,HTML-FILE,HTML-HEADER,HTML-UNCOVEREDLINE -input-file %t.dir/coverage/tmp/showProjectSummary.cpp.html %s
2828
// RUN: llvm-cov show %S/Inputs/showProjectSummary.covmapping -format=html -o %t.dir -instr-profile %t.profdata -project-title "Test Suite" -filename-equivalence %s
29-
// RUN: FileCheck -check-prefixes=HTML-TITLE,HTML,HTML-FILE,HTML-HEADER -input-file %t.dir/coverage/tmp/showProjectSummary.cpp.html %s
29+
// RUN: FileCheck -check-prefixes=HTML-TITLE,HTML,HTML-FILE,HTML-HEADER,HTML-UNCOVEREDLINE -input-file %t.dir/coverage/tmp/showProjectSummary.cpp.html %s
3030
// RUN: FileCheck -check-prefixes=HTML-TITLE,HTML -input-file %t.dir/index.html %s
3131
// RUN: llvm-cov show %S/Inputs/showProjectSummary.covmapping -format=html -o %t.dir -instr-profile %t.profdata -project-title "Test Suite" -filename-equivalence -name=main %s
3232
// RUN: FileCheck -check-prefixes=HTML-FUNCTION,HTML-HEADER -input-file %t.dir/functions.html %s
@@ -41,6 +41,7 @@ int main(int argc, char ** argv) {
4141
// HTML-FILE: <pre>Binary:
4242
// HTML-FILE: showProjectSummary.covmapping</pre>
4343
// HTML-FUNCTION: <pre>Function: main</pre>
44+
// HTML-UNCOVEREDLINE: <a href='#L8'>Go to first unexecuted line</a>
4445
// HTML-HEADER: <tr><td><span><pre>Line No.</pre></span></td>
4546
// HTML-HEADER: <td><span><pre>Count</pre></span></td>
4647
// HTML-HEADER: <td><span><pre>Source</pre></span></td>

‎llvm/tools/llvm-cov/SourceCoverageView.cpp

+24-1
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,25 @@ CoveragePrinter::create(const CoverageViewOptions &Opts) {
8282
llvm_unreachable("Unknown coverage output format!");
8383
}
8484

85+
unsigned SourceCoverageView::getFirstUncoveredLineNo() {
86+
auto CheckIfUncovered = [](const coverage::CoverageSegment &S) {
87+
return S.HasCount && S.Count == 0;
88+
};
89+
// L is less than R if (1) it's an uncovered segment (has a 0 count), and (2)
90+
// either R is not an uncovered segment, or L has a lower line number than R.
91+
const auto MinSegIt =
92+
std::min_element(CoverageInfo.begin(), CoverageInfo.end(),
93+
[CheckIfUncovered](const coverage::CoverageSegment &L,
94+
const coverage::CoverageSegment &R) {
95+
return (CheckIfUncovered(L) &&
96+
(!CheckIfUncovered(R) || (L.Line < R.Line)));
97+
});
98+
if (CheckIfUncovered(*MinSegIt))
99+
return (*MinSegIt).Line;
100+
// There is no uncovered line, return zero.
101+
return 0;
102+
}
103+
85104
std::string SourceCoverageView::formatCount(uint64_t N) {
86105
std::string Number = utostr(N);
87106
int Len = Number.size();
@@ -142,8 +161,12 @@ void SourceCoverageView::print(raw_ostream &OS, bool WholeFile,
142161

143162
renderViewHeader(OS);
144163

164+
unsigned FirstUncoveredLineNo = 0;
165+
if (WholeFile)
166+
FirstUncoveredLineNo = getFirstUncoveredLineNo();
167+
145168
if (ShowSourceName)
146-
renderSourceName(OS, WholeFile);
169+
renderSourceName(OS, WholeFile, FirstUncoveredLineNo);
147170

148171
renderTableHeader(OS, ViewDepth);
149172
// We need the expansions and instantiations sorted so we can go through them

‎llvm/tools/llvm-cov/SourceCoverageView.h

+5-1
Original file line numberDiff line numberDiff line change
@@ -175,6 +175,9 @@ class SourceCoverageView {
175175
/// Specifies whether or not the view is a function view.
176176
bool FunctionView;
177177

178+
/// Get the first uncovered line number for the source file.
179+
unsigned getFirstUncoveredLineNo();
180+
178181
protected:
179182
struct LineRef {
180183
StringRef Line;
@@ -195,7 +198,8 @@ class SourceCoverageView {
195198
virtual void renderViewFooter(raw_ostream &OS) = 0;
196199

197200
/// \brief Render the source name for the view.
198-
virtual void renderSourceName(raw_ostream &OS, bool WholeFile) = 0;
201+
virtual void renderSourceName(raw_ostream &OS, bool WholeFile,
202+
unsigned FirstUncoveredLineNo) = 0;
199203

200204
/// \brief Render the line prefix at the given \p ViewDepth.
201205
virtual void renderLinePrefix(raw_ostream &OS, unsigned ViewDepth) = 0;

‎llvm/tools/llvm-cov/SourceCoverageViewHTML.cpp

+12-3
Original file line numberDiff line numberDiff line change
@@ -338,7 +338,8 @@ void SourceCoverageViewHTML::renderViewFooter(raw_ostream &OS) {
338338
OS << EndTable << EndCenteredDiv;
339339
}
340340

341-
void SourceCoverageViewHTML::renderSourceName(raw_ostream &OS, bool WholeFile) {
341+
void SourceCoverageViewHTML::renderSourceName(raw_ostream &OS, bool WholeFile,
342+
unsigned FirstUncoveredLineNo) {
342343
OS << BeginSourceNameDiv;
343344
// Render the source name for the view.
344345
std::string SourceFile = isFunctionView() ? "Function: " : "Source: ";
@@ -347,10 +348,18 @@ void SourceCoverageViewHTML::renderSourceName(raw_ostream &OS, bool WholeFile) {
347348
sys::path::remove_dots(SourceText, /*remove_dot_dots=*/true);
348349
sys::path::native(SourceText);
349350
OS << tag("pre", escape(SourceText, getOptions()));
350-
// Render the object file name for the view.
351-
if (WholeFile)
351+
if (WholeFile) {
352+
// Render the object file name for the view.
352353
OS << tag("pre",
353354
escape("Binary: " + getOptions().ObjectFilename, getOptions()));
355+
// Render the "Go to first unexecuted line" link for the view.
356+
if (FirstUncoveredLineNo != 0) { // The file is not fully covered
357+
std::string LinkText =
358+
escape("Go to first unexecuted line", getOptions());
359+
std::string LinkTarget = "#L" + utostr(uint64_t(FirstUncoveredLineNo));
360+
OS << tag("pre", a(LinkTarget, LinkText));
361+
}
362+
}
354363
OS << EndSourceNameDiv;
355364
}
356365

‎llvm/tools/llvm-cov/SourceCoverageViewHTML.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,8 @@ class SourceCoverageViewHTML : public SourceCoverageView {
3838

3939
void renderViewFooter(raw_ostream &OS) override;
4040

41-
void renderSourceName(raw_ostream &OS, bool WholeFile) override;
41+
void renderSourceName(raw_ostream &OS, bool WholeFile,
42+
unsigned FirstUncoveredLineNo) override;
4243

4344
void renderLinePrefix(raw_ostream &OS, unsigned ViewDepth) override;
4445

‎llvm/tools/llvm-cov/SourceCoverageViewText.cpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,8 @@ void SourceCoverageViewText::renderViewHeader(raw_ostream &) {}
6363

6464
void SourceCoverageViewText::renderViewFooter(raw_ostream &) {}
6565

66-
void SourceCoverageViewText::renderSourceName(raw_ostream &OS, bool WholeFile) {
66+
void SourceCoverageViewText::renderSourceName(raw_ostream &OS, bool WholeFile,
67+
unsigned FirstUncoveredLineNo) {
6768
getOptions().colored_ostream(OS, raw_ostream::CYAN) << getSourceName()
6869
<< ":\n";
6970
if (WholeFile) {

‎llvm/tools/llvm-cov/SourceCoverageViewText.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,8 @@ class SourceCoverageViewText : public SourceCoverageView {
3838

3939
void renderViewFooter(raw_ostream &OS) override;
4040

41-
void renderSourceName(raw_ostream &OS, bool WholeFile) override;
41+
void renderSourceName(raw_ostream &OS, bool WholeFile,
42+
unsigned FirstUncoveredLineNo) override;
4243

4344
void renderLinePrefix(raw_ostream &OS, unsigned ViewDepth) override;
4445

0 commit comments

Comments
 (0)
Please sign in to comment.