llvm-cov: Add support for gcov's --long-file-names option

GCOV provides an option to prepend output file names with the source
file name, to disambiguate between covered data that's included from
multiple sources. Add a flag to llvm-cov that does the same.

llvm-svn: 207035
This commit is contained in:
Justin Bogner 2014-04-23 21:44:55 +00:00
parent bac905c684
commit c67f0250ef
6 changed files with 46 additions and 10 deletions

View File

@ -37,9 +37,9 @@ namespace GCOV {
/// GCOVOptions - A struct for passing gcov options between functions. /// GCOVOptions - A struct for passing gcov options between functions.
struct GCOVOptions { struct GCOVOptions {
GCOVOptions(bool A, bool B, bool C, bool F, bool P, bool U) GCOVOptions(bool A, bool B, bool C, bool F, bool P, bool U, bool L)
: AllBlocks(A), BranchInfo(B), BranchCount(C), FuncCoverage(F), : AllBlocks(A), BranchInfo(B), BranchCount(C), FuncCoverage(F),
PreservePaths(P), UncondBranch(U) {} PreservePaths(P), UncondBranch(U), LongFileNames(L) {}
bool AllBlocks; bool AllBlocks;
bool BranchInfo; bool BranchInfo;
@ -47,6 +47,7 @@ struct GCOVOptions {
bool FuncCoverage; bool FuncCoverage;
bool PreservePaths; bool PreservePaths;
bool UncondBranch; bool UncondBranch;
bool LongFileNames;
}; };
/// GCOVBuffer - A wrapper around MemoryBuffer to provide GCOV specific /// GCOVBuffer - A wrapper around MemoryBuffer to provide GCOV specific
@ -385,7 +386,8 @@ public:
} }
void setRunCount(uint32_t Runs) { RunCount = Runs; } void setRunCount(uint32_t Runs) { RunCount = Runs; }
void setProgramCount(uint32_t Programs) { ProgramCount = Programs; } void setProgramCount(uint32_t Programs) { ProgramCount = Programs; }
void print(StringRef GCNOFile, StringRef GCDAFile); void print(StringRef MainFilename, StringRef GCNOFile, StringRef GCDAFile);
private: private:
void printFunctionSummary(raw_fd_ostream &OS, void printFunctionSummary(raw_fd_ostream &OS,
const FunctionVector &Funcs) const; const FunctionVector &Funcs) const;

View File

