OpenCores
URL https://opencores.org/ocsvn/openrisc_me/openrisc_me/trunk

Subversion Repositories openrisc_me

[/] [openrisc/] [trunk/] [gnu-src/] [gcc-4.2.2/] [gcc/] [tree-ssa-structalias.c] - Blame information for rev 192

Go to most recent revision | Details | Compare with Previous | View Log

Line No. Rev Author Line
1 38 julius
/* Tree based points-to analysis
2
   Copyright (C) 2005, 2006 Free Software Foundation, Inc.
3
   Contributed by Daniel Berlin <dberlin@dberlin.org>
4
 
5
   This file is part of GCC.
6
 
7
   GCC is free software; you can redistribute it and/or modify
8
   under the terms of the GNU General Public License as published by
9
   the Free Software Foundation; either version 3 of the License, or
10
   (at your option) any later version.
11
 
12
   GCC is distributed in the hope that it will be useful,
13
   but WITHOUT ANY WARRANTY; without even the implied warranty of
14
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15
   GNU General Public License for more details.
16
 
17
   You should have received a copy of the GNU General Public License
18
   along with GCC; see the file COPYING3.  If not see
19
   <http://www.gnu.org/licenses/>.  */
20
 
21
#include "config.h"
22
#include "system.h"
23
#include "coretypes.h"
24
#include "tm.h"
25
#include "ggc.h"
26
#include "obstack.h"
27
#include "bitmap.h"
28
#include "flags.h"
29
#include "rtl.h"
30
#include "tm_p.h"
31
#include "hard-reg-set.h"
32
#include "basic-block.h"
33
#include "output.h"
34
#include "errors.h"
35
#include "diagnostic.h"
36
#include "tree.h"
37
#include "c-common.h"
38
#include "tree-flow.h"
39
#include "tree-inline.h"
40
#include "varray.h"
41
#include "c-tree.h"
42
#include "tree-gimple.h"
43
#include "hashtab.h"
44
#include "function.h"
45
#include "cgraph.h"
46
#include "tree-pass.h"
47
#include "timevar.h"
48
#include "alloc-pool.h"
49
#include "splay-tree.h"
50
#include "params.h"
51
#include "tree-ssa-structalias.h"
52
#include "cgraph.h"
53
#include "pointer-set.h"
54
 
55
/* The idea behind this analyzer is to generate set constraints from the
56
   program, then solve the resulting constraints in order to generate the
57
   points-to sets.
58
 
59
   Set constraints are a way of modeling program analysis problems that
60
   involve sets.  They consist of an inclusion constraint language,
61
   describing the variables (each variable is a set) and operations that
62
   are involved on the variables, and a set of rules that derive facts
63
   from these operations.  To solve a system of set constraints, you derive
64
   all possible facts under the rules, which gives you the correct sets
65
   as a consequence.
66
 
67
   See  "Efficient Field-sensitive pointer analysis for C" by "David
68
   J. Pearce and Paul H. J. Kelly and Chris Hankin, at
69
   http://citeseer.ist.psu.edu/pearce04efficient.html
70
 
71
   Also see "Ultra-fast Aliasing Analysis using CLA: A Million Lines
72
   of C Code in a Second" by ""Nevin Heintze and Olivier Tardieu" at
73
   http://citeseer.ist.psu.edu/heintze01ultrafast.html
74
 
75
   There are three types of real constraint expressions, DEREF,
76
   ADDRESSOF, and SCALAR.  Each constraint expression consists
77
   of a constraint type, a variable, and an offset.
78
 
79
   SCALAR is a constraint expression type used to represent x, whether
80
   it appears on the LHS or the RHS of a statement.
81
   DEREF is a constraint expression type used to represent *x, whether
82
   it appears on the LHS or the RHS of a statement.
83
   ADDRESSOF is a constraint expression used to represent &x, whether
84
   it appears on the LHS or the RHS of a statement.
85
 
86
   Each pointer variable in the program is assigned an integer id, and
87
   each field of a structure variable is assigned an integer id as well.
88
 
89
   Structure variables are linked to their list of fields through a "next
90
   field" in each variable that points to the next field in offset
91
   order.
92
   Each variable for a structure field has
93
 
94
   1. "size", that tells the size in bits of that field.
95
   2. "fullsize, that tells the size in bits of the entire structure.
96
   3. "offset", that tells the offset in bits from the beginning of the
97
   structure to this field.
98
 
99
   Thus,
100
   struct f
101
   {
102
     int a;
103
     int b;
104
   } foo;
105
   int *bar;
106
 
107
   looks like
108
 
109
   foo.a -> id 1, size 32, offset 0, fullsize 64, next foo.b
110
   foo.b -> id 2, size 32, offset 32, fullsize 64, next NULL
111
   bar -> id 3, size 32, offset 0, fullsize 32, next NULL
112
 
113
 
114
  In order to solve the system of set constraints, the following is
115
  done:
116
 
117
  1. Each constraint variable x has a solution set associated with it,
118
  Sol(x).
119
 
120
  2. Constraints are separated into direct, copy, and complex.
121
  Direct constraints are ADDRESSOF constraints that require no extra
122
  processing, such as P = &Q
123
  Copy constraints are those of the form P = Q.
124
  Complex constraints are all the constraints involving dereferences
125
  and offsets (including offsetted copies).
126
 
127
  3. All direct constraints of the form P = &Q are processed, such
128
  that Q is added to Sol(P)
129
 
130
  4. All complex constraints for a given constraint variable are stored in a
131
  linked list attached to that variable's node.
132
 
133
  5. A directed graph is built out of the copy constraints. Each
134
  constraint variable is a node in the graph, and an edge from
135
  Q to P is added for each copy constraint of the form P = Q
136
 
137
  6. The graph is then walked, and solution sets are
138
  propagated along the copy edges, such that an edge from Q to P
139
  causes Sol(P) <- Sol(P) union Sol(Q).
140
 
141
  7.  As we visit each node, all complex constraints associated with
142
  that node are processed by adding appropriate copy edges to the graph, or the
143
  appropriate variables to the solution set.
144
 
145
  8. The process of walking the graph is iterated until no solution
146
  sets change.
147
 
148
  Prior to walking the graph in steps 6 and 7, We perform static
149
  cycle elimination on the constraint graph, as well
150
  as off-line variable substitution.
151
 
152
  TODO: Adding offsets to pointer-to-structures can be handled (IE not punted
153
  on and turned into anything), but isn't.  You can just see what offset
154
  inside the pointed-to struct it's going to access.
155
 
156
  TODO: Constant bounded arrays can be handled as if they were structs of the
157
  same number of elements.
158
 
159
  TODO: Modeling heap and incoming pointers becomes much better if we
160
  add fields to them as we discover them, which we could do.
161
 
162
  TODO: We could handle unions, but to be honest, it's probably not
163
  worth the pain or slowdown.  */
164
 
165
static GTY ((if_marked ("tree_map_marked_p"), param_is (struct tree_map))) htab_t heapvar_for_stmt;
166
 
167
/* One variable to represent all non-local accesses.  */
168
tree nonlocal_all;
169
 
170
static bool use_field_sensitive = true;
171
static int in_ipa_mode = 0;
172
 
173
/* Used for predecessor bitmaps. */
174
static bitmap_obstack predbitmap_obstack;
175
 
176
/* Used for points-to sets.  */
177
static bitmap_obstack pta_obstack;
178
 
179
/* Used for oldsolution members of variables. */
180
static bitmap_obstack oldpta_obstack;
181
 
182
/* Used for per-solver-iteration bitmaps.  */
183
static bitmap_obstack iteration_obstack;
184
 
185
static unsigned int create_variable_info_for (tree, const char *);
186
typedef struct constraint_graph *constraint_graph_t;
187
static void unify_nodes (constraint_graph_t, unsigned int, unsigned int, bool);
188
 
189
DEF_VEC_P(constraint_t);
190
DEF_VEC_ALLOC_P(constraint_t,heap);
191
 
192
#define EXECUTE_IF_IN_NONNULL_BITMAP(a, b, c, d)        \
193
  if (a)                                                \
194
    EXECUTE_IF_SET_IN_BITMAP (a, b, c, d)
195
 
196
static struct constraint_stats
197
{
198
  unsigned int total_vars;
199
  unsigned int nonpointer_vars;
200
  unsigned int unified_vars_static;
201
  unsigned int unified_vars_dynamic;
202
  unsigned int iterations;
203
  unsigned int num_edges;
204
  unsigned int num_implicit_edges;
205
  unsigned int points_to_sets_created;
206
} stats;
207
 
208
struct variable_info
209
{
210
  /* ID of this variable  */
211
  unsigned int id;
212
 
213
  /* Name of this variable */
214
  const char *name;
215
 
216
  /* Tree that this variable is associated with.  */
217
  tree decl;
218
 
219
  /* Offset of this variable, in bits, from the base variable  */
220
  unsigned HOST_WIDE_INT offset;
221
 
222
  /* Size of the variable, in bits.  */
223
  unsigned HOST_WIDE_INT size;
224
 
225
  /* Full size of the base variable, in bits.  */
226
  unsigned HOST_WIDE_INT fullsize;
227
 
228
  /* A link to the variable for the next field in this structure.  */
229
  struct variable_info *next;
230
 
231
  /* True if the variable is directly the target of a dereference.
232
     This is used to track which variables are *actually* dereferenced
233
     so we can prune their points to listed. */
234
  unsigned int directly_dereferenced:1;
235
 
236
  /* True if this is a variable created by the constraint analysis, such as
237
     heap variables and constraints we had to break up.  */
238
  unsigned int is_artificial_var:1;
239
 
240
  /* True if this is a special variable whose solution set should not be
241
     changed.  */
242
  unsigned int is_special_var:1;
243
 
244
  /* True for variables whose size is not known or variable.  */
245
  unsigned int is_unknown_size_var:1;
246
 
247
  /* True for variables that have unions somewhere in them.  */
248
  unsigned int has_union:1;
249
 
250
  /* True if this is a heap variable.  */
251
  unsigned int is_heap_var:1;
252
 
253
  /* Points-to set for this variable.  */
254
  bitmap solution;
255
 
256
  /* Old points-to set for this variable.  */
257
  bitmap oldsolution;
258
 
259
  /* Variable ids represented by this node.  */
260
  bitmap variables;
261
 
262
  /* Variable id this was collapsed to due to type unsafety.  This
263
     should be unused completely after build_succ_graph, or something
264
     is broken.  */
265
  struct variable_info *collapsed_to;
266
};
267
typedef struct variable_info *varinfo_t;
268
 
269
static varinfo_t first_vi_for_offset (varinfo_t, unsigned HOST_WIDE_INT);
270
 
271
/* Pool of variable info structures.  */
272
static alloc_pool variable_info_pool;
273
 
274
DEF_VEC_P(varinfo_t);
275
 
276
DEF_VEC_ALLOC_P(varinfo_t, heap);
277
 
278
/* Table of variable info structures for constraint variables.
279
   Indexed directly by variable info id.  */
280
static VEC(varinfo_t,heap) *varmap;
281
 
282
/* Return the varmap element N */
283
 
284
static inline varinfo_t
285
get_varinfo (unsigned int n)
286
{
287
  return VEC_index (varinfo_t, varmap, n);
288
}
289
 
290
/* Return the varmap element N, following the collapsed_to link.  */
291
 
292
static inline varinfo_t
293
get_varinfo_fc (unsigned int n)
294
{
295
  varinfo_t v = VEC_index (varinfo_t, varmap, n);
296
 
297
  if (v->collapsed_to)
298
    return v->collapsed_to;
299
  return v;
300
}
301
 
302
/* Variable that represents the unknown pointer.  */
303
static varinfo_t var_anything;
304
static tree anything_tree;
305
static unsigned int anything_id;
306
 
307
/* Variable that represents the NULL pointer.  */
308
static varinfo_t var_nothing;
309
static tree nothing_tree;
310
static unsigned int nothing_id;
311
 
312
/* Variable that represents read only memory.  */
313
static varinfo_t var_readonly;
314
static tree readonly_tree;
315
static unsigned int readonly_id;
316
 
317
/* Variable that represents integers.  This is used for when people do things
318
   like &0->a.b.  */
319
static varinfo_t var_integer;
320
static tree integer_tree;
321
static unsigned int integer_id;
322
 
323
/* Variable that represents escaped variables.  This is used to give
324
   incoming pointer variables a better set than ANYTHING.  */
325
static varinfo_t var_escaped_vars;
326
static tree escaped_vars_tree;
327
static unsigned int escaped_vars_id;
328
 
329
/* Variable that represents non-local variables before we expand it to
330
   one for each type.  */
331
static unsigned int nonlocal_vars_id;
332
/* Lookup a heap var for FROM, and return it if we find one.  */
333
 
334
static tree
335
heapvar_lookup (tree from)
336
{
337
  struct tree_map *h, in;
338
  in.from = from;
339
 
340
  h = htab_find_with_hash (heapvar_for_stmt, &in, htab_hash_pointer (from));
341
  if (h)
342
    return h->to;
343
  return NULL_TREE;
344
}
345
 
346
/* Insert a mapping FROM->TO in the heap var for statement
347
   hashtable.  */
348
 
349
static void
350
heapvar_insert (tree from, tree to)
351
{
352
  struct tree_map *h;
353
  void **loc;
354
 
355
  h = ggc_alloc (sizeof (struct tree_map));
356
  h->hash = htab_hash_pointer (from);
357
  h->from = from;
358
  h->to = to;
359
  loc = htab_find_slot_with_hash (heapvar_for_stmt, h, h->hash, INSERT);
360
  *(struct tree_map **) loc = h;
361
}
362
 
363
/* Return a new variable info structure consisting for a variable
364
   named NAME, and using constraint graph node NODE.  */
365
 
366
static varinfo_t
367
new_var_info (tree t, unsigned int id, const char *name)
368
{
369
  varinfo_t ret = pool_alloc (variable_info_pool);
370
 
371
  ret->id = id;
372
  ret->name = name;
373
  ret->decl = t;
374
  ret->directly_dereferenced = false;
375
  ret->is_artificial_var = false;
376
  ret->is_heap_var = false;
377
  ret->is_special_var = false;
378
  ret->is_unknown_size_var = false;
379
  ret->has_union = false;
380
  ret->solution = BITMAP_ALLOC (&pta_obstack);
381
  ret->oldsolution = BITMAP_ALLOC (&oldpta_obstack);
382
  ret->next = NULL;
383
  ret->collapsed_to = NULL;
384
  return ret;
385
}
386
 
387
typedef enum {SCALAR, DEREF, ADDRESSOF} constraint_expr_type;
388
 
389
/* An expression that appears in a constraint.  */
390
 
391
struct constraint_expr
392
{
393
  /* Constraint type.  */
394
  constraint_expr_type type;
395
 
396
  /* Variable we are referring to in the constraint.  */
397
  unsigned int var;
398
 
399
  /* Offset, in bits, of this constraint from the beginning of
400
     variables it ends up referring to.
401
 
402
     IOW, in a deref constraint, we would deref, get the result set,
403
     then add OFFSET to each member.   */
404
  unsigned HOST_WIDE_INT offset;
405
};
406
 
407
typedef struct constraint_expr ce_s;
408
DEF_VEC_O(ce_s);
409
DEF_VEC_ALLOC_O(ce_s, heap);
410
static void get_constraint_for (tree, VEC(ce_s, heap) **);
411
static void do_deref (VEC (ce_s, heap) **);
412
 
413
/* Our set constraints are made up of two constraint expressions, one
414
   LHS, and one RHS.
415
 
416
   As described in the introduction, our set constraints each represent an
417
   operation between set valued variables.
418
*/
419
struct constraint
420
{
421
  struct constraint_expr lhs;
422
  struct constraint_expr rhs;
423
};
424
 
425
/* List of constraints that we use to build the constraint graph from.  */
426
 
427
static VEC(constraint_t,heap) *constraints;
428
static alloc_pool constraint_pool;
429
 
430
 
431
DEF_VEC_I(int);
432
DEF_VEC_ALLOC_I(int, heap);
433
 
434
/* The constraint graph is represented as an array of bitmaps
435
   containing successor nodes.  */
436
 
437
struct constraint_graph
438
{
439
  /* Size of this graph, which may be different than the number of
440
     nodes in the variable map.  */
441
  unsigned int size;
442
 
443
  /* Explicit successors of each node. */
444
  bitmap *succs;
445
 
446
  /* Implicit predecessors of each node (Used for variable
447
     substitution). */
448
  bitmap *implicit_preds;
449
 
450
  /* Explicit predecessors of each node (Used for variable substitution).  */
451
  bitmap *preds;
452
 
453
  /* Indirect cycle representatives, or -1 if the node has no indirect
454
     cycles.  */
455
  int *indirect_cycles;
456
 
457
  /* Representative node for a node.  rep[a] == a unless the node has
458
     been unified. */
459
  unsigned int *rep;
460
 
461
  /* Equivalence class representative for a node.  This is used for
462
     variable substitution.  */
463
  int *eq_rep;
464
 
465
  /* Label for each node, used during variable substitution.  */
466
  unsigned int *label;
467
 
468
  /* Bitmap of nodes where the bit is set if the node is a direct
469
     node.  Used for variable substitution.  */
470
  sbitmap direct_nodes;
471
 
472
  /* Vector of complex constraints for each graph node.  Complex
473
     constraints are those involving dereferences or offsets that are
474
     not 0.  */
475
  VEC(constraint_t,heap) **complex;
476
};
477
 
478
static constraint_graph_t graph;
479
 
480
/* During variable substitution and the offline version of indirect
481
   cycle finding, we create nodes to represent dereferences and
482
   address taken constraints.  These represent where these start and
483
   end.  */
484
#define FIRST_REF_NODE (VEC_length (varinfo_t, varmap))
485
#define LAST_REF_NODE (FIRST_REF_NODE + (FIRST_REF_NODE - 1))
486
#define FIRST_ADDR_NODE (LAST_REF_NODE + 1)
487
 
488
/* Return the representative node for NODE, if NODE has been unioned
489
   with another NODE.
490
   This function performs path compression along the way to finding
491
   the representative.  */
492
 
493
static unsigned int
494
find (unsigned int node)
495
{
496
  gcc_assert (node < graph->size);
497
  if (graph->rep[node] != node)
498
    return graph->rep[node] = find (graph->rep[node]);
499
  return node;
500
}
501
 
502
/* Union the TO and FROM nodes to the TO nodes.
503
   Note that at some point in the future, we may want to do
504
   union-by-rank, in which case we are going to have to return the
505
   node we unified to.  */
506
 
507
static bool
508
unite (unsigned int to, unsigned int from)
509
{
510
  gcc_assert (to < graph->size && from < graph->size);
511
  if (to != from && graph->rep[from] != to)
512
    {
513
      graph->rep[from] = to;
514
      return true;
515
    }
516
  return false;
517
}
518
 
519
/* Create a new constraint consisting of LHS and RHS expressions.  */
520
 
521
static constraint_t
522
new_constraint (const struct constraint_expr lhs,
523
                const struct constraint_expr rhs)
524
{
525
  constraint_t ret = pool_alloc (constraint_pool);
526
  ret->lhs = lhs;
527
  ret->rhs = rhs;
528
  return ret;
529
}
530
 
531
/* Print out constraint C to FILE.  */
532
 
533
void
534
dump_constraint (FILE *file, constraint_t c)
535
{
536
  if (c->lhs.type == ADDRESSOF)
537
    fprintf (file, "&");
538
  else if (c->lhs.type == DEREF)
539
    fprintf (file, "*");
540
  fprintf (file, "%s", get_varinfo_fc (c->lhs.var)->name);
541
  if (c->lhs.offset != 0)
542
    fprintf (file, " + " HOST_WIDE_INT_PRINT_DEC, c->lhs.offset);
543
  fprintf (file, " = ");
544
  if (c->rhs.type == ADDRESSOF)
545
    fprintf (file, "&");
546
  else if (c->rhs.type == DEREF)
547
    fprintf (file, "*");
548
  fprintf (file, "%s", get_varinfo_fc (c->rhs.var)->name);
549
  if (c->rhs.offset != 0)
550
    fprintf (file, " + " HOST_WIDE_INT_PRINT_DEC, c->rhs.offset);
551
  fprintf (file, "\n");
552
}
553
 
554
/* Print out constraint C to stderr.  */
555
 
556
void
557
debug_constraint (constraint_t c)
558
{
559
  dump_constraint (stderr, c);
560
}
561
 
562
/* Print out all constraints to FILE */
563
 
564
void
565
dump_constraints (FILE *file)
566
{
567
  int i;
568
  constraint_t c;
569
  for (i = 0; VEC_iterate (constraint_t, constraints, i, c); i++)
570
    dump_constraint (file, c);
571
}
572
 
573
/* Print out all constraints to stderr.  */
574
 
575
void
576
debug_constraints (void)
577
{
578
  dump_constraints (stderr);
579
}
580
 
581
/* SOLVER FUNCTIONS
582
 
583
   The solver is a simple worklist solver, that works on the following
584
   algorithm:
585
 
586
   sbitmap changed_nodes = all zeroes;
587
   changed_count = 0;
588
   For each node that is not already collapsed:
589
       changed_count++;
590
       set bit in changed nodes
591
 
592
   while (changed_count > 0)
593
   {
594
     compute topological ordering for constraint graph
595
 
596
     find and collapse cycles in the constraint graph (updating
597
     changed if necessary)
598
 
599
     for each node (n) in the graph in topological order:
600
       changed_count--;
601
 
602
       Process each complex constraint associated with the node,
603
       updating changed if necessary.
604
 
605
       For each outgoing edge from n, propagate the solution from n to
606
       the destination of the edge, updating changed as necessary.
607
 
608
   }  */
609
 
610
/* Return true if two constraint expressions A and B are equal.  */
611
 
612
static bool
613
constraint_expr_equal (struct constraint_expr a, struct constraint_expr b)
614
{
615
  return a.type == b.type && a.var == b.var && a.offset == b.offset;
616
}
617
 
618
/* Return true if constraint expression A is less than constraint expression
619
   B.  This is just arbitrary, but consistent, in order to give them an
620
   ordering.  */
621
 
622
static bool
623
constraint_expr_less (struct constraint_expr a, struct constraint_expr b)
624
{
625
  if (a.type == b.type)
626
    {
627
      if (a.var == b.var)
628
        return a.offset < b.offset;
629
      else
630
        return a.var < b.var;
631
    }
632
  else
633
    return a.type < b.type;
634
}
635
 
636
/* Return true if constraint A is less than constraint B.  This is just
637
   arbitrary, but consistent, in order to give them an ordering.  */
638
 
639
static bool
640
constraint_less (const constraint_t a, const constraint_t b)
641
{
642
  if (constraint_expr_less (a->lhs, b->lhs))
643
    return true;
644
  else if (constraint_expr_less (b->lhs, a->lhs))
645
    return false;
646
  else
647
    return constraint_expr_less (a->rhs, b->rhs);
648
}
649
 
650
/* Return true if two constraints A and B are equal.  */
651
 
652
static bool
653
constraint_equal (struct constraint a, struct constraint b)
654
{
655
  return constraint_expr_equal (a.lhs, b.lhs)
656
    && constraint_expr_equal (a.rhs, b.rhs);
657
}
658
 
659
 
660
/* Find a constraint LOOKFOR in the sorted constraint vector VEC */
661
 
662
static constraint_t
663
constraint_vec_find (VEC(constraint_t,heap) *vec,
664
                     struct constraint lookfor)
665
{
666
  unsigned int place;
667
  constraint_t found;
668
 
669
  if (vec == NULL)
670
    return NULL;
671
 
672
  place = VEC_lower_bound (constraint_t, vec, &lookfor, constraint_less);
673
  if (place >= VEC_length (constraint_t, vec))
674
    return NULL;
675
  found = VEC_index (constraint_t, vec, place);
676
  if (!constraint_equal (*found, lookfor))
677
    return NULL;
678
  return found;
679
}
680
 
681
/* Union two constraint vectors, TO and FROM.  Put the result in TO.  */
682
 
683
static void
684
constraint_set_union (VEC(constraint_t,heap) **to,
685
                      VEC(constraint_t,heap) **from)
686
{
687
  int i;
688
  constraint_t c;
689
 
690
  for (i = 0; VEC_iterate (constraint_t, *from, i, c); i++)
691
    {
692
      if (constraint_vec_find (*to, *c) == NULL)
693
        {
694
          unsigned int place = VEC_lower_bound (constraint_t, *to, c,
695
                                                constraint_less);
696
          VEC_safe_insert (constraint_t, heap, *to, place, c);
697
        }
698
    }
699
}
700
 
701
/* Take a solution set SET, add OFFSET to each member of the set, and
702
   overwrite SET with the result when done.  */
703
 
