annotate gcc/lra-eliminations.c @ 131:84e7813d76e9

gcc-8.2
author mir3636
date Thu, 25 Oct 2018 07:37:49 +0900
parents 04ced10e8804
children 1830386684a0
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 /* Code for RTL register eliminations.
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
2 Copyright (C) 2010-2018 Free Software Foundation, Inc.
111
kono
parents:
diff changeset
3 Contributed by Vladimir Makarov <vmakarov@redhat.com>.
kono
parents:
diff changeset
4
kono
parents:
diff changeset
5 This file is part of GCC.
kono
parents:
diff changeset
6
kono
parents:
diff changeset
7 GCC is free software; you can redistribute it and/or modify it under
kono
parents:
diff changeset
8 the terms of the GNU General Public License as published by the Free
kono
parents:
diff changeset
9 Software Foundation; either version 3, or (at your option) any later
kono
parents:
diff changeset
10 version.
kono
parents:
diff changeset
11
kono
parents:
diff changeset
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
kono
parents:
diff changeset
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
kono
parents:
diff changeset
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
kono
parents:
diff changeset
15 for more details.
kono
parents:
diff changeset
16
kono
parents:
diff changeset
17 You should have received a copy of the GNU General Public License
kono
parents:
diff changeset
18 along with GCC; see the file COPYING3. If not see
kono
parents:
diff changeset
19 <http://www.gnu.org/licenses/>. */
kono
parents:
diff changeset
20
kono
parents:
diff changeset
21 /* Eliminable registers (like a soft argument or frame pointer) are
kono
parents:
diff changeset
22 widely used in RTL. These eliminable registers should be replaced
kono
parents:
diff changeset
23 by real hard registers (like the stack pointer or hard frame
kono
parents:
diff changeset
24 pointer) plus some offset. The offsets usually change whenever the
kono
parents:
diff changeset
25 stack is expanded. We know the final offsets only at the very end
kono
parents:
diff changeset
26 of LRA.
kono
parents:
diff changeset
27
kono
parents:
diff changeset
28 Within LRA, we usually keep the RTL in such a state that the
kono
parents:
diff changeset
29 eliminable registers can be replaced by just the corresponding hard
kono
parents:
diff changeset
30 register (without any offset). To achieve this we should add the
kono
parents:
diff changeset
31 initial elimination offset at the beginning of LRA and update the
kono
parents:
diff changeset
32 offsets whenever the stack is expanded. We need to do this before
kono
parents:
diff changeset
33 every constraint pass because the choice of offset often affects
kono
parents:
diff changeset
34 whether a particular address or memory constraint is satisfied.
kono
parents:
diff changeset
35
kono
parents:
diff changeset
36 We keep RTL code at most time in such state that the virtual
kono
parents:
diff changeset
37 registers can be changed by just the corresponding hard registers
kono
parents:
diff changeset
38 (with zero offsets) and we have the right RTL code. To achieve this
kono
parents:
diff changeset
39 we should add initial offset at the beginning of LRA work and update
kono
parents:
diff changeset
40 offsets after each stack expanding. But actually we update virtual
kono
parents:
diff changeset
41 registers to the same virtual registers + corresponding offsets
kono
parents:
diff changeset
42 before every constraint pass because it affects constraint
kono
parents:
diff changeset
43 satisfaction (e.g. an address displacement became too big for some
kono
parents:
diff changeset
44 target).
kono
parents:
diff changeset
45
kono
parents:
diff changeset
46 The final change of eliminable registers to the corresponding hard
kono
parents:
diff changeset
47 registers are done at the very end of LRA when there were no change
kono
parents:
diff changeset
48 in offsets anymore:
kono
parents:
diff changeset
49
kono
parents:
diff changeset
50 fp + 42 => sp + 42
kono
parents:
diff changeset
51
kono
parents:
diff changeset
52 */
kono
parents:
diff changeset
53
kono
parents:
diff changeset
54 #include "config.h"
kono
parents:
diff changeset
55 #include "system.h"
kono
parents:
diff changeset
56 #include "coretypes.h"
kono
parents:
diff changeset
57 #include "backend.h"
kono
parents:
diff changeset
58 #include "target.h"
kono
parents:
diff changeset
59 #include "rtl.h"
kono
parents:
diff changeset
60 #include "tree.h"
kono
parents:
diff changeset
61 #include "df.h"
kono
parents:
diff changeset
62 #include "memmodel.h"
kono
parents:
diff changeset
63 #include "tm_p.h"
kono
parents:
diff changeset
64 #include "optabs.h"
kono
parents:
diff changeset
65 #include "regs.h"
kono
parents:
diff changeset
66 #include "ira.h"
kono
parents:
diff changeset
67 #include "recog.h"
kono
parents:
diff changeset
68 #include "output.h"
kono
parents:
diff changeset
69 #include "rtl-error.h"
kono
parents:
diff changeset
70 #include "lra-int.h"
kono
parents:
diff changeset
71
kono
parents:
diff changeset
72 /* This structure is used to record information about hard register
kono
parents:
diff changeset
73 eliminations. */
kono
parents:
diff changeset
74 struct lra_elim_table
kono
parents:
diff changeset
75 {
kono
parents:
diff changeset
76 /* Hard register number to be eliminated. */
kono
parents:
diff changeset
77 int from;
kono
parents:
diff changeset
78 /* Hard register number used as replacement. */
kono
parents:
diff changeset
79 int to;
kono
parents:
diff changeset
80 /* Difference between values of the two hard registers above on
kono
parents:
diff changeset
81 previous iteration. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
82 poly_int64 previous_offset;
111
kono
parents:
diff changeset
83 /* Difference between the values on the current iteration. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
84 poly_int64 offset;
111
kono
parents:
diff changeset
85 /* Nonzero if this elimination can be done. */
kono
parents:
diff changeset
86 bool can_eliminate;
kono
parents:
diff changeset
87 /* CAN_ELIMINATE since the last check. */
kono
parents:
diff changeset
88 bool prev_can_eliminate;
kono
parents:
diff changeset
89 /* REG rtx for the register to be eliminated. We cannot simply
kono
parents:
diff changeset
90 compare the number since we might then spuriously replace a hard
kono
parents:
diff changeset
91 register corresponding to a pseudo assigned to the reg to be
kono
parents:
diff changeset
92 eliminated. */
kono
parents:
diff changeset
93 rtx from_rtx;
kono
parents:
diff changeset
94 /* REG rtx for the replacement. */
kono
parents:
diff changeset
95 rtx to_rtx;
kono
parents:
diff changeset
96 };
kono
parents:
diff changeset
97
kono
parents:
diff changeset
98 /* The elimination table. Each array entry describes one possible way
kono
parents:
diff changeset
99 of eliminating a register in favor of another. If there is more
kono
parents:
diff changeset
100 than one way of eliminating a particular register, the most
kono
parents:
diff changeset
101 preferred should be specified first. */
kono
parents:
diff changeset
102 static struct lra_elim_table *reg_eliminate = 0;
kono
parents:
diff changeset
103
kono
parents:
diff changeset
104 /* This is an intermediate structure to initialize the table. It has
kono
parents:
diff changeset
105 exactly the members provided by ELIMINABLE_REGS. */
kono
parents:
diff changeset
106 static const struct elim_table_1
kono
parents:
diff changeset
107 {
kono
parents:
diff changeset
108 const int from;
kono
parents:
diff changeset
109 const int to;
kono
parents:
diff changeset
110 } reg_eliminate_1[] =
kono
parents:
diff changeset
111
kono
parents:
diff changeset
112 ELIMINABLE_REGS;
kono
parents:
diff changeset
113
kono
parents:
diff changeset
114 #define NUM_ELIMINABLE_REGS ARRAY_SIZE (reg_eliminate_1)
kono
parents:
diff changeset
115
kono
parents:
diff changeset
116 /* Print info about elimination table to file F. */
kono
parents:
diff changeset
117 static void
kono
parents:
diff changeset
118 print_elim_table (FILE *f)
kono
parents:
diff changeset
119 {
kono
parents:
diff changeset
120 struct lra_elim_table *ep;
kono
parents:
diff changeset
121
kono
parents:
diff changeset
122 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
123 {
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
124 fprintf (f, "%s eliminate %d to %d (offset=",
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
125 ep->can_eliminate ? "Can" : "Can't", ep->from, ep->to);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
126 print_dec (ep->offset, f);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
127 fprintf (f, ", prev_offset=");
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
128 print_dec (ep->previous_offset, f);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
129 fprintf (f, ")\n");
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
130 }
111
kono
parents:
diff changeset
131 }
kono
parents:
diff changeset
132
kono
parents:
diff changeset
133 /* Print info about elimination table to stderr. */
kono
parents:
diff changeset
134 void
kono
parents:
diff changeset
135 lra_debug_elim_table (void)
kono
parents:
diff changeset
136 {
kono
parents:
diff changeset
137 print_elim_table (stderr);
kono
parents:
diff changeset
138 }
kono
parents:
diff changeset
139
kono
parents:
diff changeset
140 /* Setup possibility of elimination in elimination table element EP to
kono
parents:
diff changeset
141 VALUE. Setup FRAME_POINTER_NEEDED if elimination from frame
kono
parents:
diff changeset
142 pointer to stack pointer is not possible anymore. */
kono
parents:
diff changeset
143 static void
kono
parents:
diff changeset
144 setup_can_eliminate (struct lra_elim_table *ep, bool value)
kono
parents:
diff changeset
145 {
kono
parents:
diff changeset
146 ep->can_eliminate = ep->prev_can_eliminate = value;
kono
parents:
diff changeset
147 if (! value
kono
parents:
diff changeset
148 && ep->from == FRAME_POINTER_REGNUM && ep->to == STACK_POINTER_REGNUM)
kono
parents:
diff changeset
149 frame_pointer_needed = 1;
kono
parents:
diff changeset
150 if (!frame_pointer_needed)
kono
parents:
diff changeset
151 REGNO_POINTER_ALIGN (HARD_FRAME_POINTER_REGNUM) = 0;
kono
parents:
diff changeset
152 }
kono
parents:
diff changeset
153
kono
parents:
diff changeset
154 /* Map: eliminable "from" register -> its current elimination,
kono
parents:
diff changeset
155 or NULL if none. The elimination table may contain more than
kono
parents:
diff changeset
156 one elimination for the same hard register, but this map specifies
kono
parents:
diff changeset
157 the one that we are currently using. */
kono
parents:
diff changeset
158 static struct lra_elim_table *elimination_map[FIRST_PSEUDO_REGISTER];
kono
parents:
diff changeset
159
kono
parents:
diff changeset
160 /* When an eliminable hard register becomes not eliminable, we use the
kono
parents:
diff changeset
161 following special structure to restore original offsets for the
kono
parents:
diff changeset
162 register. */
kono
parents:
diff changeset
163 static struct lra_elim_table self_elim_table;
kono
parents:
diff changeset
164
kono
parents:
diff changeset
165 /* Offsets should be used to restore original offsets for eliminable
kono
parents:
diff changeset
166 hard register which just became not eliminable. Zero,
kono
parents:
diff changeset
167 otherwise. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
168 static poly_int64_pod self_elim_offsets[FIRST_PSEUDO_REGISTER];
111
kono
parents:
diff changeset
169
kono
parents:
diff changeset
170 /* Map: hard regno -> RTL presentation. RTL presentations of all
kono
parents:
diff changeset
171 potentially eliminable hard registers are stored in the map. */
kono
parents:
diff changeset
172 static rtx eliminable_reg_rtx[FIRST_PSEUDO_REGISTER];
kono
parents:
diff changeset
173
kono
parents:
diff changeset
174 /* Set up ELIMINATION_MAP of the currently used eliminations. */
kono
parents:
diff changeset
175 static void
kono
parents:
diff changeset
176 setup_elimination_map (void)
kono
parents:
diff changeset
177 {
kono
parents:
diff changeset
178 int i;
kono
parents:
diff changeset
179 struct lra_elim_table *ep;
kono
parents:
diff changeset
180
kono
parents:
diff changeset
181 for (i = 0; i < FIRST_PSEUDO_REGISTER; i++)
kono
parents:
diff changeset
182 elimination_map[i] = NULL;
kono
parents:
diff changeset
183 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
184 if (ep->can_eliminate && elimination_map[ep->from] == NULL)
kono
parents:
diff changeset
185 elimination_map[ep->from] = ep;
kono
parents:
diff changeset
186 }
kono
parents:
diff changeset
187
kono
parents:
diff changeset
188
kono
parents:
diff changeset
189
kono
parents:
diff changeset
190 /* Compute the sum of X and Y, making canonicalizations assumed in an
kono
parents:
diff changeset
191 address, namely: sum constant integers, surround the sum of two
kono
parents:
diff changeset
192 constants with a CONST, put the constant as the second operand, and
kono
parents:
diff changeset
193 group the constant on the outermost sum.
kono
parents:
diff changeset
194
kono
parents:
diff changeset
195 This routine assumes both inputs are already in canonical form. */
kono
parents:
diff changeset
196 static rtx
kono
parents:
diff changeset
197 form_sum (rtx x, rtx y)
kono
parents:
diff changeset
198 {
kono
parents:
diff changeset
199 machine_mode mode = GET_MODE (x);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
200 poly_int64 offset;
111
kono
parents:
diff changeset
201
kono
parents:
diff changeset
202 if (mode == VOIDmode)
kono
parents:
diff changeset
203 mode = GET_MODE (y);
kono
parents:
diff changeset
204
kono
parents:
diff changeset
205 if (mode == VOIDmode)
kono
parents:
diff changeset
206 mode = Pmode;
kono
parents:
diff changeset
207
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
208 if (poly_int_rtx_p (x, &offset))
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
209 return plus_constant (mode, y, offset);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
210 else if (poly_int_rtx_p (y, &offset))
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
211 return plus_constant (mode, x, offset);
111
kono
parents:
diff changeset
212 else if (CONSTANT_P (x))
kono
parents:
diff changeset
213 std::swap (x, y);
kono
parents:
diff changeset
214
kono
parents:
diff changeset
215 if (GET_CODE (x) == PLUS && CONSTANT_P (XEXP (x, 1)))
kono
parents:
diff changeset
216 return form_sum (XEXP (x, 0), form_sum (XEXP (x, 1), y));
kono
parents:
diff changeset
217
kono
parents:
diff changeset
218 /* Note that if the operands of Y are specified in the opposite
kono
parents:
diff changeset
219 order in the recursive calls below, infinite recursion will
kono
parents:
diff changeset
220 occur. */
kono
parents:
diff changeset
221 if (GET_CODE (y) == PLUS && CONSTANT_P (XEXP (y, 1)))
kono
parents:
diff changeset
222 return form_sum (form_sum (x, XEXP (y, 0)), XEXP (y, 1));
kono
parents:
diff changeset
223
kono
parents:
diff changeset
224 /* If both constant, encapsulate sum. Otherwise, just form sum. A
kono
parents:
diff changeset
225 constant will have been placed second. */
kono
parents:
diff changeset
226 if (CONSTANT_P (x) && CONSTANT_P (y))
kono
parents:
diff changeset
227 {
kono
parents:
diff changeset
228 if (GET_CODE (x) == CONST)
kono
parents:
diff changeset
229 x = XEXP (x, 0);
kono
parents:
diff changeset
230 if (GET_CODE (y) == CONST)
kono
parents:
diff changeset
231 y = XEXP (y, 0);
kono
parents:
diff changeset
232
kono
parents:
diff changeset
233 return gen_rtx_CONST (VOIDmode, gen_rtx_PLUS (mode, x, y));
kono
parents:
diff changeset
234 }
kono
parents:
diff changeset
235
kono
parents:
diff changeset
236 return gen_rtx_PLUS (mode, x, y);
kono
parents:
diff changeset
237 }
kono
parents:
diff changeset
238
kono
parents:
diff changeset
239 /* Return the current substitution hard register of the elimination of
kono
parents:
diff changeset
240 HARD_REGNO. If HARD_REGNO is not eliminable, return itself. */
kono
parents:
diff changeset
241 int
kono
parents:
diff changeset
242 lra_get_elimination_hard_regno (int hard_regno)
kono
parents:
diff changeset
243 {
kono
parents:
diff changeset
244 struct lra_elim_table *ep;
kono
parents:
diff changeset
245
kono
parents:
diff changeset
246 if (hard_regno < 0 || hard_regno >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
247 return hard_regno;
kono
parents:
diff changeset
248 if ((ep = elimination_map[hard_regno]) == NULL)
kono
parents:
diff changeset
249 return hard_regno;
kono
parents:
diff changeset
250 return ep->to;
kono
parents:
diff changeset
251 }
kono
parents:
diff changeset
252
kono
parents:
diff changeset
253 /* Return elimination which will be used for hard reg REG, NULL
kono
parents:
diff changeset
254 otherwise. */
kono
parents:
diff changeset
255 static struct lra_elim_table *
kono
parents:
diff changeset
256 get_elimination (rtx reg)
kono
parents:
diff changeset
257 {
kono
parents:
diff changeset
258 int hard_regno;
kono
parents:
diff changeset
259 struct lra_elim_table *ep;
kono
parents:
diff changeset
260
kono
parents:
diff changeset
261 lra_assert (REG_P (reg));
kono
parents:
diff changeset
262 if ((hard_regno = REGNO (reg)) < 0 || hard_regno >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
263 return NULL;
kono
parents:
diff changeset
264 if ((ep = elimination_map[hard_regno]) != NULL)
kono
parents:
diff changeset
265 return ep->from_rtx != reg ? NULL : ep;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
266 poly_int64 offset = self_elim_offsets[hard_regno];
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
267 if (known_eq (offset, 0))
111
kono
parents:
diff changeset
268 return NULL;
kono
parents:
diff changeset
269 /* This is an iteration to restore offsets just after HARD_REGNO
kono
parents:
diff changeset
270 stopped to be eliminable. */
kono
parents:
diff changeset
271 self_elim_table.from = self_elim_table.to = hard_regno;
kono
parents:
diff changeset
272 self_elim_table.from_rtx
kono
parents:
diff changeset
273 = self_elim_table.to_rtx
kono
parents:
diff changeset
274 = eliminable_reg_rtx[hard_regno];
kono
parents:
diff changeset
275 lra_assert (self_elim_table.from_rtx != NULL);
kono
parents:
diff changeset
276 self_elim_table.offset = offset;
kono
parents:
diff changeset
277 return &self_elim_table;
kono
parents:
diff changeset
278 }
kono
parents:
diff changeset
279
kono
parents:
diff changeset
280 /* Transform (subreg (plus reg const)) to (plus (subreg reg) const)
kono
parents:
diff changeset
281 when it is possible. Return X or the transformation result if the
kono
parents:
diff changeset
282 transformation is done. */
kono
parents:
diff changeset
283 static rtx
kono
parents:
diff changeset
284 move_plus_up (rtx x)
kono
parents:
diff changeset
285 {
kono
parents:
diff changeset
286 rtx subreg_reg;
kono
parents:
diff changeset
287 machine_mode x_mode, subreg_reg_mode;
kono
parents:
diff changeset
288
kono
parents:
diff changeset
289 if (GET_CODE (x) != SUBREG || !subreg_lowpart_p (x))
kono
parents:
diff changeset
290 return x;
kono
parents:
diff changeset
291 subreg_reg = SUBREG_REG (x);
kono
parents:
diff changeset
292 x_mode = GET_MODE (x);
kono
parents:
diff changeset
293 subreg_reg_mode = GET_MODE (subreg_reg);
kono
parents:
diff changeset
294 if (!paradoxical_subreg_p (x)
kono
parents:
diff changeset
295 && GET_CODE (subreg_reg) == PLUS
kono
parents:
diff changeset
296 && CONSTANT_P (XEXP (subreg_reg, 1))
kono
parents:
diff changeset
297 && GET_MODE_CLASS (x_mode) == MODE_INT
kono
parents:
diff changeset
298 && GET_MODE_CLASS (subreg_reg_mode) == MODE_INT)
kono
parents:
diff changeset
299 {
kono
parents:
diff changeset
300 rtx cst = simplify_subreg (x_mode, XEXP (subreg_reg, 1), subreg_reg_mode,
kono
parents:
diff changeset
301 subreg_lowpart_offset (x_mode,
kono
parents:
diff changeset
302 subreg_reg_mode));
kono
parents:
diff changeset
303 if (cst && CONSTANT_P (cst))
kono
parents:
diff changeset
304 return gen_rtx_PLUS (x_mode, lowpart_subreg (x_mode,
kono
parents:
diff changeset
305 XEXP (subreg_reg, 0),
kono
parents:
diff changeset
306 subreg_reg_mode), cst);
kono
parents:
diff changeset
307 }
kono
parents:
diff changeset
308 return x;
kono
parents:
diff changeset
309 }
kono
parents:
diff changeset
310
kono
parents:
diff changeset
311 /* Scan X and replace any eliminable registers (such as fp) with a
kono
parents:
diff changeset
312 replacement (such as sp) if SUBST_P, plus an offset. The offset is
kono
parents:
diff changeset
313 a change in the offset between the eliminable register and its
kono
parents:
diff changeset
314 substitution if UPDATE_P, or the full offset if FULL_P, or
kono
parents:
diff changeset
315 otherwise zero. If FULL_P, we also use the SP offsets for
kono
parents:
diff changeset
316 elimination to SP. If UPDATE_P, use UPDATE_SP_OFFSET for updating
kono
parents:
diff changeset
317 offsets of register elimnable to SP. If UPDATE_SP_OFFSET is
kono
parents:
diff changeset
318 non-zero, don't use difference of the offset and the previous
kono
parents:
diff changeset
319 offset.
kono
parents:
diff changeset
320
kono
parents:
diff changeset
321 MEM_MODE is the mode of an enclosing MEM. We need this to know how
kono
parents:
diff changeset
322 much to adjust a register for, e.g., PRE_DEC. Also, if we are
kono
parents:
diff changeset
323 inside a MEM, we are allowed to replace a sum of a hard register
kono
parents:
diff changeset
324 and the constant zero with the hard register, which we cannot do
kono
parents:
diff changeset
325 outside a MEM. In addition, we need to record the fact that a
kono
parents:
diff changeset
326 hard register is referenced outside a MEM.
kono
parents:
diff changeset
327
kono
parents:
diff changeset
328 If we make full substitution to SP for non-null INSN, add the insn
kono
parents:
diff changeset
329 sp offset. */
kono
parents:
diff changeset
330 rtx
kono
parents:
diff changeset
331 lra_eliminate_regs_1 (rtx_insn *insn, rtx x, machine_mode mem_mode,
kono
parents:
diff changeset
332 bool subst_p, bool update_p,
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
333 poly_int64 update_sp_offset, bool full_p)
111
kono
parents:
diff changeset
334 {
kono
parents:
diff changeset
335 enum rtx_code code = GET_CODE (x);
kono
parents:
diff changeset
336 struct lra_elim_table *ep;
kono
parents:
diff changeset
337 rtx new_rtx;
kono
parents:
diff changeset
338 int i, j;
kono
parents:
diff changeset
339 const char *fmt;
kono
parents:
diff changeset
340 int copied = 0;
kono
parents:
diff changeset
341
kono
parents:
diff changeset
342 lra_assert (!update_p || !full_p);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
343 lra_assert (known_eq (update_sp_offset, 0)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
344 || (!subst_p && update_p && !full_p));
111
kono
parents:
diff changeset
345 if (! current_function_decl)
kono
parents:
diff changeset
346 return x;
kono
parents:
diff changeset
347
kono
parents:
diff changeset
348 switch (code)
kono
parents:
diff changeset
349 {
kono
parents:
diff changeset
350 CASE_CONST_ANY:
kono
parents:
diff changeset
351 case CONST:
kono
parents:
diff changeset
352 case SYMBOL_REF:
kono
parents:
diff changeset
353 case CODE_LABEL:
kono
parents:
diff changeset
354 case PC:
kono
parents:
diff changeset
355 case CC0:
kono
parents:
diff changeset
356 case ASM_INPUT:
kono
parents:
diff changeset
357 case ADDR_VEC:
kono
parents:
diff changeset
358 case ADDR_DIFF_VEC:
kono
parents:
diff changeset
359 case RETURN:
kono
parents:
diff changeset
360 return x;
kono
parents:
diff changeset
361
kono
parents:
diff changeset
362 case REG:
kono
parents:
diff changeset
363 /* First handle the case where we encounter a bare hard register
kono
parents:
diff changeset
364 that is eliminable. Replace it with a PLUS. */
kono
parents:
diff changeset
365 if ((ep = get_elimination (x)) != NULL)
kono
parents:
diff changeset
366 {
kono
parents:
diff changeset
367 rtx to = subst_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
368
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
369 if (maybe_ne (update_sp_offset, 0))
111
kono
parents:
diff changeset
370 {
kono
parents:
diff changeset
371 if (ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
372 return plus_constant (Pmode, to, update_sp_offset);
kono
parents:
diff changeset
373 return to;
kono
parents:
diff changeset
374 }
kono
parents:
diff changeset
375 else if (update_p)
kono
parents:
diff changeset
376 return plus_constant (Pmode, to, ep->offset - ep->previous_offset);
kono
parents:
diff changeset
377 else if (full_p)
kono
parents:
diff changeset
378 return plus_constant (Pmode, to,
kono
parents:
diff changeset
379 ep->offset
kono
parents:
diff changeset
380 - (insn != NULL_RTX
kono
parents:
diff changeset
381 && ep->to_rtx == stack_pointer_rtx
kono
parents:
diff changeset
382 ? lra_get_insn_recog_data (insn)->sp_offset
kono
parents:
diff changeset
383 : 0));
kono
parents:
diff changeset
384 else
kono
parents:
diff changeset
385 return to;
kono
parents:
diff changeset
386 }
kono
parents:
diff changeset
387 return x;
kono
parents:
diff changeset
388
kono
parents:
diff changeset
389 case PLUS:
kono
parents:
diff changeset
390 /* If this is the sum of an eliminable register and a constant, rework
kono
parents:
diff changeset
391 the sum. */
kono
parents:
diff changeset
392 if (REG_P (XEXP (x, 0)) && CONSTANT_P (XEXP (x, 1)))
kono
parents:
diff changeset
393 {
kono
parents:
diff changeset
394 if ((ep = get_elimination (XEXP (x, 0))) != NULL)
kono
parents:
diff changeset
395 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
396 poly_int64 offset, curr_offset;
111
kono
parents:
diff changeset
397 rtx to = subst_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
398
kono
parents:
diff changeset
399 if (! update_p && ! full_p)
kono
parents:
diff changeset
400 return gen_rtx_PLUS (Pmode, to, XEXP (x, 1));
kono
parents:
diff changeset
401
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
402 if (maybe_ne (update_sp_offset, 0))
111
kono
parents:
diff changeset
403 offset = ep->to_rtx == stack_pointer_rtx ? update_sp_offset : 0;
kono
parents:
diff changeset
404 else
kono
parents:
diff changeset
405 offset = (update_p
kono
parents:
diff changeset
406 ? ep->offset - ep->previous_offset : ep->offset);
kono
parents:
diff changeset
407 if (full_p && insn != NULL_RTX && ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
408 offset -= lra_get_insn_recog_data (insn)->sp_offset;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
409 if (poly_int_rtx_p (XEXP (x, 1), &curr_offset)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
410 && known_eq (curr_offset, -offset))
111
kono
parents:
diff changeset
411 return to;
kono
parents:
diff changeset
412 else
kono
parents:
diff changeset
413 return gen_rtx_PLUS (Pmode, to,
kono
parents:
diff changeset
414 plus_constant (Pmode,
kono
parents:
diff changeset
415 XEXP (x, 1), offset));
kono
parents:
diff changeset
416 }
kono
parents:
diff changeset
417
kono
parents:
diff changeset
418 /* If the hard register is not eliminable, we are done since
kono
parents:
diff changeset
419 the other operand is a constant. */
kono
parents:
diff changeset
420 return x;
kono
parents:
diff changeset
421 }
kono
parents:
diff changeset
422
kono
parents:
diff changeset
423 /* If this is part of an address, we want to bring any constant
kono
parents:
diff changeset
424 to the outermost PLUS. We will do this by doing hard
kono
parents:
diff changeset
425 register replacement in our operands and seeing if a constant
kono
parents:
diff changeset
426 shows up in one of them.
kono
parents:
diff changeset
427
kono
parents:
diff changeset
428 Note that there is no risk of modifying the structure of the
kono
parents:
diff changeset
429 insn, since we only get called for its operands, thus we are
kono
parents:
diff changeset
430 either modifying the address inside a MEM, or something like
kono
parents:
diff changeset
431 an address operand of a load-address insn. */
kono
parents:
diff changeset
432
kono
parents:
diff changeset
433 {
kono
parents:
diff changeset
434 rtx new0 = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
435 subst_p, update_p,
kono
parents:
diff changeset
436 update_sp_offset, full_p);
kono
parents:
diff changeset
437 rtx new1 = lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
438 subst_p, update_p,
kono
parents:
diff changeset
439 update_sp_offset, full_p);
kono
parents:
diff changeset
440
kono
parents:
diff changeset
441 new0 = move_plus_up (new0);
kono
parents:
diff changeset
442 new1 = move_plus_up (new1);
kono
parents:
diff changeset
443 if (new0 != XEXP (x, 0) || new1 != XEXP (x, 1))
kono
parents:
diff changeset
444 return form_sum (new0, new1);
kono
parents:
diff changeset
445 }
kono
parents:
diff changeset
446 return x;
kono
parents:
diff changeset
447
kono
parents:
diff changeset
448 case MULT:
kono
parents:
diff changeset
449 /* If this is the product of an eliminable hard register and a
kono
parents:
diff changeset
450 constant, apply the distribute law and move the constant out
kono
parents:
diff changeset
451 so that we have (plus (mult ..) ..). This is needed in order
kono
parents:
diff changeset
452 to keep load-address insns valid. This case is pathological.
kono
parents:
diff changeset
453 We ignore the possibility of overflow here. */
kono
parents:
diff changeset
454 if (REG_P (XEXP (x, 0)) && CONST_INT_P (XEXP (x, 1))
kono
parents:
diff changeset
455 && (ep = get_elimination (XEXP (x, 0))) != NULL)
kono
parents:
diff changeset
456 {
kono
parents:
diff changeset
457 rtx to = subst_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
458
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
459 if (maybe_ne (update_sp_offset, 0))
111
kono
parents:
diff changeset
460 {
kono
parents:
diff changeset
461 if (ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
462 return plus_constant (Pmode,
kono
parents:
diff changeset
463 gen_rtx_MULT (Pmode, to, XEXP (x, 1)),
kono
parents:
diff changeset
464 update_sp_offset * INTVAL (XEXP (x, 1)));
kono
parents:
diff changeset
465 return gen_rtx_MULT (Pmode, to, XEXP (x, 1));
kono
parents:
diff changeset
466 }
kono
parents:
diff changeset
467 else if (update_p)
kono
parents:
diff changeset
468 return plus_constant (Pmode,
kono
parents:
diff changeset
469 gen_rtx_MULT (Pmode, to, XEXP (x, 1)),
kono
parents:
diff changeset
470 (ep->offset - ep->previous_offset)
kono
parents:
diff changeset
471 * INTVAL (XEXP (x, 1)));
kono
parents:
diff changeset
472 else if (full_p)
kono
parents:
diff changeset
473 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
474 poly_int64 offset = ep->offset;
111
kono
parents:
diff changeset
475
kono
parents:
diff changeset
476 if (insn != NULL_RTX && ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
477 offset -= lra_get_insn_recog_data (insn)->sp_offset;
kono
parents:
diff changeset
478 return
kono
parents:
diff changeset
479 plus_constant (Pmode,
kono
parents:
diff changeset
480 gen_rtx_MULT (Pmode, to, XEXP (x, 1)),
kono
parents:
diff changeset
481 offset * INTVAL (XEXP (x, 1)));
kono
parents:
diff changeset
482 }
kono
parents:
diff changeset
483 else
kono
parents:
diff changeset
484 return gen_rtx_MULT (Pmode, to, XEXP (x, 1));
kono
parents:
diff changeset
485 }
kono
parents:
diff changeset
486
kono
parents:
diff changeset
487 /* fall through */
kono
parents:
diff changeset
488
kono
parents:
diff changeset
489 case CALL:
kono
parents:
diff changeset
490 case COMPARE:
kono
parents:
diff changeset
491 /* See comments before PLUS about handling MINUS. */
kono
parents:
diff changeset
492 case MINUS:
kono
parents:
diff changeset
493 case DIV: case UDIV:
kono
parents:
diff changeset
494 case MOD: case UMOD:
kono
parents:
diff changeset
495 case AND: case IOR: case XOR:
kono
parents:
diff changeset
496 case ROTATERT: case ROTATE:
kono
parents:
diff changeset
497 case ASHIFTRT: case LSHIFTRT: case ASHIFT:
kono
parents:
diff changeset
498 case NE: case EQ:
kono
parents:
diff changeset
499 case GE: case GT: case GEU: case GTU:
kono
parents:
diff changeset
500 case LE: case LT: case LEU: case LTU:
kono
parents:
diff changeset
501 {
kono
parents:
diff changeset
502 rtx new0 = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
503 subst_p, update_p,
kono
parents:
diff changeset
504 update_sp_offset, full_p);
kono
parents:
diff changeset
505 rtx new1 = XEXP (x, 1)
kono
parents:
diff changeset
506 ? lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
507 subst_p, update_p,
kono
parents:
diff changeset
508 update_sp_offset, full_p) : 0;
kono
parents:
diff changeset
509
kono
parents:
diff changeset
510 if (new0 != XEXP (x, 0) || new1 != XEXP (x, 1))
kono
parents:
diff changeset
511 return gen_rtx_fmt_ee (code, GET_MODE (x), new0, new1);
kono
parents:
diff changeset
512 }
kono
parents:
diff changeset
513 return x;
kono
parents:
diff changeset
514
kono
parents:
diff changeset
515 case EXPR_LIST:
kono
parents:
diff changeset
516 /* If we have something in XEXP (x, 0), the usual case,
kono
parents:
diff changeset
517 eliminate it. */
kono
parents:
diff changeset
518 if (XEXP (x, 0))
kono
parents:
diff changeset
519 {
kono
parents:
diff changeset
520 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
521 subst_p, update_p,
kono
parents:
diff changeset
522 update_sp_offset, full_p);
kono
parents:
diff changeset
523 if (new_rtx != XEXP (x, 0))
kono
parents:
diff changeset
524 {
kono
parents:
diff changeset
525 /* If this is a REG_DEAD note, it is not valid anymore.
kono
parents:
diff changeset
526 Using the eliminated version could result in creating a
kono
parents:
diff changeset
527 REG_DEAD note for the stack or frame pointer. */
kono
parents:
diff changeset
528 if (REG_NOTE_KIND (x) == REG_DEAD)
kono
parents:
diff changeset
529 return (XEXP (x, 1)
kono
parents:
diff changeset
530 ? lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
531 subst_p, update_p,
kono
parents:
diff changeset
532 update_sp_offset, full_p)
kono
parents:
diff changeset
533 : NULL_RTX);
kono
parents:
diff changeset
534
kono
parents:
diff changeset
535 x = alloc_reg_note (REG_NOTE_KIND (x), new_rtx, XEXP (x, 1));
kono
parents:
diff changeset
536 }
kono
parents:
diff changeset
537 }
kono
parents:
diff changeset
538
kono
parents:
diff changeset
539 /* fall through */
kono
parents:
diff changeset
540
kono
parents:
diff changeset
541 case INSN_LIST:
kono
parents:
diff changeset
542 case INT_LIST:
kono
parents:
diff changeset
543 /* Now do eliminations in the rest of the chain. If this was
kono
parents:
diff changeset
544 an EXPR_LIST, this might result in allocating more memory than is
kono
parents:
diff changeset
545 strictly needed, but it simplifies the code. */
kono
parents:
diff changeset
546 if (XEXP (x, 1))
kono
parents:
diff changeset
547 {
kono
parents:
diff changeset
548 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
549 subst_p, update_p,
kono
parents:
diff changeset
550 update_sp_offset, full_p);
kono
parents:
diff changeset
551 if (new_rtx != XEXP (x, 1))
kono
parents:
diff changeset
552 return
kono
parents:
diff changeset
553 gen_rtx_fmt_ee (GET_CODE (x), GET_MODE (x),
kono
parents:
diff changeset
554 XEXP (x, 0), new_rtx);
kono
parents:
diff changeset
555 }
kono
parents:
diff changeset
556 return x;
kono
parents:
diff changeset
557
kono
parents:
diff changeset
558 case PRE_INC:
kono
parents:
diff changeset
559 case POST_INC:
kono
parents:
diff changeset
560 case PRE_DEC:
kono
parents:
diff changeset
561 case POST_DEC:
kono
parents:
diff changeset
562 /* We do not support elimination of a register that is modified.
kono
parents:
diff changeset
563 elimination_effects has already make sure that this does not
kono
parents:
diff changeset
564 happen. */
kono
parents:
diff changeset
565 return x;
kono
parents:
diff changeset
566
kono
parents:
diff changeset
567 case PRE_MODIFY:
kono
parents:
diff changeset
568 case POST_MODIFY:
kono
parents:
diff changeset
569 /* We do not support elimination of a hard register that is
kono
parents:
diff changeset
570 modified. LRA has already make sure that this does not
kono
parents:
diff changeset
571 happen. The only remaining case we need to consider here is
kono
parents:
diff changeset
572 that the increment value may be an eliminable register. */
kono
parents:
diff changeset
573 if (GET_CODE (XEXP (x, 1)) == PLUS
kono
parents:
diff changeset
574 && XEXP (XEXP (x, 1), 0) == XEXP (x, 0))
kono
parents:
diff changeset
575 {
kono
parents:
diff changeset
576 rtx new_rtx = lra_eliminate_regs_1 (insn, XEXP (XEXP (x, 1), 1),
kono
parents:
diff changeset
577 mem_mode, subst_p, update_p,
kono
parents:
diff changeset
578 update_sp_offset, full_p);
kono
parents:
diff changeset
579
kono
parents:
diff changeset
580 if (new_rtx != XEXP (XEXP (x, 1), 1))
kono
parents:
diff changeset
581 return gen_rtx_fmt_ee (code, GET_MODE (x), XEXP (x, 0),
kono
parents:
diff changeset
582 gen_rtx_PLUS (GET_MODE (x),
kono
parents:
diff changeset
583 XEXP (x, 0), new_rtx));
kono
parents:
diff changeset
584 }
kono
parents:
diff changeset
585 return x;
kono
parents:
diff changeset
586
kono
parents:
diff changeset
587 case STRICT_LOW_PART:
kono
parents:
diff changeset
588 case NEG: case NOT:
kono
parents:
diff changeset
589 case SIGN_EXTEND: case ZERO_EXTEND:
kono
parents:
diff changeset
590 case TRUNCATE: case FLOAT_EXTEND: case FLOAT_TRUNCATE:
kono
parents:
diff changeset
591 case FLOAT: case FIX:
kono
parents:
diff changeset
592 case UNSIGNED_FIX: case UNSIGNED_FLOAT:
kono
parents:
diff changeset
593 case ABS:
kono
parents:
diff changeset
594 case SQRT:
kono
parents:
diff changeset
595 case FFS:
kono
parents:
diff changeset
596 case CLZ:
kono
parents:
diff changeset
597 case CTZ:
kono
parents:
diff changeset
598 case POPCOUNT:
kono
parents:
diff changeset
599 case PARITY:
kono
parents:
diff changeset
600 case BSWAP:
kono
parents:
diff changeset
601 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
602 subst_p, update_p,
kono
parents:
diff changeset
603 update_sp_offset, full_p);
kono
parents:
diff changeset
604 if (new_rtx != XEXP (x, 0))
kono
parents:
diff changeset
605 return gen_rtx_fmt_e (code, GET_MODE (x), new_rtx);
kono
parents:
diff changeset
606 return x;
kono
parents:
diff changeset
607
kono
parents:
diff changeset
608 case SUBREG:
kono
parents:
diff changeset
609 new_rtx = lra_eliminate_regs_1 (insn, SUBREG_REG (x), mem_mode,
kono
parents:
diff changeset
610 subst_p, update_p,
kono
parents:
diff changeset
611 update_sp_offset, full_p);
kono
parents:
diff changeset
612
kono
parents:
diff changeset
613 if (new_rtx != SUBREG_REG (x))
kono
parents:
diff changeset
614 {
kono
parents:
diff changeset
615 if (MEM_P (new_rtx) && !paradoxical_subreg_p (x))
kono
parents:
diff changeset
616 {
kono
parents:
diff changeset
617 SUBREG_REG (x) = new_rtx;
kono
parents:
diff changeset
618 alter_subreg (&x, false);
kono
parents:
diff changeset
619 return x;
kono
parents:
diff changeset
620 }
kono
parents:
diff changeset
621 else if (! subst_p)
kono
parents:
diff changeset
622 {
kono
parents:
diff changeset
623 /* LRA can transform subregs itself. So don't call
kono
parents:
diff changeset
624 simplify_gen_subreg until LRA transformations are
kono
parents:
diff changeset
625 finished. Function simplify_gen_subreg can do
kono
parents:
diff changeset
626 non-trivial transformations (like truncation) which
kono
parents:
diff changeset
627 might make LRA work to fail. */
kono
parents:
diff changeset
628 SUBREG_REG (x) = new_rtx;
kono
parents:
diff changeset
629 return x;
kono
parents:
diff changeset
630 }
kono
parents:
diff changeset
631 else
kono
parents:
diff changeset
632 return simplify_gen_subreg (GET_MODE (x), new_rtx,
kono
parents:
diff changeset
633 GET_MODE (new_rtx), SUBREG_BYTE (x));
kono
parents:
diff changeset
634 }
kono
parents:
diff changeset
635
kono
parents:
diff changeset
636 return x;
kono
parents:
diff changeset
637
kono
parents:
diff changeset
638 case MEM:
kono
parents:
diff changeset
639 /* Our only special processing is to pass the mode of the MEM to our
kono
parents:
diff changeset
640 recursive call and copy the flags. While we are here, handle this
kono
parents:
diff changeset
641 case more efficiently. */
kono
parents:
diff changeset
642 return
kono
parents:
diff changeset
643 replace_equiv_address_nv
kono
parents:
diff changeset
644 (x,
kono
parents:
diff changeset
645 lra_eliminate_regs_1 (insn, XEXP (x, 0), GET_MODE (x),
kono
parents:
diff changeset
646 subst_p, update_p, update_sp_offset, full_p));
kono
parents:
diff changeset
647
kono
parents:
diff changeset
648 case USE:
kono
parents:
diff changeset
649 /* Handle insn_list USE that a call to a pure function may generate. */
kono
parents:
diff changeset
650 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 0), VOIDmode,
kono
parents:
diff changeset
651 subst_p, update_p, update_sp_offset, full_p);
kono
parents:
diff changeset
652 if (new_rtx != XEXP (x, 0))
kono
parents:
diff changeset
653 return gen_rtx_USE (GET_MODE (x), new_rtx);
kono
parents:
diff changeset
654 return x;
kono
parents:
diff changeset
655
kono
parents:
diff changeset
656 case CLOBBER:
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
657 case CLOBBER_HIGH:
111
kono
parents:
diff changeset
658 case SET:
kono
parents:
diff changeset
659 gcc_unreachable ();
kono
parents:
diff changeset
660
kono
parents:
diff changeset
661 default:
kono
parents:
diff changeset
662 break;
kono
parents:
diff changeset
663 }
kono
parents:
diff changeset
664
kono
parents:
diff changeset
665 /* Process each of our operands recursively. If any have changed, make a
kono
parents:
diff changeset
666 copy of the rtx. */
kono
parents:
diff changeset
667 fmt = GET_RTX_FORMAT (code);
kono
parents:
diff changeset
668 for (i = 0; i < GET_RTX_LENGTH (code); i++, fmt++)
kono
parents:
diff changeset
669 {
kono
parents:
diff changeset
670 if (*fmt == 'e')
kono
parents:
diff changeset
671 {
kono
parents:
diff changeset
672 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, i), mem_mode,
kono
parents:
diff changeset
673 subst_p, update_p,
kono
parents:
diff changeset
674 update_sp_offset, full_p);
kono
parents:
diff changeset
675 if (new_rtx != XEXP (x, i) && ! copied)
kono
parents:
diff changeset
676 {
kono
parents:
diff changeset
677 x = shallow_copy_rtx (x);
kono
parents:
diff changeset
678 copied = 1;
kono
parents:
diff changeset
679 }
kono
parents:
diff changeset
680 XEXP (x, i) = new_rtx;
kono
parents:
diff changeset
681 }
kono
parents:
diff changeset
682 else if (*fmt == 'E')
kono
parents:
diff changeset
683 {
kono
parents:
diff changeset
684 int copied_vec = 0;
kono
parents:
diff changeset
685 for (j = 0; j < XVECLEN (x, i); j++)
kono
parents:
diff changeset
686 {
kono
parents:
diff changeset
687 new_rtx = lra_eliminate_regs_1 (insn, XVECEXP (x, i, j), mem_mode,
kono
parents:
diff changeset
688 subst_p, update_p,
kono
parents:
diff changeset
689 update_sp_offset, full_p);
kono
parents:
diff changeset
690 if (new_rtx != XVECEXP (x, i, j) && ! copied_vec)
kono
parents:
diff changeset
691 {
kono
parents:
diff changeset
692 rtvec new_v = gen_rtvec_v (XVECLEN (x, i),
kono
parents:
diff changeset
693 XVEC (x, i)->elem);
kono
parents:
diff changeset
694 if (! copied)
kono
parents:
diff changeset
695 {
kono
parents:
diff changeset
696 x = shallow_copy_rtx (x);
kono
parents:
diff changeset
697 copied = 1;
kono
parents:
diff changeset
698 }
kono
parents:
diff changeset
699 XVEC (x, i) = new_v;
kono
parents:
diff changeset
700 copied_vec = 1;
kono
parents:
diff changeset
701 }
kono
parents:
diff changeset
702 XVECEXP (x, i, j) = new_rtx;
kono
parents:
diff changeset
703 }
kono
parents:
diff changeset
704 }
kono
parents:
diff changeset
705 }
kono
parents:
diff changeset
706
kono
parents:
diff changeset
707 return x;
kono
parents:
diff changeset
708 }
kono
parents:
diff changeset
709
kono
parents:
diff changeset
710 /* This function is used externally in subsequent passes of GCC. It
kono
parents:
diff changeset
711 always does a full elimination of X. */
kono
parents:
diff changeset
712 rtx
kono
parents:
diff changeset
713 lra_eliminate_regs (rtx x, machine_mode mem_mode,
kono
parents:
diff changeset
714 rtx insn ATTRIBUTE_UNUSED)
kono
parents:
diff changeset
715 {
kono
parents:
diff changeset
716 return lra_eliminate_regs_1 (NULL, x, mem_mode, true, false, 0, true);
kono
parents:
diff changeset
717 }
kono
parents:
diff changeset
718
kono
parents:
diff changeset
719 /* Stack pointer offset before the current insn relative to one at the
kono
parents:
diff changeset
720 func start. RTL insns can change SP explicitly. We keep the
kono
parents:
diff changeset
721 changes from one insn to another through this variable. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
722 static poly_int64 curr_sp_change;
111
kono
parents:
diff changeset
723
kono
parents:
diff changeset
724 /* Scan rtx X for references to elimination source or target registers
kono
parents:
diff changeset
725 in contexts that would prevent the elimination from happening.
kono
parents:
diff changeset
726 Update the table of eliminables to reflect the changed state.
kono
parents:
diff changeset
727 MEM_MODE is the mode of an enclosing MEM rtx, or VOIDmode if not
kono
parents:
diff changeset
728 within a MEM. */
kono
parents:
diff changeset
729 static void
kono
parents:
diff changeset
730 mark_not_eliminable (rtx x, machine_mode mem_mode)
kono
parents:
diff changeset
731 {
kono
parents:
diff changeset
732 enum rtx_code code = GET_CODE (x);
kono
parents:
diff changeset
733 struct lra_elim_table *ep;
kono
parents:
diff changeset
734 int i, j;
kono
parents:
diff changeset
735 const char *fmt;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
736 poly_int64 offset = 0;
111
kono
parents:
diff changeset
737
kono
parents:
diff changeset
738 switch (code)
kono
parents:
diff changeset
739 {
kono
parents:
diff changeset
740 case PRE_INC:
kono
parents:
diff changeset
741 case POST_INC:
kono
parents:
diff changeset
742 case PRE_DEC:
kono
parents:
diff changeset
743 case POST_DEC:
kono
parents:
diff changeset
744 case POST_MODIFY:
kono
parents:
diff changeset
745 case PRE_MODIFY:
kono
parents:
diff changeset
746 if (XEXP (x, 0) == stack_pointer_rtx
kono
parents:
diff changeset
747 && ((code != PRE_MODIFY && code != POST_MODIFY)
kono
parents:
diff changeset
748 || (GET_CODE (XEXP (x, 1)) == PLUS
kono
parents:
diff changeset
749 && XEXP (x, 0) == XEXP (XEXP (x, 1), 0)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
750 && poly_int_rtx_p (XEXP (XEXP (x, 1), 1), &offset))))
111
kono
parents:
diff changeset
751 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
752 poly_int64 size = GET_MODE_SIZE (mem_mode);
111
kono
parents:
diff changeset
753
kono
parents:
diff changeset
754 #ifdef PUSH_ROUNDING
kono
parents:
diff changeset
755 /* If more bytes than MEM_MODE are pushed, account for
kono
parents:
diff changeset
756 them. */
kono
parents:
diff changeset
757 size = PUSH_ROUNDING (size);
kono
parents:
diff changeset
758 #endif
kono
parents:
diff changeset
759 if (code == PRE_DEC || code == POST_DEC)
kono
parents:
diff changeset
760 curr_sp_change -= size;
kono
parents:
diff changeset
761 else if (code == PRE_INC || code == POST_INC)
kono
parents:
diff changeset
762 curr_sp_change += size;
kono
parents:
diff changeset
763 else if (code == PRE_MODIFY || code == POST_MODIFY)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
764 curr_sp_change += offset;
111
kono
parents:
diff changeset
765 }
kono
parents:
diff changeset
766 else if (REG_P (XEXP (x, 0))
kono
parents:
diff changeset
767 && REGNO (XEXP (x, 0)) >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
768 {
kono
parents:
diff changeset
769 /* If we modify the source of an elimination rule, disable
kono
parents:
diff changeset
770 it. Do the same if it is the destination and not the
kono
parents:
diff changeset
771 hard frame register. */
kono
parents:
diff changeset
772 for (ep = reg_eliminate;
kono
parents:
diff changeset
773 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
kono
parents:
diff changeset
774 ep++)
kono
parents:
diff changeset
775 if (ep->from_rtx == XEXP (x, 0)
kono
parents:
diff changeset
776 || (ep->to_rtx == XEXP (x, 0)
kono
parents:
diff changeset
777 && ep->to_rtx != hard_frame_pointer_rtx))
kono
parents:
diff changeset
778 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
779 }
kono
parents:
diff changeset
780 return;
kono
parents:
diff changeset
781
kono
parents:
diff changeset
782 case USE:
kono
parents:
diff changeset
783 if (REG_P (XEXP (x, 0)) && REGNO (XEXP (x, 0)) < FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
784 /* If using a hard register that is the source of an eliminate
kono
parents:
diff changeset
785 we still think can be performed, note it cannot be
kono
parents:
diff changeset
786 performed since we don't know how this hard register is
kono
parents:
diff changeset
787 used. */
kono
parents:
diff changeset
788 for (ep = reg_eliminate;
kono
parents:
diff changeset
789 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
kono
parents:
diff changeset
790 ep++)
kono
parents:
diff changeset
791 if (ep->from_rtx == XEXP (x, 0)
kono
parents:
diff changeset
792 && ep->to_rtx != hard_frame_pointer_rtx)
kono
parents:
diff changeset
793 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
794 return;
kono
parents:
diff changeset
795
kono
parents:
diff changeset
796 case CLOBBER:
kono
parents:
diff changeset
797 if (REG_P (XEXP (x, 0)) && REGNO (XEXP (x, 0)) < FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
798 /* If clobbering a hard register that is the replacement
kono
parents:
diff changeset
799 register for an elimination we still think can be
kono
parents:
diff changeset
800 performed, note that it cannot be performed. Otherwise, we
kono
parents:
diff changeset
801 need not be concerned about it. */
kono
parents:
diff changeset
802 for (ep = reg_eliminate;
kono
parents:
diff changeset
803 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
kono
parents:
diff changeset
804 ep++)
kono
parents:
diff changeset
805 if (ep->to_rtx == XEXP (x, 0)
kono
parents:
diff changeset
806 && ep->to_rtx != hard_frame_pointer_rtx)
kono
parents:
diff changeset
807 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
808 return;
kono
parents:
diff changeset
809
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
810 case CLOBBER_HIGH:
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
811 gcc_assert (REG_P (XEXP (x, 0)));
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
812 gcc_assert (REGNO (XEXP (x, 0)) < FIRST_PSEUDO_REGISTER);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
813 for (ep = reg_eliminate;
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
814 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
815 ep++)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
816 if (reg_is_clobbered_by_clobber_high (ep->to_rtx, XEXP (x, 0)))
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
817 setup_can_eliminate (ep, false);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
818 return;
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
819
111
kono
parents:
diff changeset
820 case SET:
kono
parents:
diff changeset
821 if (SET_DEST (x) == stack_pointer_rtx
kono
parents:
diff changeset
822 && GET_CODE (SET_SRC (x)) == PLUS
kono
parents:
diff changeset
823 && XEXP (SET_SRC (x), 0) == SET_DEST (x)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
824 && poly_int_rtx_p (XEXP (SET_SRC (x), 1), &offset))
111
kono
parents:
diff changeset
825 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
826 curr_sp_change += offset;
111
kono
parents:
diff changeset
827 return;
kono
parents:
diff changeset
828 }
kono
parents:
diff changeset
829 if (! REG_P (SET_DEST (x))
kono
parents:
diff changeset
830 || REGNO (SET_DEST (x)) >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
831 mark_not_eliminable (SET_DEST (x), mem_mode);
kono
parents:
diff changeset
832 else
kono
parents:
diff changeset
833 {
kono
parents:
diff changeset
834 /* See if this is setting the replacement hard register for
kono
parents:
diff changeset
835 an elimination.
kono
parents:
diff changeset
836
kono
parents:
diff changeset
837 If DEST is the hard frame pointer, we do nothing because
kono
parents:
diff changeset
838 we assume that all assignments to the frame pointer are
kono
parents:
diff changeset
839 for non-local gotos and are being done at a time when
kono
parents:
diff changeset
840 they are valid and do not disturb anything else. Some
kono
parents:
diff changeset
841 machines want to eliminate a fake argument pointer (or
kono
parents:
diff changeset
842 even a fake frame pointer) with either the real frame
kono
parents:
diff changeset
843 pointer or the stack pointer. Assignments to the hard
kono
parents:
diff changeset
844 frame pointer must not prevent this elimination. */
kono
parents:
diff changeset
845 for (ep = reg_eliminate;
kono
parents:
diff changeset
846 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
kono
parents:
diff changeset
847 ep++)
kono
parents:
diff changeset
848 if (ep->to_rtx == SET_DEST (x)
kono
parents:
diff changeset
849 && SET_DEST (x) != hard_frame_pointer_rtx)
kono
parents:
diff changeset
850 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
851 }
kono
parents:
diff changeset
852
kono
parents:
diff changeset
853 mark_not_eliminable (SET_SRC (x), mem_mode);
kono
parents:
diff changeset
854 return;
kono
parents:
diff changeset
855
kono
parents:
diff changeset
856 case MEM:
kono
parents:
diff changeset
857 /* Our only special processing is to pass the mode of the MEM to
kono
parents:
diff changeset
858 our recursive call. */
kono
parents:
diff changeset
859 mark_not_eliminable (XEXP (x, 0), GET_MODE (x));
kono
parents:
diff changeset
860 return;
kono
parents:
diff changeset
861
kono
parents:
diff changeset
862 default:
kono
parents:
diff changeset
863 break;
kono
parents:
diff changeset
864 }
kono
parents:
diff changeset
865
kono
parents:
diff changeset
866 fmt = GET_RTX_FORMAT (code);
kono
parents:
diff changeset
867 for (i = 0; i < GET_RTX_LENGTH (code); i++, fmt++)
kono
parents:
diff changeset
868 {
kono
parents:
diff changeset
869 if (*fmt == 'e')
kono
parents:
diff changeset
870 mark_not_eliminable (XEXP (x, i), mem_mode);
kono
parents:
diff changeset
871 else if (*fmt == 'E')
kono
parents:
diff changeset
872 for (j = 0; j < XVECLEN (x, i); j++)
kono
parents:
diff changeset
873 mark_not_eliminable (XVECEXP (x, i, j), mem_mode);
kono
parents:
diff changeset
874 }
kono
parents:
diff changeset
875 }
kono
parents:
diff changeset
876
kono
parents:
diff changeset
877
kono
parents:
diff changeset
878
kono
parents:
diff changeset
879 #ifdef HARD_FRAME_POINTER_REGNUM
kono
parents:
diff changeset
880
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
881 /* Search INSN's reg notes to see whether the destination is equal to
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
882 WHAT + C for some constant C. Return true if so, storing C in
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
883 *OFFSET_OUT and removing the reg note. */
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
884 static bool
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
885 remove_reg_equal_offset_note (rtx_insn *insn, rtx what, poly_int64 *offset_out)
111
kono
parents:
diff changeset
886 {
kono
parents:
diff changeset
887 rtx link, *link_loc;
kono
parents:
diff changeset
888
kono
parents:
diff changeset
889 for (link_loc = &REG_NOTES (insn);
kono
parents:
diff changeset
890 (link = *link_loc) != NULL_RTX;
kono
parents:
diff changeset
891 link_loc = &XEXP (link, 1))
kono
parents:
diff changeset
892 if (REG_NOTE_KIND (link) == REG_EQUAL
kono
parents:
diff changeset
893 && GET_CODE (XEXP (link, 0)) == PLUS
kono
parents:
diff changeset
894 && XEXP (XEXP (link, 0), 0) == what
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
895 && poly_int_rtx_p (XEXP (XEXP (link, 0), 1), offset_out))
111
kono
parents:
diff changeset
896 {
kono
parents:
diff changeset
897 *link_loc = XEXP (link, 1);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
898 return true;
111
kono
parents:
diff changeset
899 }
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
900 return false;
111
kono
parents:
diff changeset
901 }
kono
parents:
diff changeset
902
kono
parents:
diff changeset
903 #endif
kono
parents:
diff changeset
904
kono
parents:
diff changeset
905 /* Scan INSN and eliminate all eliminable hard registers in it.
kono
parents:
diff changeset
906
kono
parents:
diff changeset
907 If REPLACE_P is true, do the replacement destructively. Also
kono
parents:
diff changeset
908 delete the insn as dead it if it is setting an eliminable register.
kono
parents:
diff changeset
909
kono
parents:
diff changeset
910 If REPLACE_P is false, just update the offsets while keeping the
kono
parents:
diff changeset
911 base register the same. If FIRST_P, use the sp offset for
kono
parents:
diff changeset
912 elimination to sp. Otherwise, use UPDATE_SP_OFFSET for this. If
kono
parents:
diff changeset
913 UPDATE_SP_OFFSET is non-zero, don't use difference of the offset
kono
parents:
diff changeset
914 and the previous offset. Attach the note about used elimination
kono
parents:
diff changeset
915 for insns setting frame pointer to update elimination easy (without
kono
parents:
diff changeset
916 parsing already generated elimination insns to find offset
kono
parents:
diff changeset
917 previously used) in future. */
kono
parents:
diff changeset
918
kono
parents:
diff changeset
919 void
kono
parents:
diff changeset
920 eliminate_regs_in_insn (rtx_insn *insn, bool replace_p, bool first_p,
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
921 poly_int64 update_sp_offset)
111
kono
parents:
diff changeset
922 {
kono
parents:
diff changeset
923 int icode = recog_memoized (insn);
kono
parents:
diff changeset
924 rtx old_set = single_set (insn);
kono
parents:
diff changeset
925 bool validate_p;
kono
parents:
diff changeset
926 int i;
kono
parents:
diff changeset
927 rtx substed_operand[MAX_RECOG_OPERANDS];
kono
parents:
diff changeset
928 rtx orig_operand[MAX_RECOG_OPERANDS];
kono
parents:
diff changeset
929 struct lra_elim_table *ep;
kono
parents:
diff changeset
930 rtx plus_src, plus_cst_src;
kono
parents:
diff changeset
931 lra_insn_recog_data_t id;
kono
parents:
diff changeset
932 struct lra_static_insn_data *static_id;
kono
parents:
diff changeset
933
kono
parents:
diff changeset
934 if (icode < 0 && asm_noperands (PATTERN (insn)) < 0 && ! DEBUG_INSN_P (insn))
kono
parents:
diff changeset
935 {
kono
parents:
diff changeset
936 lra_assert (GET_CODE (PATTERN (insn)) == USE
kono
parents:
diff changeset
937 || GET_CODE (PATTERN (insn)) == CLOBBER
kono
parents:
diff changeset
938 || GET_CODE (PATTERN (insn)) == ASM_INPUT);
kono
parents:
diff changeset
939 return;
kono
parents:
diff changeset
940 }
kono
parents:
diff changeset
941
kono
parents:
diff changeset
942 /* Check for setting an eliminable register. */
kono
parents:
diff changeset
943 if (old_set != 0 && REG_P (SET_DEST (old_set))
kono
parents:
diff changeset
944 && (ep = get_elimination (SET_DEST (old_set))) != NULL)
kono
parents:
diff changeset
945 {
kono
parents:
diff changeset
946 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
947 if (ep->from_rtx == SET_DEST (old_set) && ep->can_eliminate)
kono
parents:
diff changeset
948 {
kono
parents:
diff changeset
949 bool delete_p = replace_p;
kono
parents:
diff changeset
950
kono
parents:
diff changeset
951 #ifdef HARD_FRAME_POINTER_REGNUM
kono
parents:
diff changeset
952 if (ep->from == FRAME_POINTER_REGNUM
kono
parents:
diff changeset
953 && ep->to == HARD_FRAME_POINTER_REGNUM)
kono
parents:
diff changeset
954 /* If this is setting the frame pointer register to the
kono
parents:
diff changeset
955 hardware frame pointer register and this is an
kono
parents:
diff changeset
956 elimination that will be done (tested above), this
kono
parents:
diff changeset
957 insn is really adjusting the frame pointer downward
kono
parents:
diff changeset
958 to compensate for the adjustment done before a
kono
parents:
diff changeset
959 nonlocal goto. */
kono
parents:
diff changeset
960 {
kono
parents:
diff changeset
961 rtx src = SET_SRC (old_set);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
962 poly_int64 offset = 0;
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
963
111
kono
parents:
diff changeset
964 /* We should never process such insn with non-zero
kono
parents:
diff changeset
965 UPDATE_SP_OFFSET. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
966 lra_assert (known_eq (update_sp_offset, 0));
111
kono
parents:
diff changeset
967
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
968 if (remove_reg_equal_offset_note (insn, ep->to_rtx, &offset)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
969 || strip_offset (src, &offset) == ep->to_rtx)
111
kono
parents:
diff changeset
970 {
kono
parents:
diff changeset
971 if (replace_p)
kono
parents:
diff changeset
972 {
kono
parents:
diff changeset
973 SET_DEST (old_set) = ep->to_rtx;
kono
parents:
diff changeset
974 lra_update_insn_recog_data (insn);
kono
parents:
diff changeset
975 return;
kono
parents:
diff changeset
976 }
kono
parents:
diff changeset
977 offset -= (ep->offset - ep->previous_offset);
kono
parents:
diff changeset
978 src = plus_constant (Pmode, ep->to_rtx, offset);
kono
parents:
diff changeset
979
kono
parents:
diff changeset
980 /* First see if this insn remains valid when we
kono
parents:
diff changeset
981 make the change. If not, keep the INSN_CODE
kono
parents:
diff changeset
982 the same and let the constraint pass fit it
kono
parents:
diff changeset
983 up. */
kono
parents:
diff changeset
984 validate_change (insn, &SET_SRC (old_set), src, 1);
kono
parents:
diff changeset
985 validate_change (insn, &SET_DEST (old_set),
kono
parents:
diff changeset
986 ep->from_rtx, 1);
kono
parents:
diff changeset
987 if (! apply_change_group ())
kono
parents:
diff changeset
988 {
kono
parents:
diff changeset
989 SET_SRC (old_set) = src;
kono
parents:
diff changeset
990 SET_DEST (old_set) = ep->from_rtx;
kono
parents:
diff changeset
991 }
kono
parents:
diff changeset
992 lra_update_insn_recog_data (insn);
kono
parents:
diff changeset
993 /* Add offset note for future updates. */
kono
parents:
diff changeset
994 add_reg_note (insn, REG_EQUAL, copy_rtx (src));
kono
parents:
diff changeset
995 return;
kono
parents:
diff changeset
996 }
kono
parents:
diff changeset
997 }
kono
parents:
diff changeset
998 #endif
kono
parents:
diff changeset
999
kono
parents:
diff changeset
1000 /* This insn isn't serving a useful purpose. We delete it
kono
parents:
diff changeset
1001 when REPLACE is set. */
kono
parents:
diff changeset
1002 if (delete_p)
kono
parents:
diff changeset
1003 lra_delete_dead_insn (insn);
kono
parents:
diff changeset
1004 return;
kono
parents:
diff changeset
1005 }
kono
parents:
diff changeset
1006 }
kono
parents:
diff changeset
1007
kono
parents:
diff changeset
1008 /* We allow one special case which happens to work on all machines we
kono
parents:
diff changeset
1009 currently support: a single set with the source or a REG_EQUAL
kono
parents:
diff changeset
1010 note being a PLUS of an eliminable register and a constant. */
kono
parents:
diff changeset
1011 plus_src = plus_cst_src = 0;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1012 poly_int64 offset = 0;
111
kono
parents:
diff changeset
1013 if (old_set && REG_P (SET_DEST (old_set)))
kono
parents:
diff changeset
1014 {
kono
parents:
diff changeset
1015 if (GET_CODE (SET_SRC (old_set)) == PLUS)
kono
parents:
diff changeset
1016 plus_src = SET_SRC (old_set);
kono
parents:
diff changeset
1017 /* First see if the source is of the form (plus (...) CST). */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1018 if (plus_src && poly_int_rtx_p (XEXP (plus_src, 1), &offset))
111
kono
parents:
diff changeset
1019 plus_cst_src = plus_src;
kono
parents:
diff changeset
1020 /* Check that the first operand of the PLUS is a hard reg or
kono
parents:
diff changeset
1021 the lowpart subreg of one. */
kono
parents:
diff changeset
1022 if (plus_cst_src)
kono
parents:
diff changeset
1023 {
kono
parents:
diff changeset
1024 rtx reg = XEXP (plus_cst_src, 0);
kono
parents:
diff changeset
1025
kono
parents:
diff changeset
1026 if (GET_CODE (reg) == SUBREG && subreg_lowpart_p (reg))
kono
parents:
diff changeset
1027 reg = SUBREG_REG (reg);
kono
parents:
diff changeset
1028
kono
parents:
diff changeset
1029 if (!REG_P (reg) || REGNO (reg) >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
1030 plus_cst_src = 0;
kono
parents:
diff changeset
1031 }
kono
parents:
diff changeset
1032 }
kono
parents:
diff changeset
1033 if (plus_cst_src)
kono
parents:
diff changeset
1034 {
kono
parents:
diff changeset
1035 rtx reg = XEXP (plus_cst_src, 0);
kono
parents:
diff changeset
1036
kono
parents:
diff changeset
1037 if (GET_CODE (reg) == SUBREG)
kono
parents:
diff changeset
1038 reg = SUBREG_REG (reg);
kono
parents:
diff changeset
1039
kono
parents:
diff changeset
1040 if (REG_P (reg) && (ep = get_elimination (reg)) != NULL)
kono
parents:
diff changeset
1041 {
kono
parents:
diff changeset
1042 rtx to_rtx = replace_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
1043
kono
parents:
diff changeset
1044 if (! replace_p)
kono
parents:
diff changeset
1045 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1046 if (known_eq (update_sp_offset, 0))
111
kono
parents:
diff changeset
1047 offset += (ep->offset - ep->previous_offset);
kono
parents:
diff changeset
1048 if (ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
1049 {
kono
parents:
diff changeset
1050 if (first_p)
kono
parents:
diff changeset
1051 offset -= lra_get_insn_recog_data (insn)->sp_offset;
kono
parents:
diff changeset
1052 else
kono
parents:
diff changeset
1053 offset += update_sp_offset;
kono
parents:
diff changeset
1054 }
kono
parents:
diff changeset
1055 offset = trunc_int_for_mode (offset, GET_MODE (plus_cst_src));
kono
parents:
diff changeset
1056 }
kono
parents:
diff changeset
1057
kono
parents:
diff changeset
1058 if (GET_CODE (XEXP (plus_cst_src, 0)) == SUBREG)
kono
parents:
diff changeset
1059 to_rtx = gen_lowpart (GET_MODE (XEXP (plus_cst_src, 0)), to_rtx);
kono
parents:
diff changeset
1060 /* If we have a nonzero offset, and the source is already a
kono
parents:
diff changeset
1061 simple REG, the following transformation would increase
kono
parents:
diff changeset
1062 the cost of the insn by replacing a simple REG with (plus
kono
parents:
diff changeset
1063 (reg sp) CST). So try only when we already had a PLUS
kono
parents:
diff changeset
1064 before. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1065 if (known_eq (offset, 0) || plus_src)
111
kono
parents:
diff changeset
1066 {
kono
parents:
diff changeset
1067 rtx new_src = plus_constant (GET_MODE (to_rtx), to_rtx, offset);
kono
parents:
diff changeset
1068
kono
parents:
diff changeset
1069 old_set = single_set (insn);
kono
parents:
diff changeset
1070
kono
parents:
diff changeset
1071 /* First see if this insn remains valid when we make the
kono
parents:
diff changeset
1072 change. If not, try to replace the whole pattern
kono
parents:
diff changeset
1073 with a simple set (this may help if the original insn
kono
parents:
diff changeset
1074 was a PARALLEL that was only recognized as single_set
kono
parents:
diff changeset
1075 due to REG_UNUSED notes). If this isn't valid
kono
parents:
diff changeset
1076 either, keep the INSN_CODE the same and let the
kono
parents:
diff changeset
1077 constraint pass fix it up. */
kono
parents:
diff changeset
1078 if (! validate_change (insn, &SET_SRC (old_set), new_src, 0))
kono
parents:
diff changeset
1079 {
kono
parents:
diff changeset
1080 rtx new_pat = gen_rtx_SET (SET_DEST (old_set), new_src);
kono
parents:
diff changeset
1081
kono
parents:
diff changeset
1082 if (! validate_change (insn, &PATTERN (insn), new_pat, 0))
kono
parents:
diff changeset
1083 SET_SRC (old_set) = new_src;
kono
parents:
diff changeset
1084 }
kono
parents:
diff changeset
1085 lra_update_insn_recog_data (insn);
kono
parents:
diff changeset
1086 /* This can't have an effect on elimination offsets, so skip
kono
parents:
diff changeset
1087 right to the end. */
kono
parents:
diff changeset
1088 return;
kono
parents:
diff changeset
1089 }
kono
parents:
diff changeset
1090 }
kono
parents:
diff changeset
1091 }
kono
parents:
diff changeset
1092
kono
parents:
diff changeset
1093 /* Eliminate all eliminable registers occurring in operands that
kono
parents:
diff changeset
1094 can be handled by the constraint pass. */
kono
parents:
diff changeset
1095 id = lra_get_insn_recog_data (insn);
kono
parents:
diff changeset
1096 static_id = id->insn_static_data;
kono
parents:
diff changeset
1097 validate_p = false;
kono
parents:
diff changeset
1098 for (i = 0; i < static_id->n_operands; i++)
kono
parents:
diff changeset
1099 {
kono
parents:
diff changeset
1100 orig_operand[i] = *id->operand_loc[i];
kono
parents:
diff changeset
1101 substed_operand[i] = *id->operand_loc[i];
kono
parents:
diff changeset
1102
kono
parents:
diff changeset
1103 /* For an asm statement, every operand is eliminable. */
kono
parents:
diff changeset
1104 if (icode < 0 || insn_data[icode].operand[i].eliminable)
kono
parents:
diff changeset
1105 {
kono
parents:
diff changeset
1106 /* Check for setting a hard register that we know about. */
kono
parents:
diff changeset
1107 if (static_id->operand[i].type != OP_IN
kono
parents:
diff changeset
1108 && REG_P (orig_operand[i]))
kono
parents:
diff changeset
1109 {
kono
parents:
diff changeset
1110 /* If we are assigning to a hard register that can be
kono
parents:
diff changeset
1111 eliminated, it must be as part of a PARALLEL, since
kono
parents:
diff changeset
1112 the code above handles single SETs. This reg can not
kono
parents:
diff changeset
1113 be longer eliminated -- it is forced by
kono
parents:
diff changeset
1114 mark_not_eliminable. */
kono
parents:
diff changeset
1115 for (ep = reg_eliminate;
kono
parents:
diff changeset
1116 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
kono
parents:
diff changeset
1117 ep++)
kono
parents:
diff changeset
1118 lra_assert (ep->from_rtx != orig_operand[i]
kono
parents:
diff changeset
1119 || ! ep->can_eliminate);
kono
parents:
diff changeset
1120 }
kono
parents:
diff changeset
1121
kono
parents:
diff changeset
1122 /* Companion to the above plus substitution, we can allow
kono
parents:
diff changeset
1123 invariants as the source of a plain move. */
kono
parents:
diff changeset
1124 substed_operand[i]
kono
parents:
diff changeset
1125 = lra_eliminate_regs_1 (insn, *id->operand_loc[i], VOIDmode,
kono
parents:
diff changeset
1126 replace_p, ! replace_p && ! first_p,
kono
parents:
diff changeset
1127 update_sp_offset, first_p);
kono
parents:
diff changeset
1128 if (substed_operand[i] != orig_operand[i])
kono
parents:
diff changeset
1129 validate_p = true;
kono
parents:
diff changeset
1130 }
kono
parents:
diff changeset
1131 }
kono
parents:
diff changeset
1132
kono
parents:
diff changeset
1133 if (! validate_p)
kono
parents:
diff changeset
1134 return;
kono
parents:
diff changeset
1135
kono
parents:
diff changeset
1136 /* Substitute the operands; the new values are in the substed_operand
kono
parents:
diff changeset
1137 array. */
kono
parents:
diff changeset
1138 for (i = 0; i < static_id->n_operands; i++)
kono
parents:
diff changeset
1139 *id->operand_loc[i] = substed_operand[i];
kono
parents:
diff changeset
1140 for (i = 0; i < static_id->n_dups; i++)
kono
parents:
diff changeset
1141 *id->dup_loc[i] = substed_operand[(int) static_id->dup_num[i]];
kono
parents:
diff changeset
1142
kono
parents:
diff changeset
1143 /* If we had a move insn but now we don't, re-recognize it.
kono
parents:
diff changeset
1144 This will cause spurious re-recognition if the old move had a
kono
parents:
diff changeset
1145 PARALLEL since the new one still will, but we can't call
kono
parents:
diff changeset
1146 single_set without having put new body into the insn and the
kono
parents:
diff changeset
1147 re-recognition won't hurt in this rare case. */
kono
parents:
diff changeset
1148 id = lra_update_insn_recog_data (insn);
kono
parents:
diff changeset
1149 static_id = id->insn_static_data;
kono
parents:
diff changeset
1150 }
kono
parents:
diff changeset
1151
kono
parents:
diff changeset
1152 /* Spill pseudos which are assigned to hard registers in SET. Add
kono
parents:
diff changeset
1153 affected insns for processing in the subsequent constraint
kono
parents:
diff changeset
1154 pass. */
kono
parents:
diff changeset
1155 static void
kono
parents:
diff changeset
1156 spill_pseudos (HARD_REG_SET set)
kono
parents:
diff changeset
1157 {
kono
parents:
diff changeset
1158 int i;
kono
parents:
diff changeset
1159 bitmap_head to_process;
kono
parents:
diff changeset
1160 rtx_insn *insn;
kono
parents:
diff changeset
1161
kono
parents:
diff changeset
1162 if (hard_reg_set_empty_p (set))
kono
parents:
diff changeset
1163 return;
kono
parents:
diff changeset
1164 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1165 {
kono
parents:
diff changeset
1166 fprintf (lra_dump_file, " Spilling non-eliminable hard regs:");
kono
parents:
diff changeset
1167 for (i = 0; i < FIRST_PSEUDO_REGISTER; i++)
kono
parents:
diff changeset
1168 if (TEST_HARD_REG_BIT (set, i))
kono
parents:
diff changeset
1169 fprintf (lra_dump_file, " %d", i);
kono
parents:
diff changeset
1170 fprintf (lra_dump_file, "\n");
kono
parents:
diff changeset
1171 }
kono
parents:
diff changeset
1172 bitmap_initialize (&to_process, &reg_obstack);
kono
parents:
diff changeset
1173 for (i = FIRST_PSEUDO_REGISTER; i < max_reg_num (); i++)
kono
parents:
diff changeset
1174 if (lra_reg_info[i].nrefs != 0 && reg_renumber[i] >= 0
kono
parents:
diff changeset
1175 && overlaps_hard_reg_set_p (set,
kono
parents:
diff changeset
1176 PSEUDO_REGNO_MODE (i), reg_renumber[i]))
kono
parents:
diff changeset
1177 {
kono
parents:
diff changeset
1178 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1179 fprintf (lra_dump_file, " Spilling r%d(%d)\n",
kono
parents:
diff changeset
1180 i, reg_renumber[i]);
kono
parents:
diff changeset
1181 reg_renumber[i] = -1;
kono
parents:
diff changeset
1182 bitmap_ior_into (&to_process, &lra_reg_info[i].insn_bitmap);
kono
parents:
diff changeset
1183 }
kono
parents:
diff changeset
1184 IOR_HARD_REG_SET (lra_no_alloc_regs, set);
kono
parents:
diff changeset
1185 for (insn = get_insns (); insn != NULL_RTX; insn = NEXT_INSN (insn))
kono
parents:
diff changeset
1186 if (bitmap_bit_p (&to_process, INSN_UID (insn)))
kono
parents:
diff changeset
1187 {
kono
parents:
diff changeset
1188 lra_push_insn (insn);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1189 lra_set_used_insn_alternative (insn, LRA_UNKNOWN_ALT);
111
kono
parents:
diff changeset
1190 }
kono
parents:
diff changeset
1191 bitmap_clear (&to_process);
kono
parents:
diff changeset
1192 }
kono
parents:
diff changeset
1193
kono
parents:
diff changeset
1194 /* Update all offsets and possibility for elimination on eliminable
kono
parents:
diff changeset
1195 registers. Spill pseudos assigned to registers which are
kono
parents:
diff changeset
1196 uneliminable, update LRA_NO_ALLOC_REGS and ELIMINABLE_REG_SET. Add
kono
parents:
diff changeset
1197 insns to INSNS_WITH_CHANGED_OFFSETS containing eliminable hard
kono
parents:
diff changeset
1198 registers whose offsets should be changed. Return true if any
kono
parents:
diff changeset
1199 elimination offset changed. */
kono
parents:
diff changeset
1200 static bool
kono
parents:
diff changeset
1201 update_reg_eliminate (bitmap insns_with_changed_offsets)
kono
parents:
diff changeset
1202 {
kono
parents:
diff changeset
1203 bool prev, result;
kono
parents:
diff changeset
1204 struct lra_elim_table *ep, *ep1;
kono
parents:
diff changeset
1205 HARD_REG_SET temp_hard_reg_set;
kono
parents:
diff changeset
1206
kono
parents:
diff changeset
1207 targetm.compute_frame_layout ();
kono
parents:
diff changeset
1208
kono
parents:
diff changeset
1209 /* Clear self elimination offsets. */
kono
parents:
diff changeset
1210 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1211 self_elim_offsets[ep->from] = 0;
kono
parents:
diff changeset
1212 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1213 {
kono
parents:
diff changeset
1214 /* If it is a currently used elimination: update the previous
kono
parents:
diff changeset
1215 offset. */
kono
parents:
diff changeset
1216 if (elimination_map[ep->from] == ep)
kono
parents:
diff changeset
1217 ep->previous_offset = ep->offset;
kono
parents:
diff changeset
1218
kono
parents:
diff changeset
1219 prev = ep->prev_can_eliminate;
kono
parents:
diff changeset
1220 setup_can_eliminate (ep, targetm.can_eliminate (ep->from, ep->to));
kono
parents:
diff changeset
1221 if (ep->can_eliminate && ! prev)
kono
parents:
diff changeset
1222 {
kono
parents:
diff changeset
1223 /* It is possible that not eliminable register becomes
kono
parents:
diff changeset
1224 eliminable because we took other reasons into account to
kono
parents:
diff changeset
1225 set up eliminable regs in the initial set up. Just
kono
parents:
diff changeset
1226 ignore new eliminable registers. */
kono
parents:
diff changeset
1227 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
1228 continue;
kono
parents:
diff changeset
1229 }
kono
parents:
diff changeset
1230 if (ep->can_eliminate != prev && elimination_map[ep->from] == ep)
kono
parents:
diff changeset
1231 {
kono
parents:
diff changeset
1232 /* We cannot use this elimination anymore -- find another
kono
parents:
diff changeset
1233 one. */
kono
parents:
diff changeset
1234 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1235 fprintf (lra_dump_file,
kono
parents:
diff changeset
1236 " Elimination %d to %d is not possible anymore\n",
kono
parents:
diff changeset
1237 ep->from, ep->to);
kono
parents:
diff changeset
1238 /* If after processing RTL we decides that SP can be used as
kono
parents:
diff changeset
1239 a result of elimination, it can not be changed. */
kono
parents:
diff changeset
1240 gcc_assert ((ep->to_rtx != stack_pointer_rtx)
kono
parents:
diff changeset
1241 || (ep->from < FIRST_PSEUDO_REGISTER
kono
parents:
diff changeset
1242 && fixed_regs [ep->from]));
kono
parents:
diff changeset
1243 /* Mark that is not eliminable anymore. */
kono
parents:
diff changeset
1244 elimination_map[ep->from] = NULL;
kono
parents:
diff changeset
1245 for (ep1 = ep + 1; ep1 < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep1++)
kono
parents:
diff changeset
1246 if (ep1->can_eliminate && ep1->from == ep->from)
kono
parents:
diff changeset
1247 break;
kono
parents:
diff changeset
1248 if (ep1 < &reg_eliminate[NUM_ELIMINABLE_REGS])
kono
parents:
diff changeset
1249 {
kono
parents:
diff changeset
1250 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1251 fprintf (lra_dump_file, " Using elimination %d to %d now\n",
kono
parents:
diff changeset
1252 ep1->from, ep1->to);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1253 lra_assert (known_eq (ep1->previous_offset, 0));
111
kono
parents:
diff changeset
1254 ep1->previous_offset = ep->offset;
kono
parents:
diff changeset
1255 }
kono
parents:
diff changeset
1256 else
kono
parents:
diff changeset
1257 {
kono
parents:
diff changeset
1258 /* There is no elimination anymore just use the hard
kono
parents:
diff changeset
1259 register `from' itself. Setup self elimination
kono
parents:
diff changeset
1260 offset to restore the original offset values. */
kono
parents:
diff changeset
1261 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1262 fprintf (lra_dump_file, " %d is not eliminable at all\n",
kono
parents:
diff changeset
1263 ep->from);
kono
parents:
diff changeset
1264 self_elim_offsets[ep->from] = -ep->offset;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1265 if (maybe_ne (ep->offset, 0))
111
kono
parents:
diff changeset
1266 bitmap_ior_into (insns_with_changed_offsets,
kono
parents:
diff changeset
1267 &lra_reg_info[ep->from].insn_bitmap);
kono
parents:
diff changeset
1268 }
kono
parents:
diff changeset
1269 }
kono
parents:
diff changeset
1270
kono
parents:
diff changeset
1271 INITIAL_ELIMINATION_OFFSET (ep->from, ep->to, ep->offset);
kono
parents:
diff changeset
1272 }
kono
parents:
diff changeset
1273 setup_elimination_map ();
kono
parents:
diff changeset
1274 result = false;
kono
parents:
diff changeset
1275 CLEAR_HARD_REG_SET (temp_hard_reg_set);
kono
parents:
diff changeset
1276 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1277 if (elimination_map[ep->from] == NULL)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1278 add_to_hard_reg_set (&temp_hard_reg_set, Pmode, ep->from);
111
kono
parents:
diff changeset
1279 else if (elimination_map[ep->from] == ep)
kono
parents:
diff changeset
1280 {
kono
parents:
diff changeset
1281 /* Prevent the hard register into which we eliminate from
kono
parents:
diff changeset
1282 the usage for pseudos. */
kono
parents:
diff changeset
1283 if (ep->from != ep->to)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1284 add_to_hard_reg_set (&temp_hard_reg_set, Pmode, ep->to);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1285 if (maybe_ne (ep->previous_offset, ep->offset))
111
kono
parents:
diff changeset
1286 {
kono
parents:
diff changeset
1287 bitmap_ior_into (insns_with_changed_offsets,
kono
parents:
diff changeset
1288 &lra_reg_info[ep->from].insn_bitmap);
kono
parents:
diff changeset
1289
kono
parents:
diff changeset
1290 /* Update offset when the eliminate offset have been
kono
parents:
diff changeset
1291 changed. */
kono
parents:
diff changeset
1292 lra_update_reg_val_offset (lra_reg_info[ep->from].val,
kono
parents:
diff changeset
1293 ep->offset - ep->previous_offset);
kono
parents:
diff changeset
1294 result = true;
kono
parents:
diff changeset
1295 }
kono
parents:
diff changeset
1296 }
kono
parents:
diff changeset
1297 IOR_HARD_REG_SET (lra_no_alloc_regs, temp_hard_reg_set);
kono
parents:
diff changeset
1298 AND_COMPL_HARD_REG_SET (eliminable_regset, temp_hard_reg_set);
kono
parents:
diff changeset
1299 spill_pseudos (temp_hard_reg_set);
kono
parents:
diff changeset
1300 return result;
kono
parents:
diff changeset
1301 }
kono
parents:
diff changeset
1302
kono
parents:
diff changeset
1303 /* Initialize the table of hard registers to eliminate.
kono
parents:
diff changeset
1304 Pre-condition: global flag frame_pointer_needed has been set before
kono
parents:
diff changeset
1305 calling this function. */
kono
parents:
diff changeset
1306 static void
kono
parents:
diff changeset
1307 init_elim_table (void)
kono
parents:
diff changeset
1308 {
kono
parents:
diff changeset
1309 struct lra_elim_table *ep;
kono
parents:
diff changeset
1310 bool value_p;
kono
parents:
diff changeset
1311 const struct elim_table_1 *ep1;
kono
parents:
diff changeset
1312
kono
parents:
diff changeset
1313 if (!reg_eliminate)
kono
parents:
diff changeset
1314 reg_eliminate = XCNEWVEC (struct lra_elim_table, NUM_ELIMINABLE_REGS);
kono
parents:
diff changeset
1315
kono
parents:
diff changeset
1316 memset (self_elim_offsets, 0, sizeof (self_elim_offsets));
kono
parents:
diff changeset
1317 /* Initiate member values which will be never changed. */
kono
parents:
diff changeset
1318 self_elim_table.can_eliminate = self_elim_table.prev_can_eliminate = true;
kono
parents:
diff changeset
1319 self_elim_table.previous_offset = 0;
kono
parents:
diff changeset
1320
kono
parents:
diff changeset
1321 for (ep = reg_eliminate, ep1 = reg_eliminate_1;
kono
parents:
diff changeset
1322 ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++, ep1++)
kono
parents:
diff changeset
1323 {
kono
parents:
diff changeset
1324 ep->offset = ep->previous_offset = 0;
kono
parents:
diff changeset
1325 ep->from = ep1->from;
kono
parents:
diff changeset
1326 ep->to = ep1->to;
kono
parents:
diff changeset
1327 value_p = (targetm.can_eliminate (ep->from, ep->to)
kono
parents:
diff changeset
1328 && ! (ep->to == STACK_POINTER_REGNUM
kono
parents:
diff changeset
1329 && frame_pointer_needed
kono
parents:
diff changeset
1330 && (! SUPPORTS_STACK_ALIGNMENT
kono
parents:
diff changeset
1331 || ! stack_realign_fp)));
kono
parents:
diff changeset
1332 setup_can_eliminate (ep, value_p);
kono
parents:
diff changeset
1333 }
kono
parents:
diff changeset
1334
kono
parents:
diff changeset
1335 /* Build the FROM and TO REG rtx's. Note that code in gen_rtx_REG
kono
parents:
diff changeset
1336 will cause, e.g., gen_rtx_REG (Pmode, STACK_POINTER_REGNUM) to
kono
parents:
diff changeset
1337 equal stack_pointer_rtx. We depend on this. Threfore we switch
kono
parents:
diff changeset
1338 off that we are in LRA temporarily. */
kono
parents:
diff changeset
1339 lra_in_progress = 0;
kono
parents:
diff changeset
1340 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1341 {
kono
parents:
diff changeset
1342 ep->from_rtx = gen_rtx_REG (Pmode, ep->from);
kono
parents:
diff changeset
1343 ep->to_rtx = gen_rtx_REG (Pmode, ep->to);
kono
parents:
diff changeset
1344 eliminable_reg_rtx[ep->from] = ep->from_rtx;
kono
parents:
diff changeset
1345 }
kono
parents:
diff changeset
1346 lra_in_progress = 1;
kono
parents:
diff changeset
1347 }
kono
parents:
diff changeset
1348
kono
parents:
diff changeset
1349 /* Function for initialization of elimination once per function. It
kono
parents:
diff changeset
1350 sets up sp offset for each insn. */
kono
parents:
diff changeset
1351 static void
kono
parents:
diff changeset
1352 init_elimination (void)
kono
parents:
diff changeset
1353 {
kono
parents:
diff changeset
1354 bool stop_to_sp_elimination_p;
kono
parents:
diff changeset
1355 basic_block bb;
kono
parents:
diff changeset
1356 rtx_insn *insn;
kono
parents:
diff changeset
1357 struct lra_elim_table *ep;
kono
parents:
diff changeset
1358
kono
parents:
diff changeset
1359 init_elim_table ();
kono
parents:
diff changeset
1360 FOR_EACH_BB_FN (bb, cfun)
kono
parents:
diff changeset
1361 {
kono
parents:
diff changeset
1362 curr_sp_change = 0;
kono
parents:
diff changeset
1363 stop_to_sp_elimination_p = false;
kono
parents:
diff changeset
1364 FOR_BB_INSNS (bb, insn)
kono
parents:
diff changeset
1365 if (INSN_P (insn))
kono
parents:
diff changeset
1366 {
kono
parents:
diff changeset
1367 lra_get_insn_recog_data (insn)->sp_offset = curr_sp_change;
kono
parents:
diff changeset
1368 if (NONDEBUG_INSN_P (insn))
kono
parents:
diff changeset
1369 {
kono
parents:
diff changeset
1370 mark_not_eliminable (PATTERN (insn), VOIDmode);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1371 if (maybe_ne (curr_sp_change, 0)
111
kono
parents:
diff changeset
1372 && find_reg_note (insn, REG_LABEL_OPERAND, NULL_RTX))
kono
parents:
diff changeset
1373 stop_to_sp_elimination_p = true;
kono
parents:
diff changeset
1374 }
kono
parents:
diff changeset
1375 }
kono
parents:
diff changeset
1376 if (! frame_pointer_needed
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1377 && (maybe_ne (curr_sp_change, 0) || stop_to_sp_elimination_p)
111
kono
parents:
diff changeset
1378 && bb->succs && bb->succs->length () != 0)
kono
parents:
diff changeset
1379 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1380 if (ep->to == STACK_POINTER_REGNUM)
kono
parents:
diff changeset
1381 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
1382 }
kono
parents:
diff changeset
1383 setup_elimination_map ();
kono
parents:
diff changeset
1384 }
kono
parents:
diff changeset
1385
kono
parents:
diff changeset
1386 /* Eliminate hard reg given by its location LOC. */
kono
parents:
diff changeset
1387 void
kono
parents:
diff changeset
1388 lra_eliminate_reg_if_possible (rtx *loc)
kono
parents:
diff changeset
1389 {
kono
parents:
diff changeset
1390 int regno;
kono
parents:
diff changeset
1391 struct lra_elim_table *ep;
kono
parents:
diff changeset
1392
kono
parents:
diff changeset
1393 lra_assert (REG_P (*loc));
kono
parents:
diff changeset
1394 if ((regno = REGNO (*loc)) >= FIRST_PSEUDO_REGISTER
kono
parents:
diff changeset
1395 || ! TEST_HARD_REG_BIT (lra_no_alloc_regs, regno))
kono
parents:
diff changeset
1396 return;
kono
parents:
diff changeset
1397 if ((ep = get_elimination (*loc)) != NULL)
kono
parents:
diff changeset
1398 *loc = ep->to_rtx;
kono
parents:
diff changeset
1399 }
kono
parents:
diff changeset
1400
kono
parents:
diff changeset
1401 /* Do (final if FINAL_P or first if FIRST_P) elimination in INSN. Add
kono
parents:
diff changeset
1402 the insn for subsequent processing in the constraint pass, update
kono
parents:
diff changeset
1403 the insn info. */
kono
parents:
diff changeset
1404 static void
kono
parents:
diff changeset
1405 process_insn_for_elimination (rtx_insn *insn, bool final_p, bool first_p)
kono
parents:
diff changeset
1406 {
kono
parents:
diff changeset
1407 eliminate_regs_in_insn (insn, final_p, first_p, 0);
kono
parents:
diff changeset
1408 if (! final_p)
kono
parents:
diff changeset
1409 {
kono
parents:
diff changeset
1410 /* Check that insn changed its code. This is a case when a move
kono
parents:
diff changeset
1411 insn becomes an add insn and we do not want to process the
kono
parents:
diff changeset
1412 insn as a move anymore. */
kono
parents:
diff changeset
1413 int icode = recog (PATTERN (insn), insn, 0);
kono
parents:
diff changeset
1414
kono
parents:
diff changeset
1415 if (icode >= 0 && icode != INSN_CODE (insn))
kono
parents:
diff changeset
1416 {
kono
parents:
diff changeset
1417 INSN_CODE (insn) = icode;
kono
parents:
diff changeset
1418 lra_update_insn_recog_data (insn);
kono
parents:
diff changeset
1419 }
kono
parents:
diff changeset
1420 lra_update_insn_regno_info (insn);
kono
parents:
diff changeset
1421 lra_push_insn (insn);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1422 lra_set_used_insn_alternative (insn, LRA_UNKNOWN_ALT);
111
kono
parents:
diff changeset
1423 }
kono
parents:
diff changeset
1424 }
kono
parents:
diff changeset
1425
kono
parents:
diff changeset
1426 /* Entry function to do final elimination if FINAL_P or to update
kono
parents:
diff changeset
1427 elimination register offsets (FIRST_P if we are doing it the first
kono
parents:
diff changeset
1428 time). */
kono
parents:
diff changeset
1429 void
kono
parents:
diff changeset
1430 lra_eliminate (bool final_p, bool first_p)
kono
parents:
diff changeset
1431 {
kono
parents:
diff changeset
1432 unsigned int uid;
kono
parents:
diff changeset
1433 bitmap_head insns_with_changed_offsets;
kono
parents:
diff changeset
1434 bitmap_iterator bi;
kono
parents:
diff changeset
1435 struct lra_elim_table *ep;
kono
parents:
diff changeset
1436
kono
parents:
diff changeset
1437 gcc_assert (! final_p || ! first_p);
kono
parents:
diff changeset
1438
kono
parents:
diff changeset
1439 timevar_push (TV_LRA_ELIMINATE);
kono
parents:
diff changeset
1440
kono
parents:
diff changeset
1441 if (first_p)
kono
parents:
diff changeset
1442 init_elimination ();
kono
parents:
diff changeset
1443
kono
parents:
diff changeset
1444 bitmap_initialize (&insns_with_changed_offsets, &reg_obstack);
kono
parents:
diff changeset
1445 if (final_p)
kono
parents:
diff changeset
1446 {
kono
parents:
diff changeset
1447 if (flag_checking)
kono
parents:
diff changeset
1448 {
kono
parents:
diff changeset
1449 update_reg_eliminate (&insns_with_changed_offsets);
kono
parents:
diff changeset
1450 gcc_assert (bitmap_empty_p (&insns_with_changed_offsets));
kono
parents:
diff changeset
1451 }
kono
parents:
diff changeset
1452 /* We change eliminable hard registers in insns so we should do
kono
parents:
diff changeset
1453 this for all insns containing any eliminable hard
kono
parents:
diff changeset
1454 register. */
kono
parents:
diff changeset
1455 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1456 if (elimination_map[ep->from] != NULL)
kono
parents:
diff changeset
1457 bitmap_ior_into (&insns_with_changed_offsets,
kono
parents:
diff changeset
1458 &lra_reg_info[ep->from].insn_bitmap);
kono
parents:
diff changeset
1459 }
kono
parents:
diff changeset
1460 else if (! update_reg_eliminate (&insns_with_changed_offsets))
kono
parents:
diff changeset
1461 goto lra_eliminate_done;
kono
parents:
diff changeset
1462 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1463 {
kono
parents:
diff changeset
1464 fprintf (lra_dump_file, "New elimination table:\n");
kono
parents:
diff changeset
1465 print_elim_table (lra_dump_file);
kono
parents:
diff changeset
1466 }
kono
parents:
diff changeset
1467 EXECUTE_IF_SET_IN_BITMAP (&insns_with_changed_offsets, 0, uid, bi)
kono
parents:
diff changeset
1468 /* A dead insn can be deleted in process_insn_for_elimination. */
kono
parents:
diff changeset
1469 if (lra_insn_recog_data[uid] != NULL)
kono
parents:
diff changeset
1470 process_insn_for_elimination (lra_insn_recog_data[uid]->insn,
kono
parents:
diff changeset
1471 final_p, first_p);
kono
parents:
diff changeset
1472 bitmap_clear (&insns_with_changed_offsets);
kono
parents:
diff changeset
1473
kono
parents:
diff changeset
1474 lra_eliminate_done:
kono
parents:
diff changeset
1475 timevar_pop (TV_LRA_ELIMINATE);
kono
parents:
diff changeset
1476 }