@ -424,7 +424,7 @@ static raw_ostream &operator<<(raw_ostream &OS, const formatBranchInfo &FBI) {
/// translates "/" to "#", ".." to "^", and drops ".", to match gcov. /// translates "/" to "#", ".." to "^", and drops ".", to match gcov.
static std::string mangleCoveragePath(StringRef Filename, bool PreservePaths) { static std::string mangleCoveragePath(StringRef Filename, bool PreservePaths) {
if (!PreservePaths) if (!PreservePaths)
return (sys::path::filename(Filename) + ".gcov").str(); return sys::path::filename(Filename).str();
// This behaviour is defined by gcov in terms of text replacements, so it's // This behaviour is defined by gcov in terms of text replacements, so it's
// not likely to do anything useful on filesystems with different textual // not likely to do anything useful on filesystems with different textual
@ -452,12 +452,12 @@ static std::string mangleCoveragePath(StringRef Filename, bool PreservePaths) {
if (S < I) if (S < I)
Result.append(S, I); Result.append(S, I);
Result.append(".gcov");
return Result.str(); return Result.str();
} }
/// print - Print source files with collected line count information. /// print - Print source files with collected line count information.
void FileInfo::print(StringRef GCNOFile, StringRef GCDAFile) { void FileInfo::print(StringRef MainFilename, StringRef GCNOFile,
StringRef GCDAFile) {
for (StringMap<LineData>::const_iterator I = LineInfo.begin(), for (StringMap<LineData>::const_iterator I = LineInfo.begin(),
E = LineInfo.end(); I != E; ++I) { E = LineInfo.end(); I != E; ++I) {
StringRef Filename = I->first(); StringRef Filename = I->first();
@ -468,8 +468,12 @@ void FileInfo::print(StringRef GCNOFile, StringRef GCDAFile) {
} }
StringRef AllLines = Buff->getBuffer(); StringRef AllLines = Buff->getBuffer();
std::string CoveragePath = mangleCoveragePath(Filename, std::string CoveragePath;
Options.PreservePaths); if (Options.LongFileNames && !Filename.equals(MainFilename))
CoveragePath =
mangleCoveragePath(MainFilename, Options.PreservePaths) + "##";
CoveragePath +=
mangleCoveragePath(Filename, Options.PreservePaths) + ".gcov";
std::string ErrorInfo; std::string ErrorInfo;
raw_fd_ostream OS(CoveragePath.c_str(), ErrorInfo, sys::fs::F_Text); raw_fd_ostream OS(CoveragePath.c_str(), ErrorInfo, sys::fs::F_Text);
if (!ErrorInfo.empty()) if (!ErrorInfo.empty())

View File

@ -0,0 +1,8 @@
File 'srcdir/./nested_dir/../test.h'
Lines executed:100.00% of 1
srcdir/./nested_dir/../test.h:creating 'test_paths.cpp##test.h.gcov'
File 'srcdir/./nested_dir/../test.cpp'
Lines executed:84.21% of 38
srcdir/./nested_dir/../test.cpp:creating 'test_paths.cpp##test.cpp.gcov'

View File

@ -0,0 +1,8 @@
File 'srcdir/./nested_dir/../test.h'
Lines executed:100.00% of 1
srcdir/./nested_dir/../test.h:creating 'srcdir#^#test_paths.cpp##srcdir#nested_dir#^#test.h.gcov'
File 'srcdir/./nested_dir/../test.cpp'
Lines executed:84.21% of 38
srcdir/./nested_dir/../test.cpp:creating 'srcdir#^#test_paths.cpp##srcdir#nested_dir#^#test.cpp.gcov'

View File

@ -43,6 +43,16 @@ RUN: llvm-cov test_paths.cpp | diff -u test_no_preserve_paths.output -
RUN: diff -aub test_paths.cpp.gcov test.cpp.gcov RUN: diff -aub test_paths.cpp.gcov test.cpp.gcov
RUN: diff -aub test_paths.h.gcov test.h.gcov RUN: diff -aub test_paths.h.gcov test.h.gcov
# Long file names.
RUN: llvm-cov -l test_paths.cpp | diff -u test_long_file_names.output -
RUN: diff -aub test_paths.cpp.gcov test_paths.cpp##test.cpp.gcov
RUN: diff -aub test_paths.h.gcov test_paths.cpp##test.h.gcov
# Long file names and preserve paths.
RUN: llvm-cov -lp -gcno test_paths.gcno -gcda test_paths.gcda srcdir/../test_paths.cpp | diff -u test_long_paths.output -
RUN: diff -aub test_paths.cpp.gcov srcdir#^#test_paths.cpp##srcdir#nested_dir#^#test.cpp.gcov
RUN: diff -aub test_paths.h.gcov srcdir#^#test_paths.cpp##srcdir#nested_dir#^#test.h.gcov
# Function summaries. This changes stdout, but not the gcov files. # Function summaries. This changes stdout, but not the gcov files.
RUN: llvm-cov test.c -f | diff -u test_-f.output - RUN: llvm-cov test.c -f | diff -u test_-f.output -
RUN: diff -aub test_no_options.cpp.gcov test.cpp.gcov RUN: diff -aub test_no_options.cpp.gcov test.cpp.gcov

View File

@ -39,6 +39,10 @@ static cl::opt<bool> BranchCount("c", cl::Grouping, cl::init(false),
"of percentages (requires -b)")); "of percentages (requires -b)"));
static cl::alias BranchCountA("branch-counts", cl::aliasopt(BranchCount)); static cl::alias BranchCountA("branch-counts", cl::aliasopt(BranchCount));
static cl::opt<bool> LongNames("l", cl::Grouping, cl::init(false),
cl::desc("Prefix filenames with the main file"));
static cl::alias LongNamesA("long-file-names", cl::aliasopt(LongNames));
static cl::opt<bool> FuncSummary("f", cl::Grouping, cl::init(false), static cl::opt<bool> FuncSummary("f", cl::Grouping, cl::init(false),
cl::desc("Show coverage for each function")); cl::desc("Show coverage for each function"));
static cl::alias FuncSummaryA("function-summaries", cl::aliasopt(FuncSummary)); static cl::alias FuncSummaryA("function-summaries", cl::aliasopt(FuncSummary));
@ -126,9 +130,9 @@ int main(int argc, char **argv) {
GF.dump(); GF.dump();
GCOVOptions Options(AllBlocks, BranchProb, BranchCount, FuncSummary, GCOVOptions Options(AllBlocks, BranchProb, BranchCount, FuncSummary,
PreservePaths, UncondBranch); PreservePaths, UncondBranch, LongNames);
FileInfo FI(Options); FileInfo FI(Options);
GF.collectLineCounts(FI); GF.collectLineCounts(FI);
FI.print(InputGCNO, InputGCDA); FI.print(SourceFile, InputGCNO, InputGCDA);
return 0; return 0;
} }