704
static void
705
solution_set_add (bitmap set, unsigned HOST_WIDE_INT offset)
706
{
707
  bitmap result = BITMAP_ALLOC (&iteration_obstack);
708
  unsigned int i;
709
  bitmap_iterator bi;
710
  unsigned HOST_WIDE_INT min = -1, max = 0;
711
 
712
  /* Compute set of vars we can reach from set + offset.  */
713
 
714
  EXECUTE_IF_SET_IN_BITMAP (set, 0, i, bi)
715
    {
716
      if (get_varinfo (i)->is_artificial_var
717
          || get_varinfo (i)->has_union
718
          || get_varinfo (i)->is_unknown_size_var)
719
        continue;
720
 
721
      if (get_varinfo (i)->offset + offset < min)
722
        min = get_varinfo (i)->offset + offset;
723
      if (get_varinfo (i)->offset + get_varinfo (i)->size + offset > max)
724
        {
725
          max = get_varinfo (i)->offset + get_varinfo (i)->size + offset;
726
          if (max > get_varinfo (i)->fullsize)
727
            max = get_varinfo (i)->fullsize;
728
        }
729
    }
730
 
731
  EXECUTE_IF_SET_IN_BITMAP (set, 0, i, bi)
732
    {
733
      /* If this is a properly sized variable, only add offset if it's
734
         less than end.  Otherwise, it is globbed to a single
735
         variable.  */
736
 
737
      if (get_varinfo (i)->offset + get_varinfo (i)->size - 1 >= min
738
          && get_varinfo (i)->offset < max)
739
        {
740
          bitmap_set_bit (result, i);
741
        }
742
      else if (get_varinfo (i)->is_artificial_var
743
               || get_varinfo (i)->has_union
744
               || get_varinfo (i)->is_unknown_size_var)
745
        {
746
          bitmap_set_bit (result, i);
747
        }
748
    }
749
 
750
  bitmap_copy (set, result);
751
  BITMAP_FREE (result);
752
}
753
 
754
/* Union solution sets TO and FROM, and add INC to each member of FROM in the
755
   process.  */
756
 
757
static bool
758
set_union_with_increment  (bitmap to, bitmap from, unsigned HOST_WIDE_INT inc)
759
{
760
  if (inc == 0)
761
    return bitmap_ior_into (to, from);
762
  else
763
    {
764
      bitmap tmp;
765
      bool res;
766
 
767
      tmp = BITMAP_ALLOC (&iteration_obstack);
768
      bitmap_copy (tmp, from);
769
      solution_set_add (tmp, inc);
770
      res = bitmap_ior_into (to, tmp);
771
      BITMAP_FREE (tmp);
772
      return res;
773
    }
774
}
775
 
776
/* Insert constraint C into the list of complex constraints for graph
777
   node VAR.  */
778
 
779
static void
780
insert_into_complex (constraint_graph_t graph,
781
                     unsigned int var, constraint_t c)
782
{
783
  VEC (constraint_t, heap) *complex = graph->complex[var];
784
  unsigned int place = VEC_lower_bound (constraint_t, complex, c,
785
                                        constraint_less);
786
 
787
  /* Only insert constraints that do not already exist.  */
788
  if (place >= VEC_length (constraint_t, complex)
789
      || !constraint_equal (*c, *VEC_index (constraint_t, complex, place)))
790
    VEC_safe_insert (constraint_t, heap, graph->complex[var], place, c);
791
}
792
 
793
 
794
/* Condense two variable nodes into a single variable node, by moving
795
   all associated info from SRC to TO.  */
796
 
797
static void
798
merge_node_constraints (constraint_graph_t graph, unsigned int to,
799
                        unsigned int from)
800
{
801
  unsigned int i;
802
  constraint_t c;
803
 
804
  gcc_assert (find (from) == to);
805
 
806
  /* Move all complex constraints from src node into to node  */
807
  for (i = 0; VEC_iterate (constraint_t, graph->complex[from], i, c); i++)
808
    {
809
      /* In complex constraints for node src, we may have either
810
         a = *src, and *src = a, or an offseted constraint which are
811
         always added to the rhs node's constraints.  */
812
 
813
      if (c->rhs.type == DEREF)
814
        c->rhs.var = to;
815
      else if (c->lhs.type == DEREF)
816
        c->lhs.var = to;
817
      else
818
        c->rhs.var = to;
819
    }
820
  constraint_set_union (&graph->complex[to], &graph->complex[from]);
821
  VEC_free (constraint_t, heap, graph->complex[from]);
822
  graph->complex[from] = NULL;
823
}
824
 
825
 
826
/* Remove edges involving NODE from GRAPH.  */
827
 
828
static void
829
clear_edges_for_node (constraint_graph_t graph, unsigned int node)
830
{
831
  if (graph->succs[node])
832
    BITMAP_FREE (graph->succs[node]);
833
}
834
 
835
/* Merge GRAPH nodes FROM and TO into node TO.  */
836
 
837
static void
838
merge_graph_nodes (constraint_graph_t graph, unsigned int to,
839
                   unsigned int from)
840
{
841
  if (graph->indirect_cycles[from] != -1)
842
    {
843
      /* If we have indirect cycles with the from node, and we have
844
         none on the to node, the to node has indirect cycles from the
845
         from node now that they are unified.
846
         If indirect cycles exist on both, unify the nodes that they
847
         are in a cycle with, since we know they are in a cycle with
848
         each other.  */
849
      if (graph->indirect_cycles[to] == -1)
850
        {
851
          graph->indirect_cycles[to] = graph->indirect_cycles[from];
852
        }
853
      else
854
        {
855
          unsigned int tonode = find (graph->indirect_cycles[to]);
856
          unsigned int fromnode = find (graph->indirect_cycles[from]);
857
 
858
          if (unite (tonode, fromnode))
859
            unify_nodes (graph, tonode, fromnode, true);
860
        }
861
    }
862
 
863
  /* Merge all the successor edges.  */
864
  if (graph->succs[from])
865
    {
866
      if (!graph->succs[to])
867
        graph->succs[to] = BITMAP_ALLOC (&pta_obstack);
868
      bitmap_ior_into (graph->succs[to],
869
                       graph->succs[from]);
870
    }
871
 
872
  clear_edges_for_node (graph, from);
873
}
874
 
875
 
876
/* Add an indirect graph edge to GRAPH, going from TO to FROM if
877
   it doesn't exist in the graph already.  */
878
 
879
static void
880
add_implicit_graph_edge (constraint_graph_t graph, unsigned int to,
881
                         unsigned int from)
882
{
883
  if (to == from)
884
    return;
885
 
886
  if (!graph->implicit_preds[to])
887
    graph->implicit_preds[to] = BITMAP_ALLOC (&predbitmap_obstack);
888
 
889
  if (!bitmap_bit_p (graph->implicit_preds[to], from))
890
    {
891
      stats.num_implicit_edges++;
892
      bitmap_set_bit (graph->implicit_preds[to], from);
893
    }
894
}
895
 
896
/* Add a predecessor graph edge to GRAPH, going from TO to FROM if
897
   it doesn't exist in the graph already.
898
   Return false if the edge already existed, true otherwise.  */
899
 
900
static void
901
add_pred_graph_edge (constraint_graph_t graph, unsigned int to,
902
                     unsigned int from)
903
{
904
  if (!graph->preds[to])
905
    graph->preds[to] = BITMAP_ALLOC (&predbitmap_obstack);
906
  if (!bitmap_bit_p (graph->preds[to], from))
907
    bitmap_set_bit (graph->preds[to], from);
908
}
909
 
910
/* Add a graph edge to GRAPH, going from FROM to TO if
911
   it doesn't exist in the graph already.
912
   Return false if the edge already existed, true otherwise.  */
913
 
914
static bool
915
add_graph_edge (constraint_graph_t graph, unsigned int to,
916
                unsigned int from)
917
{
918
  if (to == from)
919
    {
920
      return false;
921
    }
922
  else
923
    {
924
      bool r = false;
925
 
926
      if (!graph->succs[from])
927
        graph->succs[from] = BITMAP_ALLOC (&pta_obstack);
928
      if (!bitmap_bit_p (graph->succs[from], to))
929
        {
930
          r = true;
931
          if (to < FIRST_REF_NODE && from < FIRST_REF_NODE)
932
            stats.num_edges++;
933
          bitmap_set_bit (graph->succs[from], to);
934
        }
935
      return r;
936
    }
937
}
938
 
939
 
940
/* Return true if {DEST.SRC} is an existing graph edge in GRAPH.  */
941
 
942
static bool
943
valid_graph_edge (constraint_graph_t graph, unsigned int src,
944
                  unsigned int dest)
945
{
946
  return (graph->succs[dest]
947
          && bitmap_bit_p (graph->succs[dest], src));
948
}
949
 
950
/* Build the constraint graph, adding only predecessor edges right now.  */
951
 
952
static void
953
build_pred_graph (void)
954
{
955
  int i;
956
  constraint_t c;
957
  unsigned int j;
958
 
959
  graph = XNEW (struct constraint_graph);
960
  graph->size = (VEC_length (varinfo_t, varmap)) * 3;
961
  graph->succs = XCNEWVEC (bitmap, graph->size);
962
  graph->implicit_preds = XCNEWVEC (bitmap, graph->size);
963
  graph->preds = XCNEWVEC (bitmap, graph->size);
964
  graph->indirect_cycles = XNEWVEC (int, VEC_length (varinfo_t, varmap));
965
  graph->label = XCNEWVEC (unsigned int, graph->size);
966
  graph->rep = XNEWVEC (unsigned int, graph->size);
967
  graph->eq_rep = XNEWVEC (int, graph->size);
968
  graph->complex = XCNEWVEC (VEC(constraint_t, heap) *,
969
                             VEC_length (varinfo_t, varmap));
970
  graph->direct_nodes = sbitmap_alloc (graph->size);
971
  sbitmap_zero (graph->direct_nodes);
972
 
973
  for (j = 0; j < FIRST_REF_NODE; j++)
974
    {
975
      if (!get_varinfo (j)->is_special_var)
976
        SET_BIT (graph->direct_nodes, j);
977
    }
978
 
979
  for (j = 0; j < graph->size; j++)
980
    {
981
      graph->rep[j] = j;
982
      graph->eq_rep[j] = -1;
983
    }
984
 
985
  for (j = 0; j < VEC_length (varinfo_t, varmap); j++)
986
    graph->indirect_cycles[j] = -1;
987
 
988
  for (i = 0; VEC_iterate (constraint_t, constraints, i, c); i++)
989
    {
990
      struct constraint_expr lhs = c->lhs;
991
      struct constraint_expr rhs = c->rhs;
992
      unsigned int lhsvar = get_varinfo_fc (lhs.var)->id;
993
      unsigned int rhsvar = get_varinfo_fc (rhs.var)->id;
994
 
995
      if (lhs.type == DEREF)
996
        {
997
          /* *x = y.  */
998
          if (rhs.offset == 0 && lhs.offset == 0 && rhs.type == SCALAR)
999
            add_pred_graph_edge (graph, FIRST_REF_NODE + lhsvar, rhsvar);
1000
          if (rhs.type == ADDRESSOF)
1001
            RESET_BIT (graph->direct_nodes, rhsvar);
1002
        }
1003
      else if (rhs.type == DEREF)
1004
        {
1005
          /* x = *y */
1006
          if (rhs.offset == 0 && lhs.offset == 0 && lhs.type == SCALAR)
1007
            add_pred_graph_edge (graph, lhsvar, FIRST_REF_NODE + rhsvar);
1008
          else
1009
            RESET_BIT (graph->direct_nodes, lhsvar);
1010
        }
1011
      else if (rhs.type == ADDRESSOF)
1012
        {
1013
          /* x = &y */
1014
          add_pred_graph_edge (graph, lhsvar, FIRST_ADDR_NODE + rhsvar);
1015
          /* Implicitly, *x = y */
1016
          add_implicit_graph_edge (graph, FIRST_REF_NODE + lhsvar, rhsvar);
1017
 
1018
          RESET_BIT (graph->direct_nodes, rhsvar);
1019
        }
1020
      else if (lhsvar > anything_id
1021
               && lhsvar != rhsvar && lhs.offset == 0 && rhs.offset == 0)
1022
        {
1023
          /* x = y */
1024
          add_pred_graph_edge (graph, lhsvar, rhsvar);
1025
          /* Implicitly, *x = *y */
1026
          add_implicit_graph_edge (graph, FIRST_REF_NODE + lhsvar,
1027
                                   FIRST_REF_NODE + rhsvar);
1028
        }
1029
      else if (lhs.offset != 0 || rhs.offset != 0)
1030
        {
1031
          if (rhs.offset != 0)
1032
            RESET_BIT (graph->direct_nodes, lhs.var);
1033
          if (lhs.offset != 0)
1034
            RESET_BIT (graph->direct_nodes, rhs.var);
1035
        }
1036
    }
1037
}
1038
 
1039
/* Build the constraint graph, adding successor edges.  */
1040
 
1041
static void
1042
build_succ_graph (void)
1043
{
1044
  int i;
1045
  constraint_t c;
1046
 
1047
  for (i = 0; VEC_iterate (constraint_t, constraints, i, c); i++)
1048
    {
1049
      struct constraint_expr lhs;
1050
      struct constraint_expr rhs;
1051
      unsigned int lhsvar;
1052
      unsigned int rhsvar;
1053
 
1054
      if (!c)
1055
        continue;
1056
 
1057
      lhs = c->lhs;
1058
      rhs = c->rhs;
1059
      lhsvar = find (get_varinfo_fc (lhs.var)->id);
1060
      rhsvar = find (get_varinfo_fc (rhs.var)->id);
1061
 
1062
      if (lhs.type == DEREF)
1063
        {
1064
          if (rhs.offset == 0 && lhs.offset == 0 && rhs.type == SCALAR)
1065
            add_graph_edge (graph, FIRST_REF_NODE + lhsvar, rhsvar);
1066
        }
1067
      else if (rhs.type == DEREF)
1068
        {
1069
          if (rhs.offset == 0 && lhs.offset == 0 && lhs.type == SCALAR)
1070
            add_graph_edge (graph, lhsvar, FIRST_REF_NODE + rhsvar);
1071
        }
1072
      else if (rhs.type == ADDRESSOF)
1073
        {
1074
          /* x = &y */
1075
          gcc_assert (find (get_varinfo_fc (rhs.var)->id)
1076
                      == get_varinfo_fc (rhs.var)->id);
1077
          bitmap_set_bit (get_varinfo (lhsvar)->solution, rhsvar);
1078
        }
1079
      else if (lhsvar > anything_id
1080
               && lhsvar != rhsvar && lhs.offset == 0 && rhs.offset == 0)
1081
        {
1082
          add_graph_edge (graph, lhsvar, rhsvar);
1083
        }
1084
    }
1085
}
1086
 
1087
 
1088
/* Changed variables on the last iteration.  */
1089
static unsigned int changed_count;
1090
static sbitmap changed;
1091
 
1092
DEF_VEC_I(unsigned);
1093
DEF_VEC_ALLOC_I(unsigned,heap);
1094
 
1095
 
1096
/* Strongly Connected Component visitation info.  */
1097
 
1098
struct scc_info
1099
{
1100
  sbitmap visited;
1101
  sbitmap roots;
1102
  unsigned int *dfs;
1103
  unsigned int *node_mapping;
1104
  int current_index;
1105
  VEC(unsigned,heap) *scc_stack;
1106
};
1107
 
1108
 
1109
/* Recursive routine to find strongly connected components in GRAPH.
1110
   SI is the SCC info to store the information in, and N is the id of current
1111
   graph node we are processing.
1112
 
1113
   This is Tarjan's strongly connected component finding algorithm, as
1114
   modified by Nuutila to keep only non-root nodes on the stack.
1115
   The algorithm can be found in "On finding the strongly connected
1116
   connected components in a directed graph" by Esko Nuutila and Eljas
1117
   Soisalon-Soininen, in Information Processing Letters volume 49,
1118
   number 1, pages 9-14.  */
1119
 
1120
static void
1121
scc_visit (constraint_graph_t graph, struct scc_info *si, unsigned int n)
1122
{
1123
  unsigned int i;
1124
  bitmap_iterator bi;
1125
  unsigned int my_dfs;
1126
 
1127
  SET_BIT (si->visited, n);
1128
  si->dfs[n] = si->current_index ++;
1129
  my_dfs = si->dfs[n];
1130
 
1131
  /* Visit all the successors.  */
1132
  EXECUTE_IF_IN_NONNULL_BITMAP (graph->succs[n], 0, i, bi)
1133
    {
1134
      unsigned int w;
1135
 
1136
      if (i > LAST_REF_NODE)
1137
        break;
1138
 
1139
      w = find (i);
1140
      if (TEST_BIT (si->roots, w))
1141
        continue;
1142
 
1143
      if (!TEST_BIT (si->visited, w))
1144
        scc_visit (graph, si, w);
1145
      {
1146
        unsigned int t = find (w);
1147
        unsigned int nnode = find (n);
1148
        gcc_assert (nnode == n);
1149
 
1150
        if (si->dfs[t] < si->dfs[nnode])
1151
          si->dfs[n] = si->dfs[t];
1152
      }
1153
    }
1154
 
1155
  /* See if any components have been identified.  */
1156
  if (si->dfs[n] == my_dfs)
1157
    {
1158
      if (VEC_length (unsigned, si->scc_stack) > 0
1159
          && si->dfs[VEC_last (unsigned, si->scc_stack)] >= my_dfs)
1160
        {
1161
          bitmap scc = BITMAP_ALLOC (NULL);
1162
          bool have_ref_node = n >= FIRST_REF_NODE;
1163
          unsigned int lowest_node;
1164
          bitmap_iterator bi;
1165
 
1166
          bitmap_set_bit (scc, n);
1167
 
1168
          while (VEC_length (unsigned, si->scc_stack) != 0
1169
                 && si->dfs[VEC_last (unsigned, si->scc_stack)] >= my_dfs)
1170
            {
1171
              unsigned int w = VEC_pop (unsigned, si->scc_stack);
1172
 
1173
              bitmap_set_bit (scc, w);
1174
              if (w >= FIRST_REF_NODE)
1175
                have_ref_node = true;
1176
            }
1177
 
1178
          lowest_node = bitmap_first_set_bit (scc);
1179
          gcc_assert (lowest_node < FIRST_REF_NODE);
1180
          EXECUTE_IF_SET_IN_BITMAP (scc, 0, i, bi)
1181
            {
1182
              if (i < FIRST_REF_NODE)
1183
                {
1184
                  /* Mark this node for collapsing.  */
1185
                  if (unite (lowest_node, i))
1186
                    unify_nodes (graph, lowest_node, i, false);
1187
                }
1188
              else
1189
                {
1190
                  unite (lowest_node, i);
1191
                  graph->indirect_cycles[i - FIRST_REF_NODE] = lowest_node;
1192
                }
1193
            }
1194
        }
1195
      SET_BIT (si->roots, n);
1196
    }
1197
  else
1198
    VEC_safe_push (unsigned, heap, si->scc_stack, n);
1199
}
1200
 
1201
/* Unify node FROM into node TO, updating the changed count if
1202
   necessary when UPDATE_CHANGED is true.  */
1203
 
1204
static void
1205
unify_nodes (constraint_graph_t graph, unsigned int to, unsigned int from,
1206
             bool update_changed)
1207
{
1208
 
1209
  gcc_assert (to != from && find (to) == to);
1210
  if (dump_file && (dump_flags & TDF_DETAILS))
1211
    fprintf (dump_file, "Unifying %s to %s\n",
1212
             get_varinfo (from)->name,
1213
             get_varinfo (to)->name);
1214
 
1215
  if (update_changed)
1216
    stats.unified_vars_dynamic++;
1217
  else
1218
    stats.unified_vars_static++;
1219
 
1220
  merge_graph_nodes (graph, to, from);
1221
  merge_node_constraints (graph, to, from);
1222
 
1223
  if (update_changed && TEST_BIT (changed, from))
1224
    {
1225
      RESET_BIT (changed, from);
1226
      if (!TEST_BIT (changed, to))
1227
        SET_BIT (changed, to);
1228
      else
1229
        {
1230
          gcc_assert (changed_count > 0);
1231
          changed_count--;
1232
        }
1233
    }
1234
 
1235
  /* If the solution changes because of the merging, we need to mark
1236
     the variable as changed.  */
1237
  if (bitmap_ior_into (get_varinfo (to)->solution,
1238
                       get_varinfo (from)->solution))
1239
    {
1240
      if (update_changed && !TEST_BIT (changed, to))
1241
        {
1242
          SET_BIT (changed, to);
1243
          changed_count++;
1244
        }
1245
    }
1246
 
1247
  BITMAP_FREE (get_varinfo (from)->solution);
1248
  BITMAP_FREE (get_varinfo (from)->oldsolution);
1249
 
1250
  if (stats.iterations > 0)
1251
    {
1252
      BITMAP_FREE (get_varinfo (to)->oldsolution);
1253
      get_varinfo (to)->oldsolution = BITMAP_ALLOC (&oldpta_obstack);
1254
    }
1255
 
1256
  if (valid_graph_edge (graph, to, to))
1257
    {
1258
      if (graph->succs[to])
1259
        bitmap_clear_bit (graph->succs[to], to);
1260
    }
1261
}
1262
 
1263
/* Information needed to compute the topological ordering of a graph.  */
1264
 
1265
struct topo_info
1266
{
1267
  /* sbitmap of visited nodes.  */
1268
  sbitmap visited;
1269
  /* Array that stores the topological order of the graph, *in
1270
     reverse*.  */
1271
  VEC(unsigned,heap) *topo_order;
1272
};
1273
 
1274
 
1275
/* Initialize and return a topological info structure.  */
1276
 
1277
static struct topo_info *
1278
init_topo_info (void)
1279
{
1280
  size_t size = VEC_length (varinfo_t, varmap);
1281
  struct topo_info *ti = XNEW (struct topo_info);
1282
  ti->visited = sbitmap_alloc (size);
1283
  sbitmap_zero (ti->visited);
1284
  ti->topo_order = VEC_alloc (unsigned, heap, 1);
1285
  return ti;
1286
}
1287
 
1288
 
1289
/* Free the topological sort info pointed to by TI.  */
1290
 
1291
static void
1292
free_topo_info (struct topo_info *ti)
1293
{
1294
  sbitmap_free (ti->visited);
1295
  VEC_free (unsigned, heap, ti->topo_order);
1296
  free (ti);
1297
}
1298
 
1299
/* Visit the graph in topological order, and store the order in the
1300
   topo_info structure.  */
1301
 
1302
static void
1303
topo_visit (constraint_graph_t graph, struct topo_info *ti,
1304
            unsigned int n)
1305
{
1306
  bitmap_iterator bi;
1307
  unsigned int j;
1308
 
1309
  SET_BIT (ti->visited, n);
1310
 
1311
  if (graph->succs[n])
1312
    EXECUTE_IF_SET_IN_BITMAP (graph->succs[n], 0, j, bi)
1313
      {
1314
        if (!TEST_BIT (ti->visited, j))
1315
          topo_visit (graph, ti, j);
1316
      }
1317
 
1318
  VEC_safe_push (unsigned, heap, ti->topo_order, n);
1319
}
1320
 
1321
/* Return true if variable N + OFFSET is a legal field of N.  */
1322
 
1323
static bool
1324
type_safe (unsigned int n, unsigned HOST_WIDE_INT *offset)
1325
{
1326
  varinfo_t ninfo = get_varinfo (n);
1327
 
1328
  /* For things we've globbed to single variables, any offset into the
1329
     variable acts like the entire variable, so that it becomes offset
1330
     0.  */
1331
  if (ninfo->is_special_var
1332
      || ninfo->is_artificial_var
1333
      || ninfo->is_unknown_size_var)
1334
    {
1335
      *offset = 0;
1336
      return true;
1337
    }
1338
  return (get_varinfo (n)->offset + *offset) < get_varinfo (n)->fullsize;
1339
}
1340
 
1341
/* Process a constraint C that represents *x = &y.  */
1342
 
1343
static void
1344
do_da_constraint (constraint_graph_t graph ATTRIBUTE_UNUSED,
1345
                  constraint_t c, bitmap delta)
1346
{
1347
  unsigned int rhs = c->rhs.var;
1348
  unsigned int j;
1349
  bitmap_iterator bi;
1350
 
1351
  /* For each member j of Delta (Sol(x)), add x to Sol(j)  */
1352
  EXECUTE_IF_SET_IN_BITMAP (delta, 0, j, bi)
1353
    {
1354
      unsigned HOST_WIDE_INT offset = c->lhs.offset;
1355
      if (type_safe (j, &offset) && !(get_varinfo (j)->is_special_var))
1356
        {
1357
        /* *x != NULL && *x != ANYTHING*/
1358
          varinfo_t v;
1359
          unsigned int t;
1360
          bitmap sol;
1361
          unsigned HOST_WIDE_INT fieldoffset = get_varinfo (j)->offset + offset;
1362
 
1363
          v = first_vi_for_offset (get_varinfo (j), fieldoffset);
1364
          if (!v)
1365
            continue;
1366
          t = find (v->id);
1367
          sol = get_varinfo (t)->solution;
1368
          if (!bitmap_bit_p (sol, rhs))
1369
            {
1370
              bitmap_set_bit (sol, rhs);
1371
              if (!TEST_BIT (changed, t))
1372
                {
1373
                  SET_BIT (changed, t);
1374
                  changed_count++;
1375
                }
1376
            }
1377
        }
1378
      else if (0 && dump_file && !(get_varinfo (j)->is_special_var))
1379
        fprintf (dump_file, "Untypesafe usage in do_da_constraint.\n");
1380
 
1381
    }
1382
}
1383
 
