Add PR number
[official-gcc.git] / gcc / config / sh / sh_optimize_sett_clrt.cc
blob1c112703c1be6bb5079e344a475dd91a3c81bac5
1 /* An SH specific RTL pass that tries to optimize clrt and sett insns.
2 Copyright (C) 2013-2018 Free Software Foundation, Inc.
4 This file is part of GCC.
6 GCC is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3, or (at your option)
9 any later version.
11 GCC is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
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/>. */
20 #define IN_TARGET_CODE 1
22 #include "config.h"
23 #define INCLUDE_ALGORITHM
24 #define INCLUDE_VECTOR
25 #include "system.h"
26 #include "coretypes.h"
27 #include "backend.h"
28 #include "target.h"
29 #include "rtl.h"
30 #include "df.h"
31 #include "cfgrtl.h"
32 #include "tree-pass.h"
35 This pass tries to eliminate unnecessary sett or clrt instructions in cases
36 where the ccreg value is already known to be the same as the constant set
37 would set it to. This is done as follows:
39 Check every BB's insn and see if it's a sett or clrt.
40 Once a sett or clrt insn is hit, walk insns and predecessor basic blocks
41 backwards from that insn and determine all possible ccreg values from all
42 basic block paths.
43 Insns that set the ccreg value in some way (simple set, clobber etc) are
44 recorded. Conditional branches where one edge leads to the sett / clrt insn
45 are also recorded, since for each edge in the conditional branch the ccreg
46 value is known constant.
47 After collecting all possible ccreg values at the sett / clrt insn, check that
48 all the values are the same. If that value is the same as the sett / clrt
49 insn would set the ccreg to, the sett / clrt insn can be eliminated.
53 // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
54 // Helper functions
56 #define log_msg(...)\
57 do { if (dump_file != NULL) fprintf (dump_file, __VA_ARGS__); } while (0)
59 #define log_insn(i)\
60 do { if (dump_file != NULL) print_rtl_single (dump_file, \
61 (const_rtx)i); } while (0)
63 #define log_rtx(r)\
64 do { if (dump_file != NULL) print_rtl (dump_file, (const_rtx)r); } while (0)
66 #define log_return(retval, ...)\
67 do { if (dump_file != NULL) fprintf (dump_file, __VA_ARGS__); \
68 return retval; } while (0)
70 #define log_return_void(...)\
71 do { if (dump_file != NULL) fprintf (dump_file, __VA_ARGS__); \
72 return; } while (0)
74 // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
75 // RTL pass class
77 class sh_optimize_sett_clrt : public rtl_opt_pass
79 public:
80 sh_optimize_sett_clrt (gcc::context* ctx, const char* name);
81 virtual ~sh_optimize_sett_clrt (void);
82 virtual bool gate (function*);
83 virtual unsigned int execute (function* fun);
85 private:
86 static const pass_data default_pass_data;
88 struct ccreg_value
90 // The insn at which the ccreg value was determined.
91 // Might be NULL if e.g. an unknown value is recorded for an
92 // empty basic block.
93 rtx_insn *insn;
95 // The basic block where the insn was discovered.
96 basic_block bb;
98 // The value of ccreg. If NULL_RTX, the exact value is not known, but
99 // the ccreg is changed in some way (e.g. clobbered).
100 rtx value;
103 // Update the mode of the captured m_ccreg with the specified mode.
104 void update_ccreg_mode (machine_mode m);
106 // Given an insn pattern, check if it sets the ccreg to a constant value
107 // of either zero or STORE_FLAG_VALUE. If so, return the value rtx,
108 // NULL_RTX otherwise.
109 rtx const_setcc_value (rtx pat) const;
111 // Given a start insn and its basic block, recursively determine all
112 // possible ccreg values in all basic block paths that can lead to the
113 // start insn.
114 bool find_last_ccreg_values (rtx_insn *start_insn, basic_block bb,
115 std::vector<ccreg_value>& values_out,
116 std::vector<basic_block>& prev_visited_bb) const;
118 // Given a cbranch insn, its basic block and another basic block, determine
119 // the value to which the ccreg will be set after jumping/falling through to
120 // the specified target basic block.
121 bool sh_cbranch_ccreg_value (rtx_insn *cbranch_insn,
122 basic_block cbranch_insn_bb,
123 basic_block branch_target_bb) const;
125 // Check whether all of the ccreg values are the same.
126 static bool all_ccreg_values_equal (const std::vector<ccreg_value>& values);
128 // Remove REG_DEAD and REG_UNUSED notes from insns of the specified
129 // ccreg_value entries.
130 void remove_ccreg_dead_unused_notes (std::vector<ccreg_value>& values) const;
132 // rtx of the ccreg that is obtained from the target.
133 rtx m_ccreg;
136 const pass_data sh_optimize_sett_clrt::default_pass_data =
138 RTL_PASS, // type
139 "", // name (overwritten by the constructor)
140 OPTGROUP_NONE, // optinfo_flags
141 TV_OPTIMIZE, // tv_id
142 0, // properties_required
143 0, // properties_provided
144 0, // properties_destroyed
145 0, // todo_flags_start
146 0 // todo_flags_finish
149 sh_optimize_sett_clrt::sh_optimize_sett_clrt (gcc::context* ctx,
150 const char* name)
151 : rtl_opt_pass (default_pass_data, ctx),
152 m_ccreg (NULL_RTX)
154 // Overwrite default name in pass_data base class.
155 this->name = name;
158 sh_optimize_sett_clrt::~sh_optimize_sett_clrt (void)
162 bool
163 sh_optimize_sett_clrt::gate (function*)
165 return optimize > 0;
168 unsigned int
169 sh_optimize_sett_clrt::execute (function* fun)
171 unsigned int ccr0 = INVALID_REGNUM;
172 unsigned int ccr1 = INVALID_REGNUM;
174 if (targetm.fixed_condition_code_regs (&ccr0, &ccr1)
175 && ccr0 != INVALID_REGNUM)
177 // Initially create a reg rtx with VOIDmode.
178 // When the constant setcc is discovered, the mode is changed
179 // to the mode that is actually used by the target.
180 m_ccreg = gen_rtx_REG (VOIDmode, ccr0);
183 if (m_ccreg == NULL_RTX)
184 log_return (0, "no ccreg.\n\n");
186 if (STORE_FLAG_VALUE != 1)
187 log_return (0, "unsupported STORE_FLAG_VALUE %d", STORE_FLAG_VALUE);
189 log_msg ("ccreg: ");
190 log_rtx (m_ccreg);
191 log_msg (" STORE_FLAG_VALUE = %d\n", STORE_FLAG_VALUE);
193 if (!df_regs_ever_live_p (ccr0))
194 log_return (0, "ccreg never live\n\n");
196 // Output vector for find_known_ccreg_values.
197 std::vector<ccreg_value> ccreg_values;
198 ccreg_values.reserve (32);
200 // Something for recording visited basic blocks to avoid infinite recursion.
201 std::vector<basic_block> visited_bbs;
202 visited_bbs.reserve (32);
204 // Look for insns that set the ccreg to a constant value and see if it can
205 // be optimized.
206 basic_block bb;
207 FOR_EACH_BB_REVERSE_FN (bb, fun)
208 for (rtx_insn *next_i, *i = NEXT_INSN (BB_HEAD (bb));
209 i != NULL_RTX && i != BB_END (bb); i = next_i)
211 next_i = NEXT_INSN (i);
213 if (!INSN_P (i) || !NONDEBUG_INSN_P (i))
214 continue;
216 rtx setcc_val = const_setcc_value (PATTERN (i));
217 if (setcc_val != NULL_RTX)
219 update_ccreg_mode (GET_MODE (XEXP (PATTERN (i), 0)));
221 log_msg ("\n\nfound const setcc insn in [bb %d]: \n", bb->index);
222 log_insn (i);
223 log_msg ("\n");
225 ccreg_values.clear ();
226 visited_bbs.clear ();
227 bool ok = find_last_ccreg_values (PREV_INSN (i), bb, ccreg_values,
228 visited_bbs);
230 log_msg ("number of ccreg values collected: %u\n",
231 (unsigned int)ccreg_values.size ());
233 // If all the collected values are equal and are equal to the
234 // constant value of the setcc insn, the setcc insn can be
235 // removed.
236 if (ok && all_ccreg_values_equal (ccreg_values)
237 && rtx_equal_p (ccreg_values.front ().value, setcc_val))
239 log_msg ("all values are ");
240 log_rtx (setcc_val);
241 log_msg ("\n");
243 delete_insn (i);
244 remove_ccreg_dead_unused_notes (ccreg_values);
249 log_return (0, "\n\n");
252 void
253 sh_optimize_sett_clrt::update_ccreg_mode (machine_mode m)
255 if (GET_MODE (m_ccreg) == m)
256 return;
258 PUT_MODE (m_ccreg, m);
259 log_msg ("updated ccreg mode: ");
260 log_rtx (m_ccreg);
261 log_msg ("\n\n");
265 sh_optimize_sett_clrt::const_setcc_value (rtx pat) const
267 if (GET_CODE (pat) == SET
268 && REG_P (XEXP (pat, 0)) && REGNO (XEXP (pat, 0)) == REGNO (m_ccreg)
269 && CONST_INT_P (XEXP (pat, 1))
270 && (INTVAL (XEXP (pat, 1)) == 0
271 || INTVAL (XEXP (pat, 1)) == STORE_FLAG_VALUE))
272 return XEXP (pat, 1);
273 else
274 return NULL_RTX;
277 bool
278 sh_optimize_sett_clrt
279 ::sh_cbranch_ccreg_value (rtx_insn *cbranch_insn, basic_block cbranch_insn_bb,
280 basic_block branch_target_bb) const
282 rtx pc_set_rtx = pc_set (cbranch_insn);
283 gcc_assert (pc_set_rtx != NULL_RTX);
284 gcc_assert (branch_target_bb != NULL);
286 rtx cond = XEXP (XEXP (pc_set_rtx, 1), 0);
287 bool branch_if;
289 if (GET_CODE (cond) == NE
290 && REG_P (XEXP (cond, 0)) && REGNO (XEXP (cond, 0)) == REGNO (m_ccreg)
291 && XEXP (cond, 1) == const0_rtx)
292 branch_if = true;
294 else if (GET_CODE (cond) == EQ
295 && REG_P (XEXP (cond, 0)) && REGNO (XEXP (cond, 0)) == REGNO (m_ccreg)
296 && XEXP (cond, 1) == const0_rtx)
297 branch_if = false;
299 else
300 gcc_unreachable ();
302 if (branch_target_bb == BRANCH_EDGE (cbranch_insn_bb)->dest)
303 return branch_if;
304 else if (branch_target_bb == FALLTHRU_EDGE (cbranch_insn_bb)->dest)
305 return !branch_if;
306 else
307 gcc_unreachable ();
310 bool
311 sh_optimize_sett_clrt
312 ::find_last_ccreg_values (rtx_insn *start_insn, basic_block bb,
313 std::vector<ccreg_value>& values_out,
314 std::vector<basic_block>& prev_visited_bb) const
316 // FIXME: For larger CFGs this will unnecessarily re-visit basic blocks.
317 // Once a basic block has been visited, the result should be stored in
318 // some container so that it can be looked up quickly eliminating the
319 // re-visits.
320 log_msg ("looking for ccreg values in [bb %d] ", bb->index);
321 if (!prev_visited_bb.empty ())
322 log_msg ("(prev visited [bb %d])", prev_visited_bb.back ()->index);
323 log_msg ("\n");
325 for (rtx_insn *i = start_insn; i != NULL && i != PREV_INSN (BB_HEAD (bb));
326 i = PREV_INSN (i))
328 if (!INSN_P (i))
329 continue;
331 if (reg_set_p (m_ccreg, i))
333 const_rtx set_rtx = set_of (m_ccreg, i);
335 ccreg_value v;
336 v.insn = i;
337 v.bb = bb;
338 v.value = set_rtx != NULL_RTX && GET_CODE (set_rtx) == SET
339 ? XEXP (set_rtx, 1)
340 : NULL_RTX;
342 log_msg ("found setcc in [bb %d] in insn:\n", bb->index);
343 log_insn (i);
344 log_msg ("\nccreg value: ");
345 log_rtx (v.value);
346 log_msg ("\n");
348 values_out.push_back (v);
349 return true;
352 if (any_condjump_p (i) && onlyjump_p (i) && !prev_visited_bb.empty ())
354 // For a conditional branch the ccreg value will be a known constant
355 // of either 0 or STORE_FLAG_VALUE after branching/falling through
356 // to one of the two successor BBs. Record the value for the BB
357 // where we came from.
358 log_msg ("found cbranch in [bb %d]:\n", bb->index);
359 log_insn (i);
361 ccreg_value v;
362 v.insn = i;
363 v.bb = bb;
364 v.value = GEN_INT (sh_cbranch_ccreg_value (i, bb,
365 prev_visited_bb.back ()));
367 log_msg (" branches to [bb %d] with ccreg value ",
368 prev_visited_bb.back ()->index);
369 log_rtx (v.value);
370 log_msg ("\n");
372 values_out.push_back (v);
373 return true;
377 // If here, we've walked up all the insns of the current basic block
378 // and none of them seems to modify the ccreg.
379 // In this case, check the predecessor basic blocks.
380 unsigned int pred_bb_count = 0;
382 // If the current basic block is not in the stack of previously visited
383 // basic blocks yet, we can recursively check the predecessor basic blocks.
384 // Otherwise we have a loop in the CFG and recursing again will result in
385 // an infinite loop.
386 if (std::find (prev_visited_bb.rbegin (), prev_visited_bb.rend (), bb)
387 == prev_visited_bb.rend ())
389 prev_visited_bb.push_back (bb);
391 for (edge_iterator ei = ei_start (bb->preds); !ei_end_p (ei);
392 ei_next (&ei))
394 if (ei_edge (ei)->flags & EDGE_COMPLEX)
395 log_return (false, "aborting due to complex edge\n");
397 basic_block pred_bb = ei_edge (ei)->src;
398 pred_bb_count += 1;
399 if (!find_last_ccreg_values (BB_END (pred_bb), pred_bb, values_out,
400 prev_visited_bb))
401 return false;
404 prev_visited_bb.pop_back ();
406 else
407 log_msg ("loop detected for [bb %d]\n", bb->index);
409 log_msg ("[bb %d] pred_bb_count = %u\n", bb->index, pred_bb_count);
411 if (pred_bb_count == 0)
413 // If we haven't checked a single predecessor basic block, the current
414 // basic block is probably a leaf block and we don't know the ccreg value.
415 log_msg ("unknown ccreg value for [bb %d]\n", bb->index);
417 ccreg_value v;
418 v.insn = BB_END (bb);
419 v.bb = bb;
420 v.value = NULL_RTX;
422 values_out.push_back (v);
425 return true;
428 bool
429 sh_optimize_sett_clrt
430 ::all_ccreg_values_equal (const std::vector<ccreg_value>& values)
432 if (values.empty ())
433 return false;
435 rtx last_value = values.front ().value;
437 // If the ccreg is modified in the insn but the exact value is not known
438 // the value rtx might be null.
439 if (last_value == NULL_RTX)
440 return false;
442 for (std::vector<ccreg_value>::const_iterator i = values.begin ();
443 i != values.end (); ++i)
444 if (i->value == NULL_RTX || !rtx_equal_p (last_value, i->value))
445 return false;
447 return true;
450 void
451 sh_optimize_sett_clrt
452 ::remove_ccreg_dead_unused_notes (std::vector<ccreg_value>& values) const
454 for (std::vector<ccreg_value>::iterator i = values.begin ();
455 i != values.end (); ++i)
457 if (i->insn == NULL_RTX)
458 continue;
460 rtx n = find_regno_note (i->insn, REG_DEAD, REGNO (m_ccreg));
461 if (n != NULL_RTX)
462 remove_note (i->insn, n);
464 n = find_regno_note (i->insn, REG_UNUSED, REGNO (m_ccreg));
465 if (n != NULL_RTX)
466 remove_note (i->insn, n);
470 // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
471 // This allows instantiating the pass somewhere else without having to pull
472 // in a header file.
473 opt_pass*
474 make_pass_sh_optimize_sett_clrt (gcc::context* ctx, const char* name)
476 return new sh_optimize_sett_clrt (ctx, name);