InstrProf: Add unit tests for the profile reader and writer
[oota-llvm.git] / unittests / ProfileData / CoverageMappingTest.cpp
1 //===- unittest/ProfileData/CoverageMappingTest.cpp -------------------------=//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9
10 #include "llvm/ProfileData/CoverageMapping.h"
11 #include "llvm/ProfileData/CoverageMappingReader.h"
12 #include "llvm/ProfileData/CoverageMappingWriter.h"
13 #include "llvm/Support/raw_ostream.h"
14 #include "gtest/gtest.h"
15
16 #include <sstream>
17
18 using namespace llvm;
19 using namespace coverage;
20
21 namespace llvm {
22 namespace coverage {
23 void PrintTo(const Counter &C, ::std::ostream *os) {
24   if (C.isZero())
25     *os << "Zero";
26   else if (C.isExpression())
27     *os << "Expression " << C.getExpressionID();
28   else
29     *os << "Counter " << C.getCounterID();
30 }
31 }
32 }
33
34 namespace {
35
36 static std::string writeCoverage(MutableArrayRef<CounterMappingRegion> Regions,
37                                  int NumFiles) {
38   SmallVector<unsigned, 8> FileIDs;
39   for (int I = 0; I < NumFiles; ++I)
40     FileIDs.push_back(I);
41
42   std::string Coverage;
43   llvm::raw_string_ostream OS(Coverage);
44   CoverageMappingWriter(FileIDs, None, Regions).write(OS);
45   OS.flush();
46
47   return Coverage;
48 }
49
50 static std::vector<CounterMappingRegion>
51 readCoverageRegions(std::string Coverage, int NumFiles) {
52   SmallVector<std::string, 8> Filenames;
53   SmallVector<StringRef, 8> FilenameRefs;
54   for (int I = 0; I < NumFiles; ++I) {
55     std::ostringstream S;
56     S << "file" << I;
57     Filenames.push_back(S.str());
58     FilenameRefs.push_back(Filenames.back());
59   }
60
61   std::vector<StringRef> FuncFiles;
62   std::vector<CounterExpression> Expressions;
63   std::vector<CounterMappingRegion> Regions;
64   RawCoverageMappingReader Reader(Coverage, FilenameRefs, FuncFiles,
65                                   Expressions, Regions);
66   if (Reader.read())
67     // ASSERT doesn't work here, we'll just return an empty vector.
68     Regions.clear();
69   return Regions;
70 }
71
72 TEST(CoverageMappingTest, basic_write_read) {
73   int NumFiles = 2;
74   CounterMappingRegion InputRegions[] = {
75       CounterMappingRegion::makeRegion(Counter::getCounter(0), 0, 1, 1, 1, 1),
76       CounterMappingRegion::makeRegion(Counter::getCounter(1), 0, 2, 1, 2, 2),
77       CounterMappingRegion::makeRegion(Counter::getZero(), 0, 3, 1, 3, 4),
78       CounterMappingRegion::makeRegion(Counter::getCounter(2), 0, 4, 1, 4, 8),
79       CounterMappingRegion::makeRegion(Counter::getCounter(3), 1, 1, 2, 3, 4),
80   };
81   std::string Coverage = writeCoverage(InputRegions, NumFiles);
82   std::vector<CounterMappingRegion> OutputRegions =
83       readCoverageRegions(Coverage, NumFiles);
84   ASSERT_FALSE(OutputRegions.empty());
85
86   size_t N = makeArrayRef(InputRegions).size();
87   ASSERT_EQ(N, OutputRegions.size());
88   for (size_t I = 0; I < N; ++I) {
89     ASSERT_EQ(InputRegions[I].Count, OutputRegions[I].Count);
90     ASSERT_EQ(InputRegions[I].FileID, OutputRegions[I].FileID);
91     ASSERT_EQ(InputRegions[I].startLoc(), OutputRegions[I].startLoc());
92     ASSERT_EQ(InputRegions[I].endLoc(), OutputRegions[I].endLoc());
93     ASSERT_EQ(InputRegions[I].Kind, OutputRegions[I].Kind);
94   }
95 }
96
97 TEST(CoverageMappingTest, expansion_gets_first_counter) {
98   int NumFiles = 2;
99   CounterMappingRegion InputRegions[] = {
100       CounterMappingRegion::makeRegion(Counter::getCounter(1), 0, 10, 1, 10, 2),
101       // This starts earlier in file 0, so the expansion should get its counter.
102       CounterMappingRegion::makeRegion(Counter::getCounter(2), 0, 1, 1, 20, 1),
103       CounterMappingRegion::makeExpansion(1, 0, 3, 3, 3, 3),
104   };
105   std::string Coverage = writeCoverage(InputRegions, NumFiles);
106   std::vector<CounterMappingRegion> OutputRegions =
107       readCoverageRegions(Coverage, NumFiles);
108   ASSERT_FALSE(OutputRegions.empty());
109
110   ASSERT_EQ(CounterMappingRegion::ExpansionRegion, OutputRegions[2].Kind);
111   ASSERT_EQ(Counter::getCounter(2), OutputRegions[2].Count);
112   ASSERT_EQ(3U, OutputRegions[2].LineStart);
113 }
114
115
116 } // end anonymous namespace