1384
/* Process a constraint C that represents x = *y, using DELTA as the
1385
   starting solution.  */
1386
 
1387
static void
1388
do_sd_constraint (constraint_graph_t graph, constraint_t c,
1389
                  bitmap delta)
1390
{
1391
  unsigned int lhs = find (c->lhs.var);
1392
  bool flag = false;
1393
  bitmap sol = get_varinfo (lhs)->solution;
1394
  unsigned int j;
1395
  bitmap_iterator bi;
1396
 
1397
 if (bitmap_bit_p (delta, anything_id))
1398
   {
1399
     flag = !bitmap_bit_p (sol, anything_id);
1400
     if (flag)
1401
       bitmap_set_bit (sol, anything_id);
1402
     goto done;
1403
   }
1404
  /* For each variable j in delta (Sol(y)), add
1405
     an edge in the graph from j to x, and union Sol(j) into Sol(x).  */
1406
  EXECUTE_IF_SET_IN_BITMAP (delta, 0, j, bi)
1407
    {
1408
      unsigned HOST_WIDE_INT roffset = c->rhs.offset;
1409
      if (type_safe (j, &roffset))
1410
        {
1411
          varinfo_t v;
1412
          unsigned HOST_WIDE_INT fieldoffset = get_varinfo (j)->offset + roffset;
1413
          unsigned int t;
1414
 
1415
          v = first_vi_for_offset (get_varinfo (j), fieldoffset);
1416
          if (!v)
1417
            continue;
1418
          t = find (v->id);
1419
 
1420
          /* Adding edges from the special vars is pointless.
1421
             They don't have sets that can change.  */
1422
          if (get_varinfo (t) ->is_special_var)
1423
            flag |= bitmap_ior_into (sol, get_varinfo (t)->solution);
1424
          else if (add_graph_edge (graph, lhs, t))
1425
            flag |= bitmap_ior_into (sol, get_varinfo (t)->solution);
1426
        }
1427
      else if (0 && dump_file && !(get_varinfo (j)->is_special_var))
1428
        fprintf (dump_file, "Untypesafe usage in do_sd_constraint\n");
1429
 
1430
    }
1431
 
1432
done:
1433
  /* If the LHS solution changed, mark the var as changed.  */
1434
  if (flag)
1435
    {
1436
      get_varinfo (lhs)->solution = sol;
1437
      if (!TEST_BIT (changed, lhs))
1438
        {
1439
          SET_BIT (changed, lhs);
1440
          changed_count++;
1441
        }
1442
    }
1443
}
1444
 
1445
/* Process a constraint C that represents *x = y.  */
1446
 
1447
static void
1448
do_ds_constraint (constraint_t c, bitmap delta)
1449
{
1450
  unsigned int rhs = find (c->rhs.var);
1451
  unsigned HOST_WIDE_INT roff = c->rhs.offset;
1452
  bitmap sol = get_varinfo (rhs)->solution;
1453
  unsigned int j;
1454
  bitmap_iterator bi;
1455
 
1456
 if (bitmap_bit_p (sol, anything_id))
1457
   {
1458
     EXECUTE_IF_SET_IN_BITMAP (delta, 0, j, bi)
1459
       {
1460
         varinfo_t jvi = get_varinfo (j);
1461
         unsigned int t;
1462
         unsigned int loff = c->lhs.offset;
1463
         unsigned HOST_WIDE_INT fieldoffset = jvi->offset + loff;
1464
         varinfo_t v;
1465
 
1466
         v = first_vi_for_offset (get_varinfo (j), fieldoffset);
1467
         if (!v)
1468
           continue;
1469
         t = find (v->id);
1470
 
1471
         if (!bitmap_bit_p (get_varinfo (t)->solution, anything_id))
1472
           {
1473
             bitmap_set_bit (get_varinfo (t)->solution, anything_id);
1474
             if (!TEST_BIT (changed, t))
1475
               {
1476
                 SET_BIT (changed, t);
1477
                 changed_count++;
1478
               }
1479
           }
1480
       }
1481
     return;
1482
   }
1483
 
1484
  /* For each member j of delta (Sol(x)), add an edge from y to j and
1485
     union Sol(y) into Sol(j) */
1486
  EXECUTE_IF_SET_IN_BITMAP (delta, 0, j, bi)
1487
    {
1488
      unsigned HOST_WIDE_INT loff = c->lhs.offset;
1489
      if (type_safe (j, &loff) && !(get_varinfo (j)->is_special_var))
1490
        {
1491
          varinfo_t v;
1492
          unsigned int t;
1493
          unsigned HOST_WIDE_INT fieldoffset = get_varinfo (j)->offset + loff;
1494
          bitmap tmp;
1495
 
1496
          v = first_vi_for_offset (get_varinfo (j), fieldoffset);
1497
          if (!v)
1498
            continue;
1499
          t = find (v->id);
1500
          tmp = get_varinfo (t)->solution;
1501
 
1502
          if (set_union_with_increment (tmp, sol, roff))
1503
            {
1504
              get_varinfo (t)->solution = tmp;
1505
              if (t == rhs)
1506
                sol = get_varinfo (rhs)->solution;
1507
              if (!TEST_BIT (changed, t))
1508
                {
1509
                  SET_BIT (changed, t);
1510
                  changed_count++;
1511
                }
1512
            }
1513
        }
1514
      else if (0 && dump_file && !(get_varinfo (j)->is_special_var))
1515
        fprintf (dump_file, "Untypesafe usage in do_ds_constraint\n");
1516
    }
1517
}
1518
 
1519
/* Handle a non-simple (simple meaning requires no iteration),
1520
   constraint (IE *x = &y, x = *y, *x = y, and x = y with offsets involved).  */
1521
 
1522
static void
1523
do_complex_constraint (constraint_graph_t graph, constraint_t c, bitmap delta)
1524
{
1525
  if (c->lhs.type == DEREF)
1526
    {
1527
      if (c->rhs.type == ADDRESSOF)
1528
        {
1529
          /* *x = &y */
1530
          do_da_constraint (graph, c, delta);
1531
        }
1532
      else
1533
        {
1534
          /* *x = y */
1535
          do_ds_constraint (c, delta);
1536
        }
1537
    }
1538
  else if (c->rhs.type == DEREF)
1539
    {
1540
      /* x = *y */
1541
      if (!(get_varinfo (c->lhs.var)->is_special_var))
1542
        do_sd_constraint (graph, c, delta);
1543
    }
1544
  else
1545
    {
1546
      bitmap tmp;
1547
      bitmap solution;
1548
      bool flag = false;
1549
      unsigned int t;
1550
 
1551
      gcc_assert (c->rhs.type == SCALAR && c->lhs.type == SCALAR);
1552
      t = find (c->rhs.var);
1553
      solution = get_varinfo (t)->solution;
1554
      t = find (c->lhs.var);
1555
      tmp = get_varinfo (t)->solution;
1556
 
1557
      flag = set_union_with_increment (tmp, solution, c->rhs.offset);
1558
 
1559
      if (flag)
1560
        {
1561
          get_varinfo (t)->solution = tmp;
1562
          if (!TEST_BIT (changed, t))
1563
            {
1564
              SET_BIT (changed, t);
1565
              changed_count++;
1566
            }
1567
        }
1568
    }
1569
}
1570
 
1571
/* Initialize and return a new SCC info structure.  */
1572
 
1573
static struct scc_info *
1574
init_scc_info (size_t size)
1575
{
1576
  struct scc_info *si = XNEW (struct scc_info);
1577
  size_t i;
1578
 
1579
  si->current_index = 0;
1580
  si->visited = sbitmap_alloc (size);
1581
  sbitmap_zero (si->visited);
1582
  si->roots = sbitmap_alloc (size);
1583
  sbitmap_zero (si->roots);
1584
  si->node_mapping = XNEWVEC (unsigned int, size);
1585
  si->dfs = XCNEWVEC (unsigned int, size);
1586
 
1587
  for (i = 0; i < size; i++)
1588
    si->node_mapping[i] = i;
1589
 
1590
  si->scc_stack = VEC_alloc (unsigned, heap, 1);
1591
  return si;
1592
}
1593
 
1594
/* Free an SCC info structure pointed to by SI */
1595
 
1596
static void
1597
free_scc_info (struct scc_info *si)
1598
{
1599
  sbitmap_free (si->visited);
1600
  sbitmap_free (si->roots);
1601
  free (si->node_mapping);
1602
  free (si->dfs);
1603
  VEC_free (unsigned, heap, si->scc_stack);
1604
  free (si);
1605
}
1606
 
1607
 
1608
/* Find indirect cycles in GRAPH that occur, using strongly connected
1609
   components, and note them in the indirect cycles map.
1610
 
1611
   This technique comes from Ben Hardekopf and Calvin Lin,
1612
   "It Pays to be Lazy: Fast and Accurate Pointer Analysis for Millions of
1613
   Lines of Code", submitted to PLDI 2007.  */
1614
 
1615
static void
1616
find_indirect_cycles (constraint_graph_t graph)
1617
{
1618
  unsigned int i;
1619
  unsigned int size = graph->size;
1620
  struct scc_info *si = init_scc_info (size);
1621
 
1622
  for (i = 0; i < MIN (LAST_REF_NODE, size); i ++ )
1623
    if (!TEST_BIT (si->visited, i) && find (i) == i)
1624
      scc_visit (graph, si, i);
1625
 
1626
  free_scc_info (si);
1627
}
1628
 
1629
/* Compute a topological ordering for GRAPH, and store the result in the
1630
   topo_info structure TI.  */
1631
 
1632
static void
1633
compute_topo_order (constraint_graph_t graph,
1634
                    struct topo_info *ti)
1635
{
1636
  unsigned int i;
1637
  unsigned int size = VEC_length (varinfo_t, varmap);
1638
 
1639
  for (i = 0; i != size; ++i)
1640
    if (!TEST_BIT (ti->visited, i) && find (i) == i)
1641
      topo_visit (graph, ti, i);
1642
}
1643
 
1644
/* Perform offline variable substitution.
1645
 
1646
   This is a linear time way of identifying variables that must have
1647
   equivalent points-to sets, including those caused by static cycles,
1648
   and single entry subgraphs, in the constraint graph.
1649
 
1650
   The technique is described in "Off-line variable substitution for
1651
   scaling points-to analysis" by Atanas Rountev and Satish Chandra,
1652
   in "ACM SIGPLAN Notices" volume 35, number 5, pages 47-56.
1653
 
1654
   There is an optimal way to do this involving hash based value
1655
   numbering, once the technique is published i will implement it
1656
   here.
1657
 
1658
   The general method of finding equivalence classes is as follows:
1659
   Add fake nodes (REF nodes) and edges for *a = b and a = *b constraints.
1660
   Add fake nodes (ADDRESS nodes) and edges for a = &b constraints.
1661
   Initialize all non-REF/ADDRESS nodes to be direct nodes
1662
   For each SCC in the predecessor graph:
1663
      for each member (x) of the SCC
1664
         if x is not a direct node:
1665
           set rootnode(SCC) to be not a direct node
1666
         collapse node x into rootnode(SCC).
1667
      if rootnode(SCC) is not a direct node:
1668
        label rootnode(SCC) with a new equivalence class
1669
      else:
1670
        if all labeled predecessors of rootnode(SCC) have the same
1671
        label:
1672
          label rootnode(SCC) with this label
1673
        else:
1674
          label rootnode(SCC) with a new equivalence class
1675
 
1676
   All direct nodes with the same equivalence class can be replaced
1677
   with a single representative node.
1678
   All unlabeled nodes (label == 0) are not pointers and all edges
1679
   involving them can be eliminated.
1680
   We perform these optimizations during move_complex_constraints.
1681
*/
1682
 
1683
static int equivalence_class;
1684
 
1685
/* Recursive routine to find strongly connected components in GRAPH,
1686
   and label it's nodes with equivalence classes.
1687
   This is used during variable substitution to find cycles involving
1688
   the regular or implicit predecessors, and label them as equivalent.
1689
   The SCC finding algorithm used is the same as that for scc_visit.  */
1690
 
1691
static void
1692
label_visit (constraint_graph_t graph, struct scc_info *si, unsigned int n)
1693
{
1694
  unsigned int i;
1695
  bitmap_iterator bi;
1696
  unsigned int my_dfs;
1697
 
1698
  gcc_assert (si->node_mapping[n] == n);
1699
  SET_BIT (si->visited, n);
1700
  si->dfs[n] = si->current_index ++;
1701
  my_dfs = si->dfs[n];
1702
 
1703
  /* Visit all the successors.  */
1704
  EXECUTE_IF_IN_NONNULL_BITMAP (graph->preds[n], 0, i, bi)
1705
    {
1706
      unsigned int w = si->node_mapping[i];
1707
 
1708
      if (TEST_BIT (si->roots, w))
1709
        continue;
1710
 
1711
      if (!TEST_BIT (si->visited, w))
1712
        label_visit (graph, si, w);
1713
      {
1714
        unsigned int t = si->node_mapping[w];
1715
        unsigned int nnode = si->node_mapping[n];
1716
        gcc_assert (nnode == n);
1717
 
1718
        if (si->dfs[t] < si->dfs[nnode])
1719
          si->dfs[n] = si->dfs[t];
1720
      }
1721
    }
1722
 
1723
  /* Visit all the implicit predecessors.  */
1724
  EXECUTE_IF_IN_NONNULL_BITMAP (graph->implicit_preds[n], 0, i, bi)
1725
    {
1726
      unsigned int w = si->node_mapping[i];
1727
 
1728
      if (TEST_BIT (si->roots, w))
1729
        continue;
1730
 
1731
      if (!TEST_BIT (si->visited, w))
1732
        label_visit (graph, si, w);
1733
      {
1734
        unsigned int t = si->node_mapping[w];
1735
        unsigned int nnode = si->node_mapping[n];
1736
        gcc_assert (nnode == n);
1737
 
1738
        if (si->dfs[t] < si->dfs[nnode])
1739
          si->dfs[n] = si->dfs[t];
1740
      }
1741
    }
1742
 
1743
  /* See if any components have been identified.  */
1744
  if (si->dfs[n] == my_dfs)
1745
    {
1746
      while (VEC_length (unsigned, si->scc_stack) != 0
1747
             && si->dfs[VEC_last (unsigned, si->scc_stack)] >= my_dfs)
1748
        {
1749
          unsigned int w = VEC_pop (unsigned, si->scc_stack);
1750
          si->node_mapping[w] = n;
1751
 
1752
          if (!TEST_BIT (graph->direct_nodes, w))
1753
            RESET_BIT (graph->direct_nodes, n);
1754
        }
1755
      SET_BIT (si->roots, n);
1756
 
1757
      if (!TEST_BIT (graph->direct_nodes, n))
1758
        {
1759
          graph->label[n] = equivalence_class++;
1760
        }
1761
      else
1762
        {
1763
          unsigned int size = 0;
1764
          unsigned int firstlabel = ~0;
1765
 
1766
          EXECUTE_IF_IN_NONNULL_BITMAP (graph->preds[n], 0, i, bi)
1767
            {
1768
              unsigned int j = si->node_mapping[i];
1769
 
1770
              if (j == n || graph->label[j] == 0)
1771
                continue;
1772
 
1773
              if (firstlabel == (unsigned int)~0)
1774
                {
1775
                  firstlabel = graph->label[j];
1776
                  size++;
1777
                }
1778
              else if (graph->label[j] != firstlabel)
1779
                size++;
1780
            }
1781
 
1782
          if (size == 0)
1783
            graph->label[n] = 0;
1784
          else if (size == 1)
1785
            graph->label[n] = firstlabel;
1786
          else
1787
            graph->label[n] = equivalence_class++;
1788
        }
1789
    }
1790
  else
1791
    VEC_safe_push (unsigned, heap, si->scc_stack, n);
1792
}
1793
 
1794
/* Perform offline variable substitution, discovering equivalence
1795
   classes, and eliminating non-pointer variables.  */
1796
 
1797
static struct scc_info *
1798
perform_var_substitution (constraint_graph_t graph)
1799
{
1800
  unsigned int i;
1801
  unsigned int size = graph->size;
1802
  struct scc_info *si = init_scc_info (size);
1803
 
1804
  bitmap_obstack_initialize (&iteration_obstack);
1805
  equivalence_class = 0;
1806
 
1807
  /* We only need to visit the non-address nodes for labeling
1808
     purposes, as the address nodes will never have any predecessors,
1809
     because &x never appears on the LHS of a constraint.  */
1810
  for (i = 0; i < LAST_REF_NODE; i++)
1811
    if (!TEST_BIT (si->visited, si->node_mapping[i]))
1812
      label_visit (graph, si, si->node_mapping[i]);
1813
 
1814
  if (dump_file && (dump_flags & TDF_DETAILS))
1815
    for (i = 0; i < FIRST_REF_NODE; i++)
1816
      {
1817
        bool direct_node = TEST_BIT (graph->direct_nodes, i);
1818
        fprintf (dump_file,
1819
                 "Equivalence class for %s node id %d:%s is %d\n",
1820
                 direct_node ? "Direct node" : "Indirect node", i,
1821
                 get_varinfo (i)->name,
1822
                 graph->label[si->node_mapping[i]]);
1823
      }
1824
 
1825
  /* Quickly eliminate our non-pointer variables.  */
1826
 
1827
  for (i = 0; i < FIRST_REF_NODE; i++)
1828
    {
1829
      unsigned int node = si->node_mapping[i];
1830
 
1831
      if (graph->label[node] == 0 && TEST_BIT (graph->direct_nodes, node))
1832
        {
1833
          if (dump_file && (dump_flags & TDF_DETAILS))
1834
            fprintf (dump_file,
1835
                     "%s is a non-pointer variable, eliminating edges.\n",
1836
                     get_varinfo (node)->name);
1837
          stats.nonpointer_vars++;
1838
          clear_edges_for_node (graph, node);
1839
        }
1840
    }
1841
  return si;
1842
}
1843
 
1844
/* Free information that was only necessary for variable
1845
   substitution.  */
1846
 
1847
static void
1848
free_var_substitution_info (struct scc_info *si)
1849
{
1850
  free_scc_info (si);
1851
  free (graph->label);
1852
  free (graph->eq_rep);
1853
  sbitmap_free (graph->direct_nodes);
1854
  bitmap_obstack_release (&iteration_obstack);
1855
}
1856
 
1857
/* Return an existing node that is equivalent to NODE, which has
1858
   equivalence class LABEL, if one exists.  Return NODE otherwise.  */
1859
 
1860
static unsigned int
1861
find_equivalent_node (constraint_graph_t graph,
1862
                      unsigned int node, unsigned int label)
1863
{
1864
  /* If the address version of this variable is unused, we can
1865
     substitute it for anything else with the same label.
1866
     Otherwise, we know the pointers are equivalent, but not the
1867
     locations.  */
1868
 
1869
  if (graph->label[FIRST_ADDR_NODE + node] == 0)
1870
    {
1871
      gcc_assert (label < graph->size);
1872
 
1873
      if (graph->eq_rep[label] != -1)
1874
        {
1875
          /* Unify the two variables since we know they are equivalent.  */
1876
          if (unite (graph->eq_rep[label], node))
1877
            unify_nodes (graph, graph->eq_rep[label], node, false);
1878
          return graph->eq_rep[label];
1879
        }
1880
      else
1881
        {
1882
          graph->eq_rep[label] = node;
1883
        }
1884
    }
1885
  return node;
1886
}
1887
 
1888
/* Move complex constraints to the appropriate nodes, and collapse
1889
   variables we've discovered are equivalent during variable
1890
   substitution.  SI is the SCC_INFO that is the result of
1891
   perform_variable_substitution.  */
1892
 
1893
static void
1894
move_complex_constraints (constraint_graph_t graph,
1895
                          struct scc_info *si)
1896
{
1897
  int i;
1898
  unsigned int j;
1899
  constraint_t c;
1900
 
1901
  for (j = 0; j < graph->size; j++)
1902
    gcc_assert (find (j) == j);
1903
 
1904
  for (i = 0; VEC_iterate (constraint_t, constraints, i, c); i++)
1905
    {
1906
      struct constraint_expr lhs = c->lhs;
1907
      struct constraint_expr rhs = c->rhs;
1908
      unsigned int lhsvar = find (get_varinfo_fc (lhs.var)->id);
1909
      unsigned int rhsvar = find (get_varinfo_fc (rhs.var)->id);
1910
      unsigned int lhsnode, rhsnode;
1911
      unsigned int lhslabel, rhslabel;
1912
 
1913
      lhsnode = si->node_mapping[lhsvar];
1914
      rhsnode = si->node_mapping[rhsvar];
1915
      lhslabel = graph->label[lhsnode];
1916
      rhslabel = graph->label[rhsnode];
1917
 
1918
      /* See if it is really a non-pointer variable, and if so, ignore
1919
         the constraint.  */
1920
      if (lhslabel == 0)
1921
        {
1922
          if (!TEST_BIT (graph->direct_nodes, lhsnode))
1923
            lhslabel = graph->label[lhsnode] = equivalence_class++;
1924
          else
1925
            {
1926
              if (dump_file && (dump_flags & TDF_DETAILS))
1927
                {
1928
 
1929
                  fprintf (dump_file, "%s is a non-pointer variable,"
1930
                           "ignoring constraint:",
1931
                           get_varinfo (lhs.var)->name);
1932
                  dump_constraint (dump_file, c);
1933
                }
1934
              VEC_replace (constraint_t, constraints, i, NULL);
1935
              continue;
1936
            }
1937
        }
1938
 
1939
      if (rhslabel == 0)
1940
        {
1941
          if (!TEST_BIT (graph->direct_nodes, rhsnode))
1942
            rhslabel = graph->label[rhsnode] = equivalence_class++;
1943
          else
1944
            {
1945
              if (dump_file && (dump_flags & TDF_DETAILS))
1946
                {
1947
 
1948
                  fprintf (dump_file, "%s is a non-pointer variable,"
1949
                           "ignoring constraint:",
1950
                           get_varinfo (rhs.var)->name);
1951
                  dump_constraint (dump_file, c);
1952
                }
1953
              VEC_replace (constraint_t, constraints, i, NULL);
1954
              continue;
1955
            }
1956
        }
1957
 
1958
      lhsvar = find_equivalent_node (graph, lhsvar, lhslabel);
1959
      rhsvar = find_equivalent_node (graph, rhsvar, rhslabel);
1960
      c->lhs.var = lhsvar;
1961
      c->rhs.var = rhsvar;
1962
 
1963
      if (lhs.type == DEREF)
1964
        {
1965
          if (rhs.type == ADDRESSOF || rhsvar > anything_id)
1966
            insert_into_complex (graph, lhsvar, c);
1967
        }
1968
      else if (rhs.type == DEREF)
1969
        {
1970
          if (!(get_varinfo (lhsvar)->is_special_var))
1971
            insert_into_complex (graph, rhsvar, c);
1972
        }
1973
      else if (rhs.type != ADDRESSOF && lhsvar > anything_id
1974
               && (lhs.offset != 0 || rhs.offset != 0))
1975
        {
1976
          insert_into_complex (graph, rhsvar, c);
1977
        }
1978
 
1979
    }
1980
}
1981
 
1982
/* Eliminate indirect cycles involving NODE.  Return true if NODE was
1983
   part of an SCC, false otherwise.  */
1984
 
1985
static bool
1986
eliminate_indirect_cycles (unsigned int node)
1987
{
1988
  if (graph->indirect_cycles[node] != -1
1989
      && !bitmap_empty_p (get_varinfo (node)->solution))
1990
    {
1991
      unsigned int i;
1992
      VEC(unsigned,heap) *queue = NULL;
1993
      int queuepos;
1994
      unsigned int to = find (graph->indirect_cycles[node]);
1995
      bitmap_iterator bi;
1996
 
1997
      /* We can't touch the solution set and call unify_nodes
1998
         at the same time, because unify_nodes is going to do
1999
         bitmap unions into it. */
2000
 
2001
      EXECUTE_IF_SET_IN_BITMAP (get_varinfo (node)->solution, 0, i, bi)
2002
        {
2003
          if (find (i) == i && i != to)
2004
            {
2005
              if (unite (to, i))
2006
                VEC_safe_push (unsigned, heap, queue, i);
2007
            }
2008
        }
2009
 
2010
      for (queuepos = 0;
2011
           VEC_iterate (unsigned, queue, queuepos, i);
2012
           queuepos++)
2013
        {
2014
          unify_nodes (graph, to, i, true);
2015
        }
2016
      VEC_free (unsigned, heap, queue);
2017
      return true;
2018
    }
2019
  return false;
2020
}
2021
 
