1 /* Functions for LTO dump tool.
2 Copyright (C) 2018-2024 Free Software Foundation, Inc.
4 This file is part of GCC.
6 GCC is free software; you can redistribute it and/or modify it under
7 the terms of the GNU General Public License as published by the Free
8 Software Foundation; either version 3, or (at your option) any later
11 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
12 WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
16 You should have received a copy of the GNU General Public License
17 along with GCC; see the file COPYING3. If not see
18 <http://www.gnu.org/licenses/>. */
22 #include "coretypes.h"
25 #include "basic-block.h"
30 #include "tree-pass.h"
31 #include "tree-streamer.h"
35 #include "lto-partition.h"
36 #include "tree-pretty-print.h"
37 #include "lto-common.h"
39 /* Stores details of symbols for dumping symbol list. */
45 symbol_entry (symtab_node
*node_
): node (node_
)
48 virtual ~symbol_entry ()
51 char* get_name () const
53 if (flag_lto_dump_demangle
)
54 return xstrdup (node
->name ());
56 return xstrdup (node
->asm_name ());
59 virtual size_t get_size () const = 0;
63 const char *name
= get_name ();
64 const char *type_name
= node
->get_symtab_type_string ();
65 const char *visibility
= node
->get_visibility_string ();
66 size_t sz
= get_size ();
67 printf ("%s %s %4" PRIu64
" %s ", type_name
, visibility
, (uint64_t) sz
,
72 /* Stores variable specific details of symbols for dumping symbol list. */
74 class variable_entry
: public symbol_entry
77 variable_entry (varpool_node
*node_
): symbol_entry (node_
)
80 virtual ~variable_entry ()
83 size_t get_size () const final override
85 varpool_node
*vnode
= dyn_cast
<varpool_node
*> (node
);
86 if (DECL_SIZE (vnode
->decl
) && tree_fits_shwi_p (DECL_SIZE (vnode
->decl
)))
87 return tree_to_shwi (DECL_SIZE (vnode
->decl
));
91 void dump () final override
93 symbol_entry :: dump ();
94 varpool_node
*vnode
= dyn_cast
<varpool_node
*> (node
);
95 vnode
->get_constructor ();
96 tree value_tree
= DECL_INITIAL (vnode
->decl
);
97 if (flag_lto_print_value
&& value_tree
)
98 print_generic_expr (stdout
, value_tree
, TDF_NONE
);
103 /* Stores function specific details of symbols for dumping symbol list. */
105 class function_entry
: public symbol_entry
108 function_entry (cgraph_node
*node_
): symbol_entry (node_
)
111 virtual ~function_entry ()
114 void dump () final override
116 symbol_entry :: dump ();
120 size_t get_size () const final override
122 cgraph_node
*cnode
= dyn_cast
<cgraph_node
*> (node
);
125 return (cnode
->definition
&& !cnode
->thunk
&& !cnode
->alias
)
126 ? n_basic_blocks_for_fn (DECL_STRUCT_FUNCTION (cnode
->decl
))
131 /* Comparing symbols based on size. */
133 int size_compare (const void *a
, const void *b
)
135 const symbol_entry
*e1
= *(const symbol_entry
* const*) a
;
136 const symbol_entry
*e2
= *(const symbol_entry
* const*) b
;
138 return e1
->get_size () - e2
->get_size ();
141 /* Comparing symbols based on name. */
143 int name_compare (const void *a
, const void *b
)
145 const symbol_entry
*e1
= *(const symbol_entry
* const*) a
;
146 const symbol_entry
*e2
= *(const symbol_entry
* const*) b
;
148 return strcmp (e1
->get_name (), e2
->get_name ());
151 /* Dump list of functions and their details. */
153 void dump_list_functions (void)
155 auto_vec
<symbol_entry
*> v
;
158 FOR_EACH_FUNCTION (cnode
)
160 if (cnode
->definition
&& !cnode
->alias
)
161 cnode
->get_untransformed_body ();
162 symbol_entry
*e
= new function_entry (cnode
);
163 if (!flag_lto_dump_defined
|| (cnode
->definition
&& !cnode
->alias
))
167 if (flag_lto_size_sort
)
168 v
.qsort (size_compare
);
169 else if (flag_lto_name_sort
)
170 v
.qsort (name_compare
);
171 if (flag_lto_reverse_sort
)
174 printf ("Type Visibility Size Name");
175 if (flag_lto_print_value
)
180 FOR_EACH_VEC_ELT (v
, i
, e
)
187 /* Dump list of variables and their details. */
189 void dump_list_variables (void)
191 auto_vec
<symbol_entry
*> v
;
194 FOR_EACH_VARIABLE (vnode
)
196 symbol_entry
*e
= new variable_entry (vnode
);
197 if (!flag_lto_dump_defined
|| vnode
->definition
)
201 if (flag_lto_size_sort
)
202 v
.qsort (size_compare
);
203 else if (flag_lto_name_sort
)
204 v
.qsort (name_compare
);
205 if (flag_lto_reverse_sort
)
211 FOR_EACH_VEC_ELT (v
, i
, e
)
218 /* Dump symbol table in graphviz format. */
219 void dump_symtab_graphviz (void)
221 symtab
->dump_graphviz (stdout
);
224 /* Dump symbol list. */
226 void dump_list (void)
228 dump_list_functions ();
229 dump_list_variables ();
232 /* Dump specific variables and functions used in IL. */
236 printf ("Symbol: %s\n", flag_lto_dump_symbol
);
237 FOR_EACH_SYMBOL (node
)
239 if (!strcmp (flag_lto_dump_symbol
, node
->name ()))
247 /* Dump specific gimple body of specified function. */
251 dump_flags_t flags
= TDF_NONE
;
253 flags
= parse_dump_option (flag_dump_level
, NULL
);
254 if (flags
== TDF_ERROR
)
256 error_at (input_location
, "Level not found, use none, slim, blocks, vops.");
260 FOR_EACH_DEFINED_FUNCTION (cnode
)
262 && !strcmp (cnode
->asm_name (), flag_dump_body
))
264 printf ("GIMPLE body of function: %s\n\n", cnode
->asm_name ());
265 cnode
->get_untransformed_body ();
266 debug_function (cnode
->decl
, flags
);
270 error_at (input_location
, "Function not found.");
273 /* List of command line options for dumping. */
274 void dump_tool_help ()
277 "Usage: lto-dump [OPTION]... SUB_COMMAND [OPTION]...\n\n"
278 "LTO dump tool command line options.\n\n"
279 " -list [options] Dump the symbol list.\n"
280 " -demangle Dump the demangled output.\n"
281 " -defined-only Dump only the defined symbols.\n"
282 " -print-value Dump initial values of the variables.\n"
283 " -name-sort Sort the symbols alphabetically.\n"
284 " -size-sort Sort the symbols according to size.\n"
285 " -reverse-sort Dump the symbols in reverse order.\n"
286 " -symbol= Dump the details of specific symbol.\n"
287 " -objects Dump the details of LTO objects.\n"
288 " -callgraph Dump the callgraph in graphviz format.\n"
289 " -type-stats Dump statistics of tree types.\n"
290 " -tree-stats Dump statistics of trees.\n"
291 " -gimple-stats Dump statistics of GIMPLE statements.\n"
292 " -dump-body= Dump the specific GIMPLE body.\n"
293 " -dump-level= Deciding the optimization level of body.\n"
294 " -help Display the dump tool help.\n";
300 lto_option_lang_mask (void)
305 /* Functions for dumping various details in LTO dump tool are called
306 in lto_main(). The purpose of this dump tool is to analyze the LTO
313 if (flag_lto_dump_tool_help
)
316 exit (SUCCESS_EXIT_CODE
);
319 /* LTO is called as a front end, even though it is not a front end.
320 Because it is called as a front end, TV_PHASE_PARSING and
321 TV_PARSE_GLOBAL are active, and we need to turn them off while
322 doing LTO. Later we turn them back on so they are active up in
325 /* Initialize the LTO front end. */
328 /* Read all the symbols and call graph from all the files in the
330 read_cgraph_and_symbols (num_in_fnames
, in_fnames
);
332 /* Dump symbol list. */
333 if (flag_lto_dump_list
)
335 else if (flag_lto_dump_symbol
)
337 /* Dump specific variables and functions used in IL. */
340 else if (flag_lto_gimple_stats
)
342 /* Dump gimple statement statistics. */
344 FOR_EACH_DEFINED_FUNCTION (node
)
346 node
->get_untransformed_body ();
347 if (!GATHER_STATISTICS
)
348 warning_at (input_location
, 0,
349 "Not configured with "
350 "%<--enable-gather-detailed-mem-stats%>.");
352 dump_gimple_statistics ();
354 else if (flag_lto_tree_stats
)
356 /* Dump tree statistics. */
357 if (!GATHER_STATISTICS
)
358 warning_at (input_location
, 0,
359 "Not configured with "
360 "%<--enable-gather-detailed-mem-stats%>.");
363 printf ("Tree statistics\n");
364 dump_tree_statistics ();
367 else if (flag_dump_body
)
369 /* Dump specific gimple body of specified function. */
372 else if (flag_dump_callgraph
)
373 dump_symtab_graphviz ();
377 /* Exit right now. */
378 exit (SUCCESS_EXIT_CODE
);