1 /* Callgraph based analysis of static variables.
2 Copyright (C) 2015-2019 Free Software Foundation, Inc.
3 Contributed by Martin Liska <mliska@suse.cz>
5 This file is part of GCC.
7 GCC is free software; you can redistribute it and/or modify it under
8 the terms of the GNU General Public License as published by the Free
9 Software Foundation; either version 3, or (at your option) any later
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
17 You should have received a copy of the GNU General Public License
18 along with GCC; see the file COPYING3. If not see
19 <http://www.gnu.org/licenses/>. */
21 /* Interprocedural HSA pass is responsible for creation of HSA clones.
22 For all these HSA clones, we emit HSAIL instructions and pass processing
27 #include "coretypes.h"
33 #include "tree-pass.h"
35 #include "basic-block.h"
38 #include "gimple-pretty-print.h"
39 #include "tree-streamer.h"
40 #include "stringpool.h"
42 #include "print-tree.h"
43 #include "symbol-summary.h"
44 #include "hsa-common.h"
48 /* If NODE is not versionable, warn about not emiting HSAIL and return false.
49 Otherwise return true. */
52 check_warn_node_versionable (cgraph_node
*node
)
54 if (!node
->local
.versionable
)
56 warning_at (EXPR_LOCATION (node
->decl
), OPT_Whsa
,
57 "could not emit HSAIL for function %s: function cannot be "
58 "cloned", node
->name ());
64 /* The function creates HSA clones for all functions that were either
65 marked as HSA kernels or are callable HSA functions. Apart from that,
66 we redirect all edges that come from an HSA clone and end in another
67 HSA clone to connect these two functions. */
70 process_hsa_functions (void)
72 struct cgraph_node
*node
;
74 if (hsa_summaries
== NULL
)
75 hsa_summaries
= new hsa_summary_t (symtab
);
77 FOR_EACH_DEFINED_FUNCTION (node
)
79 hsa_function_summary
*s
= hsa_summaries
->get (node
);
81 /* A linked function is skipped. */
82 if (s
!= NULL
&& s
->m_bound_function
!= NULL
)
87 if (!check_warn_node_versionable (node
))
90 = node
->create_virtual_clone (vec
<cgraph_edge
*> (),
91 NULL
, NULL
, "hsa", 0);
92 TREE_PUBLIC (clone
->decl
) = TREE_PUBLIC (node
->decl
);
93 clone
->externally_visible
= node
->externally_visible
;
95 clone
->force_output
= true;
96 hsa_summaries
->link_functions (clone
, node
, s
->m_kind
, false);
99 fprintf (dump_file
, "Created a new HSA clone: %s, type: %s\n",
101 s
->m_kind
== HSA_KERNEL
? "kernel" : "function");
103 else if (hsa_callable_function_p (node
->decl
)
104 /* At this point, this is enough to identify clones for
105 parallel, which for HSA would need to be kernels anyway. */
106 && !DECL_ARTIFICIAL (node
->decl
))
108 if (!check_warn_node_versionable (node
))
111 = node
->create_virtual_clone (vec
<cgraph_edge
*> (),
112 NULL
, NULL
, "hsa", 0);
113 TREE_PUBLIC (clone
->decl
) = TREE_PUBLIC (node
->decl
);
114 clone
->externally_visible
= node
->externally_visible
;
116 if (!node
->local
.local
)
117 clone
->force_output
= true;
118 hsa_summaries
->link_functions (clone
, node
, HSA_FUNCTION
, false);
121 fprintf (dump_file
, "Created a new HSA function clone: %s\n",
126 /* Redirect all edges that are between HSA clones. */
127 FOR_EACH_DEFINED_FUNCTION (node
)
129 cgraph_edge
*e
= node
->callees
;
133 hsa_function_summary
*src
= hsa_summaries
->get (node
);
134 if (src
!= NULL
&& src
->m_gpu_implementation_p
)
136 hsa_function_summary
*dst
= hsa_summaries
->get (e
->callee
);
137 if (dst
!= NULL
&& !dst
->m_gpu_implementation_p
)
139 e
->redirect_callee (dst
->m_bound_function
);
142 "Redirecting edge to HSA function: %s->%s\n",
143 xstrdup_for_dump (e
->caller
->name ()),
144 xstrdup_for_dump (e
->callee
->name ()));
155 /* Iterate all HSA functions and stream out HSA function summary. */
158 ipa_hsa_write_summary (void)
161 struct cgraph_node
*node
;
162 struct output_block
*ob
;
163 unsigned int count
= 0;
164 lto_symtab_encoder_iterator lsei
;
165 lto_symtab_encoder_t encoder
;
170 ob
= create_output_block (LTO_section_ipa_hsa
);
171 encoder
= ob
->decl_state
->symtab_node_encoder
;
173 for (lsei
= lsei_start_function_in_partition (encoder
); !lsei_end_p (lsei
);
174 lsei_next_function_in_partition (&lsei
))
176 node
= lsei_cgraph_node (lsei
);
177 hsa_function_summary
*s
= hsa_summaries
->get (node
);
183 streamer_write_uhwi (ob
, count
);
185 /* Process all of the functions. */
186 for (lsei
= lsei_start_function_in_partition (encoder
); !lsei_end_p (lsei
);
187 lsei_next_function_in_partition (&lsei
))
189 node
= lsei_cgraph_node (lsei
);
190 hsa_function_summary
*s
= hsa_summaries
->get (node
);
194 encoder
= ob
->decl_state
->symtab_node_encoder
;
195 int node_ref
= lto_symtab_encoder_encode (encoder
, node
);
196 streamer_write_uhwi (ob
, node_ref
);
198 bp
= bitpack_create (ob
->main_stream
);
199 bp_pack_value (&bp
, s
->m_kind
, 2);
200 bp_pack_value (&bp
, s
->m_gpu_implementation_p
, 1);
201 bp_pack_value (&bp
, s
->m_bound_function
!= NULL
, 1);
202 streamer_write_bitpack (&bp
);
203 if (s
->m_bound_function
)
204 stream_write_tree (ob
, s
->m_bound_function
->decl
, true);
208 streamer_write_char_stream (ob
->main_stream
, 0);
209 produce_asm (ob
, NULL
);
210 destroy_output_block (ob
);
213 /* Read section in file FILE_DATA of length LEN with data DATA. */
216 ipa_hsa_read_section (struct lto_file_decl_data
*file_data
, const char *data
,
219 const struct lto_function_header
*header
220 = (const struct lto_function_header
*) data
;
221 const int cfg_offset
= sizeof (struct lto_function_header
);
222 const int main_offset
= cfg_offset
+ header
->cfg_size
;
223 const int string_offset
= main_offset
+ header
->main_size
;
224 struct data_in
*data_in
;
228 lto_input_block
ib_main ((const char *) data
+ main_offset
,
229 header
->main_size
, file_data
->mode_table
);
232 = lto_data_in_create (file_data
, (const char *) data
+ string_offset
,
233 header
->string_size
, vNULL
);
234 count
= streamer_read_uhwi (&ib_main
);
236 for (i
= 0; i
< count
; i
++)
239 struct cgraph_node
*node
;
240 lto_symtab_encoder_t encoder
;
242 index
= streamer_read_uhwi (&ib_main
);
243 encoder
= file_data
->symtab_node_encoder
;
244 node
= dyn_cast
<cgraph_node
*> (lto_symtab_encoder_deref (encoder
,
246 gcc_assert (node
->definition
);
247 hsa_function_summary
*s
= hsa_summaries
->get_create (node
);
249 struct bitpack_d bp
= streamer_read_bitpack (&ib_main
);
250 s
->m_kind
= (hsa_function_kind
) bp_unpack_value (&bp
, 2);
251 s
->m_gpu_implementation_p
= bp_unpack_value (&bp
, 1);
252 bool has_tree
= bp_unpack_value (&bp
, 1);
256 tree decl
= stream_read_tree (&ib_main
, data_in
);
257 s
->m_bound_function
= cgraph_node::get_create (decl
);
260 lto_free_section_data (file_data
, LTO_section_ipa_hsa
, NULL
, data
,
262 lto_data_in_delete (data_in
);
265 /* Load streamed HSA functions summary and assign the summary to a function. */
268 ipa_hsa_read_summary (void)
270 struct lto_file_decl_data
**file_data_vec
= lto_get_file_decl_data ();
271 struct lto_file_decl_data
*file_data
;
274 if (hsa_summaries
== NULL
)
275 hsa_summaries
= new hsa_summary_t (symtab
);
277 while ((file_data
= file_data_vec
[j
++]))
280 const char *data
= lto_get_section_data (file_data
, LTO_section_ipa_hsa
,
284 ipa_hsa_read_section (file_data
, data
, len
);
288 const pass_data pass_data_ipa_hsa
=
292 OPTGROUP_OMP
, /* optinfo_flags */
293 TV_IPA_HSA
, /* tv_id */
294 0, /* properties_required */
295 0, /* properties_provided */
296 0, /* properties_destroyed */
297 0, /* todo_flags_start */
298 TODO_dump_symtab
, /* todo_flags_finish */
301 class pass_ipa_hsa
: public ipa_opt_pass_d
304 pass_ipa_hsa (gcc::context
*ctxt
)
305 : ipa_opt_pass_d (pass_data_ipa_hsa
, ctxt
,
306 NULL
, /* generate_summary */
307 ipa_hsa_write_summary
, /* write_summary */
308 ipa_hsa_read_summary
, /* read_summary */
309 ipa_hsa_write_summary
, /* write_optimization_summary */
310 ipa_hsa_read_summary
, /* read_optimization_summary */
311 NULL
, /* stmt_fixup */
312 0, /* function_transform_todo_flags_start */
313 NULL
, /* function_transform */
314 NULL
) /* variable_transform */
317 /* opt_pass methods: */
318 virtual bool gate (function
*);
320 virtual unsigned int execute (function
*) { return process_hsa_functions (); }
322 }; // class pass_ipa_reference
325 pass_ipa_hsa::gate (function
*)
327 return hsa_gen_requested_p ();
333 make_pass_ipa_hsa (gcc::context
*ctxt
)
335 return new pass_ipa_hsa (ctxt
);