2022
/* Solve the constraint graph GRAPH using our worklist solver.
2023
   This is based on the PW* family of solvers from the "Efficient Field
2024
   Sensitive Pointer Analysis for C" paper.
2025
   It works by iterating over all the graph nodes, processing the complex
2026
   constraints and propagating the copy constraints, until everything stops
2027
   changed.  This corresponds to steps 6-8 in the solving list given above.  */
2028
 
2029
static void
2030
solve_graph (constraint_graph_t graph)
2031
{
2032
  unsigned int size = VEC_length (varinfo_t, varmap);
2033
  unsigned int i;
2034
  bitmap pts;
2035
 
2036
  changed_count = 0;
2037
  changed = sbitmap_alloc (size);
2038
  sbitmap_zero (changed);
2039
 
2040
  /* Mark all initial non-collapsed nodes as changed.  */
2041
  for (i = 0; i < size; i++)
2042
    {
2043
      varinfo_t ivi = get_varinfo (i);
2044
      if (find (i) == i && !bitmap_empty_p (ivi->solution)
2045
          && ((graph->succs[i] && !bitmap_empty_p (graph->succs[i]))
2046
              || VEC_length (constraint_t, graph->complex[i]) > 0))
2047
        {
2048
          SET_BIT (changed, i);
2049
          changed_count++;
2050
        }
2051
    }
2052
 
2053
  /* Allocate a bitmap to be used to store the changed bits.  */
2054
  pts = BITMAP_ALLOC (&pta_obstack);
2055
 
2056
  while (changed_count > 0)
2057
    {
2058
      unsigned int i;
2059
      struct topo_info *ti = init_topo_info ();
2060
      stats.iterations++;
2061
 
2062
      bitmap_obstack_initialize (&iteration_obstack);
2063
 
2064
      compute_topo_order (graph, ti);
2065
 
2066
      while (VEC_length (unsigned, ti->topo_order) != 0)
2067
        {
2068
 
2069
          i = VEC_pop (unsigned, ti->topo_order);
2070
 
2071
          /* If this variable is not a representative, skip it.  */
2072
          if (find (i) != i)
2073
            continue;
2074
 
2075
          /* In certain indirect cycle cases, we may merge this
2076
             variable to another.  */
2077
          if (eliminate_indirect_cycles (i) && find (i) != i)
2078
            continue;
2079
 
2080
          /* If the node has changed, we need to process the
2081
             complex constraints and outgoing edges again.  */
2082
          if (TEST_BIT (changed, i))
2083
            {
2084
              unsigned int j;
2085
              constraint_t c;
2086
              bitmap solution;
2087
              VEC(constraint_t,heap) *complex = graph->complex[i];
2088
              bool solution_empty;
2089
 
2090
              RESET_BIT (changed, i);
2091
              changed_count--;
2092
 
2093
              /* Compute the changed set of solution bits.  */
2094
              bitmap_and_compl (pts, get_varinfo (i)->solution,
2095
                                get_varinfo (i)->oldsolution);
2096
 
2097
              if (bitmap_empty_p (pts))
2098
                continue;
2099
 
2100
              bitmap_ior_into (get_varinfo (i)->oldsolution, pts);
2101
 
2102
              solution = get_varinfo (i)->solution;
2103
              solution_empty = bitmap_empty_p (solution);
2104
 
2105
              /* Process the complex constraints */
2106
              for (j = 0; VEC_iterate (constraint_t, complex, j, c); j++)
2107
                {
2108
                  /* The only complex constraint that can change our
2109
                     solution to non-empty, given an empty solution,
2110
                     is a constraint where the lhs side is receiving
2111
                     some set from elsewhere.  */
2112
                  if (!solution_empty || c->lhs.type != DEREF)
2113
                    do_complex_constraint (graph, c, pts);
2114
                }
2115
 
2116
              solution_empty = bitmap_empty_p (solution);
2117
 
2118
              if (!solution_empty)
2119
                {
2120
                  bitmap_iterator bi;
2121
 
2122
                  /* Propagate solution to all successors.  */
2123
                  EXECUTE_IF_IN_NONNULL_BITMAP (graph->succs[i],
2124
                                                0, j, bi)
2125
                    {
2126
                      bitmap tmp;
2127
                      bool flag;
2128
 
2129
                      unsigned int to = find (j);
2130
                      tmp = get_varinfo (to)->solution;
2131
                      flag = false;
2132
 
2133
                      /* Don't try to propagate to ourselves.  */
2134
                      if (to == i)
2135
                        continue;
2136
 
2137
                      flag = set_union_with_increment (tmp, pts, 0);
2138
 
2139
                      if (flag)
2140
                        {
2141
                          get_varinfo (to)->solution = tmp;
2142
                          if (!TEST_BIT (changed, to))
2143
                            {
2144
                              SET_BIT (changed, to);
2145
                              changed_count++;
2146
                            }
2147
                        }
2148
                    }
2149
                }
2150
            }
2151
        }
2152
      free_topo_info (ti);
2153
      bitmap_obstack_release (&iteration_obstack);
2154
    }
2155
 
2156
  BITMAP_FREE (pts);
2157
  sbitmap_free (changed);
2158
  bitmap_obstack_release (&oldpta_obstack);
2159
}
2160
 
2161
/* Map from trees to variable infos.  */
2162
static struct pointer_map_t *vi_for_tree;
2163
 
2164
 
2165
/* Insert ID as the variable id for tree T in the vi_for_tree map.  */
2166
 
2167
static void
2168
insert_vi_for_tree (tree t, varinfo_t vi)
2169
{
2170
  void **slot = pointer_map_insert (vi_for_tree, t);
2171
  gcc_assert (vi);
2172
  gcc_assert (*slot == NULL);
2173
  *slot = vi;
2174
}
2175
 
2176
/* Find the variable info for tree T in VI_FOR_TREE.  If T does not
2177
   exist in the map, return NULL, otherwise, return the varinfo we found.  */
2178
 
2179
static varinfo_t
2180
lookup_vi_for_tree (tree t)
2181
{
2182
  void **slot = pointer_map_contains (vi_for_tree, t);
2183
  if (slot == NULL)
2184
    return NULL;
2185
 
2186
  return (varinfo_t) *slot;
2187
}
2188
 
2189
/* Return a printable name for DECL  */
2190
 
2191
static const char *
2192
alias_get_name (tree decl)
2193
{
2194
  const char *res = get_name (decl);
2195
  char *temp;
2196
  int num_printed = 0;
2197
 
2198
  if (res != NULL)
2199
    return res;
2200
 
2201
  res = "NULL";
2202
  if (!dump_file)
2203
    return res;
2204
 
2205
  if (TREE_CODE (decl) == SSA_NAME)
2206
    {
2207
      num_printed = asprintf (&temp, "%s_%u",
2208
                              alias_get_name (SSA_NAME_VAR (decl)),
2209
                              SSA_NAME_VERSION (decl));
2210
    }
2211
  else if (DECL_P (decl))
2212
    {
2213
      num_printed = asprintf (&temp, "D.%u", DECL_UID (decl));
2214
    }
2215
  if (num_printed > 0)
2216
    {
2217
      res = ggc_strdup (temp);
2218
      free (temp);
2219
    }
2220
  return res;
2221
}
2222
 
2223
/* Find the variable id for tree T in the map.
2224
   If T doesn't exist in the map, create an entry for it and return it.  */
2225
 
2226
static varinfo_t
2227
get_vi_for_tree (tree t)
2228
{
2229
  void **slot = pointer_map_contains (vi_for_tree, t);
2230
  if (slot == NULL)
2231
    return get_varinfo (create_variable_info_for (t, alias_get_name (t)));
2232
 
2233
  return (varinfo_t) *slot;
2234
}
2235
 
2236
/* Get a constraint expression from an SSA_VAR_P node.  */
2237
 
2238
static struct constraint_expr
2239
get_constraint_exp_from_ssa_var (tree t)
2240
{
2241
  struct constraint_expr cexpr;
2242
 
2243
  gcc_assert (SSA_VAR_P (t) || DECL_P (t));
2244
 
2245
  /* For parameters, get at the points-to set for the actual parm
2246
     decl.  */
2247
  if (TREE_CODE (t) == SSA_NAME
2248
      && TREE_CODE (SSA_NAME_VAR (t)) == PARM_DECL
2249
      && default_def (SSA_NAME_VAR (t)) == t)
2250
    return get_constraint_exp_from_ssa_var (SSA_NAME_VAR (t));
2251
 
2252
  cexpr.type = SCALAR;
2253
 
2254
  cexpr.var = get_vi_for_tree (t)->id;
2255
  /* If we determine the result is "anything", and we know this is readonly,
2256
     say it points to readonly memory instead.  */
2257
  if (cexpr.var == anything_id && TREE_READONLY (t))
2258
    {
2259
      cexpr.type = ADDRESSOF;
2260
      cexpr.var = readonly_id;
2261
    }
2262
 
2263
  cexpr.offset = 0;
2264
  return cexpr;
2265
}
2266
 
2267
/* Process a completed constraint T, and add it to the constraint
2268
   list.  */
2269
 
2270
static void
2271
process_constraint (constraint_t t)
2272
{
2273
  struct constraint_expr rhs = t->rhs;
2274
  struct constraint_expr lhs = t->lhs;
2275
 
2276
  gcc_assert (rhs.var < VEC_length (varinfo_t, varmap));
2277
  gcc_assert (lhs.var < VEC_length (varinfo_t, varmap));
2278
 
2279
  if (lhs.type == DEREF)
2280
    get_varinfo (lhs.var)->directly_dereferenced = true;
2281
  if (rhs.type == DEREF)
2282
    get_varinfo (rhs.var)->directly_dereferenced = true;
2283
 
2284
  if (!use_field_sensitive)
2285
    {
2286
      t->rhs.offset = 0;
2287
      t->lhs.offset = 0;
2288
    }
2289
 
2290
  /* ANYTHING == ANYTHING is pointless.  */
2291
  if (lhs.var == anything_id && rhs.var == anything_id)
2292
    return;
2293
 
2294
  /* If we have &ANYTHING = something, convert to SOMETHING = &ANYTHING) */
2295
  else if (lhs.var == anything_id && lhs.type == ADDRESSOF)
2296
    {
2297
      rhs = t->lhs;
2298
      t->lhs = t->rhs;
2299
      t->rhs = rhs;
2300
      process_constraint (t);
2301
    }
2302
  /* This can happen in our IR with things like n->a = *p */
2303
  else if (rhs.type == DEREF && lhs.type == DEREF && rhs.var != anything_id)
2304
    {
2305
      /* Split into tmp = *rhs, *lhs = tmp */
2306
      tree rhsdecl = get_varinfo (rhs.var)->decl;
2307
      tree pointertype = TREE_TYPE (rhsdecl);
2308
      tree pointedtotype = TREE_TYPE (pointertype);
2309
      tree tmpvar = create_tmp_var_raw (pointedtotype, "doubledereftmp");
2310
      struct constraint_expr tmplhs = get_constraint_exp_from_ssa_var (tmpvar);
2311
 
2312
      /* If this is an aggregate of known size, we should have passed
2313
         this off to do_structure_copy, and it should have broken it
2314
         up.  */
2315
      gcc_assert (!AGGREGATE_TYPE_P (pointedtotype)
2316
                  || get_varinfo (rhs.var)->is_unknown_size_var);
2317
 
2318
      process_constraint (new_constraint (tmplhs, rhs));
2319
      process_constraint (new_constraint (lhs, tmplhs));
2320
    }
2321
  else
2322
    {
2323
      gcc_assert (rhs.type != ADDRESSOF || rhs.offset == 0);
2324
      VEC_safe_push (constraint_t, heap, constraints, t);
2325
    }
2326
}
2327
 
2328
/* Return true if T is a variable of a type that could contain
2329
   pointers.  */
2330
 
2331
static bool
2332
could_have_pointers (tree t)
2333
{
2334
  tree type = TREE_TYPE (t);
2335
 
2336
  if (POINTER_TYPE_P (type)
2337
      || AGGREGATE_TYPE_P (type)
2338
      || TREE_CODE (type) == COMPLEX_TYPE)
2339
    return true;
2340
 
2341
  return false;
2342
}
2343
 
2344
/* Return the position, in bits, of FIELD_DECL from the beginning of its
2345
   structure.  */
2346
 
2347
static unsigned HOST_WIDE_INT
2348
bitpos_of_field (const tree fdecl)
2349
{
2350
 
2351
  if (TREE_CODE (DECL_FIELD_OFFSET (fdecl)) != INTEGER_CST
2352
      || TREE_CODE (DECL_FIELD_BIT_OFFSET (fdecl)) != INTEGER_CST)
2353
    return -1;
2354
 
2355
  return (tree_low_cst (DECL_FIELD_OFFSET (fdecl), 1) * 8)
2356
         + tree_low_cst (DECL_FIELD_BIT_OFFSET (fdecl), 1);
2357
}
2358
 
2359
 
2360
/* Return true if an access to [ACCESSPOS, ACCESSSIZE]
2361
   overlaps with a field at [FIELDPOS, FIELDSIZE] */
2362
 
2363
static bool
2364
offset_overlaps_with_access (const unsigned HOST_WIDE_INT fieldpos,
2365
                             const unsigned HOST_WIDE_INT fieldsize,
2366
                             const unsigned HOST_WIDE_INT accesspos,
2367
                             const unsigned HOST_WIDE_INT accesssize)
2368
{
2369
  if (fieldpos == accesspos && fieldsize == accesssize)
2370
    return true;
2371
  if (accesspos >= fieldpos && accesspos < (fieldpos + fieldsize))
2372
    return true;
2373
  if (accesspos < fieldpos && (accesspos + accesssize > fieldpos))
2374
    return true;
2375
 
2376
  return false;
2377
}
2378
 
2379
/* Given a COMPONENT_REF T, return the constraint_expr for it.  */
2380
 
2381
static void
2382
get_constraint_for_component_ref (tree t, VEC(ce_s, heap) **results)
2383
{
2384
  tree orig_t = t;
2385
  HOST_WIDE_INT bitsize = -1;
2386
  HOST_WIDE_INT bitmaxsize = -1;
2387
  HOST_WIDE_INT bitpos;
2388
  tree forzero;
2389
  struct constraint_expr *result;
2390
  unsigned int beforelength = VEC_length (ce_s, *results);
2391
 
2392
  /* Some people like to do cute things like take the address of
2393
     &0->a.b */
2394
  forzero = t;
2395
  while (!SSA_VAR_P (forzero) && !CONSTANT_CLASS_P (forzero))
2396
    forzero = TREE_OPERAND (forzero, 0);
2397
 
2398
  if (CONSTANT_CLASS_P (forzero) && integer_zerop (forzero))
2399
    {
2400
      struct constraint_expr temp;
2401
 
2402
      temp.offset = 0;
2403
      temp.var = integer_id;
2404
      temp.type = SCALAR;
2405
      VEC_safe_push (ce_s, heap, *results, &temp);
2406
      return;
2407
    }
2408
 
2409
  t = get_ref_base_and_extent (t, &bitpos, &bitsize, &bitmaxsize);
2410
 
2411
  /* String constants are readonly, so there is nothing to really do
2412
     here.  */
2413
  if (TREE_CODE (t) == STRING_CST)
2414
    return;
2415
 
2416
  get_constraint_for (t, results);
2417
  result = VEC_last (ce_s, *results);
2418
  result->offset = bitpos;
2419
 
2420
  gcc_assert (beforelength + 1 == VEC_length (ce_s, *results));
2421
 
2422
  /* This can also happen due to weird offsetof type macros.  */
2423
  if (TREE_CODE (t) != ADDR_EXPR && result->type == ADDRESSOF)
2424
    result->type = SCALAR;
2425
 
2426
  if (result->type == SCALAR)
2427
    {
2428
      /* In languages like C, you can access one past the end of an
2429
         array.  You aren't allowed to dereference it, so we can
2430
         ignore this constraint. When we handle pointer subtraction,
2431
         we may have to do something cute here.  */
2432
 
2433
      if (result->offset < get_varinfo (result->var)->fullsize
2434
          && bitmaxsize != 0)
2435
        {
2436
          /* It's also not true that the constraint will actually start at the
2437
             right offset, it may start in some padding.  We only care about
2438
             setting the constraint to the first actual field it touches, so
2439
             walk to find it.  */
2440
          varinfo_t curr;
2441
          for (curr = get_varinfo (result->var); curr; curr = curr->next)
2442
            {
2443
              if (offset_overlaps_with_access (curr->offset, curr->size,
2444
                                               result->offset, bitmaxsize))
2445
                {
2446
                  result->var = curr->id;
2447
                  break;
2448
                }
2449
            }
2450
          /* assert that we found *some* field there. The user couldn't be
2451
             accessing *only* padding.  */
2452
          /* Still the user could access one past the end of an array
2453
             embedded in a struct resulting in accessing *only* padding.  */
2454
          gcc_assert (curr || TREE_CODE (TREE_TYPE (orig_t)) == ARRAY_TYPE
2455
                      || ref_contains_array_ref (orig_t));
2456
        }
2457
      else if (bitmaxsize == 0)
2458
        {
2459
          if (dump_file && (dump_flags & TDF_DETAILS))
2460
            fprintf (dump_file, "Access to zero-sized part of variable,"
2461
                     "ignoring\n");
2462
        }
2463
      else
2464
        if (dump_file && (dump_flags & TDF_DETAILS))
2465
          fprintf (dump_file, "Access to past the end of variable, ignoring\n");
2466
 
2467
      result->offset = 0;
2468
    }
2469
}
2470
 
2471
 
2472
/* Dereference the constraint expression CONS, and return the result.
2473
   DEREF (ADDRESSOF) = SCALAR
2474
   DEREF (SCALAR) = DEREF
2475
   DEREF (DEREF) = (temp = DEREF1; result = DEREF(temp))
2476
   This is needed so that we can handle dereferencing DEREF constraints.  */
2477
 
2478
static void
2479
do_deref (VEC (ce_s, heap) **constraints)
2480
{
2481
  struct constraint_expr *c;
2482
  unsigned int i = 0;
2483
 
2484
  for (i = 0; VEC_iterate (ce_s, *constraints, i, c); i++)
2485
    {
2486
      if (c->type == SCALAR)
2487
        c->type = DEREF;
2488
      else if (c->type == ADDRESSOF)
2489
        c->type = SCALAR;
2490
      else if (c->type == DEREF)
2491
        {
2492
          tree tmpvar = create_tmp_var_raw (ptr_type_node, "dereftmp");
2493
          struct constraint_expr tmplhs = get_constraint_exp_from_ssa_var (tmpvar);
2494
          process_constraint (new_constraint (tmplhs, *c));
2495
          c->var = tmplhs.var;
2496
        }
2497
      else
2498
        gcc_unreachable ();
2499
    }
2500
}
2501
 
2502
/* Create a nonlocal variable of TYPE to represent nonlocals we can
2503
   alias.  */
2504
 
2505
static tree
2506
create_nonlocal_var (tree type)
2507
{
2508
  tree nonlocal = create_tmp_var_raw (type, "NONLOCAL");
2509
 
2510
  if (referenced_vars)
2511
    add_referenced_var (nonlocal);
2512
 
2513
  DECL_EXTERNAL (nonlocal) = 1;
2514
  return nonlocal;
2515
}
2516
 
2517
/* Given a tree T, return the constraint expression for it.  */
2518
 
2519
static void
2520
get_constraint_for (tree t, VEC (ce_s, heap) **results)
2521
{
2522
  struct constraint_expr temp;
2523
 
2524
  /* x = integer is all glommed to a single variable, which doesn't
2525
     point to anything by itself.  That is, of course, unless it is an
2526
     integer constant being treated as a pointer, in which case, we
2527
     will return that this is really the addressof anything.  This
2528
     happens below, since it will fall into the default case. The only
2529
     case we know something about an integer treated like a pointer is
2530
     when it is the NULL pointer, and then we just say it points to
2531
     NULL.  */
2532
  if (TREE_CODE (t) == INTEGER_CST
2533
      && !POINTER_TYPE_P (TREE_TYPE (t)))
2534
    {
2535
      temp.var = integer_id;
2536
      temp.type = SCALAR;
2537
      temp.offset = 0;
2538
      VEC_safe_push (ce_s, heap, *results, &temp);
2539
      return;
2540
    }
2541
  else if (TREE_CODE (t) == INTEGER_CST
2542
           && integer_zerop (t))
2543
    {
2544
      temp.var = nothing_id;
2545
      temp.type = ADDRESSOF;
2546
      temp.offset = 0;
2547
      VEC_safe_push (ce_s, heap, *results, &temp);
2548
      return;
2549
    }
2550
 
2551
  switch (TREE_CODE_CLASS (TREE_CODE (t)))
2552
    {
2553
    case tcc_expression:
2554
      {
2555
        switch (TREE_CODE (t))
2556
          {
2557
          case ADDR_EXPR:
2558
            {
2559
              struct constraint_expr *c;
2560
              unsigned int i;
2561
              tree exp = TREE_OPERAND (t, 0);
2562
              tree pttype = TREE_TYPE (TREE_TYPE (t));
2563
 
2564
              get_constraint_for (exp, results);
2565
 
2566
              /* Make sure we capture constraints to all elements
2567
                 of an array.  */
2568
              if ((handled_component_p (exp)
2569
                   && ref_contains_array_ref (exp))
2570
                  || TREE_CODE (TREE_TYPE (exp)) == ARRAY_TYPE)
2571
                {
2572
                  struct constraint_expr *origrhs;
2573
                  varinfo_t origvar;
2574
                  struct constraint_expr tmp;
2575
 
2576
                  if (VEC_length (ce_s, *results) == 0)
2577
                    return;
2578
 
2579
                  gcc_assert (VEC_length (ce_s, *results) == 1);
2580
                  origrhs = VEC_last (ce_s, *results);
2581
                  tmp = *origrhs;
2582
                  VEC_pop (ce_s, *results);
2583
                  origvar = get_varinfo (origrhs->var);
2584
                  for (; origvar; origvar = origvar->next)
2585
                    {
2586
                      tmp.var = origvar->id;
2587
                      VEC_safe_push (ce_s, heap, *results, &tmp);
2588
                    }
2589
                }
2590
              else if (VEC_length (ce_s, *results) == 1
2591
                       && (AGGREGATE_TYPE_P (pttype)
2592
                           || TREE_CODE (pttype) == COMPLEX_TYPE))
2593
                {
2594
                  struct constraint_expr *origrhs;
2595
                  varinfo_t origvar;
2596
                  struct constraint_expr tmp;
2597
 
2598
                  gcc_assert (VEC_length (ce_s, *results) == 1);
2599
                  origrhs = VEC_last (ce_s, *results);
2600
                  tmp = *origrhs;
2601
                  VEC_pop (ce_s, *results);
2602
                  origvar = get_varinfo (origrhs->var);
2603
                  for (; origvar; origvar = origvar->next)
2604
                    {
2605
                      tmp.var = origvar->id;
2606
                      VEC_safe_push (ce_s, heap, *results, &tmp);
2607
                    }
2608
                }
2609
 
2610
              for (i = 0; VEC_iterate (ce_s, *results, i, c); i++)
2611
                {
2612
                  if (c->type == DEREF)
2613
                    c->type = SCALAR;
2614
                  else
2615
                    c->type = ADDRESSOF;
2616
                }
2617
              return;
2618
            }
2619
            break;
2620
          case CALL_EXPR:
2621
            /* XXX: In interprocedural mode, if we didn't have the
2622
               body, we would need to do *each pointer argument =
2623
               &ANYTHING added.  */
2624
            if (call_expr_flags (t) & (ECF_MALLOC | ECF_MAY_BE_ALLOCA))
2625
              {
2626
                varinfo_t vi;
2627
                tree heapvar = heapvar_lookup (t);
2628
 
2629
                if (heapvar == NULL)
2630
                  {
2631
                    heapvar = create_tmp_var_raw (ptr_type_node, "HEAP");
2632
                    DECL_EXTERNAL (heapvar) = 1;
2633
                    if (referenced_vars)
2634
                      add_referenced_var (heapvar);
2635
                    heapvar_insert (t, heapvar);
2636
                  }
2637
 
2638
                temp.var = create_variable_info_for (heapvar,
2639
                                                     alias_get_name (heapvar));
2640
 
2641
                vi = get_varinfo (temp.var);
2642
                vi->is_artificial_var = 1;
2643
                vi->is_heap_var = 1;
2644
                temp.type = ADDRESSOF;
2645
                temp.offset = 0;
2646
                VEC_safe_push (ce_s, heap, *results, &temp);
2647
                return;
2648
              }
2649
            else
2650
              {
2651
                temp.var = escaped_vars_id;
2652
                temp.type = SCALAR;
2653
                temp.offset = 0;
2654
                VEC_safe_push (ce_s, heap, *results, &temp);
2655
                return;
2656
              }
2657
            break;
2658
          default:
2659
            {
2660
              temp.type = ADDRESSOF;
2661
              temp.var = anything_id;
2662
              temp.offset = 0;
2663
              VEC_safe_push (ce_s, heap, *results, &temp);
2664
              return;
2665
            }
2666
          }
2667
      }
2668
    case tcc_reference:
2669
      {
2670
        switch (TREE_CODE (t))
2671
          {
2672
          case INDIRECT_REF:
2673
            {
2674
              get_constraint_for (TREE_OPERAND (t, 0), results);
2675
              do_deref (results);
2676
              return;
2677
            }
2678
          case ARRAY_REF:
2679
          case ARRAY_RANGE_REF:
2680
          case COMPONENT_REF:
2681
            get_constraint_for_component_ref (t, results);
2682
            return;
2683
          default:
2684
            {
2685
              temp.type = ADDRESSOF;
2686
              temp.var = anything_id;
2687
              temp.offset = 0;
2688
              VEC_safe_push (ce_s, heap, *results, &temp);
2689
              return;
2690
            }
2691
          }
2692
      }
2693
    case tcc_unary:
2694
      {
2695
        switch (TREE_CODE (t))
2696
          {
2697
          case NOP_EXPR:
2698
          case CONVERT_EXPR:
2699
          case NON_LVALUE_EXPR:
2700
            {
2701
              tree op = TREE_OPERAND (t, 0);
2702
 
2703
              /* Cast from non-pointer to pointers are bad news for us.
2704
                 Anything else, we see through */
2705
              if (!(POINTER_TYPE_P (TREE_TYPE (t))
2706
                    && ! POINTER_TYPE_P (TREE_TYPE (op))))
2707
                {
2708
                  get_constraint_for (op, results);
2709
                  return;
2710
                }
2711
 
2712
              /* FALLTHRU  */
2713
            }
2714
          default:
2715
            {
2716
              temp.type = ADDRESSOF;
2717
              temp.var = anything_id;
2718
              temp.offset = 0;
2719
              VEC_safe_push (ce_s, heap, *results, &temp);
2720
              return;
2721
            }
2722
          }
2723
      }
2724
    case tcc_exceptional:
2725
      {
2726
        switch (TREE_CODE (t))
2727
          {
2728
          case PHI_NODE:
2729
            {
2730
              get_constraint_for (PHI_RESULT (t), results);
2731
              return;
2732
            }
2733
            break;
2734
          case SSA_NAME:
2735
            {
2736
              struct constraint_expr temp;
2737
              temp = get_constraint_exp_from_ssa_var (t);
2738
              VEC_safe_push (ce_s, heap, *results, &temp);
2739
              return;
2740
            }
2741
            break;
2742
          default:
2743
            {
2744
              temp.type = ADDRESSOF;
2745
              temp.var = anything_id;
2746
              temp.offset = 0;
2747
              VEC_safe_push (ce_s, heap, *results, &temp);
2748
              return;
2749
            }
2750
          }
2751
      }
2752
    case tcc_declaration:
2753
      {
2754
        struct constraint_expr temp;
2755
        temp = get_constraint_exp_from_ssa_var (t);
2756
        VEC_safe_push (ce_s, heap, *results, &temp);
2757
        return;
2758
      }
2759
    default:
2760
      {
2761
        temp.type = ADDRESSOF;
2762
        temp.var = anything_id;
2763
        temp.offset = 0;
2764
        VEC_safe_push (ce_s, heap, *results, &temp);
2765
        return;
2766
      }
2767
    }
2768
}
2769
 
