Branch data Line data Source code
1 : : /* Common block and equivalence list handling
2 : : Copyright (C) 2000-2025 Free Software Foundation, Inc.
3 : : Contributed by Canqun Yang <canqun@nudt.edu.cn>
4 : :
5 : : This file is part of GCC.
6 : :
7 : : GCC is free software; you can redistribute it and/or modify it under
8 : : the terms of the GNU General Public License as published by the Free
9 : : Software Foundation; either version 3, or (at your option) any later
10 : : version.
11 : :
12 : : GCC is distributed in the hope that it will be useful, but WITHOUT ANY
13 : : WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 : : FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
15 : : 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 : : /* The core algorithm is based on Andy Vaught's g95 tree. Also the
22 : : way to build UNION_TYPE is borrowed from Richard Henderson.
23 : :
24 : : Transform common blocks. An integral part of this is processing
25 : : equivalence variables. Equivalenced variables that are not in a
26 : : common block end up in a private block of their own.
27 : :
28 : : Each common block or local equivalence list is declared as a union.
29 : : Variables within the block are represented as a field within the
30 : : block with the proper offset.
31 : :
32 : : So if two variables are equivalenced, they just point to a common
33 : : area in memory.
34 : :
35 : : Mathematically, laying out an equivalence block is equivalent to
36 : : solving a linear system of equations. The matrix is usually a
37 : : sparse matrix in which each row contains all zero elements except
38 : : for a +1 and a -1, a sort of a generalized Vandermonde matrix. The
39 : : matrix is usually block diagonal. The system can be
40 : : overdetermined, underdetermined or have a unique solution. If the
41 : : system is inconsistent, the program is not standard conforming.
42 : : The solution vector is integral, since all of the pivots are +1 or -1.
43 : :
44 : : How we lay out an equivalence block is a little less complicated.
45 : : In an equivalence list with n elements, there are n-1 conditions to
46 : : be satisfied. The conditions partition the variables into what we
47 : : will call segments. If A and B are equivalenced then A and B are
48 : : in the same segment. If B and C are equivalenced as well, then A,
49 : : B and C are in a segment and so on. Each segment is a block of
50 : : memory that has one or more variables equivalenced in some way. A
51 : : common block is made up of a series of segments that are joined one
52 : : after the other. In the linear system, a segment is a block
53 : : diagonal.
54 : :
55 : : To lay out a segment we first start with some variable and
56 : : determine its length. The first variable is assumed to start at
57 : : offset one and extends to however long it is. We then traverse the
58 : : list of equivalences to find an unused condition that involves at
59 : : least one of the variables currently in the segment.
60 : :
61 : : Each equivalence condition amounts to the condition B+b=C+c where B
62 : : and C are the offsets of the B and C variables, and b and c are
63 : : constants which are nonzero for array elements, substrings or
64 : : structure components. So for
65 : :
66 : : EQUIVALENCE(B(2), C(3))
67 : : we have
68 : : B + 2*size of B's elements = C + 3*size of C's elements.
69 : :
70 : : If B and C are known we check to see if the condition already
71 : : holds. If B is known we can solve for C. Since we know the length
72 : : of C, we can see if the minimum and maximum extents of the segment
73 : : are affected. Eventually, we make a full pass through the
74 : : equivalence list without finding any new conditions and the segment
75 : : is fully specified.
76 : :
77 : : At this point, the segment is added to the current common block.
78 : : Since we know the minimum extent of the segment, everything in the
79 : : segment is translated to its position in the common block. The
80 : : usual case here is that there are no equivalence statements and the
81 : : common block is series of segments with one variable each, which is
82 : : a diagonal matrix in the matrix formulation.
83 : :
84 : : Each segment is described by a chain of segment_info structures. Each
85 : : segment_info structure describes the extents of a single variable within
86 : : the segment. This list is maintained in the order the elements are
87 : : positioned within the segment. If two elements have the same starting
88 : : offset the smaller will come first. If they also have the same size their
89 : : ordering is undefined.
90 : :
91 : : Once all common blocks have been created, the list of equivalences
92 : : is examined for still-unused equivalence conditions. We create a
93 : : block for each merged equivalence list. */
94 : :
95 : : #include "config.h"
96 : : #define INCLUDE_MAP
97 : : #include "system.h"
98 : : #include "coretypes.h"
99 : : #include "tm.h"
100 : : #include "tree.h"
101 : : #include "cgraph.h"
102 : : #include "context.h"
103 : : #include "omp-offload.h"
104 : : #include "gfortran.h"
105 : : #include "trans.h"
106 : : #include "stringpool.h"
107 : : #include "fold-const.h"
108 : : #include "stor-layout.h"
109 : : #include "varasm.h"
110 : : #include "trans-types.h"
111 : : #include "trans-const.h"
112 : : #include "target-memory.h"
113 : :
114 : :
115 : : /* Holds a single variable in an equivalence set. */
116 : : typedef struct segment_info
117 : : {
118 : : gfc_symbol *sym;
119 : : HOST_WIDE_INT offset;
120 : : HOST_WIDE_INT length;
121 : : /* This will contain the field type until the field is created. */
122 : : tree field;
123 : : struct segment_info *next;
124 : : } segment_info;
125 : :
126 : : static segment_info * current_segment;
127 : :
128 : : /* Store decl of all common blocks in this translation unit; the first
129 : : tree is the identifier. */
130 : : static std::map<tree, tree> gfc_map_of_all_commons;
131 : :
132 : :
133 : : /* Make a segment_info based on a symbol. */
134 : :
135 : : static segment_info *
136 : 8115 : get_segment_info (gfc_symbol * sym, HOST_WIDE_INT offset)
137 : : {
138 : 8115 : segment_info *s;
139 : :
140 : : /* Make sure we've got the character length. */
141 : 8115 : if (sym->ts.type == BT_CHARACTER)
142 : 641 : gfc_conv_const_charlen (sym->ts.u.cl);
143 : :
144 : : /* Create the segment_info and fill it in. */
145 : 8115 : s = XCNEW (segment_info);
146 : 8115 : s->sym = sym;
147 : : /* We will use this type when building the segment aggregate type. */
148 : 8115 : s->field = gfc_sym_type (sym);
149 : 8115 : s->length = int_size_in_bytes (s->field);
150 : 8115 : s->offset = offset;
151 : :
152 : 8115 : return s;
153 : : }
154 : :
155 : :
156 : : /* Add a copy of a segment list to the namespace. This is specifically for
157 : : equivalence segments, so that dependency checking can be done on
158 : : equivalence group members. */
159 : :
160 : : static void
161 : 6554 : copy_equiv_list_to_ns (segment_info *c)
162 : : {
163 : 6554 : segment_info *f;
164 : 6554 : gfc_equiv_info *s;
165 : 6554 : gfc_equiv_list *l;
166 : :
167 : 6554 : l = XCNEW (gfc_equiv_list);
168 : :
169 : 6554 : l->next = c->sym->ns->equiv_lists;
170 : 6554 : c->sym->ns->equiv_lists = l;
171 : :
172 : 14669 : for (f = c; f; f = f->next)
173 : : {
174 : 8115 : s = XCNEW (gfc_equiv_info);
175 : 8115 : s->next = l->equiv;
176 : 8115 : l->equiv = s;
177 : 8115 : s->sym = f->sym;
178 : 8115 : s->offset = f->offset;
179 : 8115 : s->length = f->length;
180 : : }
181 : 6554 : }
182 : :
183 : :
184 : : /* Add combine segment V and segment LIST. */
185 : :
186 : : static segment_info *
187 : 7322 : add_segments (segment_info *list, segment_info *v)
188 : : {
189 : 7322 : segment_info *s;
190 : 7322 : segment_info *p;
191 : 7322 : segment_info *next;
192 : :
193 : 7322 : p = NULL;
194 : 7322 : s = list;
195 : :
196 : 14991 : while (v)
197 : : {
198 : : /* Find the location of the new element. */
199 : 42136 : while (s)
200 : : {
201 : 35575 : if (v->offset < s->offset)
202 : : break;
203 : 35153 : if (v->offset == s->offset
204 : 732 : && v->length <= s->length)
205 : : break;
206 : :
207 : 34467 : p = s;
208 : 34467 : s = s->next;
209 : : }
210 : :
211 : : /* Insert the new element in between p and s. */
212 : 7669 : next = v->next;
213 : 7669 : v->next = s;
214 : 7669 : if (p == NULL)
215 : : list = v;
216 : : else
217 : 4770 : p->next = v;
218 : :
219 : : p = v;
220 : : v = next;
221 : : }
222 : :
223 : 7322 : return list;
224 : : }
225 : :
226 : :
227 : : /* Construct mangled common block name from symbol name. */
228 : :
229 : : /* We need the bind(c) flag to tell us how/if we should mangle the symbol
230 : : name. There are few calls to this function, so few places that this
231 : : would need to be added. At the moment, there is only one call, in
232 : : build_common_decl(). We can't attempt to look up the common block
233 : : because we may be building it for the first time and therefore, it won't
234 : : be in the common_root. We also need the binding label, if it's bind(c).
235 : : Therefore, send in the pointer to the common block, so whatever info we
236 : : have so far can be used. All of the necessary info should be available
237 : : in the gfc_common_head by now, so it should be accurate to test the
238 : : isBindC flag and use the binding label given if it is bind(c).
239 : :
240 : : We may NOT know yet if it's bind(c) or not, but we can try at least.
241 : : Will have to figure out what to do later if it's labeled bind(c)
242 : : after this is called. */
243 : :
244 : : static tree
245 : 2061 : gfc_sym_mangled_common_id (gfc_common_head *com)
246 : : {
247 : 2061 : int has_underscore;
248 : : /* Provide sufficient space to hold "symbol.symbol.eq.1234567890__". */
249 : 2061 : char mangled_name[2*GFC_MAX_MANGLED_SYMBOL_LEN + 1 + 16 + 1];
250 : 2061 : char name[sizeof (mangled_name) - 2];
251 : :
252 : : /* Get the name out of the common block pointer. */
253 : 2061 : size_t len = strlen (com->name);
254 : 2061 : gcc_assert (len < sizeof (name));
255 : 2061 : strcpy (name, com->name);
256 : :
257 : : /* If we're suppose to do a bind(c). */
258 : 2061 : if (com->is_bind_c == 1 && com->binding_label)
259 : 69 : return get_identifier (com->binding_label);
260 : :
261 : 1992 : if (strcmp (name, BLANK_COMMON_NAME) == 0)
262 : 192 : return get_identifier (name);
263 : :
264 : 1800 : if (flag_underscoring)
265 : : {
266 : 1800 : has_underscore = strchr (name, '_') != 0;
267 : 1800 : if (flag_second_underscore && has_underscore)
268 : 4 : snprintf (mangled_name, sizeof mangled_name, "%s__", name);
269 : : else
270 : 1796 : snprintf (mangled_name, sizeof mangled_name, "%s_", name);
271 : :
272 : 1800 : return get_identifier (mangled_name);
273 : : }
274 : : else
275 : 0 : return get_identifier (name);
276 : : }
277 : :
278 : :
279 : : /* Build a field declaration for a common variable or a local equivalence
280 : : object. */
281 : :
282 : : static void
283 : 8115 : build_field (segment_info *h, tree union_type, record_layout_info rli)
284 : : {
285 : 8115 : tree field;
286 : 8115 : tree name;
287 : 8115 : HOST_WIDE_INT offset = h->offset;
288 : 8115 : unsigned HOST_WIDE_INT desired_align, known_align;
289 : :
290 : 8115 : name = get_identifier (h->sym->name);
291 : 8115 : field = build_decl (gfc_get_location (&h->sym->declared_at),
292 : : FIELD_DECL, name, h->field);
293 : 8115 : known_align = (offset & -offset) * BITS_PER_UNIT;
294 : 12887 : if (known_align == 0 || known_align > BIGGEST_ALIGNMENT)
295 : 4900 : known_align = BIGGEST_ALIGNMENT;
296 : :
297 : 8115 : desired_align = update_alignment_for_field (rli, field, known_align);
298 : 8115 : if (desired_align > known_align)
299 : 7 : DECL_PACKED (field) = 1;
300 : :
301 : 8115 : DECL_FIELD_CONTEXT (field) = union_type;
302 : 8115 : DECL_FIELD_OFFSET (field) = size_int (offset);
303 : 8115 : DECL_FIELD_BIT_OFFSET (field) = bitsize_zero_node;
304 : 8115 : SET_DECL_OFFSET_ALIGN (field, known_align);
305 : :
306 : 8115 : rli->offset = size_binop (MAX_EXPR, rli->offset,
307 : : size_binop (PLUS_EXPR,
308 : : DECL_FIELD_OFFSET (field),
309 : : DECL_SIZE_UNIT (field)));
310 : : /* If this field is assigned to a label, we create another two variables.
311 : : One will hold the address of target label or format label. The other will
312 : : hold the length of format label string. */
313 : 8115 : if (h->sym->attr.assign)
314 : : {
315 : 14 : tree len;
316 : 14 : tree addr;
317 : :
318 : 14 : gfc_allocate_lang_decl (field);
319 : 14 : GFC_DECL_ASSIGN (field) = 1;
320 : 14 : len = gfc_create_var_np (gfc_charlen_type_node,h->sym->name);
321 : 14 : addr = gfc_create_var_np (pvoid_type_node, h->sym->name);
322 : 14 : TREE_STATIC (len) = 1;
323 : 14 : TREE_STATIC (addr) = 1;
324 : 14 : DECL_INITIAL (len) = build_int_cst (gfc_charlen_type_node, -2);
325 : 14 : gfc_set_decl_location (len, &h->sym->declared_at);
326 : 14 : gfc_set_decl_location (addr, &h->sym->declared_at);
327 : 14 : GFC_DECL_STRING_LEN (field) = pushdecl_top_level (len);
328 : 14 : GFC_DECL_ASSIGN_ADDR (field) = pushdecl_top_level (addr);
329 : : }
330 : :
331 : : /* If this field is volatile, mark it. */
332 : 8115 : if (h->sym->attr.volatile_)
333 : : {
334 : 3 : tree new_type;
335 : 3 : TREE_THIS_VOLATILE (field) = 1;
336 : 3 : TREE_SIDE_EFFECTS (field) = 1;
337 : 3 : new_type = build_qualified_type (TREE_TYPE (field), TYPE_QUAL_VOLATILE);
338 : 3 : TREE_TYPE (field) = new_type;
339 : : }
340 : :
341 : 8115 : h->field = field;
342 : 8115 : }
343 : :
344 : : #if !defined (NO_DOT_IN_LABEL)
345 : : #define GFC_EQUIV_FMT "equiv.%d"
346 : : #elif !defined (NO_DOLLAR_IN_LABEL)
347 : : #define GFC_EQUIV_FMT "_Equiv$%d"
348 : : #else
349 : : #define GFC_EQUIV_FMT "_Equiv_%d"
350 : : #endif
351 : :
352 : : /* Get storage for local equivalence. */
353 : :
354 : : static tree
355 : 688 : build_equiv_decl (tree union_type, bool is_init, bool is_saved, bool is_auto)
356 : : {
357 : 688 : tree decl;
358 : 688 : char name[18];
359 : 688 : static int serial = 0;
360 : :
361 : 688 : if (is_init)
362 : : {
363 : 141 : decl = gfc_create_var (union_type, "equiv");
364 : 141 : TREE_STATIC (decl) = 1;
365 : 141 : GFC_DECL_COMMON_OR_EQUIV (decl) = 1;
366 : 141 : return decl;
367 : : }
368 : :
369 : 547 : snprintf (name, sizeof (name), GFC_EQUIV_FMT, serial++);
370 : 547 : decl = build_decl (input_location,
371 : : VAR_DECL, get_identifier (name), union_type);
372 : 547 : DECL_ARTIFICIAL (decl) = 1;
373 : 547 : DECL_IGNORED_P (decl) = 1;
374 : :
375 : 547 : if (!is_auto && (!gfc_can_put_var_on_stack (DECL_SIZE_UNIT (decl))
376 : 526 : || is_saved))
377 : 27 : TREE_STATIC (decl) = 1;
378 : :
379 : 547 : TREE_ADDRESSABLE (decl) = 1;
380 : 547 : TREE_USED (decl) = 1;
381 : 547 : GFC_DECL_COMMON_OR_EQUIV (decl) = 1;
382 : :
383 : : /* The source location has been lost, and doesn't really matter.
384 : : We need to set it to something though. */
385 : 547 : DECL_SOURCE_LOCATION (decl) = input_location;
386 : :
387 : 547 : gfc_add_decl_to_function (decl);
388 : :
389 : 547 : return decl;
390 : : }
391 : :
392 : :
393 : : /* Get storage for common block. */
394 : :
395 : : static tree
396 : 2061 : build_common_decl (gfc_common_head *com, tree union_type, bool is_init)
397 : : {
398 : 2061 : tree decl, identifier;
399 : :
400 : 2061 : identifier = gfc_sym_mangled_common_id (com);
401 : 2061 : decl = gfc_map_of_all_commons.count(identifier)
402 : 969 : ? gfc_map_of_all_commons[identifier] : NULL_TREE;
403 : :
404 : : /* Update the size of this common block as needed. */
405 : 2061 : if (decl != NULL_TREE)
406 : : {
407 : 969 : tree size = TYPE_SIZE_UNIT (union_type);
408 : :
409 : : /* Named common blocks of the same name shall be of the same size
410 : : in all scoping units of a program in which they appear, but
411 : : blank common blocks may be of different sizes. */
412 : 969 : if (!tree_int_cst_equal (DECL_SIZE_UNIT (decl), size)
413 : 969 : && strcmp (com->name, BLANK_COMMON_NAME))
414 : 45 : gfc_warning (0, "Named COMMON block %qs at %L shall be of the "
415 : : "same size as elsewhere (%wu vs %wu bytes)", com->name,
416 : : &com->where,
417 : 15 : TREE_INT_CST_LOW (size),
418 : 15 : TREE_INT_CST_LOW (DECL_SIZE_UNIT (decl)));
419 : :
420 : 969 : if (tree_int_cst_lt (DECL_SIZE_UNIT (decl), size))
421 : : {
422 : 11 : DECL_SIZE (decl) = TYPE_SIZE (union_type);
423 : 11 : DECL_SIZE_UNIT (decl) = size;
424 : 11 : SET_DECL_MODE (decl, TYPE_MODE (union_type));
425 : 11 : TREE_TYPE (decl) = union_type;
426 : 11 : layout_decl (decl, 0);
427 : : }
428 : : }
429 : :
430 : : /* If this common block has been declared in a previous program unit,
431 : : and either it is already initialized or there is no new initialization
432 : : for it, just return. */
433 : 969 : if ((decl != NULL_TREE) && (!is_init || DECL_INITIAL (decl)))
434 : : return decl;
435 : :
436 : : /* If there is no backend_decl for the common block, build it. */
437 : 1110 : if (decl == NULL_TREE)
438 : : {
439 : 1092 : tree omp_clauses = NULL_TREE;
440 : :
441 : 1092 : if (com->is_bind_c == 1 && com->binding_label)
442 : 51 : decl = build_decl (input_location, VAR_DECL, identifier, union_type);
443 : : else
444 : : {
445 : 1041 : decl = build_decl (input_location, VAR_DECL, get_identifier (com->name),
446 : : union_type);
447 : 1041 : gfc_set_decl_assembler_name (decl, identifier);
448 : : }
449 : :
450 : 1092 : TREE_PUBLIC (decl) = 1;
451 : 1092 : TREE_STATIC (decl) = 1;
452 : 1092 : DECL_IGNORED_P (decl) = 1;
453 : 1092 : if (!com->is_bind_c)
454 : 2053 : SET_DECL_ALIGN (decl, BIGGEST_ALIGNMENT);
455 : : else
456 : : {
457 : : /* Do not set the alignment for bind(c) common blocks to
458 : : BIGGEST_ALIGNMENT because that won't match what C does. Also,
459 : : for common blocks with one element, the alignment must be
460 : : that of the field within the common block in order to match
461 : : what C will do. */
462 : 59 : tree field = NULL_TREE;
463 : 59 : field = TYPE_FIELDS (TREE_TYPE (decl));
464 : 59 : if (DECL_CHAIN (field) == NULL_TREE)
465 : 23 : SET_DECL_ALIGN (decl, TYPE_ALIGN (TREE_TYPE (field)));
466 : : }
467 : 1092 : DECL_USER_ALIGN (decl) = 0;
468 : 1092 : GFC_DECL_COMMON_OR_EQUIV (decl) = 1;
469 : :
470 : 1092 : gfc_set_decl_location (decl, &com->where);
471 : :
472 : 1092 : if (com->omp_device_type != OMP_DEVICE_TYPE_UNSET)
473 : : {
474 : 15 : tree c = build_omp_clause (UNKNOWN_LOCATION, OMP_CLAUSE_DEVICE_TYPE);
475 : 15 : switch (com->omp_device_type)
476 : : {
477 : 4 : case OMP_DEVICE_TYPE_HOST:
478 : 4 : OMP_CLAUSE_DEVICE_TYPE_KIND (c) = OMP_CLAUSE_DEVICE_TYPE_HOST;
479 : 4 : break;
480 : 2 : case OMP_DEVICE_TYPE_NOHOST:
481 : 2 : OMP_CLAUSE_DEVICE_TYPE_KIND (c) = OMP_CLAUSE_DEVICE_TYPE_NOHOST;
482 : 2 : break;
483 : 9 : case OMP_DEVICE_TYPE_ANY:
484 : 9 : OMP_CLAUSE_DEVICE_TYPE_KIND (c) = OMP_CLAUSE_DEVICE_TYPE_ANY;
485 : 9 : break;
486 : 0 : default:
487 : 0 : gcc_unreachable ();
488 : : }
489 : : omp_clauses = c;
490 : : }
491 : : /* Also check trans-decl.cc when updating/removing the following;
492 : : also update f95.c's gfc_gnu_attributes.
493 : : For the warning, see also OpenMP spec issue 4663. */
494 : 1092 : if (com->omp_groupprivate && com->threadprivate)
495 : : {
496 : : /* Unset this flag; implicit 'declare target local(...)' remains. */
497 : 1 : com->omp_groupprivate = 0;
498 : 1 : gfc_warning (OPT_Wopenmp,
499 : : "Ignoring the %<groupprivate%> attribute for "
500 : : "%<threadprivate%> common block %</%s/%> declared at %L",
501 : 1 : com->name, &com->where);
502 : : }
503 : 1092 : if (com->omp_groupprivate)
504 : 5 : gfc_error ("Sorry, OMP GROUPPRIVATE not implemented, used by common "
505 : 5 : "block %</%s/%> declared at %L", com->name, &com->where);
506 : 1087 : else if (com->omp_declare_target_local)
507 : : /* Use 'else if' as groupprivate implies 'local'. */
508 : 1 : gfc_error ("Sorry, OMP DECLARE TARGET with LOCAL clause not implemented"
509 : : ", used by common block %</%s/%> declared at %L",
510 : 1 : com->name, &com->where);
511 : :
512 : 1092 : if (com->omp_declare_target_link)
513 : 3 : DECL_ATTRIBUTES (decl)
514 : 6 : = tree_cons (get_identifier ("omp declare target link"),
515 : 3 : omp_clauses, DECL_ATTRIBUTES (decl));
516 : 1089 : else if (com->omp_declare_target)
517 : 6 : DECL_ATTRIBUTES (decl)
518 : 12 : = tree_cons (get_identifier ("omp declare target"),
519 : 6 : omp_clauses, DECL_ATTRIBUTES (decl));
520 : :
521 : 1092 : if (com->omp_declare_target_link || com->omp_declare_target
522 : : /* FIXME: || com->omp_declare_target_local */)
523 : : {
524 : : /* Add to offload_vars; get_create does so for omp_declare_target
525 : : and omp_declare_target_local, omp_declare_target_link requires
526 : : manual work. */
527 : 9 : gcc_assert (symtab_node::get (decl) == 0);
528 : 9 : symtab_node *node = symtab_node::get_create (decl);
529 : 9 : if (node != NULL && com->omp_declare_target_link)
530 : : {
531 : 3 : node->offloadable = 1;
532 : 3 : if (ENABLE_OFFLOADING)
533 : : {
534 : : g->have_offload = true;
535 : : if (is_a <varpool_node *> (node))
536 : : vec_safe_push (offload_vars, decl);
537 : : }
538 : : }
539 : : }
540 : :
541 : : /* Place the back end declaration for this common block in
542 : : GLOBAL_BINDING_LEVEL. */
543 : 1092 : gfc_map_of_all_commons[identifier] = pushdecl_top_level (decl);
544 : : }
545 : :
546 : : /* Has no initial values. */
547 : 1110 : if (!is_init)
548 : : {
549 : 1008 : DECL_INITIAL (decl) = NULL_TREE;
550 : 1008 : DECL_COMMON (decl) = 1;
551 : 1008 : DECL_DEFER_OUTPUT (decl) = 1;
552 : : }
553 : : else
554 : : {
555 : 102 : DECL_INITIAL (decl) = error_mark_node;
556 : 102 : DECL_COMMON (decl) = 0;
557 : 102 : DECL_DEFER_OUTPUT (decl) = 0;
558 : : }
559 : :
560 : 1110 : if (com->threadprivate)
561 : 44 : set_decl_tls_model (decl, decl_default_tls_model (decl));
562 : :
563 : : return decl;
564 : : }
565 : :
566 : :
567 : : /* Return a field that is the size of the union, if an equivalence has
568 : : overlapping initializers. Merge the initializers into a single
569 : : initializer for this new field, then free the old ones. */
570 : :
571 : : static tree
572 : 941 : get_init_field (segment_info *head, tree union_type, tree *field_init,
573 : : record_layout_info rli)
574 : : {
575 : 941 : segment_info *s;
576 : 941 : HOST_WIDE_INT length = 0;
577 : 941 : HOST_WIDE_INT offset = 0;
578 : 941 : unsigned HOST_WIDE_INT known_align, desired_align;
579 : 941 : bool overlap = false;
580 : 941 : tree tmp, field;
581 : 941 : tree init;
582 : 941 : unsigned char *data, *chk;
583 : 941 : vec<constructor_elt, va_gc> *v = NULL;
584 : :
585 : 941 : tree type = unsigned_char_type_node;
586 : 941 : int i;
587 : :
588 : : /* Obtain the size of the union and check if there are any overlapping
589 : : initializers. */
590 : 4070 : for (s = head; s; s = s->next)
591 : : {
592 : 3129 : HOST_WIDE_INT slen = s->offset + s->length;
593 : 3129 : if (s->sym->value)
594 : : {
595 : 229 : if (s->offset < offset)
596 : 50 : overlap = true;
597 : : offset = slen;
598 : : }
599 : 3129 : length = length < slen ? slen : length;
600 : : }
601 : :
602 : 941 : if (!overlap)
603 : : return NULL_TREE;
604 : :
605 : : /* Now absorb all the initializer data into a single vector,
606 : : whilst checking for overlapping, unequal values. */
607 : 46 : data = XCNEWVEC (unsigned char, (size_t)length);
608 : 46 : chk = XCNEWVEC (unsigned char, (size_t)length);
609 : :
610 : : /* TODO - change this when default initialization is implemented. */
611 : 46 : memset (data, '\0', (size_t)length);
612 : 46 : memset (chk, '\0', (size_t)length);
613 : 156 : for (s = head; s; s = s->next)
614 : 110 : if (s->sym->value)
615 : : {
616 : 96 : locus *loc = NULL;
617 : 96 : if (s->sym->ns->equiv && s->sym->ns->equiv->eq)
618 : 96 : loc = &s->sym->ns->equiv->eq->expr->where;
619 : 96 : gfc_merge_initializers (s->sym->ts, s->sym->value, loc,
620 : : &data[s->offset],
621 : 96 : &chk[s->offset],
622 : 96 : (size_t)s->length);
623 : : }
624 : :
625 : 1142 : for (i = 0; i < length; i++)
626 : 1096 : CONSTRUCTOR_APPEND_ELT (v, NULL, build_int_cst (type, data[i]));
627 : :
628 : 46 : free (data);
629 : 46 : free (chk);
630 : :
631 : : /* Build a char[length] array to hold the initializers. Much of what
632 : : follows is borrowed from build_field, above. */
633 : :
634 : 46 : tmp = build_int_cst (gfc_array_index_type, length - 1);
635 : 46 : tmp = build_range_type (gfc_array_index_type,
636 : : gfc_index_zero_node, tmp);
637 : 46 : tmp = build_array_type (type, tmp);
638 : 46 : field = build_decl (input_location, FIELD_DECL, NULL_TREE, tmp);
639 : :
640 : 46 : known_align = BIGGEST_ALIGNMENT;
641 : :
642 : 46 : desired_align = update_alignment_for_field (rli, field, known_align);
643 : 46 : if (desired_align > known_align)
644 : 0 : DECL_PACKED (field) = 1;
645 : :
646 : 46 : DECL_FIELD_CONTEXT (field) = union_type;
647 : 46 : DECL_FIELD_OFFSET (field) = size_int (0);
648 : 46 : DECL_FIELD_BIT_OFFSET (field) = bitsize_zero_node;
649 : 46 : SET_DECL_OFFSET_ALIGN (field, known_align);
650 : :
651 : 46 : rli->offset = size_binop (MAX_EXPR, rli->offset,
652 : : size_binop (PLUS_EXPR,
653 : : DECL_FIELD_OFFSET (field),
654 : : DECL_SIZE_UNIT (field)));
655 : :
656 : 46 : init = build_constructor (TREE_TYPE (field), v);
657 : 46 : TREE_CONSTANT (init) = 1;
658 : :
659 : 46 : *field_init = init;
660 : :
661 : 156 : for (s = head; s; s = s->next)
662 : : {
663 : 110 : if (s->sym->value == NULL)
664 : 14 : continue;
665 : :
666 : 96 : gfc_free_expr (s->sym->value);
667 : 96 : s->sym->value = NULL;
668 : : }
669 : :
670 : : return field;
671 : : }
672 : :
673 : :
674 : : /* Declare memory for the common block or local equivalence, and create
675 : : backend declarations for all of the elements. */
676 : :
677 : : static void
678 : 2749 : create_common (gfc_common_head *com, segment_info *head, bool saw_equiv)
679 : : {
680 : 2749 : segment_info *s, *next_s;
681 : 2749 : tree union_type;
682 : 2749 : tree *field_link;
683 : 2749 : tree field;
684 : 2749 : tree field_init = NULL_TREE;
685 : 2749 : record_layout_info rli;
686 : 2749 : tree decl;
687 : 2749 : bool is_init = false;
688 : 2749 : bool is_saved = false;
689 : 2749 : bool is_auto = false;
690 : :
691 : : /* Declare the variables inside the common block.
692 : : If the current common block contains any equivalence object, then
693 : : make a UNION_TYPE node, otherwise RECORD_TYPE. This will let the
694 : : alias analyzer work well when there is no address overlapping for
695 : : common variables in the current common block. */
696 : 2749 : if (saw_equiv)
697 : 941 : union_type = make_node (UNION_TYPE);
698 : : else
699 : 1808 : union_type = make_node (RECORD_TYPE);
700 : :
701 : 2749 : rli = start_record_layout (union_type);
702 : 2749 : field_link = &TYPE_FIELDS (union_type);
703 : :
704 : : /* Check for overlapping initializers and replace them with a single,
705 : : artificial field that contains all the data. */
706 : 2749 : if (saw_equiv)
707 : 941 : field = get_init_field (head, union_type, &field_init, rli);
708 : : else
709 : : field = NULL_TREE;
710 : :
711 : 941 : if (field != NULL_TREE)
712 : : {
713 : 46 : is_init = true;
714 : 46 : *field_link = field;
715 : 46 : field_link = &DECL_CHAIN (field);
716 : : }
717 : :
718 : 10864 : for (s = head; s; s = s->next)
719 : : {
720 : 8115 : build_field (s, union_type, rli);
721 : :
722 : : /* Link the field into the type. */
723 : 8115 : *field_link = s->field;
724 : 8115 : field_link = &DECL_CHAIN (s->field);
725 : :
726 : : /* Has initial value. */
727 : 8115 : if (s->sym->value)
728 : 254 : is_init = true;
729 : :
730 : : /* Has SAVE attribute. */
731 : 8115 : if (s->sym->attr.save)
732 : 582 : is_saved = true;
733 : :
734 : : /* Has AUTOMATIC attribute. */
735 : 8115 : if (s->sym->attr.automatic)
736 : 14 : is_auto = true;
737 : : }
738 : :
739 : 2749 : finish_record_layout (rli, true);
740 : :
741 : 2749 : if (com)
742 : 2061 : decl = build_common_decl (com, union_type, is_init);
743 : : else
744 : 688 : decl = build_equiv_decl (union_type, is_init, is_saved, is_auto);
745 : :
746 : 2749 : if (is_init)
747 : : {
748 : 243 : tree ctor, tmp;
749 : 243 : vec<constructor_elt, va_gc> *v = NULL;
750 : :
751 : 243 : if (field != NULL_TREE && field_init != NULL_TREE)
752 : 46 : CONSTRUCTOR_APPEND_ELT (v, field, field_init);
753 : : else
754 : 605 : for (s = head; s; s = s->next)
755 : : {
756 : 408 : if (s->sym->value)
757 : : {
758 : : /* Add the initializer for this field. */
759 : 254 : tmp = gfc_conv_initializer (s->sym->value, &s->sym->ts,
760 : 254 : TREE_TYPE (s->field),
761 : : s->sym->attr.dimension,
762 : 254 : s->sym->attr.pointer
763 : 254 : || s->sym->attr.allocatable, false);
764 : :
765 : 254 : CONSTRUCTOR_APPEND_ELT (v, s->field, tmp);
766 : : }
767 : : }
768 : :
769 : 243 : gcc_assert (!v->is_empty ());
770 : 243 : ctor = build_constructor (union_type, v);
771 : 243 : TREE_CONSTANT (ctor) = 1;
772 : 243 : TREE_STATIC (ctor) = 1;
773 : 243 : DECL_INITIAL (decl) = ctor;
774 : :
775 : 243 : if (flag_checking)
776 : : {
777 : : tree field, value;
778 : : unsigned HOST_WIDE_INT idx;
779 : 543 : FOR_EACH_CONSTRUCTOR_ELT (CONSTRUCTOR_ELTS (ctor), idx, field, value)
780 : 300 : gcc_assert (TREE_CODE (field) == FIELD_DECL);
781 : : }
782 : : }
783 : :
784 : : /* Build component reference for each variable. */
785 : 10864 : for (s = head; s; s = next_s)
786 : : {
787 : 8115 : tree var_decl;
788 : :
789 : 8115 : var_decl = build_decl (gfc_get_location (&s->sym->declared_at),
790 : 8115 : VAR_DECL, DECL_NAME (s->field),
791 : 8115 : TREE_TYPE (s->field));
792 : 8115 : TREE_STATIC (var_decl) = TREE_STATIC (decl);
793 : : /* Mark the variable as used in order to avoid warnings about
794 : : unused variables. */
795 : 8115 : TREE_USED (var_decl) = 1;
796 : 8115 : if (s->sym->attr.use_assoc)
797 : 436 : DECL_IGNORED_P (var_decl) = 1;
798 : 8115 : if (s->sym->attr.target)
799 : 66 : TREE_ADDRESSABLE (var_decl) = 1;
800 : : /* Fake variables are not visible from other translation units. */
801 : 8115 : TREE_PUBLIC (var_decl) = 0;
802 : 8115 : gfc_finish_decl_attrs (var_decl, &s->sym->attr);
803 : :
804 : : /* To preserve identifier names in COMMON, chain to procedure
805 : : scope unless at top level in a module definition. */
806 : 8115 : if (com
807 : 6355 : && s->sym->ns->proc_name
808 : 6284 : && s->sym->ns->proc_name->attr.flavor == FL_MODULE)
809 : 491 : var_decl = pushdecl_top_level (var_decl);
810 : : else
811 : 7624 : gfc_add_decl_to_function (var_decl);
812 : :
813 : 8115 : tree comp = build3_loc (input_location, COMPONENT_REF,
814 : 8115 : TREE_TYPE (s->field), decl, s->field, NULL_TREE);
815 : 8115 : if (TREE_THIS_VOLATILE (s->field))
816 : 3 : TREE_THIS_VOLATILE (comp) = 1;
817 : 8115 : SET_DECL_VALUE_EXPR (var_decl, comp);
818 : 8115 : DECL_HAS_VALUE_EXPR_P (var_decl) = 1;
819 : 8115 : GFC_DECL_COMMON_OR_EQUIV (var_decl) = 1;
820 : :
821 : 8115 : if (s->sym->attr.assign)
822 : : {
823 : 14 : gfc_allocate_lang_decl (var_decl);
824 : 14 : GFC_DECL_ASSIGN (var_decl) = 1;
825 : 14 : GFC_DECL_STRING_LEN (var_decl) = GFC_DECL_STRING_LEN (s->field);
826 : 14 : GFC_DECL_ASSIGN_ADDR (var_decl) = GFC_DECL_ASSIGN_ADDR (s->field);
827 : : }
828 : :
829 : 8115 : s->sym->backend_decl = var_decl;
830 : :
831 : 8115 : next_s = s->next;
832 : 8115 : free (s);
833 : : }
834 : 2749 : }
835 : :
836 : :
837 : : /* Given a symbol, find it in the current segment list. Returns NULL if
838 : : not found. */
839 : :
840 : : static segment_info *
841 : 7331 : find_segment_info (gfc_symbol *symbol)
842 : : {
843 : 7331 : segment_info *n;
844 : :
845 : 43385 : for (n = current_segment; n; n = n->next)
846 : : {
847 : 36063 : if (n->sym == symbol)
848 : : return n;
849 : : }
850 : :
851 : : return NULL;
852 : : }
853 : :
854 : :
855 : : /* Given an expression node, make sure it is a constant integer and return
856 : : the mpz_t value. */
857 : :
858 : : static mpz_t *
859 : 6084 : get_mpz (gfc_expr *e)
860 : : {
861 : :
862 : 0 : if (e->expr_type != EXPR_CONSTANT)
863 : 0 : gfc_internal_error ("get_mpz(): Not an integer constant");
864 : :
865 : 6084 : return &e->value.integer;
866 : : }
867 : :
868 : :
869 : : /* Given an array specification and an array reference, figure out the
870 : : array element number (zero based). Bounds and elements are guaranteed
871 : : to be constants. If something goes wrong we generate an error and
872 : : return zero. */
873 : :
874 : : static HOST_WIDE_INT
875 : 1427 : element_number (gfc_array_ref *ar)
876 : : {
877 : 1427 : mpz_t multiplier, offset, extent, n;
878 : 1427 : gfc_array_spec *as;
879 : 1427 : HOST_WIDE_INT i, rank;
880 : :
881 : 1427 : as = ar->as;
882 : 1427 : rank = as->rank;
883 : 1427 : mpz_init_set_ui (multiplier, 1);
884 : 1427 : mpz_init_set_ui (offset, 0);
885 : 1427 : mpz_init (extent);
886 : 1427 : mpz_init (n);
887 : :
888 : 4291 : for (i = 0; i < rank; i++)
889 : : {
890 : 1437 : if (ar->dimen_type[i] != DIMEN_ELEMENT)
891 : 0 : gfc_internal_error ("element_number(): Bad dimension type");
892 : :
893 : 1437 : if (as && as->lower[i])
894 : 1436 : mpz_sub (n, *get_mpz (ar->start[i]), *get_mpz (as->lower[i]));
895 : : else
896 : 1 : mpz_sub_ui (n, *get_mpz (ar->start[i]), 1);
897 : :
898 : 1437 : mpz_mul (n, n, multiplier);
899 : 1437 : mpz_add (offset, offset, n);
900 : :
901 : 1437 : if (as && as->upper[i] && as->lower[i])
902 : : {
903 : 1436 : mpz_sub (extent, *get_mpz (as->upper[i]), *get_mpz (as->lower[i]));
904 : 1436 : mpz_add_ui (extent, extent, 1);
905 : : }
906 : : else
907 : 1 : mpz_set_ui (extent, 0);
908 : :
909 : 1437 : if (mpz_sgn (extent) < 0)
910 : 0 : mpz_set_ui (extent, 0);
911 : :
912 : 1437 : mpz_mul (multiplier, multiplier, extent);
913 : : }
914 : :
915 : 1427 : i = mpz_get_ui (offset);
916 : :
917 : 1427 : mpz_clear (multiplier);
918 : 1427 : mpz_clear (offset);
919 : 1427 : mpz_clear (extent);
920 : 1427 : mpz_clear (n);
921 : :
922 : 1427 : return i;
923 : : }
924 : :
925 : :
926 : : /* Given a single element of an equivalence list, figure out the offset
927 : : from the base symbol. For simple variables or full arrays, this is
928 : : simply zero. For an array element we have to calculate the array
929 : : element number and multiply by the element size. For a substring we
930 : : have to calculate the further reference. */
931 : :
932 : : static HOST_WIDE_INT
933 : 3126 : calculate_offset (gfc_expr *e)
934 : : {
935 : 3126 : HOST_WIDE_INT n, element_size, offset;
936 : 3126 : gfc_typespec *element_type;
937 : 3126 : gfc_ref *reference;
938 : :
939 : 3126 : offset = 0;
940 : 3126 : element_type = &e->symtree->n.sym->ts;
941 : :
942 : 5320 : for (reference = e->ref; reference; reference = reference->next)
943 : 2194 : switch (reference->type)
944 : : {
945 : 1855 : case REF_ARRAY:
946 : 1855 : switch (reference->u.ar.type)
947 : : {
948 : : case AR_FULL:
949 : : break;
950 : :
951 : 1427 : case AR_ELEMENT:
952 : 1427 : n = element_number (&reference->u.ar);
953 : 1427 : if (element_type->type == BT_CHARACTER)
954 : 221 : gfc_conv_const_charlen (element_type->u.cl);
955 : 1427 : element_size =
956 : 1427 : int_size_in_bytes (gfc_typenode_for_spec (element_type));
957 : 1427 : offset += n * element_size;
958 : 1427 : break;
959 : :
960 : 0 : default:
961 : 0 : gfc_error ("Bad array reference at %L", &e->where);
962 : : }
963 : : break;
964 : 339 : case REF_SUBSTRING:
965 : 339 : if (reference->u.ss.start != NULL)
966 : 339 : offset += mpz_get_ui (*get_mpz (reference->u.ss.start)) - 1;
967 : : break;
968 : 0 : default:
969 : 0 : gfc_error ("Illegal reference type at %L as EQUIVALENCE object",
970 : : &e->where);
971 : : }
972 : 3126 : return offset;
973 : : }
974 : :
975 : :
976 : : /* Add a new segment_info structure to the current segment. eq1 is already
977 : : in the list, eq2 is not. */
978 : :
979 : : static void
980 : 1561 : new_condition (segment_info *v, gfc_equiv *eq1, gfc_equiv *eq2)
981 : : {
982 : 1561 : HOST_WIDE_INT offset1, offset2;
983 : 1561 : segment_info *a;
984 : :
985 : 1561 : offset1 = calculate_offset (eq1->expr);
986 : 1561 : offset2 = calculate_offset (eq2->expr);
987 : :
988 : 3122 : a = get_segment_info (eq2->expr->symtree->n.sym,
989 : 1561 : v->offset + offset1 - offset2);
990 : :
991 : 1561 : current_segment = add_segments (current_segment, a);
992 : 1561 : }
993 : :
994 : :
995 : : /* Given two equivalence structures that are both already in the list, make
996 : : sure that this new condition is not violated, generating an error if it
997 : : is. */
998 : :
999 : : static void
1000 : 2 : confirm_condition (segment_info *s1, gfc_equiv *eq1, segment_info *s2,
1001 : : gfc_equiv *eq2)
1002 : : {
1003 : 2 : HOST_WIDE_INT offset1, offset2;
1004 : :
1005 : 2 : offset1 = calculate_offset (eq1->expr);
1006 : 2 : offset2 = calculate_offset (eq2->expr);
1007 : :
1008 : 2 : if (s1->offset + offset1 != s2->offset + offset2)
1009 : 2 : gfc_error ("Inconsistent equivalence rules involving %qs at %L and "
1010 : 2 : "%qs at %L", s1->sym->name, &s1->sym->declared_at,
1011 : 2 : s2->sym->name, &s2->sym->declared_at);
1012 : 2 : }
1013 : :
1014 : :
1015 : : /* Process a new equivalence condition. eq1 is know to be in segment f.
1016 : : If eq2 is also present then confirm that the condition holds.
1017 : : Otherwise add a new variable to the segment list. */
1018 : :
1019 : : static void
1020 : 1563 : add_condition (segment_info *f, gfc_equiv *eq1, gfc_equiv *eq2)
1021 : : {
1022 : 1563 : segment_info *n;
1023 : :
1024 : 1563 : n = find_segment_info (eq2->expr->symtree->n.sym);
1025 : :
1026 : 1563 : if (n == NULL)
1027 : 1561 : new_condition (f, eq1, eq2);
1028 : : else
1029 : 2 : confirm_condition (f, eq1, n, eq2);
1030 : 1563 : }
1031 : :
1032 : : static void
1033 : 44745 : accumulate_equivalence_attributes (symbol_attribute *dummy_symbol, gfc_equiv *e)
1034 : : {
1035 : 44745 : symbol_attribute attr = e->expr->symtree->n.sym->attr;
1036 : :
1037 : 44745 : dummy_symbol->dummy |= attr.dummy;
1038 : 44745 : dummy_symbol->pointer |= attr.pointer;
1039 : 44745 : dummy_symbol->target |= attr.target;
1040 : 44745 : dummy_symbol->external |= attr.external;
1041 : 44745 : dummy_symbol->intrinsic |= attr.intrinsic;
1042 : 44745 : dummy_symbol->allocatable |= attr.allocatable;
1043 : 44745 : dummy_symbol->elemental |= attr.elemental;
1044 : 44745 : dummy_symbol->recursive |= attr.recursive;
1045 : 44745 : dummy_symbol->in_common |= attr.in_common;
1046 : 44745 : dummy_symbol->result |= attr.result;
1047 : 44745 : dummy_symbol->in_namelist |= attr.in_namelist;
1048 : 44745 : dummy_symbol->optional |= attr.optional;
1049 : 44745 : dummy_symbol->entry |= attr.entry;
1050 : 44745 : dummy_symbol->function |= attr.function;
1051 : 44745 : dummy_symbol->subroutine |= attr.subroutine;
1052 : 44745 : dummy_symbol->dimension |= attr.dimension;
1053 : 44745 : dummy_symbol->in_equivalence |= attr.in_equivalence;
1054 : 44745 : dummy_symbol->use_assoc |= attr.use_assoc;
1055 : 44745 : dummy_symbol->cray_pointer |= attr.cray_pointer;
1056 : 44745 : dummy_symbol->cray_pointee |= attr.cray_pointee;
1057 : 44745 : dummy_symbol->data |= attr.data;
1058 : 44745 : dummy_symbol->value |= attr.value;
1059 : 44745 : dummy_symbol->volatile_ |= attr.volatile_;
1060 : 44745 : dummy_symbol->is_protected |= attr.is_protected;
1061 : 44745 : dummy_symbol->is_bind_c |= attr.is_bind_c;
1062 : 44745 : dummy_symbol->procedure |= attr.procedure;
1063 : 44745 : dummy_symbol->proc_pointer |= attr.proc_pointer;
1064 : 44745 : dummy_symbol->abstract |= attr.abstract;
1065 : 44745 : dummy_symbol->asynchronous |= attr.asynchronous;
1066 : 44745 : dummy_symbol->codimension |= attr.codimension;
1067 : 44745 : dummy_symbol->contiguous |= attr.contiguous;
1068 : 44745 : dummy_symbol->generic |= attr.generic;
1069 : 44745 : dummy_symbol->automatic |= attr.automatic;
1070 : 44745 : dummy_symbol->threadprivate |= attr.threadprivate;
1071 : 44745 : dummy_symbol->omp_groupprivate |= attr.omp_groupprivate;
1072 : 44745 : dummy_symbol->omp_declare_target |= attr.omp_declare_target;
1073 : 44745 : dummy_symbol->omp_declare_target_link |= attr.omp_declare_target_link;
1074 : 44745 : dummy_symbol->omp_declare_target_local |= attr.omp_declare_target_local;
1075 : 44745 : dummy_symbol->oacc_declare_copyin |= attr.oacc_declare_copyin;
1076 : 44745 : dummy_symbol->oacc_declare_create |= attr.oacc_declare_create;
1077 : 44745 : dummy_symbol->oacc_declare_deviceptr |= attr.oacc_declare_deviceptr;
1078 : 44745 : dummy_symbol->oacc_declare_device_resident
1079 : 44745 : |= attr.oacc_declare_device_resident;
1080 : :
1081 : : /* Not strictly correct, but probably close enough. */
1082 : 44745 : if (attr.save > dummy_symbol->save)
1083 : 691 : dummy_symbol->save = attr.save;
1084 : 44745 : if (attr.access > dummy_symbol->access)
1085 : 4 : dummy_symbol->access = attr.access;
1086 : 44745 : }
1087 : :
1088 : : /* Given a segment element, search through the equivalence lists for unused
1089 : : conditions that involve the symbol. Add these rules to the segment. */
1090 : :
1091 : : static bool
1092 : 8103 : find_equivalence (segment_info *n)
1093 : : {
1094 : 8103 : gfc_equiv *e1, *e2, *eq;
1095 : 8103 : bool found;
1096 : :
1097 : 8103 : found = false;
1098 : :
1099 : 31010 : for (e1 = n->sym->ns->equiv; e1; e1 = e1->next)
1100 : : {
1101 : 22907 : eq = NULL;
1102 : :
1103 : : /* Search the equivalence list, including the root (first) element
1104 : : for the symbol that owns the segment. */
1105 : 22907 : symbol_attribute dummy_symbol;
1106 : 22907 : memset (&dummy_symbol, 0, sizeof (dummy_symbol));
1107 : 66136 : for (e2 = e1; e2; e2 = e2->eq)
1108 : : {
1109 : 44745 : accumulate_equivalence_attributes (&dummy_symbol, e2);
1110 : 44745 : if (!e2->used && e2->expr->symtree->n.sym == n->sym)
1111 : : {
1112 : : eq = e2;
1113 : : break;
1114 : : }
1115 : : }
1116 : :
1117 : 22907 : gfc_check_conflict (&dummy_symbol, e1->expr->symtree->name, &e1->expr->where);
1118 : :
1119 : : /* Go to the next root element. */
1120 : 22907 : if (eq == NULL)
1121 : 21391 : continue;
1122 : :
1123 : 1516 : eq->used = 1;
1124 : :
1125 : : /* Now traverse the equivalence list matching the offsets. */
1126 : 4595 : for (e2 = e1; e2; e2 = e2->eq)
1127 : : {
1128 : 3079 : if (!e2->used && e2 != eq)
1129 : : {
1130 : 1563 : add_condition (n, eq, e2);
1131 : 1563 : e2->used = 1;
1132 : 1563 : found = true;
1133 : : }
1134 : : }
1135 : : }
1136 : 8103 : return found;
1137 : : }
1138 : :
1139 : :
1140 : : /* Add all symbols equivalenced within a segment. We need to scan the
1141 : : segment list multiple times to include indirect equivalences. Since
1142 : : a new segment_info can inserted at the beginning of the segment list,
1143 : : depending on its offset, we have to force a final pass through the
1144 : : loop by demanding that completion sees a pass with no matches; i.e.,
1145 : : all symbols with equiv_built set and no new equivalences found. */
1146 : :
1147 : : static void
1148 : 6554 : add_equivalences (bool *saw_equiv)
1149 : : {
1150 : 6554 : segment_info *f;
1151 : 6554 : bool more = true;
1152 : :
1153 : 14264 : while (more)
1154 : : {
1155 : 7710 : more = false;
1156 : 17699 : for (f = current_segment; f; f = f->next)
1157 : : {
1158 : 9989 : if (!f->sym->equiv_built)
1159 : : {
1160 : 8103 : f->sym->equiv_built = 1;
1161 : 8103 : bool seen_one = find_equivalence (f);
1162 : 8103 : if (seen_one)
1163 : : {
1164 : 1163 : *saw_equiv = true;
1165 : 1163 : more = true;
1166 : : }
1167 : : }
1168 : : }
1169 : : }
1170 : :
1171 : : /* Add a copy of this segment list to the namespace. */
1172 : 6554 : copy_equiv_list_to_ns (current_segment);
1173 : 6554 : }
1174 : :
1175 : :
1176 : : /* Returns the offset necessary to properly align the current equivalence.
1177 : : Sets *palign to the required alignment. */
1178 : :
1179 : : static HOST_WIDE_INT
1180 : 6536 : align_segment (unsigned HOST_WIDE_INT *palign)
1181 : : {
1182 : 6536 : segment_info *s;
1183 : 6536 : unsigned HOST_WIDE_INT offset;
1184 : 6536 : unsigned HOST_WIDE_INT max_align;
1185 : 6536 : unsigned HOST_WIDE_INT this_align;
1186 : 6536 : unsigned HOST_WIDE_INT this_offset;
1187 : :
1188 : 6536 : max_align = 1;
1189 : 6536 : offset = 0;
1190 : 14633 : for (s = current_segment; s; s = s->next)
1191 : : {
1192 : 8097 : this_align = TYPE_ALIGN_UNIT (s->field);
1193 : 8097 : if (s->offset & (this_align - 1))
1194 : : {
1195 : : /* Field is misaligned. */
1196 : 128 : this_offset = this_align - ((s->offset + offset) & (this_align - 1));
1197 : 128 : if (this_offset & (max_align - 1))
1198 : : {
1199 : : /* Aligning this field would misalign a previous field. */
1200 : 0 : gfc_error ("The equivalence set for variable %qs "
1201 : : "declared at %L violates alignment requirements",
1202 : 0 : s->sym->name, &s->sym->declared_at);
1203 : : }
1204 : 128 : offset += this_offset;
1205 : : }
1206 : 8097 : max_align = this_align;
1207 : : }
1208 : 6536 : if (palign)
1209 : 6536 : *palign = max_align;
1210 : 6536 : return offset;
1211 : : }
1212 : :
1213 : :
1214 : : /* Adjust segment offsets by the given amount. */
1215 : :
1216 : : static void
1217 : 6554 : apply_segment_offset (segment_info *s, HOST_WIDE_INT offset)
1218 : : {
1219 : 14669 : for (; s; s = s->next)
1220 : 8115 : s->offset += offset;
1221 : 0 : }
1222 : :
1223 : :
1224 : : /* Lay out a symbol in a common block. If the symbol has already been seen
1225 : : then check the location is consistent. Otherwise create segments
1226 : : for that symbol and all the symbols equivalenced with it. */
1227 : :
1228 : : /* Translate a single common block. */
1229 : :
1230 : : static void
1231 : 1957 : translate_common (gfc_common_head *common, gfc_symbol *var_list)
1232 : : {
1233 : 1957 : gfc_symbol *sym;
1234 : 1957 : segment_info *s;
1235 : 1957 : segment_info *common_segment;
1236 : 1957 : HOST_WIDE_INT offset;
1237 : 1957 : HOST_WIDE_INT current_offset;
1238 : 1957 : unsigned HOST_WIDE_INT align;
1239 : 1957 : bool saw_equiv;
1240 : :
1241 : 1957 : common_segment = NULL;
1242 : 1957 : offset = 0;
1243 : 1957 : current_offset = 0;
1244 : 1957 : align = 1;
1245 : 1957 : saw_equiv = false;
1246 : :
1247 : 1957 : if (var_list && var_list->attr.omp_allocate)
1248 : 6 : gfc_error ("Sorry, !$OMP allocate for COMMON block variable %qs at %L "
1249 : 6 : "not supported", common->name, &common->where);
1250 : :
1251 : : /* Add symbols to the segment. */
1252 : 7725 : for (sym = var_list; sym; sym = sym->common_next)
1253 : : {
1254 : 5768 : current_segment = common_segment;
1255 : 5768 : s = find_segment_info (sym);
1256 : :
1257 : : /* Symbol has already been added via an equivalence. Multiple
1258 : : use associations of the same common block result in equiv_built
1259 : : being set but no information about the symbol in the segment. */
1260 : 5768 : if (s && sym->equiv_built)
1261 : : {
1262 : : /* Ensure the current location is properly aligned. */
1263 : 7 : align = TYPE_ALIGN_UNIT (s->field);
1264 : 7 : current_offset = (current_offset + align - 1) &~ (align - 1);
1265 : :
1266 : : /* Verify that it ended up where we expect it. */
1267 : 7 : if (s->offset != current_offset)
1268 : : {
1269 : 1 : gfc_error ("Equivalence for %qs does not match ordering of "
1270 : : "COMMON %qs at %L", sym->name,
1271 : 1 : common->name, &common->where);
1272 : : }
1273 : : }
1274 : : else
1275 : : {
1276 : : /* A symbol we haven't seen before. */
1277 : 5761 : s = current_segment = get_segment_info (sym, current_offset);
1278 : :
1279 : : /* Add all objects directly or indirectly equivalenced with this
1280 : : symbol. */
1281 : 5761 : add_equivalences (&saw_equiv);
1282 : :
1283 : 5761 : if (current_segment->offset < 0)
1284 : 0 : gfc_error ("The equivalence set for %qs cause an invalid "
1285 : : "extension to COMMON %qs at %L", sym->name,
1286 : 0 : common->name, &common->where);
1287 : :
1288 : 5761 : if (flag_align_commons)
1289 : 5743 : offset = align_segment (&align);
1290 : :
1291 : 5761 : if (offset)
1292 : : {
1293 : : /* The required offset conflicts with previous alignment
1294 : : requirements. Insert padding immediately before this
1295 : : segment. */
1296 : 37 : if (warn_align_commons)
1297 : : {
1298 : 35 : if (strcmp (common->name, BLANK_COMMON_NAME))
1299 : 23 : gfc_warning (OPT_Walign_commons,
1300 : : "Padding of %d bytes required before %qs in "
1301 : : "COMMON %qs at %L; reorder elements or use "
1302 : : "%<-fno-align-commons%>", (int)offset,
1303 : 23 : s->sym->name, common->name, &common->where);
1304 : : else
1305 : 12 : gfc_warning (OPT_Walign_commons,
1306 : : "Padding of %d bytes required before %qs in "
1307 : : "COMMON at %L; reorder elements or use "
1308 : : "%<-fno-align-commons%>", (int)offset,
1309 : 12 : s->sym->name, &common->where);
1310 : : }
1311 : : }
1312 : :
1313 : : /* Apply the offset to the new segments. */
1314 : 5761 : apply_segment_offset (current_segment, offset);
1315 : 5761 : current_offset += offset;
1316 : :
1317 : : /* Add the new segments to the common block. */
1318 : 5761 : common_segment = add_segments (common_segment, current_segment);
1319 : : }
1320 : :
1321 : : /* The offset of the next common variable. */
1322 : 5768 : current_offset += s->length;
1323 : : }
1324 : :
1325 : 1957 : if (common_segment == NULL)
1326 : : {
1327 : 1 : gfc_error ("COMMON %qs at %L does not exist",
1328 : 1 : common->name, &common->where);
1329 : 1 : return;
1330 : : }
1331 : :
1332 : 1956 : if (common_segment->offset != 0 && warn_align_commons)
1333 : : {
1334 : 0 : if (strcmp (common->name, BLANK_COMMON_NAME))
1335 : 0 : gfc_warning (OPT_Walign_commons,
1336 : : "COMMON %qs at %L requires %d bytes of padding; "
1337 : : "reorder elements or use %<-fno-align-commons%>",
1338 : : common->name, &common->where, (int)common_segment->offset);
1339 : : else
1340 : 0 : gfc_warning (OPT_Walign_commons,
1341 : : "COMMON at %L requires %d bytes of padding; "
1342 : : "reorder elements or use %<-fno-align-commons%>",
1343 : : &common->where, (int)common_segment->offset);
1344 : : }
1345 : :
1346 : 1956 : create_common (common, common_segment, saw_equiv);
1347 : : }
1348 : :
1349 : :
1350 : : /* Create a new block for each merged equivalence list. */
1351 : :
1352 : : static void
1353 : 91558 : finish_equivalences (gfc_namespace *ns)
1354 : : {
1355 : 91558 : gfc_equiv *z, *y;
1356 : 91558 : gfc_symbol *sym;
1357 : 91558 : gfc_common_head * c;
1358 : 91558 : HOST_WIDE_INT offset;
1359 : 91558 : unsigned HOST_WIDE_INT align;
1360 : 91558 : bool dummy;
1361 : :
1362 : 93074 : for (z = ns->equiv; z; z = z->next)
1363 : 2269 : for (y = z->eq; y; y = y->eq)
1364 : : {
1365 : 1546 : if (y->used)
1366 : 753 : continue;
1367 : 793 : sym = z->expr->symtree->n.sym;
1368 : 793 : current_segment = get_segment_info (sym, 0);
1369 : :
1370 : : /* All objects directly or indirectly equivalenced with this
1371 : : symbol. */
1372 : 793 : add_equivalences (&dummy);
1373 : :
1374 : : /* Align the block. */
1375 : 793 : offset = align_segment (&align);
1376 : :
1377 : : /* Ensure all offsets are positive. */
1378 : 793 : offset -= current_segment->offset & ~(align - 1);
1379 : :
1380 : 793 : apply_segment_offset (current_segment, offset);
1381 : :
1382 : : /* Create the decl. If this is a module equivalence, it has a
1383 : : unique name, pointed to by z->module. This is written to a
1384 : : gfc_common_header to push create_common into using
1385 : : build_common_decl, so that the equivalence appears as an
1386 : : external symbol. Otherwise, a local declaration is built using
1387 : : build_equiv_decl. */
1388 : 793 : if (z->module)
1389 : : {
1390 : 105 : c = gfc_get_common_head ();
1391 : : /* We've lost the real location, so use the location of the
1392 : : enclosing procedure. If we're in a BLOCK DATA block, then
1393 : : use the location in the sym_root. */
1394 : 105 : if (ns->proc_name)
1395 : 104 : c->where = ns->proc_name->declared_at;
1396 : 1 : else if (ns->is_block_data)
1397 : 1 : c->where = ns->sym_root->n.sym->declared_at;
1398 : :
1399 : 105 : size_t len = strlen (z->module);
1400 : 105 : gcc_assert (len < sizeof (c->name));
1401 : 105 : memcpy (c->name, z->module, len);
1402 : 105 : c->name[len] = '\0';
1403 : : }
1404 : : else
1405 : : c = NULL;
1406 : :
1407 : 793 : create_common (c, current_segment, true);
1408 : 793 : break;
1409 : : }
1410 : 91558 : }
1411 : :
1412 : :
1413 : : /* Work function for translating a named common block. */
1414 : :
1415 : : static void
1416 : 1771 : named_common (gfc_symtree *st)
1417 : : {
1418 : 1771 : translate_common (st->n.common, st->n.common->head);
1419 : 1771 : }
1420 : :
1421 : :
1422 : : /* Translate the common blocks in a namespace. Unlike other variables,
1423 : : these have to be created before code, because the backend_decl depends
1424 : : on the rest of the common block. */
1425 : :
1426 : : void
1427 : 91558 : gfc_trans_common (gfc_namespace *ns)
1428 : : {
1429 : 91558 : gfc_common_head *c;
1430 : :
1431 : : /* Translate the blank common block. */
1432 : 91558 : if (ns->blank_common.head != NULL)
1433 : : {
1434 : 186 : c = gfc_get_common_head ();
1435 : 186 : c->where = ns->blank_common.head->common_head->where;
1436 : 186 : strcpy (c->name, BLANK_COMMON_NAME);
1437 : 186 : translate_common (c, ns->blank_common.head);
1438 : : }
1439 : :
1440 : : /* Translate all named common blocks. */
1441 : 91558 : gfc_traverse_symtree (ns->common_root, named_common);
1442 : :
1443 : : /* Translate local equivalence. */
1444 : 91558 : finish_equivalences (ns);
1445 : :
1446 : : /* Commit the newly created symbols for common blocks and module
1447 : : equivalences. */
1448 : 91558 : gfc_commit_symbols ();
1449 : 91558 : }
|