OSDN Git Service

PR debug/43670
[pf3gnuchains/gcc-fork.git] / gcc / web.c
1 /* Web construction code for GNU compiler.
2    Contributed by Jan Hubicka.
3    Copyright (C) 2001, 2002, 2004, 2006, 2007, 2008, 2010
4    Free Software Foundation, Inc.
5
6 This file is part of GCC.
7
8 GCC is free software; you can redistribute it and/or modify it under
9 the terms of the GNU General Public License as published by the Free
10 Software Foundation; either version 3, or (at your option) any later
11 version.
12
13 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
14 WARRANTY; without even the implied warranty of MERCHANTABILITY or
15 FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
16 for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with GCC; see the file COPYING3.  If not see
20 <http://www.gnu.org/licenses/>.  */
21
22 /* Simple optimization pass that splits independent uses of each pseudo,
23    increasing effectiveness of other optimizations.  The optimization can
24    serve as an example of use for the dataflow module.
25
26    We don't split registers with REG_USERVAR set unless -fmessy-debugging
27    is specified, because debugging information about such split variables
28    is almost unusable.
29
30    TODO
31     - We may use profile information and ignore infrequent use for the
32       purpose of web unifying, inserting the compensation code later to
33       implement full induction variable expansion for loops (currently
34       we expand only if the induction variable is dead afterward, which
35       is often the case).  */
36
37 #include "config.h"
38 #include "system.h"
39 #include "coretypes.h"
40 #include "tm.h"
41 #include "toplev.h"
42
43 #include "rtl.h"
44 #include "hard-reg-set.h"
45 #include "flags.h"
46 #include "obstack.h"
47 #include "basic-block.h"
48 #include "output.h"
49 #include "df.h"
50 #include "function.h"
51 #include "insn-config.h"
52 #include "recog.h"
53 #include "timevar.h"
54 #include "tree-pass.h"
55
56
57 /* Find the root of unionfind tree (the representative of set).  */
58
59 struct web_entry *
60 unionfind_root (struct web_entry *element)
61 {
62   struct web_entry *element1 = element, *element2;
63
64   while (element->pred)
65     element = element->pred;
66   while (element1->pred)
67     {
68       element2 = element1->pred;
69       element1->pred = element;
70       element1 = element2;
71     }
72   return element;
73 }
74
75 /* Union sets.
76    Return true if FIRST and SECOND points to the same web entry structure and
77    nothing is done.  Otherwise, return false.  */
78
79 bool
80 unionfind_union (struct web_entry *first, struct web_entry *second)
81 {
82   first = unionfind_root (first);
83   second = unionfind_root (second);
84   if (first == second)
85     return true;
86   second->pred = first;
87   return false;
88 }
89
90 /* For INSN, union all defs and uses that are linked by match_dup.
91    FUN is the function that does the union.  */
92
93 static void
94 union_match_dups (rtx insn, struct web_entry *def_entry,
95                   struct web_entry *use_entry,
96                   bool (*fun) (struct web_entry *, struct web_entry *))
97 {
98   struct df_insn_info *insn_info = DF_INSN_INFO_GET (insn);
99   df_ref *use_link = DF_INSN_INFO_USES (insn_info);
100   df_ref *def_link = DF_INSN_INFO_DEFS (insn_info);
101   int i;
102
103   extract_insn (insn);
104
105   for (i = 0; i < recog_data.n_dups; i++)
106     {
107       int op = recog_data.dup_num[i];
108       enum op_type type = recog_data.operand_type[op];
109       df_ref *ref, *dupref;
110       struct web_entry *entry;
111
112       for (dupref = use_link; *dupref; dupref++)
113         if (DF_REF_LOC (*dupref) == recog_data.dup_loc[i])
114           break;
115
116       if (*dupref == NULL
117           || DF_REF_REGNO (*dupref) < FIRST_PSEUDO_REGISTER)
118         continue;
119
120       ref = type == OP_IN ? use_link : def_link;
121       entry = type == OP_IN ? use_entry : def_entry;
122       for (; *ref; ref++)
123         if (DF_REF_LOC (*ref) == recog_data.operand_loc[op])
124           break;
125
126       (*fun) (use_entry + DF_REF_ID (*dupref), entry + DF_REF_ID (*ref));
127     }
128 }
129
130 /* For each use, all possible defs reaching it must come in the same
131    register, union them.
132    FUN is the function that does the union.
133
134    In USED, we keep the DF_REF_ID of the first uninitialized uses of a
135    register, so that all uninitialized uses of the register can be
136    combined into a single web.  We actually offset it by 2, because
137    the values 0 and 1 are reserved for use by entry_register.  */
138
139 void
140 union_defs (df_ref use, struct web_entry *def_entry,
141             unsigned int *used, struct web_entry *use_entry,
142             bool (*fun) (struct web_entry *, struct web_entry *))
143 {
144   struct df_insn_info *insn_info = DF_REF_INSN_INFO (use);
145   struct df_link *link = DF_REF_CHAIN (use);
146   df_ref *eq_use_link;
147   df_ref *def_link;
148   rtx set;
149
150   if (insn_info)
151     {
152       rtx insn = insn_info->insn;
153       eq_use_link = DF_INSN_INFO_EQ_USES (insn_info);
154       def_link = DF_INSN_INFO_DEFS (insn_info);
155       set = single_set (insn);
156     }
157   else
158     {
159       /* An artificial use.  It links up with nothing.  */
160       eq_use_link = NULL;
161       def_link = NULL;
162       set = NULL;
163     }
164
165   /* Union all occurrences of the same register in reg notes.  */
166
167   if (eq_use_link)
168     while (*eq_use_link)
169       {
170         if (use != *eq_use_link
171             && DF_REF_REAL_REG (use) == DF_REF_REAL_REG (*eq_use_link))
172           (*fun) (use_entry + DF_REF_ID (use),
173                   use_entry + DF_REF_ID (*eq_use_link));
174         eq_use_link++;
175     }
176
177   /* Recognize trivial noop moves and attempt to keep them as noop.  */
178
179   if (set
180       && SET_SRC (set) == DF_REF_REG (use)
181       && SET_SRC (set) == SET_DEST (set))
182     {
183       if (def_link)
184         while (*def_link)
185           {
186             if (DF_REF_REAL_REG (use) == DF_REF_REAL_REG (*def_link))
187               (*fun) (use_entry + DF_REF_ID (use),
188                       def_entry + DF_REF_ID (*def_link));
189             def_link++;
190           }
191     }
192
193   /* UD chains of uninitialized REGs are empty.  Keeping all uses of
194      the same uninitialized REG in a single web is not necessary for
195      correctness, since the uses are undefined, but it's wasteful to
196      allocate one register or slot for each reference.  Furthermore,
197      creating new pseudos for uninitialized references in debug insns
198      (see PR 42631) causes -fcompare-debug failures.  We record the
199      number of the first uninitialized reference we found, and merge
200      with it any other uninitialized references to the same
201      register.  */
202   if (!link)
203     {
204       int regno = REGNO (DF_REF_REAL_REG (use));
205       if (used[regno])
206         (*fun) (use_entry + DF_REF_ID (use), use_entry + used[regno] - 2);
207       else
208         used[regno] = DF_REF_ID (use) + 2;
209     }
210
211   while (link)
212     {
213       (*fun) (use_entry + DF_REF_ID (use),
214               def_entry + DF_REF_ID (link->ref));
215       link = link->next;
216     }
217
218   /* A READ_WRITE use requires the corresponding def to be in the same
219      register.  Find it and union.  */
220   if (DF_REF_FLAGS (use) & DF_REF_READ_WRITE)
221     {
222       df_ref *link;
223
224       if (insn_info)
225         link = DF_INSN_INFO_DEFS (insn_info);
226       else
227         link = NULL;
228
229       if (link)
230         while (*link)
231           {
232             if (DF_REF_REAL_REG (*link) == DF_REF_REAL_REG (use))
233               (*fun) (use_entry + DF_REF_ID (use),
234                       def_entry + DF_REF_ID (*link));
235             link++;
236           }
237     }
238 }
239
240 /* Find the corresponding register for the given entry.  */
241
242 static rtx
243 entry_register (struct web_entry *entry, df_ref ref, unsigned int *used)
244 {
245   struct web_entry *root;
246   rtx reg, newreg;
247
248   /* Find the corresponding web and see if it has been visited.  */
249   root = unionfind_root (entry);
250   if (root->reg)
251     return root->reg;
252
253   /* We are seeing this web for the first time, do the assignment.  */
254   reg = DF_REF_REAL_REG (ref);
255
256   /* In case the original register is already assigned, generate new
257      one.  Since we use USED to merge uninitialized refs into a single
258      web, we might found an element to be nonzero without our having
259      used it.  Test for 1, because union_defs saves it for our use,
260      and there won't be any use for the other values when we get to
261      this point.  */
262   if (used[REGNO (reg)] != 1)
263     newreg = reg, used[REGNO (reg)] = 1;
264   else
265     {
266       newreg = gen_reg_rtx (GET_MODE (reg));
267       REG_USERVAR_P (newreg) = REG_USERVAR_P (reg);
268       REG_POINTER (newreg) = REG_POINTER (reg);
269       REG_ATTRS (newreg) = REG_ATTRS (reg);
270       if (dump_file)
271         fprintf (dump_file, "Web oldreg=%i newreg=%i\n", REGNO (reg),
272                  REGNO (newreg));
273     }
274
275   root->reg = newreg;
276   return newreg;
277 }
278
279 /* Replace the reference by REG.  */
280
281 static void
282 replace_ref (df_ref ref, rtx reg)
283 {
284   rtx oldreg = DF_REF_REAL_REG (ref);
285   rtx *loc = DF_REF_REAL_LOC (ref);
286   unsigned int uid = DF_REF_INSN_UID (ref);
287
288   if (oldreg == reg)
289     return;
290   if (dump_file)
291     fprintf (dump_file, "Updating insn %i (%i->%i)\n",
292              uid, REGNO (oldreg), REGNO (reg));
293   *loc = reg;
294   df_insn_rescan (DF_REF_INSN (ref));
295 }
296
297 \f
298 static bool
299 gate_handle_web (void)
300 {
301   return (optimize > 0 && flag_web);
302 }
303
304 /* Main entry point.  */
305
306 static unsigned int
307 web_main (void)
308 {
309   struct web_entry *def_entry;
310   struct web_entry *use_entry;
311   unsigned int max = max_reg_num ();
312   unsigned int *used;
313   basic_block bb;
314   unsigned int uses_num = 0;
315   rtx insn;
316
317   df_set_flags (DF_NO_HARD_REGS + DF_EQ_NOTES);
318   df_chain_add_problem (DF_UD_CHAIN);
319   df_analyze ();
320   df_set_flags (DF_DEFER_INSN_RESCAN);
321
322   /* Assign ids to the uses.  */
323   FOR_ALL_BB (bb)
324     FOR_BB_INSNS (bb, insn)
325     {
326       unsigned int uid = INSN_UID (insn);
327       if (NONDEBUG_INSN_P (insn))
328         {
329           df_ref *use_rec;
330           for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
331             {
332               df_ref use = *use_rec;
333               if (DF_REF_REGNO (use) >= FIRST_PSEUDO_REGISTER)
334                 DF_REF_ID (use) = uses_num++;
335             }
336           for (use_rec = DF_INSN_UID_EQ_USES (uid); *use_rec; use_rec++)
337             {
338               df_ref use = *use_rec;
339               if (DF_REF_REGNO (use) >= FIRST_PSEUDO_REGISTER)
340                 DF_REF_ID (use) = uses_num++;
341             }
342         }
343     }
344
345   /* Record the number of uses and defs at the beginning of the optimization.  */
346   def_entry = XCNEWVEC (struct web_entry, DF_DEFS_TABLE_SIZE());
347   used = XCNEWVEC (unsigned, max);
348   use_entry = XCNEWVEC (struct web_entry, uses_num);
349
350   /* Produce the web.  */
351   FOR_ALL_BB (bb)
352     FOR_BB_INSNS (bb, insn)
353     {
354       unsigned int uid = INSN_UID (insn);
355       if (NONDEBUG_INSN_P (insn))
356         {
357           df_ref *use_rec;
358           union_match_dups (insn, def_entry, use_entry, unionfind_union);
359           for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
360             {
361               df_ref use = *use_rec;
362               if (DF_REF_REGNO (use) >= FIRST_PSEUDO_REGISTER)
363                 union_defs (use, def_entry, used, use_entry, unionfind_union);
364             }
365           for (use_rec = DF_INSN_UID_EQ_USES (uid); *use_rec; use_rec++)
366             {
367               df_ref use = *use_rec;
368               if (DF_REF_REGNO (use) >= FIRST_PSEUDO_REGISTER)
369                 union_defs (use, def_entry, used, use_entry, unionfind_union);
370             }
371         }
372     }
373
374   /* Update the instruction stream, allocating new registers for split pseudos
375      in progress.  */
376   FOR_ALL_BB (bb)
377     FOR_BB_INSNS (bb, insn)
378     {
379       unsigned int uid = INSN_UID (insn);
380       if (NONDEBUG_INSN_P (insn))
381         {
382           df_ref *use_rec;
383           df_ref *def_rec;
384           for (use_rec = DF_INSN_UID_USES (uid); *use_rec; use_rec++)
385             {
386               df_ref use = *use_rec;
387               if (DF_REF_REGNO (use) >= FIRST_PSEUDO_REGISTER)
388                 replace_ref (use, entry_register (use_entry + DF_REF_ID (use), use, used));
389             }
390           for (use_rec = DF_INSN_UID_EQ_USES (uid); *use_rec; use_rec++)
391             {
392               df_ref use = *use_rec;
393               if (DF_REF_REGNO (use) >= FIRST_PSEUDO_REGISTER)
394                 replace_ref (use, entry_register (use_entry + DF_REF_ID (use), use, used));
395             }
396           for (def_rec = DF_INSN_UID_DEFS (uid); *def_rec; def_rec++)
397             {
398               df_ref def = *def_rec;
399               if (DF_REF_REGNO (def) >= FIRST_PSEUDO_REGISTER)
400                 replace_ref (def, entry_register (def_entry + DF_REF_ID (def), def, used));
401             }
402         }
403     }
404
405   free (def_entry);
406   free (use_entry);
407   free (used);
408   return 0;
409 }
410 \f
411 struct rtl_opt_pass pass_web =
412 {
413  {
414   RTL_PASS,
415   "web",                                /* name */
416   gate_handle_web,                      /* gate */
417   web_main,                             /* execute */
418   NULL,                                 /* sub */
419   NULL,                                 /* next */
420   0,                                    /* static_pass_number */
421   TV_WEB,                               /* tv_id */
422   0,                                    /* properties_required */
423   0,                                    /* properties_provided */
424   0,                                    /* properties_destroyed */
425   0,                                    /* todo_flags_start */
426   TODO_df_finish | TODO_verify_rtl_sharing |
427   TODO_dump_func                        /* todo_flags_finish */
428  }
429 };
430