2770
 
2771
/* Handle the structure copy case where we have a simple structure copy
2772
   between LHS and RHS that is of SIZE (in bits)
2773
 
2774
   For each field of the lhs variable (lhsfield)
2775
     For each field of the rhs variable at lhsfield.offset (rhsfield)
2776
       add the constraint lhsfield = rhsfield
2777
 
2778
   If we fail due to some kind of type unsafety or other thing we
2779
   can't handle, return false.  We expect the caller to collapse the
2780
   variable in that case.  */
2781
 
2782
static bool
2783
do_simple_structure_copy (const struct constraint_expr lhs,
2784
                          const struct constraint_expr rhs,
2785
                          const unsigned HOST_WIDE_INT size)
2786
{
2787
  varinfo_t p = get_varinfo (lhs.var);
2788
  unsigned HOST_WIDE_INT pstart, last;
2789
  pstart = p->offset;
2790
  last = p->offset + size;
2791
  for (; p && p->offset < last; p = p->next)
2792
    {
2793
      varinfo_t q;
2794
      struct constraint_expr templhs = lhs;
2795
      struct constraint_expr temprhs = rhs;
2796
      unsigned HOST_WIDE_INT fieldoffset;
2797
 
2798
      templhs.var = p->id;
2799
      q = get_varinfo (temprhs.var);
2800
      fieldoffset = p->offset - pstart;
2801
      q = first_vi_for_offset (q, q->offset + fieldoffset);
2802
      if (!q)
2803
        return false;
2804
      temprhs.var = q->id;
2805
      process_constraint (new_constraint (templhs, temprhs));
2806
    }
2807
  return true;
2808
}
2809
 
2810
 
2811
/* Handle the structure copy case where we have a  structure copy between a
2812
   aggregate on the LHS and a dereference of a pointer on the RHS
2813
   that is of SIZE (in bits)
2814
 
2815
   For each field of the lhs variable (lhsfield)
2816
       rhs.offset = lhsfield->offset
2817
       add the constraint lhsfield = rhs
2818
*/
2819
 
2820
static void
2821
do_rhs_deref_structure_copy (const struct constraint_expr lhs,
2822
                             const struct constraint_expr rhs,
2823
                             const unsigned HOST_WIDE_INT size)
2824
{
2825
  varinfo_t p = get_varinfo (lhs.var);
2826
  unsigned HOST_WIDE_INT pstart,last;
2827
  pstart = p->offset;
2828
  last = p->offset + size;
2829
 
2830
  for (; p && p->offset < last; p = p->next)
2831
    {
2832
      varinfo_t q;
2833
      struct constraint_expr templhs = lhs;
2834
      struct constraint_expr temprhs = rhs;
2835
      unsigned HOST_WIDE_INT fieldoffset;
2836
 
2837
 
2838
      if (templhs.type == SCALAR)
2839
        templhs.var = p->id;
2840
      else
2841
        templhs.offset = p->offset;
2842
 
2843
      q = get_varinfo (temprhs.var);
2844
      fieldoffset = p->offset - pstart;
2845
      temprhs.offset += fieldoffset;
2846
      process_constraint (new_constraint (templhs, temprhs));
2847
    }
2848
}
2849
 
2850
/* Handle the structure copy case where we have a structure copy
2851
   between a aggregate on the RHS and a dereference of a pointer on
2852
   the LHS that is of SIZE (in bits)
2853
 
2854
   For each field of the rhs variable (rhsfield)
2855
       lhs.offset = rhsfield->offset
2856
       add the constraint lhs = rhsfield
2857
*/
2858
 
2859
static void
2860
do_lhs_deref_structure_copy (const struct constraint_expr lhs,
2861
                             const struct constraint_expr rhs,
2862
                             const unsigned HOST_WIDE_INT size)
2863
{
2864
  varinfo_t p = get_varinfo (rhs.var);
2865
  unsigned HOST_WIDE_INT pstart,last;
2866
  pstart = p->offset;
2867
  last = p->offset + size;
2868
 
2869
  for (; p && p->offset < last; p = p->next)
2870
    {
2871
      varinfo_t q;
2872
      struct constraint_expr templhs = lhs;
2873
      struct constraint_expr temprhs = rhs;
2874
      unsigned HOST_WIDE_INT fieldoffset;
2875
 
2876
 
2877
      if (temprhs.type == SCALAR)
2878
        temprhs.var = p->id;
2879
      else
2880
        temprhs.offset = p->offset;
2881
 
2882
      q = get_varinfo (templhs.var);
2883
      fieldoffset = p->offset - pstart;
2884
      templhs.offset += fieldoffset;
2885
      process_constraint (new_constraint (templhs, temprhs));
2886
    }
2887
}
2888
 
2889
/* Sometimes, frontends like to give us bad type information.  This
2890
   function will collapse all the fields from VAR to the end of VAR,
2891
   into VAR, so that we treat those fields as a single variable.
2892
   We return the variable they were collapsed into.  */
2893
 
2894
static unsigned int
2895
collapse_rest_of_var (unsigned int var)
2896
{
2897
  varinfo_t currvar = get_varinfo (var);
2898
  varinfo_t field;
2899
 
2900
  for (field = currvar->next; field; field = field->next)
2901
    {
2902
      if (dump_file)
2903
        fprintf (dump_file, "Type safety: Collapsing var %s into %s\n",
2904
                 field->name, currvar->name);
2905
 
2906
      gcc_assert (!field->collapsed_to);
2907
      field->collapsed_to = currvar;
2908
    }
2909
 
2910
  currvar->next = NULL;
2911
  currvar->size = currvar->fullsize - currvar->offset;
2912
 
2913
  return currvar->id;
2914
}
2915
 
2916
/* Handle aggregate copies by expanding into copies of the respective
2917
   fields of the structures.  */
2918
 
2919
static void
2920
do_structure_copy (tree lhsop, tree rhsop)
2921
{
2922
  struct constraint_expr lhs, rhs, tmp;
2923
  VEC (ce_s, heap) *lhsc = NULL, *rhsc = NULL;
2924
  varinfo_t p;
2925
  unsigned HOST_WIDE_INT lhssize;
2926
  unsigned HOST_WIDE_INT rhssize;
2927
 
2928
  get_constraint_for (lhsop, &lhsc);
2929
  get_constraint_for (rhsop, &rhsc);
2930
  gcc_assert (VEC_length (ce_s, lhsc) == 1);
2931
  gcc_assert (VEC_length (ce_s, rhsc) == 1);
2932
  lhs = *(VEC_last (ce_s, lhsc));
2933
  rhs = *(VEC_last (ce_s, rhsc));
2934
 
2935
  VEC_free (ce_s, heap, lhsc);
2936
  VEC_free (ce_s, heap, rhsc);
2937
 
2938
  /* If we have special var = x, swap it around.  */
2939
  if (lhs.var <= integer_id && !(get_varinfo (rhs.var)->is_special_var))
2940
    {
2941
      tmp = lhs;
2942
      lhs = rhs;
2943
      rhs = tmp;
2944
    }
2945
 
2946
  /*  This is fairly conservative for the RHS == ADDRESSOF case, in that it's
2947
      possible it's something we could handle.  However, most cases falling
2948
      into this are dealing with transparent unions, which are slightly
2949
      weird. */
2950
  if (rhs.type == ADDRESSOF && !(get_varinfo (rhs.var)->is_special_var))
2951
    {
2952
      rhs.type = ADDRESSOF;
2953
      rhs.var = anything_id;
2954
    }
2955
 
2956
  /* If the RHS is a special var, or an addressof, set all the LHS fields to
2957
     that special var.  */
2958
  if (rhs.var <= integer_id)
2959
    {
2960
      for (p = get_varinfo (lhs.var); p; p = p->next)
2961
        {
2962
          struct constraint_expr templhs = lhs;
2963
          struct constraint_expr temprhs = rhs;
2964
 
2965
          if (templhs.type == SCALAR )
2966
            templhs.var = p->id;
2967
          else
2968
            templhs.offset += p->offset;
2969
          process_constraint (new_constraint (templhs, temprhs));
2970
        }
2971
    }
2972
  else
2973
    {
2974
      tree rhstype = TREE_TYPE (rhsop);
2975
      tree lhstype = TREE_TYPE (lhsop);
2976
      tree rhstypesize;
2977
      tree lhstypesize;
2978
 
2979
      lhstypesize = DECL_P (lhsop) ? DECL_SIZE (lhsop) : TYPE_SIZE (lhstype);
2980
      rhstypesize = DECL_P (rhsop) ? DECL_SIZE (rhsop) : TYPE_SIZE (rhstype);
2981
 
2982
      /* If we have a variably sized types on the rhs or lhs, and a deref
2983
         constraint, add the constraint, lhsconstraint = &ANYTHING.
2984
         This is conservatively correct because either the lhs is an unknown
2985
         sized var (if the constraint is SCALAR), or the lhs is a DEREF
2986
         constraint, and every variable it can point to must be unknown sized
2987
         anyway, so we don't need to worry about fields at all.  */
2988
      if ((rhs.type == DEREF && TREE_CODE (rhstypesize) != INTEGER_CST)
2989
          || (lhs.type == DEREF && TREE_CODE (lhstypesize) != INTEGER_CST))
2990
        {
2991
          rhs.var = anything_id;
2992
          rhs.type = ADDRESSOF;
2993
          rhs.offset = 0;
2994
          process_constraint (new_constraint (lhs, rhs));
2995
          return;
2996
        }
2997
 
2998
      /* The size only really matters insofar as we don't set more or less of
2999
         the variable.  If we hit an unknown size var, the size should be the
3000
         whole darn thing.  */
3001
      if (get_varinfo (rhs.var)->is_unknown_size_var)
3002
        rhssize = ~0;
3003
      else
3004
        rhssize = TREE_INT_CST_LOW (rhstypesize);
3005
 
3006
      if (get_varinfo (lhs.var)->is_unknown_size_var)
3007
        lhssize = ~0;
3008
      else
3009
        lhssize = TREE_INT_CST_LOW (lhstypesize);
3010
 
3011
 
3012
      if (rhs.type == SCALAR && lhs.type == SCALAR)
3013
        {
3014
          if (!do_simple_structure_copy (lhs, rhs, MIN (lhssize, rhssize)))
3015
            {
3016
              lhs.var = collapse_rest_of_var (lhs.var);
3017
              rhs.var = collapse_rest_of_var (rhs.var);
3018
              lhs.offset = 0;
3019
              rhs.offset = 0;
3020
              lhs.type = SCALAR;
3021
              rhs.type = SCALAR;
3022
              process_constraint (new_constraint (lhs, rhs));
3023
            }
3024
        }
3025
      else if (lhs.type != DEREF && rhs.type == DEREF)
3026
        do_rhs_deref_structure_copy (lhs, rhs, MIN (lhssize, rhssize));
3027
      else if (lhs.type == DEREF && rhs.type != DEREF)
3028
        do_lhs_deref_structure_copy (lhs, rhs, MIN (lhssize, rhssize));
3029
      else
3030
        {
3031
          tree pointedtotype = lhstype;
3032
          tree tmpvar;
3033
 
3034
          gcc_assert (rhs.type == DEREF && lhs.type == DEREF);
3035
          tmpvar = create_tmp_var_raw (pointedtotype, "structcopydereftmp");
3036
          do_structure_copy (tmpvar, rhsop);
3037
          do_structure_copy (lhsop, tmpvar);
3038
        }
3039
    }
3040
}
3041
 
3042
 
3043
/* Update related alias information kept in AI.  This is used when
3044
   building name tags, alias sets and deciding grouping heuristics.
3045
   STMT is the statement to process.  This function also updates
3046
   ADDRESSABLE_VARS.  */
3047
 
3048
static void
3049
update_alias_info (tree stmt, struct alias_info *ai)
3050
{
3051
  bitmap addr_taken;
3052
  use_operand_p use_p;
3053
  ssa_op_iter iter;
3054
  enum escape_type stmt_escape_type = is_escape_site (stmt);
3055
  tree op;
3056
 
3057
  if (stmt_escape_type == ESCAPE_TO_CALL
3058
      || stmt_escape_type == ESCAPE_TO_PURE_CONST)
3059
    {
3060
      ai->num_calls_found++;
3061
      if (stmt_escape_type == ESCAPE_TO_PURE_CONST)
3062
        ai->num_pure_const_calls_found++;
3063
    }
3064
 
3065
  /* Mark all the variables whose address are taken by the statement.  */
3066
  addr_taken = addresses_taken (stmt);
3067
  if (addr_taken)
3068
    {
3069
      bitmap_ior_into (addressable_vars, addr_taken);
3070
 
3071
      /* If STMT is an escape point, all the addresses taken by it are
3072
         call-clobbered.  */
3073
      if (stmt_escape_type != NO_ESCAPE)
3074
        {
3075
          bitmap_iterator bi;
3076
          unsigned i;
3077
 
3078
          EXECUTE_IF_SET_IN_BITMAP (addr_taken, 0, i, bi)
3079
            {
3080
              tree rvar = referenced_var (i);
3081
              if (!unmodifiable_var_p (rvar))
3082
                mark_call_clobbered (rvar, stmt_escape_type);
3083
            }
3084
        }
3085
    }
3086
 
3087
  /* Process each operand use.  If an operand may be aliased, keep
3088
     track of how many times it's being used.  For pointers, determine
3089
     whether they are dereferenced by the statement, or whether their
3090
     value escapes, etc.  */
3091
  FOR_EACH_PHI_OR_STMT_USE (use_p, stmt, iter, SSA_OP_USE)
3092
    {
3093
      tree op, var;
3094
      var_ann_t v_ann;
3095
      struct ptr_info_def *pi;
3096
      bool is_store, is_potential_deref;
3097
      unsigned num_uses, num_derefs;
3098
 
3099
      op = USE_FROM_PTR (use_p);
3100
 
3101
      /* If STMT is a PHI node, OP may be an ADDR_EXPR.  If so, add it
3102
         to the set of addressable variables.  */
3103
      if (TREE_CODE (op) == ADDR_EXPR)
3104
        {
3105
          gcc_assert (TREE_CODE (stmt) == PHI_NODE);
3106
 
3107
          /* PHI nodes don't have annotations for pinning the set
3108
             of addresses taken, so we collect them here.
3109
 
3110
             FIXME, should we allow PHI nodes to have annotations
3111
             so that they can be treated like regular statements?
3112
             Currently, they are treated as second-class
3113
             statements.  */
3114
          add_to_addressable_set (TREE_OPERAND (op, 0), &addressable_vars);
3115
          continue;
3116
        }
3117
 
3118
      /* Ignore constants.  */
3119
      if (TREE_CODE (op) != SSA_NAME)
3120
        continue;
3121
 
3122
      var = SSA_NAME_VAR (op);
3123
      v_ann = var_ann (var);
3124
 
3125
      /* The base variable of an ssa name must be a GIMPLE register, and thus
3126
         it cannot be aliased.  */
3127
      gcc_assert (!may_be_aliased (var));
3128
 
3129
      /* We are only interested in pointers.  */
3130
      if (!POINTER_TYPE_P (TREE_TYPE (op)))
3131
        continue;
3132
 
3133
      pi = get_ptr_info (op);
3134
 
3135
      /* Add OP to AI->PROCESSED_PTRS, if it's not there already.  */
3136
      if (!TEST_BIT (ai->ssa_names_visited, SSA_NAME_VERSION (op)))
3137
        {
3138
          SET_BIT (ai->ssa_names_visited, SSA_NAME_VERSION (op));
3139
          VEC_safe_push (tree, heap, ai->processed_ptrs, op);
3140
        }
3141
 
3142
      /* If STMT is a PHI node, then it will not have pointer
3143
         dereferences and it will not be an escape point.  */
3144
      if (TREE_CODE (stmt) == PHI_NODE)
3145
        continue;
3146
 
3147
      /* Determine whether OP is a dereferenced pointer, and if STMT
3148
         is an escape point, whether OP escapes.  */
3149
      count_uses_and_derefs (op, stmt, &num_uses, &num_derefs, &is_store);
3150
 
3151
      /* Handle a corner case involving address expressions of the
3152
         form '&PTR->FLD'.  The problem with these expressions is that
3153
         they do not represent a dereference of PTR.  However, if some
3154
         other transformation propagates them into an INDIRECT_REF
3155
         expression, we end up with '*(&PTR->FLD)' which is folded
3156
         into 'PTR->FLD'.
3157
 
3158
         So, if the original code had no other dereferences of PTR,
3159
         the aliaser will not create memory tags for it, and when
3160
         &PTR->FLD gets propagated to INDIRECT_REF expressions, the
3161
         memory operations will receive no V_MAY_DEF/VUSE operands.
3162
 
3163
         One solution would be to have count_uses_and_derefs consider
3164
         &PTR->FLD a dereference of PTR.  But that is wrong, since it
3165
         is not really a dereference but an offset calculation.
3166
 
3167
         What we do here is to recognize these special ADDR_EXPR
3168
         nodes.  Since these expressions are never GIMPLE values (they
3169
         are not GIMPLE invariants), they can only appear on the RHS
3170
         of an assignment and their base address is always an
3171
         INDIRECT_REF expression.  */
3172
      is_potential_deref = false;
3173
      if (TREE_CODE (stmt) == MODIFY_EXPR
3174
          && TREE_CODE (TREE_OPERAND (stmt, 1)) == ADDR_EXPR
3175
          && !is_gimple_val (TREE_OPERAND (stmt, 1)))
3176
        {
3177
          /* If the RHS if of the form &PTR->FLD and PTR == OP, then
3178
             this represents a potential dereference of PTR.  */
3179
          tree rhs = TREE_OPERAND (stmt, 1);
3180
          tree base = get_base_address (TREE_OPERAND (rhs, 0));
3181
          if (TREE_CODE (base) == INDIRECT_REF
3182
              && TREE_OPERAND (base, 0) == op)
3183
            is_potential_deref = true;
3184
        }
3185
 
3186
      if (num_derefs > 0 || is_potential_deref)
3187
        {
3188
          /* Mark OP as dereferenced.  In a subsequent pass,
3189
             dereferenced pointers that point to a set of
3190
             variables will be assigned a name tag to alias
3191
             all the variables OP points to.  */
3192
          pi->is_dereferenced = 1;
3193
 
3194
          /* Keep track of how many time we've dereferenced each
3195
             pointer.  */
3196
          NUM_REFERENCES_INC (v_ann);
3197
 
3198
          /* If this is a store operation, mark OP as being
3199
             dereferenced to store, otherwise mark it as being
3200
             dereferenced to load.  */
3201
          if (is_store)
3202
            bitmap_set_bit (ai->dereferenced_ptrs_store, DECL_UID (var));
3203
          else
3204
            bitmap_set_bit (ai->dereferenced_ptrs_load, DECL_UID (var));
3205
        }
3206
 
3207
      if (stmt_escape_type != NO_ESCAPE && num_derefs < num_uses)
3208
        {
3209
          /* If STMT is an escape point and STMT contains at
3210
             least one direct use of OP, then the value of OP
3211
             escapes and so the pointed-to variables need to
3212
             be marked call-clobbered.  */
3213
          pi->value_escapes_p = 1;
3214
          pi->escape_mask |= stmt_escape_type;
3215
 
3216
          /* If the statement makes a function call, assume
3217
             that pointer OP will be dereferenced in a store
3218
             operation inside the called function.  */
3219
          if (get_call_expr_in (stmt)
3220
              || stmt_escape_type == ESCAPE_STORED_IN_GLOBAL)
3221
            {
3222
              bitmap_set_bit (ai->dereferenced_ptrs_store, DECL_UID (var));
3223
              pi->is_dereferenced = 1;
3224
            }
3225
        }
3226
    }
3227
 
3228
  if (TREE_CODE (stmt) == PHI_NODE)
3229
    return;
3230
 
3231
  /* Update reference counter for definitions to any
3232
     potentially aliased variable.  This is used in the alias
3233
     grouping heuristics.  */
3234
  FOR_EACH_SSA_TREE_OPERAND (op, stmt, iter, SSA_OP_DEF)
3235
    {
3236
      tree var = SSA_NAME_VAR (op);
3237
      var_ann_t ann = var_ann (var);
3238
      bitmap_set_bit (ai->written_vars, DECL_UID (var));
3239
      if (may_be_aliased (var))
3240
        NUM_REFERENCES_INC (ann);
3241
 
3242
    }
3243
 
3244
  /* Mark variables in V_MAY_DEF operands as being written to.  */
3245
  FOR_EACH_SSA_TREE_OPERAND (op, stmt, iter, SSA_OP_VIRTUAL_DEFS)
3246
    {
3247
      tree var = DECL_P (op) ? op : SSA_NAME_VAR (op);
3248
      bitmap_set_bit (ai->written_vars, DECL_UID (var));
3249
    }
3250
}
3251
 
3252
/* Handle pointer arithmetic EXPR when creating aliasing constraints.
3253
   Expressions of the type PTR + CST can be handled in two ways:
3254
 
3255
   1- If the constraint for PTR is ADDRESSOF for a non-structure
3256
      variable, then we can use it directly because adding or
3257
      subtracting a constant may not alter the original ADDRESSOF
3258
      constraint (i.e., pointer arithmetic may not legally go outside
3259
      an object's boundaries).
3260
 
3261
   2- If the constraint for PTR is ADDRESSOF for a structure variable,
3262
      then if CST is a compile-time constant that can be used as an
3263
      offset, we can determine which sub-variable will be pointed-to
3264
      by the expression.
3265
 
3266
   Return true if the expression is handled.  For any other kind of
3267
   expression, return false so that each operand can be added as a
3268
   separate constraint by the caller.  */
