summaryrefslogtreecommitdiffstats
path: root/tester/covoar/Explanations.cc
blob: d9d7a4febe9797d6bdb2c60fd318c9e0c41aed45 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
/*! @file Explanations.cc
 *  @brief Explanations Implementation
 *
 *  This file contains the implementation of the functions
 *  which provide a base level of functionality of a Explanations.
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#include <fstream>

#include <rld.h>

#include "Explanations.h"

namespace Coverage {

  Explanations::Explanations()
  {
  }

  Explanations::~Explanations()
  {
  }

  void Explanations::load( const std::string& explanations )
  {
    std::ifstream explain;
    Explanation   e;
    int           line = 1;

    if ( explanations.empty() ) {
      return;
    }

    explain.open( explanations );
    if ( !explain ) {
      std::ostringstream what;
      what << "Unable to open " << explanations;
      throw rld::error( what, "Explanations::load" );
    }

    std::string input_line;
    while ( 1 ) {
      // Read the starting line of this explanation and
      // skip blank lines between
      do {
        std::getline( explain, input_line );
        if ( explain.fail() ) {
          return;
        }

        line++;
      } while ( input_line.empty() );

      // Have we already seen this one?
      if ( set.find( input_line ) != set.end() ) {
        std::ostringstream what;
        what << "line " << line
             << "contains a duplicate explanation ("
             << input_line << ")";
        throw rld::error( what, "Explanations::load" );
      }

      // Add the starting line and file
      e.startingPoint = input_line;
      e.found = false;

      // Get the classification
      std::getline( explain, input_line );
      if ( explain.fail() ) {
        std::ostringstream what;
        what << "line " << line
             << "out of sync at the classification";
        throw rld::error( what, "Explanations::load" );
      }
      e.classification = input_line;
      line++;

      // Get the explanation
      while ( std::getline( explain, input_line ) ) {
        line++;

        const std::string delimiter = "+++";
        if ( input_line.compare( delimiter ) == 0 ) {
          break;
        }
        // XXX only taking last line.  Needs to be a vector
        e.explanation.push_back( input_line );
      }

      if ( explain.fail() ) {
        std::ostringstream what;
        what << "line " << line
              << "out of sync at the explanation";
        throw rld::error( what, "Explanations::load" );
      }

      // Add this to the set of Explanations
      set[ e.startingPoint ] = e;
    }

  }

  const Explanation *Explanations::lookupExplanation(
    const std::string& start
  )
  {
    if ( set.find( start ) == set.end() ) {
      #if 0
        std::cerr << "Warning: Unable to find explanation for "
                  << start << std::endl;
      #endif
      return NULL;
    }
    set[ start ].found = true;
    return &set[ start ];
  }

  void Explanations::writeNotFound( const std::string& fileName )
  {
    std::ofstream notFoundFile;
    bool  notFoundOccurred = false;

    if ( fileName.empty() ) {
      return;
    }

    notFoundFile.open( fileName );
    if ( !notFoundFile ) {
      std::ostringstream what;
      what << "Unable to open " << fileName
           << " out of sync at the explanation";
      throw rld::error( what, "Explanations::writeNotFound" );
    }

    for (
      std::map<std::string, Explanation>::iterator itr = set.begin();
      itr != set.end();
      itr++
    ) {
      Explanation e = (*itr).second;
      std::string key = (*itr).first;

      if ( !e.found ) {
        notFoundOccurred = true;
        notFoundFile << e.startingPoint << std::endl;
      }
    }

    if ( !notFoundOccurred ) {
      if ( !unlink( fileName.c_str() ) ) {
        std::cerr << "Warning: Unable to unlink " << fileName
                  << std::endl
                  << std::endl;
      }
    }
  }

}