annotate gcc/lra-eliminations.c @ 158:494b0b89df80 default tip

...
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Mon, 25 May 2020 18:13:55 +0900
parents 1830386684a0
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 /* Code for RTL register eliminations.
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
2 Copyright (C) 2010-2020 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. */
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
74 class lra_elim_table
111
kono
parents:
diff changeset
75 {
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
76 public:
111
kono
parents:
diff changeset
77 /* Hard register number to be eliminated. */
kono
parents:
diff changeset
78 int from;
kono
parents:
diff changeset
79 /* Hard register number used as replacement. */
kono
parents:
diff changeset
80 int to;
kono
parents:
diff changeset
81 /* Difference between values of the two hard registers above on
kono
parents:
diff changeset
82 previous iteration. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
83 poly_int64 previous_offset;
111
kono
parents:
diff changeset
84 /* Difference between the values on the current iteration. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
85 poly_int64 offset;
111
kono
parents:
diff changeset
86 /* Nonzero if this elimination can be done. */
kono
parents:
diff changeset
87 bool can_eliminate;
kono
parents:
diff changeset
88 /* CAN_ELIMINATE since the last check. */
kono
parents:
diff changeset
89 bool prev_can_eliminate;
kono
parents:
diff changeset
90 /* REG rtx for the register to be eliminated. We cannot simply
kono
parents:
diff changeset
91 compare the number since we might then spuriously replace a hard
kono
parents:
diff changeset
92 register corresponding to a pseudo assigned to the reg to be
kono
parents:
diff changeset
93 eliminated. */
kono
parents:
diff changeset
94 rtx from_rtx;
kono
parents:
diff changeset
95 /* REG rtx for the replacement. */
kono
parents:
diff changeset
96 rtx to_rtx;
kono
parents:
diff changeset
97 };
kono
parents:
diff changeset
98
kono
parents:
diff changeset
99 /* The elimination table. Each array entry describes one possible way
kono
parents:
diff changeset
100 of eliminating a register in favor of another. If there is more
kono
parents:
diff changeset
101 than one way of eliminating a particular register, the most
kono
parents:
diff changeset
102 preferred should be specified first. */
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
103 static class lra_elim_table *reg_eliminate = 0;
111
kono
parents:
diff changeset
104
kono
parents:
diff changeset
105 /* This is an intermediate structure to initialize the table. It has
kono
parents:
diff changeset
106 exactly the members provided by ELIMINABLE_REGS. */
kono
parents:
diff changeset
107 static const struct elim_table_1
kono
parents:
diff changeset
108 {
kono
parents:
diff changeset
109 const int from;
kono
parents:
diff changeset
110 const int to;
kono
parents:
diff changeset
111 } reg_eliminate_1[] =
kono
parents:
diff changeset
112
kono
parents:
diff changeset
113 ELIMINABLE_REGS;
kono
parents:
diff changeset
114
kono
parents:
diff changeset
115 #define NUM_ELIMINABLE_REGS ARRAY_SIZE (reg_eliminate_1)
kono
parents:
diff changeset
116
kono
parents:
diff changeset
117 /* Print info about elimination table to file F. */
kono
parents:
diff changeset
118 static void
kono
parents:
diff changeset
119 print_elim_table (FILE *f)
kono
parents:
diff changeset
120 {
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
121 class lra_elim_table *ep;
111
kono
parents:
diff changeset
122
kono
parents:
diff changeset
123 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
124 {
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
125 fprintf (f, "%s eliminate %d to %d (offset=",
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
126 ep->can_eliminate ? "Can" : "Can't", ep->from, ep->to);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
127 print_dec (ep->offset, f);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
128 fprintf (f, ", prev_offset=");
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
129 print_dec (ep->previous_offset, f);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
130 fprintf (f, ")\n");
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
131 }
111
kono
parents:
diff changeset
132 }
kono
parents:
diff changeset
133
kono
parents:
diff changeset
134 /* Print info about elimination table to stderr. */
kono
parents:
diff changeset
135 void
kono
parents:
diff changeset
136 lra_debug_elim_table (void)
kono
parents:
diff changeset
137 {
kono
parents:
diff changeset
138 print_elim_table (stderr);
kono
parents:
diff changeset
139 }
kono
parents:
diff changeset
140
kono
parents:
diff changeset
141 /* Setup possibility of elimination in elimination table element EP to
kono
parents:
diff changeset
142 VALUE. Setup FRAME_POINTER_NEEDED if elimination from frame
kono
parents:
diff changeset
143 pointer to stack pointer is not possible anymore. */
kono
parents:
diff changeset
144 static void
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
145 setup_can_eliminate (class lra_elim_table *ep, bool value)
111
kono
parents:
diff changeset
146 {
kono
parents:
diff changeset
147 ep->can_eliminate = ep->prev_can_eliminate = value;
kono
parents:
diff changeset
148 if (! value
kono
parents:
diff changeset
149 && ep->from == FRAME_POINTER_REGNUM && ep->to == STACK_POINTER_REGNUM)
kono
parents:
diff changeset
150 frame_pointer_needed = 1;
kono
parents:
diff changeset
151 if (!frame_pointer_needed)
kono
parents:
diff changeset
152 REGNO_POINTER_ALIGN (HARD_FRAME_POINTER_REGNUM) = 0;
kono
parents:
diff changeset
153 }
kono
parents:
diff changeset
154
kono
parents:
diff changeset
155 /* Map: eliminable "from" register -> its current elimination,
kono
parents:
diff changeset
156 or NULL if none. The elimination table may contain more than
kono
parents:
diff changeset
157 one elimination for the same hard register, but this map specifies
kono
parents:
diff changeset
158 the one that we are currently using. */
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
159 static class lra_elim_table *elimination_map[FIRST_PSEUDO_REGISTER];
111
kono
parents:
diff changeset
160
kono
parents:
diff changeset
161 /* When an eliminable hard register becomes not eliminable, we use the
kono
parents:
diff changeset
162 following special structure to restore original offsets for the
kono
parents:
diff changeset
163 register. */
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
164 static class lra_elim_table self_elim_table;
111
kono
parents:
diff changeset
165
kono
parents:
diff changeset
166 /* Offsets should be used to restore original offsets for eliminable
kono
parents:
diff changeset
167 hard register which just became not eliminable. Zero,
kono
parents:
diff changeset
168 otherwise. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
169 static poly_int64_pod self_elim_offsets[FIRST_PSEUDO_REGISTER];
111
kono
parents:
diff changeset
170
kono
parents:
diff changeset
171 /* Map: hard regno -> RTL presentation. RTL presentations of all
kono
parents:
diff changeset
172 potentially eliminable hard registers are stored in the map. */
kono
parents:
diff changeset
173 static rtx eliminable_reg_rtx[FIRST_PSEUDO_REGISTER];
kono
parents:
diff changeset
174
kono
parents:
diff changeset
175 /* Set up ELIMINATION_MAP of the currently used eliminations. */
kono
parents:
diff changeset
176 static void
kono
parents:
diff changeset
177 setup_elimination_map (void)
kono
parents:
diff changeset
178 {
kono
parents:
diff changeset
179 int i;
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
180 class lra_elim_table *ep;
111
kono
parents:
diff changeset
181
kono
parents:
diff changeset
182 for (i = 0; i < FIRST_PSEUDO_REGISTER; i++)
kono
parents:
diff changeset
183 elimination_map[i] = NULL;
kono
parents:
diff changeset
184 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
185 if (ep->can_eliminate && elimination_map[ep->from] == NULL)
kono
parents:
diff changeset
186 elimination_map[ep->from] = ep;
kono
parents:
diff changeset
187 }
kono
parents:
diff changeset
188
kono
parents:
diff changeset
189
kono
parents:
diff changeset
190
kono
parents:
diff changeset
191 /* Compute the sum of X and Y, making canonicalizations assumed in an
kono
parents:
diff changeset
192 address, namely: sum constant integers, surround the sum of two
kono
parents:
diff changeset
193 constants with a CONST, put the constant as the second operand, and
kono
parents:
diff changeset
194 group the constant on the outermost sum.
kono
parents:
diff changeset
195
kono
parents:
diff changeset
196 This routine assumes both inputs are already in canonical form. */
kono
parents:
diff changeset
197 static rtx
kono
parents:
diff changeset
198 form_sum (rtx x, rtx y)
kono
parents:
diff changeset
199 {
kono
parents:
diff changeset
200 machine_mode mode = GET_MODE (x);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
201 poly_int64 offset;
111
kono
parents:
diff changeset
202
kono
parents:
diff changeset
203 if (mode == VOIDmode)
kono
parents:
diff changeset
204 mode = GET_MODE (y);
kono
parents:
diff changeset
205
kono
parents:
diff changeset
206 if (mode == VOIDmode)
kono
parents:
diff changeset
207 mode = Pmode;
kono
parents:
diff changeset
208
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
209 if (poly_int_rtx_p (x, &offset))
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
210 return plus_constant (mode, y, offset);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
211 else if (poly_int_rtx_p (y, &offset))
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
212 return plus_constant (mode, x, offset);
111
kono
parents:
diff changeset
213 else if (CONSTANT_P (x))
kono
parents:
diff changeset
214 std::swap (x, y);
kono
parents:
diff changeset
215
kono
parents:
diff changeset
216 if (GET_CODE (x) == PLUS && CONSTANT_P (XEXP (x, 1)))
kono
parents:
diff changeset
217 return form_sum (XEXP (x, 0), form_sum (XEXP (x, 1), y));
kono
parents:
diff changeset
218
kono
parents:
diff changeset
219 /* Note that if the operands of Y are specified in the opposite
kono
parents:
diff changeset
220 order in the recursive calls below, infinite recursion will
kono
parents:
diff changeset
221 occur. */
kono
parents:
diff changeset
222 if (GET_CODE (y) == PLUS && CONSTANT_P (XEXP (y, 1)))
kono
parents:
diff changeset
223 return form_sum (form_sum (x, XEXP (y, 0)), XEXP (y, 1));
kono
parents:
diff changeset
224
kono
parents:
diff changeset
225 /* If both constant, encapsulate sum. Otherwise, just form sum. A
kono
parents:
diff changeset
226 constant will have been placed second. */
kono
parents:
diff changeset
227 if (CONSTANT_P (x) && CONSTANT_P (y))
kono
parents:
diff changeset
228 {
kono
parents:
diff changeset
229 if (GET_CODE (x) == CONST)
kono
parents:
diff changeset
230 x = XEXP (x, 0);
kono
parents:
diff changeset
231 if (GET_CODE (y) == CONST)
kono
parents:
diff changeset
232 y = XEXP (y, 0);
kono
parents:
diff changeset
233
kono
parents:
diff changeset
234 return gen_rtx_CONST (VOIDmode, gen_rtx_PLUS (mode, x, y));
kono
parents:
diff changeset
235 }
kono
parents:
diff changeset
236
kono
parents:
diff changeset
237 return gen_rtx_PLUS (mode, x, y);
kono
parents:
diff changeset
238 }
kono
parents:
diff changeset
239
kono
parents:
diff changeset
240 /* Return the current substitution hard register of the elimination of
kono
parents:
diff changeset
241 HARD_REGNO. If HARD_REGNO is not eliminable, return itself. */
kono
parents:
diff changeset
242 int
kono
parents:
diff changeset
243 lra_get_elimination_hard_regno (int hard_regno)
kono
parents:
diff changeset
244 {
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
245 class lra_elim_table *ep;
111
kono
parents:
diff changeset
246
kono
parents:
diff changeset
247 if (hard_regno < 0 || hard_regno >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
248 return hard_regno;
kono
parents:
diff changeset
249 if ((ep = elimination_map[hard_regno]) == NULL)
kono
parents:
diff changeset
250 return hard_regno;
kono
parents:
diff changeset
251 return ep->to;
kono
parents:
diff changeset
252 }
kono
parents:
diff changeset
253
kono
parents:
diff changeset
254 /* Return elimination which will be used for hard reg REG, NULL
kono
parents:
diff changeset
255 otherwise. */
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
256 static class lra_elim_table *
111
kono
parents:
diff changeset
257 get_elimination (rtx reg)
kono
parents:
diff changeset
258 {
kono
parents:
diff changeset
259 int hard_regno;
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
260 class lra_elim_table *ep;
111
kono
parents:
diff changeset
261
kono
parents:
diff changeset
262 lra_assert (REG_P (reg));
kono
parents:
diff changeset
263 if ((hard_regno = REGNO (reg)) < 0 || hard_regno >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
264 return NULL;
kono
parents:
diff changeset
265 if ((ep = elimination_map[hard_regno]) != NULL)
kono
parents:
diff changeset
266 return ep->from_rtx != reg ? NULL : ep;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
267 poly_int64 offset = self_elim_offsets[hard_regno];
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
268 if (known_eq (offset, 0))
111
kono
parents:
diff changeset
269 return NULL;
kono
parents:
diff changeset
270 /* This is an iteration to restore offsets just after HARD_REGNO
kono
parents:
diff changeset
271 stopped to be eliminable. */
kono
parents:
diff changeset
272 self_elim_table.from = self_elim_table.to = hard_regno;
kono
parents:
diff changeset
273 self_elim_table.from_rtx
kono
parents:
diff changeset
274 = self_elim_table.to_rtx
kono
parents:
diff changeset
275 = eliminable_reg_rtx[hard_regno];
kono
parents:
diff changeset
276 lra_assert (self_elim_table.from_rtx != NULL);
kono
parents:
diff changeset
277 self_elim_table.offset = offset;
kono
parents:
diff changeset
278 return &self_elim_table;
kono
parents:
diff changeset
279 }
kono
parents:
diff changeset
280
kono
parents:
diff changeset
281 /* Transform (subreg (plus reg const)) to (plus (subreg reg) const)
kono
parents:
diff changeset
282 when it is possible. Return X or the transformation result if the
kono
parents:
diff changeset
283 transformation is done. */
kono
parents:
diff changeset
284 static rtx
kono
parents:
diff changeset
285 move_plus_up (rtx x)
kono
parents:
diff changeset
286 {
kono
parents:
diff changeset
287 rtx subreg_reg;
kono
parents:
diff changeset
288 machine_mode x_mode, subreg_reg_mode;
kono
parents:
diff changeset
289
kono
parents:
diff changeset
290 if (GET_CODE (x) != SUBREG || !subreg_lowpart_p (x))
kono
parents:
diff changeset
291 return x;
kono
parents:
diff changeset
292 subreg_reg = SUBREG_REG (x);
kono
parents:
diff changeset
293 x_mode = GET_MODE (x);
kono
parents:
diff changeset
294 subreg_reg_mode = GET_MODE (subreg_reg);
kono
parents:
diff changeset
295 if (!paradoxical_subreg_p (x)
kono
parents:
diff changeset
296 && GET_CODE (subreg_reg) == PLUS
kono
parents:
diff changeset
297 && CONSTANT_P (XEXP (subreg_reg, 1))
kono
parents:
diff changeset
298 && GET_MODE_CLASS (x_mode) == MODE_INT
kono
parents:
diff changeset
299 && GET_MODE_CLASS (subreg_reg_mode) == MODE_INT)
kono
parents:
diff changeset
300 {
kono
parents:
diff changeset
301 rtx cst = simplify_subreg (x_mode, XEXP (subreg_reg, 1), subreg_reg_mode,
kono
parents:
diff changeset
302 subreg_lowpart_offset (x_mode,
kono
parents:
diff changeset
303 subreg_reg_mode));
kono
parents:
diff changeset
304 if (cst && CONSTANT_P (cst))
kono
parents:
diff changeset
305 return gen_rtx_PLUS (x_mode, lowpart_subreg (x_mode,
kono
parents:
diff changeset
306 XEXP (subreg_reg, 0),
kono
parents:
diff changeset
307 subreg_reg_mode), cst);
kono
parents:
diff changeset
308 }
kono
parents:
diff changeset
309 return x;
kono
parents:
diff changeset
310 }
kono
parents:
diff changeset
311
kono
parents:
diff changeset
312 /* Scan X and replace any eliminable registers (such as fp) with a
kono
parents:
diff changeset
313 replacement (such as sp) if SUBST_P, plus an offset. The offset is
kono
parents:
diff changeset
314 a change in the offset between the eliminable register and its
kono
parents:
diff changeset
315 substitution if UPDATE_P, or the full offset if FULL_P, or
kono
parents:
diff changeset
316 otherwise zero. If FULL_P, we also use the SP offsets for
kono
parents:
diff changeset
317 elimination to SP. If UPDATE_P, use UPDATE_SP_OFFSET for updating
kono
parents:
diff changeset
318 offsets of register elimnable to SP. If UPDATE_SP_OFFSET is
kono
parents:
diff changeset
319 non-zero, don't use difference of the offset and the previous
kono
parents:
diff changeset
320 offset.
kono
parents:
diff changeset
321
kono
parents:
diff changeset
322 MEM_MODE is the mode of an enclosing MEM. We need this to know how
kono
parents:
diff changeset
323 much to adjust a register for, e.g., PRE_DEC. Also, if we are
kono
parents:
diff changeset
324 inside a MEM, we are allowed to replace a sum of a hard register
kono
parents:
diff changeset
325 and the constant zero with the hard register, which we cannot do
kono
parents:
diff changeset
326 outside a MEM. In addition, we need to record the fact that a
kono
parents:
diff changeset
327 hard register is referenced outside a MEM.
kono
parents:
diff changeset
328
kono
parents:
diff changeset
329 If we make full substitution to SP for non-null INSN, add the insn
kono
parents:
diff changeset
330 sp offset. */
kono
parents:
diff changeset
331 rtx
kono
parents:
diff changeset
332 lra_eliminate_regs_1 (rtx_insn *insn, rtx x, machine_mode mem_mode,
kono
parents:
diff changeset
333 bool subst_p, bool update_p,
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
334 poly_int64 update_sp_offset, bool full_p)
111
kono
parents:
diff changeset
335 {
kono
parents:
diff changeset
336 enum rtx_code code = GET_CODE (x);
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
337 class lra_elim_table *ep;
111
kono
parents:
diff changeset
338 rtx new_rtx;
kono
parents:
diff changeset
339 int i, j;
kono
parents:
diff changeset
340 const char *fmt;
kono
parents:
diff changeset
341 int copied = 0;
kono
parents:
diff changeset
342
kono
parents:
diff changeset
343 lra_assert (!update_p || !full_p);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
344 lra_assert (known_eq (update_sp_offset, 0)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
345 || (!subst_p && update_p && !full_p));
111
kono
parents:
diff changeset
346 if (! current_function_decl)
kono
parents:
diff changeset
347 return x;
kono
parents:
diff changeset
348
kono
parents:
diff changeset
349 switch (code)
kono
parents:
diff changeset
350 {
kono
parents:
diff changeset
351 CASE_CONST_ANY:
kono
parents:
diff changeset
352 case CONST:
kono
parents:
diff changeset
353 case SYMBOL_REF:
kono
parents:
diff changeset
354 case CODE_LABEL:
kono
parents:
diff changeset
355 case PC:
kono
parents:
diff changeset
356 case CC0:
kono
parents:
diff changeset
357 case ASM_INPUT:
kono
parents:
diff changeset
358 case ADDR_VEC:
kono
parents:
diff changeset
359 case ADDR_DIFF_VEC:
kono
parents:
diff changeset
360 case RETURN:
kono
parents:
diff changeset
361 return x;
kono
parents:
diff changeset
362
kono
parents:
diff changeset
363 case REG:
kono
parents:
diff changeset
364 /* First handle the case where we encounter a bare hard register
kono
parents:
diff changeset
365 that is eliminable. Replace it with a PLUS. */
kono
parents:
diff changeset
366 if ((ep = get_elimination (x)) != NULL)
kono
parents:
diff changeset
367 {
kono
parents:
diff changeset
368 rtx to = subst_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
369
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
370 if (maybe_ne (update_sp_offset, 0))
111
kono
parents:
diff changeset
371 {
kono
parents:
diff changeset
372 if (ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
373 return plus_constant (Pmode, to, update_sp_offset);
kono
parents:
diff changeset
374 return to;
kono
parents:
diff changeset
375 }
kono
parents:
diff changeset
376 else if (update_p)
kono
parents:
diff changeset
377 return plus_constant (Pmode, to, ep->offset - ep->previous_offset);
kono
parents:
diff changeset
378 else if (full_p)
kono
parents:
diff changeset
379 return plus_constant (Pmode, to,
kono
parents:
diff changeset
380 ep->offset
kono
parents:
diff changeset
381 - (insn != NULL_RTX
kono
parents:
diff changeset
382 && ep->to_rtx == stack_pointer_rtx
kono
parents:
diff changeset
383 ? lra_get_insn_recog_data (insn)->sp_offset
kono
parents:
diff changeset
384 : 0));
kono
parents:
diff changeset
385 else
kono
parents:
diff changeset
386 return to;
kono
parents:
diff changeset
387 }
kono
parents:
diff changeset
388 return x;
kono
parents:
diff changeset
389
kono
parents:
diff changeset
390 case PLUS:
kono
parents:
diff changeset
391 /* If this is the sum of an eliminable register and a constant, rework
kono
parents:
diff changeset
392 the sum. */
kono
parents:
diff changeset
393 if (REG_P (XEXP (x, 0)) && CONSTANT_P (XEXP (x, 1)))
kono
parents:
diff changeset
394 {
kono
parents:
diff changeset
395 if ((ep = get_elimination (XEXP (x, 0))) != NULL)
kono
parents:
diff changeset
396 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
397 poly_int64 offset, curr_offset;
111
kono
parents:
diff changeset
398 rtx to = subst_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
399
kono
parents:
diff changeset
400 if (! update_p && ! full_p)
kono
parents:
diff changeset
401 return gen_rtx_PLUS (Pmode, to, XEXP (x, 1));
kono
parents:
diff changeset
402
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
403 if (maybe_ne (update_sp_offset, 0))
111
kono
parents:
diff changeset
404 offset = ep->to_rtx == stack_pointer_rtx ? update_sp_offset : 0;
kono
parents:
diff changeset
405 else
kono
parents:
diff changeset
406 offset = (update_p
kono
parents:
diff changeset
407 ? ep->offset - ep->previous_offset : ep->offset);
kono
parents:
diff changeset
408 if (full_p && insn != NULL_RTX && ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
409 offset -= lra_get_insn_recog_data (insn)->sp_offset;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
410 if (poly_int_rtx_p (XEXP (x, 1), &curr_offset)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
411 && known_eq (curr_offset, -offset))
111
kono
parents:
diff changeset
412 return to;
kono
parents:
diff changeset
413 else
kono
parents:
diff changeset
414 return gen_rtx_PLUS (Pmode, to,
kono
parents:
diff changeset
415 plus_constant (Pmode,
kono
parents:
diff changeset
416 XEXP (x, 1), offset));
kono
parents:
diff changeset
417 }
kono
parents:
diff changeset
418
kono
parents:
diff changeset
419 /* If the hard register is not eliminable, we are done since
kono
parents:
diff changeset
420 the other operand is a constant. */
kono
parents:
diff changeset
421 return x;
kono
parents:
diff changeset
422 }
kono
parents:
diff changeset
423
kono
parents:
diff changeset
424 /* If this is part of an address, we want to bring any constant
kono
parents:
diff changeset
425 to the outermost PLUS. We will do this by doing hard
kono
parents:
diff changeset
426 register replacement in our operands and seeing if a constant
kono
parents:
diff changeset
427 shows up in one of them.
kono
parents:
diff changeset
428
kono
parents:
diff changeset
429 Note that there is no risk of modifying the structure of the
kono
parents:
diff changeset
430 insn, since we only get called for its operands, thus we are
kono
parents:
diff changeset
431 either modifying the address inside a MEM, or something like
kono
parents:
diff changeset
432 an address operand of a load-address insn. */
kono
parents:
diff changeset
433
kono
parents:
diff changeset
434 {
kono
parents:
diff changeset
435 rtx new0 = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
436 subst_p, update_p,
kono
parents:
diff changeset
437 update_sp_offset, full_p);
kono
parents:
diff changeset
438 rtx new1 = lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
439 subst_p, update_p,
kono
parents:
diff changeset
440 update_sp_offset, full_p);
kono
parents:
diff changeset
441
kono
parents:
diff changeset
442 new0 = move_plus_up (new0);
kono
parents:
diff changeset
443 new1 = move_plus_up (new1);
kono
parents:
diff changeset
444 if (new0 != XEXP (x, 0) || new1 != XEXP (x, 1))
kono
parents:
diff changeset
445 return form_sum (new0, new1);
kono
parents:
diff changeset
446 }
kono
parents:
diff changeset
447 return x;
kono
parents:
diff changeset
448
kono
parents:
diff changeset
449 case MULT:
kono
parents:
diff changeset
450 /* If this is the product of an eliminable hard register and a
kono
parents:
diff changeset
451 constant, apply the distribute law and move the constant out
kono
parents:
diff changeset
452 so that we have (plus (mult ..) ..). This is needed in order
kono
parents:
diff changeset
453 to keep load-address insns valid. This case is pathological.
kono
parents:
diff changeset
454 We ignore the possibility of overflow here. */
kono
parents:
diff changeset
455 if (REG_P (XEXP (x, 0)) && CONST_INT_P (XEXP (x, 1))
kono
parents:
diff changeset
456 && (ep = get_elimination (XEXP (x, 0))) != NULL)
kono
parents:
diff changeset
457 {
kono
parents:
diff changeset
458 rtx to = subst_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
459
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
460 if (maybe_ne (update_sp_offset, 0))
111
kono
parents:
diff changeset
461 {
kono
parents:
diff changeset
462 if (ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
463 return plus_constant (Pmode,
kono
parents:
diff changeset
464 gen_rtx_MULT (Pmode, to, XEXP (x, 1)),
kono
parents:
diff changeset
465 update_sp_offset * INTVAL (XEXP (x, 1)));
kono
parents:
diff changeset
466 return gen_rtx_MULT (Pmode, to, XEXP (x, 1));
kono
parents:
diff changeset
467 }
kono
parents:
diff changeset
468 else if (update_p)
kono
parents:
diff changeset
469 return plus_constant (Pmode,
kono
parents:
diff changeset
470 gen_rtx_MULT (Pmode, to, XEXP (x, 1)),
kono
parents:
diff changeset
471 (ep->offset - ep->previous_offset)
kono
parents:
diff changeset
472 * INTVAL (XEXP (x, 1)));
kono
parents:
diff changeset
473 else if (full_p)
kono
parents:
diff changeset
474 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
475 poly_int64 offset = ep->offset;
111
kono
parents:
diff changeset
476
kono
parents:
diff changeset
477 if (insn != NULL_RTX && ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
478 offset -= lra_get_insn_recog_data (insn)->sp_offset;
kono
parents:
diff changeset
479 return
kono
parents:
diff changeset
480 plus_constant (Pmode,
kono
parents:
diff changeset
481 gen_rtx_MULT (Pmode, to, XEXP (x, 1)),
kono
parents:
diff changeset
482 offset * INTVAL (XEXP (x, 1)));
kono
parents:
diff changeset
483 }
kono
parents:
diff changeset
484 else
kono
parents:
diff changeset
485 return gen_rtx_MULT (Pmode, to, XEXP (x, 1));
kono
parents:
diff changeset
486 }
kono
parents:
diff changeset
487
kono
parents:
diff changeset
488 /* fall through */
kono
parents:
diff changeset
489
kono
parents:
diff changeset
490 case CALL:
kono
parents:
diff changeset
491 case COMPARE:
kono
parents:
diff changeset
492 /* See comments before PLUS about handling MINUS. */
kono
parents:
diff changeset
493 case MINUS:
kono
parents:
diff changeset
494 case DIV: case UDIV:
kono
parents:
diff changeset
495 case MOD: case UMOD:
kono
parents:
diff changeset
496 case AND: case IOR: case XOR:
kono
parents:
diff changeset
497 case ROTATERT: case ROTATE:
kono
parents:
diff changeset
498 case ASHIFTRT: case LSHIFTRT: case ASHIFT:
kono
parents:
diff changeset
499 case NE: case EQ:
kono
parents:
diff changeset
500 case GE: case GT: case GEU: case GTU:
kono
parents:
diff changeset
501 case LE: case LT: case LEU: case LTU:
kono
parents:
diff changeset
502 {
kono
parents:
diff changeset
503 rtx new0 = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
504 subst_p, update_p,
kono
parents:
diff changeset
505 update_sp_offset, full_p);
kono
parents:
diff changeset
506 rtx new1 = XEXP (x, 1)
kono
parents:
diff changeset
507 ? lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
508 subst_p, update_p,
kono
parents:
diff changeset
509 update_sp_offset, full_p) : 0;
kono
parents:
diff changeset
510
kono
parents:
diff changeset
511 if (new0 != XEXP (x, 0) || new1 != XEXP (x, 1))
kono
parents:
diff changeset
512 return gen_rtx_fmt_ee (code, GET_MODE (x), new0, new1);
kono
parents:
diff changeset
513 }
kono
parents:
diff changeset
514 return x;
kono
parents:
diff changeset
515
kono
parents:
diff changeset
516 case EXPR_LIST:
kono
parents:
diff changeset
517 /* If we have something in XEXP (x, 0), the usual case,
kono
parents:
diff changeset
518 eliminate it. */
kono
parents:
diff changeset
519 if (XEXP (x, 0))
kono
parents:
diff changeset
520 {
kono
parents:
diff changeset
521 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
522 subst_p, update_p,
kono
parents:
diff changeset
523 update_sp_offset, full_p);
kono
parents:
diff changeset
524 if (new_rtx != XEXP (x, 0))
kono
parents:
diff changeset
525 {
kono
parents:
diff changeset
526 /* If this is a REG_DEAD note, it is not valid anymore.
kono
parents:
diff changeset
527 Using the eliminated version could result in creating a
kono
parents:
diff changeset
528 REG_DEAD note for the stack or frame pointer. */
kono
parents:
diff changeset
529 if (REG_NOTE_KIND (x) == REG_DEAD)
kono
parents:
diff changeset
530 return (XEXP (x, 1)
kono
parents:
diff changeset
531 ? lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
532 subst_p, update_p,
kono
parents:
diff changeset
533 update_sp_offset, full_p)
kono
parents:
diff changeset
534 : NULL_RTX);
kono
parents:
diff changeset
535
kono
parents:
diff changeset
536 x = alloc_reg_note (REG_NOTE_KIND (x), new_rtx, XEXP (x, 1));
kono
parents:
diff changeset
537 }
kono
parents:
diff changeset
538 }
kono
parents:
diff changeset
539
kono
parents:
diff changeset
540 /* fall through */
kono
parents:
diff changeset
541
kono
parents:
diff changeset
542 case INSN_LIST:
kono
parents:
diff changeset
543 case INT_LIST:
kono
parents:
diff changeset
544 /* Now do eliminations in the rest of the chain. If this was
kono
parents:
diff changeset
545 an EXPR_LIST, this might result in allocating more memory than is
kono
parents:
diff changeset
546 strictly needed, but it simplifies the code. */
kono
parents:
diff changeset
547 if (XEXP (x, 1))
kono
parents:
diff changeset
548 {
kono
parents:
diff changeset
549 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 1), mem_mode,
kono
parents:
diff changeset
550 subst_p, update_p,
kono
parents:
diff changeset
551 update_sp_offset, full_p);
kono
parents:
diff changeset
552 if (new_rtx != XEXP (x, 1))
kono
parents:
diff changeset
553 return
kono
parents:
diff changeset
554 gen_rtx_fmt_ee (GET_CODE (x), GET_MODE (x),
kono
parents:
diff changeset
555 XEXP (x, 0), new_rtx);
kono
parents:
diff changeset
556 }
kono
parents:
diff changeset
557 return x;
kono
parents:
diff changeset
558
kono
parents:
diff changeset
559 case PRE_INC:
kono
parents:
diff changeset
560 case POST_INC:
kono
parents:
diff changeset
561 case PRE_DEC:
kono
parents:
diff changeset
562 case POST_DEC:
kono
parents:
diff changeset
563 /* We do not support elimination of a register that is modified.
kono
parents:
diff changeset
564 elimination_effects has already make sure that this does not
kono
parents:
diff changeset
565 happen. */
kono
parents:
diff changeset
566 return x;
kono
parents:
diff changeset
567
kono
parents:
diff changeset
568 case PRE_MODIFY:
kono
parents:
diff changeset
569 case POST_MODIFY:
kono
parents:
diff changeset
570 /* We do not support elimination of a hard register that is
kono
parents:
diff changeset
571 modified. LRA has already make sure that this does not
kono
parents:
diff changeset
572 happen. The only remaining case we need to consider here is
kono
parents:
diff changeset
573 that the increment value may be an eliminable register. */
kono
parents:
diff changeset
574 if (GET_CODE (XEXP (x, 1)) == PLUS
kono
parents:
diff changeset
575 && XEXP (XEXP (x, 1), 0) == XEXP (x, 0))
kono
parents:
diff changeset
576 {
kono
parents:
diff changeset
577 rtx new_rtx = lra_eliminate_regs_1 (insn, XEXP (XEXP (x, 1), 1),
kono
parents:
diff changeset
578 mem_mode, subst_p, update_p,
kono
parents:
diff changeset
579 update_sp_offset, full_p);
kono
parents:
diff changeset
580
kono
parents:
diff changeset
581 if (new_rtx != XEXP (XEXP (x, 1), 1))
kono
parents:
diff changeset
582 return gen_rtx_fmt_ee (code, GET_MODE (x), XEXP (x, 0),
kono
parents:
diff changeset
583 gen_rtx_PLUS (GET_MODE (x),
kono
parents:
diff changeset
584 XEXP (x, 0), new_rtx));
kono
parents:
diff changeset
585 }
kono
parents:
diff changeset
586 return x;
kono
parents:
diff changeset
587
kono
parents:
diff changeset
588 case STRICT_LOW_PART:
kono
parents:
diff changeset
589 case NEG: case NOT:
kono
parents:
diff changeset
590 case SIGN_EXTEND: case ZERO_EXTEND:
kono
parents:
diff changeset
591 case TRUNCATE: case FLOAT_EXTEND: case FLOAT_TRUNCATE:
kono
parents:
diff changeset
592 case FLOAT: case FIX:
kono
parents:
diff changeset
593 case UNSIGNED_FIX: case UNSIGNED_FLOAT:
kono
parents:
diff changeset
594 case ABS:
kono
parents:
diff changeset
595 case SQRT:
kono
parents:
diff changeset
596 case FFS:
kono
parents:
diff changeset
597 case CLZ:
kono
parents:
diff changeset
598 case CTZ:
kono
parents:
diff changeset
599 case POPCOUNT:
kono
parents:
diff changeset
600 case PARITY:
kono
parents:
diff changeset
601 case BSWAP:
kono
parents:
diff changeset
602 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 0), mem_mode,
kono
parents:
diff changeset
603 subst_p, update_p,
kono
parents:
diff changeset
604 update_sp_offset, full_p);
kono
parents:
diff changeset
605 if (new_rtx != XEXP (x, 0))
kono
parents:
diff changeset
606 return gen_rtx_fmt_e (code, GET_MODE (x), new_rtx);
kono
parents:
diff changeset
607 return x;
kono
parents:
diff changeset
608
kono
parents:
diff changeset
609 case SUBREG:
kono
parents:
diff changeset
610 new_rtx = lra_eliminate_regs_1 (insn, SUBREG_REG (x), mem_mode,
kono
parents:
diff changeset
611 subst_p, update_p,
kono
parents:
diff changeset
612 update_sp_offset, full_p);
kono
parents:
diff changeset
613
kono
parents:
diff changeset
614 if (new_rtx != SUBREG_REG (x))
kono
parents:
diff changeset
615 {
kono
parents:
diff changeset
616 if (MEM_P (new_rtx) && !paradoxical_subreg_p (x))
kono
parents:
diff changeset
617 {
kono
parents:
diff changeset
618 SUBREG_REG (x) = new_rtx;
kono
parents:
diff changeset
619 alter_subreg (&x, false);
kono
parents:
diff changeset
620 return x;
kono
parents:
diff changeset
621 }
kono
parents:
diff changeset
622 else if (! subst_p)
kono
parents:
diff changeset
623 {
kono
parents:
diff changeset
624 /* LRA can transform subregs itself. So don't call
kono
parents:
diff changeset
625 simplify_gen_subreg until LRA transformations are
kono
parents:
diff changeset
626 finished. Function simplify_gen_subreg can do
kono
parents:
diff changeset
627 non-trivial transformations (like truncation) which
kono
parents:
diff changeset
628 might make LRA work to fail. */
kono
parents:
diff changeset
629 SUBREG_REG (x) = new_rtx;
kono
parents:
diff changeset
630 return x;
kono
parents:
diff changeset
631 }
kono
parents:
diff changeset
632 else
kono
parents:
diff changeset
633 return simplify_gen_subreg (GET_MODE (x), new_rtx,
kono
parents:
diff changeset
634 GET_MODE (new_rtx), SUBREG_BYTE (x));
kono
parents:
diff changeset
635 }
kono
parents:
diff changeset
636
kono
parents:
diff changeset
637 return x;
kono
parents:
diff changeset
638
kono
parents:
diff changeset
639 case MEM:
kono
parents:
diff changeset
640 /* Our only special processing is to pass the mode of the MEM to our
kono
parents:
diff changeset
641 recursive call and copy the flags. While we are here, handle this
kono
parents:
diff changeset
642 case more efficiently. */
kono
parents:
diff changeset
643 return
kono
parents:
diff changeset
644 replace_equiv_address_nv
kono
parents:
diff changeset
645 (x,
kono
parents:
diff changeset
646 lra_eliminate_regs_1 (insn, XEXP (x, 0), GET_MODE (x),
kono
parents:
diff changeset
647 subst_p, update_p, update_sp_offset, full_p));
kono
parents:
diff changeset
648
kono
parents:
diff changeset
649 case USE:
kono
parents:
diff changeset
650 /* Handle insn_list USE that a call to a pure function may generate. */
kono
parents:
diff changeset
651 new_rtx = lra_eliminate_regs_1 (insn, XEXP (x, 0), VOIDmode,
kono
parents:
diff changeset
652 subst_p, update_p, update_sp_offset, full_p);
kono
parents:
diff changeset
653 if (new_rtx != XEXP (x, 0))
kono
parents:
diff changeset
654 return gen_rtx_USE (GET_MODE (x), new_rtx);
kono
parents:
diff changeset
655 return x;
kono
parents:
diff changeset
656
kono
parents:
diff changeset
657 case CLOBBER:
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);
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
733 class lra_elim_table *ep;
111
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
kono
parents:
diff changeset
810 case SET:
kono
parents:
diff changeset
811 if (SET_DEST (x) == stack_pointer_rtx
kono
parents:
diff changeset
812 && GET_CODE (SET_SRC (x)) == PLUS
kono
parents:
diff changeset
813 && XEXP (SET_SRC (x), 0) == SET_DEST (x)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
814 && poly_int_rtx_p (XEXP (SET_SRC (x), 1), &offset))
111
kono
parents:
diff changeset
815 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
816 curr_sp_change += offset;
111
kono
parents:
diff changeset
817 return;
kono
parents:
diff changeset
818 }
kono
parents:
diff changeset
819 if (! REG_P (SET_DEST (x))
kono
parents:
diff changeset
820 || REGNO (SET_DEST (x)) >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
821 mark_not_eliminable (SET_DEST (x), mem_mode);
kono
parents:
diff changeset
822 else
kono
parents:
diff changeset
823 {
kono
parents:
diff changeset
824 /* See if this is setting the replacement hard register for
kono
parents:
diff changeset
825 an elimination.
kono
parents:
diff changeset
826
kono
parents:
diff changeset
827 If DEST is the hard frame pointer, we do nothing because
kono
parents:
diff changeset
828 we assume that all assignments to the frame pointer are
kono
parents:
diff changeset
829 for non-local gotos and are being done at a time when
kono
parents:
diff changeset
830 they are valid and do not disturb anything else. Some
kono
parents:
diff changeset
831 machines want to eliminate a fake argument pointer (or
kono
parents:
diff changeset
832 even a fake frame pointer) with either the real frame
kono
parents:
diff changeset
833 pointer or the stack pointer. Assignments to the hard
kono
parents:
diff changeset
834 frame pointer must not prevent this elimination. */
kono
parents:
diff changeset
835 for (ep = reg_eliminate;
kono
parents:
diff changeset
836 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
kono
parents:
diff changeset
837 ep++)
kono
parents:
diff changeset
838 if (ep->to_rtx == SET_DEST (x)
kono
parents:
diff changeset
839 && SET_DEST (x) != hard_frame_pointer_rtx)
kono
parents:
diff changeset
840 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
841 }
kono
parents:
diff changeset
842
kono
parents:
diff changeset
843 mark_not_eliminable (SET_SRC (x), mem_mode);
kono
parents:
diff changeset
844 return;
kono
parents:
diff changeset
845
kono
parents:
diff changeset
846 case MEM:
kono
parents:
diff changeset
847 /* Our only special processing is to pass the mode of the MEM to
kono
parents:
diff changeset
848 our recursive call. */
kono
parents:
diff changeset
849 mark_not_eliminable (XEXP (x, 0), GET_MODE (x));
kono
parents:
diff changeset
850 return;
kono
parents:
diff changeset
851
kono
parents:
diff changeset
852 default:
kono
parents:
diff changeset
853 break;
kono
parents:
diff changeset
854 }
kono
parents:
diff changeset
855
kono
parents:
diff changeset
856 fmt = GET_RTX_FORMAT (code);
kono
parents:
diff changeset
857 for (i = 0; i < GET_RTX_LENGTH (code); i++, fmt++)
kono
parents:
diff changeset
858 {
kono
parents:
diff changeset
859 if (*fmt == 'e')
kono
parents:
diff changeset
860 mark_not_eliminable (XEXP (x, i), mem_mode);
kono
parents:
diff changeset
861 else if (*fmt == 'E')
kono
parents:
diff changeset
862 for (j = 0; j < XVECLEN (x, i); j++)
kono
parents:
diff changeset
863 mark_not_eliminable (XVECEXP (x, i, j), mem_mode);
kono
parents:
diff changeset
864 }
kono
parents:
diff changeset
865 }
kono
parents:
diff changeset
866
kono
parents:
diff changeset
867
kono
parents:
diff changeset
868
kono
parents:
diff changeset
869 /* Scan INSN and eliminate all eliminable hard registers in it.
kono
parents:
diff changeset
870
kono
parents:
diff changeset
871 If REPLACE_P is true, do the replacement destructively. Also
kono
parents:
diff changeset
872 delete the insn as dead it if it is setting an eliminable register.
kono
parents:
diff changeset
873
kono
parents:
diff changeset
874 If REPLACE_P is false, just update the offsets while keeping the
kono
parents:
diff changeset
875 base register the same. If FIRST_P, use the sp offset for
kono
parents:
diff changeset
876 elimination to sp. Otherwise, use UPDATE_SP_OFFSET for this. If
kono
parents:
diff changeset
877 UPDATE_SP_OFFSET is non-zero, don't use difference of the offset
kono
parents:
diff changeset
878 and the previous offset. Attach the note about used elimination
kono
parents:
diff changeset
879 for insns setting frame pointer to update elimination easy (without
kono
parents:
diff changeset
880 parsing already generated elimination insns to find offset
kono
parents:
diff changeset
881 previously used) in future. */
kono
parents:
diff changeset
882
kono
parents:
diff changeset
883 void
kono
parents:
diff changeset
884 eliminate_regs_in_insn (rtx_insn *insn, bool replace_p, bool first_p,
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
885 poly_int64 update_sp_offset)
111
kono
parents:
diff changeset
886 {
kono
parents:
diff changeset
887 int icode = recog_memoized (insn);
kono
parents:
diff changeset
888 rtx old_set = single_set (insn);
kono
parents:
diff changeset
889 bool validate_p;
kono
parents:
diff changeset
890 int i;
kono
parents:
diff changeset
891 rtx substed_operand[MAX_RECOG_OPERANDS];
kono
parents:
diff changeset
892 rtx orig_operand[MAX_RECOG_OPERANDS];
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
893 class lra_elim_table *ep;
111
kono
parents:
diff changeset
894 rtx plus_src, plus_cst_src;
kono
parents:
diff changeset
895 lra_insn_recog_data_t id;
kono
parents:
diff changeset
896 struct lra_static_insn_data *static_id;
kono
parents:
diff changeset
897
kono
parents:
diff changeset
898 if (icode < 0 && asm_noperands (PATTERN (insn)) < 0 && ! DEBUG_INSN_P (insn))
kono
parents:
diff changeset
899 {
kono
parents:
diff changeset
900 lra_assert (GET_CODE (PATTERN (insn)) == USE
kono
parents:
diff changeset
901 || GET_CODE (PATTERN (insn)) == CLOBBER
kono
parents:
diff changeset
902 || GET_CODE (PATTERN (insn)) == ASM_INPUT);
kono
parents:
diff changeset
903 return;
kono
parents:
diff changeset
904 }
kono
parents:
diff changeset
905
kono
parents:
diff changeset
906 /* We allow one special case which happens to work on all machines we
kono
parents:
diff changeset
907 currently support: a single set with the source or a REG_EQUAL
kono
parents:
diff changeset
908 note being a PLUS of an eliminable register and a constant. */
kono
parents:
diff changeset
909 plus_src = plus_cst_src = 0;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
910 poly_int64 offset = 0;
111
kono
parents:
diff changeset
911 if (old_set && REG_P (SET_DEST (old_set)))
kono
parents:
diff changeset
912 {
kono
parents:
diff changeset
913 if (GET_CODE (SET_SRC (old_set)) == PLUS)
kono
parents:
diff changeset
914 plus_src = SET_SRC (old_set);
kono
parents:
diff changeset
915 /* First see if the source is of the form (plus (...) CST). */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
916 if (plus_src && poly_int_rtx_p (XEXP (plus_src, 1), &offset))
111
kono
parents:
diff changeset
917 plus_cst_src = plus_src;
kono
parents:
diff changeset
918 /* Check that the first operand of the PLUS is a hard reg or
kono
parents:
diff changeset
919 the lowpart subreg of one. */
kono
parents:
diff changeset
920 if (plus_cst_src)
kono
parents:
diff changeset
921 {
kono
parents:
diff changeset
922 rtx reg = XEXP (plus_cst_src, 0);
kono
parents:
diff changeset
923
kono
parents:
diff changeset
924 if (GET_CODE (reg) == SUBREG && subreg_lowpart_p (reg))
kono
parents:
diff changeset
925 reg = SUBREG_REG (reg);
kono
parents:
diff changeset
926
kono
parents:
diff changeset
927 if (!REG_P (reg) || REGNO (reg) >= FIRST_PSEUDO_REGISTER)
kono
parents:
diff changeset
928 plus_cst_src = 0;
kono
parents:
diff changeset
929 }
kono
parents:
diff changeset
930 }
kono
parents:
diff changeset
931 if (plus_cst_src)
kono
parents:
diff changeset
932 {
kono
parents:
diff changeset
933 rtx reg = XEXP (plus_cst_src, 0);
kono
parents:
diff changeset
934
kono
parents:
diff changeset
935 if (GET_CODE (reg) == SUBREG)
kono
parents:
diff changeset
936 reg = SUBREG_REG (reg);
kono
parents:
diff changeset
937
kono
parents:
diff changeset
938 if (REG_P (reg) && (ep = get_elimination (reg)) != NULL)
kono
parents:
diff changeset
939 {
kono
parents:
diff changeset
940 rtx to_rtx = replace_p ? ep->to_rtx : ep->from_rtx;
kono
parents:
diff changeset
941
kono
parents:
diff changeset
942 if (! replace_p)
kono
parents:
diff changeset
943 {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
944 if (known_eq (update_sp_offset, 0))
111
kono
parents:
diff changeset
945 offset += (ep->offset - ep->previous_offset);
kono
parents:
diff changeset
946 if (ep->to_rtx == stack_pointer_rtx)
kono
parents:
diff changeset
947 {
kono
parents:
diff changeset
948 if (first_p)
kono
parents:
diff changeset
949 offset -= lra_get_insn_recog_data (insn)->sp_offset;
kono
parents:
diff changeset
950 else
kono
parents:
diff changeset
951 offset += update_sp_offset;
kono
parents:
diff changeset
952 }
kono
parents:
diff changeset
953 offset = trunc_int_for_mode (offset, GET_MODE (plus_cst_src));
kono
parents:
diff changeset
954 }
kono
parents:
diff changeset
955
kono
parents:
diff changeset
956 if (GET_CODE (XEXP (plus_cst_src, 0)) == SUBREG)
kono
parents:
diff changeset
957 to_rtx = gen_lowpart (GET_MODE (XEXP (plus_cst_src, 0)), to_rtx);
kono
parents:
diff changeset
958 /* If we have a nonzero offset, and the source is already a
kono
parents:
diff changeset
959 simple REG, the following transformation would increase
kono
parents:
diff changeset
960 the cost of the insn by replacing a simple REG with (plus
kono
parents:
diff changeset
961 (reg sp) CST). So try only when we already had a PLUS
kono
parents:
diff changeset
962 before. */
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
963 if (known_eq (offset, 0) || plus_src)
111
kono
parents:
diff changeset
964 {
kono
parents:
diff changeset
965 rtx new_src = plus_constant (GET_MODE (to_rtx), to_rtx, offset);
kono
parents:
diff changeset
966
kono
parents:
diff changeset
967 old_set = single_set (insn);
kono
parents:
diff changeset
968
kono
parents:
diff changeset
969 /* First see if this insn remains valid when we make the
kono
parents:
diff changeset
970 change. If not, try to replace the whole pattern
kono
parents:
diff changeset
971 with a simple set (this may help if the original insn
kono
parents:
diff changeset
972 was a PARALLEL that was only recognized as single_set
kono
parents:
diff changeset
973 due to REG_UNUSED notes). If this isn't valid
kono
parents:
diff changeset
974 either, keep the INSN_CODE the same and let the
kono
parents:
diff changeset
975 constraint pass fix it up. */
kono
parents:
diff changeset
976 if (! validate_change (insn, &SET_SRC (old_set), new_src, 0))
kono
parents:
diff changeset
977 {
kono
parents:
diff changeset
978 rtx new_pat = gen_rtx_SET (SET_DEST (old_set), new_src);
kono
parents:
diff changeset
979
kono
parents:
diff changeset
980 if (! validate_change (insn, &PATTERN (insn), new_pat, 0))
kono
parents:
diff changeset
981 SET_SRC (old_set) = new_src;
kono
parents:
diff changeset
982 }
kono
parents:
diff changeset
983 lra_update_insn_recog_data (insn);
kono
parents:
diff changeset
984 /* This can't have an effect on elimination offsets, so skip
kono
parents:
diff changeset
985 right to the end. */
kono
parents:
diff changeset
986 return;
kono
parents:
diff changeset
987 }
kono
parents:
diff changeset
988 }
kono
parents:
diff changeset
989 }
kono
parents:
diff changeset
990
kono
parents:
diff changeset
991 /* Eliminate all eliminable registers occurring in operands that
kono
parents:
diff changeset
992 can be handled by the constraint pass. */
kono
parents:
diff changeset
993 id = lra_get_insn_recog_data (insn);
kono
parents:
diff changeset
994 static_id = id->insn_static_data;
kono
parents:
diff changeset
995 validate_p = false;
kono
parents:
diff changeset
996 for (i = 0; i < static_id->n_operands; i++)
kono
parents:
diff changeset
997 {
kono
parents:
diff changeset
998 orig_operand[i] = *id->operand_loc[i];
kono
parents:
diff changeset
999 substed_operand[i] = *id->operand_loc[i];
kono
parents:
diff changeset
1000
kono
parents:
diff changeset
1001 /* For an asm statement, every operand is eliminable. */
kono
parents:
diff changeset
1002 if (icode < 0 || insn_data[icode].operand[i].eliminable)
kono
parents:
diff changeset
1003 {
kono
parents:
diff changeset
1004 /* Check for setting a hard register that we know about. */
kono
parents:
diff changeset
1005 if (static_id->operand[i].type != OP_IN
kono
parents:
diff changeset
1006 && REG_P (orig_operand[i]))
kono
parents:
diff changeset
1007 {
kono
parents:
diff changeset
1008 /* If we are assigning to a hard register that can be
kono
parents:
diff changeset
1009 eliminated, it must be as part of a PARALLEL, since
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1010 the code above handles single SETs. This reg cannot
111
kono
parents:
diff changeset
1011 be longer eliminated -- it is forced by
kono
parents:
diff changeset
1012 mark_not_eliminable. */
kono
parents:
diff changeset
1013 for (ep = reg_eliminate;
kono
parents:
diff changeset
1014 ep < &reg_eliminate[NUM_ELIMINABLE_REGS];
kono
parents:
diff changeset
1015 ep++)
kono
parents:
diff changeset
1016 lra_assert (ep->from_rtx != orig_operand[i]
kono
parents:
diff changeset
1017 || ! ep->can_eliminate);
kono
parents:
diff changeset
1018 }
kono
parents:
diff changeset
1019
kono
parents:
diff changeset
1020 /* Companion to the above plus substitution, we can allow
kono
parents:
diff changeset
1021 invariants as the source of a plain move. */
kono
parents:
diff changeset
1022 substed_operand[i]
kono
parents:
diff changeset
1023 = lra_eliminate_regs_1 (insn, *id->operand_loc[i], VOIDmode,
kono
parents:
diff changeset
1024 replace_p, ! replace_p && ! first_p,
kono
parents:
diff changeset
1025 update_sp_offset, first_p);
kono
parents:
diff changeset
1026 if (substed_operand[i] != orig_operand[i])
kono
parents:
diff changeset
1027 validate_p = true;
kono
parents:
diff changeset
1028 }
kono
parents:
diff changeset
1029 }
kono
parents:
diff changeset
1030
kono
parents:
diff changeset
1031 if (! validate_p)
kono
parents:
diff changeset
1032 return;
kono
parents:
diff changeset
1033
kono
parents:
diff changeset
1034 /* Substitute the operands; the new values are in the substed_operand
kono
parents:
diff changeset
1035 array. */
kono
parents:
diff changeset
1036 for (i = 0; i < static_id->n_operands; i++)
kono
parents:
diff changeset
1037 *id->operand_loc[i] = substed_operand[i];
kono
parents:
diff changeset
1038 for (i = 0; i < static_id->n_dups; i++)
kono
parents:
diff changeset
1039 *id->dup_loc[i] = substed_operand[(int) static_id->dup_num[i]];
kono
parents:
diff changeset
1040
kono
parents:
diff changeset
1041 /* If we had a move insn but now we don't, re-recognize it.
kono
parents:
diff changeset
1042 This will cause spurious re-recognition if the old move had a
kono
parents:
diff changeset
1043 PARALLEL since the new one still will, but we can't call
kono
parents:
diff changeset
1044 single_set without having put new body into the insn and the
kono
parents:
diff changeset
1045 re-recognition won't hurt in this rare case. */
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1046 lra_update_insn_recog_data (insn);
111
kono
parents:
diff changeset
1047 }
kono
parents:
diff changeset
1048
kono
parents:
diff changeset
1049 /* Spill pseudos which are assigned to hard registers in SET. Add
kono
parents:
diff changeset
1050 affected insns for processing in the subsequent constraint
kono
parents:
diff changeset
1051 pass. */
kono
parents:
diff changeset
1052 static void
kono
parents:
diff changeset
1053 spill_pseudos (HARD_REG_SET set)
kono
parents:
diff changeset
1054 {
kono
parents:
diff changeset
1055 int i;
kono
parents:
diff changeset
1056 bitmap_head to_process;
kono
parents:
diff changeset
1057 rtx_insn *insn;
kono
parents:
diff changeset
1058
kono
parents:
diff changeset
1059 if (hard_reg_set_empty_p (set))
kono
parents:
diff changeset
1060 return;
kono
parents:
diff changeset
1061 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1062 {
kono
parents:
diff changeset
1063 fprintf (lra_dump_file, " Spilling non-eliminable hard regs:");
kono
parents:
diff changeset
1064 for (i = 0; i < FIRST_PSEUDO_REGISTER; i++)
kono
parents:
diff changeset
1065 if (TEST_HARD_REG_BIT (set, i))
kono
parents:
diff changeset
1066 fprintf (lra_dump_file, " %d", i);
kono
parents:
diff changeset
1067 fprintf (lra_dump_file, "\n");
kono
parents:
diff changeset
1068 }
kono
parents:
diff changeset
1069 bitmap_initialize (&to_process, &reg_obstack);
kono
parents:
diff changeset
1070 for (i = FIRST_PSEUDO_REGISTER; i < max_reg_num (); i++)
kono
parents:
diff changeset
1071 if (lra_reg_info[i].nrefs != 0 && reg_renumber[i] >= 0
kono
parents:
diff changeset
1072 && overlaps_hard_reg_set_p (set,
kono
parents:
diff changeset
1073 PSEUDO_REGNO_MODE (i), reg_renumber[i]))
kono
parents:
diff changeset
1074 {
kono
parents:
diff changeset
1075 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1076 fprintf (lra_dump_file, " Spilling r%d(%d)\n",
kono
parents:
diff changeset
1077 i, reg_renumber[i]);
kono
parents:
diff changeset
1078 reg_renumber[i] = -1;
kono
parents:
diff changeset
1079 bitmap_ior_into (&to_process, &lra_reg_info[i].insn_bitmap);
kono
parents:
diff changeset
1080 }
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1081 lra_no_alloc_regs |= set;
111
kono
parents:
diff changeset
1082 for (insn = get_insns (); insn != NULL_RTX; insn = NEXT_INSN (insn))
kono
parents:
diff changeset
1083 if (bitmap_bit_p (&to_process, INSN_UID (insn)))
kono
parents:
diff changeset
1084 {
kono
parents:
diff changeset
1085 lra_push_insn (insn);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1086 lra_set_used_insn_alternative (insn, LRA_UNKNOWN_ALT);
111
kono
parents:
diff changeset
1087 }
kono
parents:
diff changeset
1088 bitmap_clear (&to_process);
kono
parents:
diff changeset
1089 }
kono
parents:
diff changeset
1090
kono
parents:
diff changeset
1091 /* Update all offsets and possibility for elimination on eliminable
kono
parents:
diff changeset
1092 registers. Spill pseudos assigned to registers which are
kono
parents:
diff changeset
1093 uneliminable, update LRA_NO_ALLOC_REGS and ELIMINABLE_REG_SET. Add
kono
parents:
diff changeset
1094 insns to INSNS_WITH_CHANGED_OFFSETS containing eliminable hard
kono
parents:
diff changeset
1095 registers whose offsets should be changed. Return true if any
kono
parents:
diff changeset
1096 elimination offset changed. */
kono
parents:
diff changeset
1097 static bool
kono
parents:
diff changeset
1098 update_reg_eliminate (bitmap insns_with_changed_offsets)
kono
parents:
diff changeset
1099 {
kono
parents:
diff changeset
1100 bool prev, result;
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1101 class lra_elim_table *ep, *ep1;
111
kono
parents:
diff changeset
1102 HARD_REG_SET temp_hard_reg_set;
kono
parents:
diff changeset
1103
kono
parents:
diff changeset
1104 targetm.compute_frame_layout ();
kono
parents:
diff changeset
1105
kono
parents:
diff changeset
1106 /* Clear self elimination offsets. */
kono
parents:
diff changeset
1107 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1108 self_elim_offsets[ep->from] = 0;
kono
parents:
diff changeset
1109 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1110 {
kono
parents:
diff changeset
1111 /* If it is a currently used elimination: update the previous
kono
parents:
diff changeset
1112 offset. */
kono
parents:
diff changeset
1113 if (elimination_map[ep->from] == ep)
kono
parents:
diff changeset
1114 ep->previous_offset = ep->offset;
kono
parents:
diff changeset
1115
kono
parents:
diff changeset
1116 prev = ep->prev_can_eliminate;
kono
parents:
diff changeset
1117 setup_can_eliminate (ep, targetm.can_eliminate (ep->from, ep->to));
kono
parents:
diff changeset
1118 if (ep->can_eliminate && ! prev)
kono
parents:
diff changeset
1119 {
kono
parents:
diff changeset
1120 /* It is possible that not eliminable register becomes
kono
parents:
diff changeset
1121 eliminable because we took other reasons into account to
kono
parents:
diff changeset
1122 set up eliminable regs in the initial set up. Just
kono
parents:
diff changeset
1123 ignore new eliminable registers. */
kono
parents:
diff changeset
1124 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
1125 continue;
kono
parents:
diff changeset
1126 }
kono
parents:
diff changeset
1127 if (ep->can_eliminate != prev && elimination_map[ep->from] == ep)
kono
parents:
diff changeset
1128 {
kono
parents:
diff changeset
1129 /* We cannot use this elimination anymore -- find another
kono
parents:
diff changeset
1130 one. */
kono
parents:
diff changeset
1131 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1132 fprintf (lra_dump_file,
kono
parents:
diff changeset
1133 " Elimination %d to %d is not possible anymore\n",
kono
parents:
diff changeset
1134 ep->from, ep->to);
kono
parents:
diff changeset
1135 /* If after processing RTL we decides that SP can be used as
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1136 a result of elimination, it cannot be changed. */
111
kono
parents:
diff changeset
1137 gcc_assert ((ep->to_rtx != stack_pointer_rtx)
kono
parents:
diff changeset
1138 || (ep->from < FIRST_PSEUDO_REGISTER
kono
parents:
diff changeset
1139 && fixed_regs [ep->from]));
kono
parents:
diff changeset
1140 /* Mark that is not eliminable anymore. */
kono
parents:
diff changeset
1141 elimination_map[ep->from] = NULL;
kono
parents:
diff changeset
1142 for (ep1 = ep + 1; ep1 < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep1++)
kono
parents:
diff changeset
1143 if (ep1->can_eliminate && ep1->from == ep->from)
kono
parents:
diff changeset
1144 break;
kono
parents:
diff changeset
1145 if (ep1 < &reg_eliminate[NUM_ELIMINABLE_REGS])
kono
parents:
diff changeset
1146 {
kono
parents:
diff changeset
1147 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1148 fprintf (lra_dump_file, " Using elimination %d to %d now\n",
kono
parents:
diff changeset
1149 ep1->from, ep1->to);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1150 lra_assert (known_eq (ep1->previous_offset, 0));
111
kono
parents:
diff changeset
1151 ep1->previous_offset = ep->offset;
kono
parents:
diff changeset
1152 }
kono
parents:
diff changeset
1153 else
kono
parents:
diff changeset
1154 {
kono
parents:
diff changeset
1155 /* There is no elimination anymore just use the hard
kono
parents:
diff changeset
1156 register `from' itself. Setup self elimination
kono
parents:
diff changeset
1157 offset to restore the original offset values. */
kono
parents:
diff changeset
1158 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1159 fprintf (lra_dump_file, " %d is not eliminable at all\n",
kono
parents:
diff changeset
1160 ep->from);
kono
parents:
diff changeset
1161 self_elim_offsets[ep->from] = -ep->offset;
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1162 if (maybe_ne (ep->offset, 0))
111
kono
parents:
diff changeset
1163 bitmap_ior_into (insns_with_changed_offsets,
kono
parents:
diff changeset
1164 &lra_reg_info[ep->from].insn_bitmap);
kono
parents:
diff changeset
1165 }
kono
parents:
diff changeset
1166 }
kono
parents:
diff changeset
1167
kono
parents:
diff changeset
1168 INITIAL_ELIMINATION_OFFSET (ep->from, ep->to, ep->offset);
kono
parents:
diff changeset
1169 }
kono
parents:
diff changeset
1170 setup_elimination_map ();
kono
parents:
diff changeset
1171 result = false;
kono
parents:
diff changeset
1172 CLEAR_HARD_REG_SET (temp_hard_reg_set);
kono
parents:
diff changeset
1173 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1174 if (elimination_map[ep->from] == NULL)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1175 add_to_hard_reg_set (&temp_hard_reg_set, Pmode, ep->from);
111
kono
parents:
diff changeset
1176 else if (elimination_map[ep->from] == ep)
kono
parents:
diff changeset
1177 {
kono
parents:
diff changeset
1178 /* Prevent the hard register into which we eliminate from
kono
parents:
diff changeset
1179 the usage for pseudos. */
kono
parents:
diff changeset
1180 if (ep->from != ep->to)
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1181 add_to_hard_reg_set (&temp_hard_reg_set, Pmode, ep->to);
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1182 if (maybe_ne (ep->previous_offset, ep->offset))
111
kono
parents:
diff changeset
1183 {
kono
parents:
diff changeset
1184 bitmap_ior_into (insns_with_changed_offsets,
kono
parents:
diff changeset
1185 &lra_reg_info[ep->from].insn_bitmap);
kono
parents:
diff changeset
1186
kono
parents:
diff changeset
1187 /* Update offset when the eliminate offset have been
kono
parents:
diff changeset
1188 changed. */
kono
parents:
diff changeset
1189 lra_update_reg_val_offset (lra_reg_info[ep->from].val,
kono
parents:
diff changeset
1190 ep->offset - ep->previous_offset);
kono
parents:
diff changeset
1191 result = true;
kono
parents:
diff changeset
1192 }
kono
parents:
diff changeset
1193 }
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1194 lra_no_alloc_regs |= temp_hard_reg_set;
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1195 eliminable_regset &= ~temp_hard_reg_set;
111
kono
parents:
diff changeset
1196 spill_pseudos (temp_hard_reg_set);
kono
parents:
diff changeset
1197 return result;
kono
parents:
diff changeset
1198 }
kono
parents:
diff changeset
1199
kono
parents:
diff changeset
1200 /* Initialize the table of hard registers to eliminate.
kono
parents:
diff changeset
1201 Pre-condition: global flag frame_pointer_needed has been set before
kono
parents:
diff changeset
1202 calling this function. */
kono
parents:
diff changeset
1203 static void
kono
parents:
diff changeset
1204 init_elim_table (void)
kono
parents:
diff changeset
1205 {
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1206 class lra_elim_table *ep;
111
kono
parents:
diff changeset
1207 bool value_p;
kono
parents:
diff changeset
1208 const struct elim_table_1 *ep1;
kono
parents:
diff changeset
1209
kono
parents:
diff changeset
1210 if (!reg_eliminate)
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1211 reg_eliminate = XCNEWVEC (class lra_elim_table, NUM_ELIMINABLE_REGS);
111
kono
parents:
diff changeset
1212
kono
parents:
diff changeset
1213 memset (self_elim_offsets, 0, sizeof (self_elim_offsets));
kono
parents:
diff changeset
1214 /* Initiate member values which will be never changed. */
kono
parents:
diff changeset
1215 self_elim_table.can_eliminate = self_elim_table.prev_can_eliminate = true;
kono
parents:
diff changeset
1216 self_elim_table.previous_offset = 0;
kono
parents:
diff changeset
1217
kono
parents:
diff changeset
1218 for (ep = reg_eliminate, ep1 = reg_eliminate_1;
kono
parents:
diff changeset
1219 ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++, ep1++)
kono
parents:
diff changeset
1220 {
kono
parents:
diff changeset
1221 ep->offset = ep->previous_offset = 0;
kono
parents:
diff changeset
1222 ep->from = ep1->from;
kono
parents:
diff changeset
1223 ep->to = ep1->to;
kono
parents:
diff changeset
1224 value_p = (targetm.can_eliminate (ep->from, ep->to)
kono
parents:
diff changeset
1225 && ! (ep->to == STACK_POINTER_REGNUM
kono
parents:
diff changeset
1226 && frame_pointer_needed
kono
parents:
diff changeset
1227 && (! SUPPORTS_STACK_ALIGNMENT
kono
parents:
diff changeset
1228 || ! stack_realign_fp)));
kono
parents:
diff changeset
1229 setup_can_eliminate (ep, value_p);
kono
parents:
diff changeset
1230 }
kono
parents:
diff changeset
1231
kono
parents:
diff changeset
1232 /* Build the FROM and TO REG rtx's. Note that code in gen_rtx_REG
kono
parents:
diff changeset
1233 will cause, e.g., gen_rtx_REG (Pmode, STACK_POINTER_REGNUM) to
kono
parents:
diff changeset
1234 equal stack_pointer_rtx. We depend on this. Threfore we switch
kono
parents:
diff changeset
1235 off that we are in LRA temporarily. */
kono
parents:
diff changeset
1236 lra_in_progress = 0;
kono
parents:
diff changeset
1237 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1238 {
kono
parents:
diff changeset
1239 ep->from_rtx = gen_rtx_REG (Pmode, ep->from);
kono
parents:
diff changeset
1240 ep->to_rtx = gen_rtx_REG (Pmode, ep->to);
kono
parents:
diff changeset
1241 eliminable_reg_rtx[ep->from] = ep->from_rtx;
kono
parents:
diff changeset
1242 }
kono
parents:
diff changeset
1243 lra_in_progress = 1;
kono
parents:
diff changeset
1244 }
kono
parents:
diff changeset
1245
kono
parents:
diff changeset
1246 /* Function for initialization of elimination once per function. It
kono
parents:
diff changeset
1247 sets up sp offset for each insn. */
kono
parents:
diff changeset
1248 static void
kono
parents:
diff changeset
1249 init_elimination (void)
kono
parents:
diff changeset
1250 {
kono
parents:
diff changeset
1251 bool stop_to_sp_elimination_p;
kono
parents:
diff changeset
1252 basic_block bb;
kono
parents:
diff changeset
1253 rtx_insn *insn;
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1254 class lra_elim_table *ep;
111
kono
parents:
diff changeset
1255
kono
parents:
diff changeset
1256 init_elim_table ();
kono
parents:
diff changeset
1257 FOR_EACH_BB_FN (bb, cfun)
kono
parents:
diff changeset
1258 {
kono
parents:
diff changeset
1259 curr_sp_change = 0;
kono
parents:
diff changeset
1260 stop_to_sp_elimination_p = false;
kono
parents:
diff changeset
1261 FOR_BB_INSNS (bb, insn)
kono
parents:
diff changeset
1262 if (INSN_P (insn))
kono
parents:
diff changeset
1263 {
kono
parents:
diff changeset
1264 lra_get_insn_recog_data (insn)->sp_offset = curr_sp_change;
kono
parents:
diff changeset
1265 if (NONDEBUG_INSN_P (insn))
kono
parents:
diff changeset
1266 {
kono
parents:
diff changeset
1267 mark_not_eliminable (PATTERN (insn), VOIDmode);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1268 if (maybe_ne (curr_sp_change, 0)
111
kono
parents:
diff changeset
1269 && find_reg_note (insn, REG_LABEL_OPERAND, NULL_RTX))
kono
parents:
diff changeset
1270 stop_to_sp_elimination_p = true;
kono
parents:
diff changeset
1271 }
kono
parents:
diff changeset
1272 }
kono
parents:
diff changeset
1273 if (! frame_pointer_needed
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1274 && (maybe_ne (curr_sp_change, 0) || stop_to_sp_elimination_p)
111
kono
parents:
diff changeset
1275 && bb->succs && bb->succs->length () != 0)
kono
parents:
diff changeset
1276 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1277 if (ep->to == STACK_POINTER_REGNUM)
kono
parents:
diff changeset
1278 setup_can_eliminate (ep, false);
kono
parents:
diff changeset
1279 }
kono
parents:
diff changeset
1280 setup_elimination_map ();
kono
parents:
diff changeset
1281 }
kono
parents:
diff changeset
1282
kono
parents:
diff changeset
1283 /* Eliminate hard reg given by its location LOC. */
kono
parents:
diff changeset
1284 void
kono
parents:
diff changeset
1285 lra_eliminate_reg_if_possible (rtx *loc)
kono
parents:
diff changeset
1286 {
kono
parents:
diff changeset
1287 int regno;
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1288 class lra_elim_table *ep;
111
kono
parents:
diff changeset
1289
kono
parents:
diff changeset
1290 lra_assert (REG_P (*loc));
kono
parents:
diff changeset
1291 if ((regno = REGNO (*loc)) >= FIRST_PSEUDO_REGISTER
kono
parents:
diff changeset
1292 || ! TEST_HARD_REG_BIT (lra_no_alloc_regs, regno))
kono
parents:
diff changeset
1293 return;
kono
parents:
diff changeset
1294 if ((ep = get_elimination (*loc)) != NULL)
kono
parents:
diff changeset
1295 *loc = ep->to_rtx;
kono
parents:
diff changeset
1296 }
kono
parents:
diff changeset
1297
kono
parents:
diff changeset
1298 /* Do (final if FINAL_P or first if FIRST_P) elimination in INSN. Add
kono
parents:
diff changeset
1299 the insn for subsequent processing in the constraint pass, update
kono
parents:
diff changeset
1300 the insn info. */
kono
parents:
diff changeset
1301 static void
kono
parents:
diff changeset
1302 process_insn_for_elimination (rtx_insn *insn, bool final_p, bool first_p)
kono
parents:
diff changeset
1303 {
kono
parents:
diff changeset
1304 eliminate_regs_in_insn (insn, final_p, first_p, 0);
kono
parents:
diff changeset
1305 if (! final_p)
kono
parents:
diff changeset
1306 {
kono
parents:
diff changeset
1307 /* Check that insn changed its code. This is a case when a move
kono
parents:
diff changeset
1308 insn becomes an add insn and we do not want to process the
kono
parents:
diff changeset
1309 insn as a move anymore. */
kono
parents:
diff changeset
1310 int icode = recog (PATTERN (insn), insn, 0);
kono
parents:
diff changeset
1311
kono
parents:
diff changeset
1312 if (icode >= 0 && icode != INSN_CODE (insn))
kono
parents:
diff changeset
1313 {
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1314 if (INSN_CODE (insn) >= 0)
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1315 /* Insn code is changed. It may change its operand type
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1316 from IN to INOUT. Inform the subsequent assignment
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1317 subpass about this situation. */
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1318 check_and_force_assignment_correctness_p = true;
111
kono
parents:
diff changeset
1319 INSN_CODE (insn) = icode;
kono
parents:
diff changeset
1320 lra_update_insn_recog_data (insn);
kono
parents:
diff changeset
1321 }
kono
parents:
diff changeset
1322 lra_update_insn_regno_info (insn);
kono
parents:
diff changeset
1323 lra_push_insn (insn);
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1324 lra_set_used_insn_alternative (insn, LRA_UNKNOWN_ALT);
111
kono
parents:
diff changeset
1325 }
kono
parents:
diff changeset
1326 }
kono
parents:
diff changeset
1327
kono
parents:
diff changeset
1328 /* Entry function to do final elimination if FINAL_P or to update
kono
parents:
diff changeset
1329 elimination register offsets (FIRST_P if we are doing it the first
kono
parents:
diff changeset
1330 time). */
kono
parents:
diff changeset
1331 void
kono
parents:
diff changeset
1332 lra_eliminate (bool final_p, bool first_p)
kono
parents:
diff changeset
1333 {
kono
parents:
diff changeset
1334 unsigned int uid;
kono
parents:
diff changeset
1335 bitmap_head insns_with_changed_offsets;
kono
parents:
diff changeset
1336 bitmap_iterator bi;
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
1337 class lra_elim_table *ep;
111
kono
parents:
diff changeset
1338
kono
parents:
diff changeset
1339 gcc_assert (! final_p || ! first_p);
kono
parents:
diff changeset
1340
kono
parents:
diff changeset
1341 timevar_push (TV_LRA_ELIMINATE);
kono
parents:
diff changeset
1342
kono
parents:
diff changeset
1343 if (first_p)
kono
parents:
diff changeset
1344 init_elimination ();
kono
parents:
diff changeset
1345
kono
parents:
diff changeset
1346 bitmap_initialize (&insns_with_changed_offsets, &reg_obstack);
kono
parents:
diff changeset
1347 if (final_p)
kono
parents:
diff changeset
1348 {
kono
parents:
diff changeset
1349 if (flag_checking)
kono
parents:
diff changeset
1350 {
kono
parents:
diff changeset
1351 update_reg_eliminate (&insns_with_changed_offsets);
kono
parents:
diff changeset
1352 gcc_assert (bitmap_empty_p (&insns_with_changed_offsets));
kono
parents:
diff changeset
1353 }
kono
parents:
diff changeset
1354 /* We change eliminable hard registers in insns so we should do
kono
parents:
diff changeset
1355 this for all insns containing any eliminable hard
kono
parents:
diff changeset
1356 register. */
kono
parents:
diff changeset
1357 for (ep = reg_eliminate; ep < &reg_eliminate[NUM_ELIMINABLE_REGS]; ep++)
kono
parents:
diff changeset
1358 if (elimination_map[ep->from] != NULL)
kono
parents:
diff changeset
1359 bitmap_ior_into (&insns_with_changed_offsets,
kono
parents:
diff changeset
1360 &lra_reg_info[ep->from].insn_bitmap);
kono
parents:
diff changeset
1361 }
kono
parents:
diff changeset
1362 else if (! update_reg_eliminate (&insns_with_changed_offsets))
kono
parents:
diff changeset
1363 goto lra_eliminate_done;
kono
parents:
diff changeset
1364 if (lra_dump_file != NULL)
kono
parents:
diff changeset
1365 {
kono
parents:
diff changeset
1366 fprintf (lra_dump_file, "New elimination table:\n");
kono
parents:
diff changeset
1367 print_elim_table (lra_dump_file);
kono
parents:
diff changeset
1368 }
kono
parents:
diff changeset
1369 EXECUTE_IF_SET_IN_BITMAP (&insns_with_changed_offsets, 0, uid, bi)
kono
parents:
diff changeset
1370 /* A dead insn can be deleted in process_insn_for_elimination. */
kono
parents:
diff changeset
1371 if (lra_insn_recog_data[uid] != NULL)
kono
parents:
diff changeset
1372 process_insn_for_elimination (lra_insn_recog_data[uid]->insn,
kono
parents:
diff changeset
1373 final_p, first_p);
kono
parents:
diff changeset
1374 bitmap_clear (&insns_with_changed_offsets);
kono
parents:
diff changeset
1375
kono
parents:
diff changeset
1376 lra_eliminate_done:
kono
parents:
diff changeset
1377 timevar_pop (TV_LRA_ELIMINATE);
kono
parents:
diff changeset
1378 }