3269
 
3270
static bool
3271
handle_ptr_arith (VEC (ce_s, heap) *lhsc, tree expr)
3272
{
3273
  tree op0, op1;
3274
  struct constraint_expr *c, *c2;
3275
  unsigned int i = 0;
3276
  unsigned int j = 0;
3277
  VEC (ce_s, heap) *temp = NULL;
3278
  unsigned HOST_WIDE_INT rhsoffset = 0;
3279
 
3280
  if (TREE_CODE (expr) != PLUS_EXPR
3281
      && TREE_CODE (expr) != MINUS_EXPR)
3282
    return false;
3283
 
3284
  op0 = TREE_OPERAND (expr, 0);
3285
  op1 = TREE_OPERAND (expr, 1);
3286
 
3287
  get_constraint_for (op0, &temp);
3288
  if (POINTER_TYPE_P (TREE_TYPE (op0))
3289
      && host_integerp (op1, 1)
3290
      && TREE_CODE (expr) == PLUS_EXPR)
3291
    {
3292
      if ((TREE_INT_CST_LOW (op1) * BITS_PER_UNIT) / BITS_PER_UNIT
3293
          != TREE_INT_CST_LOW (op1))
3294
        return false;
3295
      rhsoffset = TREE_INT_CST_LOW (op1) * BITS_PER_UNIT;
3296
    }
3297
  else
3298
    return false;
3299
 
3300
 
3301
  for (i = 0; VEC_iterate (ce_s, lhsc, i, c); i++)
3302
    for (j = 0; VEC_iterate (ce_s, temp, j, c2); j++)
3303
      {
3304
        if (c2->type == ADDRESSOF && rhsoffset != 0)
3305
          {
3306
            varinfo_t temp = get_varinfo (c2->var);
3307
 
3308
            /* An access one after the end of an array is valid,
3309
               so simply punt on accesses we cannot resolve.  */
3310
            temp = first_vi_for_offset (temp, rhsoffset);
3311
            if (temp == NULL)
3312
              continue;
3313
            c2->var = temp->id;
3314
            c2->offset = 0;
3315
          }
3316
        else
3317
          c2->offset = rhsoffset;
3318
        process_constraint (new_constraint (*c, *c2));
3319
      }
3320
 
3321
  VEC_free (ce_s, heap, temp);
3322
 
3323
  return true;
3324
}
3325
 
3326
 
3327
/* Walk statement T setting up aliasing constraints according to the
3328
   references found in T.  This function is the main part of the
3329
   constraint builder.  AI points to auxiliary alias information used
3330
   when building alias sets and computing alias grouping heuristics.  */
3331
 
3332
static void
3333
find_func_aliases (tree origt)
3334
{
3335
  tree t = origt;
3336
  VEC(ce_s, heap) *lhsc = NULL;
3337
  VEC(ce_s, heap) *rhsc = NULL;
3338
  struct constraint_expr *c;
3339
 
3340
  if (TREE_CODE (t) == RETURN_EXPR && TREE_OPERAND (t, 0))
3341
    t = TREE_OPERAND (t, 0);
3342
 
3343
  /* Now build constraints expressions.  */
3344
  if (TREE_CODE (t) == PHI_NODE)
3345
    {
3346
      gcc_assert (!AGGREGATE_TYPE_P (TREE_TYPE (PHI_RESULT (t))));
3347
 
3348
      /* Only care about pointers and structures containing
3349
         pointers.  */
3350
      if (could_have_pointers (PHI_RESULT (t)))
3351
        {
3352
          int i;
3353
          unsigned int j;
3354
 
3355
          /* For a phi node, assign all the arguments to
3356
             the result.  */
3357
          get_constraint_for (PHI_RESULT (t), &lhsc);
3358
          for (i = 0; i < PHI_NUM_ARGS (t); i++)
3359
            {
3360
              tree rhstype;
3361
              tree strippedrhs = PHI_ARG_DEF (t, i);
3362
 
3363
              STRIP_NOPS (strippedrhs);
3364
              rhstype = TREE_TYPE (strippedrhs);
3365
              get_constraint_for (PHI_ARG_DEF (t, i), &rhsc);
3366
 
3367
              for (j = 0; VEC_iterate (ce_s, lhsc, j, c); j++)
3368
                {
3369
                  struct constraint_expr *c2;
3370
                  while (VEC_length (ce_s, rhsc) > 0)
3371
                    {
3372
                      c2 = VEC_last (ce_s, rhsc);
3373
                      process_constraint (new_constraint (*c, *c2));
3374
                      VEC_pop (ce_s, rhsc);
3375
                    }
3376
                }
3377
            }
3378
        }
3379
    }
3380
  /* In IPA mode, we need to generate constraints to pass call
3381
     arguments through their calls.   There are two case, either a
3382
     modify_expr when we are returning a value, or just a plain
3383
     call_expr when we are not.   */
3384
  else if (in_ipa_mode
3385
           && ((TREE_CODE (t) == MODIFY_EXPR
3386
                && TREE_CODE (TREE_OPERAND (t, 1)) == CALL_EXPR
3387
               && !(call_expr_flags (TREE_OPERAND (t, 1))
3388
                    & (ECF_MALLOC | ECF_MAY_BE_ALLOCA)))
3389
               || (TREE_CODE (t) == CALL_EXPR
3390
                   && !(call_expr_flags (t)
3391
                        & (ECF_MALLOC | ECF_MAY_BE_ALLOCA)))))
3392
    {
3393
      tree lhsop;
3394
      tree rhsop;
3395
      tree arglist;
3396
      varinfo_t fi;
3397
      int i = 1;
3398
      tree decl;
3399
      if (TREE_CODE (t) == MODIFY_EXPR)
3400
        {
3401
          lhsop = TREE_OPERAND (t, 0);
3402
          rhsop = TREE_OPERAND (t, 1);
3403
        }
3404
      else
3405
        {
3406
          lhsop = NULL;
3407
          rhsop = t;
3408
        }
3409
      decl = get_callee_fndecl (rhsop);
3410
 
3411
      /* If we can directly resolve the function being called, do so.
3412
         Otherwise, it must be some sort of indirect expression that
3413
         we should still be able to handle.  */
3414
      if (decl)
3415
        {
3416
          fi = get_vi_for_tree (decl);
3417
        }
3418
      else
3419
        {
3420
          decl = TREE_OPERAND (rhsop, 0);
3421
          fi = get_vi_for_tree (decl);
3422
        }
3423
 
3424
      /* Assign all the passed arguments to the appropriate incoming
3425
         parameters of the function.  */
3426
      arglist = TREE_OPERAND (rhsop, 1);
3427
 
3428
      for (;arglist; arglist = TREE_CHAIN (arglist))
3429
        {
3430
          tree arg = TREE_VALUE (arglist);
3431
          struct constraint_expr lhs ;
3432
          struct constraint_expr *rhsp;
3433
 
3434
          get_constraint_for (arg, &rhsc);
3435
          if (TREE_CODE (decl) != FUNCTION_DECL)
3436
            {
3437
              lhs.type = DEREF;
3438
              lhs.var = fi->id;
3439
              lhs.offset = i;
3440
            }
3441
          else
3442
            {
3443
              lhs.type = SCALAR;
3444
              lhs.var = first_vi_for_offset (fi, i)->id;
3445
              lhs.offset = 0;
3446
            }
3447
          while (VEC_length (ce_s, rhsc) != 0)
3448
            {
3449
              rhsp = VEC_last (ce_s, rhsc);
3450
              process_constraint (new_constraint (lhs, *rhsp));
3451
              VEC_pop (ce_s, rhsc);
3452
            }
3453
          i++;
3454
        }
3455
 
3456
      /* If we are returning a value, assign it to the result.  */
3457
      if (lhsop)
3458
        {
3459
          struct constraint_expr rhs;
3460
          struct constraint_expr *lhsp;
3461
          unsigned int j = 0;
3462
 
3463
          get_constraint_for (lhsop, &lhsc);
3464
          if (TREE_CODE (decl) != FUNCTION_DECL)
3465
            {
3466
              rhs.type = DEREF;
3467
              rhs.var = fi->id;
3468
              rhs.offset = i;
3469
            }
3470
          else
3471
            {
3472
              rhs.type = SCALAR;
3473
              rhs.var = first_vi_for_offset (fi, i)->id;
3474
              rhs.offset = 0;
3475
            }
3476
          for (j = 0; VEC_iterate (ce_s, lhsc, j, lhsp); j++)
3477
            process_constraint (new_constraint (*lhsp, rhs));
3478
        }
3479
    }
3480
  /* Otherwise, just a regular assignment statement.  */
3481
  else if (TREE_CODE (t) == MODIFY_EXPR)
3482
    {
3483
      tree lhsop = TREE_OPERAND (t, 0);
3484
      tree rhsop = TREE_OPERAND (t, 1);
3485
      int i;
3486
 
3487
      if ((AGGREGATE_TYPE_P (TREE_TYPE (lhsop))
3488
           || TREE_CODE (TREE_TYPE (lhsop)) == COMPLEX_TYPE)
3489
          && (AGGREGATE_TYPE_P (TREE_TYPE (rhsop))
3490
              || TREE_CODE (TREE_TYPE (lhsop)) == COMPLEX_TYPE))
3491
        {
3492
          do_structure_copy (lhsop, rhsop);
3493
        }
3494
      else
3495
        {
3496
          /* Only care about operations with pointers, structures
3497
             containing pointers, dereferences, and call expressions.  */
3498
          if (could_have_pointers (lhsop)
3499
              || TREE_CODE (rhsop) == CALL_EXPR)
3500
            {
3501
              get_constraint_for (lhsop, &lhsc);
3502
              switch (TREE_CODE_CLASS (TREE_CODE (rhsop)))
3503
                {
3504
                  /* RHS that consist of unary operations,
3505
                     exceptional types, or bare decls/constants, get
3506
                     handled directly by get_constraint_for.  */
3507
                  case tcc_reference:
3508
                  case tcc_declaration:
3509
                  case tcc_constant:
3510
                  case tcc_exceptional:
3511
                  case tcc_expression:
3512
                  case tcc_unary:
3513
                      {
3514
                        unsigned int j;
3515
 
3516
                        get_constraint_for (rhsop, &rhsc);
3517
                        for (j = 0; VEC_iterate (ce_s, lhsc, j, c); j++)
3518
                          {
3519
                            struct constraint_expr *c2;
3520
                            unsigned int k;
3521
 
3522
                            for (k = 0; VEC_iterate (ce_s, rhsc, k, c2); k++)
3523
                              process_constraint (new_constraint (*c, *c2));
3524
                          }
3525
 
3526
                      }
3527
                    break;
3528
 
3529
                  case tcc_binary:
3530
                      {
3531
                        /* For pointer arithmetic of the form
3532
                           PTR + CST, we can simply use PTR's
3533
                           constraint because pointer arithmetic is
3534
                           not allowed to go out of bounds.  */
3535
                        if (handle_ptr_arith (lhsc, rhsop))
3536
                          break;
3537
                      }
3538
                    /* FALLTHRU  */
3539
 
3540
                  /* Otherwise, walk each operand.  Notice that we
3541
                     can't use the operand interface because we need
3542
                     to process expressions other than simple operands
3543
                     (e.g. INDIRECT_REF, ADDR_EXPR, CALL_EXPR).  */
3544
                  default:
3545
                    for (i = 0; i < TREE_CODE_LENGTH (TREE_CODE (rhsop)); i++)
3546
                      {
3547
                        tree op = TREE_OPERAND (rhsop, i);
3548
                        unsigned int j;
3549
 
3550
                        gcc_assert (VEC_length (ce_s, rhsc) == 0);
3551
                        get_constraint_for (op, &rhsc);
3552
                        for (j = 0; VEC_iterate (ce_s, lhsc, j, c); j++)
3553
                          {
3554
                            struct constraint_expr *c2;
3555
                            while (VEC_length (ce_s, rhsc) > 0)
3556
                              {
3557
                                c2 = VEC_last (ce_s, rhsc);
3558
                                process_constraint (new_constraint (*c, *c2));
3559
                                VEC_pop (ce_s, rhsc);
3560
                              }
3561
                          }
3562
                      }
3563
                }
3564
            }
3565
        }
3566
    }
3567
 
3568
  /* After promoting variables and computing aliasing we will
3569
     need to re-scan most statements.  FIXME: Try to minimize the
3570
     number of statements re-scanned.  It's not really necessary to
3571
     re-scan *all* statements.  */
3572
  mark_stmt_modified (origt);
3573
  VEC_free (ce_s, heap, rhsc);
3574
  VEC_free (ce_s, heap, lhsc);
3575
}
3576
 
3577
 
3578
/* Find the first varinfo in the same variable as START that overlaps with
3579
   OFFSET.
3580
   Effectively, walk the chain of fields for the variable START to find the
3581
   first field that overlaps with OFFSET.
3582
   Return NULL if we can't find one.  */
3583
 
3584
static varinfo_t
3585
first_vi_for_offset (varinfo_t start, unsigned HOST_WIDE_INT offset)
3586
{
3587
  varinfo_t curr = start;
3588
  while (curr)
3589
    {
3590
      /* We may not find a variable in the field list with the actual
3591
         offset when when we have glommed a structure to a variable.
3592
         In that case, however, offset should still be within the size
3593
         of the variable. */
3594
      if (offset >= curr->offset && offset < (curr->offset +  curr->size))
3595
        return curr;
3596
      curr = curr->next;
3597
    }
3598
  return NULL;
3599
}
3600
 
3601
 
3602
/* Insert the varinfo FIELD into the field list for BASE, at the front
3603
   of the list.  */
3604
 
3605
static void
3606
insert_into_field_list (varinfo_t base, varinfo_t field)
3607
{
3608
  varinfo_t prev = base;
3609
  varinfo_t curr = base->next;
3610
 
3611
  field->next = curr;
3612
  prev->next = field;
3613
}
3614
 
3615
/* Insert the varinfo FIELD into the field list for BASE, ordered by
3616
   offset.  */
3617
 
3618
static void
3619
insert_into_field_list_sorted (varinfo_t base, varinfo_t field)
3620
{
3621
  varinfo_t prev = base;
3622
  varinfo_t curr = base->next;
3623
 
3624
  if (curr == NULL)
3625
    {
3626
      prev->next = field;
3627
      field->next = NULL;
3628
    }
3629
  else
3630
    {
3631
      while (curr)
3632
        {
3633
          if (field->offset <= curr->offset)
3634
            break;
3635
          prev = curr;
3636
          curr = curr->next;
3637
        }
3638
      field->next = prev->next;
3639
      prev->next = field;
3640
    }
3641
}
3642
 
3643
/* qsort comparison function for two fieldoff's PA and PB */
3644
 
3645
static int
3646
fieldoff_compare (const void *pa, const void *pb)
3647
{
3648
  const fieldoff_s *foa = (const fieldoff_s *)pa;
3649
  const fieldoff_s *fob = (const fieldoff_s *)pb;
3650
  HOST_WIDE_INT foasize, fobsize;
3651
 
3652
  if (foa->offset != fob->offset)
3653
    return foa->offset - fob->offset;
3654
 
3655
  foasize = TREE_INT_CST_LOW (foa->size);
3656
  fobsize = TREE_INT_CST_LOW (fob->size);
3657
  return foasize - fobsize;
3658
}
3659
 
3660
/* Sort a fieldstack according to the field offset and sizes.  */
3661
void
3662
sort_fieldstack (VEC(fieldoff_s,heap) *fieldstack)
3663
{
3664
  qsort (VEC_address (fieldoff_s, fieldstack),
3665
         VEC_length (fieldoff_s, fieldstack),
3666
         sizeof (fieldoff_s),
3667
         fieldoff_compare);
3668
}
3669
 
3670
/* Given a TYPE, and a vector of field offsets FIELDSTACK, push all the fields
3671
   of TYPE onto fieldstack, recording their offsets along the way.
3672
   OFFSET is used to keep track of the offset in this entire structure, rather
3673
   than just the immediately containing structure.  Returns the number
3674
   of fields pushed.
3675
   HAS_UNION is set to true if we find a union type as a field of
3676
   TYPE.  */
3677
 
3678
int
3679
push_fields_onto_fieldstack (tree type, VEC(fieldoff_s,heap) **fieldstack,
3680
                             HOST_WIDE_INT offset, bool *has_union)
3681
{
3682
  tree field;
3683
  int count = 0;
3684
  unsigned HOST_WIDE_INT minoffset = -1;
3685
 
3686
  if (TREE_CODE (type) == COMPLEX_TYPE)
3687
    {
3688
      fieldoff_s *real_part, *img_part;
3689
      real_part = VEC_safe_push (fieldoff_s, heap, *fieldstack, NULL);
3690
      real_part->type = TREE_TYPE (type);
3691
      real_part->size = TYPE_SIZE (TREE_TYPE (type));
3692
      real_part->offset = offset;
3693
      real_part->decl = NULL_TREE;
3694
 
3695
      img_part = VEC_safe_push (fieldoff_s, heap, *fieldstack, NULL);
3696
      img_part->type = TREE_TYPE (type);
3697
      img_part->size = TYPE_SIZE (TREE_TYPE (type));
3698
      img_part->offset = offset + TREE_INT_CST_LOW (TYPE_SIZE (TREE_TYPE (type)));
3699
      img_part->decl = NULL_TREE;
3700
 
3701
      return 2;
3702
    }
3703
 
3704
  if (TREE_CODE (type) == ARRAY_TYPE)
3705
    {
3706
      tree sz = TYPE_SIZE (type);
3707
      tree elsz = TYPE_SIZE (TREE_TYPE (type));
3708
      HOST_WIDE_INT nr;
3709
      int i;
3710
 
3711
      if (! sz
3712
          || ! host_integerp (sz, 1)
3713
          || TREE_INT_CST_LOW (sz) == 0
3714
          || ! elsz
3715
          || ! host_integerp (elsz, 1)
3716
          || TREE_INT_CST_LOW (elsz) == 0)
3717
        return 0;
3718
 
3719
      nr = TREE_INT_CST_LOW (sz) / TREE_INT_CST_LOW (elsz);
3720
      if (nr > SALIAS_MAX_ARRAY_ELEMENTS)
3721
        return 0;
3722
 
3723
      for (i = 0; i < nr; ++i)
3724
        {
3725
          bool push = false;
3726
          int pushed = 0;
3727
 
3728
          if (has_union
3729
              && (TREE_CODE (TREE_TYPE (type)) == QUAL_UNION_TYPE
3730
                  || TREE_CODE (TREE_TYPE (type)) == UNION_TYPE))
3731
            *has_union = true;
3732
 
3733
          if (!AGGREGATE_TYPE_P (TREE_TYPE (type))) /* var_can_have_subvars */
3734
            push = true;
3735
          else if (!(pushed = push_fields_onto_fieldstack
3736
                     (TREE_TYPE (type), fieldstack,
3737
                      offset + i * TREE_INT_CST_LOW (elsz), has_union)))
3738
            /* Empty structures may have actual size, like in C++. So
3739
               see if we didn't push any subfields and the size is
3740
               nonzero, push the field onto the stack */
3741
            push = true;
3742
 
3743
          if (push)
3744
            {
3745
              fieldoff_s *pair;
3746
 
3747
              pair = VEC_safe_push (fieldoff_s, heap, *fieldstack, NULL);
3748
              pair->type = TREE_TYPE (type);
3749
              pair->size = elsz;
3750
              pair->decl = NULL_TREE;
3751
              pair->offset = offset + i * TREE_INT_CST_LOW (elsz);
3752
              count++;
3753
            }
3754
          else
3755
            count += pushed;
3756
        }
3757
 
3758
      return count;
3759
    }
3760
 
3761
  for (field = TYPE_FIELDS (type); field; field = TREE_CHAIN (field))
3762
    if (TREE_CODE (field) == FIELD_DECL)
3763
      {
3764
        bool push = false;
3765
        int pushed = 0;
3766
 
3767
        if (has_union
3768
            && (TREE_CODE (TREE_TYPE (field)) == QUAL_UNION_TYPE
3769
                || TREE_CODE (TREE_TYPE (field)) == UNION_TYPE))
3770
          *has_union = true;
3771
 
3772
        if (!var_can_have_subvars (field))
3773
          push = true;
3774
        else if (!(pushed = push_fields_onto_fieldstack
3775
                   (TREE_TYPE (field), fieldstack,
3776
                    offset + bitpos_of_field (field), has_union))
3777
                 && DECL_SIZE (field)
3778
                 && !integer_zerop (DECL_SIZE (field)))
3779
          /* Empty structures may have actual size, like in C++. So
3780
             see if we didn't push any subfields and the size is
3781
             nonzero, push the field onto the stack */
3782
          push = true;
3783
 
3784
        if (push)
3785
          {
3786
            fieldoff_s *pair;
3787
 
3788
            pair = VEC_safe_push (fieldoff_s, heap, *fieldstack, NULL);
3789
            pair->type = TREE_TYPE (field);
3790
            pair->size = DECL_SIZE (field);
3791
            pair->decl = field;
3792
            pair->offset = offset + bitpos_of_field (field);
3793
            count++;
3794
          }
3795
        else
3796
          count += pushed;
3797
 
3798
        if (bitpos_of_field (field) < minoffset)
3799
          minoffset = bitpos_of_field (field);
3800
      }
3801
 
3802
  /* We need to create a fake subvar for empty bases.  But _only_ for non-empty
3803
     classes.  */
3804
  if (minoffset != 0 && count != 0)
3805
    {
3806
      fieldoff_s *pair;
3807
 
3808
      pair = VEC_safe_push (fieldoff_s, heap, *fieldstack, NULL);
3809
      pair->type = void_type_node;
3810
      pair->size = build_int_cst (size_type_node, minoffset);
3811
      pair->decl = NULL;
3812
      pair->offset = offset;
3813
      count++;
3814
    }
3815
 
3816
  return count;
3817
}
3818
 
3819
/* Create a constraint from ESCAPED_VARS variable to VI.  */
3820
static void
3821
make_constraint_from_escaped (varinfo_t vi)
3822
{
3823
  struct constraint_expr lhs, rhs;
3824
 
3825
  lhs.var = vi->id;
3826
  lhs.offset = 0;
3827
  lhs.type = SCALAR;
3828
 
3829
  rhs.var = escaped_vars_id;
3830
  rhs.offset = 0;
3831
  rhs.type = SCALAR;
3832
  process_constraint (new_constraint (lhs, rhs));
3833
}
3834
 
3835
/* Create a constraint to the ESCAPED_VARS variable from constraint
3836
   expression RHS. */
3837
 
3838
static void
3839
make_constraint_to_escaped (struct constraint_expr rhs)
3840
{
3841
  struct constraint_expr lhs;
3842
 
3843
  lhs.var = escaped_vars_id;
3844
  lhs.offset = 0;
3845
  lhs.type = SCALAR;
3846
 
3847
  process_constraint (new_constraint (lhs, rhs));
3848
}
3849
 
3850
/* Count the number of arguments DECL has, and set IS_VARARGS to true
3851
   if it is a varargs function.  */
3852
 
3853
static unsigned int
3854
count_num_arguments (tree decl, bool *is_varargs)
3855
{
3856
  unsigned int i = 0;
3857
  tree t;
3858
 
3859
  for (t = TYPE_ARG_TYPES (TREE_TYPE (decl));
3860
       t;
3861
       t = TREE_CHAIN (t))
3862
    {
3863
      if (TREE_VALUE (t) == void_type_node)
3864
        break;
3865
      i++;
3866
    }
3867
 
3868
  if (!t)
3869
    *is_varargs = true;
3870
  return i;
3871
}
3872
 
3873
/* Creation function node for DECL, using NAME, and return the index
3874
   of the variable we've created for the function.  */
3875
 
