Line data Source code
1 : /* Handle modules, which amounts to loading and saving symbols and
2 : their attendant structures.
3 : Copyright (C) 2000-2026 Free Software Foundation, Inc.
4 : Contributed by Andy Vaught
5 :
6 : This file is part of GCC.
7 :
8 : GCC is free software; you can redistribute it and/or modify it under
9 : the terms of the GNU General Public License as published by the Free
10 : Software Foundation; either version 3, or (at your option) any later
11 : version.
12 :
13 : GCC is distributed in the hope that it will be useful, but WITHOUT ANY
14 : WARRANTY; without even the implied warranty of MERCHANTABILITY or
15 : FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
16 : for more details.
17 :
18 : You should have received a copy of the GNU General Public License
19 : along with GCC; see the file COPYING3. If not see
20 : <http://www.gnu.org/licenses/>. */
21 :
22 : /* The syntax of gfortran modules resembles that of lisp lists, i.e. a
23 : sequence of atoms, which can be left or right parenthesis, names,
24 : integers or strings. Parenthesis are always matched which allows
25 : us to skip over sections at high speed without having to know
26 : anything about the internal structure of the lists. A "name" is
27 : usually a fortran 95 identifier, but can also start with '@' in
28 : order to reference a hidden symbol.
29 :
30 : The first line of a module is an informational message about what
31 : created the module, the file it came from and when it was created.
32 : The second line is a warning for people not to edit the module.
33 : The rest of the module looks like:
34 :
35 : ( ( <Interface info for UPLUS> )
36 : ( <Interface info for UMINUS> )
37 : ...
38 : )
39 : ( ( <name of operator interface> <module of op interface> <i/f1> ... )
40 : ...
41 : )
42 : ( ( <name of generic interface> <module of generic interface> <i/f1> ... )
43 : ...
44 : )
45 : ( ( <common name> <symbol> <saved flag>)
46 : ...
47 : )
48 :
49 : ( equivalence list )
50 :
51 : ( <Symbol Number (in no particular order)>
52 : <True name of symbol>
53 : <Module name of symbol>
54 : ( <symbol information> )
55 : ...
56 : )
57 : ( <Symtree name>
58 : <Ambiguous flag>
59 : <Symbol number>
60 : ...
61 : )
62 :
63 : In general, symbols refer to other symbols by their symbol number,
64 : which are zero based. Symbols are written to the module in no
65 : particular order. */
66 :
67 : #include "config.h"
68 : #include "system.h"
69 : #include "coretypes.h"
70 : #include "options.h"
71 : #include "tree.h"
72 : #include "gfortran.h"
73 : #include "stringpool.h"
74 : #include "arith.h"
75 : #include "match.h"
76 : #include "parse.h" /* FIXME */
77 : #include "constructor.h"
78 : #include "cpp.h"
79 : #include "diagnostic-core.h"
80 : #include "scanner.h"
81 : #include <zlib.h>
82 :
83 : #define MODULE_EXTENSION ".mod"
84 : #define SUBMODULE_EXTENSION ".smod"
85 :
86 : /* Don't put any single quote (') in MOD_VERSION, if you want it to be
87 : recognized. */
88 : #define MOD_VERSION "16"
89 : #define MOD_VERSION_NUMERIC 16
90 : /* Older mod versions we can still parse. */
91 : #define COMPAT_MOD_VERSIONS { "15" }
92 :
93 :
94 : /* Structure that describes a position within a module file. */
95 :
96 : typedef struct
97 : {
98 : int column, line;
99 : long pos;
100 : }
101 : module_locus;
102 :
103 : /* Structure for list of symbols of intrinsic modules. */
104 : typedef struct
105 : {
106 : int id;
107 : const char *name;
108 : int value;
109 : int standard;
110 : }
111 : intmod_sym;
112 :
113 :
114 : typedef enum
115 : {
116 : P_UNKNOWN = 0, P_OTHER, P_NAMESPACE, P_COMPONENT, P_SYMBOL
117 : }
118 : pointer_t;
119 :
120 : /* The fixup structure lists pointers to pointers that have to
121 : be updated when a pointer value becomes known. */
122 :
123 : typedef struct fixup_t
124 : {
125 : void **pointer;
126 : struct fixup_t *next;
127 : }
128 : fixup_t;
129 :
130 :
131 : /* Structure for holding extra info needed for pointers being read. */
132 :
133 : enum gfc_rsym_state
134 : {
135 : UNUSED,
136 : NEEDED,
137 : USED
138 : };
139 :
140 : enum gfc_wsym_state
141 : {
142 : UNREFERENCED = 0,
143 : NEEDS_WRITE,
144 : WRITTEN
145 : };
146 :
147 : typedef struct pointer_info
148 : {
149 : BBT_HEADER (pointer_info);
150 : HOST_WIDE_INT integer;
151 : pointer_t type;
152 :
153 : /* The first component of each member of the union is the pointer
154 : being stored. */
155 :
156 : fixup_t *fixup;
157 :
158 : union
159 : {
160 : void *pointer; /* Member for doing pointer searches. */
161 :
162 : struct
163 : {
164 : gfc_symbol *sym;
165 : char *true_name, *module, *binding_label;
166 : fixup_t *stfixup;
167 : gfc_symtree *symtree;
168 : enum gfc_rsym_state state;
169 : int ns, referenced, renamed;
170 : module_locus where;
171 : }
172 : rsym;
173 :
174 : struct
175 : {
176 : gfc_symbol *sym;
177 : enum gfc_wsym_state state;
178 : }
179 : wsym;
180 : }
181 : u;
182 :
183 : }
184 : pointer_info;
185 :
186 : #define gfc_get_pointer_info() XCNEW (pointer_info)
187 :
188 :
189 : /* Local variables */
190 :
191 : /* The gzFile for the module we're reading or writing. */
192 : static gzFile module_fp;
193 :
194 : /* Fully qualified module path */
195 : static char *module_fullpath = NULL;
196 :
197 : /* The name of the module we're reading (USE'ing) or writing. */
198 : static const char *module_name;
199 : /* The name of the .smod file that the submodule will write to. */
200 : static const char *submodule_name;
201 :
202 : /* The list of use statements to apply to the current namespace
203 : before parsing the non-use statements. */
204 : static gfc_use_list *module_list;
205 : /* The end of the MODULE_LIST list above at the time the recognition
206 : of the current statement started. */
207 : static gfc_use_list **old_module_list_tail;
208 :
209 : /* If we're reading an intrinsic module, this is its ID. */
210 : static intmod_id current_intmod;
211 :
212 : /* Content of module. */
213 : static char* module_content;
214 :
215 : static long module_pos;
216 : static int module_line, module_column, only_flag;
217 : static int prev_module_line, prev_module_column;
218 :
219 : static enum
220 : { IO_INPUT, IO_OUTPUT }
221 : iomode;
222 :
223 : static gfc_use_rename *gfc_rename_list;
224 : static pointer_info *pi_root;
225 : static int symbol_number; /* Counter for assigning symbol numbers */
226 :
227 : /* Tells mio_expr_ref to make symbols for unused equivalence members. */
228 : static bool in_load_equiv;
229 :
230 :
231 :
232 : /*****************************************************************/
233 :
234 : /* Pointer/integer conversion. Pointers between structures are stored
235 : as integers in the module file. The next couple of subroutines
236 : handle this translation for reading and writing. */
237 :
238 : /* Recursively free the tree of pointer structures. */
239 :
240 : static void
241 4198895 : free_pi_tree (pointer_info *p)
242 : {
243 4198895 : if (p == NULL)
244 : return;
245 :
246 2087716 : if (p->fixup != NULL)
247 0 : gfc_internal_error ("free_pi_tree(): Unresolved fixup");
248 :
249 2087716 : free_pi_tree (p->left);
250 2087716 : free_pi_tree (p->right);
251 :
252 2087716 : if (iomode == IO_INPUT)
253 : {
254 1678558 : XDELETEVEC (p->u.rsym.true_name);
255 1678558 : XDELETEVEC (p->u.rsym.module);
256 1678558 : XDELETEVEC (p->u.rsym.binding_label);
257 : }
258 :
259 2087716 : free (p);
260 : }
261 :
262 :
263 : /* Compare pointers when searching by pointer. Used when writing a
264 : module. */
265 :
266 : static int
267 2372216 : compare_pointers (void *_sn1, void *_sn2)
268 : {
269 2372216 : pointer_info *sn1, *sn2;
270 :
271 2372216 : sn1 = (pointer_info *) _sn1;
272 2372216 : sn2 = (pointer_info *) _sn2;
273 :
274 2372216 : if (sn1->u.pointer < sn2->u.pointer)
275 : return -1;
276 1395596 : if (sn1->u.pointer > sn2->u.pointer)
277 1395596 : return 1;
278 :
279 : return 0;
280 : }
281 :
282 :
283 : /* Compare integers when searching by integer. Used when reading a
284 : module. */
285 :
286 : static int
287 78062554 : compare_integers (void *_sn1, void *_sn2)
288 : {
289 78062554 : pointer_info *sn1, *sn2;
290 :
291 78062554 : sn1 = (pointer_info *) _sn1;
292 78062554 : sn2 = (pointer_info *) _sn2;
293 :
294 12602959 : if (sn1->integer < sn2->integer)
295 : return -1;
296 35450135 : if (sn1->integer > sn2->integer)
297 8513885 : return 1;
298 :
299 : return 0;
300 : }
301 :
302 :
303 : /* Initialize the pointer_info tree. */
304 :
305 : static void
306 23463 : init_pi_tree (void)
307 : {
308 23463 : compare_fn compare;
309 23463 : pointer_info *p;
310 :
311 23463 : pi_root = NULL;
312 23463 : compare = (iomode == IO_INPUT) ? compare_integers : compare_pointers;
313 :
314 : /* Pointer 0 is the NULL pointer. */
315 23463 : p = gfc_get_pointer_info ();
316 23463 : p->u.pointer = NULL;
317 23463 : p->integer = 0;
318 23463 : p->type = P_OTHER;
319 :
320 23463 : gfc_insert_bbt (&pi_root, p, compare);
321 :
322 : /* Pointer 1 is the current namespace. */
323 23463 : p = gfc_get_pointer_info ();
324 23463 : p->u.pointer = gfc_current_ns;
325 23463 : p->integer = 1;
326 23463 : p->type = P_NAMESPACE;
327 :
328 23463 : gfc_insert_bbt (&pi_root, p, compare);
329 :
330 23463 : symbol_number = 2;
331 23463 : }
332 :
333 :
334 : /* During module writing, call here with a pointer to something,
335 : returning the pointer_info node. */
336 :
337 : static pointer_info *
338 1997305 : find_pointer (void *gp)
339 : {
340 1997305 : pointer_info *p;
341 :
342 1997305 : p = pi_root;
343 10587246 : while (p != NULL)
344 : {
345 10197872 : if (p->u.pointer == gp)
346 : break;
347 8589941 : p = (gp < p->u.pointer) ? p->left : p->right;
348 : }
349 :
350 1997305 : return p;
351 : }
352 :
353 :
354 : /* Given a pointer while writing, returns the pointer_info tree node,
355 : creating it if it doesn't exist. */
356 :
357 : static pointer_info *
358 1871181 : get_pointer (void *gp)
359 : {
360 1871181 : pointer_info *p;
361 :
362 1871181 : p = find_pointer (gp);
363 1871181 : if (p != NULL)
364 : return p;
365 :
366 : /* Pointer doesn't have an integer. Give it one. */
367 389374 : p = gfc_get_pointer_info ();
368 :
369 389374 : p->u.pointer = gp;
370 389374 : p->integer = symbol_number++;
371 :
372 389374 : gfc_insert_bbt (&pi_root, p, compare_pointers);
373 :
374 389374 : return p;
375 : }
376 :
377 :
378 : /* Given an integer during reading, find it in the pointer_info tree,
379 : creating the node if not found. */
380 :
381 : static pointer_info *
382 8642831 : get_integer (HOST_WIDE_INT integer)
383 : {
384 8642831 : pointer_info *p, t;
385 8642831 : int c;
386 :
387 8642831 : t.integer = integer;
388 :
389 8642831 : p = pi_root;
390 67111011 : while (p != NULL)
391 : {
392 65459595 : c = compare_integers (&t, p);
393 : if (c == 0)
394 : break;
395 :
396 58468180 : p = (c < 0) ? p->left : p->right;
397 : }
398 :
399 8642831 : if (p != NULL)
400 : return p;
401 :
402 1651416 : p = gfc_get_pointer_info ();
403 1651416 : p->integer = integer;
404 1651416 : p->u.pointer = NULL;
405 :
406 1651416 : gfc_insert_bbt (&pi_root, p, compare_integers);
407 :
408 1651416 : return p;
409 : }
410 :
411 :
412 : /* Resolve any fixups using a known pointer. */
413 :
414 : static void
415 1680308 : resolve_fixups (fixup_t *f, void *gp)
416 : {
417 1680308 : fixup_t *next;
418 :
419 2532825 : for (; f; f = next)
420 : {
421 852517 : next = f->next;
422 852517 : *(f->pointer) = gp;
423 852517 : free (f);
424 : }
425 1680308 : }
426 :
427 :
428 : /* Convert a string such that it starts with a lower-case character. Used
429 : to convert the symtree name of a derived-type to the symbol name or to
430 : the name of the associated generic function. */
431 :
432 : const char *
433 1059966 : gfc_dt_lower_string (const char *name)
434 : {
435 1059966 : if (name[0] != (char) TOLOWER ((unsigned char) name[0]))
436 62708 : return gfc_get_string ("%c%s", (char) TOLOWER ((unsigned char) name[0]),
437 62708 : &name[1]);
438 997258 : return gfc_get_string ("%s", name);
439 : }
440 :
441 :
442 : /* Convert a string such that it starts with an upper-case character. Used to
443 : return the symtree-name for a derived type; the symbol name itself and the
444 : symtree/symbol name of the associated generic function start with a lower-
445 : case character. */
446 :
447 : const char *
448 1514101 : gfc_dt_upper_string (const char *name)
449 : {
450 1514101 : if (name[0] != (char) TOUPPER ((unsigned char) name[0]))
451 1489311 : return gfc_get_string ("%c%s", (char) TOUPPER ((unsigned char) name[0]),
452 1489311 : &name[1]);
453 24790 : return gfc_get_string ("%s", name);
454 : }
455 :
456 : /* Call here during module reading when we know what pointer to
457 : associate with an integer. Any fixups that exist are resolved at
458 : this time. */
459 :
460 : static void
461 1024685 : associate_integer_pointer (pointer_info *p, void *gp)
462 : {
463 1024685 : if (p->u.pointer != NULL)
464 0 : gfc_internal_error ("associate_integer_pointer(): Already associated");
465 :
466 1024685 : p->u.pointer = gp;
467 :
468 1024685 : resolve_fixups (p->fixup, gp);
469 :
470 1024685 : p->fixup = NULL;
471 1024685 : }
472 :
473 :
474 : /* During module reading, given an integer and a pointer to a pointer,
475 : either store the pointer from an already-known value or create a
476 : fixup structure in order to store things later. Returns zero if
477 : the reference has been actually stored, or nonzero if the reference
478 : must be fixed later (i.e., associate_integer_pointer must be called
479 : sometime later. Returns the pointer_info structure. */
480 :
481 : static pointer_info *
482 5395950 : add_fixup (HOST_WIDE_INT integer, void *gp)
483 : {
484 5395950 : pointer_info *p;
485 5395950 : fixup_t *f;
486 5395950 : char **cp;
487 :
488 5395950 : p = get_integer (integer);
489 :
490 5395950 : if (p->integer == 0 || p->u.pointer != NULL)
491 : {
492 4554724 : cp = (char **) gp;
493 4554724 : *cp = (char *) p->u.pointer;
494 : }
495 : else
496 : {
497 841226 : f = XCNEW (fixup_t);
498 :
499 841226 : f->next = p->fixup;
500 841226 : p->fixup = f;
501 :
502 841226 : f->pointer = (void **) gp;
503 : }
504 :
505 5395950 : return p;
506 : }
507 :
508 :
509 : /*****************************************************************/
510 :
511 : /* Parser related subroutines */
512 :
513 : /* Free the rename list left behind by a USE statement. */
514 :
515 : static void
516 91734 : free_rename (gfc_use_rename *list)
517 : {
518 91734 : gfc_use_rename *next;
519 :
520 101957 : for (; list; list = next)
521 : {
522 10223 : next = list->next;
523 10223 : free (list);
524 : }
525 0 : }
526 :
527 :
528 : /* Match a USE statement. */
529 :
530 : match
531 23594 : gfc_match_use (void)
532 : {
533 23594 : char name[GFC_MAX_SYMBOL_LEN + 1], module_nature[GFC_MAX_SYMBOL_LEN + 1];
534 23594 : gfc_use_rename *tail = NULL, *new_use;
535 23594 : interface_type type, type2;
536 23594 : gfc_intrinsic_op op;
537 23594 : match m;
538 23594 : gfc_use_list *use_list;
539 23594 : gfc_symtree *st;
540 23594 : locus loc;
541 :
542 23594 : use_list = gfc_get_use_list ();
543 :
544 23594 : if (gfc_match (" , ") == MATCH_YES)
545 : {
546 3497 : if ((m = gfc_match (" %n ::", module_nature)) == MATCH_YES)
547 : {
548 3495 : if (!gfc_notify_std (GFC_STD_F2003, "module "
549 : "nature in USE statement at %C"))
550 0 : goto cleanup;
551 :
552 3495 : if (strcmp (module_nature, "intrinsic") == 0)
553 3481 : use_list->intrinsic = true;
554 : else
555 : {
556 14 : if (strcmp (module_nature, "non_intrinsic") == 0)
557 13 : use_list->non_intrinsic = true;
558 : else
559 : {
560 1 : gfc_error ("Module nature in USE statement at %C shall "
561 : "be either INTRINSIC or NON_INTRINSIC");
562 1 : goto cleanup;
563 : }
564 : }
565 : }
566 : else
567 : {
568 : /* Help output a better error message than "Unclassifiable
569 : statement". */
570 2 : gfc_match (" %n", module_nature);
571 2 : if (strcmp (module_nature, "intrinsic") == 0
572 1 : || strcmp (module_nature, "non_intrinsic") == 0)
573 2 : gfc_error ("\"::\" was expected after module nature at %C "
574 : "but was not found");
575 2 : free (use_list);
576 2 : return m;
577 : }
578 : }
579 : else
580 : {
581 20097 : m = gfc_match (" ::");
582 20429 : if (m == MATCH_YES &&
583 332 : !gfc_notify_std(GFC_STD_F2003, "\"USE :: module\" at %C"))
584 0 : goto cleanup;
585 :
586 20097 : if (m != MATCH_YES)
587 : {
588 19765 : m = gfc_match ("% ");
589 19765 : if (m != MATCH_YES)
590 : {
591 17 : free (use_list);
592 17 : return m;
593 : }
594 : }
595 : }
596 :
597 23574 : use_list->where = gfc_current_locus;
598 :
599 23574 : m = gfc_match_name (name);
600 23574 : if (m != MATCH_YES)
601 : {
602 12 : free (use_list);
603 12 : return m;
604 : }
605 :
606 23562 : use_list->module_name = gfc_get_string ("%s", name);
607 :
608 23562 : if (gfc_match_eos () == MATCH_YES)
609 14849 : goto done;
610 :
611 8713 : if (gfc_match_char (',') != MATCH_YES)
612 0 : goto syntax;
613 :
614 8713 : if (gfc_match (" only :") == MATCH_YES)
615 8466 : use_list->only_flag = true;
616 :
617 8713 : if (gfc_match_eos () == MATCH_YES)
618 1 : goto done;
619 :
620 13183 : for (;;)
621 : {
622 : /* Get a new rename struct and add it to the rename list. */
623 13183 : new_use = gfc_get_use_rename ();
624 13183 : new_use->where = gfc_current_locus;
625 13183 : new_use->found = 0;
626 :
627 13183 : if (use_list->rename == NULL)
628 8712 : use_list->rename = new_use;
629 : else
630 4471 : tail->next = new_use;
631 13183 : tail = new_use;
632 :
633 : /* See what kind of interface we're dealing with. Assume it is
634 : not an operator. */
635 13183 : new_use->op = INTRINSIC_NONE;
636 13183 : if (gfc_match_generic_spec (&type, name, &op) == MATCH_ERROR)
637 0 : goto cleanup;
638 :
639 13183 : switch (type)
640 : {
641 1 : case INTERFACE_NAMELESS:
642 1 : gfc_error ("Missing generic specification in USE statement at %C");
643 1 : goto cleanup;
644 :
645 13065 : case INTERFACE_USER_OP:
646 13065 : case INTERFACE_GENERIC:
647 13065 : case INTERFACE_DTIO:
648 13065 : loc = gfc_current_locus;
649 :
650 13065 : m = gfc_match (" =>");
651 :
652 79 : if (type == INTERFACE_USER_OP && m == MATCH_YES
653 13111 : && (!gfc_notify_std(GFC_STD_F2003, "Renaming "
654 : "operators in USE statements at %C")))
655 2 : goto cleanup;
656 :
657 13063 : if (type == INTERFACE_USER_OP)
658 77 : new_use->op = INTRINSIC_USER;
659 :
660 13063 : if (use_list->only_flag)
661 : {
662 12713 : if (m != MATCH_YES)
663 12380 : strcpy (new_use->use_name, name);
664 : else
665 : {
666 333 : strcpy (new_use->local_name, name);
667 333 : m = gfc_match_generic_spec (&type2, new_use->use_name, &op);
668 333 : if (type != type2)
669 1 : goto syntax;
670 332 : if (m == MATCH_NO)
671 0 : goto syntax;
672 332 : if (m == MATCH_ERROR)
673 0 : goto cleanup;
674 : }
675 : }
676 : else
677 : {
678 350 : if (m != MATCH_YES)
679 0 : goto syntax;
680 350 : strcpy (new_use->local_name, name);
681 :
682 350 : m = gfc_match_generic_spec (&type2, new_use->use_name, &op);
683 350 : if (type != type2)
684 2 : goto syntax;
685 348 : if (m == MATCH_NO)
686 0 : goto syntax;
687 348 : if (m == MATCH_ERROR)
688 0 : goto cleanup;
689 : }
690 :
691 13060 : st = gfc_find_symtree (gfc_current_ns->sym_root, name);
692 13060 : if (st && type != INTERFACE_USER_OP
693 13 : && (st->n.sym->module != use_list->module_name
694 3 : || strcmp (st->n.sym->name, new_use->use_name) != 0))
695 : {
696 10 : if (m == MATCH_YES)
697 7 : gfc_error ("Symbol %qs at %L conflicts with the rename symbol "
698 : "at %L", name, &st->n.sym->declared_at, &loc);
699 : else
700 3 : gfc_error ("Symbol %qs at %L conflicts with the symbol "
701 : "at %L", name, &st->n.sym->declared_at, &loc);
702 10 : goto cleanup;
703 : }
704 :
705 13050 : if (strcmp (new_use->use_name, use_list->module_name) == 0
706 13048 : || strcmp (new_use->local_name, use_list->module_name) == 0)
707 : {
708 3 : gfc_error ("The name %qs at %C has already been used as "
709 : "an external module name", use_list->module_name);
710 3 : goto cleanup;
711 : }
712 : break;
713 :
714 117 : case INTERFACE_INTRINSIC_OP:
715 117 : new_use->op = op;
716 117 : break;
717 :
718 0 : default:
719 0 : gcc_unreachable ();
720 : }
721 :
722 13164 : if (gfc_match_eos () == MATCH_YES)
723 : break;
724 4473 : if (gfc_match_char (',') != MATCH_YES)
725 2 : goto syntax;
726 : }
727 :
728 8691 : done:
729 23541 : if (module_list)
730 : {
731 : gfc_use_list *last = module_list;
732 4175 : while (last->next)
733 : last = last->next;
734 3335 : last->next = use_list;
735 : }
736 : else
737 20206 : module_list = use_list;
738 :
739 : return MATCH_YES;
740 :
741 5 : syntax:
742 5 : gfc_syntax_error (ST_USE);
743 :
744 22 : cleanup:
745 22 : free_rename (use_list->rename);
746 22 : free (use_list);
747 22 : return MATCH_ERROR;
748 : }
749 :
750 :
751 : /* Match a SUBMODULE statement.
752 :
753 : According to F2008:11.2.3.2, "The submodule identifier is the
754 : ordered pair whose first element is the ancestor module name and
755 : whose second element is the submodule name. 'Submodule_name' is
756 : used for the submodule filename and uses '@' as a separator, whilst
757 : the name of the symbol for the module uses '.' as a separator.
758 : The reasons for these choices are:
759 : (i) To follow another leading brand in the submodule filenames;
760 : (ii) Since '.' is not particularly visible in the filenames; and
761 : (iii) The linker does not permit '@' in mnemonics. */
762 :
763 : match
764 268 : gfc_match_submodule (void)
765 : {
766 268 : match m;
767 268 : char name[GFC_MAX_SYMBOL_LEN + 1];
768 268 : gfc_use_list *use_list;
769 268 : bool seen_colon = false;
770 :
771 268 : if (!gfc_notify_std (GFC_STD_F2008, "SUBMODULE declaration at %C"))
772 : return MATCH_ERROR;
773 :
774 267 : if (gfc_current_state () != COMP_NONE)
775 : {
776 3 : gfc_error ("SUBMODULE declaration at %C cannot appear within "
777 : "another scoping unit");
778 3 : return MATCH_ERROR;
779 : }
780 :
781 264 : gfc_new_block = NULL;
782 264 : gcc_assert (module_list == NULL);
783 :
784 264 : if (gfc_match_char ('(') != MATCH_YES)
785 0 : goto syntax;
786 :
787 290 : while (1)
788 : {
789 290 : m = gfc_match (" %n", name);
790 290 : if (m != MATCH_YES)
791 0 : goto syntax;
792 :
793 290 : use_list = gfc_get_use_list ();
794 290 : use_list->where = gfc_current_locus;
795 :
796 290 : if (module_list)
797 : {
798 : gfc_use_list *last = module_list;
799 26 : while (last->next)
800 : last = last->next;
801 26 : last->next = use_list;
802 26 : use_list->module_name
803 26 : = gfc_get_string ("%s.%s", module_list->module_name, name);
804 26 : use_list->submodule_name
805 26 : = gfc_get_string ("%s@%s", module_list->module_name, name);
806 : }
807 : else
808 : {
809 264 : module_list = use_list;
810 264 : use_list->module_name = gfc_get_string ("%s", name);
811 264 : use_list->submodule_name = use_list->module_name;
812 : }
813 :
814 290 : if (gfc_match_char (')') == MATCH_YES)
815 : break;
816 :
817 54 : if (gfc_match_char (':') != MATCH_YES
818 27 : || seen_colon)
819 1 : goto syntax;
820 :
821 : seen_colon = true;
822 : }
823 :
824 263 : m = gfc_match (" %s%t", &gfc_new_block);
825 263 : if (m != MATCH_YES)
826 0 : goto syntax;
827 :
828 263 : submodule_name = gfc_get_string ("%s@%s", module_list->module_name,
829 : gfc_new_block->name);
830 :
831 263 : gfc_new_block->name = gfc_get_string ("%s.%s",
832 : module_list->module_name,
833 : gfc_new_block->name);
834 :
835 263 : if (!gfc_add_flavor (&gfc_new_block->attr, FL_MODULE,
836 : gfc_new_block->name, NULL))
837 : return MATCH_ERROR;
838 :
839 : /* Just retain the ultimate .(s)mod file for reading, since it
840 : contains all the information in its ancestors. */
841 263 : use_list = module_list;
842 288 : for (; module_list->next; use_list = module_list)
843 : {
844 25 : module_list = use_list->next;
845 25 : free (use_list);
846 : }
847 :
848 : return MATCH_YES;
849 :
850 1 : syntax:
851 1 : gfc_error ("Syntax error in SUBMODULE statement at %C");
852 1 : return MATCH_ERROR;
853 : }
854 :
855 :
856 : /* Given a name and a number, inst, return the inst name
857 : under which to load this symbol. Returns NULL if this
858 : symbol shouldn't be loaded. If inst is zero, returns
859 : the number of instances of this name. If interface is
860 : true, a user-defined operator is sought, otherwise only
861 : non-operators are sought. */
862 :
863 : static const char *
864 1143839 : find_use_name_n (const char *name, int *inst, bool interface)
865 : {
866 1143839 : gfc_use_rename *u;
867 1143839 : const char *low_name = NULL;
868 1143839 : int i;
869 :
870 : /* For derived types. */
871 1143839 : if (name[0] != (char) TOLOWER ((unsigned char) name[0]))
872 28944 : low_name = gfc_dt_lower_string (name);
873 :
874 1143839 : i = 0;
875 1287082 : for (u = gfc_rename_list; u; u = u->next)
876 : {
877 147560 : if ((!low_name && strcmp (u->use_name, name) != 0)
878 3629 : || (low_name && strcmp (u->use_name, low_name) != 0)
879 8675 : || (u->op == INTRINSIC_USER && !interface)
880 8673 : || (u->op != INTRINSIC_USER && interface))
881 138887 : continue;
882 8673 : if (++i == *inst)
883 : break;
884 : }
885 :
886 1143839 : if (!*inst)
887 : {
888 571848 : *inst = i;
889 571848 : return NULL;
890 : }
891 :
892 571991 : if (u == NULL)
893 622822 : return only_flag ? NULL : name;
894 :
895 4317 : u->found = 1;
896 :
897 4317 : if (low_name)
898 : {
899 719 : if (u->local_name[0] == '\0')
900 : return name;
901 108 : return gfc_dt_upper_string (u->local_name);
902 : }
903 :
904 3598 : return (u->local_name[0] != '\0') ? u->local_name : name;
905 : }
906 :
907 :
908 : /* Given a name, return the name under which to load this symbol.
909 : Returns NULL if this symbol shouldn't be loaded. */
910 :
911 : static const char *
912 86 : find_use_name (const char *name, bool interface)
913 : {
914 86 : int i = 1;
915 50 : return find_use_name_n (name, &i, interface);
916 : }
917 :
918 :
919 : /* Given a real name, return the number of use names associated with it. */
920 :
921 : static int
922 571848 : number_use_names (const char *name, bool interface)
923 : {
924 571848 : int i = 0;
925 0 : find_use_name_n (name, &i, interface);
926 571848 : return i;
927 : }
928 :
929 :
930 : /* Try to find the operator in the current list. */
931 :
932 : static gfc_use_rename *
933 70401 : find_use_operator (gfc_intrinsic_op op)
934 : {
935 70401 : gfc_use_rename *u;
936 :
937 173187 : for (u = gfc_rename_list; u; u = u->next)
938 102994 : if (u->op == op)
939 : return u;
940 :
941 : return NULL;
942 : }
943 :
944 :
945 : /*****************************************************************/
946 :
947 : /* The next couple of subroutines maintain a tree used to avoid a
948 : brute-force search for a combination of true name and module name.
949 : While symtree names, the name that a particular symbol is known by
950 : can changed with USE statements, we still have to keep track of the
951 : true names to generate the correct reference, and also avoid
952 : loading the same real symbol twice in a program unit.
953 :
954 : When we start reading, the true name tree is built and maintained
955 : as symbols are read. The tree is searched as we load new symbols
956 : to see if it already exists someplace in the namespace. */
957 :
958 : typedef struct true_name
959 : {
960 : BBT_HEADER (true_name);
961 : const char *name;
962 : gfc_symbol *sym;
963 : }
964 : true_name;
965 :
966 : static true_name *true_name_root;
967 :
968 :
969 : /* Compare two true_name structures. */
970 :
971 : static int
972 3382174 : compare_true_names (void *_t1, void *_t2)
973 : {
974 3382174 : true_name *t1, *t2;
975 3382174 : int c;
976 :
977 3382174 : t1 = (true_name *) _t1;
978 3382174 : t2 = (true_name *) _t2;
979 :
980 3382174 : c = ((t1->sym->module > t2->sym->module)
981 3382174 : - (t1->sym->module < t2->sym->module));
982 3382174 : if (c != 0)
983 : return c;
984 :
985 1036705 : return strcmp (t1->name, t2->name);
986 : }
987 :
988 :
989 : /* Given a true name, search the true name tree to see if it exists
990 : within the main namespace. */
991 :
992 : static gfc_symbol *
993 1338391 : find_true_name (const char *name, const char *module)
994 : {
995 1338391 : true_name t, *p;
996 1338391 : gfc_symbol sym;
997 1338391 : int c;
998 :
999 1338391 : t.name = gfc_get_string ("%s", name);
1000 1338391 : if (module != NULL)
1001 1317336 : sym.module = gfc_get_string ("%s", module);
1002 : else
1003 21055 : sym.module = NULL;
1004 1338391 : t.sym = &sym;
1005 :
1006 1338391 : p = true_name_root;
1007 4247697 : while (p != NULL)
1008 : {
1009 2959785 : c = compare_true_names ((void *) (&t), (void *) p);
1010 2959785 : if (c == 0)
1011 50479 : return p->sym;
1012 :
1013 2909306 : p = (c < 0) ? p->left : p->right;
1014 : }
1015 :
1016 : return NULL;
1017 : }
1018 :
1019 :
1020 : /* Given a gfc_symbol pointer that is not in the true name tree, add it. */
1021 :
1022 : static void
1023 103539 : add_true_name (gfc_symbol *sym)
1024 : {
1025 103539 : true_name *t;
1026 :
1027 103539 : t = XCNEW (true_name);
1028 103539 : t->sym = sym;
1029 103539 : if (gfc_fl_struct (sym->attr.flavor))
1030 5775 : t->name = gfc_dt_upper_string (sym->name);
1031 : else
1032 97764 : t->name = sym->name;
1033 :
1034 103539 : gfc_insert_bbt (&true_name_root, t, compare_true_names);
1035 103539 : }
1036 :
1037 :
1038 : /* Recursive function to build the initial true name tree by
1039 : recursively traversing the current namespace. */
1040 :
1041 : static void
1042 222189 : build_tnt (gfc_symtree *st)
1043 : {
1044 222189 : const char *name;
1045 222189 : if (st == NULL)
1046 : return;
1047 :
1048 104309 : build_tnt (st->left);
1049 104309 : build_tnt (st->right);
1050 :
1051 104309 : if (gfc_fl_struct (st->n.sym->attr.flavor))
1052 6217 : name = gfc_dt_upper_string (st->n.sym->name);
1053 : else
1054 98092 : name = st->n.sym->name;
1055 :
1056 104309 : if (find_true_name (name, st->n.sym->module) != NULL)
1057 : return;
1058 :
1059 103539 : add_true_name (st->n.sym);
1060 : }
1061 :
1062 :
1063 : /* Initialize the true name tree with the current namespace. */
1064 :
1065 : static void
1066 13571 : init_true_name_tree (void)
1067 : {
1068 13571 : true_name_root = NULL;
1069 13571 : build_tnt (gfc_current_ns->sym_root);
1070 13571 : }
1071 :
1072 :
1073 : /* Recursively free a true name tree node. */
1074 :
1075 : static void
1076 220649 : free_true_name (true_name *t)
1077 : {
1078 220649 : if (t == NULL)
1079 : return;
1080 103539 : free_true_name (t->left);
1081 103539 : free_true_name (t->right);
1082 :
1083 103539 : free (t);
1084 : }
1085 :
1086 :
1087 : /*****************************************************************/
1088 :
1089 : /* Module reading and writing. */
1090 :
1091 : /* The following are versions similar to the ones in scanner.cc, but
1092 : for dealing with compressed module files. */
1093 :
1094 : static gzFile
1095 10035 : gzopen_included_file_1 (const char *name, gfc_directorylist *list,
1096 : bool module, bool system)
1097 : {
1098 10035 : char *fullname;
1099 10035 : gfc_directorylist *p;
1100 10035 : gzFile f;
1101 :
1102 35021 : for (p = list; p; p = p->next)
1103 : {
1104 27175 : if (module && !p->use_for_modules)
1105 4268 : continue;
1106 :
1107 22907 : fullname = (char *) alloca(strlen (p->path) + strlen (name) + 2);
1108 22907 : strcpy (fullname, p->path);
1109 22907 : strcat (fullname, "/");
1110 22907 : strcat (fullname, name);
1111 :
1112 22907 : f = gzopen (fullname, "r");
1113 22907 : if (f != NULL)
1114 : {
1115 2189 : if (gfc_cpp_makedep ())
1116 0 : gfc_cpp_add_dep (fullname, system);
1117 :
1118 2189 : free (module_fullpath);
1119 2189 : module_fullpath = xstrdup (fullname);
1120 2189 : return f;
1121 : }
1122 : }
1123 :
1124 : return NULL;
1125 : }
1126 :
1127 : static gzFile
1128 20187 : gzopen_included_file (const char *name, bool include_cwd, bool module)
1129 : {
1130 20187 : gzFile f = NULL;
1131 :
1132 20187 : if (IS_ABSOLUTE_PATH (name) || include_cwd)
1133 : {
1134 20187 : f = gzopen (name, "r");
1135 20187 : if (f)
1136 : {
1137 11382 : if (gfc_cpp_makedep ())
1138 0 : gfc_cpp_add_dep (name, false);
1139 :
1140 11382 : free (module_fullpath);
1141 11382 : module_fullpath = xstrdup (name);
1142 : }
1143 : }
1144 :
1145 11382 : if (!f)
1146 8805 : f = gzopen_included_file_1 (name, include_dirs, module, false);
1147 :
1148 20187 : return f;
1149 : }
1150 :
1151 : static gzFile
1152 1230 : gzopen_intrinsic_module (const char* name)
1153 : {
1154 1230 : gzFile f = NULL;
1155 :
1156 1230 : if (IS_ABSOLUTE_PATH (name))
1157 : {
1158 0 : f = gzopen (name, "r");
1159 0 : if (f)
1160 : {
1161 0 : if (gfc_cpp_makedep ())
1162 0 : gfc_cpp_add_dep (name, true);
1163 :
1164 0 : free (module_fullpath);
1165 0 : module_fullpath = xstrdup (name);
1166 : }
1167 : }
1168 :
1169 0 : if (!f)
1170 1230 : f = gzopen_included_file_1 (name, intrinsic_modules_dirs, true, true);
1171 :
1172 1230 : return f;
1173 : }
1174 :
1175 :
1176 : enum atom_type
1177 : {
1178 : ATOM_NAME, ATOM_LPAREN, ATOM_RPAREN, ATOM_INTEGER, ATOM_STRING
1179 : };
1180 :
1181 : static atom_type last_atom;
1182 :
1183 :
1184 : /* The name buffer must be at least as long as a symbol name. Right
1185 : now it's not clear how we're going to store numeric constants--
1186 : probably as a hexadecimal string, since this will allow the exact
1187 : number to be preserved (this can't be done by a decimal
1188 : representation). Worry about that later. TODO! */
1189 :
1190 : #define MAX_ATOM_SIZE 100
1191 :
1192 : static HOST_WIDE_INT atom_int;
1193 : static char *atom_string, atom_name[MAX_ATOM_SIZE];
1194 :
1195 :
1196 : /* Report problems with a module. Error reporting is not very
1197 : elaborate, since this sorts of errors shouldn't really happen.
1198 : This subroutine never returns. */
1199 :
1200 : static void bad_module (const char *) ATTRIBUTE_NORETURN;
1201 :
1202 : static void
1203 0 : bad_module (const char *msgid)
1204 : {
1205 0 : XDELETEVEC (module_content);
1206 0 : module_content = NULL;
1207 :
1208 0 : switch (iomode)
1209 : {
1210 0 : case IO_INPUT:
1211 0 : gfc_fatal_error ("Reading module %qs at line %d column %d: %s",
1212 : module_fullpath, module_line, module_column, msgid);
1213 0 : break;
1214 0 : case IO_OUTPUT:
1215 0 : gfc_fatal_error ("Writing module %qs at line %d column %d: %s",
1216 : module_name, module_line, module_column, msgid);
1217 0 : break;
1218 0 : default:
1219 0 : gfc_fatal_error ("Module %qs at line %d column %d: %s",
1220 : module_name, module_line, module_column, msgid);
1221 : break;
1222 : }
1223 : }
1224 :
1225 :
1226 : /* Set the module's input pointer. */
1227 :
1228 : static void
1229 1083451 : set_module_locus (module_locus *m)
1230 : {
1231 1083451 : module_column = m->column;
1232 1083451 : module_line = m->line;
1233 1083451 : module_pos = m->pos;
1234 0 : }
1235 :
1236 :
1237 : /* Get the module's input pointer so that we can restore it later. */
1238 :
1239 : static void
1240 1288386 : get_module_locus (module_locus *m)
1241 : {
1242 1288386 : m->column = module_column;
1243 1288386 : m->line = module_line;
1244 1288386 : m->pos = module_pos;
1245 0 : }
1246 :
1247 : /* Peek at the next character in the module. */
1248 :
1249 : static int
1250 532 : module_peek_char (void)
1251 : {
1252 532 : return module_content[module_pos];
1253 : }
1254 :
1255 : /* Get the next character in the module, updating our reckoning of
1256 : where we are. */
1257 :
1258 : static int
1259 549063062 : module_char (void)
1260 : {
1261 549063062 : const char c = module_content[module_pos++];
1262 549063062 : if (c == '\0')
1263 0 : bad_module ("Unexpected EOF");
1264 :
1265 549063062 : prev_module_line = module_line;
1266 549063062 : prev_module_column = module_column;
1267 :
1268 549063062 : if (c == '\n')
1269 : {
1270 11768180 : module_line++;
1271 11768180 : module_column = 0;
1272 : }
1273 :
1274 549063062 : module_column++;
1275 549063062 : return c;
1276 : }
1277 :
1278 : /* Unget a character while remembering the line and column. Works for
1279 : a single character only. */
1280 :
1281 : static void
1282 81091095 : module_unget_char (void)
1283 : {
1284 81091095 : module_line = prev_module_line;
1285 81091095 : module_column = prev_module_column;
1286 81091095 : module_pos--;
1287 0 : }
1288 :
1289 : /* Parse a string constant. The delimiter is guaranteed to be a
1290 : single quote. */
1291 :
1292 : static void
1293 5592967 : parse_string (void)
1294 : {
1295 5592967 : int c;
1296 5592967 : size_t cursz = 30;
1297 5592967 : size_t len = 0;
1298 :
1299 5592967 : atom_string = XNEWVEC (char, cursz);
1300 :
1301 83866505 : for ( ; ; )
1302 : {
1303 44729736 : c = module_char ();
1304 :
1305 44729736 : if (c == '\'')
1306 : {
1307 5592967 : int c2 = module_char ();
1308 5592967 : if (c2 != '\'')
1309 : {
1310 5592967 : module_unget_char ();
1311 5592967 : break;
1312 : }
1313 : }
1314 :
1315 39136769 : if (len >= cursz)
1316 : {
1317 59946 : cursz *= 2;
1318 59946 : atom_string = XRESIZEVEC (char, atom_string, cursz);
1319 : }
1320 39136769 : atom_string[len] = c;
1321 39136769 : len++;
1322 39136769 : }
1323 :
1324 5592967 : atom_string = XRESIZEVEC (char, atom_string, len + 1);
1325 5592967 : atom_string[len] = '\0'; /* C-style string for debug purposes. */
1326 5592967 : }
1327 :
1328 :
1329 : /* Parse an integer. Should fit in a HOST_WIDE_INT. */
1330 :
1331 : static void
1332 36292628 : parse_integer (int c)
1333 : {
1334 36292628 : int sign = 1;
1335 :
1336 36292628 : atom_int = 0;
1337 36292628 : switch (c)
1338 : {
1339 : case ('-'):
1340 36292628 : sign = -1;
1341 : case ('+'):
1342 : break;
1343 36292096 : default:
1344 36292096 : atom_int = c - '0';
1345 36292096 : break;
1346 : }
1347 :
1348 54110728 : for (;;)
1349 : {
1350 45201678 : c = module_char ();
1351 45201678 : if (!ISDIGIT (c))
1352 : {
1353 36292628 : module_unget_char ();
1354 36292628 : break;
1355 : }
1356 :
1357 8909050 : atom_int = 10 * atom_int + c - '0';
1358 : }
1359 :
1360 36292628 : atom_int *= sign;
1361 36292628 : }
1362 :
1363 :
1364 : /* Parse a name. */
1365 :
1366 : static void
1367 26369164 : parse_name (int c)
1368 : {
1369 26369164 : char *p;
1370 26369164 : int len;
1371 :
1372 26369164 : p = atom_name;
1373 :
1374 26369164 : *p++ = c;
1375 26369164 : len = 1;
1376 :
1377 224655027 : for (;;)
1378 : {
1379 224655027 : c = module_char ();
1380 224655027 : if (!ISALNUM (c) && c != '_' && c != '-')
1381 : {
1382 26369164 : module_unget_char ();
1383 26369164 : break;
1384 : }
1385 :
1386 198285863 : *p++ = c;
1387 198285863 : if (++len > GFC_MAX_SYMBOL_LEN)
1388 0 : bad_module ("Name too long");
1389 : }
1390 :
1391 26369164 : *p = '\0';
1392 :
1393 26369164 : }
1394 :
1395 :
1396 : /* Read the next atom in the module's input stream. */
1397 :
1398 : static atom_type
1399 127610959 : parse_atom (void)
1400 : {
1401 213224788 : int c;
1402 :
1403 213224788 : do
1404 : {
1405 213224788 : c = module_char ();
1406 : }
1407 213224788 : while (c == ' ' || c == '\r' || c == '\n');
1408 :
1409 127610959 : switch (c)
1410 : {
1411 : case '(':
1412 : return ATOM_LPAREN;
1413 :
1414 29691661 : case ')':
1415 29691661 : return ATOM_RPAREN;
1416 :
1417 5592967 : case '\'':
1418 5592967 : parse_string ();
1419 5592967 : return ATOM_STRING;
1420 :
1421 36292096 : case '0':
1422 36292096 : case '1':
1423 36292096 : case '2':
1424 36292096 : case '3':
1425 36292096 : case '4':
1426 36292096 : case '5':
1427 36292096 : case '6':
1428 36292096 : case '7':
1429 36292096 : case '8':
1430 36292096 : case '9':
1431 36292096 : parse_integer (c);
1432 36292096 : return ATOM_INTEGER;
1433 :
1434 532 : case '+':
1435 532 : case '-':
1436 532 : if (ISDIGIT (module_peek_char ()))
1437 : {
1438 532 : parse_integer (c);
1439 532 : return ATOM_INTEGER;
1440 : }
1441 : else
1442 0 : bad_module ("Bad name");
1443 :
1444 26328451 : case 'a':
1445 26328451 : case 'b':
1446 26328451 : case 'c':
1447 26328451 : case 'd':
1448 26328451 : case 'e':
1449 26328451 : case 'f':
1450 26328451 : case 'g':
1451 26328451 : case 'h':
1452 26328451 : case 'i':
1453 26328451 : case 'j':
1454 26328451 : case 'k':
1455 26328451 : case 'l':
1456 26328451 : case 'm':
1457 26328451 : case 'n':
1458 26328451 : case 'o':
1459 26328451 : case 'p':
1460 26328451 : case 'q':
1461 26328451 : case 'r':
1462 26328451 : case 's':
1463 26328451 : case 't':
1464 26328451 : case 'u':
1465 26328451 : case 'v':
1466 26328451 : case 'w':
1467 26328451 : case 'x':
1468 26328451 : case 'y':
1469 26328451 : case 'z':
1470 26328451 : case 'A':
1471 26328451 : case 'B':
1472 26328451 : case 'C':
1473 26328451 : case 'D':
1474 26328451 : case 'E':
1475 26328451 : case 'F':
1476 26328451 : case 'G':
1477 26328451 : case 'H':
1478 26328451 : case 'I':
1479 26328451 : case 'J':
1480 26328451 : case 'K':
1481 26328451 : case 'L':
1482 26328451 : case 'M':
1483 26328451 : case 'N':
1484 26328451 : case 'O':
1485 26328451 : case 'P':
1486 26328451 : case 'Q':
1487 26328451 : case 'R':
1488 26328451 : case 'S':
1489 26328451 : case 'T':
1490 26328451 : case 'U':
1491 26328451 : case 'V':
1492 26328451 : case 'W':
1493 26328451 : case 'X':
1494 26328451 : case 'Y':
1495 26328451 : case 'Z':
1496 26328451 : parse_name (c);
1497 26328451 : return ATOM_NAME;
1498 :
1499 0 : default:
1500 0 : bad_module ("Bad name");
1501 : }
1502 :
1503 : /* Not reached. */
1504 : }
1505 :
1506 :
1507 : /* Peek at the next atom on the input. */
1508 :
1509 : static atom_type
1510 12836336 : peek_atom (void)
1511 : {
1512 15157261 : int c;
1513 :
1514 15157261 : do
1515 : {
1516 15157261 : c = module_char ();
1517 : }
1518 15157261 : while (c == ' ' || c == '\r' || c == '\n');
1519 :
1520 12836336 : switch (c)
1521 : {
1522 303655 : case '(':
1523 303655 : module_unget_char ();
1524 303655 : return ATOM_LPAREN;
1525 :
1526 10129487 : case ')':
1527 10129487 : module_unget_char ();
1528 10129487 : return ATOM_RPAREN;
1529 :
1530 516798 : case '\'':
1531 516798 : module_unget_char ();
1532 516798 : return ATOM_STRING;
1533 :
1534 1881326 : case '0':
1535 1881326 : case '1':
1536 1881326 : case '2':
1537 1881326 : case '3':
1538 1881326 : case '4':
1539 1881326 : case '5':
1540 1881326 : case '6':
1541 1881326 : case '7':
1542 1881326 : case '8':
1543 1881326 : case '9':
1544 1881326 : module_unget_char ();
1545 1881326 : return ATOM_INTEGER;
1546 :
1547 0 : case '+':
1548 0 : case '-':
1549 0 : if (ISDIGIT (module_peek_char ()))
1550 : {
1551 0 : module_unget_char ();
1552 0 : return ATOM_INTEGER;
1553 : }
1554 : else
1555 0 : bad_module ("Bad name");
1556 :
1557 5070 : case 'a':
1558 5070 : case 'b':
1559 5070 : case 'c':
1560 5070 : case 'd':
1561 5070 : case 'e':
1562 5070 : case 'f':
1563 5070 : case 'g':
1564 5070 : case 'h':
1565 5070 : case 'i':
1566 5070 : case 'j':
1567 5070 : case 'k':
1568 5070 : case 'l':
1569 5070 : case 'm':
1570 5070 : case 'n':
1571 5070 : case 'o':
1572 5070 : case 'p':
1573 5070 : case 'q':
1574 5070 : case 'r':
1575 5070 : case 's':
1576 5070 : case 't':
1577 5070 : case 'u':
1578 5070 : case 'v':
1579 5070 : case 'w':
1580 5070 : case 'x':
1581 5070 : case 'y':
1582 5070 : case 'z':
1583 5070 : case 'A':
1584 5070 : case 'B':
1585 5070 : case 'C':
1586 5070 : case 'D':
1587 5070 : case 'E':
1588 5070 : case 'F':
1589 5070 : case 'G':
1590 5070 : case 'H':
1591 5070 : case 'I':
1592 5070 : case 'J':
1593 5070 : case 'K':
1594 5070 : case 'L':
1595 5070 : case 'M':
1596 5070 : case 'N':
1597 5070 : case 'O':
1598 5070 : case 'P':
1599 5070 : case 'Q':
1600 5070 : case 'R':
1601 5070 : case 'S':
1602 5070 : case 'T':
1603 5070 : case 'U':
1604 5070 : case 'V':
1605 5070 : case 'W':
1606 5070 : case 'X':
1607 5070 : case 'Y':
1608 5070 : case 'Z':
1609 5070 : module_unget_char ();
1610 5070 : return ATOM_NAME;
1611 :
1612 0 : default:
1613 0 : bad_module ("Bad name");
1614 : }
1615 : }
1616 :
1617 :
1618 : /* Read the next atom from the input, requiring that it be a
1619 : particular kind. */
1620 :
1621 : static void
1622 58278826 : require_atom (atom_type type)
1623 : {
1624 58278826 : atom_type t;
1625 58278826 : const char *p;
1626 58278826 : int column, line;
1627 :
1628 58278826 : column = module_column;
1629 58278826 : line = module_line;
1630 :
1631 58278826 : t = parse_atom ();
1632 58278826 : if (t != type)
1633 : {
1634 0 : switch (type)
1635 : {
1636 0 : case ATOM_NAME:
1637 0 : p = _("Expected name");
1638 0 : break;
1639 0 : case ATOM_LPAREN:
1640 0 : p = _("Expected left parenthesis");
1641 0 : break;
1642 0 : case ATOM_RPAREN:
1643 0 : p = _("Expected right parenthesis");
1644 0 : break;
1645 0 : case ATOM_INTEGER:
1646 0 : p = _("Expected integer");
1647 0 : break;
1648 0 : case ATOM_STRING:
1649 0 : p = _("Expected string");
1650 0 : break;
1651 0 : default:
1652 0 : gfc_internal_error ("require_atom(): bad atom type required");
1653 : }
1654 :
1655 0 : module_column = column;
1656 0 : module_line = line;
1657 0 : bad_module (p);
1658 : }
1659 58278826 : }
1660 :
1661 :
1662 : /* Given a pointer to an mstring array, require that the current input
1663 : be one of the strings in the array. We return the enum value. */
1664 :
1665 : static int
1666 12019635 : find_enum (const mstring *m)
1667 : {
1668 12019635 : int i;
1669 :
1670 12019635 : i = gfc_string2code (m, atom_name);
1671 12019635 : if (i >= 0)
1672 12019635 : return i;
1673 :
1674 0 : bad_module ("find_enum(): Enum not found");
1675 :
1676 : /* Not reached. */
1677 : }
1678 :
1679 :
1680 : /* Read a string. The caller is responsible for freeing. */
1681 :
1682 : static char*
1683 3702454 : read_string (void)
1684 : {
1685 3702454 : char* p;
1686 0 : require_atom (ATOM_STRING);
1687 3702454 : p = atom_string;
1688 3702454 : atom_string = NULL;
1689 3702454 : return p;
1690 : }
1691 :
1692 :
1693 : /**************** Module output subroutines ***************************/
1694 :
1695 : /* Output a character to a module file. */
1696 :
1697 : static void
1698 67313064 : write_char (char out)
1699 : {
1700 67313064 : if (gzputc (module_fp, out) == EOF)
1701 0 : gfc_fatal_error ("Error writing modules file: %s", xstrerror (errno));
1702 :
1703 67313064 : if (out != '\n')
1704 66086710 : module_column++;
1705 : else
1706 : {
1707 1226354 : module_column = 1;
1708 1226354 : module_line++;
1709 : }
1710 67313064 : }
1711 :
1712 :
1713 : /* Write an atom to a module. The line wrapping isn't perfect, but it
1714 : should work most of the time. This isn't that big of a deal, since
1715 : the file really isn't meant to be read by people anyway. */
1716 :
1717 : static void
1718 17916614 : write_atom (atom_type atom, const void *v)
1719 : {
1720 17916614 : char buffer[32];
1721 :
1722 : /* Workaround -Wmaybe-uninitialized false positive during
1723 : profiledbootstrap by initializing them. */
1724 17916614 : int len;
1725 17916614 : HOST_WIDE_INT i = 0;
1726 17916614 : const char *p;
1727 :
1728 17916614 : switch (atom)
1729 : {
1730 : case ATOM_STRING:
1731 : case ATOM_NAME:
1732 : p = (const char *) v;
1733 : break;
1734 :
1735 : case ATOM_LPAREN:
1736 : p = "(";
1737 : break;
1738 :
1739 : case ATOM_RPAREN:
1740 : p = ")";
1741 : break;
1742 :
1743 4869804 : case ATOM_INTEGER:
1744 4869804 : i = *((const HOST_WIDE_INT *) v);
1745 :
1746 4869804 : snprintf (buffer, sizeof (buffer), HOST_WIDE_INT_PRINT_DEC, i);
1747 4869804 : p = buffer;
1748 4869804 : break;
1749 :
1750 0 : default:
1751 0 : gfc_internal_error ("write_atom(): Trying to write dab atom");
1752 :
1753 : }
1754 :
1755 17916614 : if(p == NULL || *p == '\0')
1756 : len = 0;
1757 : else
1758 17541679 : len = strlen (p);
1759 :
1760 17916614 : if (atom != ATOM_RPAREN)
1761 : {
1762 13807694 : if (module_column + len > 72)
1763 835949 : write_char ('\n');
1764 : else
1765 : {
1766 :
1767 12971745 : if (last_atom != ATOM_LPAREN && module_column != 1)
1768 11269359 : write_char (' ');
1769 : }
1770 : }
1771 :
1772 13807694 : if (atom == ATOM_STRING)
1773 1114410 : write_char ('\'');
1774 :
1775 70505145 : while (p != NULL && *p)
1776 : {
1777 52588531 : if (atom == ATOM_STRING && *p == '\'')
1778 0 : write_char ('\'');
1779 52588531 : write_char (*p++);
1780 : }
1781 :
1782 17916614 : if (atom == ATOM_STRING)
1783 1114410 : write_char ('\'');
1784 :
1785 17916614 : last_atom = atom;
1786 17916614 : }
1787 :
1788 :
1789 :
1790 : /***************** Mid-level I/O subroutines *****************/
1791 :
1792 : /* These subroutines let their caller read or write atoms without
1793 : caring about which of the two is actually happening. This lets a
1794 : subroutine concentrate on the actual format of the data being
1795 : written. */
1796 :
1797 : static void mio_expr (gfc_expr **);
1798 : pointer_info *mio_symbol_ref (gfc_symbol **);
1799 : pointer_info *mio_interface_rest (gfc_interface **);
1800 : static void mio_symtree_ref (gfc_symtree **);
1801 :
1802 : /* Read or write an enumerated value. On writing, we return the input
1803 : value for the convenience of callers. We avoid using an integer
1804 : pointer because enums are sometimes inside bitfields. */
1805 :
1806 : static int
1807 13247670 : mio_name (int t, const mstring *m)
1808 : {
1809 13247670 : if (iomode == IO_OUTPUT)
1810 3713262 : write_atom (ATOM_NAME, gfc_code2string (m, t));
1811 : else
1812 : {
1813 9534408 : require_atom (ATOM_NAME);
1814 9534408 : t = find_enum (m);
1815 : }
1816 :
1817 13247670 : return t;
1818 : }
1819 :
1820 : /* Specialization of mio_name. */
1821 :
1822 : #define DECL_MIO_NAME(TYPE) \
1823 : static inline TYPE \
1824 : MIO_NAME(TYPE) (TYPE t, const mstring *m) \
1825 : { \
1826 : return (TYPE) mio_name ((int) t, m); \
1827 : }
1828 : #define MIO_NAME(TYPE) mio_name_##TYPE
1829 :
1830 : static void
1831 17694538 : mio_lparen (void)
1832 : {
1833 17694538 : if (iomode == IO_OUTPUT)
1834 4108920 : write_atom (ATOM_LPAREN, NULL);
1835 : else
1836 13585618 : require_atom (ATOM_LPAREN);
1837 17694538 : }
1838 :
1839 :
1840 : static void
1841 16231093 : mio_rparen (void)
1842 : {
1843 16231093 : if (iomode == IO_OUTPUT)
1844 4108920 : write_atom (ATOM_RPAREN, NULL);
1845 : else
1846 12122173 : require_atom (ATOM_RPAREN);
1847 16231093 : }
1848 :
1849 :
1850 : static void
1851 13422229 : mio_integer (int *ip)
1852 : {
1853 13422229 : if (iomode == IO_OUTPUT)
1854 : {
1855 2999809 : HOST_WIDE_INT hwi = *ip;
1856 2999809 : write_atom (ATOM_INTEGER, &hwi);
1857 : }
1858 : else
1859 : {
1860 10422420 : require_atom (ATOM_INTEGER);
1861 10422420 : *ip = atom_int;
1862 : }
1863 13422229 : }
1864 :
1865 : static void
1866 396164 : mio_hwi (HOST_WIDE_INT *hwi)
1867 : {
1868 396164 : if (iomode == IO_OUTPUT)
1869 235588 : write_atom (ATOM_INTEGER, hwi);
1870 : else
1871 : {
1872 160576 : require_atom (ATOM_INTEGER);
1873 160576 : *hwi = atom_int;
1874 : }
1875 396164 : }
1876 :
1877 :
1878 : /* Read or write a gfc_intrinsic_op value. */
1879 :
1880 : static void
1881 1332 : mio_intrinsic_op (gfc_intrinsic_op* op)
1882 : {
1883 : /* FIXME: Would be nicer to do this via the operators symbolic name. */
1884 1332 : if (iomode == IO_OUTPUT)
1885 : {
1886 708 : HOST_WIDE_INT converted = (HOST_WIDE_INT) *op;
1887 708 : write_atom (ATOM_INTEGER, &converted);
1888 : }
1889 : else
1890 : {
1891 624 : require_atom (ATOM_INTEGER);
1892 624 : *op = (gfc_intrinsic_op) atom_int;
1893 : }
1894 1332 : }
1895 :
1896 :
1897 : /* Read or write a character pointer that points to a string on the heap. */
1898 :
1899 : static const char *
1900 9393 : mio_allocated_string (const char *s)
1901 : {
1902 9393 : if (iomode == IO_OUTPUT)
1903 : {
1904 9393 : write_atom (ATOM_STRING, s);
1905 9393 : return s;
1906 : }
1907 : else
1908 : {
1909 0 : require_atom (ATOM_STRING);
1910 0 : return atom_string;
1911 : }
1912 : }
1913 :
1914 :
1915 : /* Functions for quoting and unquoting strings. */
1916 :
1917 : static char *
1918 5534 : quote_string (const gfc_char_t *s, const size_t slength)
1919 : {
1920 5534 : const gfc_char_t *p;
1921 5534 : char *res, *q;
1922 5534 : size_t len = 0, i;
1923 :
1924 : /* Calculate the length we'll need: a backslash takes two ("\\"),
1925 : non-printable characters take 10 ("\Uxxxxxxxx") and others take 1. */
1926 20621 : for (p = s, i = 0; i < slength; p++, i++)
1927 : {
1928 15087 : if (*p == '\\')
1929 1 : len += 2;
1930 15086 : else if (!gfc_wide_is_printable (*p))
1931 4779 : len += 10;
1932 : else
1933 10307 : len++;
1934 : }
1935 :
1936 5534 : q = res = XCNEWVEC (char, len + 1);
1937 26155 : for (p = s, i = 0; i < slength; p++, i++)
1938 : {
1939 15087 : if (*p == '\\')
1940 1 : *q++ = '\\', *q++ = '\\';
1941 15086 : else if (!gfc_wide_is_printable (*p))
1942 : {
1943 4779 : sprintf (q, "\\U%08" HOST_WIDE_INT_PRINT "x",
1944 4779 : (unsigned HOST_WIDE_INT) *p);
1945 4779 : q += 10;
1946 : }
1947 : else
1948 10307 : *q++ = (unsigned char) *p;
1949 : }
1950 :
1951 5534 : res[len] = '\0';
1952 5534 : return res;
1953 : }
1954 :
1955 : static gfc_char_t *
1956 3010 : unquote_string (const char *s)
1957 : {
1958 3010 : size_t len, i;
1959 3010 : const char *p;
1960 3010 : gfc_char_t *res;
1961 :
1962 14746 : for (p = s, len = 0; *p; p++, len++)
1963 : {
1964 11736 : if (*p != '\\')
1965 9534 : continue;
1966 :
1967 2202 : if (p[1] == '\\')
1968 0 : p++;
1969 2202 : else if (p[1] == 'U')
1970 2202 : p += 9; /* That is a "\U????????". */
1971 : else
1972 0 : gfc_internal_error ("unquote_string(): got bad string");
1973 : }
1974 :
1975 3010 : res = gfc_get_wide_string (len + 1);
1976 14746 : for (i = 0, p = s; i < len; i++, p++)
1977 : {
1978 11736 : gcc_assert (*p);
1979 :
1980 11736 : if (*p != '\\')
1981 9534 : res[i] = (unsigned char) *p;
1982 2202 : else if (p[1] == '\\')
1983 : {
1984 0 : res[i] = (unsigned char) '\\';
1985 0 : p++;
1986 : }
1987 : else
1988 : {
1989 : /* We read the 8-digits hexadecimal constant that follows. */
1990 2202 : int j;
1991 2202 : unsigned n;
1992 2202 : gfc_char_t c = 0;
1993 :
1994 2202 : gcc_assert (p[1] == 'U');
1995 19818 : for (j = 0; j < 8; j++)
1996 : {
1997 17616 : c = c << 4;
1998 17616 : gcc_assert (sscanf (&p[j+2], "%01x", &n) == 1);
1999 17616 : c += n;
2000 : }
2001 :
2002 2202 : res[i] = c;
2003 2202 : p += 9;
2004 : }
2005 : }
2006 :
2007 3010 : res[len] = '\0';
2008 3010 : return res;
2009 : }
2010 :
2011 :
2012 : /* Read or write a character pointer that points to a wide string on the
2013 : heap, performing quoting/unquoting of nonprintable characters using the
2014 : form \U???????? (where each ? is a hexadecimal digit).
2015 : Length is the length of the string, only known and used in output mode. */
2016 :
2017 : static const gfc_char_t *
2018 8544 : mio_allocated_wide_string (const gfc_char_t *s, const size_t length)
2019 : {
2020 8544 : if (iomode == IO_OUTPUT)
2021 : {
2022 5534 : char *quoted = quote_string (s, length);
2023 5534 : write_atom (ATOM_STRING, quoted);
2024 5534 : free (quoted);
2025 5534 : return s;
2026 : }
2027 : else
2028 : {
2029 3010 : gfc_char_t *unquoted;
2030 :
2031 3010 : require_atom (ATOM_STRING);
2032 3010 : unquoted = unquote_string (atom_string);
2033 3010 : free (atom_string);
2034 3010 : return unquoted;
2035 : }
2036 : }
2037 :
2038 :
2039 : /* Read or write a string that is in static memory. */
2040 :
2041 : static void
2042 1016139 : mio_pool_string (const char **stringp)
2043 : {
2044 : /* TODO: one could write the string only once, and refer to it via a
2045 : fixup pointer. */
2046 :
2047 : /* As a special case we have to deal with a NULL string. This
2048 : happens for the 'module' member of 'gfc_symbol's that are not in a
2049 : module. We read / write these as the empty string. */
2050 1016139 : if (iomode == IO_OUTPUT)
2051 : {
2052 786785 : const char *p = *stringp == NULL ? "" : *stringp;
2053 786785 : write_atom (ATOM_STRING, p);
2054 : }
2055 : else
2056 : {
2057 229354 : require_atom (ATOM_STRING);
2058 458708 : *stringp = (atom_string[0] == '\0'
2059 229354 : ? NULL : gfc_get_string ("%s", atom_string));
2060 229354 : free (atom_string);
2061 : }
2062 1016139 : }
2063 :
2064 :
2065 : /* Read or write a string that is inside of some already-allocated
2066 : structure. */
2067 :
2068 : static void
2069 628702 : mio_internal_string (char *string)
2070 : {
2071 628702 : if (iomode == IO_OUTPUT)
2072 0 : write_atom (ATOM_STRING, string);
2073 : else
2074 : {
2075 628702 : require_atom (ATOM_STRING);
2076 628702 : strcpy (string, atom_string);
2077 628702 : free (atom_string);
2078 : }
2079 628702 : }
2080 :
2081 :
2082 : enum ab_attribute
2083 : { AB_ALLOCATABLE, AB_DIMENSION, AB_EXTERNAL, AB_INTRINSIC, AB_OPTIONAL,
2084 : AB_POINTER, AB_TARGET, AB_DUMMY, AB_RESULT, AB_DATA,
2085 : AB_IN_NAMELIST, AB_IN_COMMON, AB_FUNCTION, AB_SUBROUTINE, AB_SEQUENCE,
2086 : AB_ELEMENTAL, AB_PURE, AB_RECURSIVE, AB_GENERIC, AB_ALWAYS_EXPLICIT,
2087 : AB_CRAY_POINTER, AB_CRAY_POINTEE, AB_THREADPRIVATE,
2088 : AB_ALLOC_COMP, AB_POINTER_COMP, AB_PROC_POINTER_COMP, AB_PRIVATE_COMP,
2089 : AB_VALUE, AB_VOLATILE, AB_PROTECTED, AB_LOCK_COMP, AB_EVENT_COMP,
2090 : AB_IS_BIND_C, AB_IS_C_INTEROP, AB_IS_ISO_C, AB_ABSTRACT, AB_ZERO_COMP,
2091 : AB_IS_CLASS, AB_PROCEDURE, AB_PROC_POINTER, AB_ASYNCHRONOUS, AB_CODIMENSION,
2092 : AB_COARRAY_COMP, AB_VTYPE, AB_VTAB, AB_CONTIGUOUS, AB_CLASS_POINTER,
2093 : AB_IMPLICIT_PURE, AB_ARTIFICIAL, AB_UNLIMITED_POLY, AB_OMP_DECLARE_TARGET,
2094 : AB_ARRAY_OUTER_DEPENDENCY, AB_MODULE_PROCEDURE, AB_OACC_DECLARE_CREATE,
2095 : AB_OACC_DECLARE_COPYIN, AB_OACC_DECLARE_DEVICEPTR,
2096 : AB_OACC_DECLARE_DEVICE_RESIDENT, AB_OACC_DECLARE_LINK,
2097 : AB_OMP_DECLARE_TARGET_LINK, AB_OMP_DECLARE_TARGET_LOCAL,
2098 : AB_PDT_KIND, AB_PDT_LEN, AB_PDT_TYPE,
2099 : AB_PDT_COMP, AB_PDT_TEMPLATE, AB_PDT_ARRAY, AB_PDT_STRING,
2100 : AB_OACC_ROUTINE_LOP_GANG, AB_OACC_ROUTINE_LOP_WORKER,
2101 : AB_OACC_ROUTINE_LOP_VECTOR, AB_OACC_ROUTINE_LOP_SEQ,
2102 : AB_OACC_ROUTINE_NOHOST,
2103 : AB_OMP_REQ_REVERSE_OFFLOAD, AB_OMP_REQ_UNIFIED_ADDRESS, AB_OMP_REQ_SELF_MAPS,
2104 : AB_OMP_REQ_UNIFIED_SHARED_MEMORY, AB_OMP_REQ_DYNAMIC_ALLOCATORS,
2105 : AB_OMP_REQ_MEM_ORDER_SEQ_CST, AB_OMP_REQ_MEM_ORDER_ACQ_REL,
2106 : AB_OMP_REQ_MEM_ORDER_ACQUIRE, AB_OMP_REQ_MEM_ORDER_RELEASE,
2107 : AB_OMP_REQ_MEM_ORDER_RELAXED, AB_OMP_DEVICE_TYPE_NOHOST,
2108 : AB_OMP_DEVICE_TYPE_HOST, AB_OMP_DEVICE_TYPE_ANY, AB_OMP_GROUPPRIVATE
2109 : };
2110 :
2111 : static const mstring attr_bits[] =
2112 : {
2113 : minit ("ALLOCATABLE", AB_ALLOCATABLE),
2114 : minit ("ARTIFICIAL", AB_ARTIFICIAL),
2115 : minit ("ASYNCHRONOUS", AB_ASYNCHRONOUS),
2116 : minit ("DIMENSION", AB_DIMENSION),
2117 : minit ("CODIMENSION", AB_CODIMENSION),
2118 : minit ("CONTIGUOUS", AB_CONTIGUOUS),
2119 : minit ("EXTERNAL", AB_EXTERNAL),
2120 : minit ("INTRINSIC", AB_INTRINSIC),
2121 : minit ("OPTIONAL", AB_OPTIONAL),
2122 : minit ("POINTER", AB_POINTER),
2123 : minit ("VOLATILE", AB_VOLATILE),
2124 : minit ("TARGET", AB_TARGET),
2125 : minit ("THREADPRIVATE", AB_THREADPRIVATE),
2126 : minit ("DUMMY", AB_DUMMY),
2127 : minit ("RESULT", AB_RESULT),
2128 : minit ("DATA", AB_DATA),
2129 : minit ("IN_NAMELIST", AB_IN_NAMELIST),
2130 : minit ("IN_COMMON", AB_IN_COMMON),
2131 : minit ("FUNCTION", AB_FUNCTION),
2132 : minit ("SUBROUTINE", AB_SUBROUTINE),
2133 : minit ("SEQUENCE", AB_SEQUENCE),
2134 : minit ("ELEMENTAL", AB_ELEMENTAL),
2135 : minit ("PURE", AB_PURE),
2136 : minit ("RECURSIVE", AB_RECURSIVE),
2137 : minit ("GENERIC", AB_GENERIC),
2138 : minit ("ALWAYS_EXPLICIT", AB_ALWAYS_EXPLICIT),
2139 : minit ("CRAY_POINTER", AB_CRAY_POINTER),
2140 : minit ("CRAY_POINTEE", AB_CRAY_POINTEE),
2141 : minit ("IS_BIND_C", AB_IS_BIND_C),
2142 : minit ("IS_C_INTEROP", AB_IS_C_INTEROP),
2143 : minit ("IS_ISO_C", AB_IS_ISO_C),
2144 : minit ("VALUE", AB_VALUE),
2145 : minit ("ALLOC_COMP", AB_ALLOC_COMP),
2146 : minit ("COARRAY_COMP", AB_COARRAY_COMP),
2147 : minit ("LOCK_COMP", AB_LOCK_COMP),
2148 : minit ("EVENT_COMP", AB_EVENT_COMP),
2149 : minit ("POINTER_COMP", AB_POINTER_COMP),
2150 : minit ("PROC_POINTER_COMP", AB_PROC_POINTER_COMP),
2151 : minit ("PRIVATE_COMP", AB_PRIVATE_COMP),
2152 : minit ("ZERO_COMP", AB_ZERO_COMP),
2153 : minit ("PROTECTED", AB_PROTECTED),
2154 : minit ("ABSTRACT", AB_ABSTRACT),
2155 : minit ("IS_CLASS", AB_IS_CLASS),
2156 : minit ("PROCEDURE", AB_PROCEDURE),
2157 : minit ("PROC_POINTER", AB_PROC_POINTER),
2158 : minit ("VTYPE", AB_VTYPE),
2159 : minit ("VTAB", AB_VTAB),
2160 : minit ("CLASS_POINTER", AB_CLASS_POINTER),
2161 : minit ("IMPLICIT_PURE", AB_IMPLICIT_PURE),
2162 : minit ("UNLIMITED_POLY", AB_UNLIMITED_POLY),
2163 : minit ("OMP_DECLARE_TARGET", AB_OMP_DECLARE_TARGET),
2164 : minit ("ARRAY_OUTER_DEPENDENCY", AB_ARRAY_OUTER_DEPENDENCY),
2165 : minit ("MODULE_PROCEDURE", AB_MODULE_PROCEDURE),
2166 : minit ("OACC_DECLARE_CREATE", AB_OACC_DECLARE_CREATE),
2167 : minit ("OACC_DECLARE_COPYIN", AB_OACC_DECLARE_COPYIN),
2168 : minit ("OACC_DECLARE_DEVICEPTR", AB_OACC_DECLARE_DEVICEPTR),
2169 : minit ("OACC_DECLARE_DEVICE_RESIDENT", AB_OACC_DECLARE_DEVICE_RESIDENT),
2170 : minit ("OACC_DECLARE_LINK", AB_OACC_DECLARE_LINK),
2171 : minit ("OMP_DECLARE_TARGET_LINK", AB_OMP_DECLARE_TARGET_LINK),
2172 : minit ("OMP_DECLARE_TARGET_LOCAL", AB_OMP_DECLARE_TARGET_LOCAL),
2173 : minit ("OMP_GROUPPRIVATE", AB_OMP_GROUPPRIVATE),
2174 : minit ("PDT_KIND", AB_PDT_KIND),
2175 : minit ("PDT_LEN", AB_PDT_LEN),
2176 : minit ("PDT_TYPE", AB_PDT_TYPE),
2177 : minit ("PDT_TEMPLATE", AB_PDT_TEMPLATE),
2178 : minit ("PDT_ARRAY", AB_PDT_ARRAY),
2179 : minit ("PDT_STRING", AB_PDT_STRING),
2180 : minit ("PDT_COMP", AB_PDT_COMP),
2181 : minit ("OACC_ROUTINE_LOP_GANG", AB_OACC_ROUTINE_LOP_GANG),
2182 : minit ("OACC_ROUTINE_LOP_WORKER", AB_OACC_ROUTINE_LOP_WORKER),
2183 : minit ("OACC_ROUTINE_LOP_VECTOR", AB_OACC_ROUTINE_LOP_VECTOR),
2184 : minit ("OACC_ROUTINE_LOP_SEQ", AB_OACC_ROUTINE_LOP_SEQ),
2185 : minit ("OACC_ROUTINE_NOHOST", AB_OACC_ROUTINE_NOHOST),
2186 : minit ("OMP_REQ_REVERSE_OFFLOAD", AB_OMP_REQ_REVERSE_OFFLOAD),
2187 : minit ("OMP_REQ_UNIFIED_ADDRESS", AB_OMP_REQ_UNIFIED_ADDRESS),
2188 : minit ("OMP_REQ_UNIFIED_SHARED_MEMORY", AB_OMP_REQ_UNIFIED_SHARED_MEMORY),
2189 : minit ("OMP_REQ_SELF_MAPS", AB_OMP_REQ_SELF_MAPS),
2190 : minit ("OMP_REQ_DYNAMIC_ALLOCATORS", AB_OMP_REQ_DYNAMIC_ALLOCATORS),
2191 : minit ("OMP_REQ_MEM_ORDER_SEQ_CST", AB_OMP_REQ_MEM_ORDER_SEQ_CST),
2192 : minit ("OMP_REQ_MEM_ORDER_ACQ_REL", AB_OMP_REQ_MEM_ORDER_ACQ_REL),
2193 : minit ("OMP_REQ_MEM_ORDER_ACQUIRE", AB_OMP_REQ_MEM_ORDER_ACQUIRE),
2194 : minit ("OMP_REQ_MEM_ORDER_RELAXED", AB_OMP_REQ_MEM_ORDER_RELAXED),
2195 : minit ("OMP_REQ_MEM_ORDER_RELEASE", AB_OMP_REQ_MEM_ORDER_RELEASE),
2196 : minit ("OMP_DEVICE_TYPE_HOST", AB_OMP_DEVICE_TYPE_HOST),
2197 : minit ("OMP_DEVICE_TYPE_NOHOST", AB_OMP_DEVICE_TYPE_NOHOST),
2198 : minit ("OMP_DEVICE_TYPE_ANYHOST", AB_OMP_DEVICE_TYPE_ANY),
2199 : minit (NULL, -1)
2200 : };
2201 :
2202 : /* For binding attributes. */
2203 : static const mstring binding_passing[] =
2204 : {
2205 : minit ("PASS", 0),
2206 : minit ("NOPASS", 1),
2207 : minit (NULL, -1)
2208 : };
2209 : static const mstring binding_overriding[] =
2210 : {
2211 : minit ("OVERRIDABLE", 0),
2212 : minit ("NON_OVERRIDABLE", 1),
2213 : minit ("DEFERRED", 2),
2214 : minit (NULL, -1)
2215 : };
2216 : static const mstring binding_generic[] =
2217 : {
2218 : minit ("SPECIFIC", 0),
2219 : minit ("GENERIC", 1),
2220 : minit (NULL, -1)
2221 : };
2222 : static const mstring binding_ppc[] =
2223 : {
2224 : minit ("NO_PPC", 0),
2225 : minit ("PPC", 1),
2226 : minit (NULL, -1)
2227 : };
2228 :
2229 : /* Specialization of mio_name. */
2230 618544 : DECL_MIO_NAME (ab_attribute)
2231 3614 : DECL_MIO_NAME (ar_type)
2232 165890 : DECL_MIO_NAME (array_type)
2233 7885592 : DECL_MIO_NAME (bt)
2234 101101 : DECL_MIO_NAME (expr_t)
2235 609178 : DECL_MIO_NAME (gfc_access)
2236 1629 : DECL_MIO_NAME (gfc_intrinsic_op)
2237 1532700 : DECL_MIO_NAME (ifsrc)
2238 1532700 : DECL_MIO_NAME (save_state)
2239 1532700 : DECL_MIO_NAME (procedure_type)
2240 5252 : DECL_MIO_NAME (ref_type)
2241 1532700 : DECL_MIO_NAME (sym_flavor)
2242 1532700 : DECL_MIO_NAME (sym_intent)
2243 0 : DECL_MIO_NAME (inquiry_type)
2244 : #undef DECL_MIO_NAME
2245 :
2246 : /* Verify OACC_ROUTINE_LOP_NONE. */
2247 :
2248 : static void
2249 96 : verify_OACC_ROUTINE_LOP_NONE (enum oacc_routine_lop lop)
2250 : {
2251 0 : if (lop != OACC_ROUTINE_LOP_NONE)
2252 0 : bad_module ("Unsupported: multiple OpenACC 'routine' levels of parallelism");
2253 0 : }
2254 :
2255 : /* Symbol attributes are stored in list with the first three elements
2256 : being the enumerated fields, while the remaining elements (if any)
2257 : indicate the individual attribute bits. The access field is not
2258 : saved-- it controls what symbols are exported when a module is
2259 : written. */
2260 :
2261 : static void
2262 1532700 : mio_symbol_attribute (symbol_attribute *attr)
2263 : {
2264 1532700 : atom_type t;
2265 1532700 : unsigned ext_attr,extension_level;
2266 :
2267 1532700 : mio_lparen ();
2268 :
2269 1532700 : attr->flavor = MIO_NAME (sym_flavor) (attr->flavor, flavors);
2270 1532700 : attr->intent = MIO_NAME (sym_intent) (attr->intent, intents);
2271 1532700 : attr->proc = MIO_NAME (procedure_type) (attr->proc, procedures);
2272 1532700 : attr->if_source = MIO_NAME (ifsrc) (attr->if_source, ifsrc_types);
2273 1532700 : attr->save = MIO_NAME (save_state) (attr->save, save_status);
2274 :
2275 1532700 : ext_attr = attr->ext_attr;
2276 1532700 : mio_integer ((int *) &ext_attr);
2277 1532700 : attr->ext_attr = ext_attr;
2278 :
2279 1532700 : extension_level = attr->extension;
2280 1532700 : mio_integer ((int *) &extension_level);
2281 1532700 : attr->extension = extension_level;
2282 :
2283 1532700 : if (iomode == IO_OUTPUT)
2284 : {
2285 345977 : if (attr->allocatable)
2286 4900 : MIO_NAME (ab_attribute) (AB_ALLOCATABLE, attr_bits);
2287 345977 : if (attr->artificial)
2288 101543 : MIO_NAME (ab_attribute) (AB_ARTIFICIAL, attr_bits);
2289 345977 : if (attr->asynchronous)
2290 0 : MIO_NAME (ab_attribute) (AB_ASYNCHRONOUS, attr_bits);
2291 345977 : if (attr->dimension)
2292 18541 : MIO_NAME (ab_attribute) (AB_DIMENSION, attr_bits);
2293 345977 : if (attr->codimension)
2294 107 : MIO_NAME (ab_attribute) (AB_CODIMENSION, attr_bits);
2295 345977 : if (attr->contiguous)
2296 3168 : MIO_NAME (ab_attribute) (AB_CONTIGUOUS, attr_bits);
2297 345977 : if (attr->external)
2298 15173 : MIO_NAME (ab_attribute) (AB_EXTERNAL, attr_bits);
2299 345977 : if (attr->intrinsic)
2300 5570 : MIO_NAME (ab_attribute) (AB_INTRINSIC, attr_bits);
2301 345977 : if (attr->optional)
2302 5165 : MIO_NAME (ab_attribute) (AB_OPTIONAL, attr_bits);
2303 345977 : if (attr->pointer)
2304 32183 : MIO_NAME (ab_attribute) (AB_POINTER, attr_bits);
2305 345977 : if (attr->class_pointer)
2306 477 : MIO_NAME (ab_attribute) (AB_CLASS_POINTER, attr_bits);
2307 345977 : if (attr->is_protected)
2308 70 : MIO_NAME (ab_attribute) (AB_PROTECTED, attr_bits);
2309 345977 : if (attr->value)
2310 10764 : MIO_NAME (ab_attribute) (AB_VALUE, attr_bits);
2311 345977 : if (attr->volatile_)
2312 14 : MIO_NAME (ab_attribute) (AB_VOLATILE, attr_bits);
2313 345977 : if (attr->target)
2314 19571 : MIO_NAME (ab_attribute) (AB_TARGET, attr_bits);
2315 345977 : if (attr->threadprivate)
2316 42 : MIO_NAME (ab_attribute) (AB_THREADPRIVATE, attr_bits);
2317 345977 : if (attr->dummy)
2318 82046 : MIO_NAME (ab_attribute) (AB_DUMMY, attr_bits);
2319 345977 : if (attr->result)
2320 7100 : MIO_NAME (ab_attribute) (AB_RESULT, attr_bits);
2321 : /* We deliberately don't preserve the "entry" flag. */
2322 :
2323 345977 : if (attr->data)
2324 22 : MIO_NAME (ab_attribute) (AB_DATA, attr_bits);
2325 345977 : if (attr->in_namelist)
2326 78 : MIO_NAME (ab_attribute) (AB_IN_NAMELIST, attr_bits);
2327 345977 : if (attr->in_common)
2328 392 : MIO_NAME (ab_attribute) (AB_IN_COMMON, attr_bits);
2329 :
2330 345977 : if (attr->function)
2331 33876 : MIO_NAME (ab_attribute) (AB_FUNCTION, attr_bits);
2332 345977 : if (attr->subroutine)
2333 29331 : MIO_NAME (ab_attribute) (AB_SUBROUTINE, attr_bits);
2334 345977 : if (attr->generic)
2335 9368 : MIO_NAME (ab_attribute) (AB_GENERIC, attr_bits);
2336 345977 : if (attr->abstract)
2337 3017 : MIO_NAME (ab_attribute) (AB_ABSTRACT, attr_bits);
2338 :
2339 345977 : if (attr->sequence)
2340 125 : MIO_NAME (ab_attribute) (AB_SEQUENCE, attr_bits);
2341 345977 : if (attr->elemental)
2342 15678 : MIO_NAME (ab_attribute) (AB_ELEMENTAL, attr_bits);
2343 345977 : if (attr->pure)
2344 19183 : MIO_NAME (ab_attribute) (AB_PURE, attr_bits);
2345 345977 : if (attr->implicit_pure)
2346 4223 : MIO_NAME (ab_attribute) (AB_IMPLICIT_PURE, attr_bits);
2347 345977 : if (attr->unlimited_polymorphic)
2348 346 : MIO_NAME (ab_attribute) (AB_UNLIMITED_POLY, attr_bits);
2349 345977 : if (attr->recursive)
2350 2888 : MIO_NAME (ab_attribute) (AB_RECURSIVE, attr_bits);
2351 345977 : if (attr->always_explicit)
2352 29996 : MIO_NAME (ab_attribute) (AB_ALWAYS_EXPLICIT, attr_bits);
2353 345977 : if (attr->cray_pointer)
2354 13 : MIO_NAME (ab_attribute) (AB_CRAY_POINTER, attr_bits);
2355 345977 : if (attr->cray_pointee)
2356 13 : MIO_NAME (ab_attribute) (AB_CRAY_POINTEE, attr_bits);
2357 345977 : if (attr->is_bind_c)
2358 6454 : MIO_NAME(ab_attribute) (AB_IS_BIND_C, attr_bits);
2359 345977 : if (attr->is_c_interop)
2360 30340 : MIO_NAME(ab_attribute) (AB_IS_C_INTEROP, attr_bits);
2361 345977 : if (attr->is_iso_c)
2362 26607 : MIO_NAME(ab_attribute) (AB_IS_ISO_C, attr_bits);
2363 345977 : if (attr->alloc_comp)
2364 2946 : MIO_NAME (ab_attribute) (AB_ALLOC_COMP, attr_bits);
2365 345977 : if (attr->pointer_comp)
2366 946 : MIO_NAME (ab_attribute) (AB_POINTER_COMP, attr_bits);
2367 345977 : if (attr->proc_pointer_comp)
2368 263 : MIO_NAME (ab_attribute) (AB_PROC_POINTER_COMP, attr_bits);
2369 345977 : if (attr->private_comp)
2370 3279 : MIO_NAME (ab_attribute) (AB_PRIVATE_COMP, attr_bits);
2371 345977 : if (attr->coarray_comp)
2372 33 : MIO_NAME (ab_attribute) (AB_COARRAY_COMP, attr_bits);
2373 345977 : if (attr->lock_comp)
2374 4 : MIO_NAME (ab_attribute) (AB_LOCK_COMP, attr_bits);
2375 345977 : if (attr->event_comp)
2376 0 : MIO_NAME (ab_attribute) (AB_EVENT_COMP, attr_bits);
2377 345977 : if (attr->zero_comp)
2378 2183 : MIO_NAME (ab_attribute) (AB_ZERO_COMP, attr_bits);
2379 345977 : if (attr->is_class)
2380 4706 : MIO_NAME (ab_attribute) (AB_IS_CLASS, attr_bits);
2381 345977 : if (attr->procedure)
2382 5734 : MIO_NAME (ab_attribute) (AB_PROCEDURE, attr_bits);
2383 345977 : if (attr->proc_pointer)
2384 37144 : MIO_NAME (ab_attribute) (AB_PROC_POINTER, attr_bits);
2385 345977 : if (attr->vtype)
2386 10603 : MIO_NAME (ab_attribute) (AB_VTYPE, attr_bits);
2387 345977 : if (attr->vtab)
2388 10058 : MIO_NAME (ab_attribute) (AB_VTAB, attr_bits);
2389 345977 : if (attr->omp_declare_target)
2390 423 : MIO_NAME (ab_attribute) (AB_OMP_DECLARE_TARGET, attr_bits);
2391 345977 : if (attr->array_outer_dependency)
2392 17825 : MIO_NAME (ab_attribute) (AB_ARRAY_OUTER_DEPENDENCY, attr_bits);
2393 345977 : if (attr->module_procedure)
2394 1632 : MIO_NAME (ab_attribute) (AB_MODULE_PROCEDURE, attr_bits);
2395 345977 : if (attr->oacc_declare_create)
2396 39 : MIO_NAME (ab_attribute) (AB_OACC_DECLARE_CREATE, attr_bits);
2397 345977 : if (attr->oacc_declare_copyin)
2398 7 : MIO_NAME (ab_attribute) (AB_OACC_DECLARE_COPYIN, attr_bits);
2399 345977 : if (attr->oacc_declare_deviceptr)
2400 1 : MIO_NAME (ab_attribute) (AB_OACC_DECLARE_DEVICEPTR, attr_bits);
2401 345977 : if (attr->oacc_declare_device_resident)
2402 33 : MIO_NAME (ab_attribute) (AB_OACC_DECLARE_DEVICE_RESIDENT, attr_bits);
2403 345977 : if (attr->oacc_declare_link)
2404 1 : MIO_NAME (ab_attribute) (AB_OACC_DECLARE_LINK, attr_bits);
2405 345977 : if (attr->omp_declare_target_link)
2406 15 : MIO_NAME (ab_attribute) (AB_OMP_DECLARE_TARGET_LINK, attr_bits);
2407 345977 : if (attr->omp_declare_target_local)
2408 12 : MIO_NAME (ab_attribute) (AB_OMP_DECLARE_TARGET_LOCAL, attr_bits);
2409 345977 : if (attr->omp_groupprivate)
2410 12 : MIO_NAME (ab_attribute) (AB_OMP_GROUPPRIVATE, attr_bits);
2411 345977 : if (attr->pdt_kind)
2412 682 : MIO_NAME (ab_attribute) (AB_PDT_KIND, attr_bits);
2413 345977 : if (attr->pdt_len)
2414 398 : MIO_NAME (ab_attribute) (AB_PDT_LEN, attr_bits);
2415 345977 : if (attr->pdt_type)
2416 324 : MIO_NAME (ab_attribute) (AB_PDT_TYPE, attr_bits);
2417 345977 : if (attr->pdt_comp)
2418 27 : MIO_NAME (ab_attribute) (AB_PDT_COMP , attr_bits);
2419 345977 : if (attr->pdt_template)
2420 283 : MIO_NAME (ab_attribute) (AB_PDT_TEMPLATE, attr_bits);
2421 345977 : if (attr->pdt_array)
2422 69 : MIO_NAME (ab_attribute) (AB_PDT_ARRAY, attr_bits);
2423 345977 : if (attr->pdt_string)
2424 2 : MIO_NAME (ab_attribute) (AB_PDT_STRING, attr_bits);
2425 345977 : switch (attr->oacc_routine_lop)
2426 : {
2427 : case OACC_ROUTINE_LOP_NONE:
2428 : /* This is the default anyway, and for maintaining compatibility with
2429 : the current MOD_VERSION, we're not emitting anything in that
2430 : case. */
2431 : break;
2432 12 : case OACC_ROUTINE_LOP_GANG:
2433 12 : MIO_NAME (ab_attribute) (AB_OACC_ROUTINE_LOP_GANG, attr_bits);
2434 12 : break;
2435 10 : case OACC_ROUTINE_LOP_WORKER:
2436 10 : MIO_NAME (ab_attribute) (AB_OACC_ROUTINE_LOP_WORKER, attr_bits);
2437 10 : break;
2438 8 : case OACC_ROUTINE_LOP_VECTOR:
2439 8 : MIO_NAME (ab_attribute) (AB_OACC_ROUTINE_LOP_VECTOR, attr_bits);
2440 8 : break;
2441 81 : case OACC_ROUTINE_LOP_SEQ:
2442 81 : MIO_NAME (ab_attribute) (AB_OACC_ROUTINE_LOP_SEQ, attr_bits);
2443 81 : break;
2444 0 : case OACC_ROUTINE_LOP_ERROR:
2445 : /* ... intentionally omitted here; it's only used internally. */
2446 0 : default:
2447 0 : gcc_unreachable ();
2448 : }
2449 345977 : if (attr->oacc_routine_nohost)
2450 21 : MIO_NAME (ab_attribute) (AB_OACC_ROUTINE_NOHOST, attr_bits);
2451 :
2452 345977 : if (attr->flavor == FL_MODULE && gfc_current_ns->omp_requires)
2453 : {
2454 29 : if (gfc_current_ns->omp_requires & OMP_REQ_REVERSE_OFFLOAD)
2455 15 : MIO_NAME (ab_attribute) (AB_OMP_REQ_REVERSE_OFFLOAD, attr_bits);
2456 29 : if (gfc_current_ns->omp_requires & OMP_REQ_UNIFIED_ADDRESS)
2457 1 : MIO_NAME (ab_attribute) (AB_OMP_REQ_UNIFIED_ADDRESS, attr_bits);
2458 29 : if (gfc_current_ns->omp_requires & OMP_REQ_UNIFIED_SHARED_MEMORY)
2459 3 : MIO_NAME (ab_attribute) (AB_OMP_REQ_UNIFIED_SHARED_MEMORY, attr_bits);
2460 29 : if (gfc_current_ns->omp_requires & OMP_REQ_SELF_MAPS)
2461 1 : MIO_NAME (ab_attribute) (AB_OMP_REQ_SELF_MAPS, attr_bits);
2462 29 : if (gfc_current_ns->omp_requires & OMP_REQ_DYNAMIC_ALLOCATORS)
2463 2 : MIO_NAME (ab_attribute) (AB_OMP_REQ_DYNAMIC_ALLOCATORS, attr_bits);
2464 29 : if ((gfc_current_ns->omp_requires & OMP_REQ_ATOMIC_MEM_ORDER_MASK)
2465 : == OMP_REQ_ATOMIC_MEM_ORDER_SEQ_CST)
2466 3 : MIO_NAME (ab_attribute) (AB_OMP_REQ_MEM_ORDER_SEQ_CST, attr_bits);
2467 29 : if ((gfc_current_ns->omp_requires & OMP_REQ_ATOMIC_MEM_ORDER_MASK)
2468 : == OMP_REQ_ATOMIC_MEM_ORDER_ACQ_REL)
2469 3 : MIO_NAME (ab_attribute) (AB_OMP_REQ_MEM_ORDER_ACQ_REL, attr_bits);
2470 29 : if ((gfc_current_ns->omp_requires & OMP_REQ_ATOMIC_MEM_ORDER_MASK)
2471 : == OMP_REQ_ATOMIC_MEM_ORDER_ACQUIRE)
2472 0 : MIO_NAME (ab_attribute) (AB_OMP_REQ_MEM_ORDER_ACQUIRE, attr_bits);
2473 29 : if ((gfc_current_ns->omp_requires & OMP_REQ_ATOMIC_MEM_ORDER_MASK)
2474 : == OMP_REQ_ATOMIC_MEM_ORDER_RELAXED)
2475 3 : MIO_NAME (ab_attribute) (AB_OMP_REQ_MEM_ORDER_RELAXED, attr_bits);
2476 29 : if ((gfc_current_ns->omp_requires & OMP_REQ_ATOMIC_MEM_ORDER_MASK)
2477 : == OMP_REQ_ATOMIC_MEM_ORDER_RELEASE)
2478 0 : MIO_NAME (ab_attribute) (AB_OMP_REQ_MEM_ORDER_RELEASE, attr_bits);
2479 : }
2480 345977 : switch (attr->omp_device_type)
2481 : {
2482 : case OMP_DEVICE_TYPE_UNSET:
2483 : break;
2484 16 : case OMP_DEVICE_TYPE_HOST:
2485 16 : MIO_NAME (ab_attribute) (AB_OMP_DEVICE_TYPE_HOST, attr_bits);
2486 16 : break;
2487 9 : case OMP_DEVICE_TYPE_NOHOST:
2488 9 : MIO_NAME (ab_attribute) (AB_OMP_DEVICE_TYPE_NOHOST, attr_bits);
2489 9 : break;
2490 288 : case OMP_DEVICE_TYPE_ANY:
2491 288 : MIO_NAME (ab_attribute) (AB_OMP_DEVICE_TYPE_ANY, attr_bits);
2492 288 : break;
2493 0 : default:
2494 0 : gcc_unreachable ();
2495 : }
2496 345977 : mio_rparen ();
2497 : }
2498 : else
2499 : {
2500 3334519 : for (;;)
2501 : {
2502 3334519 : t = parse_atom ();
2503 3334519 : if (t == ATOM_RPAREN)
2504 : break;
2505 2147796 : if (t != ATOM_NAME)
2506 0 : bad_module ("Expected attribute bit name");
2507 :
2508 2147796 : switch ((ab_attribute) find_enum (attr_bits))
2509 : {
2510 4763 : case AB_ALLOCATABLE:
2511 4763 : attr->allocatable = 1;
2512 4763 : break;
2513 141433 : case AB_ARTIFICIAL:
2514 141433 : attr->artificial = 1;
2515 141433 : break;
2516 0 : case AB_ASYNCHRONOUS:
2517 0 : attr->asynchronous = 1;
2518 0 : break;
2519 64256 : case AB_DIMENSION:
2520 64256 : attr->dimension = 1;
2521 64256 : break;
2522 103 : case AB_CODIMENSION:
2523 103 : attr->codimension = 1;
2524 103 : break;
2525 7862 : case AB_CONTIGUOUS:
2526 7862 : attr->contiguous = 1;
2527 7862 : break;
2528 185502 : case AB_EXTERNAL:
2529 185502 : attr->external = 1;
2530 185502 : break;
2531 3654 : case AB_INTRINSIC:
2532 3654 : attr->intrinsic = 1;
2533 3654 : break;
2534 9268 : case AB_OPTIONAL:
2535 9268 : attr->optional = 1;
2536 9268 : break;
2537 43997 : case AB_POINTER:
2538 43997 : attr->pointer = 1;
2539 43997 : break;
2540 417 : case AB_CLASS_POINTER:
2541 417 : attr->class_pointer = 1;
2542 417 : break;
2543 62 : case AB_PROTECTED:
2544 62 : attr->is_protected = 1;
2545 62 : break;
2546 80360 : case AB_VALUE:
2547 80360 : attr->value = 1;
2548 80360 : break;
2549 15 : case AB_VOLATILE:
2550 15 : attr->volatile_ = 1;
2551 15 : break;
2552 27557 : case AB_TARGET:
2553 27557 : attr->target = 1;
2554 27557 : break;
2555 52 : case AB_THREADPRIVATE:
2556 52 : attr->threadprivate = 1;
2557 52 : break;
2558 415095 : case AB_DUMMY:
2559 415095 : attr->dummy = 1;
2560 415095 : break;
2561 33937 : case AB_RESULT:
2562 33937 : attr->result = 1;
2563 33937 : break;
2564 22 : case AB_DATA:
2565 22 : attr->data = 1;
2566 22 : break;
2567 83 : case AB_IN_NAMELIST:
2568 83 : attr->in_namelist = 1;
2569 83 : break;
2570 301 : case AB_IN_COMMON:
2571 301 : attr->in_common = 1;
2572 301 : break;
2573 183378 : case AB_FUNCTION:
2574 183378 : attr->function = 1;
2575 183378 : break;
2576 83832 : case AB_SUBROUTINE:
2577 83832 : attr->subroutine = 1;
2578 83832 : break;
2579 25164 : case AB_GENERIC:
2580 25164 : attr->generic = 1;
2581 25164 : break;
2582 2284 : case AB_ABSTRACT:
2583 2284 : attr->abstract = 1;
2584 2284 : break;
2585 137 : case AB_SEQUENCE:
2586 137 : attr->sequence = 1;
2587 137 : break;
2588 92962 : case AB_ELEMENTAL:
2589 92962 : attr->elemental = 1;
2590 92962 : break;
2591 118401 : case AB_PURE:
2592 118401 : attr->pure = 1;
2593 118401 : break;
2594 4237 : case AB_IMPLICIT_PURE:
2595 4237 : attr->implicit_pure = 1;
2596 4237 : break;
2597 311 : case AB_UNLIMITED_POLY:
2598 311 : attr->unlimited_polymorphic = 1;
2599 311 : break;
2600 2868 : case AB_RECURSIVE:
2601 2868 : attr->recursive = 1;
2602 2868 : break;
2603 154889 : case AB_ALWAYS_EXPLICIT:
2604 154889 : attr->always_explicit = 1;
2605 154889 : break;
2606 13 : case AB_CRAY_POINTER:
2607 13 : attr->cray_pointer = 1;
2608 13 : break;
2609 13 : case AB_CRAY_POINTEE:
2610 13 : attr->cray_pointee = 1;
2611 13 : break;
2612 40876 : case AB_IS_BIND_C:
2613 40876 : attr->is_bind_c = 1;
2614 40876 : break;
2615 73248 : case AB_IS_C_INTEROP:
2616 73248 : attr->is_c_interop = 1;
2617 73248 : break;
2618 28296 : case AB_IS_ISO_C:
2619 28296 : attr->is_iso_c = 1;
2620 28296 : break;
2621 2781 : case AB_ALLOC_COMP:
2622 2781 : attr->alloc_comp = 1;
2623 2781 : break;
2624 22 : case AB_COARRAY_COMP:
2625 22 : attr->coarray_comp = 1;
2626 22 : break;
2627 4 : case AB_LOCK_COMP:
2628 4 : attr->lock_comp = 1;
2629 4 : break;
2630 0 : case AB_EVENT_COMP:
2631 0 : attr->event_comp = 1;
2632 0 : break;
2633 884 : case AB_POINTER_COMP:
2634 884 : attr->pointer_comp = 1;
2635 884 : break;
2636 251 : case AB_PROC_POINTER_COMP:
2637 251 : attr->proc_pointer_comp = 1;
2638 251 : break;
2639 20037 : case AB_PRIVATE_COMP:
2640 20037 : attr->private_comp = 1;
2641 20037 : break;
2642 1829 : case AB_ZERO_COMP:
2643 1829 : attr->zero_comp = 1;
2644 1829 : break;
2645 4180 : case AB_IS_CLASS:
2646 4180 : attr->is_class = 1;
2647 4180 : break;
2648 5107 : case AB_PROCEDURE:
2649 5107 : attr->procedure = 1;
2650 5107 : break;
2651 49931 : case AB_PROC_POINTER:
2652 49931 : attr->proc_pointer = 1;
2653 49931 : break;
2654 15051 : case AB_VTYPE:
2655 15051 : attr->vtype = 1;
2656 15051 : break;
2657 14739 : case AB_VTAB:
2658 14739 : attr->vtab = 1;
2659 14739 : break;
2660 444 : case AB_OMP_DECLARE_TARGET:
2661 444 : attr->omp_declare_target = 1;
2662 444 : break;
2663 10 : case AB_OMP_DECLARE_TARGET_LINK:
2664 10 : attr->omp_declare_target_link = 1;
2665 10 : break;
2666 0 : case AB_OMP_DECLARE_TARGET_LOCAL:
2667 0 : attr->omp_declare_target_local = 1;
2668 0 : break;
2669 0 : case AB_OMP_GROUPPRIVATE:
2670 0 : attr->omp_groupprivate = 1;
2671 0 : break;
2672 199690 : case AB_ARRAY_OUTER_DEPENDENCY:
2673 199690 : attr->array_outer_dependency =1;
2674 199690 : break;
2675 1090 : case AB_MODULE_PROCEDURE:
2676 1090 : attr->module_procedure =1;
2677 1090 : break;
2678 110 : case AB_OACC_DECLARE_CREATE:
2679 110 : attr->oacc_declare_create = 1;
2680 110 : break;
2681 2 : case AB_OACC_DECLARE_COPYIN:
2682 2 : attr->oacc_declare_copyin = 1;
2683 2 : break;
2684 0 : case AB_OACC_DECLARE_DEVICEPTR:
2685 0 : attr->oacc_declare_deviceptr = 1;
2686 0 : break;
2687 34 : case AB_OACC_DECLARE_DEVICE_RESIDENT:
2688 34 : attr->oacc_declare_device_resident = 1;
2689 34 : break;
2690 2 : case AB_OACC_DECLARE_LINK:
2691 2 : attr->oacc_declare_link = 1;
2692 2 : break;
2693 514 : case AB_PDT_KIND:
2694 514 : attr->pdt_kind = 1;
2695 514 : break;
2696 388 : case AB_PDT_LEN:
2697 388 : attr->pdt_len = 1;
2698 388 : break;
2699 266 : case AB_PDT_TYPE:
2700 266 : attr->pdt_type = 1;
2701 266 : break;
2702 16 : case AB_PDT_COMP:
2703 16 : attr->pdt_comp = 1;
2704 16 : break;
2705 240 : case AB_PDT_TEMPLATE:
2706 240 : attr->pdt_template = 1;
2707 240 : break;
2708 69 : case AB_PDT_ARRAY:
2709 69 : attr->pdt_array = 1;
2710 69 : break;
2711 0 : case AB_PDT_STRING:
2712 0 : attr->pdt_string = 1;
2713 0 : break;
2714 8 : case AB_OACC_ROUTINE_LOP_GANG:
2715 8 : verify_OACC_ROUTINE_LOP_NONE (attr->oacc_routine_lop);
2716 8 : attr->oacc_routine_lop = OACC_ROUTINE_LOP_GANG;
2717 8 : break;
2718 8 : case AB_OACC_ROUTINE_LOP_WORKER:
2719 8 : verify_OACC_ROUTINE_LOP_NONE (attr->oacc_routine_lop);
2720 8 : attr->oacc_routine_lop = OACC_ROUTINE_LOP_WORKER;
2721 8 : break;
2722 8 : case AB_OACC_ROUTINE_LOP_VECTOR:
2723 8 : verify_OACC_ROUTINE_LOP_NONE (attr->oacc_routine_lop);
2724 8 : attr->oacc_routine_lop = OACC_ROUTINE_LOP_VECTOR;
2725 8 : break;
2726 72 : case AB_OACC_ROUTINE_LOP_SEQ:
2727 72 : verify_OACC_ROUTINE_LOP_NONE (attr->oacc_routine_lop);
2728 72 : attr->oacc_routine_lop = OACC_ROUTINE_LOP_SEQ;
2729 72 : break;
2730 20 : case AB_OACC_ROUTINE_NOHOST:
2731 20 : attr->oacc_routine_nohost = 1;
2732 20 : break;
2733 24 : case AB_OMP_REQ_REVERSE_OFFLOAD:
2734 24 : gfc_omp_requires_add_clause (OMP_REQ_REVERSE_OFFLOAD,
2735 : "reverse_offload",
2736 : &gfc_current_locus,
2737 : module_name);
2738 24 : break;
2739 0 : case AB_OMP_REQ_UNIFIED_ADDRESS:
2740 0 : gfc_omp_requires_add_clause (OMP_REQ_UNIFIED_ADDRESS,
2741 : "unified_address",
2742 : &gfc_current_locus,
2743 : module_name);
2744 0 : break;
2745 0 : case AB_OMP_REQ_UNIFIED_SHARED_MEMORY:
2746 0 : gfc_omp_requires_add_clause (OMP_REQ_UNIFIED_SHARED_MEMORY,
2747 : "unified_shared_memory",
2748 : &gfc_current_locus,
2749 : module_name);
2750 0 : break;
2751 1 : case AB_OMP_REQ_SELF_MAPS:
2752 1 : gfc_omp_requires_add_clause (OMP_REQ_SELF_MAPS,
2753 : "self_maps",
2754 : &gfc_current_locus,
2755 : module_name);
2756 1 : break;
2757 0 : case AB_OMP_REQ_DYNAMIC_ALLOCATORS:
2758 0 : gfc_omp_requires_add_clause (OMP_REQ_DYNAMIC_ALLOCATORS,
2759 : "dynamic_allocators",
2760 : &gfc_current_locus,
2761 : module_name);
2762 0 : break;
2763 2 : case AB_OMP_REQ_MEM_ORDER_SEQ_CST:
2764 2 : gfc_omp_requires_add_clause (OMP_REQ_ATOMIC_MEM_ORDER_SEQ_CST,
2765 : "seq_cst", &gfc_current_locus,
2766 : module_name);
2767 2 : break;
2768 2 : case AB_OMP_REQ_MEM_ORDER_ACQ_REL:
2769 2 : gfc_omp_requires_add_clause (OMP_REQ_ATOMIC_MEM_ORDER_ACQ_REL,
2770 : "acq_rel", &gfc_current_locus,
2771 : module_name);
2772 2 : break;
2773 0 : case AB_OMP_REQ_MEM_ORDER_ACQUIRE:
2774 0 : gfc_omp_requires_add_clause (OMP_REQ_ATOMIC_MEM_ORDER_ACQUIRE,
2775 : "acquires", &gfc_current_locus,
2776 : module_name);
2777 0 : break;
2778 2 : case AB_OMP_REQ_MEM_ORDER_RELAXED:
2779 2 : gfc_omp_requires_add_clause (OMP_REQ_ATOMIC_MEM_ORDER_RELAXED,
2780 : "relaxed", &gfc_current_locus,
2781 : module_name);
2782 2 : break;
2783 0 : case AB_OMP_REQ_MEM_ORDER_RELEASE:
2784 0 : gfc_omp_requires_add_clause (OMP_REQ_ATOMIC_MEM_ORDER_RELEASE,
2785 : "release", &gfc_current_locus,
2786 : module_name);
2787 0 : break;
2788 8 : case AB_OMP_DEVICE_TYPE_HOST:
2789 8 : attr->omp_device_type = OMP_DEVICE_TYPE_HOST;
2790 8 : break;
2791 5 : case AB_OMP_DEVICE_TYPE_NOHOST:
2792 5 : attr->omp_device_type = OMP_DEVICE_TYPE_NOHOST;
2793 5 : break;
2794 297 : case AB_OMP_DEVICE_TYPE_ANY:
2795 297 : attr->omp_device_type = OMP_DEVICE_TYPE_ANY;
2796 297 : break;
2797 : }
2798 : }
2799 : }
2800 1532700 : }
2801 :
2802 :
2803 : static const mstring bt_types[] = {
2804 : minit ("INTEGER", BT_INTEGER),
2805 : minit ("REAL", BT_REAL),
2806 : minit ("COMPLEX", BT_COMPLEX),
2807 : minit ("LOGICAL", BT_LOGICAL),
2808 : minit ("CHARACTER", BT_CHARACTER),
2809 : minit ("UNION", BT_UNION),
2810 : minit ("DERIVED", BT_DERIVED),
2811 : minit ("CLASS", BT_CLASS),
2812 : minit ("PROCEDURE", BT_PROCEDURE),
2813 : minit ("UNKNOWN", BT_UNKNOWN),
2814 : minit ("VOID", BT_VOID),
2815 : minit ("ASSUMED", BT_ASSUMED),
2816 : minit ("UNSIGNED", BT_UNSIGNED),
2817 : minit (NULL, -1)
2818 : };
2819 :
2820 :
2821 : static void
2822 41358 : mio_charlen (gfc_charlen **clp)
2823 : {
2824 41358 : gfc_charlen *cl;
2825 :
2826 41358 : mio_lparen ();
2827 :
2828 41358 : if (iomode == IO_OUTPUT)
2829 : {
2830 19467 : cl = *clp;
2831 19467 : if (cl != NULL)
2832 18703 : mio_expr (&cl->length);
2833 : }
2834 : else
2835 : {
2836 21891 : if (peek_atom () != ATOM_RPAREN)
2837 : {
2838 21427 : cl = gfc_new_charlen (gfc_current_ns, NULL);
2839 21427 : mio_expr (&cl->length);
2840 21427 : *clp = cl;
2841 : }
2842 : }
2843 :
2844 41358 : mio_rparen ();
2845 41358 : }
2846 :
2847 :
2848 : /* See if a name is a generated name. */
2849 :
2850 : static int
2851 761036 : check_unique_name (const char *name)
2852 : {
2853 761036 : return *name == '@';
2854 : }
2855 :
2856 :
2857 : static void
2858 1971398 : mio_typespec (gfc_typespec *ts)
2859 : {
2860 1971398 : mio_lparen ();
2861 :
2862 1971398 : ts->type = MIO_NAME (bt) (ts->type, bt_types);
2863 :
2864 1971398 : if (!gfc_bt_struct (ts->type) && ts->type != BT_CLASS)
2865 1667318 : mio_integer (&ts->kind);
2866 : else
2867 304080 : mio_symbol_ref (&ts->u.derived);
2868 :
2869 1971398 : mio_symbol_ref (&ts->interface);
2870 :
2871 : /* Add info for C interop and is_iso_c. */
2872 1971398 : mio_integer (&ts->is_c_interop);
2873 1971398 : mio_integer (&ts->is_iso_c);
2874 :
2875 : /* If the typespec is for an identifier either from iso_c_binding, or
2876 : a constant that was initialized to an identifier from it, use the
2877 : f90_type. Otherwise, use the ts->type, since it shouldn't matter. */
2878 1971398 : if (ts->is_iso_c)
2879 125563 : ts->f90_type = MIO_NAME (bt) (ts->f90_type, bt_types);
2880 : else
2881 1845835 : ts->f90_type = MIO_NAME (bt) (ts->type, bt_types);
2882 :
2883 1971398 : if (ts->type != BT_CHARACTER)
2884 : {
2885 : /* ts->u.cl is only valid for BT_CHARACTER. */
2886 1930046 : mio_lparen ();
2887 1930046 : mio_rparen ();
2888 : }
2889 : else
2890 41352 : mio_charlen (&ts->u.cl);
2891 :
2892 : /* So as not to disturb the existing API, use an ATOM_NAME to
2893 : transmit deferred characteristic for characters (F2003). */
2894 1971398 : if (iomode == IO_OUTPUT)
2895 : {
2896 447167 : if (ts->type == BT_CHARACTER && ts->deferred)
2897 728 : write_atom (ATOM_NAME, "DEFERRED_CL");
2898 : }
2899 1524231 : else if (peek_atom () != ATOM_RPAREN)
2900 : {
2901 4831 : if (parse_atom () != ATOM_NAME)
2902 0 : bad_module ("Expected string");
2903 4831 : ts->deferred = 1;
2904 : }
2905 :
2906 1971398 : mio_rparen ();
2907 1971398 : }
2908 :
2909 :
2910 : static const mstring array_spec_types[] = {
2911 : minit ("EXPLICIT", AS_EXPLICIT),
2912 : minit ("ASSUMED_RANK", AS_ASSUMED_RANK),
2913 : minit ("ASSUMED_SHAPE", AS_ASSUMED_SHAPE),
2914 : minit ("DEFERRED", AS_DEFERRED),
2915 : minit ("ASSUMED_SIZE", AS_ASSUMED_SIZE),
2916 : minit (NULL, -1)
2917 : };
2918 :
2919 :
2920 : static void
2921 1532680 : mio_array_spec (gfc_array_spec **asp)
2922 : {
2923 1532680 : gfc_array_spec *as;
2924 1532680 : int i;
2925 :
2926 1532680 : mio_lparen ();
2927 :
2928 1532680 : if (iomode == IO_OUTPUT)
2929 : {
2930 345977 : int rank;
2931 :
2932 345977 : if (*asp == NULL)
2933 327365 : goto done;
2934 18612 : as = *asp;
2935 :
2936 : /* mio_integer expects nonnegative values. */
2937 18612 : rank = as->rank > 0 ? as->rank : 0;
2938 18612 : mio_integer (&rank);
2939 : }
2940 : else
2941 : {
2942 1186703 : if (peek_atom () == ATOM_RPAREN)
2943 : {
2944 1122370 : *asp = NULL;
2945 1122370 : goto done;
2946 : }
2947 :
2948 64333 : *asp = as = gfc_get_array_spec ();
2949 64333 : mio_integer (&as->rank);
2950 : }
2951 :
2952 82945 : mio_integer (&as->corank);
2953 82945 : as->type = MIO_NAME (array_type) (as->type, array_spec_types);
2954 :
2955 82945 : if (iomode == IO_INPUT && as->type == AS_ASSUMED_RANK)
2956 23739 : as->rank = -1;
2957 82945 : if (iomode == IO_INPUT && as->corank)
2958 166 : as->cotype = (as->type == AS_DEFERRED) ? AS_DEFERRED : AS_EXPLICIT;
2959 :
2960 82945 : if (as->rank + as->corank > 0)
2961 115868 : for (i = 0; i < as->rank + as->corank; i++)
2962 : {
2963 60684 : mio_expr (&as->lower[i]);
2964 60684 : mio_expr (&as->upper[i]);
2965 : }
2966 :
2967 27761 : done:
2968 1532680 : mio_rparen ();
2969 1532680 : }
2970 :
2971 :
2972 : /* Given a pointer to an array reference structure (which lives in a
2973 : gfc_ref structure), find the corresponding array specification
2974 : structure. Storing the pointer in the ref structure doesn't quite
2975 : work when loading from a module. Generating code for an array
2976 : reference also needs more information than just the array spec. */
2977 :
2978 : static const mstring array_ref_types[] = {
2979 : minit ("FULL", AR_FULL),
2980 : minit ("ELEMENT", AR_ELEMENT),
2981 : minit ("SECTION", AR_SECTION),
2982 : minit (NULL, -1)
2983 : };
2984 :
2985 :
2986 : static void
2987 1807 : mio_array_ref (gfc_array_ref *ar)
2988 : {
2989 1807 : int i;
2990 :
2991 1807 : mio_lparen ();
2992 1807 : ar->type = MIO_NAME (ar_type) (ar->type, array_ref_types);
2993 1807 : mio_integer (&ar->dimen);
2994 :
2995 1807 : switch (ar->type)
2996 : {
2997 : case AR_FULL:
2998 : break;
2999 :
3000 : case AR_ELEMENT:
3001 982 : for (i = 0; i < ar->dimen; i++)
3002 505 : mio_expr (&ar->start[i]);
3003 :
3004 : break;
3005 :
3006 : case AR_SECTION:
3007 0 : for (i = 0; i < ar->dimen; i++)
3008 : {
3009 0 : mio_expr (&ar->start[i]);
3010 0 : mio_expr (&ar->end[i]);
3011 0 : mio_expr (&ar->stride[i]);
3012 : }
3013 :
3014 : break;
3015 :
3016 0 : case AR_UNKNOWN:
3017 0 : gfc_internal_error ("mio_array_ref(): Unknown array ref");
3018 : }
3019 :
3020 : /* Unfortunately, ar->dimen_type is an anonymous enumerated type so
3021 : we can't call mio_integer directly. Instead loop over each element
3022 : and cast it to/from an integer. */
3023 1807 : if (iomode == IO_OUTPUT)
3024 : {
3025 2030 : for (i = 0; i < ar->dimen; i++)
3026 : {
3027 1132 : HOST_WIDE_INT tmp = (HOST_WIDE_INT)ar->dimen_type[i];
3028 1132 : write_atom (ATOM_INTEGER, &tmp);
3029 : }
3030 : }
3031 : else
3032 : {
3033 2084 : for (i = 0; i < ar->dimen; i++)
3034 : {
3035 1175 : require_atom (ATOM_INTEGER);
3036 1175 : ar->dimen_type[i] = (enum gfc_array_ref_dimen_type) atom_int;
3037 : }
3038 : }
3039 :
3040 1807 : if (iomode == IO_INPUT)
3041 : {
3042 909 : ar->where = gfc_current_locus;
3043 :
3044 2084 : for (i = 0; i < ar->dimen; i++)
3045 1175 : ar->c_where[i] = gfc_current_locus;
3046 : }
3047 :
3048 1807 : mio_rparen ();
3049 1807 : }
3050 :
3051 :
3052 : /* Saves or restores a pointer. The pointer is converted back and
3053 : forth from an integer. We return the pointer_info pointer so that
3054 : the caller can take additional action based on the pointer type. */
3055 :
3056 : static pointer_info *
3057 7028505 : mio_pointer_ref (void *gp)
3058 : {
3059 7028505 : pointer_info *p;
3060 :
3061 7028505 : if (iomode == IO_OUTPUT)
3062 : {
3063 1632567 : p = get_pointer (*((char **) gp));
3064 1632567 : HOST_WIDE_INT hwi = p->integer;
3065 1632567 : write_atom (ATOM_INTEGER, &hwi);
3066 : }
3067 : else
3068 : {
3069 5395938 : require_atom (ATOM_INTEGER);
3070 5395938 : p = add_fixup (atom_int, gp);
3071 : }
3072 :
3073 7028505 : return p;
3074 : }
3075 :
3076 :
3077 : /* Save and load references to components that occur within
3078 : expressions. We have to describe these references by a number and
3079 : by name. The number is necessary for forward references during
3080 : reading, and the name is necessary if the symbol already exists in
3081 : the namespace and is not loaded again. */
3082 :
3083 : static void
3084 813 : mio_component_ref (gfc_component **cp)
3085 : {
3086 813 : pointer_info *p;
3087 :
3088 813 : p = mio_pointer_ref (cp);
3089 813 : if (p->type == P_UNKNOWN)
3090 180 : p->type = P_COMPONENT;
3091 813 : }
3092 :
3093 :
3094 : static void mio_namespace_ref (gfc_namespace **nsp);
3095 : static void mio_formal_arglist (gfc_formal_arglist **formal);
3096 : static void mio_typebound_proc (gfc_typebound_proc** proc);
3097 : static void mio_actual_arglist (gfc_actual_arglist **ap, bool pdt);
3098 :
3099 : static void
3100 261552 : mio_component (gfc_component *c, int vtype)
3101 : {
3102 261552 : pointer_info *p;
3103 :
3104 261552 : mio_lparen ();
3105 :
3106 261552 : if (iomode == IO_OUTPUT)
3107 : {
3108 103968 : p = get_pointer (c);
3109 103968 : mio_hwi (&p->integer);
3110 : }
3111 : else
3112 : {
3113 157584 : HOST_WIDE_INT n;
3114 157584 : mio_hwi (&n);
3115 157584 : p = get_integer (n);
3116 157584 : associate_integer_pointer (p, c);
3117 : }
3118 :
3119 261552 : if (p->type == P_UNKNOWN)
3120 261402 : p->type = P_COMPONENT;
3121 :
3122 261552 : mio_pool_string (&c->name);
3123 261552 : mio_typespec (&c->ts);
3124 261552 : mio_array_spec (&c->as);
3125 :
3126 : /* PDT templates store the expression for the kind of a component here. */
3127 261552 : mio_expr (&c->kind_expr);
3128 :
3129 : /* PDT types store the component specification list here. */
3130 261552 : mio_actual_arglist (&c->param_list, true);
3131 :
3132 261552 : mio_symbol_attribute (&c->attr);
3133 261552 : if (c->ts.type == BT_CLASS)
3134 2113 : c->attr.class_ok = 1;
3135 261552 : c->attr.access = MIO_NAME (gfc_access) (c->attr.access, access_types);
3136 :
3137 261552 : if (!vtype || strcmp (c->name, "_final") == 0
3138 163230 : || strcmp (c->name, "_hash") == 0)
3139 123976 : mio_expr (&c->initializer);
3140 :
3141 261552 : if (c->attr.proc_pointer)
3142 86722 : mio_typebound_proc (&c->tb);
3143 :
3144 261552 : c->loc = gfc_current_locus;
3145 :
3146 261552 : mio_rparen ();
3147 261552 : }
3148 :
3149 :
3150 : static void
3151 1271128 : mio_component_list (gfc_component **cp, int vtype)
3152 : {
3153 1271128 : gfc_component *c, *tail;
3154 :
3155 1271128 : mio_lparen ();
3156 :
3157 1271128 : if (iomode == IO_OUTPUT)
3158 : {
3159 345977 : for (c = *cp; c; c = c->next)
3160 103968 : mio_component (c, vtype);
3161 : }
3162 : else
3163 : {
3164 1029119 : *cp = NULL;
3165 1029119 : tail = NULL;
3166 :
3167 1186703 : for (;;)
3168 : {
3169 1186703 : if (peek_atom () == ATOM_RPAREN)
3170 : break;
3171 :
3172 157584 : c = gfc_get_component ();
3173 157584 : mio_component (c, vtype);
3174 :
3175 157584 : if (tail == NULL)
3176 49308 : *cp = c;
3177 : else
3178 108276 : tail->next = c;
3179 :
3180 : tail = c;
3181 : }
3182 : }
3183 :
3184 1271128 : mio_rparen ();
3185 1271128 : }
3186 :
3187 :
3188 : static void
3189 7451 : mio_actual_arg (gfc_actual_arglist *a, bool pdt)
3190 : {
3191 7451 : mio_lparen ();
3192 7451 : mio_pool_string (&a->name);
3193 7451 : mio_expr (&a->expr);
3194 7451 : if (pdt)
3195 1487 : mio_integer ((int *)&a->spec_type);
3196 7451 : mio_rparen ();
3197 7451 : }
3198 :
3199 :
3200 : static void
3201 1973992 : mio_actual_arglist (gfc_actual_arglist **ap, bool pdt)
3202 : {
3203 1973992 : gfc_actual_arglist *a, *tail;
3204 :
3205 1973992 : mio_lparen ();
3206 :
3207 1973992 : if (iomode == IO_OUTPUT)
3208 : {
3209 452346 : for (a = *ap; a; a = a->next)
3210 3831 : mio_actual_arg (a, pdt);
3211 :
3212 : }
3213 : else
3214 : {
3215 : tail = NULL;
3216 :
3217 1532717 : for (;;)
3218 : {
3219 1529097 : if (peek_atom () != ATOM_LPAREN)
3220 : break;
3221 :
3222 3620 : a = gfc_get_actual_arglist ();
3223 :
3224 3620 : if (tail == NULL)
3225 1889 : *ap = a;
3226 : else
3227 1731 : tail->next = a;
3228 :
3229 3620 : tail = a;
3230 3620 : mio_actual_arg (a, pdt);
3231 : }
3232 : }
3233 :
3234 1973992 : mio_rparen ();
3235 1973992 : }
3236 :
3237 :
3238 : /* Read and write formal argument lists. */
3239 :
3240 : static void
3241 1271128 : mio_formal_arglist (gfc_formal_arglist **formal)
3242 : {
3243 1271128 : gfc_formal_arglist *f, *tail;
3244 :
3245 1271128 : mio_lparen ();
3246 :
3247 1271128 : if (iomode == IO_OUTPUT)
3248 : {
3249 324231 : for (f = *formal; f; f = f->next)
3250 82222 : mio_symbol_ref (&f->sym);
3251 : }
3252 : else
3253 : {
3254 1029119 : *formal = tail = NULL;
3255 :
3256 1444313 : while (peek_atom () != ATOM_RPAREN)
3257 : {
3258 415194 : f = gfc_get_formal_arglist ();
3259 415194 : mio_symbol_ref (&f->sym);
3260 :
3261 415194 : if (*formal == NULL)
3262 218401 : *formal = f;
3263 : else
3264 196793 : tail->next = f;
3265 :
3266 : tail = f;
3267 : }
3268 : }
3269 :
3270 1271128 : mio_rparen ();
3271 1271128 : }
3272 :
3273 :
3274 : /* Save or restore a reference to a symbol node. */
3275 :
3276 : pointer_info *
3277 5514532 : mio_symbol_ref (gfc_symbol **symp)
3278 : {
3279 5514532 : pointer_info *p;
3280 :
3281 5514532 : p = mio_pointer_ref (symp);
3282 5514532 : if (p->type == P_UNKNOWN)
3283 139359 : p->type = P_SYMBOL;
3284 :
3285 5514532 : if (iomode == IO_OUTPUT)
3286 : {
3287 1148053 : if (p->u.wsym.state == UNREFERENCED)
3288 156569 : p->u.wsym.state = NEEDS_WRITE;
3289 : }
3290 : else
3291 : {
3292 4366479 : if (p->u.rsym.state == UNUSED)
3293 618766 : p->u.rsym.state = NEEDED;
3294 : }
3295 5514532 : return p;
3296 : }
3297 :
3298 :
3299 : /* Save or restore a reference to a symtree node. */
3300 :
3301 : static void
3302 29884 : mio_symtree_ref (gfc_symtree **stp)
3303 : {
3304 29884 : pointer_info *p;
3305 29884 : fixup_t *f;
3306 :
3307 29884 : if (iomode == IO_OUTPUT)
3308 14957 : mio_symbol_ref (&(*stp)->n.sym);
3309 : else
3310 : {
3311 14927 : require_atom (ATOM_INTEGER);
3312 14927 : p = get_integer (atom_int);
3313 :
3314 : /* An unused equivalence member; make a symbol and a symtree
3315 : for it. */
3316 14927 : if (in_load_equiv && p->u.rsym.symtree == NULL)
3317 : {
3318 : /* Since this is not used, it must have a unique name. */
3319 87 : p->u.rsym.symtree = gfc_get_unique_symtree (gfc_current_ns);
3320 :
3321 : /* Make the symbol. */
3322 87 : if (p->u.rsym.sym == NULL)
3323 : {
3324 75 : p->u.rsym.sym = gfc_new_symbol (p->u.rsym.true_name,
3325 : gfc_current_ns);
3326 75 : p->u.rsym.sym->module = gfc_get_string ("%s", p->u.rsym.module);
3327 : }
3328 :
3329 87 : p->u.rsym.symtree->n.sym = p->u.rsym.sym;
3330 87 : p->u.rsym.symtree->n.sym->refs++;
3331 87 : p->u.rsym.referenced = 1;
3332 :
3333 : /* If the symbol is PRIVATE and in COMMON, load_commons will
3334 : generate a fixup symbol, which must be associated. */
3335 87 : if (p->fixup)
3336 2 : resolve_fixups (p->fixup, p->u.rsym.sym);
3337 87 : p->fixup = NULL;
3338 : }
3339 :
3340 14927 : if (p->type == P_UNKNOWN)
3341 0 : p->type = P_SYMBOL;
3342 :
3343 14927 : if (p->u.rsym.state == UNUSED)
3344 2591 : p->u.rsym.state = NEEDED;
3345 :
3346 14927 : if (p->u.rsym.symtree != NULL)
3347 : {
3348 3632 : *stp = p->u.rsym.symtree;
3349 : }
3350 : else
3351 : {
3352 11295 : f = XCNEW (fixup_t);
3353 :
3354 11295 : f->next = p->u.rsym.stfixup;
3355 11295 : p->u.rsym.stfixup = f;
3356 :
3357 11295 : f->pointer = (void **) stp;
3358 : }
3359 : }
3360 29884 : }
3361 :
3362 :
3363 : static void
3364 34463 : mio_iterator (gfc_iterator **ip)
3365 : {
3366 34463 : gfc_iterator *iter;
3367 :
3368 34463 : mio_lparen ();
3369 :
3370 34463 : if (iomode == IO_OUTPUT)
3371 : {
3372 9630 : if (*ip == NULL)
3373 9624 : goto done;
3374 : }
3375 : else
3376 : {
3377 24833 : if (peek_atom () == ATOM_RPAREN)
3378 : {
3379 24827 : *ip = NULL;
3380 24827 : goto done;
3381 : }
3382 :
3383 6 : *ip = gfc_get_iterator ();
3384 : }
3385 :
3386 12 : iter = *ip;
3387 :
3388 12 : mio_expr (&iter->var);
3389 12 : mio_expr (&iter->start);
3390 12 : mio_expr (&iter->end);
3391 12 : mio_expr (&iter->step);
3392 :
3393 34463 : done:
3394 34463 : mio_rparen ();
3395 34463 : }
3396 :
3397 :
3398 : static void
3399 21198 : mio_constructor (gfc_constructor_base *cp)
3400 : {
3401 21198 : gfc_constructor *c;
3402 :
3403 21198 : mio_lparen ();
3404 :
3405 21198 : if (iomode == IO_OUTPUT)
3406 : {
3407 13743 : for (c = gfc_constructor_first (*cp); c; c = gfc_constructor_next (c))
3408 : {
3409 9630 : mio_lparen ();
3410 9630 : mio_expr (&c->expr);
3411 9630 : mio_iterator (&c->iterator);
3412 9630 : mio_rparen ();
3413 : }
3414 : }
3415 : else
3416 : {
3417 41918 : while (peek_atom () != ATOM_RPAREN)
3418 : {
3419 24833 : c = gfc_constructor_append_expr (cp, NULL, NULL);
3420 :
3421 24833 : mio_lparen ();
3422 24833 : mio_expr (&c->expr);
3423 24833 : mio_iterator (&c->iterator);
3424 24833 : mio_rparen ();
3425 : }
3426 : }
3427 :
3428 21198 : mio_rparen ();
3429 21198 : }
3430 :
3431 :
3432 : static const mstring ref_types[] = {
3433 : minit ("ARRAY", REF_ARRAY),
3434 : minit ("COMPONENT", REF_COMPONENT),
3435 : minit ("SUBSTRING", REF_SUBSTRING),
3436 : minit ("INQUIRY", REF_INQUIRY),
3437 : minit (NULL, -1)
3438 : };
3439 :
3440 : static const mstring inquiry_types[] = {
3441 : minit ("RE", INQUIRY_RE),
3442 : minit ("IM", INQUIRY_IM),
3443 : minit ("KIND", INQUIRY_KIND),
3444 : minit ("LEN", INQUIRY_LEN),
3445 : minit (NULL, -1)
3446 : };
3447 :
3448 :
3449 : static void
3450 2626 : mio_ref (gfc_ref **rp)
3451 : {
3452 2626 : gfc_ref *r;
3453 :
3454 2626 : mio_lparen ();
3455 :
3456 2626 : r = *rp;
3457 2626 : r->type = MIO_NAME (ref_type) (r->type, ref_types);
3458 :
3459 2626 : switch (r->type)
3460 : {
3461 1807 : case REF_ARRAY:
3462 1807 : mio_array_ref (&r->u.ar);
3463 1807 : break;
3464 :
3465 813 : case REF_COMPONENT:
3466 813 : mio_symbol_ref (&r->u.c.sym);
3467 813 : mio_component_ref (&r->u.c.component);
3468 813 : break;
3469 :
3470 6 : case REF_SUBSTRING:
3471 6 : mio_expr (&r->u.ss.start);
3472 6 : mio_expr (&r->u.ss.end);
3473 6 : mio_charlen (&r->u.ss.length);
3474 6 : break;
3475 :
3476 0 : case REF_INQUIRY:
3477 0 : r->u.i = MIO_NAME (inquiry_type) (r->u.i, inquiry_types);
3478 0 : break;
3479 : }
3480 :
3481 2626 : mio_rparen ();
3482 2626 : }
3483 :
3484 :
3485 : static void
3486 16155 : mio_ref_list (gfc_ref **rp)
3487 : {
3488 16155 : gfc_ref *ref, *head, *tail;
3489 :
3490 16155 : mio_lparen ();
3491 :
3492 16155 : if (iomode == IO_OUTPUT)
3493 : {
3494 9070 : for (ref = *rp; ref; ref = ref->next)
3495 1382 : mio_ref (&ref);
3496 : }
3497 : else
3498 : {
3499 8467 : head = tail = NULL;
3500 :
3501 9711 : while (peek_atom () != ATOM_RPAREN)
3502 : {
3503 1244 : if (head == NULL)
3504 1087 : head = tail = gfc_get_ref ();
3505 : else
3506 : {
3507 157 : tail->next = gfc_get_ref ();
3508 157 : tail = tail->next;
3509 : }
3510 :
3511 1244 : mio_ref (&tail);
3512 : }
3513 :
3514 8467 : *rp = head;
3515 : }
3516 :
3517 16155 : mio_rparen ();
3518 16155 : }
3519 :
3520 :
3521 : /* Read and write an integer value. */
3522 :
3523 : static void
3524 364541 : mio_gmp_integer (mpz_t *integer)
3525 : {
3526 364541 : char *p;
3527 :
3528 364541 : if (iomode == IO_INPUT)
3529 : {
3530 293189 : if (parse_atom () != ATOM_STRING)
3531 0 : bad_module ("Expected integer string");
3532 :
3533 293189 : mpz_init (*integer);
3534 293189 : if (mpz_set_str (*integer, atom_string, 10))
3535 0 : bad_module ("Error converting integer");
3536 :
3537 293189 : free (atom_string);
3538 : }
3539 : else
3540 : {
3541 71352 : p = mpz_get_str (NULL, 10, *integer);
3542 71352 : write_atom (ATOM_STRING, p);
3543 71352 : free (p);
3544 : }
3545 364541 : }
3546 :
3547 :
3548 : static void
3549 2062 : mio_gmp_real (mpfr_t *real)
3550 : {
3551 2062 : mpfr_exp_t exponent;
3552 2062 : char *p;
3553 :
3554 2062 : if (iomode == IO_INPUT)
3555 : {
3556 963 : if (parse_atom () != ATOM_STRING)
3557 0 : bad_module ("Expected real string");
3558 :
3559 963 : mpfr_init (*real);
3560 963 : mpfr_set_str (*real, atom_string, 16, GFC_RND_MODE);
3561 963 : free (atom_string);
3562 : }
3563 : else
3564 : {
3565 1099 : p = mpfr_get_str (NULL, &exponent, 16, 0, *real, GFC_RND_MODE);
3566 :
3567 1099 : if (mpfr_nan_p (*real) || mpfr_inf_p (*real))
3568 : {
3569 18 : write_atom (ATOM_STRING, p);
3570 18 : free (p);
3571 18 : return;
3572 : }
3573 :
3574 1081 : atom_string = XCNEWVEC (char, strlen (p) + 20);
3575 :
3576 1081 : sprintf (atom_string, "0.%s@%ld", p, exponent);
3577 :
3578 : /* Fix negative numbers. */
3579 1081 : if (atom_string[2] == '-')
3580 : {
3581 45 : atom_string[0] = '-';
3582 45 : atom_string[1] = '0';
3583 45 : atom_string[2] = '.';
3584 : }
3585 :
3586 1081 : write_atom (ATOM_STRING, atom_string);
3587 :
3588 1081 : free (atom_string);
3589 1081 : free (p);
3590 : }
3591 : }
3592 :
3593 :
3594 : /* Save and restore the shape of an array constructor. */
3595 :
3596 : static void
3597 21198 : mio_shape (mpz_t **pshape, int rank)
3598 : {
3599 21198 : mpz_t *shape;
3600 21198 : atom_type t;
3601 21198 : int n;
3602 :
3603 : /* A NULL shape is represented by (). */
3604 21198 : mio_lparen ();
3605 :
3606 21198 : if (iomode == IO_OUTPUT)
3607 : {
3608 4113 : shape = *pshape;
3609 4113 : if (!shape)
3610 : {
3611 3527 : mio_rparen ();
3612 3527 : return;
3613 : }
3614 : }
3615 : else
3616 : {
3617 17085 : t = peek_atom ();
3618 17085 : if (t == ATOM_RPAREN)
3619 : {
3620 15724 : *pshape = NULL;
3621 15724 : mio_rparen ();
3622 15724 : return;
3623 : }
3624 :
3625 1361 : shape = gfc_get_shape (rank);
3626 1361 : *pshape = shape;
3627 : }
3628 :
3629 4033 : for (n = 0; n < rank; n++)
3630 2086 : mio_gmp_integer (&shape[n]);
3631 :
3632 1947 : mio_rparen ();
3633 : }
3634 :
3635 :
3636 : static const mstring expr_types[] = {
3637 : minit ("OP", EXPR_OP),
3638 : minit ("FUNCTION", EXPR_FUNCTION),
3639 : minit ("CONSTANT", EXPR_CONSTANT),
3640 : minit ("VARIABLE", EXPR_VARIABLE),
3641 : minit ("SUBSTRING", EXPR_SUBSTRING),
3642 : minit ("STRUCTURE", EXPR_STRUCTURE),
3643 : minit ("ARRAY", EXPR_ARRAY),
3644 : minit ("NULL", EXPR_NULL),
3645 : minit ("COMPCALL", EXPR_COMPCALL),
3646 : minit ("PPC", EXPR_PPC),
3647 : minit ("CONDITIONAL", EXPR_CONDITIONAL),
3648 : minit (NULL, -1),
3649 : };
3650 :
3651 : /* INTRINSIC_ASSIGN is missing because it is used as an index for
3652 : generic operators, not in expressions. INTRINSIC_USER is also
3653 : replaced by the correct function name by the time we see it. */
3654 :
3655 : static const mstring intrinsics[] =
3656 : {
3657 : minit ("UPLUS", INTRINSIC_UPLUS),
3658 : minit ("UMINUS", INTRINSIC_UMINUS),
3659 : minit ("PLUS", INTRINSIC_PLUS),
3660 : minit ("MINUS", INTRINSIC_MINUS),
3661 : minit ("TIMES", INTRINSIC_TIMES),
3662 : minit ("DIVIDE", INTRINSIC_DIVIDE),
3663 : minit ("POWER", INTRINSIC_POWER),
3664 : minit ("CONCAT", INTRINSIC_CONCAT),
3665 : minit ("AND", INTRINSIC_AND),
3666 : minit ("OR", INTRINSIC_OR),
3667 : minit ("EQV", INTRINSIC_EQV),
3668 : minit ("NEQV", INTRINSIC_NEQV),
3669 : minit ("EQ_SIGN", INTRINSIC_EQ),
3670 : minit ("EQ", INTRINSIC_EQ_OS),
3671 : minit ("NE_SIGN", INTRINSIC_NE),
3672 : minit ("NE", INTRINSIC_NE_OS),
3673 : minit ("GT_SIGN", INTRINSIC_GT),
3674 : minit ("GT", INTRINSIC_GT_OS),
3675 : minit ("GE_SIGN", INTRINSIC_GE),
3676 : minit ("GE", INTRINSIC_GE_OS),
3677 : minit ("LT_SIGN", INTRINSIC_LT),
3678 : minit ("LT", INTRINSIC_LT_OS),
3679 : minit ("LE_SIGN", INTRINSIC_LE),
3680 : minit ("LE", INTRINSIC_LE_OS),
3681 : minit ("NOT", INTRINSIC_NOT),
3682 : minit ("PARENTHESES", INTRINSIC_PARENTHESES),
3683 : minit ("USER", INTRINSIC_USER),
3684 : minit (NULL, -1)
3685 : };
3686 :
3687 :
3688 : /* Remedy a couple of situations where the gfc_expr's can be defective. */
3689 :
3690 : static void
3691 438532 : fix_mio_expr (gfc_expr *e)
3692 : {
3693 438532 : gfc_symtree *ns_st = NULL;
3694 438532 : const char *fname;
3695 :
3696 438532 : if (iomode != IO_OUTPUT)
3697 : return;
3698 :
3699 101101 : if (e->symtree)
3700 : {
3701 : /* If this is a symtree for a symbol that came from a contained module
3702 : namespace, it has a unique name and we should look in the current
3703 : namespace to see if the required, non-contained symbol is available
3704 : yet. If so, the latter should be written. */
3705 10345 : if (e->symtree->n.sym && check_unique_name (e->symtree->name))
3706 : {
3707 722 : const char *name = e->symtree->n.sym->name;
3708 722 : if (gfc_fl_struct (e->symtree->n.sym->attr.flavor))
3709 0 : name = gfc_dt_upper_string (name);
3710 722 : ns_st = gfc_find_symtree (gfc_current_ns->sym_root, name);
3711 : }
3712 :
3713 : /* On the other hand, if the existing symbol is the module name or the
3714 : new symbol is a dummy argument, do not do the promotion. */
3715 722 : if (ns_st && ns_st->n.sym
3716 23 : && ns_st->n.sym->attr.flavor != FL_MODULE
3717 22 : && !e->symtree->n.sym->attr.dummy)
3718 21 : e->symtree = ns_st;
3719 : }
3720 90756 : else if (e->expr_type == EXPR_FUNCTION
3721 2 : && (e->value.function.name || e->value.function.isym))
3722 : {
3723 2 : gfc_symbol *sym;
3724 :
3725 : /* In some circumstances, a function used in an initialization
3726 : expression, in one use associated module, can fail to be
3727 : coupled to its symtree when used in a specification
3728 : expression in another module. */
3729 2 : fname = e->value.function.esym ? e->value.function.esym->name
3730 2 : : e->value.function.isym->name;
3731 2 : e->symtree = gfc_find_symtree (gfc_current_ns->sym_root, fname);
3732 :
3733 2 : if (e->symtree)
3734 1 : return;
3735 :
3736 : /* This is probably a reference to a private procedure from another
3737 : module. To prevent a segfault, make a generic with no specific
3738 : instances. If this module is used, without the required
3739 : specific coming from somewhere, the appropriate error message
3740 : is issued. */
3741 1 : gfc_get_symbol (fname, gfc_current_ns, &sym);
3742 1 : sym->attr.flavor = FL_PROCEDURE;
3743 1 : sym->attr.generic = 1;
3744 1 : e->symtree = gfc_find_symtree (gfc_current_ns->sym_root, fname);
3745 1 : gfc_commit_symbol (sym);
3746 : }
3747 : }
3748 :
3749 :
3750 : /* Read and write expressions. The form "()" is allowed to indicate a
3751 : NULL expression. */
3752 :
3753 : static void
3754 848841 : mio_expr (gfc_expr **ep)
3755 : {
3756 848841 : HOST_WIDE_INT hwi;
3757 848841 : gfc_expr *e;
3758 848841 : atom_type t;
3759 848841 : int flag;
3760 :
3761 848841 : mio_lparen ();
3762 :
3763 848841 : if (iomode == IO_OUTPUT)
3764 : {
3765 253419 : if (*ep == NULL)
3766 : {
3767 152318 : mio_rparen ();
3768 562627 : return;
3769 : }
3770 :
3771 101101 : e = *ep;
3772 101101 : MIO_NAME (expr_t) (e->expr_type, expr_types);
3773 : }
3774 : else
3775 : {
3776 595422 : t = parse_atom ();
3777 595422 : if (t == ATOM_RPAREN)
3778 : {
3779 257991 : *ep = NULL;
3780 257991 : return;
3781 : }
3782 :
3783 337431 : if (t != ATOM_NAME)
3784 0 : bad_module ("Expected expression type");
3785 :
3786 337431 : e = *ep = gfc_get_expr ();
3787 337431 : e->where = gfc_current_locus;
3788 337431 : e->expr_type = (expr_t) find_enum (expr_types);
3789 : }
3790 :
3791 438532 : mio_typespec (&e->ts);
3792 438532 : mio_integer (&e->rank);
3793 :
3794 438532 : fix_mio_expr (e);
3795 :
3796 438532 : switch (e->expr_type)
3797 : {
3798 1629 : case EXPR_OP:
3799 1629 : e->value.op.op
3800 1629 : = MIO_NAME (gfc_intrinsic_op) (e->value.op.op, intrinsics);
3801 :
3802 1629 : switch (e->value.op.op)
3803 : {
3804 447 : case INTRINSIC_UPLUS:
3805 447 : case INTRINSIC_UMINUS:
3806 447 : case INTRINSIC_NOT:
3807 447 : case INTRINSIC_PARENTHESES:
3808 447 : mio_expr (&e->value.op.op1);
3809 447 : break;
3810 :
3811 1112 : case INTRINSIC_PLUS:
3812 1112 : case INTRINSIC_MINUS:
3813 1112 : case INTRINSIC_TIMES:
3814 1112 : case INTRINSIC_DIVIDE:
3815 1112 : case INTRINSIC_POWER:
3816 1112 : case INTRINSIC_CONCAT:
3817 1112 : case INTRINSIC_AND:
3818 1112 : case INTRINSIC_OR:
3819 1112 : case INTRINSIC_EQV:
3820 1112 : case INTRINSIC_NEQV:
3821 1112 : case INTRINSIC_EQ:
3822 1112 : case INTRINSIC_EQ_OS:
3823 1112 : case INTRINSIC_NE:
3824 1112 : case INTRINSIC_NE_OS:
3825 1112 : case INTRINSIC_GT:
3826 1112 : case INTRINSIC_GT_OS:
3827 1112 : case INTRINSIC_GE:
3828 1112 : case INTRINSIC_GE_OS:
3829 1112 : case INTRINSIC_LT:
3830 1112 : case INTRINSIC_LT_OS:
3831 1112 : case INTRINSIC_LE:
3832 1112 : case INTRINSIC_LE_OS:
3833 1112 : mio_expr (&e->value.op.op1);
3834 1112 : mio_expr (&e->value.op.op2);
3835 1112 : break;
3836 :
3837 70 : case INTRINSIC_USER:
3838 : /* INTRINSIC_USER should not appear in resolved expressions,
3839 : though for UDRs we need to stream unresolved ones. */
3840 70 : if (iomode == IO_OUTPUT)
3841 34 : write_atom (ATOM_STRING, e->value.op.uop->name);
3842 : else
3843 : {
3844 36 : char *name = read_string ();
3845 36 : const char *uop_name = find_use_name (name, true);
3846 36 : if (uop_name == NULL)
3847 : {
3848 0 : size_t len = strlen (name);
3849 0 : char *name2 = XCNEWVEC (char, len + 2);
3850 0 : memcpy (name2, name, len);
3851 0 : name2[len] = ' ';
3852 0 : name2[len + 1] = '\0';
3853 0 : free (name);
3854 0 : uop_name = name = name2;
3855 : }
3856 36 : e->value.op.uop = gfc_get_uop (uop_name);
3857 36 : free (name);
3858 : }
3859 70 : mio_expr (&e->value.op.op1);
3860 70 : mio_expr (&e->value.op.op2);
3861 70 : break;
3862 :
3863 0 : default:
3864 0 : bad_module ("Bad operator");
3865 : }
3866 :
3867 : break;
3868 :
3869 2 : case EXPR_CONDITIONAL:
3870 2 : mio_expr (&e->value.conditional.condition);
3871 2 : mio_expr (&e->value.conditional.true_expr);
3872 2 : mio_expr (&e->value.conditional.false_expr);
3873 2 : break;
3874 :
3875 2672 : case EXPR_FUNCTION:
3876 2672 : mio_symtree_ref (&e->symtree);
3877 2672 : mio_actual_arglist (&e->value.function.actual, false);
3878 :
3879 2672 : if (iomode == IO_OUTPUT)
3880 : {
3881 1383 : e->value.function.name
3882 1383 : = mio_allocated_string (e->value.function.name);
3883 1383 : if (e->value.function.esym)
3884 131 : flag = 1;
3885 1252 : else if (e->ref)
3886 104 : flag = 2;
3887 1148 : else if (e->value.function.isym == NULL)
3888 250 : flag = 3;
3889 : else
3890 898 : flag = 0;
3891 1383 : mio_integer (&flag);
3892 1383 : switch (flag)
3893 : {
3894 131 : case 1:
3895 131 : mio_symbol_ref (&e->value.function.esym);
3896 131 : break;
3897 104 : case 2:
3898 104 : mio_ref_list (&e->ref);
3899 104 : break;
3900 : case 3:
3901 : break;
3902 898 : default:
3903 898 : write_atom (ATOM_STRING, e->value.function.isym->name);
3904 : }
3905 : }
3906 : else
3907 : {
3908 1289 : require_atom (ATOM_STRING);
3909 1289 : if (atom_string[0] == '\0')
3910 723 : e->value.function.name = NULL;
3911 : else
3912 566 : e->value.function.name = gfc_get_string ("%s", atom_string);
3913 1289 : free (atom_string);
3914 :
3915 1289 : mio_integer (&flag);
3916 1289 : switch (flag)
3917 : {
3918 138 : case 1:
3919 138 : mio_symbol_ref (&e->value.function.esym);
3920 138 : break;
3921 72 : case 2:
3922 72 : mio_ref_list (&e->ref);
3923 72 : break;
3924 : case 3:
3925 : break;
3926 880 : default:
3927 880 : require_atom (ATOM_STRING);
3928 880 : e->value.function.isym = gfc_find_function (atom_string);
3929 880 : free (atom_string);
3930 : }
3931 : }
3932 :
3933 : break;
3934 :
3935 15979 : case EXPR_VARIABLE:
3936 15979 : mio_symtree_ref (&e->symtree);
3937 15979 : mio_ref_list (&e->ref);
3938 15979 : break;
3939 :
3940 0 : case EXPR_SUBSTRING:
3941 0 : e->value.character.string = const_cast<gfc_char_t *>
3942 0 : (mio_allocated_wide_string (e->value.character.string,
3943 0 : e->value.character.length));
3944 0 : mio_ref_list (&e->ref);
3945 0 : break;
3946 :
3947 21198 : case EXPR_STRUCTURE:
3948 21198 : case EXPR_ARRAY:
3949 21198 : mio_constructor (&e->value.constructor);
3950 21198 : mio_shape (&e->shape, e->rank);
3951 21198 : break;
3952 :
3953 373094 : case EXPR_CONSTANT:
3954 373094 : switch (e->ts.type)
3955 : {
3956 362455 : case BT_INTEGER:
3957 362455 : case BT_UNSIGNED:
3958 362455 : mio_gmp_integer (&e->value.integer);
3959 362455 : break;
3960 :
3961 1926 : case BT_REAL:
3962 1926 : gfc_set_model_kind (e->ts.kind);
3963 1926 : mio_gmp_real (&e->value.real);
3964 1926 : break;
3965 :
3966 68 : case BT_COMPLEX:
3967 68 : gfc_set_model_kind (e->ts.kind);
3968 68 : mio_gmp_real (&mpc_realref (e->value.complex));
3969 68 : mio_gmp_real (&mpc_imagref (e->value.complex));
3970 68 : break;
3971 :
3972 157 : case BT_LOGICAL:
3973 157 : mio_integer (&e->value.logical);
3974 157 : break;
3975 :
3976 8488 : case BT_CHARACTER:
3977 8488 : hwi = e->value.character.length;
3978 8488 : mio_hwi (&hwi);
3979 8488 : e->value.character.length = hwi;
3980 16976 : e->value.character.string = const_cast<gfc_char_t *>
3981 8488 : (mio_allocated_wide_string (e->value.character.string,
3982 : e->value.character.length));
3983 8488 : break;
3984 :
3985 0 : default:
3986 0 : bad_module ("Bad type in constant expression");
3987 : }
3988 :
3989 : break;
3990 :
3991 : case EXPR_NULL:
3992 : break;
3993 :
3994 0 : case EXPR_COMPCALL:
3995 0 : case EXPR_PPC:
3996 0 : case EXPR_UNKNOWN:
3997 0 : gcc_unreachable ();
3998 438532 : break;
3999 : }
4000 :
4001 : /* PDT types store the expression specification list here. */
4002 438532 : mio_actual_arglist (&e->param_list, true);
4003 :
4004 438532 : mio_rparen ();
4005 : }
4006 :
4007 :
4008 : /* Read and write namelists. */
4009 :
4010 : static void
4011 1271128 : mio_namelist (gfc_symbol *sym)
4012 : {
4013 1271128 : gfc_namelist *n, *m;
4014 :
4015 1271128 : mio_lparen ();
4016 :
4017 1271128 : if (iomode == IO_OUTPUT)
4018 : {
4019 242093 : for (n = sym->namelist; n; n = n->next)
4020 84 : mio_symbol_ref (&n->sym);
4021 : }
4022 : else
4023 : {
4024 : m = NULL;
4025 1029208 : while (peek_atom () != ATOM_RPAREN)
4026 : {
4027 89 : n = gfc_get_namelist ();
4028 89 : mio_symbol_ref (&n->sym);
4029 :
4030 89 : if (sym->namelist == NULL)
4031 53 : sym->namelist = n;
4032 : else
4033 36 : m->next = n;
4034 :
4035 : m = n;
4036 : }
4037 1029119 : sym->namelist_tail = m;
4038 : }
4039 :
4040 1271128 : mio_rparen ();
4041 1271128 : }
4042 :
4043 :
4044 : /* Save/restore lists of gfc_interface structures. When loading an
4045 : interface, we are really appending to the existing list of
4046 : interfaces. Checking for duplicate and ambiguous interfaces has to
4047 : be done later when all symbols have been loaded. */
4048 :
4049 : pointer_info *
4050 650865 : mio_interface_rest (gfc_interface **ip)
4051 : {
4052 650865 : gfc_interface *tail, *p;
4053 650865 : pointer_info *pi = NULL;
4054 :
4055 650865 : if (iomode == IO_OUTPUT)
4056 : {
4057 278082 : if (ip != NULL)
4058 268879 : for (p = *ip; p; p = p->next)
4059 16758 : mio_symbol_ref (&p->sym);
4060 : }
4061 : else
4062 : {
4063 372783 : if (*ip == NULL)
4064 : tail = NULL;
4065 : else
4066 : {
4067 : tail = *ip;
4068 5164 : while (tail->next)
4069 : tail = tail->next;
4070 : }
4071 :
4072 537840 : for (;;)
4073 : {
4074 537840 : if (peek_atom () == ATOM_RPAREN)
4075 : break;
4076 :
4077 165057 : p = gfc_get_interface ();
4078 165057 : p->where = gfc_current_locus;
4079 165057 : pi = mio_symbol_ref (&p->sym);
4080 :
4081 165057 : if (tail == NULL)
4082 54177 : *ip = p;
4083 : else
4084 110880 : tail->next = p;
4085 :
4086 : tail = p;
4087 : }
4088 : }
4089 :
4090 650865 : mio_rparen ();
4091 650865 : return pi;
4092 : }
4093 :
4094 :
4095 : /* Save/restore a nameless operator interface. */
4096 :
4097 : static void
4098 584916 : mio_interface (gfc_interface **ip)
4099 : {
4100 267084 : mio_lparen ();
4101 317832 : mio_interface_rest (ip);
4102 267084 : }
4103 :
4104 :
4105 : /* Save/restore a named operator interface. */
4106 :
4107 : static void
4108 10998 : mio_symbol_interface (const char **name, const char **module,
4109 : gfc_interface **ip)
4110 : {
4111 10998 : mio_lparen ();
4112 10998 : mio_pool_string (name);
4113 10998 : mio_pool_string (module);
4114 10998 : mio_interface_rest (ip);
4115 10998 : }
4116 :
4117 :
4118 : static void
4119 1271128 : mio_namespace_ref (gfc_namespace **nsp)
4120 : {
4121 1271128 : gfc_namespace *ns;
4122 1271128 : pointer_info *p;
4123 :
4124 1271128 : p = mio_pointer_ref (nsp);
4125 :
4126 1271128 : if (p->type == P_UNKNOWN)
4127 254860 : p->type = P_NAMESPACE;
4128 :
4129 1271128 : if (iomode == IO_INPUT && p->integer != 0)
4130 : {
4131 225660 : ns = (gfc_namespace *) p->u.pointer;
4132 225660 : if (ns == NULL)
4133 : {
4134 225390 : ns = gfc_get_namespace (NULL, 0);
4135 225390 : associate_integer_pointer (p, ns);
4136 : }
4137 : else
4138 270 : ns->refs++;
4139 : }
4140 1271128 : }
4141 :
4142 :
4143 : /* Save/restore the f2k_derived namespace of a derived-type symbol. */
4144 :
4145 : static gfc_namespace* current_f2k_derived;
4146 :
4147 : static void
4148 99184 : mio_typebound_proc (gfc_typebound_proc** proc)
4149 : {
4150 99184 : int flag;
4151 99184 : int overriding_flag;
4152 :
4153 99184 : if (iomode == IO_INPUT)
4154 : {
4155 55590 : *proc = gfc_get_typebound_proc (NULL);
4156 55590 : (*proc)->where = gfc_current_locus;
4157 : }
4158 99184 : gcc_assert (*proc);
4159 :
4160 99184 : mio_lparen ();
4161 :
4162 99184 : (*proc)->access = MIO_NAME (gfc_access) ((*proc)->access, access_types);
4163 :
4164 : /* IO the NON_OVERRIDABLE/DEFERRED combination. */
4165 99184 : gcc_assert (!((*proc)->deferred && (*proc)->non_overridable));
4166 99184 : overriding_flag = ((*proc)->deferred << 1) | (*proc)->non_overridable;
4167 99184 : overriding_flag = mio_name (overriding_flag, binding_overriding);
4168 99184 : (*proc)->deferred = ((overriding_flag & 2) != 0);
4169 99184 : (*proc)->non_overridable = ((overriding_flag & 1) != 0);
4170 99184 : gcc_assert (!((*proc)->deferred && (*proc)->non_overridable));
4171 :
4172 99184 : (*proc)->nopass = mio_name ((*proc)->nopass, binding_passing);
4173 99184 : (*proc)->is_generic = mio_name ((*proc)->is_generic, binding_generic);
4174 99184 : (*proc)->ppc = mio_name((*proc)->ppc, binding_ppc);
4175 :
4176 99184 : mio_pool_string (&((*proc)->pass_arg));
4177 :
4178 99184 : flag = (int) (*proc)->pass_arg_num;
4179 99184 : mio_integer (&flag);
4180 99184 : (*proc)->pass_arg_num = (unsigned) flag;
4181 :
4182 99184 : if ((*proc)->is_generic)
4183 : {
4184 2878 : gfc_tbp_generic* g;
4185 2878 : int iop;
4186 :
4187 2878 : mio_lparen ();
4188 :
4189 2878 : if (iomode == IO_OUTPUT)
4190 3440 : for (g = (*proc)->u.generic; g; g = g->next)
4191 : {
4192 1925 : iop = (int) g->is_operator;
4193 1925 : mio_integer (&iop);
4194 1925 : mio_allocated_string (g->specific_st->name);
4195 : }
4196 : else
4197 : {
4198 1363 : (*proc)->u.generic = NULL;
4199 3039 : while (peek_atom () != ATOM_RPAREN)
4200 : {
4201 1676 : gfc_symtree** sym_root;
4202 :
4203 1676 : g = gfc_get_tbp_generic ();
4204 1676 : g->specific = NULL;
4205 :
4206 1676 : mio_integer (&iop);
4207 1676 : g->is_operator = (bool) iop;
4208 :
4209 1676 : require_atom (ATOM_STRING);
4210 1676 : sym_root = ¤t_f2k_derived->tb_sym_root;
4211 1676 : g->specific_st = gfc_get_tbp_symtree (sym_root, atom_string);
4212 1676 : free (atom_string);
4213 :
4214 1676 : g->next = (*proc)->u.generic;
4215 1676 : (*proc)->u.generic = g;
4216 : }
4217 : }
4218 :
4219 2878 : mio_rparen ();
4220 : }
4221 96306 : else if (!(*proc)->ppc)
4222 9584 : mio_symtree_ref (&(*proc)->u.specific);
4223 :
4224 99184 : mio_rparen ();
4225 99184 : }
4226 :
4227 : /* Walker-callback function for this purpose. */
4228 : static void
4229 11130 : mio_typebound_symtree (gfc_symtree* st)
4230 : {
4231 11130 : if (iomode == IO_OUTPUT && !st->n.tb)
4232 : return;
4233 :
4234 11130 : if (iomode == IO_OUTPUT)
4235 : {
4236 5921 : mio_lparen ();
4237 5921 : mio_allocated_string (st->name);
4238 : }
4239 : /* For IO_INPUT, the above is done in mio_f2k_derived. */
4240 :
4241 11130 : mio_typebound_proc (&st->n.tb);
4242 11130 : mio_rparen ();
4243 : }
4244 :
4245 : /* IO a full symtree (in all depth). */
4246 : static void
4247 64086 : mio_full_typebound_tree (gfc_symtree** root)
4248 : {
4249 64086 : mio_lparen ();
4250 :
4251 64086 : if (iomode == IO_OUTPUT)
4252 27514 : gfc_traverse_symtree (*root, &mio_typebound_symtree);
4253 : else
4254 : {
4255 41781 : while (peek_atom () == ATOM_LPAREN)
4256 : {
4257 5209 : gfc_symtree* st;
4258 :
4259 5209 : mio_lparen ();
4260 :
4261 5209 : require_atom (ATOM_STRING);
4262 5209 : st = gfc_get_tbp_symtree (root, atom_string);
4263 5209 : free (atom_string);
4264 :
4265 5209 : mio_typebound_symtree (st);
4266 : }
4267 : }
4268 :
4269 64086 : mio_rparen ();
4270 64086 : }
4271 :
4272 : static void
4273 1227 : mio_finalizer (gfc_finalizer **f)
4274 : {
4275 1227 : if (iomode == IO_OUTPUT)
4276 : {
4277 588 : gcc_assert (*f);
4278 588 : gcc_assert ((*f)->proc_tree); /* Should already be resolved. */
4279 588 : mio_symtree_ref (&(*f)->proc_tree);
4280 : }
4281 : else
4282 : {
4283 639 : *f = gfc_get_finalizer ();
4284 639 : (*f)->where = gfc_current_locus; /* Value should not matter. */
4285 639 : (*f)->next = NULL;
4286 :
4287 639 : mio_symtree_ref (&(*f)->proc_tree);
4288 639 : (*f)->proc_sym = NULL;
4289 : }
4290 1227 : }
4291 :
4292 : static void
4293 32043 : mio_f2k_derived (gfc_namespace *f2k)
4294 : {
4295 32043 : current_f2k_derived = f2k;
4296 :
4297 : /* Handle the list of finalizer procedures. */
4298 32043 : mio_lparen ();
4299 32043 : if (iomode == IO_OUTPUT)
4300 : {
4301 13757 : gfc_finalizer *f;
4302 14345 : for (f = f2k->finalizers; f; f = f->next)
4303 588 : mio_finalizer (&f);
4304 : }
4305 : else
4306 : {
4307 18286 : f2k->finalizers = NULL;
4308 18925 : while (peek_atom () != ATOM_RPAREN)
4309 : {
4310 639 : gfc_finalizer *cur = NULL;
4311 639 : mio_finalizer (&cur);
4312 639 : cur->next = f2k->finalizers;
4313 639 : f2k->finalizers = cur;
4314 : }
4315 : }
4316 32043 : mio_rparen ();
4317 :
4318 : /* Handle type-bound procedures. */
4319 32043 : mio_full_typebound_tree (&f2k->tb_sym_root);
4320 :
4321 : /* Type-bound user operators. */
4322 32043 : mio_full_typebound_tree (&f2k->tb_uop_root);
4323 :
4324 : /* Type-bound intrinsic operators. */
4325 32043 : mio_lparen ();
4326 32043 : if (iomode == IO_OUTPUT)
4327 : {
4328 : int op;
4329 398953 : for (op = GFC_INTRINSIC_BEGIN; op != GFC_INTRINSIC_END; ++op)
4330 : {
4331 385196 : gfc_intrinsic_op realop;
4332 :
4333 385196 : if (op == INTRINSIC_USER || !f2k->tb_op[op])
4334 384488 : continue;
4335 :
4336 708 : mio_lparen ();
4337 708 : realop = (gfc_intrinsic_op) op;
4338 708 : mio_intrinsic_op (&realop);
4339 708 : mio_typebound_proc (&f2k->tb_op[op]);
4340 708 : mio_rparen ();
4341 : }
4342 : }
4343 : else
4344 18910 : while (peek_atom () != ATOM_RPAREN)
4345 : {
4346 624 : gfc_intrinsic_op op = GFC_INTRINSIC_BEGIN; /* Silence GCC. */
4347 :
4348 624 : mio_lparen ();
4349 624 : mio_intrinsic_op (&op);
4350 624 : mio_typebound_proc (&f2k->tb_op[op]);
4351 624 : mio_rparen ();
4352 : }
4353 32043 : mio_rparen ();
4354 32043 : }
4355 :
4356 : static void
4357 1271128 : mio_full_f2k_derived (gfc_symbol *sym)
4358 : {
4359 1271128 : mio_lparen ();
4360 :
4361 1271128 : if (iomode == IO_OUTPUT)
4362 : {
4363 242009 : if (sym->f2k_derived)
4364 13757 : mio_f2k_derived (sym->f2k_derived);
4365 : }
4366 : else
4367 : {
4368 1029119 : if (peek_atom () != ATOM_RPAREN)
4369 : {
4370 18286 : gfc_namespace *ns;
4371 :
4372 18286 : sym->f2k_derived = gfc_get_namespace (NULL, 0);
4373 :
4374 : /* PDT templates make use of the mechanisms for formal args
4375 : and so the parameter symbols are stored in the formal
4376 : namespace. Transfer the sym_root to f2k_derived and then
4377 : free the formal namespace since it is unneeded. */
4378 18286 : if (sym->attr.pdt_template && sym->formal && sym->formal->sym)
4379 : {
4380 6 : ns = sym->formal->sym->ns;
4381 6 : sym->f2k_derived->sym_root = ns->sym_root;
4382 6 : ns->sym_root = NULL;
4383 6 : ns->refs++;
4384 6 : gfc_free_namespace (ns);
4385 6 : ns = NULL;
4386 : }
4387 :
4388 18286 : mio_f2k_derived (sym->f2k_derived);
4389 : }
4390 : else
4391 1010833 : gcc_assert (!sym->f2k_derived);
4392 : }
4393 :
4394 1271128 : mio_rparen ();
4395 1271128 : }
4396 :
4397 : static const mstring omp_declare_simd_clauses[] =
4398 : {
4399 : minit ("INBRANCH", 0),
4400 : minit ("NOTINBRANCH", 1),
4401 : minit ("SIMDLEN", 2),
4402 : minit ("UNIFORM", 3),
4403 : minit ("LINEAR", 4),
4404 : minit ("ALIGNED", 5),
4405 : minit ("LINEAR_REF", 33),
4406 : minit ("LINEAR_VAL", 34),
4407 : minit ("LINEAR_UVAL", 35),
4408 : minit (NULL, -1)
4409 : };
4410 :
4411 : /* Handle OpenMP's declare-simd clauses. */
4412 :
4413 : static void
4414 148 : mio_omp_declare_simd_clauses (gfc_omp_clauses **clausesp)
4415 : {
4416 148 : if (iomode == IO_OUTPUT)
4417 : {
4418 94 : gfc_omp_clauses *clauses = *clausesp;
4419 94 : gfc_omp_namelist *n;
4420 :
4421 94 : write_atom (ATOM_NAME, "OMP_DECLARE_SIMD");
4422 94 : if (clauses->inbranch)
4423 10 : mio_name (0, omp_declare_simd_clauses);
4424 94 : if (clauses->notinbranch)
4425 23 : mio_name (1, omp_declare_simd_clauses);
4426 94 : if (clauses->simdlen_expr)
4427 : {
4428 37 : mio_name (2, omp_declare_simd_clauses);
4429 37 : mio_expr (&clauses->simdlen_expr);
4430 : }
4431 151 : for (n = clauses->lists[OMP_LIST_UNIFORM]; n; n = n->next)
4432 : {
4433 57 : mio_name (3, omp_declare_simd_clauses);
4434 57 : mio_symbol_ref (&n->sym);
4435 : }
4436 146 : for (n = clauses->lists[OMP_LIST_LINEAR]; n; n = n->next)
4437 : {
4438 52 : if (n->u.linear.op == OMP_LINEAR_DEFAULT)
4439 33 : mio_name (4, omp_declare_simd_clauses);
4440 : else
4441 19 : mio_name (32 + n->u.linear.op, omp_declare_simd_clauses);
4442 52 : mio_symbol_ref (&n->sym);
4443 52 : mio_expr (&n->expr);
4444 : }
4445 100 : for (n = clauses->lists[OMP_LIST_ALIGNED]; n; n = n->next)
4446 : {
4447 6 : mio_name (5, omp_declare_simd_clauses);
4448 6 : mio_symbol_ref (&n->sym);
4449 6 : mio_expr (&n->expr);
4450 : }
4451 : }
4452 : else
4453 : {
4454 54 : if (peek_atom () != ATOM_NAME)
4455 18 : return;
4456 :
4457 36 : gfc_omp_namelist **ptrs[3] = { NULL, NULL, NULL };
4458 36 : gfc_omp_clauses *clauses = *clausesp = gfc_get_omp_clauses ();
4459 36 : ptrs[0] = &clauses->lists[OMP_LIST_UNIFORM];
4460 36 : ptrs[1] = &clauses->lists[OMP_LIST_LINEAR];
4461 36 : ptrs[2] = &clauses->lists[OMP_LIST_ALIGNED];
4462 :
4463 181 : while (peek_atom () == ATOM_NAME)
4464 : {
4465 109 : gfc_omp_namelist *n;
4466 109 : int t = mio_name (0, omp_declare_simd_clauses);
4467 :
4468 109 : switch (t)
4469 : {
4470 0 : case 0: clauses->inbranch = true; break;
4471 10 : case 1: clauses->notinbranch = true; break;
4472 19 : case 2: mio_expr (&clauses->simdlen_expr); break;
4473 77 : case 3:
4474 77 : case 4:
4475 77 : case 5:
4476 77 : *ptrs[t - 3] = n = gfc_get_omp_namelist ();
4477 80 : finish_namelist:
4478 80 : n->where = gfc_current_locus;
4479 80 : ptrs[t - 3] = &n->next;
4480 80 : mio_symbol_ref (&n->sym);
4481 80 : if (t != 3)
4482 32 : mio_expr (&n->expr);
4483 : break;
4484 3 : case 33:
4485 3 : case 34:
4486 3 : case 35:
4487 3 : *ptrs[1] = n = gfc_get_omp_namelist ();
4488 3 : n->u.linear.op = (enum gfc_omp_linear_op) (t - 32);
4489 3 : t = 4;
4490 3 : goto finish_namelist;
4491 : }
4492 : }
4493 : }
4494 : }
4495 :
4496 :
4497 : /* Handle !$omp declare simd. */
4498 :
4499 : static void
4500 254743 : mio_omp_declare_simd (gfc_namespace *ns, gfc_omp_declare_simd **odsp)
4501 : {
4502 254743 : if (iomode == IO_OUTPUT)
4503 : {
4504 29029 : if (*odsp == NULL)
4505 : {
4506 28949 : if (ns->omp_declare_variant)
4507 : {
4508 98 : mio_lparen ();
4509 98 : mio_rparen ();
4510 : }
4511 28949 : return;
4512 : }
4513 : }
4514 225714 : else if (peek_atom () != ATOM_LPAREN)
4515 : return;
4516 :
4517 169 : gfc_omp_declare_simd *ods = *odsp;
4518 :
4519 169 : mio_lparen ();
4520 169 : if (iomode == IO_OUTPUT)
4521 : {
4522 80 : if (ods->clauses)
4523 80 : mio_omp_declare_simd_clauses (&ods->clauses);
4524 : }
4525 : else
4526 : {
4527 89 : if (peek_atom () == ATOM_RPAREN)
4528 : {
4529 35 : mio_rparen ();
4530 35 : return;
4531 : }
4532 :
4533 54 : require_atom (ATOM_NAME);
4534 54 : *odsp = ods = gfc_get_omp_declare_simd ();
4535 54 : ods->where = gfc_current_locus;
4536 54 : ods->proc_name = ns->proc_name;
4537 54 : mio_omp_declare_simd_clauses (&ods->clauses);
4538 : }
4539 :
4540 134 : mio_omp_declare_simd (ns, &ods->next);
4541 :
4542 134 : mio_rparen ();
4543 : }
4544 :
4545 : /* Handle !$omp declare variant. */
4546 :
4547 : static void
4548 274165 : mio_omp_declare_variant (gfc_namespace *ns, gfc_omp_declare_variant **odvp)
4549 : {
4550 274165 : if (iomode == IO_OUTPUT)
4551 : {
4552 48462 : if (*odvp == NULL)
4553 : return;
4554 : }
4555 225703 : else if (peek_atom () != ATOM_LPAREN)
4556 : return;
4557 :
4558 157 : gfc_omp_declare_variant *odv;
4559 :
4560 157 : mio_lparen ();
4561 157 : if (iomode == IO_OUTPUT)
4562 : {
4563 117 : odv = *odvp;
4564 117 : write_atom (ATOM_NAME, "OMP_DECLARE_VARIANT");
4565 117 : gfc_symtree *st;
4566 234 : st = (odv->base_proc_symtree
4567 117 : ? odv->base_proc_symtree
4568 108 : : gfc_find_symtree (ns->sym_root, ns->proc_name->name));
4569 117 : mio_symtree_ref (&st);
4570 234 : st = (st->n.sym->attr.if_source == IFSRC_IFBODY
4571 31 : && st->n.sym->formal_ns == ns
4572 118 : ? gfc_find_symtree (ns->parent->sym_root,
4573 30 : odv->variant_proc_symtree->name)
4574 : : odv->variant_proc_symtree);
4575 117 : mio_symtree_ref (&st);
4576 :
4577 117 : mio_lparen ();
4578 117 : write_atom (ATOM_NAME, "SEL");
4579 253 : for (gfc_omp_set_selector *set = odv->set_selectors; set; set = set->next)
4580 : {
4581 136 : int set_code = set->code;
4582 136 : mio_integer (&set_code);
4583 136 : mio_lparen ();
4584 312 : for (gfc_omp_selector *sel = set->trait_selectors; sel;
4585 176 : sel = sel->next)
4586 : {
4587 176 : int sel_code = sel->code;
4588 176 : mio_integer (&sel_code);
4589 176 : mio_expr (&sel->score);
4590 176 : mio_lparen ();
4591 232 : for (gfc_omp_trait_property *prop = sel->properties; prop;
4592 56 : prop = prop->next)
4593 : {
4594 56 : int kind = prop->property_kind;
4595 56 : mio_integer (&kind);
4596 56 : int is_name = prop->is_name;
4597 56 : mio_integer (&is_name);
4598 56 : switch (prop->property_kind)
4599 : {
4600 11 : case OMP_TRAIT_PROPERTY_DEV_NUM_EXPR:
4601 11 : case OMP_TRAIT_PROPERTY_BOOL_EXPR:
4602 11 : mio_expr (&prop->expr);
4603 11 : break;
4604 3 : case OMP_TRAIT_PROPERTY_ID:
4605 3 : write_atom (ATOM_STRING, prop->name);
4606 3 : break;
4607 28 : case OMP_TRAIT_PROPERTY_NAME_LIST:
4608 28 : if (prop->is_name)
4609 25 : write_atom (ATOM_STRING, prop->name);
4610 : else
4611 3 : mio_expr (&prop->expr);
4612 : break;
4613 14 : case OMP_TRAIT_PROPERTY_CLAUSE_LIST:
4614 14 : {
4615 : /* Currently only declare simd. */
4616 14 : mio_lparen ();
4617 14 : mio_omp_declare_simd_clauses (&prop->clauses);
4618 14 : mio_rparen ();
4619 : }
4620 14 : break;
4621 0 : default:
4622 0 : gcc_unreachable ();
4623 : }
4624 : }
4625 176 : mio_rparen ();
4626 : }
4627 136 : mio_rparen ();
4628 : }
4629 117 : mio_rparen ();
4630 :
4631 117 : mio_lparen ();
4632 117 : write_atom (ATOM_NAME, "ADJ");
4633 225 : for (gfc_omp_namelist *arg = odv->adjust_args_list; arg; arg = arg->next)
4634 : {
4635 108 : int need_ptr = arg->u.adj_args.need_ptr;
4636 108 : int need_addr = arg->u.adj_args.need_addr;
4637 108 : int range_start = arg->u.adj_args.range_start;
4638 108 : int omp_num_args_plus = arg->u.adj_args.omp_num_args_plus;
4639 108 : int omp_num_args_minus = arg->u.adj_args.omp_num_args_minus;
4640 108 : mio_integer (&need_ptr);
4641 108 : mio_integer (&need_addr);
4642 108 : mio_integer (&range_start);
4643 108 : mio_integer (&omp_num_args_plus);
4644 108 : mio_integer (&omp_num_args_minus);
4645 108 : mio_expr (&arg->expr);
4646 : }
4647 117 : mio_rparen ();
4648 :
4649 117 : mio_lparen ();
4650 117 : write_atom (ATOM_NAME, "APP");
4651 155 : for (gfc_omp_namelist *arg = odv->append_args_list; arg; arg = arg->next)
4652 : {
4653 38 : int target = arg->u.init.target;
4654 38 : int targetsync = arg->u.init.targetsync;
4655 38 : mio_integer (&target);
4656 38 : mio_integer (&targetsync);
4657 38 : mio_integer (&arg->u.init.len);
4658 38 : gfc_char_t *p = XALLOCAVEC (gfc_char_t, arg->u.init.len);
4659 409 : for (int i = 0; i < arg->u.init.len; i++)
4660 371 : p[i] = arg->u2.init_interop[i];
4661 38 : mio_allocated_wide_string (p, arg->u.init.len);
4662 : }
4663 117 : mio_rparen ();
4664 : }
4665 : else
4666 : {
4667 40 : if (peek_atom () == ATOM_RPAREN)
4668 : {
4669 0 : mio_rparen ();
4670 0 : return;
4671 : }
4672 :
4673 40 : require_atom (ATOM_NAME);
4674 40 : odv = *odvp = gfc_get_omp_declare_variant ();
4675 40 : odv->where = gfc_current_locus;
4676 :
4677 40 : mio_symtree_ref (&odv->base_proc_symtree);
4678 40 : mio_symtree_ref (&odv->variant_proc_symtree);
4679 :
4680 40 : mio_lparen ();
4681 40 : require_atom (ATOM_NAME); /* SEL */
4682 40 : gfc_omp_set_selector **set = &odv->set_selectors;
4683 82 : while (peek_atom () != ATOM_RPAREN)
4684 : {
4685 42 : *set = gfc_get_omp_set_selector ();
4686 42 : int set_code;
4687 42 : mio_integer (&set_code);
4688 42 : (*set)->code = (enum omp_tss_code) set_code;
4689 :
4690 42 : mio_lparen ();
4691 42 : gfc_omp_selector **sel = &(*set)->trait_selectors;
4692 86 : while (peek_atom () != ATOM_RPAREN)
4693 : {
4694 44 : *sel = gfc_get_omp_selector ();
4695 44 : int sel_code = 0;
4696 44 : mio_integer (&sel_code);
4697 44 : (*sel)->code = (enum omp_ts_code) sel_code;
4698 44 : mio_expr (&(*sel)->score);
4699 :
4700 44 : mio_lparen ();
4701 44 : gfc_omp_trait_property **prop = &(*sel)->properties;
4702 47 : while (peek_atom () != ATOM_RPAREN)
4703 : {
4704 3 : *prop = gfc_get_omp_trait_property ();
4705 3 : int kind = 0, is_name = 0;
4706 3 : mio_integer (&kind);
4707 3 : mio_integer (&is_name);
4708 3 : (*prop)->property_kind = (enum omp_tp_type) kind;
4709 3 : (*prop)->is_name = is_name;
4710 3 : switch ((*prop)->property_kind)
4711 : {
4712 0 : case OMP_TRAIT_PROPERTY_DEV_NUM_EXPR:
4713 0 : case OMP_TRAIT_PROPERTY_BOOL_EXPR:
4714 0 : mio_expr (&(*prop)->expr);
4715 0 : break;
4716 0 : case OMP_TRAIT_PROPERTY_ID:
4717 0 : (*prop)->name = read_string ();
4718 0 : break;
4719 3 : case OMP_TRAIT_PROPERTY_NAME_LIST:
4720 3 : if ((*prop)->is_name)
4721 2 : (*prop)->name = read_string ();
4722 : else
4723 1 : mio_expr (&(*prop)->expr);
4724 : break;
4725 0 : case OMP_TRAIT_PROPERTY_CLAUSE_LIST:
4726 0 : {
4727 : /* Currently only declare simd. */
4728 0 : mio_lparen ();
4729 0 : mio_omp_declare_simd_clauses (&(*prop)->clauses);
4730 0 : mio_rparen ();
4731 : }
4732 0 : break;
4733 0 : default:
4734 0 : gcc_unreachable ();
4735 : }
4736 3 : prop = &(*prop)->next;
4737 : }
4738 44 : mio_rparen ();
4739 44 : sel = &(*sel)->next;
4740 : }
4741 42 : mio_rparen ();
4742 42 : set = &(*set)->next;
4743 : }
4744 40 : mio_rparen ();
4745 :
4746 40 : mio_lparen ();
4747 40 : require_atom (ATOM_NAME); /* ADJ */
4748 40 : gfc_omp_namelist **nl = &odv->adjust_args_list;
4749 122 : while (peek_atom () != ATOM_RPAREN)
4750 : {
4751 82 : *nl = gfc_get_omp_namelist ();
4752 82 : (*nl)->where = gfc_current_locus;
4753 82 : int need_ptr, need_addr, range_start;
4754 82 : int omp_num_args_plus, omp_num_args_minus;
4755 82 : mio_integer (&need_ptr);
4756 82 : mio_integer (&need_addr);
4757 82 : mio_integer (&range_start);
4758 82 : mio_integer (&omp_num_args_plus);
4759 82 : mio_integer (&omp_num_args_minus);
4760 82 : (*nl)->u.adj_args.need_ptr = need_ptr;
4761 82 : (*nl)->u.adj_args.need_addr = need_addr;
4762 82 : (*nl)->u.adj_args.range_start = range_start;
4763 82 : (*nl)->u.adj_args.omp_num_args_plus = omp_num_args_minus;
4764 82 : (*nl)->u.adj_args.omp_num_args_plus = omp_num_args_minus;
4765 82 : mio_expr (&(*nl)->expr);
4766 82 : nl = &(*nl)->next;
4767 : }
4768 40 : mio_rparen ();
4769 :
4770 40 : mio_lparen ();
4771 40 : require_atom (ATOM_NAME); /* APP */
4772 40 : nl = &odv->append_args_list;
4773 58 : while (peek_atom () != ATOM_RPAREN)
4774 : {
4775 18 : *nl = gfc_get_omp_namelist ();
4776 18 : (*nl)->where = gfc_current_locus;
4777 18 : int target, targetsync;
4778 18 : mio_integer (&target);
4779 18 : mio_integer (&targetsync);
4780 18 : mio_integer (&(*nl)->u.init.len);
4781 18 : (*nl)->u.init.target = target;
4782 18 : (*nl)->u.init.targetsync = targetsync;
4783 18 : const gfc_char_t *p = XALLOCAVEC (gfc_char_t, (*nl)->u.init.len); // FIXME: memory handling?
4784 18 : (*nl)->u2.init_interop = XCNEWVEC (char, (*nl)->u.init.len);
4785 18 : p = mio_allocated_wide_string (NULL, (*nl)->u.init.len);
4786 101 : for (int i = 0; i < (*nl)->u.init.len; i++)
4787 83 : (*nl)->u2.init_interop[i] = p[i];
4788 18 : nl = &(*nl)->next;
4789 : }
4790 40 : mio_rparen ();
4791 : }
4792 :
4793 157 : mio_omp_declare_variant (ns, &odv->next);
4794 :
4795 157 : mio_rparen ();
4796 : }
4797 :
4798 : static const mstring omp_declare_reduction_stmt[] =
4799 : {
4800 : minit ("ASSIGN", 0),
4801 : minit ("CALL", 1),
4802 : minit (NULL, -1)
4803 : };
4804 :
4805 :
4806 : static void
4807 293 : mio_omp_udr_expr (gfc_omp_udr *udr, gfc_symbol **sym1, gfc_symbol **sym2,
4808 : gfc_namespace *ns, bool is_initializer)
4809 : {
4810 293 : if (iomode == IO_OUTPUT)
4811 : {
4812 144 : if ((*sym1)->module == NULL)
4813 : {
4814 108 : (*sym1)->module = module_name;
4815 108 : (*sym2)->module = module_name;
4816 : }
4817 144 : mio_symbol_ref (sym1);
4818 144 : mio_symbol_ref (sym2);
4819 144 : if (ns->code->op == EXEC_ASSIGN)
4820 : {
4821 90 : mio_name (0, omp_declare_reduction_stmt);
4822 90 : mio_expr (&ns->code->expr1);
4823 90 : mio_expr (&ns->code->expr2);
4824 : }
4825 : else
4826 : {
4827 54 : int flag;
4828 54 : mio_name (1, omp_declare_reduction_stmt);
4829 54 : mio_symtree_ref (&ns->code->symtree);
4830 54 : mio_actual_arglist (&ns->code->ext.actual, false);
4831 :
4832 54 : flag = ns->code->resolved_isym != NULL;
4833 54 : mio_integer (&flag);
4834 54 : if (flag)
4835 0 : write_atom (ATOM_STRING, ns->code->resolved_isym->name);
4836 : else
4837 54 : mio_symbol_ref (&ns->code->resolved_sym);
4838 : }
4839 : }
4840 : else
4841 : {
4842 149 : pointer_info *p1 = mio_symbol_ref (sym1);
4843 149 : pointer_info *p2 = mio_symbol_ref (sym2);
4844 149 : gfc_symbol *sym;
4845 149 : gcc_assert (p1->u.rsym.ns == p2->u.rsym.ns);
4846 149 : gcc_assert (p1->u.rsym.sym == NULL);
4847 : /* Add hidden symbols to the symtree. */
4848 149 : pointer_info *q = get_integer (p1->u.rsym.ns);
4849 149 : q->u.pointer = (void *) ns;
4850 231 : sym = gfc_new_symbol (is_initializer ? "omp_priv" : "omp_out", ns);
4851 149 : sym->ts = udr->ts;
4852 149 : sym->module = gfc_get_string ("%s", p1->u.rsym.module);
4853 149 : associate_integer_pointer (p1, sym);
4854 149 : sym->attr.omp_udr_artificial_var = 1;
4855 149 : gcc_assert (p2->u.rsym.sym == NULL);
4856 231 : sym = gfc_new_symbol (is_initializer ? "omp_orig" : "omp_in", ns);
4857 149 : sym->ts = udr->ts;
4858 149 : sym->module = gfc_get_string ("%s", p2->u.rsym.module);
4859 149 : associate_integer_pointer (p2, sym);
4860 149 : sym->attr.omp_udr_artificial_var = 1;
4861 149 : if (mio_name (0, omp_declare_reduction_stmt) == 0)
4862 : {
4863 95 : ns->code = gfc_get_code (EXEC_ASSIGN);
4864 95 : mio_expr (&ns->code->expr1);
4865 95 : mio_expr (&ns->code->expr2);
4866 : }
4867 : else
4868 : {
4869 54 : int flag;
4870 54 : ns->code = gfc_get_code (EXEC_CALL);
4871 54 : mio_symtree_ref (&ns->code->symtree);
4872 54 : mio_actual_arglist (&ns->code->ext.actual, false);
4873 :
4874 54 : mio_integer (&flag);
4875 54 : if (flag)
4876 : {
4877 0 : require_atom (ATOM_STRING);
4878 0 : ns->code->resolved_isym = gfc_find_subroutine (atom_string);
4879 0 : free (atom_string);
4880 : }
4881 : else
4882 54 : mio_symbol_ref (&ns->code->resolved_sym);
4883 : }
4884 149 : ns->code->loc = gfc_current_locus;
4885 149 : ns->omp_udr_ns = 1;
4886 : }
4887 293 : }
4888 :
4889 :
4890 : /* Unlike most other routines, the address of the symbol node is already
4891 : fixed on input and the name/module has already been filled in.
4892 : If you update the symbol format here, don't forget to update read_module
4893 : as well (look for "seek to the symbol's component list"). */
4894 :
4895 : static void
4896 1271128 : mio_symbol (gfc_symbol *sym)
4897 : {
4898 1271128 : int intmod = INTMOD_NONE;
4899 :
4900 1271128 : mio_lparen ();
4901 :
4902 1271128 : mio_symbol_attribute (&sym->attr);
4903 :
4904 1271128 : if (sym->attr.pdt_type)
4905 590 : sym->name = gfc_dt_upper_string (sym->name);
4906 :
4907 : /* Note that components are always saved, even if they are supposed
4908 : to be private. Component access is checked during searching. */
4909 1271128 : mio_component_list (&sym->components, sym->attr.vtype);
4910 1271128 : if (sym->components != NULL)
4911 74629 : sym->component_access
4912 74629 : = MIO_NAME (gfc_access) (sym->component_access, access_types);
4913 :
4914 1271128 : mio_typespec (&sym->ts);
4915 1271128 : if (sym->ts.type == BT_CLASS)
4916 15272 : sym->attr.class_ok = 1;
4917 :
4918 1271128 : if (iomode == IO_OUTPUT)
4919 242009 : mio_namespace_ref (&sym->formal_ns);
4920 : else
4921 : {
4922 1029119 : mio_namespace_ref (&sym->formal_ns);
4923 1029119 : if (sym->formal_ns)
4924 225660 : sym->formal_ns->proc_name = sym;
4925 : }
4926 :
4927 : /* Save/restore common block links. */
4928 1271128 : mio_symbol_ref (&sym->common_next);
4929 :
4930 1271128 : mio_formal_arglist (&sym->formal);
4931 :
4932 1271128 : if (sym->attr.flavor == FL_PARAMETER)
4933 255114 : mio_expr (&sym->value);
4934 :
4935 1271128 : mio_array_spec (&sym->as);
4936 :
4937 1271128 : mio_symbol_ref (&sym->result);
4938 :
4939 1271128 : if (sym->attr.cray_pointee)
4940 26 : mio_symbol_ref (&sym->cp_pointer);
4941 :
4942 : /* Load/save the f2k_derived namespace of a derived-type symbol. */
4943 1271128 : mio_full_f2k_derived (sym);
4944 :
4945 : /* PDT types store the symbol specification list here. */
4946 1271128 : mio_actual_arglist (&sym->param_list, true);
4947 :
4948 1271128 : mio_namelist (sym);
4949 :
4950 : /* Add the fields that say whether this is from an intrinsic module,
4951 : and if so, what symbol it is within the module. */
4952 : /* mio_integer (&(sym->from_intmod)); */
4953 1271128 : if (iomode == IO_OUTPUT)
4954 : {
4955 242009 : intmod = sym->from_intmod;
4956 242009 : mio_integer (&intmod);
4957 : }
4958 : else
4959 : {
4960 1029119 : mio_integer (&intmod);
4961 1029119 : if (current_intmod)
4962 317289 : sym->from_intmod = current_intmod;
4963 : else
4964 711830 : sym->from_intmod = (intmod_id) intmod;
4965 : }
4966 :
4967 1271128 : mio_integer (&(sym->intmod_sym_id));
4968 :
4969 1271128 : if (gfc_fl_struct (sym->attr.flavor))
4970 77960 : mio_integer (&(sym->hash_value));
4971 :
4972 1271128 : if (sym->formal_ns
4973 255152 : && sym->formal_ns->proc_name == sym
4974 254609 : && sym->formal_ns->entries == NULL)
4975 : {
4976 254609 : mio_omp_declare_simd (sym->formal_ns, &sym->formal_ns->omp_declare_simd);
4977 254609 : mio_omp_declare_variant (sym->formal_ns,
4978 254609 : &sym->formal_ns->omp_declare_variant);
4979 : }
4980 213060 : else if ((iomode == IO_OUTPUT && sym->ns->proc_name == sym)
4981 1210183 : || (iomode == IO_INPUT && peek_atom () == ATOM_LPAREN))
4982 19399 : mio_omp_declare_variant (sym->ns, &sym->ns->omp_declare_variant);
4983 :
4984 1271128 : mio_rparen ();
4985 1271128 : }
4986 :
4987 :
4988 : /************************* Top level subroutines *************************/
4989 :
4990 : /* A recursive function to look for a specific symbol by name and by
4991 : module. Whilst several symtrees might point to one symbol, its
4992 : is sufficient for the purposes here than one exist. Note that
4993 : generic interfaces are distinguished as are symbols that have been
4994 : renamed in another module. */
4995 : static gfc_symtree *
4996 44260304 : find_symbol (gfc_symtree *st, const char *name,
4997 : const char *module, int generic)
4998 : {
4999 88015846 : int c;
5000 88015846 : gfc_symtree *retval, *s;
5001 :
5002 88015846 : if (st == NULL || st->n.sym == NULL)
5003 : return NULL;
5004 :
5005 43757955 : c = strcmp (name, st->n.sym->name);
5006 96609 : if (c == 0 && st->n.sym->module
5007 96603 : && strcmp (module, st->n.sym->module) == 0
5008 43797683 : && !check_unique_name (st->name))
5009 : {
5010 39642 : s = gfc_find_symtree (gfc_current_ns->sym_root, name);
5011 :
5012 : /* Detect symbols that are renamed by use association in another
5013 : module by the absence of a symtree and null attr.use_rename,
5014 : since the latter is not transmitted in the module file. */
5015 39642 : if (((!generic && !st->n.sym->attr.generic)
5016 30711 : || (generic && st->n.sym->attr.generic))
5017 8971 : && !(s == NULL && !st->n.sym->attr.use_rename))
5018 : return st;
5019 : }
5020 :
5021 43757361 : retval = find_symbol (st->left, name, module, generic);
5022 :
5023 43757361 : if (retval == NULL)
5024 43755542 : retval = find_symbol (st->right, name, module, generic);
5025 :
5026 : return retval;
5027 : }
5028 :
5029 :
5030 : /* Skip a list between balanced left and right parens.
5031 : By setting NEST_LEVEL one assumes that a number of NEST_LEVEL opening parens
5032 : have been already parsed by hand, and the remaining of the content is to be
5033 : skipped here. The default value is 0 (balanced parens). */
5034 :
5035 : static void
5036 1381079 : skip_list (int nest_level = 0)
5037 : {
5038 1381079 : int level;
5039 :
5040 1381079 : level = nest_level;
5041 65068304 : do
5042 : {
5043 65068304 : switch (parse_atom ())
5044 : {
5045 16106063 : case ATOM_LPAREN:
5046 16106063 : level++;
5047 16106063 : break;
5048 :
5049 16123041 : case ATOM_RPAREN:
5050 16123041 : level--;
5051 16123041 : break;
5052 :
5053 712670 : case ATOM_STRING:
5054 712670 : free (atom_string);
5055 712670 : break;
5056 :
5057 : case ATOM_NAME:
5058 : case ATOM_INTEGER:
5059 : break;
5060 : }
5061 : }
5062 65068304 : while (level > 0);
5063 1381079 : }
5064 :
5065 :
5066 : /* Load operator interfaces from the module. Interfaces are unusual
5067 : in that they attach themselves to existing symbols. */
5068 :
5069 : static void
5070 13571 : load_operator_interfaces (void)
5071 : {
5072 13571 : const char *p;
5073 : /* "module" must be large enough for the case of submodules in which the name
5074 : has the form module.submodule */
5075 13571 : char name[GFC_MAX_SYMBOL_LEN + 1], module[2 * GFC_MAX_SYMBOL_LEN + 2];
5076 13571 : gfc_user_op *uop;
5077 13571 : pointer_info *pi = NULL;
5078 13571 : int n, i;
5079 :
5080 13571 : mio_lparen ();
5081 :
5082 27305 : while (peek_atom () != ATOM_RPAREN)
5083 : {
5084 163 : mio_lparen ();
5085 :
5086 163 : mio_internal_string (name);
5087 163 : mio_internal_string (module);
5088 :
5089 163 : n = number_use_names (name, true);
5090 163 : n = n ? n : 1;
5091 :
5092 344 : for (i = 1; i <= n; i++)
5093 : {
5094 : /* Decide if we need to load this one or not. */
5095 181 : p = find_use_name_n (name, &i, true);
5096 :
5097 181 : if (p == NULL)
5098 : {
5099 14 : while (parse_atom () != ATOM_RPAREN);
5100 7 : continue;
5101 : }
5102 :
5103 174 : if (i == 1)
5104 : {
5105 156 : uop = gfc_get_uop (p);
5106 156 : pi = mio_interface_rest (&uop->op);
5107 : }
5108 : else
5109 : {
5110 18 : if (gfc_find_uop (p, NULL))
5111 6 : continue;
5112 12 : uop = gfc_get_uop (p);
5113 12 : uop->op = gfc_get_interface ();
5114 12 : uop->op->where = gfc_current_locus;
5115 12 : add_fixup (pi->integer, &uop->op->sym);
5116 : }
5117 : }
5118 : }
5119 :
5120 13571 : mio_rparen ();
5121 13571 : }
5122 :
5123 :
5124 : /* Load interfaces from the module. Interfaces are unusual in that
5125 : they attach themselves to existing symbols. */
5126 :
5127 : static void
5128 13571 : load_generic_interfaces (void)
5129 : {
5130 13571 : const char *p;
5131 : /* "module" must be large enough for the case of submodules in which the name
5132 : has the form module.submodule */
5133 13571 : char name[GFC_MAX_SYMBOL_LEN + 1], module[2 * GFC_MAX_SYMBOL_LEN + 2];
5134 13571 : gfc_symbol *sym;
5135 13571 : gfc_interface *generic = NULL, *gen = NULL;
5136 13571 : int n, i, renamed;
5137 13571 : bool ambiguous_set = false;
5138 :
5139 13571 : mio_lparen ();
5140 :
5141 83663 : while (peek_atom () != ATOM_RPAREN)
5142 : {
5143 56521 : mio_lparen ();
5144 :
5145 56521 : mio_internal_string (name);
5146 56521 : mio_internal_string (module);
5147 :
5148 56521 : n = number_use_names (name, false);
5149 56521 : renamed = n ? 1 : 0;
5150 55745 : n = n ? n : 1;
5151 :
5152 113046 : for (i = 1; i <= n; i++)
5153 : {
5154 56525 : gfc_symtree *st;
5155 : /* Decide if we need to load this one or not. */
5156 56525 : p = find_use_name_n (name, &i, false);
5157 :
5158 56525 : if (!p || gfc_find_symbol (p, NULL, 0, &sym))
5159 : {
5160 : /* Skip the specific names for these cases. */
5161 9475 : while (i == 1 && parse_atom () != ATOM_RPAREN);
5162 :
5163 1726 : continue;
5164 : }
5165 :
5166 54799 : st = find_symbol (gfc_current_ns->sym_root,
5167 : name, module_name, 1);
5168 :
5169 : /* If the symbol exists already and is being USEd without being
5170 : in an ONLY clause, do not load a new symtree(11.3.2). */
5171 54799 : if (!only_flag && st)
5172 44 : sym = st->n.sym;
5173 :
5174 54799 : if (!sym)
5175 : {
5176 27323 : if (st)
5177 : {
5178 1 : sym = st->n.sym;
5179 1 : if (strcmp (st->name, p) != 0)
5180 : {
5181 1 : st = gfc_new_symtree (&gfc_current_ns->sym_root, p);
5182 1 : st->n.sym = sym;
5183 1 : sym->refs++;
5184 : }
5185 : }
5186 :
5187 : /* Since we haven't found a valid generic interface, we had
5188 : better make one. */
5189 27323 : if (!sym)
5190 : {
5191 27322 : gfc_get_symbol (p, NULL, &sym);
5192 27322 : sym->name = gfc_get_string ("%s", name);
5193 27322 : sym->module = module_name;
5194 27322 : sym->attr.flavor = FL_PROCEDURE;
5195 27322 : sym->attr.generic = 1;
5196 27322 : sym->attr.use_assoc = 1;
5197 : }
5198 : }
5199 : else
5200 : {
5201 : /* Unless sym is a generic interface, this reference
5202 : is ambiguous. */
5203 27476 : if (st == NULL)
5204 27431 : st = gfc_find_symtree (gfc_current_ns->sym_root, p);
5205 :
5206 27476 : sym = st->n.sym;
5207 :
5208 27476 : if (st && !sym->attr.generic
5209 25121 : && !st->ambiguous
5210 25121 : && sym->module
5211 25120 : && strcmp (module, sym->module))
5212 : {
5213 1 : ambiguous_set = true;
5214 1 : st->ambiguous = 1;
5215 : }
5216 : }
5217 :
5218 54799 : sym->attr.use_only = only_flag;
5219 54799 : sym->attr.use_rename = renamed;
5220 :
5221 54799 : if (i == 1)
5222 : {
5223 54795 : mio_interface_rest (&sym->generic);
5224 54795 : generic = sym->generic;
5225 : }
5226 4 : else if (!sym->generic)
5227 : {
5228 0 : sym->generic = generic;
5229 0 : sym->attr.generic_copy = 1;
5230 : }
5231 :
5232 : /* If a procedure that is not generic has generic interfaces
5233 : that include itself, it is generic! We need to take care
5234 : to retain symbols ambiguous that were already so. */
5235 54799 : if (sym->attr.use_assoc
5236 29679 : && !sym->attr.generic
5237 2 : && sym->attr.flavor == FL_PROCEDURE)
5238 : {
5239 4 : for (gen = generic; gen; gen = gen->next)
5240 : {
5241 3 : if (gen->sym == sym)
5242 : {
5243 1 : sym->attr.generic = 1;
5244 1 : if (ambiguous_set)
5245 0 : st->ambiguous = 0;
5246 : break;
5247 : }
5248 : }
5249 : }
5250 :
5251 : }
5252 : }
5253 :
5254 13571 : mio_rparen ();
5255 13571 : }
5256 :
5257 :
5258 : /* Load common blocks. */
5259 :
5260 : static void
5261 13571 : load_commons (void)
5262 : {
5263 13571 : char name[GFC_MAX_SYMBOL_LEN + 1];
5264 13571 : gfc_common_head *p;
5265 :
5266 13571 : mio_lparen ();
5267 :
5268 27312 : while (peek_atom () != ATOM_RPAREN)
5269 : {
5270 170 : int flags = 0;
5271 170 : char* label;
5272 170 : mio_lparen ();
5273 170 : mio_internal_string (name);
5274 :
5275 170 : p = gfc_get_common (name, 1);
5276 :
5277 170 : mio_symbol_ref (&p->head);
5278 170 : mio_integer (&flags);
5279 170 : if (flags & 1)
5280 0 : p->saved = 1;
5281 170 : if (flags & 2)
5282 0 : p->threadprivate = 1;
5283 170 : p->omp_device_type = (gfc_omp_device_type) ((flags >> 2) & 3);
5284 170 : if ((flags >> 4) & 1)
5285 0 : p->omp_groupprivate = 1;
5286 170 : p->use_assoc = 1;
5287 :
5288 : /* Get whether this was a bind(c) common or not. */
5289 170 : mio_integer (&p->is_bind_c);
5290 : /* Get the binding label. */
5291 170 : label = read_string ();
5292 170 : if (strlen (label))
5293 22 : p->binding_label = IDENTIFIER_POINTER (get_identifier (label));
5294 170 : XDELETEVEC (label);
5295 :
5296 170 : mio_rparen ();
5297 : }
5298 :
5299 13571 : mio_rparen ();
5300 13571 : }
5301 :
5302 :
5303 : /* Load equivalences. The flag in_load_equiv informs mio_expr_ref of this
5304 : so that unused variables are not loaded and so that the expression can
5305 : be safely freed. */
5306 :
5307 : static void
5308 13571 : load_equiv (void)
5309 : {
5310 13571 : gfc_equiv *head, *tail, *end, *eq, *equiv;
5311 13571 : bool duplicate;
5312 :
5313 13571 : mio_lparen ();
5314 13571 : in_load_equiv = true;
5315 :
5316 13571 : end = gfc_current_ns->equiv;
5317 13577 : while (end != NULL && end->next != NULL)
5318 : end = end->next;
5319 :
5320 13704 : while (peek_atom () != ATOM_RPAREN) {
5321 133 : mio_lparen ();
5322 133 : head = tail = NULL;
5323 :
5324 532 : while(peek_atom () != ATOM_RPAREN)
5325 : {
5326 266 : if (head == NULL)
5327 133 : head = tail = gfc_get_equiv ();
5328 : else
5329 : {
5330 133 : tail->eq = gfc_get_equiv ();
5331 133 : tail = tail->eq;
5332 : }
5333 :
5334 266 : mio_pool_string (&tail->module);
5335 266 : mio_expr (&tail->expr);
5336 : }
5337 :
5338 : /* Check for duplicate equivalences being loaded from different modules */
5339 133 : duplicate = false;
5340 192 : for (equiv = gfc_current_ns->equiv; equiv; equiv = equiv->next)
5341 : {
5342 65 : if (equiv->module && head->module
5343 65 : && strcmp (equiv->module, head->module) == 0)
5344 : {
5345 : duplicate = true;
5346 : break;
5347 : }
5348 : }
5349 :
5350 133 : if (duplicate)
5351 : {
5352 18 : for (eq = head; eq; eq = head)
5353 : {
5354 12 : head = eq->eq;
5355 12 : gfc_free_expr (eq->expr);
5356 12 : free (eq);
5357 : }
5358 : }
5359 :
5360 133 : if (end == NULL)
5361 80 : gfc_current_ns->equiv = head;
5362 : else
5363 53 : end->next = head;
5364 :
5365 133 : if (head != NULL)
5366 127 : end = head;
5367 :
5368 133 : mio_rparen ();
5369 : }
5370 :
5371 13571 : mio_rparen ();
5372 13571 : in_load_equiv = false;
5373 13571 : }
5374 :
5375 :
5376 : /* This function loads OpenMP user defined reductions. */
5377 : static void
5378 13571 : load_omp_udrs (void)
5379 : {
5380 13571 : mio_lparen ();
5381 27230 : while (peek_atom () != ATOM_RPAREN)
5382 : {
5383 88 : const char *name = NULL, *newname;
5384 88 : char *altname;
5385 88 : gfc_typespec ts;
5386 88 : gfc_symtree *st;
5387 88 : gfc_omp_reduction_op rop = OMP_REDUCTION_USER;
5388 :
5389 88 : mio_lparen ();
5390 88 : mio_pool_string (&name);
5391 88 : gfc_clear_ts (&ts);
5392 88 : mio_typespec (&ts);
5393 88 : if (startswith (name, "operator "))
5394 : {
5395 38 : const char *p = name + sizeof ("operator ") - 1;
5396 38 : if (strcmp (p, "+") == 0)
5397 : rop = OMP_REDUCTION_PLUS;
5398 0 : else if (strcmp (p, "*") == 0)
5399 : rop = OMP_REDUCTION_TIMES;
5400 0 : else if (strcmp (p, "-") == 0)
5401 : rop = OMP_REDUCTION_MINUS;
5402 0 : else if (strcmp (p, ".and.") == 0)
5403 : rop = OMP_REDUCTION_AND;
5404 0 : else if (strcmp (p, ".or.") == 0)
5405 : rop = OMP_REDUCTION_OR;
5406 0 : else if (strcmp (p, ".eqv.") == 0)
5407 : rop = OMP_REDUCTION_EQV;
5408 0 : else if (strcmp (p, ".neqv.") == 0)
5409 : rop = OMP_REDUCTION_NEQV;
5410 : }
5411 50 : altname = NULL;
5412 50 : if (rop == OMP_REDUCTION_USER && name[0] == '.')
5413 : {
5414 50 : size_t len = strlen (name + 1);
5415 50 : altname = XALLOCAVEC (char, len);
5416 50 : gcc_assert (name[len] == '.');
5417 50 : memcpy (altname, name + 1, len - 1);
5418 50 : altname[len - 1] = '\0';
5419 : }
5420 88 : newname = name;
5421 88 : if (rop == OMP_REDUCTION_USER)
5422 100 : newname = find_use_name (altname ? altname : name, !!altname);
5423 44 : else if (only_flag && find_use_operator ((gfc_intrinsic_op) rop) == NULL)
5424 : newname = NULL;
5425 88 : if (newname == NULL)
5426 : {
5427 0 : skip_list (1);
5428 6 : continue;
5429 : }
5430 88 : if (altname && newname != altname)
5431 : {
5432 18 : size_t len = strlen (newname);
5433 18 : altname = XALLOCAVEC (char, len + 3);
5434 18 : altname[0] = '.';
5435 18 : memcpy (altname + 1, newname, len);
5436 18 : altname[len + 1] = '.';
5437 18 : altname[len + 2] = '\0';
5438 18 : name = gfc_get_string ("%s", altname);
5439 : }
5440 88 : st = gfc_find_symtree (gfc_current_ns->omp_udr_root, name);
5441 88 : gfc_omp_udr *udr = gfc_omp_udr_find (st, &ts);
5442 88 : if (udr)
5443 : {
5444 6 : require_atom (ATOM_INTEGER);
5445 6 : pointer_info *p = get_integer (atom_int);
5446 6 : if (strcmp (p->u.rsym.module, udr->omp_out->module))
5447 : {
5448 6 : gcc_assert (!gfc_buffered_p ()); /* Cf. PR80012 comment 15. */
5449 6 : auto_diagnostic_group d;
5450 6 : gfc_error ("Ambiguous !$OMP DECLARE REDUCTION %qs for type %qs "
5451 : "from module %qs at %L", udr->name,
5452 : gfc_typename (&ts), module_name, &gfc_current_locus);
5453 6 : inform (gfc_get_location (&udr->where),
5454 : "Previous !$OMP DECLARE REDUCTION from module %qs",
5455 6 : udr->omp_out->module);
5456 6 : }
5457 6 : skip_list (1);
5458 6 : continue;
5459 6 : }
5460 82 : udr = gfc_get_omp_udr ();
5461 82 : udr->name = name;
5462 82 : udr->rop = rop;
5463 82 : udr->ts = ts;
5464 82 : udr->where = gfc_current_locus;
5465 82 : udr->combiner_ns = gfc_get_namespace (gfc_current_ns, 1);
5466 82 : udr->combiner_ns->proc_name = gfc_current_ns->proc_name;
5467 82 : mio_omp_udr_expr (udr, &udr->omp_out, &udr->omp_in, udr->combiner_ns,
5468 : false);
5469 82 : if (peek_atom () != ATOM_RPAREN)
5470 : {
5471 67 : udr->initializer_ns = gfc_get_namespace (gfc_current_ns, 1);
5472 67 : udr->initializer_ns->proc_name = gfc_current_ns->proc_name;
5473 67 : mio_omp_udr_expr (udr, &udr->omp_priv, &udr->omp_orig,
5474 : udr->initializer_ns, true);
5475 : }
5476 82 : if (st)
5477 : {
5478 1 : udr->next = st->n.omp_udr;
5479 1 : st->n.omp_udr = udr;
5480 : }
5481 : else
5482 : {
5483 81 : st = gfc_new_symtree (&gfc_current_ns->omp_udr_root, name);
5484 81 : st->n.omp_udr = udr;
5485 : }
5486 82 : mio_rparen ();
5487 : }
5488 13571 : mio_rparen ();
5489 13571 : }
5490 :
5491 :
5492 : /* In declare mapper, not all map types are permitted; hence, only
5493 : a subset is needed. */
5494 :
5495 : static const mstring omp_map_clause_ops[] =
5496 : {
5497 : minit ("ALLOC", OMP_MAP_ALLOC),
5498 : minit ("TO", OMP_MAP_TO),
5499 : minit ("FROM", OMP_MAP_FROM),
5500 : minit ("TOFROM", OMP_MAP_TOFROM),
5501 : minit ("ALWAYS_TO", OMP_MAP_ALWAYS_TO),
5502 : minit ("ALWAYS_FROM", OMP_MAP_ALWAYS_FROM),
5503 : minit ("ALWAYS_TOFROM", OMP_MAP_ALWAYS_TOFROM),
5504 : minit ("UNSET", OMP_MAP_UNSET),
5505 : minit (NULL, -1)
5506 : };
5507 :
5508 : /* This function loads OpenMP user-defined mappers. */
5509 :
5510 : static void
5511 8 : load_omp_udms (void)
5512 : {
5513 17 : while (peek_atom () != ATOM_RPAREN)
5514 : {
5515 9 : const char *mapper_id = NULL;
5516 9 : gfc_symtree *st;
5517 :
5518 9 : mio_lparen ();
5519 9 : gfc_omp_udm *udm = gfc_get_omp_udm ();
5520 :
5521 9 : require_atom (ATOM_INTEGER);
5522 9 : pointer_info *udmpi = get_integer (atom_int);
5523 9 : associate_integer_pointer (udmpi, udm);
5524 :
5525 9 : mio_pool_string (&mapper_id);
5526 :
5527 : /* Note: for a derived-type typespec, we might not have loaded the
5528 : "u.derived" symbol yet. Defer checking duplicates until
5529 : check_omp_declare_mappers is called after loading all symbols. */
5530 9 : mio_typespec (&udm->ts);
5531 :
5532 9 : if (mapper_id == NULL)
5533 8 : mapper_id = gfc_get_string ("%s", "");
5534 :
5535 9 : st = gfc_find_symtree (gfc_current_ns->omp_udm_root, mapper_id);
5536 :
5537 9 : pointer_info *p = mio_symbol_ref (&udm->var_sym);
5538 9 : pointer_info *q = get_integer (p->u.rsym.ns);
5539 :
5540 9 : udm->where = gfc_current_locus;
5541 9 : udm->mapper_id = mapper_id;
5542 9 : udm->mapper_ns = gfc_get_namespace (gfc_current_ns, 1);
5543 9 : udm->mapper_ns->proc_name = gfc_current_ns->proc_name;
5544 9 : udm->mapper_ns->omp_udm_ns = 1;
5545 :
5546 9 : associate_integer_pointer (q, udm->mapper_ns);
5547 :
5548 9 : gfc_omp_namelist *clauses = NULL;
5549 9 : gfc_omp_namelist **clausep = &clauses;
5550 :
5551 9 : mio_lparen ();
5552 35 : while (peek_atom () != ATOM_RPAREN)
5553 : {
5554 : /* Read each map clause. */
5555 17 : gfc_omp_namelist *n = gfc_get_omp_namelist ();
5556 :
5557 17 : mio_lparen ();
5558 :
5559 17 : n->u.map.op = (gfc_omp_map_op) mio_name (0, omp_map_clause_ops);
5560 17 : mio_symbol_ref (&n->sym);
5561 17 : mio_expr (&n->expr);
5562 :
5563 17 : mio_lparen ();
5564 :
5565 17 : if (peek_atom () != ATOM_RPAREN)
5566 : {
5567 7 : n->u3.udm = gfc_get_omp_namelist_udm ();
5568 7 : mio_pool_string (&n->u3.udm->requested_mapper_id);
5569 :
5570 7 : if (n->u3.udm->requested_mapper_id == NULL)
5571 7 : n->u3.udm->requested_mapper_id = gfc_get_string ("%s", "");
5572 :
5573 7 : mio_pointer_ref (&n->u3.udm->resolved_udm);
5574 : }
5575 :
5576 17 : mio_rparen ();
5577 :
5578 17 : n->where = gfc_current_locus;
5579 :
5580 17 : mio_rparen ();
5581 :
5582 17 : *clausep = n;
5583 17 : clausep = &n->next;
5584 : }
5585 9 : mio_rparen ();
5586 :
5587 9 : udm->clauses = gfc_get_omp_clauses ();
5588 9 : udm->clauses->lists[OMP_LIST_MAP] = clauses;
5589 :
5590 9 : if (st)
5591 : {
5592 3 : udm->next = st->n.omp_udm;
5593 3 : st->n.omp_udm = udm;
5594 : }
5595 : else
5596 : {
5597 6 : st = gfc_new_symtree (&gfc_current_ns->omp_udm_root, mapper_id);
5598 6 : st->n.omp_udm = udm;
5599 : }
5600 :
5601 9 : mio_rparen ();
5602 : }
5603 8 : }
5604 :
5605 :
5606 : /* Recursive function to traverse the pointer_info tree and load a
5607 : needed symbol. We return nonzero if we load a symbol and stop the
5608 : traversal, because the act of loading can alter the tree. */
5609 :
5610 : static int
5611 9940909 : load_needed (pointer_info *p)
5612 : {
5613 9940909 : gfc_namespace *ns;
5614 9940909 : pointer_info *q;
5615 9940909 : gfc_symbol *sym;
5616 9940909 : int rv;
5617 :
5618 9940909 : rv = 0;
5619 9940909 : if (p == NULL)
5620 : return rv;
5621 :
5622 4951535 : rv |= load_needed (p->left);
5623 4951535 : rv |= load_needed (p->right);
5624 :
5625 4951535 : if (p->type != P_SYMBOL || p->u.rsym.state != NEEDED)
5626 : return rv;
5627 :
5628 1029119 : p->u.rsym.state = USED;
5629 :
5630 1029119 : set_module_locus (&p->u.rsym.where);
5631 :
5632 1029119 : sym = p->u.rsym.sym;
5633 1029119 : if (sym == NULL)
5634 : {
5635 607456 : q = get_integer (p->u.rsym.ns);
5636 :
5637 607456 : ns = (gfc_namespace *) q->u.pointer;
5638 607456 : if (ns == NULL)
5639 : {
5640 : /* Create an interface namespace if necessary. These are
5641 : the namespaces that hold the formal parameters of module
5642 : procedures. */
5643 :
5644 21749 : ns = gfc_get_namespace (NULL, 0);
5645 21749 : associate_integer_pointer (q, ns);
5646 : }
5647 :
5648 : /* Use the module sym as 'proc_name' so that gfc_get_symbol_decl
5649 : doesn't go pear-shaped if the symbol is used. */
5650 607456 : if (!ns->proc_name)
5651 30494 : gfc_find_symbol (p->u.rsym.module, gfc_current_ns,
5652 : 1, &ns->proc_name);
5653 :
5654 607456 : sym = gfc_new_symbol (p->u.rsym.true_name, ns);
5655 607456 : sym->name = gfc_dt_lower_string (p->u.rsym.true_name);
5656 607456 : sym->module = gfc_get_string ("%s", p->u.rsym.module);
5657 607456 : if (p->u.rsym.binding_label)
5658 21 : sym->binding_label = IDENTIFIER_POINTER (get_identifier
5659 : (p->u.rsym.binding_label));
5660 :
5661 607456 : associate_integer_pointer (p, sym);
5662 : }
5663 :
5664 1029119 : mio_symbol (sym);
5665 1029119 : sym->attr.use_assoc = 1;
5666 :
5667 : /* Unliked derived types, a STRUCTURE may share names with other symbols.
5668 : We greedily converted the symbol name to lowercase before we knew its
5669 : type, so now we must fix it. */
5670 1029119 : if (sym->attr.flavor == FL_STRUCT)
5671 60 : sym->name = gfc_dt_upper_string (sym->name);
5672 :
5673 : /* Mark as only or rename for later diagnosis for explicitly imported
5674 : but not used warnings; don't mark internal symbols such as __vtab,
5675 : __def_init etc. Only mark them if they have been explicitly loaded. */
5676 :
5677 1029119 : if (only_flag && sym->name[0] != '_' && sym->name[1] != '_')
5678 : {
5679 12237 : gfc_use_rename *u;
5680 :
5681 : /* Search the use/rename list for the variable; if the variable is
5682 : found, mark it. */
5683 30111 : for (u = gfc_rename_list; u; u = u->next)
5684 : {
5685 20859 : if (strcmp (u->use_name, sym->name) == 0)
5686 : {
5687 2985 : sym->attr.use_only = 1;
5688 2985 : break;
5689 : }
5690 : }
5691 : }
5692 :
5693 1029119 : if (p->u.rsym.renamed)
5694 3370 : sym->attr.use_rename = 1;
5695 :
5696 : return 1;
5697 : }
5698 :
5699 :
5700 : /* Recursive function for cleaning up things after a module has been read. */
5701 :
5702 : static void
5703 3370687 : read_cleanup (pointer_info *p)
5704 : {
5705 3370687 : gfc_symtree *st;
5706 3370687 : pointer_info *q;
5707 :
5708 3370687 : if (p == NULL)
5709 : return;
5710 :
5711 1678558 : read_cleanup (p->left);
5712 1678558 : read_cleanup (p->right);
5713 :
5714 1678558 : if (p->type == P_SYMBOL && p->u.rsym.state == USED && !p->u.rsym.referenced)
5715 : {
5716 655621 : gfc_namespace *ns;
5717 : /* Add hidden symbols to the symtree. */
5718 655621 : q = get_integer (p->u.rsym.ns);
5719 655621 : ns = (gfc_namespace *) q->u.pointer;
5720 :
5721 655621 : if (!p->u.rsym.sym->attr.vtype
5722 654509 : && !p->u.rsym.sym->attr.vtab)
5723 653709 : st = gfc_get_unique_symtree (ns);
5724 : else
5725 : {
5726 : /* There is no reason to use 'unique_symtrees' for vtabs or
5727 : vtypes - their name is fine for a symtree and reduces the
5728 : namespace pollution. */
5729 1912 : st = gfc_find_symtree (ns->sym_root, p->u.rsym.sym->name);
5730 1912 : if (!st)
5731 300 : st = gfc_new_symtree (&ns->sym_root, p->u.rsym.sym->name);
5732 : }
5733 :
5734 655621 : st->n.sym = p->u.rsym.sym;
5735 655621 : st->n.sym->refs++;
5736 :
5737 : /* Fixup any symtree references. */
5738 655621 : p->u.rsym.symtree = st;
5739 655621 : resolve_fixups (p->u.rsym.stfixup, st);
5740 655621 : p->u.rsym.stfixup = NULL;
5741 : }
5742 :
5743 : /* Free unused symbols. */
5744 1678558 : if (p->type == P_SYMBOL && p->u.rsym.state == UNUSED)
5745 155279 : gfc_free_symbol (p->u.rsym.sym);
5746 : }
5747 :
5748 :
5749 : /* It is not quite enough to check for ambiguity in the symbols by
5750 : the loaded symbol and the new symbol not being identical. */
5751 : static bool
5752 43536 : check_for_ambiguous (gfc_symtree *st, pointer_info *info)
5753 : {
5754 43536 : gfc_symbol *rsym;
5755 43536 : module_locus locus;
5756 43536 : symbol_attribute attr;
5757 43536 : gfc_symbol *st_sym;
5758 :
5759 43536 : if (gfc_current_ns->proc_name && st->name == gfc_current_ns->proc_name->name)
5760 : {
5761 6 : gfc_error ("%qs of module %qs, imported at %C, is also the name of the "
5762 : "current program unit", st->name, module_name);
5763 6 : return true;
5764 : }
5765 :
5766 43530 : st_sym = st->n.sym;
5767 43530 : rsym = info->u.rsym.sym;
5768 43530 : if (st_sym == rsym)
5769 : return false;
5770 :
5771 609 : if (st_sym->attr.vtab || st_sym->attr.vtype)
5772 : return false;
5773 :
5774 : /* If the existing symbol is generic from a different module and
5775 : the new symbol is generic there can be no ambiguity. */
5776 421 : if (st_sym->attr.generic
5777 20 : && st_sym->module
5778 20 : && st_sym->module != module_name)
5779 : {
5780 : /* The new symbol's attributes have not yet been read. Since
5781 : we need attr.generic, read it directly. */
5782 20 : get_module_locus (&locus);
5783 20 : set_module_locus (&info->u.rsym.where);
5784 20 : mio_lparen ();
5785 20 : attr.generic = 0;
5786 20 : mio_symbol_attribute (&attr);
5787 20 : set_module_locus (&locus);
5788 20 : if (attr.generic)
5789 : return false;
5790 : }
5791 :
5792 : return true;
5793 : }
5794 :
5795 :
5796 : static void
5797 13589 : check_omp_declare_mappers (gfc_symtree *st)
5798 : {
5799 13589 : if (!st)
5800 13580 : return;
5801 :
5802 9 : check_omp_declare_mappers (st->left);
5803 9 : check_omp_declare_mappers (st->right);
5804 :
5805 9 : gfc_omp_udm **udmp = &st->n.omp_udm;
5806 9 : gfc_symtree tmp_st;
5807 :
5808 21 : while (*udmp)
5809 : {
5810 12 : gfc_omp_udm *udm = *udmp;
5811 12 : tmp_st.n.omp_udm = udm->next;
5812 12 : gfc_omp_udm *prev_udm = gfc_omp_udm_find (&tmp_st, &udm->ts);
5813 12 : if (prev_udm)
5814 : {
5815 2 : gcc_assert (!gfc_buffered_p ()); /* Cf. PR80012 comment 15. */
5816 2 : auto_diagnostic_group d;
5817 2 : gfc_error ("Ambiguous !$OMP DECLARE MAPPER %qs for type %qs from "
5818 : "module %qs at %L",
5819 2 : st->n.omp_udm->mapper_id[0] != '\0'
5820 : ? st->n.omp_udm->mapper_id : "default",
5821 2 : udm->ts.u.derived->name, module_name,
5822 : &udm->where);
5823 2 : inform (gfc_get_location (&prev_udm->where),
5824 : "Previous !$OMP DECLARE MAPPER from module %qs",
5825 2 : prev_udm->var_sym->module);
5826 : /* Delete the duplicate. */
5827 2 : *udmp = (*udmp)->next;
5828 2 : }
5829 : else
5830 10 : udmp = &(*udmp)->next;
5831 : }
5832 : }
5833 :
5834 :
5835 : /* Read a module file. */
5836 :
5837 : static void
5838 13571 : read_module (void)
5839 : {
5840 13571 : module_locus operator_interfaces, user_operators, omp_udrs, omp_udms;
5841 13571 : bool has_omp_udms = false;
5842 13571 : const char *p;
5843 13571 : char name[GFC_MAX_SYMBOL_LEN + 1];
5844 13571 : int i;
5845 : /* Workaround -Wmaybe-uninitialized false positive during
5846 : profiledbootstrap by initializing them. */
5847 13571 : int ambiguous = 0, j, nuse, symbol = 0;
5848 13571 : pointer_info *info, *q;
5849 13571 : gfc_use_rename *u = NULL;
5850 13571 : gfc_symtree *st;
5851 13571 : gfc_symbol *sym;
5852 :
5853 13571 : get_module_locus (&operator_interfaces); /* Skip these for now. */
5854 13571 : skip_list ();
5855 :
5856 13571 : get_module_locus (&user_operators);
5857 13571 : skip_list ();
5858 13571 : skip_list ();
5859 :
5860 : /* Skip commons and equivalences for now. */
5861 13571 : skip_list ();
5862 13571 : skip_list ();
5863 :
5864 : /* Skip OpenMP UDRs. */
5865 13571 : get_module_locus (&omp_udrs);
5866 13571 : skip_list ();
5867 :
5868 : /* Skip OpenMP's user-defined 'declare mapper' (UDM); some extra code is
5869 : required to permit reading files without USM; see write_module for
5870 : details. */
5871 13571 : get_module_locus (&omp_udms);
5872 13571 : if (peek_atom () == ATOM_LPAREN
5873 13571 : && parse_atom ()
5874 13571 : && module_char () == 'U'
5875 8 : && module_char () == 'D'
5876 13579 : && module_char () == 'M')
5877 : has_omp_udms = true;
5878 13571 : set_module_locus (&omp_udms);
5879 13571 : if (has_omp_udms)
5880 8 : skip_list ();
5881 :
5882 13571 : mio_lparen ();
5883 :
5884 : /* Create the fixup nodes for all the symbols. */
5885 :
5886 1261224 : while (peek_atom () != ATOM_RPAREN)
5887 : {
5888 1234082 : char* bind_label;
5889 1234082 : require_atom (ATOM_INTEGER);
5890 1234082 : info = get_integer (atom_int);
5891 :
5892 1234082 : info->type = P_SYMBOL;
5893 1234082 : info->u.rsym.state = UNUSED;
5894 :
5895 1234082 : info->u.rsym.true_name = read_string ();
5896 1234082 : info->u.rsym.module = read_string ();
5897 1234082 : bind_label = read_string ();
5898 1234082 : if (strlen (bind_label))
5899 33261 : info->u.rsym.binding_label = bind_label;
5900 : else
5901 1200821 : XDELETEVEC (bind_label);
5902 :
5903 1234082 : require_atom (ATOM_INTEGER);
5904 1234082 : info->u.rsym.ns = atom_int;
5905 :
5906 1234082 : get_module_locus (&info->u.rsym.where);
5907 :
5908 : /* See if the symbol has already been loaded by a previous module.
5909 : If so, we reference the existing symbol and prevent it from
5910 : being loaded again. This should not happen if the symbol being
5911 : read is an index for an assumed shape dummy array (ns != 1). */
5912 :
5913 1234082 : sym = find_true_name (info->u.rsym.true_name, info->u.rsym.module);
5914 :
5915 1234082 : if (sym == NULL
5916 49709 : || (sym->attr.flavor == FL_VARIABLE && info->u.rsym.ns !=1))
5917 : {
5918 1184398 : skip_list ();
5919 1184398 : continue;
5920 : }
5921 :
5922 49684 : info->u.rsym.state = USED;
5923 49684 : info->u.rsym.sym = sym;
5924 : /* The current symbol has already been loaded, so we can avoid loading
5925 : it again. However, if it is a derived type, some of its components
5926 : can be used in expressions in the module. To avoid the module loading
5927 : failing, we need to associate the module's component pointer indexes
5928 : with the existing symbol's component pointers. */
5929 49684 : if (gfc_fl_struct (sym->attr.flavor))
5930 : {
5931 4782 : gfc_component *c;
5932 :
5933 : /* First seek to the symbol's component list. */
5934 4782 : mio_lparen (); /* symbol opening. */
5935 4782 : skip_list (); /* skip symbol attribute. */
5936 :
5937 4782 : mio_lparen (); /* component list opening. */
5938 16972 : for (c = sym->components; c; c = c->next)
5939 : {
5940 12190 : pointer_info *p;
5941 12190 : const char *comp_name = NULL;
5942 12190 : int n = 0;
5943 :
5944 12190 : mio_lparen (); /* component opening. */
5945 12190 : mio_integer (&n);
5946 12190 : p = get_integer (n);
5947 12190 : if (p->u.pointer == NULL)
5948 12190 : associate_integer_pointer (p, c);
5949 12190 : mio_pool_string (&comp_name);
5950 12190 : if (comp_name != c->name)
5951 : {
5952 0 : gfc_fatal_error ("Mismatch in components of derived type "
5953 : "%qs from %qs at %C: expecting %qs, "
5954 : "but got %qs", sym->name, sym->module,
5955 : c->name, comp_name);
5956 : }
5957 12190 : skip_list (1); /* component end. */
5958 : }
5959 4782 : mio_rparen (); /* component list closing. */
5960 :
5961 4782 : skip_list (1); /* symbol end. */
5962 4782 : }
5963 : else
5964 44902 : skip_list ();
5965 :
5966 : /* Some symbols do not have a namespace (eg. formal arguments),
5967 : so the automatic "unique symtree" mechanism must be suppressed
5968 : by marking them as referenced. */
5969 49684 : q = get_integer (info->u.rsym.ns);
5970 49684 : if (q->u.pointer == NULL)
5971 : {
5972 1698 : info->u.rsym.referenced = 1;
5973 1698 : continue;
5974 : }
5975 : }
5976 :
5977 13571 : mio_rparen ();
5978 :
5979 : /* Parse the symtree lists. This lets us mark which symbols need to
5980 : be loaded. Renaming is also done at this point by replacing the
5981 : symtree name. */
5982 :
5983 13571 : mio_lparen ();
5984 :
5985 542306 : while (peek_atom () != ATOM_RPAREN)
5986 : {
5987 515164 : mio_internal_string (name);
5988 515164 : mio_integer (&ambiguous);
5989 515164 : mio_integer (&symbol);
5990 :
5991 515164 : info = get_integer (symbol);
5992 :
5993 : /* See how many use names there are. If none, go through the start
5994 : of the loop at least once. */
5995 515164 : nuse = number_use_names (name, false);
5996 515164 : info->u.rsym.renamed = nuse ? 1 : 0;
5997 :
5998 3393 : if (nuse == 0)
5999 511771 : nuse = 1;
6000 :
6001 1030363 : for (j = 1; j <= nuse; j++)
6002 : {
6003 : /* Get the jth local name for this symbol. */
6004 515199 : p = find_use_name_n (name, &j, false);
6005 :
6006 515199 : if (p == NULL && strcmp (name, module_name) == 0)
6007 : p = name;
6008 :
6009 : /* Exception: Always import vtabs & vtypes. */
6010 51995 : if (p == NULL && name[0] == '_'
6011 3400 : && (startswith (name, "__vtab_")
6012 2244 : || startswith (name, "__vtype_")))
6013 : p = name;
6014 :
6015 : /* Include pdt_types if their associated pdt_template is in a
6016 : USE, ONLY list. */
6017 49683 : if (p == NULL && name[0] == 'P'
6018 52 : && startswith (name, PDT_PREFIX)
6019 512903 : && module_list)
6020 : {
6021 32 : gfc_use_list *ml = module_list;
6022 32 : for (; ml; ml = ml->next)
6023 16 : if (ml->rename
6024 16 : && !strncmp (&name[PDT_PREFIX_LEN],
6025 : ml->rename->use_name,
6026 16 : strlen (ml->rename->use_name)))
6027 16 : p = name;
6028 : }
6029 :
6030 : /* Skip symtree nodes not in an ONLY clause, unless there
6031 : is an existing symtree loaded from another USE statement. */
6032 515199 : if (p == NULL)
6033 : {
6034 49667 : st = gfc_find_symtree (gfc_current_ns->sym_root, name);
6035 49667 : if (st != NULL
6036 564 : && strcmp (st->n.sym->name, info->u.rsym.true_name) == 0
6037 504 : && st->n.sym->module != NULL
6038 202 : && strcmp (st->n.sym->module, info->u.rsym.module) == 0)
6039 : {
6040 194 : info->u.rsym.symtree = st;
6041 194 : info->u.rsym.sym = st->n.sym;
6042 : }
6043 49667 : continue;
6044 : }
6045 :
6046 : /* If a symbol of the same name and module exists already,
6047 : this symbol, which is not in an ONLY clause, must not be
6048 : added to the namespace(11.3.2). Note that find_symbol
6049 : only returns the first occurrence that it finds. */
6050 458709 : if (!only_flag && !info->u.rsym.renamed
6051 458356 : && strcmp (name, module_name) != 0
6052 913676 : && find_symbol (gfc_current_ns->sym_root, name,
6053 : module_name, 0))
6054 548 : continue;
6055 :
6056 464984 : st = gfc_find_symtree (gfc_current_ns->sym_root, p);
6057 :
6058 464984 : if (st != NULL
6059 43573 : && !(st->n.sym && st->n.sym->attr.used_in_submodule))
6060 : {
6061 : /* Check for ambiguous symbols. */
6062 43536 : if (check_for_ambiguous (st, info))
6063 408 : st->ambiguous = 1;
6064 : else
6065 43128 : info->u.rsym.symtree = st;
6066 : }
6067 : else
6068 : {
6069 421448 : if (st)
6070 : {
6071 : /* This symbol is host associated from a module in a
6072 : submodule. Hide it with a unique symtree. */
6073 37 : gfc_symtree *s = gfc_get_unique_symtree (gfc_current_ns);
6074 37 : s->n.sym = st->n.sym;
6075 37 : st->n.sym = NULL;
6076 : }
6077 : else
6078 : {
6079 : /* Create a symtree node in the current namespace for this
6080 : symbol. */
6081 421411 : st = check_unique_name (p)
6082 421411 : ? gfc_get_unique_symtree (gfc_current_ns)
6083 421411 : : gfc_new_symtree (&gfc_current_ns->sym_root, p);
6084 421411 : st->ambiguous = ambiguous;
6085 : }
6086 :
6087 421448 : sym = info->u.rsym.sym;
6088 :
6089 : /* Create a symbol node if it doesn't already exist. */
6090 421448 : if (sym == NULL)
6091 : {
6092 421290 : info->u.rsym.sym = gfc_new_symbol (info->u.rsym.true_name,
6093 : gfc_current_ns);
6094 421290 : info->u.rsym.sym->name = gfc_dt_lower_string (info->u.rsym.true_name);
6095 421290 : sym = info->u.rsym.sym;
6096 421290 : sym->module = gfc_get_string ("%s", info->u.rsym.module);
6097 :
6098 421290 : if (info->u.rsym.binding_label)
6099 : {
6100 22596 : tree id = get_identifier (info->u.rsym.binding_label);
6101 22596 : sym->binding_label = IDENTIFIER_POINTER (id);
6102 : }
6103 : }
6104 :
6105 421448 : st->n.sym = sym;
6106 421448 : st->n.sym->refs++;
6107 :
6108 421448 : if (strcmp (name, p) != 0)
6109 536 : sym->attr.use_rename = 1;
6110 :
6111 421448 : if (name[0] != '_'
6112 421448 : || (!startswith (name, "__vtab_")
6113 30884 : && !startswith (name, "__vtype_")))
6114 391970 : sym->attr.use_only = only_flag;
6115 :
6116 : /* Store the symtree pointing to this symbol. */
6117 421448 : info->u.rsym.symtree = st;
6118 :
6119 421448 : if (info->u.rsym.state == UNUSED)
6120 421290 : info->u.rsym.state = NEEDED;
6121 421448 : info->u.rsym.referenced = 1;
6122 : }
6123 : }
6124 : }
6125 :
6126 13571 : mio_rparen ();
6127 :
6128 : /* Load intrinsic operator interfaces. */
6129 13571 : set_module_locus (&operator_interfaces);
6130 13571 : mio_lparen ();
6131 :
6132 393559 : for (i = GFC_INTRINSIC_BEGIN; i != GFC_INTRINSIC_END; i++)
6133 : {
6134 379988 : gfc_use_rename *u = NULL, *v = NULL;
6135 379988 : int j = i;
6136 :
6137 379988 : if (i == INTRINSIC_USER)
6138 13571 : continue;
6139 :
6140 366417 : if (only_flag)
6141 : {
6142 48735 : u = find_use_operator ((gfc_intrinsic_op) i);
6143 :
6144 : /* F2018:10.1.5.5.1 requires same interpretation of old and new-style
6145 : relational operators. Special handling for USE, ONLY. */
6146 48735 : switch (i)
6147 : {
6148 : case INTRINSIC_EQ:
6149 : j = INTRINSIC_EQ_OS;
6150 : break;
6151 : case INTRINSIC_EQ_OS:
6152 : j = INTRINSIC_EQ;
6153 : break;
6154 : case INTRINSIC_NE:
6155 : j = INTRINSIC_NE_OS;
6156 : break;
6157 : case INTRINSIC_NE_OS:
6158 : j = INTRINSIC_NE;
6159 : break;
6160 : case INTRINSIC_GT:
6161 : j = INTRINSIC_GT_OS;
6162 : break;
6163 : case INTRINSIC_GT_OS:
6164 : j = INTRINSIC_GT;
6165 : break;
6166 : case INTRINSIC_GE:
6167 : j = INTRINSIC_GE_OS;
6168 : break;
6169 : case INTRINSIC_GE_OS:
6170 : j = INTRINSIC_GE;
6171 : break;
6172 : case INTRINSIC_LT:
6173 : j = INTRINSIC_LT_OS;
6174 : break;
6175 : case INTRINSIC_LT_OS:
6176 : j = INTRINSIC_LT;
6177 : break;
6178 : case INTRINSIC_LE:
6179 : j = INTRINSIC_LE_OS;
6180 : break;
6181 : case INTRINSIC_LE_OS:
6182 : j = INTRINSIC_LE;
6183 : break;
6184 : default:
6185 : break;
6186 : }
6187 :
6188 : if (j != i)
6189 21660 : v = find_use_operator ((gfc_intrinsic_op) j);
6190 :
6191 48735 : if (u == NULL && v == NULL)
6192 : {
6193 48585 : skip_list ();
6194 48585 : continue;
6195 : }
6196 :
6197 150 : if (u)
6198 113 : u->found = 1;
6199 150 : if (v)
6200 89 : v->found = 1;
6201 : }
6202 :
6203 317832 : mio_interface (&gfc_current_ns->op[i]);
6204 317832 : if (!gfc_current_ns->op[i] && !gfc_current_ns->op[j])
6205 : {
6206 316132 : if (u)
6207 15 : u->found = 0;
6208 316132 : if (v)
6209 26 : v->found = 0;
6210 : }
6211 : }
6212 :
6213 13571 : mio_rparen ();
6214 :
6215 : /* Load generic and user operator interfaces. These must follow the
6216 : loading of symtree because otherwise symbols can be marked as
6217 : ambiguous. */
6218 :
6219 13571 : set_module_locus (&user_operators);
6220 :
6221 13571 : load_operator_interfaces ();
6222 13571 : load_generic_interfaces ();
6223 :
6224 13571 : load_commons ();
6225 13571 : load_equiv ();
6226 :
6227 : /* Load OpenMP user defined reductions. */
6228 13571 : set_module_locus (&omp_udrs);
6229 13571 : load_omp_udrs ();
6230 :
6231 : /* Load OpenMP user defined mappers. */
6232 13571 : if (has_omp_udms)
6233 : {
6234 8 : set_module_locus (&omp_udms);
6235 8 : mio_lparen ();
6236 : /* Skip 'UDM' marker, cf. above. */
6237 8 : (void) module_char ();
6238 8 : (void) module_char ();
6239 8 : (void) module_char ();
6240 8 : load_omp_udms ();
6241 8 : mio_rparen ();
6242 : }
6243 :
6244 : /* At this point, we read those symbols that are needed but haven't
6245 : been loaded yet. If one symbol requires another, the other gets
6246 : marked as NEEDED if its previous state was UNUSED. */
6247 :
6248 37839 : while (load_needed (pi_root));
6249 :
6250 : /* Make sure all elements of the rename-list were found in the module. */
6251 :
6252 16529 : for (u = gfc_rename_list; u; u = u->next)
6253 : {
6254 2958 : if (u->found)
6255 2950 : continue;
6256 :
6257 8 : if (u->op == INTRINSIC_NONE)
6258 : {
6259 3 : gfc_error ("Symbol %qs referenced at %L not found in module %qs",
6260 3 : u->use_name, &u->where, module_name);
6261 3 : continue;
6262 : }
6263 :
6264 5 : if (u->op == INTRINSIC_USER)
6265 : {
6266 2 : gfc_error ("User operator %qs referenced at %L not found "
6267 2 : "in module %qs", u->use_name, &u->where, module_name);
6268 2 : continue;
6269 : }
6270 :
6271 3 : gfc_error ("Intrinsic operator %qs referenced at %L not found "
6272 : "in module %qs", gfc_op2string (u->op), &u->where,
6273 : module_name);
6274 : }
6275 :
6276 : /* Check "omp declare mappers" for duplicates from different modules. */
6277 13571 : check_omp_declare_mappers (gfc_current_ns->omp_udm_root);
6278 :
6279 : /* Clean up symbol nodes that were never loaded, create references
6280 : to hidden symbols. */
6281 :
6282 13571 : read_cleanup (pi_root);
6283 13571 : }
6284 :
6285 :
6286 : /* Given an access type that is specific to an entity and the default
6287 : access, return nonzero if the entity is publicly accessible. If the
6288 : element is declared as PUBLIC, then it is public; if declared
6289 : PRIVATE, then private, and otherwise it is public unless the default
6290 : access in this context has been declared PRIVATE. */
6291 :
6292 : static bool dump_smod = false;
6293 :
6294 : static bool
6295 1046896 : check_access (gfc_access specific_access, gfc_access default_access)
6296 : {
6297 1046896 : if (dump_smod)
6298 : return true;
6299 :
6300 1022162 : if (specific_access == ACCESS_PUBLIC)
6301 : return true;
6302 991934 : if (specific_access == ACCESS_PRIVATE)
6303 : return false;
6304 :
6305 989589 : if (flag_module_private)
6306 91 : return default_access == ACCESS_PUBLIC;
6307 : else
6308 989498 : return default_access != ACCESS_PRIVATE;
6309 : }
6310 :
6311 :
6312 : bool
6313 869793 : gfc_check_symbol_access (gfc_symbol *sym)
6314 : {
6315 869793 : if (sym->attr.vtab || sym->attr.vtype)
6316 : return true;
6317 : else
6318 779610 : return check_access (sym->attr.access, sym->ns->default_access);
6319 : }
6320 :
6321 :
6322 : /* A structure to remember which commons we've already written. */
6323 :
6324 : struct written_common
6325 : {
6326 : BBT_HEADER(written_common);
6327 : const char *name, *label;
6328 : };
6329 :
6330 : static struct written_common *written_commons = NULL;
6331 :
6332 : /* Comparison function used for balancing the binary tree. */
6333 :
6334 : static int
6335 127 : compare_written_commons (void *a1, void *b1)
6336 : {
6337 127 : const char *aname = ((struct written_common *) a1)->name;
6338 127 : const char *alabel = ((struct written_common *) a1)->label;
6339 127 : const char *bname = ((struct written_common *) b1)->name;
6340 127 : const char *blabel = ((struct written_common *) b1)->label;
6341 127 : int c = strcmp (aname, bname);
6342 :
6343 127 : return (c != 0 ? c : strcmp (alabel, blabel));
6344 : }
6345 :
6346 : /* Free a list of written commons. */
6347 :
6348 : static void
6349 9961 : free_written_common (struct written_common *w)
6350 : {
6351 9961 : if (!w)
6352 : return;
6353 :
6354 211 : if (w->left)
6355 27 : free_written_common (w->left);
6356 211 : if (w->right)
6357 42 : free_written_common (w->right);
6358 :
6359 211 : free (w);
6360 : }
6361 :
6362 : /* Write a common block to the module -- recursive helper function. */
6363 :
6364 : static void
6365 20724 : write_common_0 (gfc_symtree *st, bool this_module)
6366 : {
6367 20724 : gfc_common_head *p;
6368 20724 : const char * name;
6369 20724 : int flags;
6370 20724 : const char *label;
6371 20724 : struct written_common *w;
6372 20724 : bool write_me = true;
6373 :
6374 20724 : if (st == NULL)
6375 20254 : return;
6376 :
6377 470 : write_common_0 (st->left, this_module);
6378 :
6379 : /* We will write out the binding label, or "" if no label given. */
6380 470 : name = st->n.common->name;
6381 470 : p = st->n.common;
6382 470 : label = (p->is_bind_c && p->binding_label) ? p->binding_label : "";
6383 :
6384 : /* Check if we've already output this common. */
6385 470 : w = written_commons;
6386 988 : while (w)
6387 : {
6388 518 : int c = strcmp (name, w->name);
6389 518 : c = (c != 0 ? c : strcmp (label, w->label));
6390 206 : if (c == 0)
6391 : write_me = false;
6392 :
6393 518 : w = (c < 0) ? w->left : w->right;
6394 : }
6395 :
6396 470 : if (this_module && p->use_assoc)
6397 : write_me = false;
6398 :
6399 417 : if (write_me)
6400 : {
6401 : /* Write the common to the module. */
6402 211 : mio_lparen ();
6403 211 : mio_pool_string (&name);
6404 :
6405 211 : mio_symbol_ref (&p->head);
6406 211 : flags = p->saved ? 1 : 0;
6407 211 : if (p->threadprivate)
6408 0 : flags |= 2;
6409 211 : flags |= p->omp_device_type << 2;
6410 211 : flags |= p->omp_groupprivate << 4;
6411 211 : mio_integer (&flags);
6412 :
6413 : /* Write out whether the common block is bind(c) or not. */
6414 211 : mio_integer (&(p->is_bind_c));
6415 :
6416 211 : mio_pool_string (&label);
6417 211 : mio_rparen ();
6418 :
6419 : /* Record that we have written this common. */
6420 211 : w = XCNEW (struct written_common);
6421 211 : w->name = p->name;
6422 211 : w->label = label;
6423 211 : gfc_insert_bbt (&written_commons, w, compare_written_commons);
6424 : }
6425 :
6426 470 : write_common_0 (st->right, this_module);
6427 : }
6428 :
6429 :
6430 : /* Write a common, by initializing the list of written commons, calling
6431 : the recursive function write_common_0() and cleaning up afterwards. */
6432 :
6433 : static void
6434 9892 : write_common (gfc_symtree *st)
6435 : {
6436 9892 : written_commons = NULL;
6437 9892 : write_common_0 (st, true);
6438 9892 : write_common_0 (st, false);
6439 9892 : free_written_common (written_commons);
6440 9892 : written_commons = NULL;
6441 9892 : }
6442 :
6443 :
6444 : /* Write the blank common block to the module. */
6445 :
6446 : static void
6447 9892 : write_blank_common (void)
6448 : {
6449 9892 : const char * name = BLANK_COMMON_NAME;
6450 9892 : int saved;
6451 : /* TODO: Blank commons are not bind(c). The F2003 standard probably says
6452 : this, but it hasn't been checked. Just making it so for now. */
6453 9892 : int is_bind_c = 0;
6454 :
6455 9892 : if (gfc_current_ns->blank_common.head == NULL)
6456 9885 : return;
6457 :
6458 7 : mio_lparen ();
6459 :
6460 7 : mio_pool_string (&name);
6461 :
6462 7 : mio_symbol_ref (&gfc_current_ns->blank_common.head);
6463 7 : saved = gfc_current_ns->blank_common.saved;
6464 7 : mio_integer (&saved);
6465 :
6466 : /* Write out whether the common block is bind(c) or not. */
6467 7 : mio_integer (&is_bind_c);
6468 :
6469 : /* Write out an empty binding label. */
6470 7 : write_atom (ATOM_STRING, "");
6471 :
6472 7 : mio_rparen ();
6473 : }
6474 :
6475 :
6476 : /* Write equivalences to the module. */
6477 :
6478 : static void
6479 9892 : write_equiv (void)
6480 : {
6481 9892 : gfc_equiv *eq, *e;
6482 9892 : int num;
6483 :
6484 9892 : num = 0;
6485 9974 : for (eq = gfc_current_ns->equiv; eq; eq = eq->next)
6486 : {
6487 82 : mio_lparen ();
6488 :
6489 328 : for (e = eq; e; e = e->eq)
6490 : {
6491 164 : if (e->module == NULL)
6492 142 : e->module = gfc_get_string ("%s.eq.%d", module_name, num);
6493 164 : mio_allocated_string (e->module);
6494 164 : mio_expr (&e->expr);
6495 : }
6496 :
6497 82 : num++;
6498 82 : mio_rparen ();
6499 : }
6500 9892 : }
6501 :
6502 :
6503 : /* Write a symbol to the module. */
6504 :
6505 : static void
6506 242009 : write_symbol (int n, gfc_symbol *sym)
6507 : {
6508 242009 : const char *label;
6509 :
6510 242009 : if (sym->attr.flavor == FL_UNKNOWN || sym->attr.flavor == FL_LABEL)
6511 0 : gfc_internal_error ("write_symbol(): bad module symbol %qs", sym->name);
6512 :
6513 242009 : mio_integer (&n);
6514 :
6515 242009 : if (gfc_fl_struct (sym->attr.flavor))
6516 : {
6517 27141 : const char *name;
6518 27141 : name = gfc_dt_upper_string (sym->name);
6519 27141 : mio_pool_string (&name);
6520 27141 : }
6521 : else
6522 214868 : mio_pool_string (&sym->name);
6523 :
6524 242009 : mio_pool_string (&sym->module);
6525 242009 : if ((sym->attr.is_bind_c || sym->attr.is_iso_c) && sym->binding_label)
6526 : {
6527 2729 : label = sym->binding_label;
6528 2729 : mio_pool_string (&label);
6529 : }
6530 : else
6531 239280 : write_atom (ATOM_STRING, "");
6532 :
6533 242009 : mio_pointer_ref (&sym->ns);
6534 :
6535 242009 : mio_symbol (sym);
6536 242009 : write_char ('\n');
6537 242009 : }
6538 :
6539 :
6540 : /* Recursive traversal function to write the initial set of symbols to
6541 : the module. We check to see if the symbol should be written
6542 : according to the access specification. */
6543 :
6544 : static void
6545 164798 : write_symbol0 (gfc_symtree *st)
6546 : {
6547 319704 : gfc_symbol *sym;
6548 319704 : pointer_info *p;
6549 319704 : bool dont_write = false;
6550 :
6551 319704 : if (st == NULL)
6552 164798 : return;
6553 :
6554 154906 : write_symbol0 (st->left);
6555 :
6556 154906 : sym = st->n.sym;
6557 154906 : if (sym->module == NULL)
6558 71432 : sym->module = module_name;
6559 :
6560 154906 : if (sym->attr.flavor == FL_PROCEDURE && sym->attr.generic
6561 11866 : && !sym->attr.subroutine && !sym->attr.function)
6562 154906 : dont_write = true;
6563 :
6564 154906 : if (!gfc_check_symbol_access (sym))
6565 : dont_write = true;
6566 :
6567 136124 : if (!dont_write)
6568 : {
6569 134646 : p = get_pointer (sym);
6570 134646 : if (p->type == P_UNKNOWN)
6571 102650 : p->type = P_SYMBOL;
6572 :
6573 134646 : if (p->u.wsym.state != WRITTEN)
6574 : {
6575 131775 : write_symbol (p->integer, sym);
6576 131775 : p->u.wsym.state = WRITTEN;
6577 : }
6578 : }
6579 :
6580 154906 : write_symbol0 (st->right);
6581 : }
6582 :
6583 :
6584 : static void
6585 100 : write_omp_udr (gfc_omp_udr *udr)
6586 : {
6587 100 : switch (udr->rop)
6588 : {
6589 65 : case OMP_REDUCTION_USER:
6590 : /* Non-operators can't be used outside of the module. */
6591 65 : if (udr->name[0] != '.')
6592 : return;
6593 : else
6594 : {
6595 47 : gfc_symtree *st;
6596 47 : size_t len = strlen (udr->name + 1);
6597 47 : char *name = XALLOCAVEC (char, len);
6598 47 : memcpy (name, udr->name, len - 1);
6599 47 : name[len - 1] = '\0';
6600 47 : st = gfc_find_symtree (gfc_current_ns->uop_root, name);
6601 : /* If corresponding user operator is private, don't write
6602 : the UDR. */
6603 47 : if (st != NULL)
6604 : {
6605 0 : gfc_user_op *uop = st->n.uop;
6606 0 : if (!check_access (uop->access, uop->ns->default_access))
6607 : return;
6608 : }
6609 : }
6610 : break;
6611 35 : case OMP_REDUCTION_PLUS:
6612 35 : case OMP_REDUCTION_MINUS:
6613 35 : case OMP_REDUCTION_TIMES:
6614 35 : case OMP_REDUCTION_AND:
6615 35 : case OMP_REDUCTION_OR:
6616 35 : case OMP_REDUCTION_EQV:
6617 35 : case OMP_REDUCTION_NEQV:
6618 : /* If corresponding operator is private, don't write the UDR. */
6619 35 : if (!check_access (gfc_current_ns->operator_access[udr->rop],
6620 : gfc_current_ns->default_access))
6621 : return;
6622 : break;
6623 : default:
6624 : break;
6625 : }
6626 81 : if (udr->ts.type == BT_DERIVED || udr->ts.type == BT_CLASS)
6627 : {
6628 : /* If derived type is private, don't write the UDR. */
6629 45 : if (!gfc_check_symbol_access (udr->ts.u.derived))
6630 : return;
6631 : }
6632 :
6633 80 : mio_lparen ();
6634 80 : mio_pool_string (&udr->name);
6635 80 : mio_typespec (&udr->ts);
6636 80 : mio_omp_udr_expr (udr, &udr->omp_out, &udr->omp_in, udr->combiner_ns, false);
6637 80 : if (udr->initializer_ns)
6638 64 : mio_omp_udr_expr (udr, &udr->omp_priv, &udr->omp_orig,
6639 : udr->initializer_ns, true);
6640 80 : mio_rparen ();
6641 : }
6642 :
6643 :
6644 : /* Write OpenMP's declare reduction (used defined reductions). */
6645 :
6646 : static void
6647 9992 : write_omp_udrs (gfc_symtree *st)
6648 : {
6649 10092 : if (st == NULL)
6650 9992 : return;
6651 :
6652 100 : write_omp_udrs (st->left);
6653 100 : gfc_omp_udr *udr;
6654 200 : for (udr = st->n.omp_udr; udr; udr = udr->next)
6655 100 : write_omp_udr (udr);
6656 100 : write_omp_udrs (st->right);
6657 : }
6658 :
6659 :
6660 : /* Write OpenMP's declare mapper (used defined mapper). */
6661 :
6662 : static void
6663 9 : write_omp_udm (gfc_omp_udm *udm)
6664 : {
6665 9 : mio_lparen ();
6666 : /* We need this pointer ref to identify this mapper so that other mappers
6667 : can refer to it. */
6668 9 : mio_pointer_ref (&udm);
6669 9 : mio_pool_string (&udm->mapper_id);
6670 9 : mio_typespec (&udm->ts);
6671 :
6672 9 : if (udm->var_sym->module == NULL)
6673 9 : udm->var_sym->module = module_name;
6674 :
6675 9 : mio_symbol_ref (&udm->var_sym);
6676 9 : mio_lparen ();
6677 9 : gfc_omp_namelist *n;
6678 26 : for (n = udm->clauses->lists[OMP_LIST_MAP]; n; n = n->next)
6679 : {
6680 17 : mio_lparen ();
6681 :
6682 17 : mio_name (n->u.map.op, omp_map_clause_ops);
6683 17 : mio_symbol_ref (&n->sym);
6684 17 : mio_expr (&n->expr);
6685 :
6686 17 : mio_lparen ();
6687 :
6688 17 : if (n->u3.udm)
6689 : {
6690 7 : mio_pool_string (&n->u3.udm->requested_mapper_id);
6691 7 : mio_pointer_ref (&n->u3.udm->resolved_udm);
6692 : }
6693 :
6694 17 : mio_rparen ();
6695 :
6696 17 : mio_rparen ();
6697 : }
6698 9 : mio_rparen ();
6699 9 : mio_rparen ();
6700 9 : }
6701 :
6702 :
6703 : static void
6704 17 : write_omp_udms (gfc_symtree *st)
6705 : {
6706 26 : if (st == NULL)
6707 17 : return;
6708 :
6709 9 : write_omp_udms (st->left);
6710 9 : gfc_omp_udm *udm;
6711 18 : for (udm = st->n.omp_udm; udm; udm = udm->next)
6712 9 : write_omp_udm (udm);
6713 9 : write_omp_udms (st->right);
6714 : }
6715 :
6716 :
6717 : /* Type for the temporary tree used when writing secondary symbols. */
6718 :
6719 : struct sorted_pointer_info
6720 : {
6721 : BBT_HEADER (sorted_pointer_info);
6722 :
6723 : pointer_info *p;
6724 : };
6725 :
6726 : #define gfc_get_sorted_pointer_info() XCNEW (sorted_pointer_info)
6727 :
6728 : /* Recursively traverse the temporary tree, free its contents. */
6729 :
6730 : static void
6731 235751 : free_sorted_pointer_info_tree (sorted_pointer_info *p)
6732 : {
6733 235751 : if (!p)
6734 : return;
6735 :
6736 110234 : free_sorted_pointer_info_tree (p->left);
6737 110234 : free_sorted_pointer_info_tree (p->right);
6738 :
6739 110234 : free (p);
6740 : }
6741 :
6742 : /* Comparison function for the temporary tree. */
6743 :
6744 : static int
6745 373155 : compare_sorted_pointer_info (void *_spi1, void *_spi2)
6746 : {
6747 373155 : sorted_pointer_info *spi1, *spi2;
6748 373155 : spi1 = (sorted_pointer_info *)_spi1;
6749 373155 : spi2 = (sorted_pointer_info *)_spi2;
6750 :
6751 373155 : if (spi1->p->integer < spi2->p->integer)
6752 : return -1;
6753 211595 : if (spi1->p->integer > spi2->p->integer)
6754 211595 : return 1;
6755 : return 0;
6756 : }
6757 :
6758 :
6759 : /* Finds the symbols that need to be written and collects them in the
6760 : sorted_pi tree so that they can be traversed in an order
6761 : independent of memory addresses. */
6762 :
6763 : static void
6764 1232155 : find_symbols_to_write(sorted_pointer_info **tree, pointer_info *p)
6765 : {
6766 2439135 : if (!p)
6767 1232155 : return;
6768 :
6769 1206980 : if (p->type == P_SYMBOL && p->u.wsym.state == NEEDS_WRITE)
6770 : {
6771 110234 : sorted_pointer_info *sp = gfc_get_sorted_pointer_info();
6772 110234 : sp->p = p;
6773 :
6774 110234 : gfc_insert_bbt (tree, sp, compare_sorted_pointer_info);
6775 : }
6776 :
6777 1206980 : find_symbols_to_write (tree, p->left);
6778 1206980 : find_symbols_to_write (tree, p->right);
6779 : }
6780 :
6781 :
6782 : /* Recursive function that traverses the tree of symbols that need to be
6783 : written and writes them in order. */
6784 :
6785 : static void
6786 125517 : write_symbol1_recursion (sorted_pointer_info *sp)
6787 : {
6788 235751 : if (!sp)
6789 125517 : return;
6790 :
6791 110234 : write_symbol1_recursion (sp->left);
6792 :
6793 110234 : pointer_info *p1 = sp->p;
6794 110234 : gcc_assert (p1->type == P_SYMBOL && p1->u.wsym.state == NEEDS_WRITE);
6795 :
6796 110234 : p1->u.wsym.state = WRITTEN;
6797 110234 : write_symbol (p1->integer, p1->u.wsym.sym);
6798 110234 : p1->u.wsym.sym->attr.public_used = 1;
6799 :
6800 110234 : write_symbol1_recursion (sp->right);
6801 : }
6802 :
6803 :
6804 : /* Write the secondary set of symbols to the module file. These are
6805 : symbols that were not public yet are needed by the public symbols
6806 : or another dependent symbol. The act of writing a symbol can add
6807 : symbols to the pointer_info tree, so we return nonzero if a symbol
6808 : was written and pass that information upwards. The caller will
6809 : then call this function again until nothing was written. It uses
6810 : the utility functions and a temporary tree to ensure a reproducible
6811 : ordering of the symbol output and thus the module file. */
6812 :
6813 : static int
6814 25175 : write_symbol1 (pointer_info *p)
6815 : {
6816 25175 : if (!p)
6817 : return 0;
6818 :
6819 : /* Put symbols that need to be written into a tree sorted on the
6820 : integer field. */
6821 :
6822 25175 : sorted_pointer_info *spi_root = NULL;
6823 25175 : find_symbols_to_write (&spi_root, p);
6824 :
6825 : /* No symbols to write, return. */
6826 25175 : if (!spi_root)
6827 : return 0;
6828 :
6829 : /* Otherwise, write and free the tree again. */
6830 15283 : write_symbol1_recursion (spi_root);
6831 15283 : free_sorted_pointer_info_tree (spi_root);
6832 :
6833 15283 : return 1;
6834 : }
6835 :
6836 :
6837 : /* Write operator interfaces associated with a symbol. */
6838 :
6839 : static void
6840 167 : write_operator (gfc_user_op *uop)
6841 : {
6842 167 : static char nullstring[] = "";
6843 167 : const char *p = nullstring;
6844 :
6845 167 : if (uop->op == NULL || !check_access (uop->access, uop->ns->default_access))
6846 1 : return;
6847 :
6848 166 : mio_symbol_interface (&uop->name, &p, &uop->op);
6849 : }
6850 :
6851 :
6852 : /* Write generic interfaces from the namespace sym_root. */
6853 :
6854 : static void
6855 164798 : write_generic (gfc_symtree *st)
6856 : {
6857 319704 : gfc_symbol *sym;
6858 :
6859 319704 : if (st == NULL)
6860 164798 : return;
6861 :
6862 154906 : write_generic (st->left);
6863 :
6864 154906 : sym = st->n.sym;
6865 154906 : if (sym && !check_unique_name (st->name)
6866 302444 : && sym->generic && gfc_check_symbol_access (sym))
6867 : {
6868 10832 : if (!sym->module)
6869 7116 : sym->module = module_name;
6870 :
6871 10832 : mio_symbol_interface (&st->name, &sym->module, &sym->generic);
6872 : }
6873 :
6874 154906 : write_generic (st->right);
6875 : }
6876 :
6877 :
6878 : static void
6879 154907 : write_symtree (gfc_symtree *st)
6880 : {
6881 154907 : gfc_symbol *sym;
6882 154907 : pointer_info *p;
6883 :
6884 154907 : sym = st->n.sym;
6885 :
6886 : /* A symbol in an interface body must not be visible in the
6887 : module file. */
6888 154907 : if (sym->ns != gfc_current_ns
6889 450 : && sym->ns->proc_name
6890 450 : && sym->ns->proc_name->attr.if_source == IFSRC_IFBODY)
6891 : return;
6892 :
6893 154907 : if ((!gfc_check_symbol_access (sym)
6894 18782 : && (!sym->attr.public_used || submodule_name == NULL))
6895 154907 : || (sym->attr.flavor == FL_PROCEDURE && sym->attr.generic
6896 10960 : && !sym->attr.subroutine && !sym->attr.function))
6897 : return;
6898 :
6899 134646 : if (check_unique_name (st->name))
6900 : return;
6901 :
6902 : /* From F2003 onwards, intrinsic procedures are no longer subject to
6903 : the restriction, "that an elemental intrinsic function here be of
6904 : type integer or character and each argument must be an initialization
6905 : expr of type integer or character" is lifted so that intrinsic
6906 : procedures can be over-ridden. This requires that the intrinsic
6907 : symbol not appear in the module file, thereby preventing ambiguity
6908 : when USEd. */
6909 128251 : if (strcmp (sym->module, "(intrinsic)") == 0
6910 2128 : && (gfc_option.allow_std & GFC_STD_F2003))
6911 : return;
6912 :
6913 126124 : p = find_pointer (sym);
6914 126124 : if (p == NULL)
6915 0 : gfc_internal_error ("write_symtree(): Symbol not written");
6916 :
6917 126124 : mio_pool_string (&st->name);
6918 126124 : mio_integer (&st->ambiguous);
6919 126124 : mio_hwi (&p->integer);
6920 : }
6921 :
6922 :
6923 : static void
6924 9892 : write_module (void)
6925 : {
6926 9892 : int i;
6927 :
6928 : /* Initialize the column counter. */
6929 9892 : module_column = 1;
6930 :
6931 : /* Write the operator interfaces. */
6932 9892 : mio_lparen ();
6933 :
6934 296760 : for (i = GFC_INTRINSIC_BEGIN; i != GFC_INTRINSIC_END; i++)
6935 : {
6936 276976 : if (i == INTRINSIC_USER)
6937 9892 : continue;
6938 :
6939 534168 : mio_interface (check_access (gfc_current_ns->operator_access[i],
6940 : gfc_current_ns->default_access)
6941 : ? &gfc_current_ns->op[i] : NULL);
6942 : }
6943 :
6944 9892 : mio_rparen ();
6945 9892 : write_char ('\n');
6946 9892 : write_char ('\n');
6947 :
6948 9892 : mio_lparen ();
6949 9892 : gfc_traverse_user_op (gfc_current_ns, write_operator);
6950 9892 : mio_rparen ();
6951 9892 : write_char ('\n');
6952 9892 : write_char ('\n');
6953 :
6954 9892 : mio_lparen ();
6955 9892 : write_generic (gfc_current_ns->sym_root);
6956 9892 : mio_rparen ();
6957 9892 : write_char ('\n');
6958 9892 : write_char ('\n');
6959 :
6960 9892 : mio_lparen ();
6961 9892 : write_blank_common ();
6962 9892 : write_common (gfc_current_ns->common_root);
6963 9892 : mio_rparen ();
6964 9892 : write_char ('\n');
6965 9892 : write_char ('\n');
6966 :
6967 9892 : mio_lparen ();
6968 9892 : write_equiv ();
6969 9892 : mio_rparen ();
6970 9892 : write_char ('\n');
6971 9892 : write_char ('\n');
6972 :
6973 9892 : mio_lparen ();
6974 9892 : write_omp_udrs (gfc_current_ns->omp_udr_root);
6975 9892 : mio_rparen ();
6976 9892 : write_char ('\n');
6977 9892 : write_char ('\n');
6978 :
6979 : /* Condition can be removed if version is bumped. Note that
6980 : write_symbol0 starts with an integer. Keep in sync with read_module;
6981 : The 'UDM' tag can be only removed when changing COMPAT_MOD_VERSIONS. */
6982 9892 : STATIC_ASSERT (MOD_VERSION_NUMERIC == 16);
6983 9892 : if (gfc_current_ns->omp_udm_root)
6984 : {
6985 8 : mio_lparen ();
6986 8 : write_atom (ATOM_NAME, "UDM"); /* Marker. */
6987 8 : write_omp_udms (gfc_current_ns->omp_udm_root);
6988 8 : mio_rparen ();
6989 8 : write_char ('\n');
6990 8 : write_char ('\n');
6991 : }
6992 :
6993 : /* Write symbol information. First we traverse all symbols in the
6994 : primary namespace, writing those that need to be written.
6995 : Sometimes writing one symbol will cause another to need to be
6996 : written. A list of these symbols ends up on the write stack, and
6997 : we end by popping the bottom of the stack and writing the symbol
6998 : until the stack is empty. */
6999 :
7000 9892 : mio_lparen ();
7001 :
7002 9892 : write_symbol0 (gfc_current_ns->sym_root);
7003 35067 : while (write_symbol1 (pi_root))
7004 : /* Nothing. */;
7005 :
7006 9892 : mio_rparen ();
7007 :
7008 9892 : write_char ('\n');
7009 9892 : write_char ('\n');
7010 :
7011 9892 : mio_lparen ();
7012 9892 : gfc_traverse_symtree (gfc_current_ns->sym_root, write_symtree);
7013 9892 : mio_rparen ();
7014 9892 : }
7015 :
7016 :
7017 : /* Read a CRC32 sum from the gzip trailer of a module file. Returns
7018 : true on success, false on failure. */
7019 :
7020 : static bool
7021 19784 : read_crc32_from_module_file (const char* filename, uLong* crc)
7022 : {
7023 19784 : FILE *file;
7024 19784 : char buf[4];
7025 19784 : unsigned int val;
7026 :
7027 : /* Open the file in binary mode. */
7028 19784 : if ((file = fopen (filename, "rb")) == NULL)
7029 : return false;
7030 :
7031 : /* The gzip crc32 value is found in the [END-8, END-4] bytes of the
7032 : file. See RFC 1952. */
7033 10106 : if (fseek (file, -8, SEEK_END) != 0)
7034 : {
7035 0 : fclose (file);
7036 0 : return false;
7037 : }
7038 :
7039 : /* Read the CRC32. */
7040 10106 : if (fread (buf, 1, 4, file) != 4)
7041 : {
7042 0 : fclose (file);
7043 0 : return false;
7044 : }
7045 :
7046 : /* Close the file. */
7047 10106 : fclose (file);
7048 :
7049 10106 : val = (buf[0] & 0xFF) + ((buf[1] & 0xFF) << 8) + ((buf[2] & 0xFF) << 16)
7050 10106 : + ((buf[3] & 0xFF) << 24);
7051 10106 : *crc = val;
7052 :
7053 : /* For debugging, the CRC value printed in hexadecimal should match
7054 : the CRC printed by "zcat -l -v filename".
7055 : printf("CRC of file %s is %x\n", filename, val); */
7056 :
7057 10106 : return true;
7058 : }
7059 :
7060 :
7061 : /* Given module, dump it to disk. If there was an error while
7062 : processing the module, dump_flag will be set to zero and we delete
7063 : the module file, even if it was already there. */
7064 :
7065 : static void
7066 10378 : dump_module (const char *name, int dump_flag)
7067 : {
7068 10378 : int n;
7069 10378 : char *filename, *filename_tmp;
7070 10378 : uLong crc, crc_old;
7071 :
7072 10378 : module_name = gfc_get_string ("%s", name);
7073 :
7074 10378 : if (dump_smod)
7075 : {
7076 560 : name = submodule_name;
7077 560 : n = strlen (name) + strlen (SUBMODULE_EXTENSION) + 1;
7078 : }
7079 : else
7080 9818 : n = strlen (name) + strlen (MODULE_EXTENSION) + 1;
7081 :
7082 10378 : if (gfc_option.module_dir != NULL)
7083 : {
7084 0 : n += strlen (gfc_option.module_dir);
7085 0 : filename = (char *) alloca (n);
7086 0 : strcpy (filename, gfc_option.module_dir);
7087 0 : strcat (filename, name);
7088 : }
7089 : else
7090 : {
7091 10378 : filename = (char *) alloca (n);
7092 10378 : strcpy (filename, name);
7093 : }
7094 :
7095 10378 : if (dump_smod)
7096 560 : strcat (filename, SUBMODULE_EXTENSION);
7097 : else
7098 9818 : strcat (filename, MODULE_EXTENSION);
7099 :
7100 : /* Name of the temporary file used to write the module. */
7101 10378 : filename_tmp = (char *) alloca (n + 1);
7102 10378 : strcpy (filename_tmp, filename);
7103 10378 : strcat (filename_tmp, "0");
7104 :
7105 : /* There was an error while processing the module. We delete the
7106 : module file, even if it was already there. */
7107 10378 : if (!dump_flag)
7108 : {
7109 486 : remove (filename);
7110 486 : return;
7111 : }
7112 :
7113 9892 : if (gfc_cpp_makedep ())
7114 0 : gfc_cpp_add_target (filename);
7115 :
7116 : /* Write the module to the temporary file. */
7117 9892 : module_fp = gzopen (filename_tmp, "w");
7118 9892 : if (module_fp == NULL)
7119 0 : gfc_fatal_error ("Cannot open module file %qs for writing at %C: %s",
7120 0 : filename_tmp, xstrerror (errno));
7121 :
7122 : /* Use lbasename to ensure module files are reproducible regardless
7123 : of the build path (see the reproducible builds project). */
7124 9892 : gzprintf (module_fp, "GFORTRAN module version '%s' created from %s\n",
7125 : MOD_VERSION, lbasename (gfc_source_file));
7126 :
7127 : /* Write the module itself. */
7128 9892 : iomode = IO_OUTPUT;
7129 :
7130 9892 : init_pi_tree ();
7131 :
7132 9892 : write_module ();
7133 :
7134 9892 : free_pi_tree (pi_root);
7135 9892 : pi_root = NULL;
7136 :
7137 9892 : write_char ('\n');
7138 :
7139 9892 : if (gzclose (module_fp))
7140 0 : gfc_fatal_error ("Error writing module file %qs for writing: %s",
7141 0 : filename_tmp, xstrerror (errno));
7142 :
7143 : /* Read the CRC32 from the gzip trailers of the module files and
7144 : compare. */
7145 9892 : if (!read_crc32_from_module_file (filename_tmp, &crc)
7146 9892 : || !read_crc32_from_module_file (filename, &crc_old)
7147 10106 : || crc_old != crc)
7148 : {
7149 : /* Module file have changed, replace the old one. */
7150 9680 : if (remove (filename) && errno != ENOENT)
7151 0 : gfc_fatal_error ("Cannot delete module file %qs: %s", filename,
7152 : xstrerror (errno));
7153 9680 : if (rename (filename_tmp, filename))
7154 0 : gfc_fatal_error ("Cannot rename module file %qs to %qs: %s",
7155 0 : filename_tmp, filename, xstrerror (errno));
7156 : }
7157 : else
7158 : {
7159 212 : if (remove (filename_tmp))
7160 0 : gfc_fatal_error ("Cannot delete temporary module file %qs: %s",
7161 0 : filename_tmp, xstrerror (errno));
7162 : }
7163 : }
7164 :
7165 :
7166 : /* Suppress the output of a .smod file by module, if no module
7167 : procedures have been seen. */
7168 : static bool no_module_procedures;
7169 :
7170 : static void
7171 154673 : check_for_module_procedures (gfc_symbol *sym)
7172 : {
7173 154673 : if (sym && sym->attr.module_procedure)
7174 1159 : no_module_procedures = false;
7175 154673 : }
7176 :
7177 :
7178 : void
7179 10080 : gfc_dump_module (const char *name, int dump_flag)
7180 : {
7181 10080 : if (gfc_state_stack->state == COMP_SUBMODULE)
7182 262 : dump_smod = true;
7183 : else
7184 9818 : dump_smod =false;
7185 :
7186 10080 : no_module_procedures = true;
7187 10080 : gfc_traverse_ns (gfc_current_ns, check_for_module_procedures);
7188 :
7189 10080 : dump_module (name, dump_flag);
7190 :
7191 10080 : if (no_module_procedures || dump_smod)
7192 : return;
7193 :
7194 : /* Write a submodule file from a module. The 'dump_smod' flag switches
7195 : off the check for PRIVATE entities. */
7196 298 : dump_smod = true;
7197 298 : submodule_name = module_name;
7198 298 : dump_module (name, dump_flag);
7199 298 : dump_smod = false;
7200 : }
7201 :
7202 : static void
7203 26907 : create_intrinsic_function (const char *name, int id,
7204 : const char *modname, intmod_id module,
7205 : bool subroutine, gfc_symbol *result_type)
7206 : {
7207 26907 : gfc_intrinsic_sym *isym;
7208 26907 : gfc_symtree *tmp_symtree;
7209 26907 : gfc_symbol *sym;
7210 :
7211 26907 : tmp_symtree = gfc_find_symtree (gfc_current_ns->sym_root, name);
7212 26907 : if (tmp_symtree)
7213 : {
7214 48 : if (tmp_symtree->n.sym && tmp_symtree->n.sym->module
7215 48 : && strcmp (modname, tmp_symtree->n.sym->module) == 0)
7216 48 : return;
7217 0 : gfc_error ("Symbol %qs at %C already declared", name);
7218 0 : return;
7219 : }
7220 :
7221 26859 : gfc_get_sym_tree (name, gfc_current_ns, &tmp_symtree, false);
7222 26859 : sym = tmp_symtree->n.sym;
7223 :
7224 26859 : if (subroutine)
7225 : {
7226 9759 : gfc_isym_id isym_id = gfc_isym_id_by_intmod (module, id);
7227 9759 : isym = gfc_intrinsic_subroutine_by_id (isym_id);
7228 9759 : sym->attr.subroutine = 1;
7229 : }
7230 : else
7231 : {
7232 17100 : gfc_isym_id isym_id = gfc_isym_id_by_intmod (module, id);
7233 17100 : isym = gfc_intrinsic_function_by_id (isym_id);
7234 :
7235 17100 : sym->attr.function = 1;
7236 17100 : if (result_type)
7237 : {
7238 6662 : sym->ts.type = BT_DERIVED;
7239 6662 : sym->ts.u.derived = result_type;
7240 6662 : sym->ts.is_c_interop = 1;
7241 6662 : isym->ts.f90_type = BT_VOID;
7242 6662 : isym->ts.type = BT_DERIVED;
7243 6662 : isym->ts.f90_type = BT_VOID;
7244 6662 : isym->ts.u.derived = result_type;
7245 6662 : isym->ts.is_c_interop = 1;
7246 : }
7247 : }
7248 26859 : gcc_assert (isym);
7249 :
7250 26859 : sym->attr.flavor = FL_PROCEDURE;
7251 26859 : sym->attr.intrinsic = 1;
7252 :
7253 26859 : sym->module = gfc_get_string ("%s", modname);
7254 26859 : sym->attr.use_assoc = 1;
7255 26859 : sym->from_intmod = module;
7256 26859 : sym->intmod_sym_id = id;
7257 : }
7258 :
7259 :
7260 : /* Import the intrinsic ISO_C_BINDING module, generating symbols in
7261 : the current namespace for all named constants, pointer types, and
7262 : procedures in the module unless the only clause was used or a rename
7263 : list was provided. */
7264 :
7265 : static void
7266 9442 : import_iso_c_binding_module (void)
7267 : {
7268 9442 : gfc_symbol *mod_sym = NULL, *return_type;
7269 9442 : gfc_symtree *mod_symtree = NULL, *tmp_symtree;
7270 9442 : gfc_symtree *c_ptr = NULL, *c_funptr = NULL;
7271 9442 : const char *iso_c_module_name = "__iso_c_binding";
7272 9442 : gfc_use_rename *u;
7273 9442 : int i;
7274 9442 : bool want_c_ptr = false, want_c_funptr = false;
7275 :
7276 : /* Look only in the current namespace. */
7277 9442 : mod_symtree = gfc_find_symtree (gfc_current_ns->sym_root, iso_c_module_name);
7278 :
7279 9442 : if (mod_symtree == NULL)
7280 : {
7281 : /* symtree doesn't already exist in current namespace. */
7282 9368 : gfc_get_sym_tree (iso_c_module_name, gfc_current_ns, &mod_symtree,
7283 : false);
7284 :
7285 9368 : if (mod_symtree != NULL)
7286 9368 : mod_sym = mod_symtree->n.sym;
7287 : else
7288 0 : gfc_internal_error ("import_iso_c_binding_module(): Unable to "
7289 : "create symbol for %s", iso_c_module_name);
7290 :
7291 9368 : mod_sym->attr.flavor = FL_MODULE;
7292 9368 : mod_sym->attr.intrinsic = 1;
7293 9368 : mod_sym->module = gfc_get_string ("%s", iso_c_module_name);
7294 9368 : mod_sym->from_intmod = INTMOD_ISO_C_BINDING;
7295 : }
7296 :
7297 : /* Check whether C_PTR or C_FUNPTR are in the include list, if so, load it;
7298 : check also whether C_NULL_(FUN)PTR or C_(FUN)LOC are requested, which
7299 : need C_(FUN)PTR. */
7300 19132 : for (u = gfc_rename_list; u; u = u->next)
7301 : {
7302 9690 : if (strcmp (c_interop_kinds_table[ISOCBINDING_NULL_PTR].name,
7303 9690 : u->use_name) == 0)
7304 : want_c_ptr = true;
7305 9629 : else if (strcmp (c_interop_kinds_table[ISOCBINDING_LOC].name,
7306 : u->use_name) == 0)
7307 : want_c_ptr = true;
7308 9499 : else if (strcmp (c_interop_kinds_table[ISOCBINDING_NULL_FUNPTR].name,
7309 : u->use_name) == 0)
7310 : want_c_funptr = true;
7311 9490 : else if (strcmp (c_interop_kinds_table[ISOCBINDING_FUNLOC].name,
7312 : u->use_name) == 0)
7313 : want_c_funptr = true;
7314 9454 : else if (strcmp (c_interop_kinds_table[ISOCBINDING_PTR].name,
7315 : u->use_name) == 0)
7316 : {
7317 2248 : c_ptr = generate_isocbinding_symbol (iso_c_module_name,
7318 : (iso_c_binding_symbol)
7319 : ISOCBINDING_PTR,
7320 2248 : u->local_name[0] ? u->local_name
7321 : : u->use_name,
7322 : NULL, false);
7323 : }
7324 7206 : else if (strcmp (c_interop_kinds_table[ISOCBINDING_FUNPTR].name,
7325 : u->use_name) == 0)
7326 : {
7327 107 : c_funptr
7328 107 : = generate_isocbinding_symbol (iso_c_module_name,
7329 : (iso_c_binding_symbol)
7330 : ISOCBINDING_FUNPTR,
7331 107 : u->local_name[0] ? u->local_name
7332 : : u->use_name,
7333 : NULL, false);
7334 : }
7335 : }
7336 :
7337 9442 : if ((want_c_ptr || !only_flag) && !c_ptr)
7338 3291 : c_ptr = generate_isocbinding_symbol (iso_c_module_name,
7339 : (iso_c_binding_symbol)
7340 : ISOCBINDING_PTR,
7341 : NULL, NULL, only_flag);
7342 9442 : if ((want_c_funptr || !only_flag) && !c_funptr)
7343 3255 : c_funptr = generate_isocbinding_symbol (iso_c_module_name,
7344 : (iso_c_binding_symbol)
7345 : ISOCBINDING_FUNPTR,
7346 : NULL, NULL, only_flag);
7347 :
7348 : /* Generate the symbols for the named constants representing
7349 : the kinds for intrinsic data types. */
7350 717592 : for (i = 0; i < ISOCBINDING_NUMBER; i++)
7351 : {
7352 708150 : bool found = false;
7353 1434900 : for (u = gfc_rename_list; u; u = u->next)
7354 726750 : if (strcmp (c_interop_kinds_table[i].name, u->use_name) == 0)
7355 : {
7356 9688 : bool not_in_std;
7357 9688 : const char *name;
7358 9688 : u->found = 1;
7359 9688 : found = true;
7360 :
7361 9688 : switch (i)
7362 : {
7363 : #define NAMED_FUNCTION(a,b,c,d) \
7364 : case a: \
7365 : not_in_std = (gfc_option.allow_std & d) == 0; \
7366 : name = b; \
7367 : break;
7368 : #define NAMED_SUBROUTINE(a,b,c,d) \
7369 : case a: \
7370 : not_in_std = (gfc_option.allow_std & d) == 0; \
7371 : name = b; \
7372 : break;
7373 : #define NAMED_INTCST(a,b,c,d) \
7374 : case a: \
7375 : not_in_std = (gfc_option.allow_std & d) == 0; \
7376 : name = b; \
7377 : break;
7378 : #define NAMED_UINTCST(a,b,c,d) \
7379 : case a: \
7380 : not_in_std = (gfc_option.allow_std & d) == 0; \
7381 : name = b; \
7382 : break;
7383 : #define NAMED_REALCST(a,b,c,d) \
7384 : case a: \
7385 : not_in_std = (gfc_option.allow_std & d) == 0; \
7386 : name = b; \
7387 : break;
7388 : #define NAMED_CMPXCST(a,b,c,d) \
7389 : case a: \
7390 : not_in_std = (gfc_option.allow_std & d) == 0; \
7391 : name = b; \
7392 : break;
7393 : #include "iso-c-binding.def"
7394 : default:
7395 : not_in_std = false;
7396 : name = "";
7397 : }
7398 :
7399 6917 : if (not_in_std)
7400 : {
7401 6 : gfc_error ("The symbol %qs, referenced at %L, is not "
7402 : "in the selected standard", name, &u->where);
7403 6 : continue;
7404 : }
7405 :
7406 9682 : switch (i)
7407 : {
7408 : #define NAMED_FUNCTION(a,b,c,d) \
7409 : case a: \
7410 : if (a == ISOCBINDING_LOC) \
7411 : return_type = c_ptr->n.sym; \
7412 : else if (a == ISOCBINDING_FUNLOC) \
7413 : return_type = c_funptr->n.sym; \
7414 : else \
7415 : return_type = NULL; \
7416 : create_intrinsic_function (u->local_name[0] \
7417 : ? u->local_name : u->use_name, \
7418 : a, iso_c_module_name, \
7419 : INTMOD_ISO_C_BINDING, false, \
7420 : return_type); \
7421 : break;
7422 : #define NAMED_SUBROUTINE(a,b,c,d) \
7423 : case a: \
7424 : create_intrinsic_function (u->local_name[0] ? u->local_name \
7425 : : u->use_name, \
7426 : a, iso_c_module_name, \
7427 : INTMOD_ISO_C_BINDING, true, NULL); \
7428 : break;
7429 : #include "iso-c-binding.def"
7430 :
7431 : case ISOCBINDING_PTR:
7432 : case ISOCBINDING_FUNPTR:
7433 : /* Already handled above. */
7434 : break;
7435 6954 : default:
7436 6954 : if (i == ISOCBINDING_NULL_PTR)
7437 : tmp_symtree = c_ptr;
7438 6893 : else if (i == ISOCBINDING_NULL_FUNPTR)
7439 : tmp_symtree = c_funptr;
7440 : else
7441 6884 : tmp_symtree = NULL;
7442 6954 : generate_isocbinding_symbol (iso_c_module_name,
7443 : (iso_c_binding_symbol) i,
7444 6954 : u->local_name[0]
7445 : ? u->local_name : u->use_name,
7446 : tmp_symtree, false);
7447 : }
7448 : }
7449 :
7450 708150 : if (!found && !only_flag)
7451 : {
7452 : /* Skip, if the symbol is not in the enabled standard. */
7453 244009 : switch (i)
7454 : {
7455 : #define NAMED_FUNCTION(a,b,c,d) \
7456 : case a: \
7457 : if ((gfc_option.allow_std & d) == 0) \
7458 : continue; \
7459 : break;
7460 : #define NAMED_SUBROUTINE(a,b,c,d) \
7461 : case a: \
7462 : if ((gfc_option.allow_std & d) == 0) \
7463 : continue; \
7464 : break;
7465 : #define NAMED_INTCST(a,b,c,d) \
7466 : case a: \
7467 : if ((gfc_option.allow_std & d) == 0) \
7468 : continue; \
7469 : break;
7470 : #define NAMED_UINTCST(a,b,c,d) \
7471 : case a: \
7472 : if ((gfc_option.allow_std & d) == 0) \
7473 : continue; \
7474 : break;
7475 : #define NAMED_REALCST(a,b,c,d) \
7476 : case a: \
7477 : if ((gfc_option.allow_std & d) == 0) \
7478 : continue; \
7479 : break;
7480 : #define NAMED_CMPXCST(a,b,c,d) \
7481 : case a: \
7482 : if ((gfc_option.allow_std & d) == 0) \
7483 : continue; \
7484 : break;
7485 : #include "iso-c-binding.def"
7486 175287 : default:
7487 175287 : ; /* Not GFC_STD_* versioned. */
7488 : }
7489 :
7490 175287 : switch (i)
7491 : {
7492 : #define NAMED_FUNCTION(a,b,c,d) \
7493 : case a: \
7494 : if (a == ISOCBINDING_LOC) \
7495 : return_type = c_ptr->n.sym; \
7496 : else if (a == ISOCBINDING_FUNLOC) \
7497 : return_type = c_funptr->n.sym; \
7498 : else \
7499 : return_type = NULL; \
7500 : create_intrinsic_function (b, a, iso_c_module_name, \
7501 : INTMOD_ISO_C_BINDING, false, \
7502 : return_type); \
7503 : break;
7504 : #define NAMED_SUBROUTINE(a,b,c,d) \
7505 : case a: \
7506 : create_intrinsic_function (b, a, iso_c_module_name, \
7507 : INTMOD_ISO_C_BINDING, true, NULL); \
7508 : break;
7509 : #include "iso-c-binding.def"
7510 :
7511 : case ISOCBINDING_PTR:
7512 : case ISOCBINDING_FUNPTR:
7513 : /* Already handled above. */
7514 : break;
7515 142881 : default:
7516 142881 : if (i == ISOCBINDING_NULL_PTR)
7517 : tmp_symtree = c_ptr;
7518 139627 : else if (i == ISOCBINDING_NULL_FUNPTR)
7519 : tmp_symtree = c_funptr;
7520 : else
7521 136373 : tmp_symtree = NULL;
7522 142881 : generate_isocbinding_symbol (iso_c_module_name,
7523 : (iso_c_binding_symbol) i, NULL,
7524 : tmp_symtree, false);
7525 : }
7526 : }
7527 : }
7528 :
7529 19132 : for (u = gfc_rename_list; u; u = u->next)
7530 : {
7531 9690 : if (u->found)
7532 9688 : continue;
7533 :
7534 2 : gfc_error ("Symbol %qs referenced at %L not found in intrinsic "
7535 2 : "module ISO_C_BINDING", u->use_name, &u->where);
7536 : }
7537 9442 : }
7538 :
7539 :
7540 : /* Add an integer named constant from a given module. */
7541 :
7542 : static void
7543 9859 : create_int_parameter (const char *name, int value, const char *modname,
7544 : intmod_id module, int id)
7545 : {
7546 9859 : gfc_symtree *tmp_symtree;
7547 9859 : gfc_symbol *sym;
7548 :
7549 9859 : tmp_symtree = gfc_find_symtree (gfc_current_ns->sym_root, name);
7550 9859 : if (tmp_symtree != NULL)
7551 : {
7552 0 : if (strcmp (modname, tmp_symtree->n.sym->module) == 0)
7553 0 : return;
7554 : else
7555 0 : gfc_error ("Symbol %qs already declared", name);
7556 : }
7557 :
7558 9859 : gfc_get_sym_tree (name, gfc_current_ns, &tmp_symtree, false);
7559 9859 : sym = tmp_symtree->n.sym;
7560 :
7561 9859 : sym->module = gfc_get_string ("%s", modname);
7562 9859 : sym->attr.flavor = FL_PARAMETER;
7563 9859 : sym->ts.type = BT_INTEGER;
7564 9859 : sym->ts.kind = gfc_default_integer_kind;
7565 9859 : sym->value = gfc_get_int_expr (gfc_default_integer_kind, NULL, value);
7566 9859 : sym->attr.use_assoc = 1;
7567 9859 : sym->from_intmod = module;
7568 9859 : sym->intmod_sym_id = id;
7569 : }
7570 :
7571 :
7572 : /* Value is already contained by the array constructor, but not
7573 : yet the shape. */
7574 :
7575 : static void
7576 1260 : create_int_parameter_array (const char *name, int size, gfc_expr *value,
7577 : const char *modname, intmod_id module, int id)
7578 : {
7579 1260 : gfc_symtree *tmp_symtree;
7580 1260 : gfc_symbol *sym;
7581 :
7582 1260 : tmp_symtree = gfc_find_symtree (gfc_current_ns->sym_root, name);
7583 1260 : if (tmp_symtree != NULL)
7584 : {
7585 1 : if (tmp_symtree->n.sym->module &&
7586 0 : strcmp (modname, tmp_symtree->n.sym->module) == 0)
7587 0 : return;
7588 : else
7589 1 : gfc_error ("Symbol %qs already declared at %L conflicts with "
7590 : "symbol in %qs at %C", name,
7591 : &tmp_symtree->n.sym->declared_at, modname);
7592 : }
7593 :
7594 1260 : gfc_get_sym_tree (name, gfc_current_ns, &tmp_symtree, false);
7595 1260 : sym = tmp_symtree->n.sym;
7596 :
7597 1260 : sym->module = gfc_get_string ("%s", modname);
7598 1260 : sym->attr.flavor = FL_PARAMETER;
7599 1260 : sym->ts.type = BT_INTEGER;
7600 1260 : sym->ts.kind = gfc_default_integer_kind;
7601 1260 : sym->attr.use_assoc = 1;
7602 1260 : sym->from_intmod = module;
7603 1260 : sym->intmod_sym_id = id;
7604 1260 : sym->attr.dimension = 1;
7605 1260 : sym->as = gfc_get_array_spec ();
7606 1260 : sym->as->rank = 1;
7607 1260 : sym->as->type = AS_EXPLICIT;
7608 1260 : sym->as->lower[0] = gfc_get_int_expr (gfc_default_integer_kind, NULL, 1);
7609 1260 : sym->as->upper[0] = gfc_get_int_expr (gfc_default_integer_kind, NULL, size);
7610 :
7611 1260 : sym->value = value;
7612 1260 : sym->value->shape = gfc_get_shape (1);
7613 1260 : mpz_init_set_ui (sym->value->shape[0], size);
7614 : }
7615 :
7616 :
7617 : /* Add an derived type for a given module. */
7618 :
7619 : static void
7620 1003 : create_derived_type (const char *name, const char *modname,
7621 : intmod_id module, int id)
7622 : {
7623 1003 : gfc_symtree *tmp_symtree;
7624 1003 : gfc_symbol *sym, *dt_sym;
7625 1003 : gfc_interface *intr, *head;
7626 :
7627 1003 : tmp_symtree = gfc_find_symtree (gfc_current_ns->sym_root, name);
7628 1003 : if (tmp_symtree != NULL)
7629 : {
7630 0 : if (strcmp (modname, tmp_symtree->n.sym->module) == 0)
7631 0 : return;
7632 : else
7633 0 : gfc_error ("Symbol %qs already declared", name);
7634 : }
7635 :
7636 1003 : gfc_get_sym_tree (name, gfc_current_ns, &tmp_symtree, false);
7637 1003 : sym = tmp_symtree->n.sym;
7638 1003 : sym->module = gfc_get_string ("%s", modname);
7639 1003 : sym->from_intmod = module;
7640 1003 : sym->intmod_sym_id = id;
7641 1003 : sym->attr.flavor = FL_PROCEDURE;
7642 1003 : sym->attr.function = 1;
7643 1003 : sym->attr.generic = 1;
7644 :
7645 1003 : gfc_get_sym_tree (gfc_dt_upper_string (sym->name),
7646 : gfc_current_ns, &tmp_symtree, false);
7647 1003 : dt_sym = tmp_symtree->n.sym;
7648 1003 : dt_sym->name = gfc_get_string ("%s", sym->name);
7649 1003 : dt_sym->attr.flavor = FL_DERIVED;
7650 1003 : dt_sym->attr.private_comp = 1;
7651 1003 : dt_sym->attr.zero_comp = 1;
7652 1003 : dt_sym->attr.use_assoc = 1;
7653 1003 : dt_sym->module = gfc_get_string ("%s", modname);
7654 1003 : dt_sym->from_intmod = module;
7655 1003 : dt_sym->intmod_sym_id = id;
7656 :
7657 1003 : head = sym->generic;
7658 1003 : intr = gfc_get_interface ();
7659 1003 : intr->sym = dt_sym;
7660 1003 : intr->where = gfc_current_locus;
7661 1003 : intr->next = head;
7662 1003 : sym->generic = intr;
7663 1003 : sym->attr.if_source = IFSRC_DECL;
7664 : }
7665 :
7666 :
7667 : /* Read the contents of the module file into a temporary buffer. */
7668 :
7669 : static void
7670 13571 : read_module_to_tmpbuf ()
7671 : {
7672 : /* We don't know the uncompressed size, so enlarge the buffer as
7673 : needed. */
7674 13571 : int cursz = 4096;
7675 13571 : int rsize = cursz;
7676 13571 : int len = 0;
7677 :
7678 13571 : module_content = XNEWVEC (char, cursz);
7679 :
7680 56983 : while (1)
7681 : {
7682 35277 : int nread = gzread (module_fp, module_content + len, rsize);
7683 35277 : len += nread;
7684 35277 : if (nread < rsize)
7685 : break;
7686 21706 : cursz *= 2;
7687 21706 : module_content = XRESIZEVEC (char, module_content, cursz);
7688 21706 : rsize = cursz - len;
7689 21706 : }
7690 :
7691 13571 : module_content = XRESIZEVEC (char, module_content, len + 1);
7692 13571 : module_content[len] = '\0';
7693 :
7694 13571 : module_pos = 0;
7695 13571 : }
7696 :
7697 :
7698 : /* USE the ISO_FORTRAN_ENV intrinsic module. */
7699 :
7700 : static void
7701 609 : use_iso_fortran_env_module (void)
7702 : {
7703 609 : static char mod[] = "iso_fortran_env";
7704 609 : gfc_use_rename *u;
7705 609 : gfc_symbol *mod_sym;
7706 609 : gfc_symtree *mod_symtree;
7707 609 : gfc_expr *expr;
7708 609 : int i, j;
7709 :
7710 609 : intmod_sym symbol[] = {
7711 : #define NAMED_INTCST(a,b,c,d) { a, b, 0, d },
7712 : #define NAMED_UINTCST(a,b,c,d) { a, b, 0, d },
7713 : #define NAMED_KINDARRAY(a,b,c,d) { a, b, 0, d },
7714 : #define NAMED_DERIVED_TYPE(a,b,c,d) { a, b, 0, d },
7715 : #define NAMED_FUNCTION(a,b,c,d) { a, b, c, d },
7716 : #define NAMED_SUBROUTINE(a,b,c,d) { a, b, c, d },
7717 : #include "iso-fortran-env.def"
7718 : { ISOFORTRANENV_INVALID, NULL, -1234, 0 } };
7719 :
7720 : /* We could have used c in the NAMED_{,U}INTCST macros
7721 : instead of 0, but then current g++ expands the initialization
7722 : as clearing the whole object followed by explicit stores of
7723 : all the non-zero elements (over 150), while by using 0s for
7724 : the non-constant initializers and initializing them afterwards
7725 : g++ will often copy everything from .rodata and then only override
7726 : over 30 non-constant ones. */
7727 609 : i = 0;
7728 : #define NAMED_INTCST(a,b,c,d) symbol[i++].value = c;
7729 : #define NAMED_UINTCST(a,b,c,d) symbol[i++].value = c;
7730 : #define NAMED_KINDARRAY(a,b,c,d) i++;
7731 : #define NAMED_DERIVED_TYPE(a,b,c,d) i++;
7732 : #define NAMED_FUNCTION(a,b,c,d) i++;
7733 : #define NAMED_SUBROUTINE(a,b,c,d) i++;
7734 : #include "iso-fortran-env.def"
7735 609 : gcc_checking_assert (i == (int) ARRAY_SIZE (symbol) - 1);
7736 :
7737 : /* Generate the symbol for the module itself. */
7738 609 : mod_symtree = gfc_find_symtree (gfc_current_ns->sym_root, mod);
7739 609 : if (mod_symtree == NULL)
7740 : {
7741 608 : gfc_get_sym_tree (mod, gfc_current_ns, &mod_symtree, false);
7742 608 : gcc_assert (mod_symtree);
7743 608 : mod_sym = mod_symtree->n.sym;
7744 :
7745 608 : mod_sym->attr.flavor = FL_MODULE;
7746 608 : mod_sym->attr.intrinsic = 1;
7747 608 : mod_sym->module = gfc_get_string ("%s", mod);
7748 608 : mod_sym->from_intmod = INTMOD_ISO_FORTRAN_ENV;
7749 : }
7750 : else
7751 1 : if (!mod_symtree->n.sym->attr.intrinsic)
7752 1 : gfc_error ("Use of intrinsic module %qs at %C conflicts with "
7753 : "non-intrinsic module name used previously", mod);
7754 :
7755 : /* Generate the symbols for the module integer named constants. */
7756 :
7757 27405 : for (i = 0; symbol[i].name; i++)
7758 : {
7759 26796 : bool found = false;
7760 49236 : for (u = gfc_rename_list; u; u = u->next)
7761 : {
7762 22440 : if (strcmp (symbol[i].name, u->use_name) == 0)
7763 : {
7764 510 : found = true;
7765 510 : u->found = 1;
7766 :
7767 510 : if (!gfc_notify_std (symbol[i].standard, "The symbol %qs, "
7768 : "referenced at %L, is not in the selected "
7769 : "standard", symbol[i].name, &u->where))
7770 11 : continue;
7771 :
7772 499 : if ((flag_default_integer || flag_default_real_8)
7773 2 : && symbol[i].id == ISOFORTRANENV_NUMERIC_STORAGE_SIZE)
7774 0 : gfc_warning_now (0, "Use of the NUMERIC_STORAGE_SIZE named "
7775 : "constant from intrinsic module "
7776 : "ISO_FORTRAN_ENV at %L is incompatible with "
7777 : "option %qs", &u->where,
7778 : flag_default_integer
7779 : ? "-fdefault-integer-8"
7780 : : "-fdefault-real-8");
7781 499 : switch (symbol[i].id)
7782 : {
7783 : #define NAMED_INTCST(a,b,c,d) \
7784 : case a:
7785 : #include "iso-fortran-env.def"
7786 333 : create_int_parameter (u->local_name[0] ? u->local_name
7787 : : u->use_name,
7788 : symbol[i].value, mod,
7789 : INTMOD_ISO_FORTRAN_ENV, symbol[i].id);
7790 333 : break;
7791 :
7792 : #define NAMED_UINTCST(a,b,c,d) \
7793 : case a:
7794 : #include "iso-fortran-env.def"
7795 30 : create_int_parameter (u->local_name[0] ? u->local_name
7796 : : u->use_name,
7797 : symbol[i].value, mod,
7798 : INTMOD_ISO_FORTRAN_ENV, symbol[i].id);
7799 30 : break;
7800 :
7801 : #define NAMED_KINDARRAY(a,b,KINDS,d) \
7802 : case a:\
7803 : expr = gfc_get_array_expr (BT_INTEGER, \
7804 : gfc_default_integer_kind,\
7805 : NULL); \
7806 : for (j = 0; KINDS[j].kind != 0; j++) \
7807 : gfc_constructor_append_expr (&expr->value.constructor, \
7808 : gfc_get_int_expr (gfc_default_integer_kind, NULL, \
7809 : KINDS[j].kind), NULL); \
7810 : create_int_parameter_array (u->local_name[0] ? u->local_name \
7811 : : u->use_name, \
7812 : j, expr, mod, \
7813 : INTMOD_ISO_FORTRAN_ENV, \
7814 : symbol[i].id); \
7815 : break;
7816 : #include "iso-fortran-env.def"
7817 :
7818 : #define NAMED_DERIVED_TYPE(a,b,TYPE,STD) \
7819 : case a:
7820 : #include "iso-fortran-env.def"
7821 85 : create_derived_type (u->local_name[0] ? u->local_name
7822 : : u->use_name,
7823 : mod, INTMOD_ISO_FORTRAN_ENV,
7824 : symbol[i].id);
7825 85 : break;
7826 :
7827 : #define NAMED_FUNCTION(a,b,c,d) \
7828 : case a:
7829 : #include "iso-fortran-env.def"
7830 15 : create_intrinsic_function (u->local_name[0] ? u->local_name
7831 : : u->use_name,
7832 : symbol[i].id, mod,
7833 : INTMOD_ISO_FORTRAN_ENV, false,
7834 : NULL);
7835 15 : break;
7836 :
7837 0 : default:
7838 0 : gcc_unreachable ();
7839 : }
7840 : }
7841 : }
7842 :
7843 26796 : if (!found && !only_flag)
7844 : {
7845 13581 : if ((gfc_option.allow_std & symbol[i].standard) == 0)
7846 1332 : continue;
7847 :
7848 12249 : if ((flag_default_integer || flag_default_real_8)
7849 0 : && symbol[i].id == ISOFORTRANENV_NUMERIC_STORAGE_SIZE)
7850 0 : gfc_warning_now (0,
7851 : "Use of the NUMERIC_STORAGE_SIZE named constant "
7852 : "from intrinsic module ISO_FORTRAN_ENV at %C is "
7853 : "incompatible with option %s",
7854 : flag_default_integer
7855 : ? "-fdefault-integer-8" : "-fdefault-real-8");
7856 :
7857 12249 : switch (symbol[i].id)
7858 : {
7859 : #define NAMED_INTCST(a,b,c,d) \
7860 : case a:
7861 : #include "iso-fortran-env.def"
7862 9492 : create_int_parameter (symbol[i].name, symbol[i].value, mod,
7863 : INTMOD_ISO_FORTRAN_ENV, symbol[i].id);
7864 9492 : break;
7865 :
7866 : #define NAMED_UINTCST(a,b,c,d) \
7867 : case a:
7868 : #include "iso-fortran-env.def"
7869 4 : create_int_parameter (symbol[i].name, symbol[i].value, mod,
7870 : INTMOD_ISO_FORTRAN_ENV, symbol[i].id);
7871 4 : break;
7872 :
7873 : #define NAMED_KINDARRAY(a,b,KINDS,d) \
7874 : case a:\
7875 : expr = gfc_get_array_expr (BT_INTEGER, gfc_default_integer_kind, \
7876 : NULL); \
7877 : for (j = 0; KINDS[j].kind != 0; j++) \
7878 : gfc_constructor_append_expr (&expr->value.constructor, \
7879 : gfc_get_int_expr (gfc_default_integer_kind, NULL, \
7880 : KINDS[j].kind), NULL); \
7881 : create_int_parameter_array (symbol[i].name, j, expr, mod, \
7882 : INTMOD_ISO_FORTRAN_ENV, symbol[i].id);\
7883 : break;
7884 : #include "iso-fortran-env.def"
7885 :
7886 : #define NAMED_DERIVED_TYPE(a,b,TYPE,STD) \
7887 : case a:
7888 : #include "iso-fortran-env.def"
7889 918 : create_derived_type (symbol[i].name, mod, INTMOD_ISO_FORTRAN_ENV,
7890 : symbol[i].id);
7891 918 : break;
7892 :
7893 : #define NAMED_FUNCTION(a,b,c,d) \
7894 : case a:
7895 : #include "iso-fortran-env.def"
7896 611 : create_intrinsic_function (symbol[i].name, symbol[i].id, mod,
7897 : INTMOD_ISO_FORTRAN_ENV, false, NULL);
7898 611 : break;
7899 :
7900 0 : default:
7901 0 : gcc_unreachable ();
7902 : }
7903 : }
7904 : }
7905 :
7906 1119 : for (u = gfc_rename_list; u; u = u->next)
7907 : {
7908 510 : if (u->found)
7909 510 : continue;
7910 :
7911 0 : gfc_error ("Symbol %qs referenced at %L not found in intrinsic "
7912 0 : "module ISO_FORTRAN_ENV", u->use_name, &u->where);
7913 : }
7914 609 : }
7915 :
7916 :
7917 : /* Process a USE directive. */
7918 :
7919 : static void
7920 23626 : gfc_use_module (gfc_use_list *module)
7921 : {
7922 23626 : char *filename;
7923 23626 : gfc_state_data *p;
7924 23626 : int c, line, start;
7925 23626 : gfc_symtree *mod_symtree;
7926 23626 : gfc_use_list *use_stmt;
7927 23626 : locus old_locus = gfc_current_locus;
7928 :
7929 23626 : gfc_current_locus = module->where;
7930 23626 : module_name = module->module_name;
7931 23626 : gfc_rename_list = module->rename;
7932 23626 : only_flag = module->only_flag;
7933 23626 : current_intmod = INTMOD_NONE;
7934 :
7935 23626 : if (!only_flag && gfc_state_stack->state != COMP_SUBMODULE)
7936 15050 : gfc_warning_now (OPT_Wuse_without_only,
7937 : "USE statement at %C has no ONLY qualifier");
7938 :
7939 23626 : if (gfc_state_stack->state == COMP_MODULE
7940 21097 : || module->submodule_name == NULL)
7941 : {
7942 23363 : filename = XALLOCAVEC (char, strlen (module_name)
7943 : + strlen (MODULE_EXTENSION) + 1);
7944 23363 : strcpy (filename, module_name);
7945 23363 : strcat (filename, MODULE_EXTENSION);
7946 : }
7947 : else
7948 : {
7949 263 : filename = XALLOCAVEC (char, strlen (module->submodule_name)
7950 : + strlen (SUBMODULE_EXTENSION) + 1);
7951 263 : strcpy (filename, module->submodule_name);
7952 263 : strcat (filename, SUBMODULE_EXTENSION);
7953 : }
7954 :
7955 : /* First, try to find an non-intrinsic module, unless the USE statement
7956 : specified that the module is intrinsic. */
7957 23626 : module_fp = NULL;
7958 23626 : if (!module->intrinsic)
7959 20187 : module_fp = gzopen_included_file (filename, true, true);
7960 :
7961 : /* Then, see if it's an intrinsic one, unless the USE statement
7962 : specified that the module is non-intrinsic. */
7963 23626 : if (module_fp == NULL && !module->non_intrinsic)
7964 : {
7965 11281 : if (strcmp (module_name, "iso_fortran_env") == 0
7966 11281 : && gfc_notify_std (GFC_STD_F2003, "ISO_FORTRAN_ENV "
7967 : "intrinsic module at %C"))
7968 : {
7969 609 : use_iso_fortran_env_module ();
7970 609 : free_rename (module->rename);
7971 609 : module->rename = NULL;
7972 609 : gfc_current_locus = old_locus;
7973 609 : module->intrinsic = true;
7974 10051 : return;
7975 : }
7976 :
7977 10672 : if (strcmp (module_name, "iso_c_binding") == 0
7978 10672 : && gfc_notify_std (GFC_STD_F2003, "ISO_C_BINDING module at %C"))
7979 : {
7980 9442 : import_iso_c_binding_module();
7981 9442 : free_rename (module->rename);
7982 9442 : module->rename = NULL;
7983 9442 : gfc_current_locus = old_locus;
7984 9442 : module->intrinsic = true;
7985 9442 : return;
7986 : }
7987 :
7988 1230 : module_fp = gzopen_intrinsic_module (filename);
7989 :
7990 1230 : if (module_fp == NULL && module->intrinsic)
7991 0 : gfc_fatal_error ("Cannot find an intrinsic module named %qs at %C",
7992 : module_name);
7993 :
7994 : /* Check for the IEEE modules, so we can mark their symbols
7995 : accordingly when we read them. */
7996 1230 : if (strcmp (module_name, "ieee_features") == 0
7997 1230 : && gfc_notify_std (GFC_STD_F2003, "IEEE_FEATURES module at %C"))
7998 : {
7999 57 : current_intmod = INTMOD_IEEE_FEATURES;
8000 : }
8001 1173 : else if (strcmp (module_name, "ieee_exceptions") == 0
8002 1173 : && gfc_notify_std (GFC_STD_F2003,
8003 : "IEEE_EXCEPTIONS module at %C"))
8004 : {
8005 60 : current_intmod = INTMOD_IEEE_EXCEPTIONS;
8006 : }
8007 1113 : else if (strcmp (module_name, "ieee_arithmetic") == 0
8008 1113 : && gfc_notify_std (GFC_STD_F2003,
8009 : "IEEE_ARITHMETIC module at %C"))
8010 : {
8011 405 : current_intmod = INTMOD_IEEE_ARITHMETIC;
8012 : }
8013 : }
8014 :
8015 13575 : if (module_fp == NULL)
8016 : {
8017 4 : if (gfc_state_stack->state != COMP_SUBMODULE
8018 3 : && module->submodule_name == NULL)
8019 3 : gfc_fatal_error ("Cannot open module file %qs for reading at %C: %s",
8020 3 : filename, xstrerror (errno));
8021 : else
8022 1 : gfc_fatal_error ("Module file %qs has not been generated, either "
8023 : "because the module does not contain a MODULE "
8024 : "PROCEDURE or there is an error in the module.",
8025 : filename);
8026 : }
8027 :
8028 : /* Check that we haven't already USEd an intrinsic module with the
8029 : same name. */
8030 :
8031 13571 : mod_symtree = gfc_find_symtree (gfc_current_ns->sym_root, module_name);
8032 13571 : if (mod_symtree && mod_symtree->n.sym->attr.intrinsic)
8033 1 : gfc_error ("Use of non-intrinsic module %qs at %C conflicts with "
8034 : "intrinsic module name used previously", module_name);
8035 :
8036 13571 : iomode = IO_INPUT;
8037 13571 : module_line = 1;
8038 13571 : module_column = 1;
8039 13571 : start = 0;
8040 :
8041 13571 : read_module_to_tmpbuf ();
8042 13571 : gzclose (module_fp);
8043 :
8044 : /* Skip the first line of the module, after checking that this is
8045 : a gfortran module file. */
8046 13571 : line = 0;
8047 501565 : while (line < 1)
8048 : {
8049 474423 : c = module_char ();
8050 474423 : if (c == EOF)
8051 0 : bad_module ("Unexpected end of module");
8052 474423 : if (start++ < 3)
8053 40713 : parse_name (c);
8054 474423 : if ((start == 1 && strcmp (atom_name, "GFORTRAN") != 0)
8055 474423 : || (start == 2 && strcmp (atom_name, " module") != 0))
8056 0 : gfc_fatal_error ("File %qs opened at %C is not a GNU Fortran"
8057 : " module file", module_fullpath);
8058 474423 : if (start == 3)
8059 : {
8060 13571 : bool fatal = false;
8061 13571 : if (strcmp (atom_name, " version") != 0
8062 13571 : || module_char () != ' '
8063 27142 : || parse_atom () != ATOM_STRING)
8064 : fatal = true;
8065 13571 : else if (strcmp (atom_string, MOD_VERSION))
8066 : {
8067 : static const char *compat_mod_versions[] = COMPAT_MOD_VERSIONS;
8068 0 : fatal = true;
8069 0 : for (unsigned i = 0; i < ARRAY_SIZE (compat_mod_versions); ++i)
8070 0 : if (!strcmp (atom_string, compat_mod_versions[i]))
8071 : {
8072 : fatal = false;
8073 : break;
8074 : }
8075 : }
8076 0 : if (fatal)
8077 0 : gfc_fatal_error ("Cannot read module file %qs opened at %C,"
8078 : " because it was created by a different"
8079 : " version of GNU Fortran", module_fullpath);
8080 :
8081 13571 : free (atom_string);
8082 : }
8083 :
8084 474423 : if (c == '\n')
8085 13571 : line++;
8086 : }
8087 :
8088 : /* Make sure we're not reading the same module that we may be building. */
8089 45702 : for (p = gfc_state_stack; p; p = p->previous)
8090 32131 : if ((p->state == COMP_MODULE || p->state == COMP_SUBMODULE)
8091 2309 : && strcmp (p->sym->name, module_name) == 0)
8092 : {
8093 0 : if (p->state == COMP_SUBMODULE)
8094 0 : gfc_fatal_error ("Cannot USE a submodule that is currently built");
8095 : else
8096 0 : gfc_fatal_error ("Cannot USE a module that is currently built");
8097 : }
8098 :
8099 13571 : init_pi_tree ();
8100 13571 : init_true_name_tree ();
8101 :
8102 13571 : read_module ();
8103 :
8104 13571 : free_true_name (true_name_root);
8105 13571 : true_name_root = NULL;
8106 :
8107 13571 : free_pi_tree (pi_root);
8108 13571 : pi_root = NULL;
8109 :
8110 13571 : XDELETEVEC (module_content);
8111 13571 : module_content = NULL;
8112 :
8113 13571 : use_stmt = gfc_get_use_list ();
8114 13571 : *use_stmt = *module;
8115 13571 : use_stmt->next = gfc_current_ns->use_stmts;
8116 13571 : gfc_current_ns->use_stmts = use_stmt;
8117 :
8118 13571 : gfc_current_locus = old_locus;
8119 : }
8120 :
8121 :
8122 : /* Remove duplicated intrinsic operators from the rename list. */
8123 :
8124 : static void
8125 23626 : rename_list_remove_duplicate (gfc_use_rename *list)
8126 : {
8127 23626 : gfc_use_rename *seek, *last;
8128 :
8129 36784 : for (; list; list = list->next)
8130 13158 : if (list->op != INTRINSIC_USER && list->op != INTRINSIC_NONE)
8131 : {
8132 113 : last = list;
8133 459 : for (seek = list->next; seek; seek = last->next)
8134 : {
8135 346 : if (list->op == seek->op)
8136 : {
8137 2 : last->next = seek->next;
8138 2 : free (seek);
8139 : }
8140 : else
8141 : last = seek;
8142 : }
8143 : }
8144 23626 : }
8145 :
8146 :
8147 : /* Process all USE directives. */
8148 :
8149 : void
8150 20469 : gfc_use_modules (void)
8151 : {
8152 20469 : gfc_use_list *next, *seek, *last;
8153 :
8154 44095 : for (next = module_list; next; next = next->next)
8155 : {
8156 23626 : bool non_intrinsic = next->non_intrinsic;
8157 23626 : bool intrinsic = next->intrinsic;
8158 23626 : bool neither = !non_intrinsic && !intrinsic;
8159 :
8160 27577 : for (seek = next->next; seek; seek = seek->next)
8161 : {
8162 3951 : if (next->module_name != seek->module_name)
8163 3776 : continue;
8164 :
8165 175 : if (seek->non_intrinsic)
8166 : non_intrinsic = true;
8167 174 : else if (seek->intrinsic)
8168 : intrinsic = true;
8169 : else
8170 134 : neither = true;
8171 : }
8172 :
8173 23626 : if (intrinsic && neither && !non_intrinsic)
8174 : {
8175 1 : char *filename;
8176 1 : FILE *fp;
8177 :
8178 1 : filename = XALLOCAVEC (char,
8179 : strlen (next->module_name)
8180 : + strlen (MODULE_EXTENSION) + 1);
8181 1 : strcpy (filename, next->module_name);
8182 1 : strcat (filename, MODULE_EXTENSION);
8183 1 : fp = gfc_open_included_file (filename, true, true);
8184 1 : if (fp != NULL)
8185 : {
8186 0 : non_intrinsic = true;
8187 0 : fclose (fp);
8188 : }
8189 : }
8190 :
8191 23626 : last = next;
8192 27577 : for (seek = next->next; seek; seek = last->next)
8193 : {
8194 3951 : if (next->module_name != seek->module_name)
8195 : {
8196 3776 : last = seek;
8197 3776 : continue;
8198 : }
8199 :
8200 175 : if ((!next->intrinsic && !seek->intrinsic)
8201 41 : || (next->intrinsic && seek->intrinsic)
8202 3 : || !non_intrinsic)
8203 : {
8204 173 : if (!seek->only_flag)
8205 18 : next->only_flag = false;
8206 173 : if (seek->rename)
8207 : {
8208 : gfc_use_rename *r = seek->rename;
8209 310 : while (r->next)
8210 : r = r->next;
8211 168 : r->next = next->rename;
8212 168 : next->rename = seek->rename;
8213 : }
8214 173 : last->next = seek->next;
8215 173 : free (seek);
8216 173 : }
8217 : else
8218 : last = seek;
8219 : }
8220 : }
8221 :
8222 44091 : for (; module_list; module_list = next)
8223 : {
8224 23626 : next = module_list->next;
8225 23626 : rename_list_remove_duplicate (module_list->rename);
8226 23626 : gfc_use_module (module_list);
8227 23622 : free (module_list);
8228 : }
8229 20465 : module_list = NULL;
8230 20465 : old_module_list_tail = &module_list;
8231 20465 : gfc_rename_list = NULL;
8232 20465 : }
8233 :
8234 :
8235 : void
8236 9591007 : gfc_free_use_stmts (gfc_use_list *use_stmts)
8237 : {
8238 9591007 : gfc_use_list *next;
8239 9604582 : for (; use_stmts; use_stmts = next)
8240 : {
8241 : gfc_use_rename *next_rename;
8242 :
8243 16533 : for (; use_stmts->rename; use_stmts->rename = next_rename)
8244 : {
8245 2958 : next_rename = use_stmts->rename->next;
8246 2958 : free (use_stmts->rename);
8247 : }
8248 13575 : next = use_stmts->next;
8249 13575 : free (use_stmts);
8250 : }
8251 9591007 : }
8252 :
8253 :
8254 : /* Remember the end of the MODULE_LIST list, so that the list can be restored
8255 : to its previous state if the current statement is erroneous. */
8256 :
8257 : void
8258 1452229 : gfc_save_module_list ()
8259 : {
8260 1452229 : gfc_use_list **tail = &module_list;
8261 1481896 : while (*tail != NULL)
8262 29667 : tail = &(*tail)->next;
8263 1452229 : old_module_list_tail = tail;
8264 1452229 : }
8265 :
8266 :
8267 : /* Restore the MODULE_LIST list to its previous value and free the use
8268 : statements that are no longer part of the list. */
8269 :
8270 : void
8271 9062168 : gfc_restore_old_module_list ()
8272 : {
8273 9062168 : gfc_free_use_stmts (*old_module_list_tail);
8274 9062168 : *old_module_list_tail = NULL;
8275 9062168 : }
8276 :
8277 :
8278 : void
8279 81313 : gfc_module_init_2 (void)
8280 : {
8281 81313 : last_atom = ATOM_LPAREN;
8282 81313 : gfc_rename_list = NULL;
8283 81313 : module_list = NULL;
8284 81313 : }
8285 :
8286 :
8287 : void
8288 81661 : gfc_module_done_2 (void)
8289 : {
8290 81661 : free_rename (gfc_rename_list);
8291 81661 : gfc_rename_list = NULL;
8292 81661 : }
|