aboutsummaryrefslogtreecommitdiff
path: root/gcc/gcov.cc
diff options
context:
space:
mode:
Diffstat (limited to 'gcc/gcov.cc')
-rw-r--r--gcc/gcov.cc537
1 files changed, 530 insertions, 7 deletions
diff --git a/gcc/gcov.cc b/gcc/gcov.cc
index 3e6f2e4..e1ad801 100644
--- a/gcc/gcov.cc
+++ b/gcc/gcov.cc
@@ -48,6 +48,7 @@ along with Gcov; see the file COPYING3. If not see
#include "json.h"
#include "hwint.h"
#include "xregex.h"
+#include "graphds.h"
#include <zlib.h>
#include <getopt.h>
@@ -84,6 +85,7 @@ class function_info;
class block_info;
class source_info;
class condition_info;
+class path_info;
/* Describes an arc between two basic blocks. */
@@ -129,6 +131,45 @@ struct arc_info
struct arc_info *pred_next;
};
+/* Describes (prime) path coverage. */
+class path_info
+{
+public:
+ path_info () : paths (), covered () {}
+
+ /* The prime paths of a function. The paths will be
+ lexicographically ordered and identified by their index. */
+ vector<vector<unsigned>> paths;
+
+ /* The covered paths. This is really a large bitset partitioned
+ into buckets of gcov_type_unsigned, and bit n is set if the nth
+ path is covered. */
+ vector<gcov_type_unsigned> covered;
+
+ /* The size (in bits) of each bucket. */
+ static const size_t
+ bucketsize = sizeof (gcov_type_unsigned) * BITS_PER_UNIT;
+
+ /* Count the covered paths. */
+ unsigned covered_paths () const
+ {
+ unsigned cnt = 0;
+ for (gcov_type_unsigned v : covered)
+ cnt += popcount_hwi (v);
+ return cnt;
+ }
+
+ /* Check if the nth path is covered. */
+ bool covered_p (size_t n) const
+ {
+ if (covered.empty ())
+ return false;
+ const size_t bucket = n / bucketsize;
+ const uint64_t bit = n % bucketsize;
+ return covered[bucket] & (gcov_type_unsigned (1) << bit);
+ }
+};
+
/* Describes which locations (lines and files) are associated with
a basic block. */
@@ -317,6 +358,9 @@ public:
vector<condition_info*> conditions;
+ /* Path coverage information. */
+ path_info paths;
+
/* Raw arc coverage counts. */
vector<gcov_type> counts;
@@ -400,6 +444,9 @@ struct coverage_info
int calls_executed;
char *name;
+
+ unsigned paths;
+ unsigned paths_covered;
};
/* Describes a file mentioned in the block graph. Contains an array
@@ -607,6 +654,17 @@ static bool flag_conditions = 0;
/* Show unconditional branches too. */
static int flag_unconditional = 0;
+/* Output path coverage. */
+static bool flag_prime_paths = false;
+
+/* Output path coverage - lines mode. */
+static bool flag_prime_paths_lines_covered = false;
+static bool flag_prime_paths_lines_uncovered = false;
+
+/* Output path coverage - source mode. */
+static bool flag_prime_paths_source_covered = false;
+static bool flag_prime_paths_source_uncovered = false;
+
/* Output a gcov file if this is true. This is on by default, and can
be turned off by the -n option. */
@@ -750,9 +808,11 @@ static unsigned find_source (const char *);
static void read_graph_file (void);
static int read_count_file (void);
static void solve_flow_graph (function_info *);
+static void find_prime_paths (function_info *fn);
static void find_exception_blocks (function_info *);
static void add_branch_counts (coverage_info *, const arc_info *);
static void add_condition_counts (coverage_info *, const block_info *);
+static void add_path_counts (coverage_info &, const function_info &);
static void add_line_counts (coverage_info *, function_info *);
static void executed_summary (unsigned, unsigned);
static void function_summary (const coverage_info *);
@@ -801,6 +861,17 @@ bool function_info::group_line_p (unsigned n, unsigned src_idx)
return is_group && src == src_idx && start_line <= n && n <= end_line;
}
+/* Find the arc that connects BLOCK to the block with id DEST. This
+ edge must exist. */
+static const arc_info&
+find_arc (const block_info &block, unsigned dest)
+{
+ for (const arc_info *arc = block.succ; arc; arc = arc->succ_next)
+ if (arc->dst->id == dest)
+ return *arc;
+ gcc_assert (false);
+}
+
/* Cycle detection!
There are a bajillion algorithms that do this. Boost's function is named
hawick_cycles, so I used the algorithm by K. A. Hawick and H. A. James in
@@ -1030,6 +1101,15 @@ print_usage (int error_p)
rather than percentages\n");
fnotice (file, " -g, --conditions Include modified condition/decision\n\
coverage (masking MC/DC) in output\n");
+ fnotice (file, " -e, --prime-paths Show prime path coverage summary\n");
+ fnotice (file, " --prime-paths-lines[=TYPE] Include paths in output\n\
+ line trace mode - does not affect json\n\
+ TYPE is 'covered', 'uncovered', or 'both'\n\
+ and defaults to 'uncovered'\n");
+ fnotice (file, " --prime-paths-source[=TYPE] Include paths in output\n\
+ source trace mode - does not affect json\n\
+ TYPE is 'covered', 'uncovered', or 'both'\n\
+ and defaults to 'uncovered'\n");
fnotice (file, " -d, --display-progress Display progress information\n");
fnotice (file, " -D, --debug Display debugging dumps\n");
fnotice (file, " -f, --function-summaries Output summaries for each function\n");
@@ -1088,6 +1168,9 @@ static const struct option options[] =
{ "branch-probabilities", no_argument, NULL, 'b' },
{ "branch-counts", no_argument, NULL, 'c' },
{ "conditions", no_argument, NULL, 'g' },
+ { "prime-paths", no_argument, NULL, 'e' },
+ { "prime-paths-lines", optional_argument, NULL, 900 },
+ { "prime-paths-source", optional_argument, NULL, 901 },
{ "json-format", no_argument, NULL, 'j' },
{ "include", required_argument, NULL, 'I' },
{ "exclude", required_argument, NULL, 'E' },
@@ -1119,7 +1202,7 @@ process_args (int argc, char **argv)
{
int opt;
- const char *opts = "abcdDfghHijklmMno:pqrs:tuvwx";
+ const char *opts = "abcdDefghHijklmMno:pqrs:tuvwx";
while ((opt = getopt_long (argc, argv, opts, options, NULL)) != -1)
{
switch (opt)
@@ -1133,6 +1216,51 @@ process_args (int argc, char **argv)
case 'c':
flag_counts = 1;
break;
+ case 'e':
+ flag_prime_paths = true;
+ break;
+ case 900:
+ flag_prime_paths = true;
+ if (!optarg)
+ flag_prime_paths_lines_uncovered = true;
+ else if (strcmp (optarg, "uncovered") == 0)
+ flag_prime_paths_lines_uncovered = true;
+ else if (strcmp (optarg, "covered") == 0)
+ flag_prime_paths_lines_covered = true;
+ else if (strcmp (optarg, "both") == 0)
+ {
+ flag_prime_paths_lines_covered = true;
+ flag_prime_paths_lines_uncovered = true;
+ }
+ else
+ {
+ fnotice (stderr, "invalid argument '%s' for "
+ "'--prime-paths-lines'. Valid arguments are: "
+ "'covered', 'uncovered', 'both'\n", optarg);
+ exit (FATAL_EXIT_CODE);
+ }
+ break;
+ case 901:
+ flag_prime_paths = true;
+ if (!optarg)
+ flag_prime_paths_source_uncovered = true;
+ else if (strcmp (optarg, "uncovered") == 0)
+ flag_prime_paths_source_uncovered = true;
+ else if (strcmp (optarg, "covered") == 0)
+ flag_prime_paths_source_covered = true;
+ else if (strcmp (optarg, "both") == 0)
+ {
+ flag_prime_paths_source_covered = true;
+ flag_prime_paths_source_uncovered = true;
+ }
+ else
+ {
+ fnotice (stderr, "invalid argument '%s' for "
+ "'--prime-paths-source'. Valid arguments are: "
+ "'covered', 'uncovered', 'both'\n", optarg);
+ exit (FATAL_EXIT_CODE);
+ }
+ break;
case 'f':
flag_function_summary = 1;
break;
@@ -1366,7 +1494,7 @@ get_md5sum (const char *input)
static string
get_gcov_intermediate_filename (const char *input_file_name)
{
- string base = basename (input_file_name);
+ string base = lbasename (input_file_name);
string str = strip_extention (base);
if (flag_hash_filenames)
@@ -1385,6 +1513,72 @@ get_gcov_intermediate_filename (const char *input_file_name)
return str.c_str ();
}
+/* Add prime path coverage from INFO to FUNCTION. */
+static void
+json_set_prime_path_coverage (json::object &function, function_info &info)
+{
+ json::array *jpaths = new json::array ();
+ function.set_integer ("total_prime_paths", info.paths.paths.size ());
+ function.set_integer ("covered_prime_paths", info.paths.covered_paths ());
+ function.set ("prime_path_coverage", jpaths);
+
+ size_t pathno = 0;
+ for (const vector<unsigned> &path : info.paths.paths)
+ {
+ if (info.paths.covered_p (pathno++))
+ continue;
+
+ gcc_assert (!path.empty ());
+
+ json::object *jpath = new json::object ();
+ jpaths->append (jpath);
+ jpath->set_integer ("id", pathno - 1);
+
+ json::array *jlist = new json::array ();
+ jpath->set ("sequence", jlist);
+
+ for (size_t i = 0; i != path.size (); ++i)
+ {
+ const unsigned bb = path[i];
+ const block_info &block = info.blocks[bb];
+ const char *edge_kind = "";
+ if (i + 1 != path.size ())
+ {
+ const arc_info &arc = find_arc (block, path[i+1]);
+ if (arc.false_value)
+ edge_kind = "true";
+ else if (arc.false_value)
+ edge_kind = "false";
+ else if (arc.fall_through)
+ edge_kind = "fallthru";
+ else if (arc.is_throw)
+ edge_kind = "throw";
+ }
+
+ json::object *jblock = new json::object ();
+ json::array *jlocs = new json::array ();
+ jblock->set_integer ("block_id", block.id);
+ jblock->set ("locations", jlocs);
+ jblock->set_string ("edge_kind", edge_kind);
+ jlist->append (jblock);
+ for (const block_location_info &loc : block.locations)
+ {
+ /* loc.lines could be empty when a statement is not anchored to a
+ source file -- see g++.dg/gcov/gcov-23.C. */
+ if (loc.lines.empty ())
+ continue;
+ json::object *jloc = new json::object ();
+ json::array *jline_numbers = new json::array ();
+ jlocs->append (jloc);
+ jloc->set_string ("file", sources[loc.source_file_idx].name);
+ jloc->set ("line_numbers", jline_numbers);
+ for (unsigned line : loc.lines)
+ jline_numbers->append (new json::integer_number (line));
+ }
+ }
+ }
+}
+
/* Output the result in JSON intermediate format.
Source info SRC is dumped into JSON_FILES which is JSON array. */
@@ -1415,6 +1609,7 @@ output_json_intermediate_file (json::array *json_files, source_info *src)
function->set_integer ("blocks_executed", (*it)->blocks_executed);
function->set_integer ("execution_count", (*it)->blocks[0].count);
+ json_set_prime_path_coverage (*function, **it);
functions->append (function);
}
@@ -1629,6 +1824,9 @@ process_all_functions (void)
solve_flow_graph (fn);
if (fn->has_catch)
find_exception_blocks (fn);
+
+ /* For path coverage. */
+ find_prime_paths (fn);
}
else
{
@@ -1687,11 +1885,21 @@ generate_results (const char *file_name)
memset (&coverage, 0, sizeof (coverage));
coverage.name = fn->get_name ();
add_line_counts (flag_function_summary ? &coverage : NULL, fn);
- if (flag_function_summary)
- {
- function_summary (&coverage);
- fnotice (stdout, "\n");
- }
+
+ if (!flag_function_summary)
+ continue;
+
+ for (const block_info& block : fn->blocks)
+ for (arc_info *arc = block.succ; arc; arc = arc->succ_next)
+ add_branch_counts (&coverage, arc);
+
+ for (const block_info& block : fn->blocks)
+ add_condition_counts (&coverage, &block);
+
+ add_path_counts (coverage, *fn);
+
+ function_summary (&coverage);
+ fnotice (stdout, "\n");
}
name_map needle;
@@ -1734,6 +1942,9 @@ generate_results (const char *file_name)
continue;
}
+ for (function_info *fn : src->functions)
+ add_path_counts (src->coverage, *fn);
+
accumulate_line_counts (src);
if (flag_debug)
src->debug ();
@@ -2193,6 +2404,13 @@ read_graph_file (void)
info->n_terms = gcov_read_unsigned ();
fn->conditions[i] = info;
}
+ }
+ else if (fn && tag == GCOV_TAG_PATHS)
+ {
+ const unsigned npaths = gcov_read_unsigned ();
+ const size_t nbits = path_info::bucketsize;
+ const size_t nbuckets = (npaths + (nbits - 1)) / nbits;
+ fn->paths.covered.assign (nbuckets, 0);
}
else if (fn && tag == GCOV_TAG_LINES)
{
@@ -2349,6 +2567,17 @@ read_count_file (void)
for (ix = 0; ix != fn->counts.size (); ix++)
fn->counts[ix] += gcov_read_counter ();
}
+ else if (tag == GCOV_TAG_FOR_COUNTER (GCOV_COUNTER_PATHS) && fn)
+ {
+ vector<gcov_type_unsigned> &covered = fn->paths.covered;
+ length = abs (read_length);
+ if (length != GCOV_TAG_COUNTER_LENGTH (covered.size ()))
+ goto mismatch;
+
+ if (read_length > 0)
+ for (ix = 0; ix != covered.size (); ix++)
+ covered[ix] = gcov_read_counter ();
+ }
if (read_length < 0)
read_length = 0;
gcov_sync (base, read_length);
@@ -2632,6 +2861,67 @@ solve_flow_graph (function_info *fn)
}
}
+/* Find the prime paths of the function from the CFG and add to FN
+ using the same function as gcc. It relies on gcc recording the CFG
+ faithfully. Storing the paths explicitly takes up way too much
+ space to be practical, but this means we need to recompute the
+ (exact) same paths in gcov. This should give paths in
+ lexicographical order so that the nth path in gcc is the nth path
+ in gcov. ENTRY_BLOCK and EXIT_BLOCK are both removed from all
+ paths. */
+static void
+find_prime_paths (function_info *fn)
+{
+ if (!flag_prime_paths)
+ return;
+
+ /* If paths.covered being empty then this function was not
+ instrumented, probably because it exceeded #-of-paths limit. In
+ this case we don't want to find the prime paths as it will take
+ too long, and covered paths are not measured. */
+ if (fn->paths.covered.empty ())
+ return;
+
+ struct graph *cfg = new_graph (fn->blocks.size ());
+ for (block_info &block : fn->blocks)
+ {
+ cfg->vertices[block.id].data = &block;
+ for (arc_info *arc = block.succ; arc; arc = arc->succ_next)
+ if (!arc->fake)
+ add_edge (cfg, arc->src->id, arc->dst->id)->data = arc;
+ }
+
+ vec<vec<int>> prime_paths (struct graph*, size_t);
+ /* TODO: Pass extra information in the PATH_TAG section. In case
+ that is empty this might still need to be tunable should the
+ coverage be requested without instrumentation. */
+ vec<vec<int>> paths = prime_paths (cfg, (size_t)-1);
+ fn->paths.paths.reserve (paths.length ());
+ for (vec<int> &path : paths)
+ {
+ const int *begin = path.begin ();
+ const int *end = path.end ();
+ if (begin != end && path.last () == EXIT_BLOCK)
+ --end;
+ if (begin != end && *begin == ENTRY_BLOCK)
+ ++begin;
+
+ if (begin == end)
+ continue;
+
+ /* If this is an isolated vertex because abnormal edges and fake
+ edges are removed, don't include it. */
+ if (end - begin == 1 && !cfg->vertices[*begin].succ
+ && !cfg->vertices[*begin].pred)
+ continue;
+
+ fn->paths.paths.emplace_back (begin, end);
+ }
+
+ release_vec_vec (paths);
+ free_graph (cfg);
+}
+
/* Mark all the blocks only reachable via an incoming catch. */
static void
@@ -2692,6 +2982,16 @@ add_condition_counts (coverage_info *coverage, const block_info *block)
coverage->conditions_covered += block->conditions.popcount ();
}
+/* Increment path totals, number of paths and number of covered paths,
+ in COVERAGE according to FN. */
+
+static void
+add_path_counts (coverage_info &coverage, const function_info &fn)
+{
+ coverage.paths += fn.paths.paths.size ();
+ coverage.paths_covered += fn.paths.covered_paths ();
+}
+
/* Format COUNT, if flag_human_readable_numbers is set, return it human
readable format. */
@@ -2764,6 +3064,46 @@ function_summary (const coverage_info *coverage)
{
fnotice (stdout, "%s '%s'\n", "Function", coverage->name);
executed_summary (coverage->lines, coverage->lines_executed);
+
+ if (coverage->branches)
+ {
+ fnotice (stdout, "Branches executed:%s of %d\n",
+ format_gcov (coverage->branches_executed, coverage->branches, 2),
+ coverage->branches);
+ fnotice (stdout, "Taken at least once:%s of %d\n",
+ format_gcov (coverage->branches_taken, coverage->branches, 2),
+ coverage->branches);
+ }
+ else
+ fnotice (stdout, "No branches\n");
+
+ if (coverage->calls)
+ fnotice (stdout, "Calls executed:%s of %d\n",
+ format_gcov (coverage->calls_executed, coverage->calls, 2),
+ coverage->calls);
+ else
+ fnotice (stdout, "No calls\n");
+
+ if (flag_conditions)
+ {
+ if (coverage->conditions)
+ fnotice (stdout, "Condition outcomes covered:%s of %d\n",
+ format_gcov (coverage->conditions_covered,
+ coverage->conditions, 2),
+ coverage->conditions);
+ else
+ fnotice (stdout, "No conditions\n");
+ }
+
+ if (flag_prime_paths)
+ {
+ if (coverage->paths)
+ fnotice (stdout, "Prime paths covered:%s of %d\n",
+ format_gcov (coverage->paths_covered, coverage->paths, 2),
+ coverage->paths);
+ else
+ fnotice (stdout, "No path information\n");
+ }
}
/* Output summary info for a file. */
@@ -2808,6 +3148,16 @@ file_summary (const coverage_info *coverage)
else
fnotice (stdout, "No conditions\n");
}
+
+ if (flag_prime_paths)
+ {
+ if (coverage->paths)
+ fnotice (stdout, "Prime paths covered:%s of %d\n",
+ format_gcov (coverage->paths_covered, coverage->paths, 2),
+ coverage->paths);
+ else
+ fnotice (stdout, "No path information\n");
+ }
}
/* Canonicalize the filename NAME by canonicalizing directory
@@ -3226,6 +3576,177 @@ output_branch_count (FILE *gcov_file, int ix, const arc_info *arc)
return 1;
}
+static void
+print_source_line (FILE *f, const vector<const char *> &source_lines,
+ unsigned line);
+
+
+/* Print a dense coverage report for PATH of FN to GCOV_FILE. PATH should be
+ number PATHNO in the sorted set of paths. This function prints a dense form
+ where only the line numbers, and optionally the source file the line comes
+ from, in the order they need to be executed to achieve coverage. This
+ produces very long lines for large functions, but is a useful and greppable
+ output.
+
+ Returns 1 if the path was printed, 0 otherwise. */
+static unsigned
+print_prime_path_lines (FILE *gcov_file, const function_info &fn,
+ const vector<unsigned> &path, size_t pathno)
+{
+ const bool is_covered = fn.paths.covered_p (pathno);
+ if (is_covered && !flag_prime_paths_lines_covered)
+ return 0;
+ if (!is_covered && !flag_prime_paths_lines_uncovered)
+ return 0;
+
+ if (is_covered)
+ fprintf (gcov_file, "path %zu covered: lines", pathno);
+ else
+ fprintf (gcov_file, "path %zu not covered: lines", pathno);
+
+ for (size_t k = 0; k != path.size (); ++k)
+ {
+ const block_info &block = fn.blocks[path[k]];
+ const char *edge_kind = "";
+ if (k + 1 != path.size ())
+ {
+ gcc_checking_assert (block.id == path[k]);
+ const arc_info &arc = find_arc (block, path[k+1]);
+ if (arc.true_value)
+ edge_kind = "(true)";
+ else if (arc.false_value)
+ edge_kind = "(false)";
+ else if (arc.is_throw)
+ edge_kind = "(throw)";
+ }
+
+ for (const block_location_info &loc : block.locations)
+ {
+ /* loc.lines could be empty when a statement is not anchored to a
+ source file -- see g++.dg/gcov/gcov-23.C. Since there is no
+ actual source line to list anyway we can skip this location. */
+ if (loc.lines.empty ())
+ continue;
+ if (loc.source_file_idx == fn.src)
+ fprintf (gcov_file, " %u%s", loc.lines.back (), edge_kind);
+ else
+ fprintf (gcov_file, " %s:%u%s", sources[loc.source_file_idx].name,
+ loc.lines.back (), edge_kind);
+ }
+ }
+
+ fprintf (gcov_file, "\n");
+ return 1;
+}
+
+static unsigned
+print_inlined_separator (FILE *gcov_file, unsigned current_index, const
+ block_location_info &loc, const function_info &fn)
+{
+ if (loc.source_file_idx != current_index && loc.source_file_idx == fn.src)
+ fprintf (gcov_file, "------------------\n");
+ if (loc.source_file_idx != current_index && loc.source_file_idx != fn.src)
+ fprintf (gcov_file, "== inlined from %s ==\n",
+ sources[loc.source_file_idx].name);
+ return loc.source_file_idx;
+}
+
+/* Print a coverage report for PATH of FN to GCOV_FILE. PATH should be number
+ PATHNO in the sorted set of paths. This function prints the lines that need
+ to be executed (and in what order) to cover it.
+
+ Returns 1 if the path was printed, 0 otherwise. */
+static unsigned
+print_prime_path_source (FILE *gcov_file, const function_info &fn,
+ const vector<unsigned> &path, size_t pathno)
+{
+ const bool is_covered = fn.paths.covered_p (pathno);
+ if (is_covered && !flag_prime_paths_source_covered)
+ return 0;
+ if (!is_covered && !flag_prime_paths_source_uncovered)
+ return 0;
+
+ if (is_covered)
+ fprintf (gcov_file, "path %zu covered:\n", pathno);
+ else
+ fprintf (gcov_file, "path %zu not covered:\n", pathno);
+ unsigned current = fn.src;
+ for (size_t k = 0; k != path.size (); ++k)
+ {
+ const unsigned bb = path[k];
+ const block_info &block = fn.blocks[bb];
+ gcc_checking_assert (block.id == bb);
+
+ const char *edge_kind = "";
+ if (k + 1 != path.size ())
+ {
+ const arc_info &arc = find_arc (block, path[k+1]);
+ if (arc.true_value)
+ edge_kind = "(true)";
+ else if (arc.false_value)
+ edge_kind = "(false)";
+ else if (arc.is_throw)
+ edge_kind = "(throw)";
+ }
+
+ for (const block_location_info &loc : block.locations)
+ {
+ /* loc.lines could be empty when a statement is not anchored to a
+ source file -- see g++.dg/gcov/gcov-24.C. Since there is no
+ actual source line to list anyway we can skip this location. */
+ if (loc.lines.empty ())
+ continue;
+ const source_info &src = sources[loc.source_file_idx];
+ const vector<const char *> &lines = slurp (src, gcov_file, "");
+ current = print_inlined_separator (gcov_file, current, loc, fn);
+ for (unsigned i = 0; i != loc.lines.size () - 1; ++i)
+ {
+ const unsigned line = loc.lines[i];
+ fprintf (gcov_file, "BB %2d: %-7s %3d", bb, "", line);
+ print_source_line (gcov_file, lines, line);
+ }
+
+ const unsigned line = loc.lines.back ();
+ fprintf (gcov_file, "BB %2d: %-7s %3d", bb, edge_kind, line);
+ print_source_line (gcov_file, lines, line);
+ }
+ }
+
+ fputc ('\n', gcov_file);
+ return 1;
+}
+
+/* Print path coverage counts for FN to GCOV_FILE. LINES is the vector of
+ source lines for FN. Note that unlike statements, branch counts, and
+ conditions, this is not anchored to source lines but the function root. */
+static int
+output_path_coverage (FILE *gcov_file, const function_info *fn)
+{
+ if (!flag_prime_paths)
+ return 0;
+
+ if (fn->paths.paths.empty ())
+ fnotice (gcov_file, "path coverage omitted\n");
+ else
+ fnotice (gcov_file, "paths covered %u of %zu\n",
+ fn->paths.covered_paths (), fn->paths.paths.size ());
+
+ if (flag_prime_paths_lines_uncovered || flag_prime_paths_lines_covered)
+ {
+ size_t pathno = 0;
+ for (const vector<unsigned> &path : fn->paths.paths)
+ print_prime_path_lines (gcov_file, *fn, path, pathno++);
+ }
+
+ if (flag_prime_paths_source_uncovered || flag_prime_paths_source_covered)
+ {
+ size_t pathno = 0;
+ for (const vector<unsigned> &path : fn->paths.paths)
+ print_prime_path_source (gcov_file, *fn, path, pathno++);
+ }
+ return 1;
+}
+
static const char *
read_line (FILE *file)
{
@@ -3560,6 +4081,7 @@ output_lines (FILE *gcov_file, const source_info *src)
{
function_info *fn = (*fns)[0];
output_function_details (gcov_file, fn);
+ output_path_coverage (gcov_file, fn);
/* If functions are filtered, only the matching functions will be in
fns and there is no need for extra checking. */
@@ -3605,6 +4127,7 @@ output_lines (FILE *gcov_file, const source_info *src)
fprintf (gcov_file, "%s:\n", fn_name.c_str ());
output_function_details (gcov_file, fn);
+ output_path_coverage (gcov_file, fn);
/* Print all lines covered by the function. */
for (unsigned i = 0; i < lines.size (); i++)