3876
static unsigned int
3877
create_function_info_for (tree decl, const char *name)
3878
{
3879
  unsigned int index = VEC_length (varinfo_t, varmap);
3880
  varinfo_t vi;
3881
  tree arg;
3882
  unsigned int i;
3883
  bool is_varargs = false;
3884
 
3885
  /* Create the variable info.  */
3886
 
3887
  vi = new_var_info (decl, index, name);
3888
  vi->decl = decl;
3889
  vi->offset = 0;
3890
  vi->has_union = 0;
3891
  vi->size = 1;
3892
  vi->fullsize = count_num_arguments (decl, &is_varargs) + 1;
3893
  insert_vi_for_tree (vi->decl, vi);
3894
  VEC_safe_push (varinfo_t, heap, varmap, vi);
3895
 
3896
  stats.total_vars++;
3897
 
3898
  /* If it's varargs, we don't know how many arguments it has, so we
3899
     can't do much.
3900
  */
3901
  if (is_varargs)
3902
    {
3903
      vi->fullsize = ~0;
3904
      vi->size = ~0;
3905
      vi->is_unknown_size_var = true;
3906
      return index;
3907
    }
3908
 
3909
 
3910
  arg = DECL_ARGUMENTS (decl);
3911
 
3912
  /* Set up variables for each argument.  */
3913
  for (i = 1; i < vi->fullsize; i++)
3914
    {
3915
      varinfo_t argvi;
3916
      const char *newname;
3917
      char *tempname;
3918
      unsigned int newindex;
3919
      tree argdecl = decl;
3920
 
3921
      if (arg)
3922
        argdecl = arg;
3923
 
3924
      newindex = VEC_length (varinfo_t, varmap);
3925
      asprintf (&tempname, "%s.arg%d", name, i-1);
3926
      newname = ggc_strdup (tempname);
3927
      free (tempname);
3928
 
3929
      argvi = new_var_info (argdecl, newindex, newname);
3930
      argvi->decl = argdecl;
3931
      VEC_safe_push (varinfo_t, heap, varmap, argvi);
3932
      argvi->offset = i;
3933
      argvi->size = 1;
3934
      argvi->fullsize = vi->fullsize;
3935
      argvi->has_union = false;
3936
      insert_into_field_list_sorted (vi, argvi);
3937
      stats.total_vars ++;
3938
      if (arg)
3939
        {
3940
          insert_vi_for_tree (arg, argvi);
3941
          arg = TREE_CHAIN (arg);
3942
        }
3943
    }
3944
 
3945
  /* Create a variable for the return var.  */
3946
  if (DECL_RESULT (decl) != NULL
3947
      || !VOID_TYPE_P (TREE_TYPE (TREE_TYPE (decl))))
3948
    {
3949
      varinfo_t resultvi;
3950
      const char *newname;
3951
      char *tempname;
3952
      unsigned int newindex;
3953
      tree resultdecl = decl;
3954
 
3955
      vi->fullsize ++;
3956
 
3957
      if (DECL_RESULT (decl))
3958
        resultdecl = DECL_RESULT (decl);
3959
 
3960
      newindex = VEC_length (varinfo_t, varmap);
3961
      asprintf (&tempname, "%s.result", name);
3962
      newname = ggc_strdup (tempname);
3963
      free (tempname);
3964
 
3965
      resultvi = new_var_info (resultdecl, newindex, newname);
3966
      resultvi->decl = resultdecl;
3967
      VEC_safe_push (varinfo_t, heap, varmap, resultvi);
3968
      resultvi->offset = i;
3969
      resultvi->size = 1;
3970
      resultvi->fullsize = vi->fullsize;
3971
      resultvi->has_union = false;
3972
      insert_into_field_list_sorted (vi, resultvi);
3973
      stats.total_vars ++;
3974
      if (DECL_RESULT (decl))
3975
        insert_vi_for_tree (DECL_RESULT (decl), resultvi);
3976
    }
3977
  return index;
3978
}
3979
 
3980
 
3981
/* Return true if FIELDSTACK contains fields that overlap.
3982
   FIELDSTACK is assumed to be sorted by offset.  */
3983
 
3984
static bool
3985
check_for_overlaps (VEC (fieldoff_s,heap) *fieldstack)
3986
{
3987
  fieldoff_s *fo = NULL;
3988
  unsigned int i;
3989
  HOST_WIDE_INT lastoffset = -1;
3990
 
3991
  for (i = 0; VEC_iterate (fieldoff_s, fieldstack, i, fo); i++)
3992
    {
3993
      if (fo->offset == lastoffset)
3994
        return true;
3995
      lastoffset = fo->offset;
3996
    }
3997
  return false;
3998
}
3999
 
4000
/* This function is called through walk_tree to walk global
4001
   initializers looking for constraints we need to add to the
4002
   constraint list.  */
4003
 
4004
static tree
4005
find_global_initializers (tree *tp, int *walk_subtrees ATTRIBUTE_UNUSED,
4006
                          void *viv)
4007
{
4008
  varinfo_t vi = (varinfo_t)viv;
4009
  tree t = *tp;
4010
 
4011
  switch (TREE_CODE (t))
4012
    {
4013
      /* Dereferences and addressofs are the only important things
4014
         here, and i don't even remember if dereferences are legal
4015
         here in initializers.  */
4016
    case INDIRECT_REF:
4017
    case ADDR_EXPR:
4018
      {
4019
        struct constraint_expr *c;
4020
        size_t i;
4021
 
4022
        VEC(ce_s, heap) *rhsc = NULL;
4023
        get_constraint_for (t, &rhsc);
4024
        for (i = 0; VEC_iterate (ce_s, rhsc, i, c); i++)
4025
          {
4026
            struct constraint_expr lhs;
4027
 
4028
            lhs.var = vi->id;
4029
            lhs.type = SCALAR;
4030
            lhs.offset = 0;
4031
            process_constraint (new_constraint (lhs, *c));
4032
          }
4033
 
4034
        VEC_free (ce_s, heap, rhsc);
4035
      }
4036
      break;
4037
    case VAR_DECL:
4038
      /* We might not have walked this because we skip
4039
         DECL_EXTERNALs during the initial scan.  */
4040
      if (referenced_vars)
4041
        {
4042
          get_var_ann (t);
4043
          if (referenced_var_check_and_insert (t))
4044
            mark_sym_for_renaming (t);
4045
        }
4046
      break;
4047
    default:
4048
      break;
4049
    }
4050
  return NULL_TREE;
4051
}
4052
 
4053
/* Create a varinfo structure for NAME and DECL, and add it to VARMAP.
4054
   This will also create any varinfo structures necessary for fields
4055
   of DECL.  */
4056
 
4057
static unsigned int
4058
create_variable_info_for (tree decl, const char *name)
4059
{
4060
  unsigned int index = VEC_length (varinfo_t, varmap);
4061
  varinfo_t vi;
4062
  tree decltype = TREE_TYPE (decl);
4063
  tree declsize = DECL_P (decl) ? DECL_SIZE (decl) : TYPE_SIZE (decltype);
4064
  bool notokay = false;
4065
  bool hasunion;
4066
  bool is_global = DECL_P (decl) ? is_global_var (decl) : false;
4067
  VEC (fieldoff_s,heap) *fieldstack = NULL;
4068
 
4069
  if (TREE_CODE (decl) == FUNCTION_DECL && in_ipa_mode)
4070
    return create_function_info_for (decl, name);
4071
 
4072
  hasunion = TREE_CODE (decltype) == UNION_TYPE
4073
             || TREE_CODE (decltype) == QUAL_UNION_TYPE;
4074
  if (var_can_have_subvars (decl) && use_field_sensitive && !hasunion)
4075
    {
4076
      push_fields_onto_fieldstack (decltype, &fieldstack, 0, &hasunion);
4077
      if (hasunion)
4078
        {
4079
          VEC_free (fieldoff_s, heap, fieldstack);
4080
          notokay = true;
4081
        }
4082
    }
4083
 
4084
 
4085
  /* If the variable doesn't have subvars, we may end up needing to
4086
     sort the field list and create fake variables for all the
4087
     fields.  */
4088
  vi = new_var_info (decl, index, name);
4089
  vi->decl = decl;
4090
  vi->offset = 0;
4091
  vi->has_union = hasunion;
4092
  if (!declsize
4093
      || TREE_CODE (declsize) != INTEGER_CST
4094
      || TREE_CODE (decltype) == UNION_TYPE
4095
      || TREE_CODE (decltype) == QUAL_UNION_TYPE)
4096
    {
4097
      vi->is_unknown_size_var = true;
4098
      vi->fullsize = ~0;
4099
      vi->size = ~0;
4100
    }
4101
  else
4102
    {
4103
      vi->fullsize = TREE_INT_CST_LOW (declsize);
4104
      vi->size = vi->fullsize;
4105
    }
4106
 
4107
  insert_vi_for_tree (vi->decl, vi);
4108
  VEC_safe_push (varinfo_t, heap, varmap, vi);
4109
  if (is_global && (!flag_whole_program || !in_ipa_mode))
4110
    {
4111
      make_constraint_from_escaped (vi);
4112
 
4113
      /* If the variable can't be aliased, there is no point in
4114
         putting it in the set of nonlocal vars.  */
4115
      if (may_be_aliased (vi->decl))
4116
        {
4117
          struct constraint_expr rhs;
4118
          rhs.var = index;
4119
          rhs.type = ADDRESSOF;
4120
          rhs.offset = 0;
4121
          make_constraint_to_escaped (rhs);
4122
        }
4123
 
4124
      if (TREE_CODE (decl) != FUNCTION_DECL && DECL_INITIAL (decl))
4125
        {
4126
          walk_tree_without_duplicates (&DECL_INITIAL (decl),
4127
                                        find_global_initializers,
4128
                                        (void *)vi);
4129
        }
4130
    }
4131
 
4132
  stats.total_vars++;
4133
  if (use_field_sensitive
4134
      && !notokay
4135
      && !vi->is_unknown_size_var
4136
      && var_can_have_subvars (decl)
4137
      && VEC_length (fieldoff_s, fieldstack) <= MAX_FIELDS_FOR_FIELD_SENSITIVE)
4138
    {
4139
      unsigned int newindex = VEC_length (varinfo_t, varmap);
4140
      fieldoff_s *fo = NULL;
4141
      unsigned int i;
4142
 
4143
      for (i = 0; !notokay && VEC_iterate (fieldoff_s, fieldstack, i, fo); i++)
4144
        {
4145
          if (! fo->size
4146
              || TREE_CODE (fo->size) != INTEGER_CST
4147
              || fo->offset < 0)
4148
            {
4149
              notokay = true;
4150
              break;
4151
            }
4152
        }
4153
 
4154
      /* We can't sort them if we have a field with a variable sized type,
4155
         which will make notokay = true.  In that case, we are going to return
4156
         without creating varinfos for the fields anyway, so sorting them is a
4157
         waste to boot.  */
4158
      if (!notokay)
4159
        {
4160
          sort_fieldstack (fieldstack);
4161
          /* Due to some C++ FE issues, like PR 22488, we might end up
4162
             what appear to be overlapping fields even though they,
4163
             in reality, do not overlap.  Until the C++ FE is fixed,
4164
             we will simply disable field-sensitivity for these cases.  */
4165
          notokay = check_for_overlaps (fieldstack);
4166
        }
4167
 
4168
 
4169
      if (VEC_length (fieldoff_s, fieldstack) != 0)
4170
        fo = VEC_index (fieldoff_s, fieldstack, 0);
4171
 
4172
      if (fo == NULL || notokay)
4173
        {
4174
          vi->is_unknown_size_var = 1;
4175
          vi->fullsize = ~0;
4176
          vi->size = ~0;
4177
          VEC_free (fieldoff_s, heap, fieldstack);
4178
          return index;
4179
        }
4180
 
4181
      vi->size = TREE_INT_CST_LOW (fo->size);
4182
      vi->offset = fo->offset;
4183
      for (i = VEC_length (fieldoff_s, fieldstack) - 1;
4184
           i >= 1 && VEC_iterate (fieldoff_s, fieldstack, i, fo);
4185
           i--)
4186
        {
4187
          varinfo_t newvi;
4188
          const char *newname = "NULL";
4189
          char *tempname;
4190
 
4191
          newindex = VEC_length (varinfo_t, varmap);
4192
          if (dump_file)
4193
            {
4194
              if (fo->decl)
4195
                asprintf (&tempname, "%s.%s",
4196
                          vi->name, alias_get_name (fo->decl));
4197
              else
4198
                asprintf (&tempname, "%s." HOST_WIDE_INT_PRINT_DEC,
4199
                          vi->name, fo->offset);
4200
              newname = ggc_strdup (tempname);
4201
              free (tempname);
4202
            }
4203
          newvi = new_var_info (decl, newindex, newname);
4204
          newvi->offset = fo->offset;
4205
          newvi->size = TREE_INT_CST_LOW (fo->size);
4206
          newvi->fullsize = vi->fullsize;
4207
          insert_into_field_list (vi, newvi);
4208
          VEC_safe_push (varinfo_t, heap, varmap, newvi);
4209
          if (is_global && (!flag_whole_program || !in_ipa_mode))
4210
            {
4211
              /* If the variable can't be aliased, there is no point in
4212
                 putting it in the set of nonlocal vars.  */
4213
              if (may_be_aliased (vi->decl))
4214
                {
4215
                  struct constraint_expr rhs;
4216
 
4217
                  rhs.var = newindex;
4218
                  rhs.type = ADDRESSOF;
4219
                  rhs.offset = 0;
4220
                  make_constraint_to_escaped (rhs);
4221
                }
4222
              make_constraint_from_escaped (newvi);
4223
            }
4224
 
4225
          stats.total_vars++;
4226
        }
4227
      VEC_free (fieldoff_s, heap, fieldstack);
4228
    }
4229
  return index;
4230
}
4231
 
4232
/* Print out the points-to solution for VAR to FILE.  */
4233
 
4234
void
4235
dump_solution_for_var (FILE *file, unsigned int var)
4236
{
4237
  varinfo_t vi = get_varinfo (var);
4238
  unsigned int i;
4239
  bitmap_iterator bi;
4240
 
4241
  if (find (var) != var)
4242
    {
4243
      varinfo_t vipt = get_varinfo (find (var));
4244
      fprintf (file, "%s = same as %s\n", vi->name, vipt->name);
4245
    }
4246
  else
4247
    {
4248
      fprintf (file, "%s = { ", vi->name);
4249
      EXECUTE_IF_SET_IN_BITMAP (vi->solution, 0, i, bi)
4250
        {
4251
          fprintf (file, "%s ", get_varinfo (i)->name);
4252
        }
4253
      fprintf (file, "}\n");
4254
    }
4255
}
4256
 
4257
/* Print the points-to solution for VAR to stdout.  */
4258
 
4259
void
4260
debug_solution_for_var (unsigned int var)
4261
{
4262
  dump_solution_for_var (stdout, var);
4263
}
4264
 
4265
/* Create varinfo structures for all of the variables in the
4266
   function for intraprocedural mode.  */
4267
 
4268
static void
4269
intra_create_variable_infos (void)
4270
{
4271
  tree t;
4272
  struct constraint_expr lhs, rhs;
4273
  varinfo_t nonlocal_vi;
4274
 
4275
  /* For each incoming pointer argument arg, ARG = ESCAPED_VARS or a
4276
     dummy variable if flag_argument_noalias > 2. */
4277
  for (t = DECL_ARGUMENTS (current_function_decl); t; t = TREE_CHAIN (t))
4278
    {
4279
      varinfo_t p;
4280
      unsigned int arg_id;
4281
 
4282
      if (!could_have_pointers (t))
4283
        continue;
4284
 
4285
      arg_id = get_vi_for_tree (t)->id;
4286
 
4287
      /* With flag_argument_noalias greater than two means that the incoming
4288
         argument cannot alias anything except for itself so create a HEAP
4289
         variable.  */
4290
      if (POINTER_TYPE_P (TREE_TYPE (t))
4291
          && flag_argument_noalias > 2)
4292
        {
4293
          varinfo_t vi;
4294
          tree heapvar = heapvar_lookup (t);
4295
 
4296
          lhs.offset = 0;
4297
          lhs.type = SCALAR;
4298
          lhs.var  = get_vi_for_tree (t)->id;
4299
 
4300
          if (heapvar == NULL_TREE)
4301
            {
4302
              heapvar = create_tmp_var_raw (TREE_TYPE (TREE_TYPE (t)),
4303
                                            "PARM_NOALIAS");
4304
              DECL_EXTERNAL (heapvar) = 1;
4305
              if (referenced_vars)
4306
                add_referenced_var (heapvar);
4307
              heapvar_insert (t, heapvar);
4308
            }
4309
 
4310
          vi = get_vi_for_tree (heapvar);
4311
          vi->is_artificial_var = 1;
4312
          vi->is_heap_var = 1;
4313
          rhs.var = vi->id;
4314
          rhs.type = ADDRESSOF;
4315
          rhs.offset = 0;
4316
          for (p = get_varinfo (lhs.var); p; p = p->next)
4317
            {
4318
              struct constraint_expr temp = lhs;
4319
              temp.var = p->id;
4320
              process_constraint (new_constraint (temp, rhs));
4321
            }
4322
        }
4323
      else
4324
        {
4325
          for (p = get_varinfo (arg_id); p; p = p->next)
4326
            make_constraint_from_escaped (p);
4327
        }
4328
    }
4329
  if (!nonlocal_all)
4330
    nonlocal_all = create_nonlocal_var (void_type_node);
4331
 
4332
  /* Create variable info for the nonlocal var if it does not
4333
     exist.  */
4334
  nonlocal_vars_id = create_variable_info_for (nonlocal_all,
4335
                                               get_name (nonlocal_all));
4336
  nonlocal_vi = get_varinfo (nonlocal_vars_id);
4337
  nonlocal_vi->is_artificial_var = 1;
4338
  nonlocal_vi->is_heap_var = 1;
4339
  nonlocal_vi->is_unknown_size_var = 1;
4340
  nonlocal_vi->directly_dereferenced = true;
4341
 
4342
  rhs.var = nonlocal_vars_id;
4343
  rhs.type = ADDRESSOF;
4344
  rhs.offset = 0;
4345
 
4346
  lhs.var = escaped_vars_id;
4347
  lhs.type = SCALAR;
4348
  lhs.offset = 0;
4349
 
4350
  process_constraint (new_constraint (lhs, rhs));
4351
}
4352
 
4353
/* Structure used to put solution bitmaps in a hashtable so they can
4354
   be shared among variables with the same points-to set.  */
4355
 
4356
typedef struct shared_bitmap_info
4357
{
4358
  bitmap pt_vars;
4359
  hashval_t hashcode;
4360
} *shared_bitmap_info_t;
4361
 
4362
static htab_t shared_bitmap_table;
4363
 
4364
/* Hash function for a shared_bitmap_info_t */
4365
 
4366
static hashval_t
4367
shared_bitmap_hash (const void *p)
4368
{
4369
  const shared_bitmap_info_t bi = (shared_bitmap_info_t) p;
4370
  return bi->hashcode;
4371
}
4372
 
4373
/* Equality function for two shared_bitmap_info_t's. */
4374
 
4375
static int
4376
shared_bitmap_eq (const void *p1, const void *p2)
4377
{
4378
  const shared_bitmap_info_t sbi1 = (shared_bitmap_info_t) p1;
4379
  const shared_bitmap_info_t sbi2 = (shared_bitmap_info_t) p2;
4380
  return bitmap_equal_p (sbi1->pt_vars, sbi2->pt_vars);
4381
}
4382
 
4383
/* Lookup a bitmap in the shared bitmap hashtable, and return an already
4384
   existing instance if there is one, NULL otherwise.  */
4385
 
4386
static bitmap
4387
shared_bitmap_lookup (bitmap pt_vars)
4388
{
4389
  void **slot;
4390
  struct shared_bitmap_info sbi;
4391
 
4392
  sbi.pt_vars = pt_vars;
4393
  sbi.hashcode = bitmap_hash (pt_vars);
4394
 
4395
  slot = htab_find_slot_with_hash (shared_bitmap_table, &sbi,
4396
                                   sbi.hashcode, NO_INSERT);
4397
  if (!slot)
4398
    return NULL;
4399
  else
4400
    return ((shared_bitmap_info_t) *slot)->pt_vars;
4401
}
4402
 
4403
 
4404
/* Add a bitmap to the shared bitmap hashtable.  */
4405
 
4406
static void
4407
shared_bitmap_add (bitmap pt_vars)
4408
{
4409
  void **slot;
4410
  shared_bitmap_info_t sbi = XNEW (struct shared_bitmap_info);
4411
 
4412
  sbi->pt_vars = pt_vars;
4413
  sbi->hashcode = bitmap_hash (pt_vars);
4414
 
4415
  slot = htab_find_slot_with_hash (shared_bitmap_table, sbi,
4416
                                   sbi->hashcode, INSERT);
4417
  gcc_assert (!*slot);
4418
  *slot = (void *) sbi;
4419
}
4420
 
4421
 
4422
/* Set bits in INTO corresponding to the variable uids in solution set
4423
   FROM, which came from variable PTR.
4424
   For variables that are actually dereferenced, we also use type
4425
   based alias analysis to prune the points-to sets.  */
4426
 
4427
static void
4428
set_uids_in_ptset (tree ptr, bitmap into, bitmap from)
4429
{
4430
  unsigned int i;
4431
  bitmap_iterator bi;
4432
  subvar_t sv;
4433
  unsigned HOST_WIDE_INT ptr_alias_set = get_alias_set (TREE_TYPE (ptr));
4434
 
4435
  EXECUTE_IF_SET_IN_BITMAP (from, 0, i, bi)
4436
    {
4437
      varinfo_t vi = get_varinfo (i);
4438
      unsigned HOST_WIDE_INT var_alias_set;
4439
 
4440
      /* The only artificial variables that are allowed in a may-alias
4441
         set are heap variables.  */
4442
      if (vi->is_artificial_var && !vi->is_heap_var)
4443
        continue;
4444
 
4445
      if (vi->has_union && get_subvars_for_var (vi->decl) != NULL)
4446
        {
4447
          /* Variables containing unions may need to be converted to
4448
             their SFT's, because SFT's can have unions and we cannot.  */
4449
          for (sv = get_subvars_for_var (vi->decl); sv; sv = sv->next)
4450
            bitmap_set_bit (into, DECL_UID (sv->var));
4451
        }
4452
      else if (TREE_CODE (vi->decl) == VAR_DECL
4453
               || TREE_CODE (vi->decl) == PARM_DECL
4454
               || TREE_CODE (vi->decl) == RESULT_DECL)
4455
        {
4456
          if (var_can_have_subvars (vi->decl)
4457
              && get_subvars_for_var (vi->decl))
4458
            {
4459
              /* If VI->DECL is an aggregate for which we created
4460
                 SFTs, add the SFT corresponding to VI->OFFSET.  */
4461
              tree sft = get_subvar_at (vi->decl, vi->offset);
4462
              if (sft)
4463
                {
4464
                  var_alias_set = get_alias_set (sft);
4465
                  if (!vi->directly_dereferenced
4466
                      || alias_sets_conflict_p (ptr_alias_set, var_alias_set))
4467
                    bitmap_set_bit (into, DECL_UID (sft));
4468
                }
4469
            }
4470
          else
4471
            {
4472
              /* Otherwise, just add VI->DECL to the alias set.
4473
                 Don't type prune artificial vars.  */
4474
              if (vi->is_artificial_var)
4475
                bitmap_set_bit (into, DECL_UID (vi->decl));
4476
              else
4477
                {
4478
                  var_alias_set = get_alias_set (vi->decl);
4479
                  if (!vi->directly_dereferenced
4480
                      || alias_sets_conflict_p (ptr_alias_set, var_alias_set))
4481
                    bitmap_set_bit (into, DECL_UID (vi->decl));
4482
                }
4483
            }
4484
        }
4485
    }
4486
}
4487
 
4488
 
4489
static bool have_alias_info = false;
4490
 
4491
/* Given a pointer variable P, fill in its points-to set, or return
4492
   false if we can't.  */
4493
 
4494
bool
4495
find_what_p_points_to (tree p)
4496
{
4497
  tree lookup_p = p;
4498
  varinfo_t vi;
4499
 
4500
  if (!have_alias_info)
4501
    return false;
4502
 
4503
  /* For parameters, get at the points-to set for the actual parm
4504
     decl.  */
4505
  if (TREE_CODE (p) == SSA_NAME
4506
      && TREE_CODE (SSA_NAME_VAR (p)) == PARM_DECL
4507
      && default_def (SSA_NAME_VAR (p)) == p)
4508
    lookup_p = SSA_NAME_VAR (p);
4509
 
4510
  vi = lookup_vi_for_tree (lookup_p);
4511
  if (vi)
4512
    {
4513
 
4514
      if (vi->is_artificial_var)
4515
        return false;
4516
 
4517
      /* See if this is a field or a structure.  */
4518
      if (vi->size != vi->fullsize)
4519
        {
4520
          /* Nothing currently asks about structure fields directly,
4521
             but when they do, we need code here to hand back the
4522
             points-to set.  */
4523
          if (!var_can_have_subvars (vi->decl)
4524
              || get_subvars_for_var (vi->decl) == NULL)
4525
            return false;
4526
        }
4527
      else
4528
        {
4529
          struct ptr_info_def *pi = get_ptr_info (p);
4530
          unsigned int i;
4531
          bitmap_iterator bi;
4532
          bitmap finished_solution;
4533
          bitmap result;
4534
 
4535
          /* This variable may have been collapsed, let's get the real
4536
             variable.  */
4537
          vi = get_varinfo (find (vi->id));
4538
 
4539
          /* Translate artificial variables into SSA_NAME_PTR_INFO
4540
             attributes.  */
4541
          EXECUTE_IF_SET_IN_BITMAP (vi->solution, 0, i, bi)
4542
            {
4543
              varinfo_t vi = get_varinfo (i);
4544
 
4545
              if (vi->is_artificial_var)
4546
                {
4547
                  /* FIXME.  READONLY should be handled better so that
4548
                     flow insensitive aliasing can disregard writable
4549
                     aliases.  */
4550
                  if (vi->id == nothing_id)
4551
                    pi->pt_null = 1;
4552
                  else if (vi->id == anything_id)
4553
                    pi->pt_anything = 1;
4554
                  else if (vi->id == readonly_id)
4555
                    pi->pt_anything = 1;
4556
                  else if (vi->id == integer_id)
4557
                    pi->pt_anything = 1;
4558
                  else if (vi->is_heap_var)
4559
                    pi->pt_global_mem = 1;
4560
                }
4561
            }
4562
 
4563
          if (pi->pt_anything)
4564
            return false;
4565
 
4566
          finished_solution = BITMAP_GGC_ALLOC ();
4567
          set_uids_in_ptset (vi->decl, finished_solution, vi->solution);
4568
          result = shared_bitmap_lookup (finished_solution);
4569
 
4570
          if (!result)
4571
            {
4572
              shared_bitmap_add (finished_solution);
4573
              pi->pt_vars = finished_solution;
4574
            }
4575
          else
4576
            {
4577
              pi->pt_vars = result;
4578
              bitmap_clear (finished_solution);
4579
            }
4580
 
4581
          if (bitmap_empty_p (pi->pt_vars))
4582
            pi->pt_vars = NULL;
4583
 
4584
          return true;
4585
        }
4586
    }
4587
 
4588
  return false;
4589
}
4590
 
