diff options
Diffstat (limited to 'gcc/profile.c')
-rw-r--r-- | gcc/profile.c | 1154 |
1 files changed, 748 insertions, 406 deletions
diff --git a/gcc/profile.c b/gcc/profile.c index a70dc617621..4e6aecd6b64 100644 --- a/gcc/profile.c +++ b/gcc/profile.c @@ -39,36 +39,19 @@ Software Foundation, 59 Temple Place - Suite 330, Boston, MA edges must be on the spanning tree. We also attempt to place EDGE_CRITICAL edges on the spanning tree. - The two auxiliary files generated are <dumpbase>.bb and - <dumpbase>.bbg. The former contains the BB->linenumber - mappings, and the latter describes the BB graph. - - The BB file contains line numbers for each block. For each basic - block, a zero count is output (to mark the start of a block), then - the line numbers of that block are listed. A zero ends the file - too. - - The BBG file contains a count of the blocks, followed by edge - information, for every edge in the graph. The edge information - lists the source and target block numbers, and a bit mask - describing the type of edge. - - The BB and BBG file formats are fully described in the gcov - documentation. */ + The auxiliary file generated is <dumpbase>.bbg. The format is + described in full in gcov-io.h. */ /* ??? Register allocation should use basic block execution counts to give preference to the most commonly executed blocks. */ -/* ??? The .da files are not safe. Changing the program after creating .da - files or using different options when compiling with -fbranch-probabilities - can result the arc data not matching the program. Maybe add instrumented - arc count to .bbg file? Maybe check whether PFG matches the .bbg file? */ - /* ??? Should calculate branch probabilities before instrumenting code, since then we can use arc counts to help decide which arcs to instrument. */ #include "config.h" #include "system.h" +#include "coretypes.h" +#include "tm.h" #include "rtl.h" #include "tree.h" #include "flags.h" @@ -86,6 +69,7 @@ Software Foundation, 59 Temple Place - Suite 330, Boston, MA #include "profile.h" #include "libfuncs.h" #include "langhooks.h" +#include "hashtab.h" /* Additional information about the edges we need. */ struct edge_info { @@ -107,6 +91,17 @@ struct bb_info { gcov_type pred_count; }; +struct function_list +{ + struct function_list *next; /* next function */ + const char *name; /* function name */ + unsigned cfg_checksum; /* function checksum */ + unsigned count_edges; /* number of intrumented edges */ +}; + +static struct function_list *functions_head = 0; +static struct function_list **functions_tail = &functions_head; + #define EDGE_INFO(e) ((struct edge_info *) (e)->aux) #define BB_INFO(b) ((struct bb_info *) (b)->aux) @@ -123,18 +118,15 @@ struct profile_info profile_info; /* Name and file pointer of the output file for the basic block graph. */ static FILE *bbg_file; +static char *bbg_file_name; /* Name and file pointer of the input file for the arc count data. */ static FILE *da_file; static char *da_file_name; -/* Pointer of the output file for the basic block/line number map. */ -static FILE *bb_file; - -/* Last source file name written to bb_file. */ - -static char *last_bb_file_name; +/* The name of the count table. Used by the edge profiling code. */ +static GTY(()) rtx profiler_label; /* Collect statistics on the performance of this pass for the entire source file. */ @@ -152,19 +144,19 @@ static int total_num_branches; /* Forward declarations. */ static void find_spanning_tree PARAMS ((struct edge_list *)); -static void init_edge_profiler PARAMS ((void)); static rtx gen_edge_profiler PARAMS ((int)); static void instrument_edges PARAMS ((struct edge_list *)); -static void output_gcov_string PARAMS ((const char *, long)); static void compute_branch_probabilities PARAMS ((void)); +static hashval_t htab_counts_index_hash PARAMS ((const void *)); +static int htab_counts_index_eq PARAMS ((const void *, const void *)); +static void htab_counts_index_del PARAMS ((void *)); +static void cleanup_counts_index PARAMS ((int)); +static int index_counts_file PARAMS ((void)); static gcov_type * get_exec_counts PARAMS ((void)); -static long compute_checksum PARAMS ((void)); +static unsigned compute_checksum PARAMS ((void)); static basic_block find_group PARAMS ((basic_block)); static void union_groups PARAMS ((basic_block, basic_block)); -/* If nonzero, we need to output a constructor to set up the - per-object-file data. */ -static int need_func_profiler = 0; /* Add edge instrumentation code to the entire insn chain. @@ -194,7 +186,6 @@ instrument_edges (el) fprintf (rtl_dump_file, "Edge %d to %d instrumented%s\n", e->src->index, e->dest->index, EDGE_CRITICAL_P (e) ? " (and split)" : ""); - need_func_profiler = 1; insert_insn_on_edge ( gen_edge_profiler (total_num_edges_instrumented + num_instr_edges++), e); @@ -213,38 +204,212 @@ instrument_edges (el) commit_edge_insertions_watch_calls (); } + +struct section_reference +{ + long offset; + int owns_summary; + long *summary; +}; + +struct da_index_entry +{ + /* We hash by */ + char *function_name; + unsigned section; + /* and store */ + unsigned checksum; + unsigned n_offsets; + struct section_reference *offsets; +}; + +static hashval_t +htab_counts_index_hash (of) + const void *of; +{ + const struct da_index_entry *entry = of; + + return htab_hash_string (entry->function_name) ^ entry->section; +} -/* Output STRING to bb_file, surrounded by DELIMITER. */ +static int +htab_counts_index_eq (of1, of2) + const void *of1; + const void *of2; +{ + const struct da_index_entry *entry1 = of1; + const struct da_index_entry *entry2 = of2; + + return !strcmp (entry1->function_name, entry2->function_name) + && entry1->section == entry2->section; +} static void -output_gcov_string (string, delimiter) - const char *string; - long delimiter; +htab_counts_index_del (what) + void *what; { - size_t temp; + struct da_index_entry *entry = what; + unsigned i; - /* Write a delimiter to indicate that a file name follows. */ - __write_long (delimiter, bb_file, 4); + for (i = 0; i < entry->n_offsets; i++) + { + struct section_reference *act = entry->offsets + i; + if (act->owns_summary) + free (act->summary); + } + free (entry->function_name); + free (entry->offsets); + free (entry); +} - /* Write the string. */ - temp = strlen (string) + 1; - fwrite (string, temp, 1, bb_file); +static char *counts_file_name; +static htab_t counts_file_index = NULL; - /* Append a few zeros, to align the output to a 4 byte boundary. */ - temp = temp & 0x3; - if (temp) +static void +cleanup_counts_index (close_file) + int close_file; +{ + if (da_file && close_file) { - char c[4]; + fclose (da_file); + da_file = NULL; + } + if (counts_file_name) + free (counts_file_name); + counts_file_name = NULL; + if (counts_file_index) + htab_delete (counts_file_index); + counts_file_index = NULL; +} + +static int +index_counts_file () +{ + char *function_name_buffer = NULL; + unsigned magic, version, ix, checksum; + long *summary; + + if (!da_file) + return 0; + counts_file_index = htab_create (10, htab_counts_index_hash, htab_counts_index_eq, htab_counts_index_del); - c[0] = c[1] = c[2] = c[3] = 0; - fwrite (c, sizeof (char), 4 - temp, bb_file); + /* No .da file, no data. */ + if (!da_file) + return 0; + + /* Now index all profile sections. */ + + rewind (da_file); + + summary = NULL; + + if (gcov_read_unsigned (da_file, &magic) || magic != GCOV_DATA_MAGIC) + { + warning ("`%s' is not a gcov data file", da_file_name); + goto cleanup; + } + if (gcov_read_unsigned (da_file, &version) || version != GCOV_VERSION) + { + char v[4], e[4]; + magic = GCOV_VERSION; + + for (ix = 4; ix--; magic >>= 8, version >>= 8) + { + v[ix] = version; + e[ix] = magic; + } + warning ("`%s' is version `%.4s', expected version `%.4s'", + da_file_name, v, e); + goto cleanup; } + + while (1) + { + unsigned tag, length; + long offset; + + offset = gcov_save_position (da_file); + if (gcov_read_unsigned (da_file, &tag) + || gcov_read_unsigned (da_file, &length)) + { + if (feof (da_file)) + break; + corrupt:; + warning ("`%s' is corrupted", da_file_name); + goto cleanup; + } + if (tag == GCOV_TAG_FUNCTION) + { + if (gcov_read_string (da_file, &function_name_buffer, NULL) + || gcov_read_unsigned (da_file, &checksum)) + goto corrupt; + continue; + } + if (tag == GCOV_TAG_PROGRAM_SUMMARY) + { + if (length != GCOV_SUMMARY_LENGTH) + goto corrupt; - /* Store another delimiter in the .bb file, just to make it easy to find - the end of the file name. */ - __write_long (delimiter, bb_file, 4); + if (summary) + *summary = offset; + summary = NULL; + } + else + { + if (function_name_buffer) + { + struct da_index_entry **slot, elt; + elt.function_name = function_name_buffer; + elt.section = tag; + + slot = (struct da_index_entry **) + htab_find_slot (counts_file_index, &elt, INSERT); + if (*slot) + { + if ((*slot)->checksum != checksum) + { + warning ("profile mismatch for `%s'", function_name_buffer); + goto cleanup; + } + (*slot)->n_offsets++; + (*slot)->offsets = xrealloc ((*slot)->offsets, + sizeof (struct section_reference) * (*slot)->n_offsets); + } + else + { + *slot = xmalloc (sizeof (struct da_index_entry)); + (*slot)->function_name = xstrdup (function_name_buffer); + (*slot)->section = tag; + (*slot)->checksum = checksum; + (*slot)->n_offsets = 1; + (*slot)->offsets = xmalloc (sizeof (struct section_reference)); + } + (*slot)->offsets[(*slot)->n_offsets - 1].offset = offset; + if (summary) + (*slot)->offsets[(*slot)->n_offsets - 1].owns_summary = 0; + else + { + summary = xmalloc (sizeof (long)); + *summary = -1; + (*slot)->offsets[(*slot)->n_offsets - 1].owns_summary = 1; + } + (*slot)->offsets[(*slot)->n_offsets - 1].summary = summary; + } + } + if (gcov_skip (da_file, length)) + goto corrupt; + } + + free (function_name_buffer); + + return 1; + +cleanup: + cleanup_counts_index (1); + if (function_name_buffer) + free (function_name_buffer); + return 0; } - /* Computes hybrid profile for all matching entries in da_file. Sets max_counter_in_program as a side effect. */ @@ -252,23 +417,25 @@ output_gcov_string (string, delimiter) static gcov_type * get_exec_counts () { - int num_edges = 0; + unsigned num_edges = 0; basic_block bb; - int okay = 1, i; - int mismatch = 0; gcov_type *profile; - char *function_name_buffer; - int function_name_buffer_len; - gcov_type max_counter_in_run; - const char *name = IDENTIFIER_POINTER - (DECL_ASSEMBLER_NAME (current_function_decl)); + gcov_type max_count; + unsigned ix, i, tag, length, num; + const char *name = IDENTIFIER_POINTER (DECL_ASSEMBLER_NAME (current_function_decl)); + struct da_index_entry *entry, what; + struct section_reference *act; + gcov_type count; + struct gcov_summary summ; profile_info.max_counter_in_program = 0; profile_info.count_profiles_merged = 0; /* No .da file, no execution counts. */ if (!da_file) - return 0; + return NULL; + if (!counts_file_index) + abort (); /* Count the edges to be (possibly) instrumented. */ @@ -283,124 +450,79 @@ get_exec_counts () /* now read and combine all matching profiles. */ profile = xmalloc (sizeof (gcov_type) * num_edges); - rewind (da_file); - function_name_buffer_len = strlen (name) + 1; - function_name_buffer = xmalloc (function_name_buffer_len + 1); - for (i = 0; i < num_edges; i++) - profile[i] = 0; + for (ix = 0; ix < num_edges; ix++) + profile[ix] = 0; - while (1) + what.function_name = (char *) name; + what.section = GCOV_TAG_ARC_COUNTS; + entry = htab_find (counts_file_index, &what); + if (!entry) { - long magic, extra_bytes; - long func_count; - int i; + warning ("No profile for function '%s' found.", name); + goto cleanup; + } + + if (entry->checksum != profile_info.current_function_cfg_checksum) + { + warning ("profile mismatch for `%s'", current_function_name); + goto cleanup; + } - if (__read_long (&magic, da_file, 4) != 0) - break; + for (i = 0; i < entry->n_offsets; i++) + { + act = entry->offsets + i; - if (magic != -123) + /* Read arc counters. */ + max_count = 0; + gcov_resync (da_file, act->offset, 0); + + if (gcov_read_unsigned (da_file, &tag) + || gcov_read_unsigned (da_file, &length) + || tag != GCOV_TAG_ARC_COUNTS) { - okay = 0; - break; + /* We have already passed through file, so any error means + something is rotten. */ + abort (); } + num = length / 8; - if (__read_long (&func_count, da_file, 4) != 0) + if (num != num_edges) { - okay = 0; - break; + warning ("profile mismatch for `%s'", current_function_name); + goto cleanup; } - - if (__read_long (&extra_bytes, da_file, 4) != 0) + + for (ix = 0; ix != num; ix++) { - okay = 0; - break; + if (gcov_read_counter (da_file, &count)) + abort (); + if (count > max_count) + max_count = count; + profile[ix] += count; } - fseek (da_file, 4 + 8, SEEK_CUR); - - /* read the maximal counter. */ - __read_gcov_type (&max_counter_in_run, da_file, 8); - - /* skip the rest of "statistics" emited by __bb_exit_func. */ - fseek (da_file, extra_bytes - (4 + 8 + 8), SEEK_CUR); - - for (i = 0; i < func_count; i++) + /* Read program summary. */ + if (*act->summary != -1) { - long arc_count; - long chksum; - int j; - - if (__read_gcov_string - (function_name_buffer, function_name_buffer_len, da_file, - -1) != 0) - { - okay = 0; - break; - } - - if (__read_long (&chksum, da_file, 4) != 0) - { - okay = 0; - break; - } - - if (__read_long (&arc_count, da_file, 4) != 0) - { - okay = 0; - break; - } - - if (strcmp (function_name_buffer, name) != 0) - { - /* skip */ - if (fseek (da_file, arc_count * 8, SEEK_CUR) < 0) - { - okay = 0; - break; - } - } - else if (arc_count != num_edges - || chksum != profile_info.current_function_cfg_checksum) - okay = 0, mismatch = 1; - else - { - gcov_type tmp; - - profile_info.max_counter_in_program += max_counter_in_run; - profile_info.count_profiles_merged++; - - for (j = 0; j < arc_count; j++) - if (__read_gcov_type (&tmp, da_file, 8) != 0) - { - okay = 0; - break; - } - else - { - profile[j] += tmp; - } - } + gcov_resync (da_file, *act->summary, 0); + if (gcov_read_unsigned (da_file, &tag) + || gcov_read_unsigned (da_file, &length) + || tag != GCOV_TAG_PROGRAM_SUMMARY + || gcov_read_summary (da_file, &summ)) + abort (); + profile_info.count_profiles_merged += summ.runs; + profile_info.max_counter_in_program += summ.arc_sum_max; } - - if (!okay) - break; - - } - - free (function_name_buffer); - - if (!okay) - { - if (mismatch) - error - ("Profile does not match flowgraph of function %s (out of date?)", - current_function_name); else - error (".da file corrupted"); - free (profile); - return 0; + summ.runs = 0; + if (!summ.runs) + { + profile_info.count_profiles_merged++; + profile_info.max_counter_in_program += max_count; + } } + if (rtl_dump_file) { fprintf(rtl_dump_file, "Merged %i profiles with maximal count %i.\n", @@ -409,6 +531,11 @@ get_exec_counts () } return profile; + +cleanup:; + free (profile); + cleanup_counts_index (1); + return NULL; } @@ -669,10 +796,11 @@ compute_branch_probabilities () num_branches++; } } - /* Otherwise distribute the probabilities evenly so we get sane sum. - Use simple heuristics that if there are normal edges, give all abnormals - frequency of 0, otherwise distribute the frequency over abnormals - (this is the case of noreturn calls). */ + /* Otherwise distribute the probabilities evenly so we get sane + sum. Use simple heuristics that if there are normal edges, + give all abnormals frequency of 0, otherwise distribute the + frequency over abnormals (this is the case of noreturn + calls). */ else { for (e = bb->succ; e; e = e->succ_next) @@ -725,30 +853,43 @@ compute_branch_probabilities () free (exec_counts); } -/* Compute checksum for the current function. */ +/* Compute checksum for the current function. We generate a CRC32. */ -#define CHSUM_HASH 500000003 -#define CHSUM_SHIFT 2 - -static long +static unsigned compute_checksum () { - long chsum = 0; + unsigned chksum = 0; basic_block bb; - + FOR_EACH_BB (bb) { - edge e; - - for (e = bb->succ; e; e = e->succ_next) + edge e = NULL; + + do { - chsum = ((chsum << CHSUM_SHIFT) + (BB_TO_GCOV_INDEX (e->dest) + 1)) % CHSUM_HASH; - } + unsigned value = BB_TO_GCOV_INDEX (e ? e->dest : bb); + unsigned ix; + + /* No need to use all bits in value identically, nearly all + functions have less than 256 blocks. */ + value ^= value << 16; + value ^= value << 8; + + for (ix = 8; ix--; value <<= 1) + { + unsigned feedback; - chsum = (chsum << CHSUM_SHIFT) % CHSUM_HASH; + feedback = (value ^ chksum) & 0x80000000 ? 0x04c11db7 : 0; + chksum <<= 1; + chksum ^= feedback; + } + + e = e ? e->succ_next : bb->succ; + } + while (e); } - return chsum; + return chksum; } /* Instrument and/or analyze program behavior based on program flow graph. @@ -775,18 +916,14 @@ branch_prob () int num_edges, ignored_edges; struct edge_list *el; const char *name = IDENTIFIER_POINTER - (DECL_ASSEMBLER_NAME (current_function_decl)); + (DECL_ASSEMBLER_NAME (current_function_decl)); profile_info.current_function_cfg_checksum = compute_checksum (); if (rtl_dump_file) - fprintf (rtl_dump_file, "CFG checksum is %ld\n", + fprintf (rtl_dump_file, "CFG checksum is %u\n", profile_info.current_function_cfg_checksum); - /* Start of a function. */ - if (flag_test_coverage) - output_gcov_string (name, (long) -2); - total_num_times_called++; flow_call_edges_add (NULL); @@ -895,70 +1032,6 @@ branch_prob () verify_flow_info (); #endif - /* Output line number information about each basic block for - GCOV utility. */ - if (flag_test_coverage) - { - FOR_EACH_BB (bb) - { - rtx insn = bb->head; - static int ignore_next_note = 0; - - /* We are looking for line number notes. Search backward before - basic block to find correct ones. */ - insn = prev_nonnote_insn (insn); - if (!insn) - insn = get_insns (); - else - insn = NEXT_INSN (insn); - - /* Output a zero to the .bb file to indicate that a new - block list is starting. */ - __write_long (0, bb_file, 4); - - while (insn != bb->end) - { - if (GET_CODE (insn) == NOTE) - { - /* Must ignore the line number notes that immediately - follow the end of an inline function to avoid counting - it twice. There is a note before the call, and one - after the call. */ - if (NOTE_LINE_NUMBER (insn) == NOTE_INSN_REPEATED_LINE_NUMBER) - ignore_next_note = 1; - else if (NOTE_LINE_NUMBER (insn) > 0) - { - if (ignore_next_note) - ignore_next_note = 0; - else - { - /* If this is a new source file, then output the - file's name to the .bb file. */ - if (! last_bb_file_name - || strcmp (NOTE_SOURCE_FILE (insn), - last_bb_file_name)) - { - if (last_bb_file_name) - free (last_bb_file_name); - last_bb_file_name - = xstrdup (NOTE_SOURCE_FILE (insn)); - output_gcov_string (NOTE_SOURCE_FILE (insn), - (long)-1); - } - /* Output the line number to the .bb file. Must be - done after the output_bb_profile_data() call, and - after the file name is written, to ensure that it - is correctly handled by gcov. */ - __write_long (NOTE_LINE_NUMBER (insn), bb_file, 4); - } - } - } - insn = NEXT_INSN (insn); - } - } - __write_long (0, bb_file, 4); - } - /* Create spanning tree from basic block graph, mark each edge that is on the spanning tree. We insert as many abnormal and critical edges as possible to minimize number of edge splits necessary. */ @@ -995,56 +1068,140 @@ branch_prob () edge output the source and target basic block numbers. NOTE: The format of this file must be compatible with gcov. */ - if (flag_test_coverage) + if (flag_test_coverage && bbg_file) { - int flag_bits; - - __write_gcov_string (name, strlen (name), bbg_file, -1); - - /* write checksum. */ - __write_long (profile_info.current_function_cfg_checksum, bbg_file, 4); - - /* The plus 2 stands for entry and exit block. */ - __write_long (n_basic_blocks + 2, bbg_file, 4); - __write_long (num_edges - ignored_edges + 1, bbg_file, 4); - + long offset; + + /* Announce function */ + if (gcov_write_unsigned (bbg_file, GCOV_TAG_FUNCTION) + || !(offset = gcov_reserve_length (bbg_file)) + || gcov_write_string (bbg_file, name, + strlen (name)) + || gcov_write_unsigned (bbg_file, + profile_info.current_function_cfg_checksum) + || gcov_write_length (bbg_file, offset)) + goto bbg_error; + + /* Basic block flags */ + if (gcov_write_unsigned (bbg_file, GCOV_TAG_BLOCKS) + || !(offset = gcov_reserve_length (bbg_file))) + goto bbg_error; + for (i = 0; i != n_basic_blocks + 2; i++) + if (gcov_write_unsigned (bbg_file, 0)) + goto bbg_error; + if (gcov_write_length (bbg_file, offset)) + goto bbg_error; + + /* Arcs */ FOR_BB_BETWEEN (bb, ENTRY_BLOCK_PTR, EXIT_BLOCK_PTR, next_bb) { edge e; - long count = 0; - for (e = bb->succ; e; e = e->succ_next) - if (!EDGE_INFO (e)->ignore) - count++; - __write_long (count, bbg_file, 4); + if (gcov_write_unsigned (bbg_file, GCOV_TAG_ARCS) + || !(offset = gcov_reserve_length (bbg_file)) + || gcov_write_unsigned (bbg_file, BB_TO_GCOV_INDEX (bb))) + goto bbg_error; for (e = bb->succ; e; e = e->succ_next) { struct edge_info *i = EDGE_INFO (e); if (!i->ignore) { - flag_bits = 0; + unsigned flag_bits = 0; + if (i->on_tree) - flag_bits |= 0x1; + flag_bits |= GCOV_ARC_ON_TREE; if (e->flags & EDGE_FAKE) - flag_bits |= 0x2; + flag_bits |= GCOV_ARC_FAKE; if (e->flags & EDGE_FALLTHRU) - flag_bits |= 0x4; + flag_bits |= GCOV_ARC_FALLTHROUGH; - __write_long (BB_TO_GCOV_INDEX (e->dest), bbg_file, 4); - __write_long (flag_bits, bbg_file, 4); + if (gcov_write_unsigned (bbg_file, + BB_TO_GCOV_INDEX (e->dest)) + || gcov_write_unsigned (bbg_file, flag_bits)) + goto bbg_error; } } + if (gcov_write_length (bbg_file, offset)) + goto bbg_error; } - /* Emit fake loopback edge for EXIT block to maintain compatibility with - old gcov format. */ - __write_long (1, bbg_file, 4); - __write_long (0, bbg_file, 4); - __write_long (0x1, bbg_file, 4); - - /* Emit a -1 to separate the list of all edges from the list of - loop back edges that follows. */ - __write_long (-1, bbg_file, 4); + + /* Output line number information about each basic block for + GCOV utility. */ + { + char const *prev_file_name = NULL; + + FOR_EACH_BB (bb) + { + rtx insn = bb->head; + int ignore_next_note = 0; + + offset = 0; + + /* We are looking for line number notes. Search backward + before basic block to find correct ones. */ + insn = prev_nonnote_insn (insn); + if (!insn) + insn = get_insns (); + else + insn = NEXT_INSN (insn); + + while (insn != bb->end) + { + if (GET_CODE (insn) == NOTE) + { + /* Must ignore the line number notes that immediately + follow the end of an inline function to avoid counting + it twice. There is a note before the call, and one + after the call. */ + if (NOTE_LINE_NUMBER (insn) + == NOTE_INSN_REPEATED_LINE_NUMBER) + ignore_next_note = 1; + else if (NOTE_LINE_NUMBER (insn) <= 0) + /*NOP*/; + else if (ignore_next_note) + ignore_next_note = 0; + else + { + if (offset) + /*NOP*/; + else if (gcov_write_unsigned (bbg_file, GCOV_TAG_LINES) + || !(offset = gcov_reserve_length (bbg_file)) + || gcov_write_unsigned (bbg_file, + BB_TO_GCOV_INDEX (bb))) + goto bbg_error; + /* If this is a new source file, then output + the file's name to the .bb file. */ + if (!prev_file_name + || strcmp (NOTE_SOURCE_FILE (insn), + prev_file_name)) + { + prev_file_name = NOTE_SOURCE_FILE (insn); + if (gcov_write_unsigned (bbg_file, 0) + || gcov_write_string (bbg_file, prev_file_name, + strlen (prev_file_name))) + goto bbg_error; + } + if (gcov_write_unsigned (bbg_file, NOTE_LINE_NUMBER (insn))) + goto bbg_error; + } + } + insn = NEXT_INSN (insn); + } + if (offset) + { + if (gcov_write_unsigned (bbg_file, 0) + || gcov_write_string (bbg_file, NULL, 0) + || gcov_write_length (bbg_file, offset)) + { + bbg_error:; + warning ("error writing `%s'", bbg_file_name); + fclose (bbg_file); + bbg_file = NULL; + } + } + } + } } if (flag_branch_probabilities) @@ -1052,10 +1209,23 @@ branch_prob () /* For each edge not on the spanning tree, add counting code as rtl. */ - if (profile_arc_flag) + if (cfun->arc_profile && profile_arc_flag) { + struct function_list *item; + instrument_edges (el); allocate_reg_info (max_reg_num (), FALSE, FALSE); + + /* ??? Probably should re-use the existing struct function. */ + item = xmalloc (sizeof (struct function_list)); + + *functions_tail = item; + functions_tail = &item->next; + + item->next = 0; + item->name = xstrdup (name); + item->cfg_checksum = profile_info.current_function_cfg_checksum; + item->count_edges = profile_info.count_edges_instrumented_now; } remove_fake_edges (); @@ -1194,30 +1364,25 @@ init_branch_prob (filename) if (flag_test_coverage) { - char *data_file, *bbg_file_name; - - /* Open an output file for the basic block/line number map. */ - data_file = (char *) alloca (len + 4); - strcpy (data_file, filename); - strcat (data_file, ".bb"); - if ((bb_file = fopen (data_file, "wb")) == 0) - fatal_io_error ("can't open %s", data_file); - - /* Open an output file for the program flow graph. */ - bbg_file_name = (char *) alloca (len + 5); + /* Open the bbg output file. */ + bbg_file_name = (char *) xmalloc (len + strlen (GCOV_GRAPH_SUFFIX) + 1); strcpy (bbg_file_name, filename); - strcat (bbg_file_name, ".bbg"); - if ((bbg_file = fopen (bbg_file_name, "wb")) == 0) - fatal_io_error ("can't open %s", bbg_file_name); + strcat (bbg_file_name, GCOV_GRAPH_SUFFIX); + bbg_file = fopen (bbg_file_name, "wb"); + if (!bbg_file) + fatal_io_error ("cannot open %s", bbg_file_name); - /* Initialize to zero, to ensure that the first file name will be - written to the .bb file. */ - last_bb_file_name = 0; + if (gcov_write_unsigned (bbg_file, GCOV_GRAPH_MAGIC) + || gcov_write_unsigned (bbg_file, GCOV_VERSION)) + { + fclose (bbg_file); + fatal_io_error ("cannot write `%s'", bbg_file_name); + } } - da_file_name = (char *) xmalloc (len + 4); + da_file_name = (char *) xmalloc (len + strlen (GCOV_DATA_SUFFIX) + 1); strcpy (da_file_name, filename); - strcat (da_file_name, ".da"); + strcat (da_file_name, GCOV_DATA_SUFFIX); if (flag_branch_probabilities) { @@ -1225,11 +1390,21 @@ init_branch_prob (filename) if (!da_file) warning ("file %s not found, execution counts assumed to be zero", da_file_name); + if (counts_file_index && strcmp (da_file_name, counts_file_name)) + cleanup_counts_index (0); + if (index_counts_file ()) + counts_file_name = xstrdup (da_file_name); } if (profile_arc_flag) - init_edge_profiler (); - + { + /* Generate and save a copy of this so it can be shared. */ + char buf[20]; + + ASM_GENERATE_INTERNAL_LABEL (buf, "LPBX", 2); + profiler_label = gen_rtx_SYMBOL_REF (Pmode, ggc_strdup (buf)); + } + total_num_blocks = 0; total_num_edges = 0; total_num_edges_ignored = 0; @@ -1251,12 +1426,32 @@ end_branch_prob () { if (flag_test_coverage) { - fclose (bb_file); - fclose (bbg_file); - unlink (da_file_name); + if (bbg_file) + { +#if __GNUC__ && !CROSS_COMPILE && SUPPORTS_WEAK + /* If __gcov_init has a value in the compiler, it means we + are instrumenting ourselves. We should not remove the + counts file, because we might be recompiling + ourselves. The .da files are all removed during copying + the stage1 files. */ + extern void __gcov_init (void *) + __attribute__ ((weak)); + + if (!__gcov_init) + unlink (da_file_name); +#else + unlink (da_file_name); +#endif + fclose (bbg_file); + } + else + { + unlink (bbg_file_name); + unlink (da_file_name); + } } - if (flag_branch_probabilities && da_file) + if (da_file) fclose (da_file); if (rtl_dump_file) @@ -1292,124 +1487,242 @@ end_branch_prob () } } } - -/* The label used by the edge profiling code. */ - -static GTY(()) rtx profiler_label; - -/* Initialize the profiler_label. */ - -static void -init_edge_profiler () -{ - /* Generate and save a copy of this so it can be shared. */ - char buf[20]; - ASM_GENERATE_INTERNAL_LABEL (buf, "LPBX", 2); - profiler_label = gen_rtx_SYMBOL_REF (Pmode, ggc_strdup (buf)); -} - -/* Output instructions as RTL to increment the edge execution count. */ - -static rtx -gen_edge_profiler (edgeno) - int edgeno; -{ - enum machine_mode mode = mode_for_size (GCOV_TYPE_SIZE, MODE_INT, 0); - rtx mem_ref, tmp; - rtx sequence; - - start_sequence (); - - tmp = force_reg (Pmode, profiler_label); - tmp = plus_constant (tmp, GCOV_TYPE_SIZE / BITS_PER_UNIT * edgeno); - mem_ref = validize_mem (gen_rtx_MEM (mode, tmp)); - - set_mem_alias_set (mem_ref, new_alias_set ()); - - tmp = expand_simple_binop (mode, PLUS, mem_ref, const1_rtx, - mem_ref, 0, OPTAB_WIDEN); - - if (tmp != mem_ref) - emit_move_insn (copy_rtx (mem_ref), tmp); - - sequence = get_insns (); - end_sequence (); - return sequence; -} -/* Output code for a constructor that will invoke __bb_init_func, if - this has not already been done. */ +/* Write out the structure which libgcc uses to locate all the arc + counters. The structures used here must match those defined in + gcov-io.h. Write out the constructor to call __gcov_init. */ void -output_func_start_profiler () +create_profiler () { - tree fnname, fndecl; - char *name; - char buf[20]; - const char *cfnname; - rtx table_address; - enum machine_mode mode = mode_for_size (GCOV_TYPE_SIZE, MODE_INT, 0); + tree fields, field, value = NULL_TREE; + tree ginfo_type; + tree string_type; + tree gcov_type, gcov_ptr_type; + char name[20]; + char *ctor_name; + tree structure, ctor; + rtx structure_address; int save_flag_inline_functions = flag_inline_functions; - /* It's either already been output, or we don't need it because we're - not doing profile-edges. */ - if (! need_func_profiler) + if (!profile_info.count_instrumented_edges) return; + + string_type = build_pointer_type + (build_qualified_type (char_type_node, TYPE_QUAL_CONST)); + gcov_type = make_signed_type (GCOV_TYPE_SIZE); + gcov_ptr_type + = build_pointer_type (build_qualified_type + (gcov_type, TYPE_QUAL_CONST)); + + ginfo_type = (*lang_hooks.types.make_type) (RECORD_TYPE); + - need_func_profiler = 0; - - /* Synthesize a constructor function to invoke __bb_init_func with a - pointer to this object file's profile block. */ - - /* Try and make a unique name given the "file function name". - - And no, I don't like this either. */ - - fnname = get_file_function_name ('I'); - cfnname = IDENTIFIER_POINTER (fnname); - name = concat (cfnname, "GCOV", NULL); - fnname = get_identifier (name); - free (name); + /* Version ident */ + fields = build_decl (FIELD_DECL, NULL_TREE, long_unsigned_type_node); + value = tree_cons (fields, convert (long_unsigned_type_node, build_int_2 + (GCOV_VERSION, 0)), value); + + /* NULL */ + field = build_decl (FIELD_DECL, NULL_TREE, build_pointer_type + (build_qualified_type + (ginfo_type, TYPE_QUAL_CONST))); + TREE_CHAIN (field) = fields; + fields = field; + value = tree_cons (fields, null_pointer_node, value); + + /* Filename */ + { + tree filename_string; + char *filename; + int filename_len; + + filename = getpwd (); + filename = (filename && da_file_name[0] != '/' + ? concat (filename, "/", da_file_name, NULL) + : da_file_name); + filename_len = strlen (filename); + filename_string = build_string (filename_len + 1, filename); + if (filename != da_file_name) + free (filename); + TREE_TYPE (filename_string) = build_array_type + (char_type_node, build_index_type + (build_int_2 (filename_len, 0))); + + field = build_decl (FIELD_DECL, NULL_TREE, string_type); + TREE_CHAIN (field) = fields; + fields = field; + value = tree_cons (fields, build1 (ADDR_EXPR, string_type, + filename_string), value); + } + + /* Workspace */ + field = build_decl (FIELD_DECL, NULL_TREE, long_integer_type_node); + TREE_CHAIN (field) = fields; + fields = field; + value = tree_cons (fields, + convert (long_integer_type_node, integer_zero_node), + value); + + /* function_info table */ + { + struct function_list *item; + int num_nodes = 0; + tree array_value = NULL_TREE; + tree finfo_type, finfo_ptr_type; + tree name, checksum, arcs; + + finfo_type = (*lang_hooks.types.make_type) (RECORD_TYPE); + name = build_decl (FIELD_DECL, NULL_TREE, string_type); + checksum = build_decl (FIELD_DECL, NULL_TREE, unsigned_type_node); + TREE_CHAIN (checksum) = name; + arcs = build_decl (FIELD_DECL, NULL_TREE, unsigned_type_node); + TREE_CHAIN (arcs) = checksum; + finish_builtin_struct (finfo_type, "__function_info", + arcs, NULL_TREE); + finfo_ptr_type = build_pointer_type + (build_qualified_type (finfo_type, TYPE_QUAL_CONST)); + + for (item = functions_head; item != 0; item = item->next, num_nodes++) + { + size_t name_len = strlen (item->name); + tree finfo_value = NULL_TREE; + tree fname = build_string (name_len + 1, item->name); + + TREE_TYPE (fname) = build_array_type + (char_type_node, build_index_type (build_int_2 (name_len, 0))); + finfo_value = tree_cons (name, build1 + (ADDR_EXPR, string_type, + fname), finfo_value); + finfo_value = tree_cons (checksum, convert + (unsigned_type_node, + build_int_2 (item->cfg_checksum, 0)), + finfo_value); + finfo_value = tree_cons (arcs, convert + (unsigned_type_node, + build_int_2 (item->count_edges, 0)), + finfo_value); + array_value = tree_cons (NULL_TREE, build + (CONSTRUCTOR, finfo_type, NULL_TREE, + nreverse (finfo_value)), array_value); + } + + /* Create constructor for array. */ + if (num_nodes) + { + tree array_type; + + array_type = build_array_type (finfo_type, build_index_type + (build_int_2 (num_nodes - 1, 0))); + array_value = build (CONSTRUCTOR, array_type, + NULL_TREE, nreverse (array_value)); + array_value = build1 + (ADDR_EXPR, finfo_ptr_type, array_value); + } + else + array_value = null_pointer_node; + + field = build_decl (FIELD_DECL, NULL_TREE, finfo_ptr_type); + TREE_CHAIN (field) = fields; + fields = field; + value = tree_cons (fields, array_value, value); + + /* number of functions */ + field = build_decl (FIELD_DECL, NULL_TREE, unsigned_type_node); + TREE_CHAIN (field) = fields; + fields = field; + value = tree_cons (fields, convert (unsigned_type_node, build_int_2 + (num_nodes, 0)), value); + } + + /* arc count table */ + { + tree counts_table = null_pointer_node; + + if (profile_info.count_instrumented_edges) + { + tree gcov_type_array_type + = build_array_type (gcov_type, build_index_type + (build_int_2 (profile_info. + count_instrumented_edges - 1, 0))); + /* No values. */ + counts_table + = build (VAR_DECL, gcov_type_array_type, NULL_TREE, NULL_TREE); + TREE_STATIC (counts_table) = 1; + DECL_NAME (counts_table) = get_identifier (XSTR (profiler_label, 0)); + assemble_variable (counts_table, 0, 0, 0); + counts_table = build1 (ADDR_EXPR, gcov_ptr_type, counts_table); + } + + field = build_decl (FIELD_DECL, NULL_TREE, gcov_ptr_type); + TREE_CHAIN (field) = fields; + fields = field; + value = tree_cons (fields, counts_table, value); + } + + /* number of arc counts */ + field = build_decl (FIELD_DECL, NULL_TREE, unsigned_type_node); + TREE_CHAIN (field) = fields; + fields = field; + value = tree_cons (fields, convert + (unsigned_type_node, + build_int_2 (profile_info + .count_instrumented_edges, 0)), + value); + + finish_builtin_struct (ginfo_type, "__gcov_info", fields, NULL_TREE); + structure = build (VAR_DECL, ginfo_type, NULL_TREE, NULL_TREE); + DECL_INITIAL (structure) + = build (CONSTRUCTOR, ginfo_type, NULL_TREE, nreverse (value)); + TREE_STATIC (structure) = 1; + ASM_GENERATE_INTERNAL_LABEL (name, "LPBX", 0); + DECL_NAME (structure) = get_identifier (name); + + /* Build structure. */ + assemble_variable (structure, 0, 0, 0); - fndecl = build_decl (FUNCTION_DECL, fnname, - build_function_type (void_type_node, NULL_TREE)); - DECL_EXTERNAL (fndecl) = 0; + /* Build the constructor function to invoke __gcov_init. */ + ctor_name = concat (IDENTIFIER_POINTER (get_file_function_name ('I')), + "_GCOV", NULL); + ctor = build_decl (FUNCTION_DECL, get_identifier (ctor_name), + build_function_type (void_type_node, NULL_TREE)); + free (ctor_name); + DECL_EXTERNAL (ctor) = 0; /* It can be a static function as long as collect2 does not have to scan the object file to find its ctor/dtor routine. */ - TREE_PUBLIC (fndecl) = ! targetm.have_ctors_dtors; - - TREE_USED (fndecl) = 1; - - DECL_RESULT (fndecl) = build_decl (RESULT_DECL, NULL_TREE, void_type_node); - - fndecl = (*lang_hooks.decls.pushdecl) (fndecl); - rest_of_decl_compilation (fndecl, 0, 1, 0); - announce_function (fndecl); - current_function_decl = fndecl; - DECL_INITIAL (fndecl) = error_mark_node; - make_decl_rtl (fndecl, NULL); - init_function_start (fndecl, input_filename, lineno); + TREE_PUBLIC (ctor) = ! targetm.have_ctors_dtors; + TREE_USED (ctor) = 1; + DECL_RESULT (ctor) = build_decl (RESULT_DECL, NULL_TREE, void_type_node); + + ctor = (*lang_hooks.decls.pushdecl) (ctor); + rest_of_decl_compilation (ctor, 0, 1, 0); + announce_function (ctor); + current_function_decl = ctor; + DECL_INITIAL (ctor) = error_mark_node; + make_decl_rtl (ctor, NULL); + init_function_start (ctor, input_filename, lineno); (*lang_hooks.decls.pushlevel) (0); - expand_function_start (fndecl, 0); + expand_function_start (ctor, 0); cfun->arc_profile = 0; - /* Actually generate the code to call __bb_init_func. */ - ASM_GENERATE_INTERNAL_LABEL (buf, "LPBX", 0); - table_address = force_reg (Pmode, - gen_rtx_SYMBOL_REF (Pmode, ggc_strdup (buf))); - emit_library_call (gen_rtx_SYMBOL_REF (Pmode, "__bb_init_func"), LCT_NORMAL, - mode, 1, table_address, Pmode); + /* Actually generate the code to call __gcov_init. */ + structure_address = force_reg (Pmode, gen_rtx_SYMBOL_REF + (Pmode, IDENTIFIER_POINTER + (DECL_NAME (structure)))); + emit_library_call (gen_rtx_SYMBOL_REF (Pmode, "__gcov_init"), + LCT_NORMAL, VOIDmode, 1, + structure_address, Pmode); expand_function_end (input_filename, lineno, 0); (*lang_hooks.decls.poplevel) (1, 0, 1); - /* Since fndecl isn't in the list of globals, it would never be emitted + /* Since ctor isn't in the list of globals, it would never be emitted when it's considered to be 'safe' for inlining, so turn off flag_inline_functions. */ flag_inline_functions = 0; - rest_of_compilation (fndecl); + rest_of_compilation (ctor); /* Reset flag_inline_functions to its original value. */ flag_inline_functions = save_flag_inline_functions; @@ -1419,8 +1732,37 @@ output_func_start_profiler () current_function_decl = NULL_TREE; if (targetm.have_ctors_dtors) - (* targetm.asm_out.constructor) (XEXP (DECL_RTL (fndecl), 0), + (* targetm.asm_out.constructor) (XEXP (DECL_RTL (ctor), 0), DEFAULT_INIT_PRIORITY); } + +/* Output instructions as RTL to increment the edge execution count. */ + +static rtx +gen_edge_profiler (edgeno) + int edgeno; +{ + enum machine_mode mode = mode_for_size (GCOV_TYPE_SIZE, MODE_INT, 0); + rtx mem_ref, tmp; + rtx sequence; + + start_sequence (); + + tmp = force_reg (Pmode, profiler_label); + tmp = plus_constant (tmp, GCOV_TYPE_SIZE / BITS_PER_UNIT * edgeno); + mem_ref = validize_mem (gen_rtx_MEM (mode, tmp)); + + set_mem_alias_set (mem_ref, new_alias_set ()); + + tmp = expand_simple_binop (mode, PLUS, mem_ref, const1_rtx, + mem_ref, 0, OPTAB_WIDEN); + + if (tmp != mem_ref) + emit_move_insn (copy_rtx (mem_ref), tmp); + + sequence = get_insns (); + end_sequence (); + return sequence; +} #include "gt-profile.h" |