4591
 
4592
 
4593
/* Dump points-to information to OUTFILE.  */
4594
 
4595
void
4596
dump_sa_points_to_info (FILE *outfile)
4597
{
4598
  unsigned int i;
4599
 
4600
  fprintf (outfile, "\nPoints-to sets\n\n");
4601
 
4602
  if (dump_flags & TDF_STATS)
4603
    {
4604
      fprintf (outfile, "Stats:\n");
4605
      fprintf (outfile, "Total vars:               %d\n", stats.total_vars);
4606
      fprintf (outfile, "Non-pointer vars:          %d\n",
4607
               stats.nonpointer_vars);
4608
      fprintf (outfile, "Statically unified vars:  %d\n",
4609
               stats.unified_vars_static);
4610
      fprintf (outfile, "Dynamically unified vars: %d\n",
4611
               stats.unified_vars_dynamic);
4612
      fprintf (outfile, "Iterations:               %d\n", stats.iterations);
4613
      fprintf (outfile, "Number of edges:          %d\n", stats.num_edges);
4614
      fprintf (outfile, "Number of implicit edges: %d\n",
4615
               stats.num_implicit_edges);
4616
    }
4617
 
4618
  for (i = 0; i < VEC_length (varinfo_t, varmap); i++)
4619
    dump_solution_for_var (outfile, i);
4620
}
4621
 
4622
 
4623
/* Debug points-to information to stderr.  */
4624
 
4625
void
4626
debug_sa_points_to_info (void)
4627
{
4628
  dump_sa_points_to_info (stderr);
4629
}
4630
 
4631
 
4632
/* Initialize the always-existing constraint variables for NULL
4633
   ANYTHING, READONLY, and INTEGER */
4634
 
4635
static void
4636
init_base_vars (void)
4637
{
4638
  struct constraint_expr lhs, rhs;
4639
 
4640
  /* Create the NULL variable, used to represent that a variable points
4641
     to NULL.  */
4642
  nothing_tree = create_tmp_var_raw (void_type_node, "NULL");
4643
  var_nothing = new_var_info (nothing_tree, 0, "NULL");
4644
  insert_vi_for_tree (nothing_tree, var_nothing);
4645
  var_nothing->is_artificial_var = 1;
4646
  var_nothing->offset = 0;
4647
  var_nothing->size = ~0;
4648
  var_nothing->fullsize = ~0;
4649
  var_nothing->is_special_var = 1;
4650
  nothing_id = 0;
4651
  VEC_safe_push (varinfo_t, heap, varmap, var_nothing);
4652
 
4653
  /* Create the ANYTHING variable, used to represent that a variable
4654
     points to some unknown piece of memory.  */
4655
  anything_tree = create_tmp_var_raw (void_type_node, "ANYTHING");
4656
  var_anything = new_var_info (anything_tree, 1, "ANYTHING");
4657
  insert_vi_for_tree (anything_tree, var_anything);
4658
  var_anything->is_artificial_var = 1;
4659
  var_anything->size = ~0;
4660
  var_anything->offset = 0;
4661
  var_anything->next = NULL;
4662
  var_anything->fullsize = ~0;
4663
  var_anything->is_special_var = 1;
4664
  anything_id = 1;
4665
 
4666
  /* Anything points to anything.  This makes deref constraints just
4667
     work in the presence of linked list and other p = *p type loops,
4668
     by saying that *ANYTHING = ANYTHING. */
4669
  VEC_safe_push (varinfo_t, heap, varmap, var_anything);
4670
  lhs.type = SCALAR;
4671
  lhs.var = anything_id;
4672
  lhs.offset = 0;
4673
  rhs.type = ADDRESSOF;
4674
  rhs.var = anything_id;
4675
  rhs.offset = 0;
4676
 
4677
  /* This specifically does not use process_constraint because
4678
     process_constraint ignores all anything = anything constraints, since all
4679
     but this one are redundant.  */
4680
  VEC_safe_push (constraint_t, heap, constraints, new_constraint (lhs, rhs));
4681
 
4682
  /* Create the READONLY variable, used to represent that a variable
4683
     points to readonly memory.  */
4684
  readonly_tree = create_tmp_var_raw (void_type_node, "READONLY");
4685
  var_readonly = new_var_info (readonly_tree, 2, "READONLY");
4686
  var_readonly->is_artificial_var = 1;
4687
  var_readonly->offset = 0;
4688
  var_readonly->size = ~0;
4689
  var_readonly->fullsize = ~0;
4690
  var_readonly->next = NULL;
4691
  var_readonly->is_special_var = 1;
4692
  insert_vi_for_tree (readonly_tree, var_readonly);
4693
  readonly_id = 2;
4694
  VEC_safe_push (varinfo_t, heap, varmap, var_readonly);
4695
 
4696
  /* readonly memory points to anything, in order to make deref
4697
     easier.  In reality, it points to anything the particular
4698
     readonly variable can point to, but we don't track this
4699
     separately. */
4700
  lhs.type = SCALAR;
4701
  lhs.var = readonly_id;
4702
  lhs.offset = 0;
4703
  rhs.type = ADDRESSOF;
4704
  rhs.var = anything_id;
4705
  rhs.offset = 0;
4706
 
4707
  process_constraint (new_constraint (lhs, rhs));
4708
 
4709
  /* Create the INTEGER variable, used to represent that a variable points
4710
     to an INTEGER.  */
4711
  integer_tree = create_tmp_var_raw (void_type_node, "INTEGER");
4712
  var_integer = new_var_info (integer_tree, 3, "INTEGER");
4713
  insert_vi_for_tree (integer_tree, var_integer);
4714
  var_integer->is_artificial_var = 1;
4715
  var_integer->size = ~0;
4716
  var_integer->fullsize = ~0;
4717
  var_integer->offset = 0;
4718
  var_integer->next = NULL;
4719
  var_integer->is_special_var = 1;
4720
  integer_id = 3;
4721
  VEC_safe_push (varinfo_t, heap, varmap, var_integer);
4722
 
4723
  /* INTEGER = ANYTHING, because we don't know where a dereference of
4724
     a random integer will point to.  */
4725
  lhs.type = SCALAR;
4726
  lhs.var = integer_id;
4727
  lhs.offset = 0;
4728
  rhs.type = ADDRESSOF;
4729
  rhs.var = anything_id;
4730
  rhs.offset = 0;
4731
  process_constraint (new_constraint (lhs, rhs));
4732
 
4733
  /* Create the ESCAPED_VARS variable used to represent variables that
4734
     escape this function.  */
4735
  escaped_vars_tree = create_tmp_var_raw (void_type_node, "ESCAPED_VARS");
4736
  var_escaped_vars = new_var_info (escaped_vars_tree, 4, "ESCAPED_VARS");
4737
  insert_vi_for_tree (escaped_vars_tree, var_escaped_vars);
4738
  var_escaped_vars->is_artificial_var = 1;
4739
  var_escaped_vars->size = ~0;
4740
  var_escaped_vars->fullsize = ~0;
4741
  var_escaped_vars->offset = 0;
4742
  var_escaped_vars->next = NULL;
4743
  escaped_vars_id = 4;
4744
  VEC_safe_push (varinfo_t, heap, varmap, var_escaped_vars);
4745
 
4746
  /* ESCAPED_VARS = *ESCAPED_VARS */
4747
  lhs.type = SCALAR;
4748
  lhs.var = escaped_vars_id;
4749
  lhs.offset = 0;
4750
  rhs.type = DEREF;
4751
  rhs.var = escaped_vars_id;
4752
  rhs.offset = 0;
4753
  process_constraint (new_constraint (lhs, rhs));
4754
 
4755
}
4756
 
4757
/* Initialize things necessary to perform PTA */
4758
 
4759
static void
4760
init_alias_vars (void)
4761
{
4762
  bitmap_obstack_initialize (&pta_obstack);
4763
  bitmap_obstack_initialize (&oldpta_obstack);
4764
  bitmap_obstack_initialize (&predbitmap_obstack);
4765
 
4766
  constraint_pool = create_alloc_pool ("Constraint pool",
4767
                                       sizeof (struct constraint), 30);
4768
  variable_info_pool = create_alloc_pool ("Variable info pool",
4769
                                          sizeof (struct variable_info), 30);
4770
  constraints = VEC_alloc (constraint_t, heap, 8);
4771
  varmap = VEC_alloc (varinfo_t, heap, 8);
4772
  vi_for_tree = pointer_map_create ();
4773
 
4774
  memset (&stats, 0, sizeof (stats));
4775
  shared_bitmap_table = htab_create (511, shared_bitmap_hash,
4776
                                     shared_bitmap_eq, free);
4777
  init_base_vars ();
4778
}
4779
 
4780
/* Given a statement STMT, generate necessary constraints to
4781
   escaped_vars for the escaping variables.  */
4782
 
4783
static void
4784
find_escape_constraints (tree stmt)
4785
{
4786
  enum escape_type stmt_escape_type = is_escape_site (stmt);
4787
  tree rhs;
4788
  VEC(ce_s, heap) *rhsc = NULL;
4789
  struct constraint_expr *c;
4790
  size_t i;
4791
 
4792
  if (stmt_escape_type == NO_ESCAPE)
4793
    return;
4794
 
4795
  if (TREE_CODE (stmt) == RETURN_EXPR)
4796
    {
4797
      /* Returns are either bare, with an embedded MODIFY_EXPR, or
4798
         just a plain old expression.  */
4799
      if (!TREE_OPERAND (stmt, 0))
4800
        return;
4801
      if (TREE_CODE (TREE_OPERAND (stmt, 0)) == MODIFY_EXPR)
4802
        rhs = TREE_OPERAND (TREE_OPERAND (stmt, 0), 1);
4803
      else
4804
        rhs = TREE_OPERAND (stmt, 0);
4805
 
4806
      get_constraint_for (rhs, &rhsc);
4807
      for (i = 0; VEC_iterate (ce_s, rhsc, i, c); i++)
4808
        make_constraint_to_escaped (*c);
4809
      VEC_free (ce_s, heap, rhsc);
4810
      return;
4811
    }
4812
  else if (TREE_CODE (stmt) == ASM_EXPR)
4813
    {
4814
      /* Whatever the inputs of the ASM are, escape.  */
4815
      tree arg;
4816
 
4817
      for (arg = ASM_INPUTS (stmt); arg; arg = TREE_CHAIN (arg))
4818
        {
4819
          rhsc = NULL;
4820
          get_constraint_for (TREE_VALUE (arg), &rhsc);
4821
          for (i = 0; VEC_iterate (ce_s, rhsc, i, c); i++)
4822
            make_constraint_to_escaped (*c);
4823
          VEC_free (ce_s, heap, rhsc);
4824
        }
4825
      return;
4826
    }
4827
  else if (TREE_CODE (stmt) == CALL_EXPR
4828
           || (TREE_CODE (stmt) == MODIFY_EXPR
4829
               && TREE_CODE (TREE_OPERAND (stmt, 1)) == CALL_EXPR))
4830
    {
4831
      /* Calls cause all of the arguments passed in to escape.  */
4832
      tree arg;
4833
 
4834
      if (TREE_CODE (stmt) == MODIFY_EXPR)
4835
        stmt = TREE_OPERAND (stmt, 1);
4836
      for (arg = TREE_OPERAND (stmt, 1); arg; arg = TREE_CHAIN (arg))
4837
        {
4838
          if (POINTER_TYPE_P (TREE_TYPE (TREE_VALUE (arg))))
4839
            {
4840
              rhsc = NULL;
4841
              get_constraint_for (TREE_VALUE (arg), &rhsc);
4842
              for (i = 0; VEC_iterate (ce_s, rhsc, i, c); i++)
4843
                make_constraint_to_escaped (*c);
4844
              VEC_free (ce_s, heap, rhsc);
4845
            }
4846
        }
4847
      return;
4848
    }
4849
  else
4850
    {
4851
      gcc_assert (TREE_CODE (stmt) == MODIFY_EXPR);
4852
    }
4853
 
4854
  gcc_assert (stmt_escape_type == ESCAPE_BAD_CAST
4855
              || stmt_escape_type == ESCAPE_STORED_IN_GLOBAL
4856
              || stmt_escape_type == ESCAPE_UNKNOWN);
4857
  rhs = TREE_OPERAND (stmt, 1);
4858
 
4859
  /* Look through casts for the real escaping variable.
4860
     Constants don't really escape, so ignore them.
4861
     Otherwise, whatever escapes must be on our RHS.  */
4862
  if (TREE_CODE (rhs) == NOP_EXPR
4863
      || TREE_CODE (rhs) == CONVERT_EXPR
4864
      || TREE_CODE (rhs) == NON_LVALUE_EXPR)
4865
    {
4866
      get_constraint_for (TREE_OPERAND (rhs, 0), &rhsc);
4867
    }
4868
  else if (CONSTANT_CLASS_P (rhs))
4869
    return;
4870
  else
4871
    {
4872
      get_constraint_for (rhs, &rhsc);
4873
    }
4874
  for (i = 0; VEC_iterate (ce_s, rhsc, i, c); i++)
4875
    make_constraint_to_escaped (*c);
4876
  VEC_free (ce_s, heap, rhsc);
4877
}
4878
 
4879
 
4880
/* Remove the REF and ADDRESS edges from GRAPH, as well as all the
4881
   predecessor edges.  */
4882
 
4883
static void
4884
remove_preds_and_fake_succs (constraint_graph_t graph)
4885
{
4886
  unsigned int i;
4887
 
4888
  /* Clear the implicit ref and address nodes from the successor
4889
     lists.  */
4890
  for (i = 0; i < FIRST_REF_NODE; i++)
4891
    {
4892
      if (graph->succs[i])
4893
        bitmap_clear_range (graph->succs[i], FIRST_REF_NODE,
4894
                            FIRST_REF_NODE * 2);
4895
    }
4896
 
4897
  /* Free the successor list for the non-ref nodes.  */
4898
  for (i = FIRST_REF_NODE; i < graph->size; i++)
4899
    {
4900
      if (graph->succs[i])
4901
        BITMAP_FREE (graph->succs[i]);
4902
    }
4903
 
4904
  /* Now reallocate the size of the successor list as, and blow away
4905
     the predecessor bitmaps.  */
4906
  graph->size = VEC_length (varinfo_t, varmap);
4907
  graph->succs = xrealloc (graph->succs, graph->size * sizeof (bitmap));
4908
 
4909
  free (graph->implicit_preds);
4910
  graph->implicit_preds = NULL;
4911
  free (graph->preds);
4912
  graph->preds = NULL;
4913
  bitmap_obstack_release (&predbitmap_obstack);
4914
}
4915
 
4916
/* Create points-to sets for the current function.  See the comments
4917
   at the start of the file for an algorithmic overview.  */
4918
 
4919
void
4920
compute_points_to_sets (struct alias_info *ai)
4921
{
4922
  basic_block bb;
4923
  struct scc_info *si;
4924
 
4925
  timevar_push (TV_TREE_PTA);
4926
 
4927
  init_alias_vars ();
4928
  init_alias_heapvars ();
4929
 
4930
  intra_create_variable_infos ();
4931
 
4932
  /* Now walk all statements and derive aliases.  */
4933
  FOR_EACH_BB (bb)
4934
    {
4935
      block_stmt_iterator bsi;
4936
      tree phi;
4937
 
4938
      for (phi = phi_nodes (bb); phi; phi = TREE_CHAIN (phi))
4939
        {
4940
          if (is_gimple_reg (PHI_RESULT (phi)))
4941
            {
4942
              find_func_aliases (phi);
4943
              /* Update various related attributes like escaped
4944
                 addresses, pointer dereferences for loads and stores.
4945
                 This is used when creating name tags and alias
4946
                 sets.  */
4947
              update_alias_info (phi, ai);
4948
            }
4949
        }
4950
 
4951
      for (bsi = bsi_start (bb); !bsi_end_p (bsi); bsi_next (&bsi))
4952
        {
4953
          tree stmt = bsi_stmt (bsi);
4954
 
4955
          find_func_aliases (stmt);
4956
          find_escape_constraints (stmt);
4957
          /* Update various related attributes like escaped
4958
             addresses, pointer dereferences for loads and stores.
4959
             This is used when creating name tags and alias
4960
             sets.  */
4961
          update_alias_info (stmt, ai);
4962
        }
4963
    }
4964
 
4965
 
4966
  if (dump_file)
4967
    {
4968
      fprintf (dump_file, "Points-to analysis\n\nConstraints:\n\n");
4969
      dump_constraints (dump_file);
4970
    }
4971
 
4972
  if (dump_file)
4973
    fprintf (dump_file,
4974
             "\nCollapsing static cycles and doing variable "
4975
             "substitution:\n");
4976
 
4977
  build_pred_graph ();
4978
  si = perform_var_substitution (graph);
4979
  move_complex_constraints (graph, si);
4980
  free_var_substitution_info (si);
4981
 
4982
  build_succ_graph ();
4983
  find_indirect_cycles (graph);
4984
 
4985
  /* Implicit nodes and predecessors are no longer necessary at this
4986
     point. */
4987
  remove_preds_and_fake_succs (graph);
4988
 
4989
  if (dump_file)
4990
    fprintf (dump_file, "\nSolving graph:\n");
4991
 
4992
  solve_graph (graph);
4993
 
4994
  if (dump_file)
4995
    dump_sa_points_to_info (dump_file);
4996
  have_alias_info = true;
4997
 
4998
  timevar_pop (TV_TREE_PTA);
4999
}
5000
 
5001
/* Delete created points-to sets.  */
5002
 
5003
void
5004
delete_points_to_sets (void)
5005
{
5006
  varinfo_t v;
5007
  int i;
5008
 
5009
  htab_delete (shared_bitmap_table);
5010
  if (dump_file && (dump_flags & TDF_STATS))
5011
    fprintf (dump_file, "Points to sets created:%d\n",
5012
             stats.points_to_sets_created);
5013
 
5014
  pointer_map_destroy (vi_for_tree);
5015
  bitmap_obstack_release (&pta_obstack);
5016
  VEC_free (constraint_t, heap, constraints);
5017
 
5018
  for (i = 0; VEC_iterate (varinfo_t, varmap, i, v); i++)
5019
    VEC_free (constraint_t, heap, graph->complex[i]);
5020
  free (graph->complex);
5021
 
5022
  free (graph->rep);
5023
  free (graph->succs);
5024
  free (graph->indirect_cycles);
5025
  free (graph);
5026
 
5027
  VEC_free (varinfo_t, heap, varmap);
5028
  free_alloc_pool (variable_info_pool);
5029
  free_alloc_pool (constraint_pool);
5030
  have_alias_info = false;
5031
}
5032
 
5033
/* Return true if we should execute IPA PTA.  */
5034
static bool
5035
gate_ipa_pta (void)
5036
{
5037
  return (flag_unit_at_a_time != 0
5038
          && flag_ipa_pta
5039
          /* Don't bother doing anything if the program has errors.  */
5040
          && !(errorcount || sorrycount));
5041
}
5042
 
5043
/* Execute the driver for IPA PTA.  */
5044
static unsigned int
5045
ipa_pta_execute (void)
5046
{
5047
#if 0
5048
  struct cgraph_node *node;
5049
  in_ipa_mode = 1;
5050
  init_alias_heapvars ();
5051
  init_alias_vars ();
5052
 
5053
  for (node = cgraph_nodes; node; node = node->next)
5054
    {
5055
      if (!node->analyzed || cgraph_is_master_clone (node))
5056
        {
5057
          unsigned int varid;
5058
 
5059
          varid = create_function_info_for (node->decl,
5060
                                            cgraph_node_name (node));
5061
          if (node->local.externally_visible)
5062
            {
5063
              varinfo_t fi = get_varinfo (varid);
5064
              for (; fi; fi = fi->next)
5065
                make_constraint_from_escaped (fi);
5066
            }
5067
        }
5068
    }
5069
  for (node = cgraph_nodes; node; node = node->next)
5070
    {
5071
      if (node->analyzed && cgraph_is_master_clone (node))
5072
        {
5073
          struct function *cfun = DECL_STRUCT_FUNCTION (node->decl);
5074
          basic_block bb;
5075
          tree old_func_decl = current_function_decl;
5076
          if (dump_file)
5077
            fprintf (dump_file,
5078
                     "Generating constraints for %s\n",
5079
                     cgraph_node_name (node));
5080
          push_cfun (cfun);
5081
          current_function_decl = node->decl;
5082
 
5083
          FOR_EACH_BB_FN (bb, cfun)
5084
            {
5085
              block_stmt_iterator bsi;
5086
              tree phi;
5087
 
5088
              for (phi = phi_nodes (bb); phi; phi = TREE_CHAIN (phi))
5089
                {
5090
                  if (is_gimple_reg (PHI_RESULT (phi)))
5091
                    {
5092
                      find_func_aliases (phi);
5093
                    }
5094
                }
5095
 
5096
              for (bsi = bsi_start (bb); !bsi_end_p (bsi); bsi_next (&bsi))
5097
                {
5098
                  tree stmt = bsi_stmt (bsi);
5099
                  find_func_aliases (stmt);
5100
                }
5101
            }
5102
          current_function_decl = old_func_decl;
5103
          pop_cfun ();
5104
        }
5105
      else
5106
        {
5107
          /* Make point to anything.  */
5108
        }
5109
    }
5110
 
5111
  build_constraint_graph ();
5112
 
5113
  if (dump_file)
5114
    {
5115
      fprintf (dump_file, "Points-to analysis\n\nConstraints:\n\n");
5116
      dump_constraints (dump_file);
5117
    }
5118
 
5119
  if (dump_file)
5120
    fprintf (dump_file,
5121
             "\nCollapsing static cycles and doing variable "
5122
             "substitution:\n");
5123
 
5124
  find_and_collapse_graph_cycles (graph, false);
5125
  perform_var_substitution (graph);
5126
 
5127
  if (dump_file)
5128
    fprintf (dump_file, "\nSolving graph:\n");
5129
 
5130
  solve_graph (graph);
5131
 
5132
  if (dump_file)
5133
    dump_sa_points_to_info (dump_file);
5134
  in_ipa_mode = 0;
5135
  delete_alias_heapvars ();
5136
  delete_points_to_sets ();
5137
#endif
5138
  return 0;
5139
}
5140
 
5141
struct tree_opt_pass pass_ipa_pta =
5142
{
5143
  "pta",                                /* name */
5144
  gate_ipa_pta,                 /* gate */
5145
  ipa_pta_execute,                      /* execute */
5146
  NULL,                                 /* sub */
5147
  NULL,                                 /* next */
5148
  0,                                     /* static_pass_number */
5149
  TV_IPA_PTA,                   /* tv_id */
5150
  0,                                     /* properties_required */
5151
  0,                                     /* properties_provided */
5152
  0,                                     /* properties_destroyed */
5153
  0,                                     /* todo_flags_start */
5154
  0,                                    /* todo_flags_finish */
5155
 
5156
};
5157
 
5158
/* Initialize the heapvar for statement mapping.  */
5159
void
5160
init_alias_heapvars (void)
5161
{
5162
  if (!heapvar_for_stmt)
5163
    heapvar_for_stmt = htab_create_ggc (11, tree_map_hash, tree_map_eq,
5164
                                        NULL);
5165
  nonlocal_all = NULL_TREE;
5166
}
5167
 
5168
void
5169
delete_alias_heapvars (void)
5170
{
5171
  nonlocal_all = NULL_TREE;
5172
  htab_delete (heapvar_for_stmt);
5173
  heapvar_for_stmt = NULL;
5174
}
5175
 
5176
#include "gt-tree-ssa-structalias.h"

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.