aboutsummaryrefslogtreecommitdiff
path: root/gcc/fortran/array.c
diff options
context:
space:
mode:
Diffstat (limited to 'gcc/fortran/array.c')
-rw-r--r--gcc/fortran/array.c2785
1 files changed, 0 insertions, 2785 deletions
diff --git a/gcc/fortran/array.c b/gcc/fortran/array.c
deleted file mode 100644
index 4723043..0000000
--- a/gcc/fortran/array.c
+++ /dev/null
@@ -1,2785 +0,0 @@
-/* Array things
- Copyright (C) 2000-2022 Free Software Foundation, Inc.
- Contributed by Andy Vaught
-
-This file is part of GCC.
-
-GCC is free software; you can redistribute it and/or modify it under
-the terms of the GNU General Public License as published by the Free
-Software Foundation; either version 3, or (at your option) any later
-version.
-
-GCC is distributed in the hope that it will be useful, but WITHOUT ANY
-WARRANTY; without even the implied warranty of MERCHANTABILITY or
-FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
-for more details.
-
-You should have received a copy of the GNU General Public License
-along with GCC; see the file COPYING3. If not see
-<http://www.gnu.org/licenses/>. */
-
-#include "config.h"
-#include "system.h"
-#include "coretypes.h"
-#include "options.h"
-#include "gfortran.h"
-#include "parse.h"
-#include "match.h"
-#include "constructor.h"
-
-/**************** Array reference matching subroutines *****************/
-
-/* Copy an array reference structure. */
-
-gfc_array_ref *
-gfc_copy_array_ref (gfc_array_ref *src)
-{
- gfc_array_ref *dest;
- int i;
-
- if (src == NULL)
- return NULL;
-
- dest = gfc_get_array_ref ();
-
- *dest = *src;
-
- for (i = 0; i < GFC_MAX_DIMENSIONS; i++)
- {
- dest->start[i] = gfc_copy_expr (src->start[i]);
- dest->end[i] = gfc_copy_expr (src->end[i]);
- dest->stride[i] = gfc_copy_expr (src->stride[i]);
- }
-
- return dest;
-}
-
-
-/* Match a single dimension of an array reference. This can be a
- single element or an array section. Any modifications we've made
- to the ar structure are cleaned up by the caller. If the init
- is set, we require the subscript to be a valid initialization
- expression. */
-
-static match
-match_subscript (gfc_array_ref *ar, int init, bool match_star)
-{
- match m = MATCH_ERROR;
- bool star = false;
- int i;
- bool saw_boz = false;
-
- i = ar->dimen + ar->codimen;
-
- gfc_gobble_whitespace ();
- ar->c_where[i] = gfc_current_locus;
- ar->start[i] = ar->end[i] = ar->stride[i] = NULL;
-
- /* We can't be sure of the difference between DIMEN_ELEMENT and
- DIMEN_VECTOR until we know the type of the element itself at
- resolution time. */
-
- ar->dimen_type[i] = DIMEN_UNKNOWN;
-
- if (gfc_match_char (':') == MATCH_YES)
- goto end_element;
-
- /* Get start element. */
- if (match_star && (m = gfc_match_char ('*')) == MATCH_YES)
- star = true;
-
- if (!star && init)
- m = gfc_match_init_expr (&ar->start[i]);
- else if (!star)
- m = gfc_match_expr (&ar->start[i]);
-
- if (ar->start[i] && ar->start[i]->ts.type == BT_BOZ)
- {
- gfc_error ("Invalid BOZ literal constant used in subscript at %C");
- saw_boz = true;
- }
-
- if (m == MATCH_NO)
- gfc_error ("Expected array subscript at %C");
- if (m != MATCH_YES)
- return MATCH_ERROR;
-
- if (gfc_match_char (':') == MATCH_NO)
- goto matched;
-
- if (star)
- {
- gfc_error ("Unexpected %<*%> in coarray subscript at %C");
- return MATCH_ERROR;
- }
-
- /* Get an optional end element. Because we've seen the colon, we
- definitely have a range along this dimension. */
-end_element:
- ar->dimen_type[i] = DIMEN_RANGE;
-
- if (match_star && (m = gfc_match_char ('*')) == MATCH_YES)
- star = true;
- else if (init)
- m = gfc_match_init_expr (&ar->end[i]);
- else
- m = gfc_match_expr (&ar->end[i]);
-
- if (ar->end[i] && ar->end[i]->ts.type == BT_BOZ)
- {
- gfc_error ("Invalid BOZ literal constant used in subscript at %C");
- saw_boz = true;
- }
-
- if (m == MATCH_ERROR)
- return MATCH_ERROR;
-
- /* See if we have an optional stride. */
- if (gfc_match_char (':') == MATCH_YES)
- {
- if (star)
- {
- gfc_error ("Strides not allowed in coarray subscript at %C");
- return MATCH_ERROR;
- }
-
- m = init ? gfc_match_init_expr (&ar->stride[i])
- : gfc_match_expr (&ar->stride[i]);
-
- if (ar->stride[i] && ar->stride[i]->ts.type == BT_BOZ)
- {
- gfc_error ("Invalid BOZ literal constant used in subscript at %C");
- saw_boz = true;
- }
-
- if (m == MATCH_NO)
- gfc_error ("Expected array subscript stride at %C");
- if (m != MATCH_YES)
- return MATCH_ERROR;
- }
-
-matched:
- if (star)
- ar->dimen_type[i] = DIMEN_STAR;
-
- return (saw_boz ? MATCH_ERROR : MATCH_YES);
-}
-
-
-/* Match an array reference, whether it is the whole array or particular
- elements or a section. If init is set, the reference has to consist
- of init expressions. */
-
-match
-gfc_match_array_ref (gfc_array_ref *ar, gfc_array_spec *as, int init,
- int corank)
-{
- match m;
- bool matched_bracket = false;
- gfc_expr *tmp;
- bool stat_just_seen = false;
- bool team_just_seen = false;
-
- memset (ar, '\0', sizeof (*ar));
-
- ar->where = gfc_current_locus;
- ar->as = as;
- ar->type = AR_UNKNOWN;
-
- if (gfc_match_char ('[') == MATCH_YES)
- {
- matched_bracket = true;
- goto coarray;
- }
-
- if (gfc_match_char ('(') != MATCH_YES)
- {
- ar->type = AR_FULL;
- ar->dimen = 0;
- return MATCH_YES;
- }
-
- for (ar->dimen = 0; ar->dimen < GFC_MAX_DIMENSIONS; ar->dimen++)
- {
- m = match_subscript (ar, init, false);
- if (m == MATCH_ERROR)
- return MATCH_ERROR;
-
- if (gfc_match_char (')') == MATCH_YES)
- {
- ar->dimen++;
- goto coarray;
- }
-
- if (gfc_match_char (',') != MATCH_YES)
- {
- gfc_error ("Invalid form of array reference at %C");
- return MATCH_ERROR;
- }
- }
-
- if (ar->dimen >= 7
- && !gfc_notify_std (GFC_STD_F2008,
- "Array reference at %C has more than 7 dimensions"))
- return MATCH_ERROR;
-
- gfc_error ("Array reference at %C cannot have more than %d dimensions",
- GFC_MAX_DIMENSIONS);
- return MATCH_ERROR;
-
-coarray:
- if (!matched_bracket && gfc_match_char ('[') != MATCH_YES)
- {
- if (ar->dimen > 0)
- return MATCH_YES;
- else
- return MATCH_ERROR;
- }
-
- if (flag_coarray == GFC_FCOARRAY_NONE)
- {
- gfc_fatal_error ("Coarrays disabled at %C, use %<-fcoarray=%> to enable");
- return MATCH_ERROR;
- }
-
- if (corank == 0)
- {
- gfc_error ("Unexpected coarray designator at %C");
- return MATCH_ERROR;
- }
-
- ar->stat = NULL;
-
- for (ar->codimen = 0; ar->codimen + ar->dimen < GFC_MAX_DIMENSIONS; ar->codimen++)
- {
- m = match_subscript (ar, init, true);
- if (m == MATCH_ERROR)
- return MATCH_ERROR;
-
- team_just_seen = false;
- stat_just_seen = false;
- if (gfc_match (" , team = %e", &tmp) == MATCH_YES && ar->team == NULL)
- {
- ar->team = tmp;
- team_just_seen = true;
- }
-
- if (ar->team && !team_just_seen)
- {
- gfc_error ("TEAM= attribute in %C misplaced");
- return MATCH_ERROR;
- }
-
- if (gfc_match (" , stat = %e",&tmp) == MATCH_YES && ar->stat == NULL)
- {
- ar->stat = tmp;
- stat_just_seen = true;
- }
-
- if (ar->stat && !stat_just_seen)
- {
- gfc_error ("STAT= attribute in %C misplaced");
- return MATCH_ERROR;
- }
-
- if (gfc_match_char (']') == MATCH_YES)
- {
- ar->codimen++;
- if (ar->codimen < corank)
- {
- gfc_error ("Too few codimensions at %C, expected %d not %d",
- corank, ar->codimen);
- return MATCH_ERROR;
- }
- if (ar->codimen > corank)
- {
- gfc_error ("Too many codimensions at %C, expected %d not %d",
- corank, ar->codimen);
- return MATCH_ERROR;
- }
- return MATCH_YES;
- }
-
- if (gfc_match_char (',') != MATCH_YES)
- {
- if (gfc_match_char ('*') == MATCH_YES)
- gfc_error ("Unexpected %<*%> for codimension %d of %d at %C",
- ar->codimen + 1, corank);
- else
- gfc_error ("Invalid form of coarray reference at %C");
- return MATCH_ERROR;
- }
- else if (ar->dimen_type[ar->codimen + ar->dimen] == DIMEN_STAR)
- {
- gfc_error ("Unexpected %<*%> for codimension %d of %d at %C",
- ar->codimen + 1, corank);
- return MATCH_ERROR;
- }
-
- if (ar->codimen >= corank)
- {
- gfc_error ("Invalid codimension %d at %C, only %d codimensions exist",
- ar->codimen + 1, corank);
- return MATCH_ERROR;
- }
- }
-
- gfc_error ("Array reference at %C cannot have more than %d dimensions",
- GFC_MAX_DIMENSIONS);
- return MATCH_ERROR;
-
-}
-
-
-/************** Array specification matching subroutines ***************/
-
-/* Free all of the expressions associated with array bounds
- specifications. */
-
-void
-gfc_free_array_spec (gfc_array_spec *as)
-{
- int i;
-
- if (as == NULL)
- return;
-
- if (as->corank == 0)
- {
- for (i = 0; i < as->rank; i++)
- {
- gfc_free_expr (as->lower[i]);
- gfc_free_expr (as->upper[i]);
- }
- }
- else
- {
- int n = as->rank + as->corank - (as->cotype == AS_EXPLICIT ? 1 : 0);
- for (i = 0; i < n; i++)
- {
- gfc_free_expr (as->lower[i]);
- gfc_free_expr (as->upper[i]);
- }
- }
-
- free (as);
-}
-
-
-/* Take an array bound, resolves the expression, that make up the
- shape and check associated constraints. */
-
-static bool
-resolve_array_bound (gfc_expr *e, int check_constant)
-{
- if (e == NULL)
- return true;
-
- if (!gfc_resolve_expr (e)
- || !gfc_specification_expr (e))
- return false;
-
- if (check_constant && !gfc_is_constant_expr (e))
- {
- if (e->expr_type == EXPR_VARIABLE)
- gfc_error ("Variable %qs at %L in this context must be constant",
- e->symtree->n.sym->name, &e->where);
- else
- gfc_error ("Expression at %L in this context must be constant",
- &e->where);
- return false;
- }
-
- return true;
-}
-
-
-/* Takes an array specification, resolves the expressions that make up
- the shape and make sure everything is integral. */
-
-bool
-gfc_resolve_array_spec (gfc_array_spec *as, int check_constant)
-{
- gfc_expr *e;
- int i;
-
- if (as == NULL)
- return true;
-
- if (as->resolved)
- return true;
-
- for (i = 0; i < as->rank + as->corank; i++)
- {
- if (i == GFC_MAX_DIMENSIONS)
- return false;
-
- e = as->lower[i];
- if (!resolve_array_bound (e, check_constant))
- return false;
-
- e = as->upper[i];
- if (!resolve_array_bound (e, check_constant))
- return false;
-
- if ((as->lower[i] == NULL) || (as->upper[i] == NULL))
- continue;
-
- /* If the size is negative in this dimension, set it to zero. */
- if (as->lower[i]->expr_type == EXPR_CONSTANT
- && as->upper[i]->expr_type == EXPR_CONSTANT
- && mpz_cmp (as->upper[i]->value.integer,
- as->lower[i]->value.integer) < 0)
- {
- gfc_free_expr (as->upper[i]);
- as->upper[i] = gfc_copy_expr (as->lower[i]);
- mpz_sub_ui (as->upper[i]->value.integer,
- as->upper[i]->value.integer, 1);
- }
- }
-
- as->resolved = true;
-
- return true;
-}
-
-
-/* Match a single array element specification. The return values as
- well as the upper and lower bounds of the array spec are filled
- in according to what we see on the input. The caller makes sure
- individual specifications make sense as a whole.
-
-
- Parsed Lower Upper Returned
- ------------------------------------
- : NULL NULL AS_DEFERRED (*)
- x 1 x AS_EXPLICIT
- x: x NULL AS_ASSUMED_SHAPE
- x:y x y AS_EXPLICIT
- x:* x NULL AS_ASSUMED_SIZE
- * 1 NULL AS_ASSUMED_SIZE
-
- (*) For non-pointer dummy arrays this is AS_ASSUMED_SHAPE. This
- is fixed during the resolution of formal interfaces.
-
- Anything else AS_UNKNOWN. */
-
-static array_type
-match_array_element_spec (gfc_array_spec *as)
-{
- gfc_expr **upper, **lower;
- match m;
- int rank;
-
- rank = as->rank == -1 ? 0 : as->rank;
- lower = &as->lower[rank + as->corank - 1];
- upper = &as->upper[rank + as->corank - 1];
-
- if (gfc_match_char ('*') == MATCH_YES)
- {
- *lower = gfc_get_int_expr (gfc_default_integer_kind, NULL, 1);
- return AS_ASSUMED_SIZE;
- }
-
- if (gfc_match_char (':') == MATCH_YES)
- return AS_DEFERRED;
-
- m = gfc_match_expr (upper);
- if (m == MATCH_NO)
- gfc_error ("Expected expression in array specification at %C");
- if (m != MATCH_YES)
- return AS_UNKNOWN;
- if (!gfc_expr_check_typed (*upper, gfc_current_ns, false))
- return AS_UNKNOWN;
-
- gfc_try_simplify_expr (*upper, 0);
-
- if (((*upper)->expr_type == EXPR_CONSTANT
- && (*upper)->ts.type != BT_INTEGER) ||
- ((*upper)->expr_type == EXPR_FUNCTION
- && (*upper)->ts.type == BT_UNKNOWN
- && (*upper)->symtree
- && strcmp ((*upper)->symtree->name, "null") == 0))
- {
- gfc_error ("Expecting a scalar INTEGER expression at %C, found %s",
- gfc_basic_typename ((*upper)->ts.type));
- return AS_UNKNOWN;
- }
-
- if (gfc_match_char (':') == MATCH_NO)
- {
- *lower = gfc_get_int_expr (gfc_default_integer_kind, NULL, 1);
- return AS_EXPLICIT;
- }
-
- *lower = *upper;
- *upper = NULL;
-
- if (gfc_match_char ('*') == MATCH_YES)
- return AS_ASSUMED_SIZE;
-
- m = gfc_match_expr (upper);
- if (m == MATCH_ERROR)
- return AS_UNKNOWN;
- if (m == MATCH_NO)
- return AS_ASSUMED_SHAPE;
- if (!gfc_expr_check_typed (*upper, gfc_current_ns, false))
- return AS_UNKNOWN;
-
- gfc_try_simplify_expr (*upper, 0);
-
- if (((*upper)->expr_type == EXPR_CONSTANT
- && (*upper)->ts.type != BT_INTEGER) ||
- ((*upper)->expr_type == EXPR_FUNCTION
- && (*upper)->ts.type == BT_UNKNOWN
- && (*upper)->symtree
- && strcmp ((*upper)->symtree->name, "null") == 0))
- {
- gfc_error ("Expecting a scalar INTEGER expression at %C, found %s",
- gfc_basic_typename ((*upper)->ts.type));
- return AS_UNKNOWN;
- }
-
- return AS_EXPLICIT;
-}
-
-
-/* Matches an array specification, incidentally figuring out what sort
- it is. Match either a normal array specification, or a coarray spec
- or both. Optionally allow [:] for coarrays. */
-
-match
-gfc_match_array_spec (gfc_array_spec **asp, bool match_dim, bool match_codim)
-{
- array_type current_type;
- gfc_array_spec *as;
- int i;
-
- as = gfc_get_array_spec ();
-
- if (!match_dim)
- goto coarray;
-
- if (gfc_match_char ('(') != MATCH_YES)
- {
- if (!match_codim)
- goto done;
- goto coarray;
- }
-
- if (gfc_match (" .. )") == MATCH_YES)
- {
- as->type = AS_ASSUMED_RANK;
- as->rank = -1;
-
- if (!gfc_notify_std (GFC_STD_F2018, "Assumed-rank array at %C"))
- goto cleanup;
-
- if (!match_codim)
- goto done;
- goto coarray;
- }
-
- for (;;)
- {
- as->rank++;
- current_type = match_array_element_spec (as);
-
- /* Note that current_type == AS_ASSUMED_SIZE for both assumed-size
- and implied-shape specifications. If the rank is at least 2, we can
- distinguish between them. But for rank 1, we currently return
- ASSUMED_SIZE; this gets adjusted later when we know for sure
- whether the symbol parsed is a PARAMETER or not. */
-
- if (as->rank == 1)
- {
- if (current_type == AS_UNKNOWN)
- goto cleanup;
- as->type = current_type;
- }
- else
- switch (as->type)
- { /* See how current spec meshes with the existing. */
- case AS_UNKNOWN:
- goto cleanup;
-
- case AS_IMPLIED_SHAPE:
- if (current_type != AS_ASSUMED_SIZE)
- {
- gfc_error ("Bad array specification for implied-shape"
- " array at %C");
- goto cleanup;
- }
- break;
-
- case AS_EXPLICIT:
- if (current_type == AS_ASSUMED_SIZE)
- {
- as->type = AS_ASSUMED_SIZE;
- break;
- }
-
- if (current_type == AS_EXPLICIT)
- break;
-
- gfc_error ("Bad array specification for an explicitly shaped "
- "array at %C");
-
- goto cleanup;
-
- case AS_ASSUMED_SHAPE:
- if ((current_type == AS_ASSUMED_SHAPE)
- || (current_type == AS_DEFERRED))
- break;
-
- gfc_error ("Bad array specification for assumed shape "
- "array at %C");
- goto cleanup;
-
- case AS_DEFERRED:
- if (current_type == AS_DEFERRED)
- break;
-
- if (current_type == AS_ASSUMED_SHAPE)
- {
- as->type = AS_ASSUMED_SHAPE;
- break;
- }
-
- gfc_error ("Bad specification for deferred shape array at %C");
- goto cleanup;
-
- case AS_ASSUMED_SIZE:
- if (as->rank == 2 && current_type == AS_ASSUMED_SIZE)
- {
- as->type = AS_IMPLIED_SHAPE;
- break;
- }
-
- gfc_error ("Bad specification for assumed size array at %C");
- goto cleanup;
-
- case AS_ASSUMED_RANK:
- gcc_unreachable ();
- }
-
- if (gfc_match_char (')') == MATCH_YES)
- break;
-
- if (gfc_match_char (',') != MATCH_YES)
- {
- gfc_error ("Expected another dimension in array declaration at %C");
- goto cleanup;
- }
-
- if (as->rank + as->corank >= GFC_MAX_DIMENSIONS)
- {
- gfc_error ("Array specification at %C has more than %d dimensions",
- GFC_MAX_DIMENSIONS);
- goto cleanup;
- }
-
- if (as->corank + as->rank >= 7
- && !gfc_notify_std (GFC_STD_F2008, "Array specification at %C "
- "with more than 7 dimensions"))
- goto cleanup;
- }
-
- if (!match_codim)
- goto done;
-
-coarray:
- if (gfc_match_char ('[') != MATCH_YES)
- goto done;
-
- if (!gfc_notify_std (GFC_STD_F2008, "Coarray declaration at %C"))
- goto cleanup;
-
- if (flag_coarray == GFC_FCOARRAY_NONE)
- {
- gfc_fatal_error ("Coarrays disabled at %C, use %<-fcoarray=%> to enable");
- goto cleanup;
- }
-
- if (as->rank >= GFC_MAX_DIMENSIONS)
- {
- gfc_error ("Array specification at %C has more than %d "
- "dimensions", GFC_MAX_DIMENSIONS);
- goto cleanup;
- }
-
- for (;;)
- {
- as->corank++;
- current_type = match_array_element_spec (as);
-
- if (current_type == AS_UNKNOWN)
- goto cleanup;
-
- if (as->corank == 1)
- as->cotype = current_type;
- else
- switch (as->cotype)
- { /* See how current spec meshes with the existing. */
- case AS_IMPLIED_SHAPE:
- case AS_UNKNOWN:
- goto cleanup;
-
- case AS_EXPLICIT:
- if (current_type == AS_ASSUMED_SIZE)
- {
- as->cotype = AS_ASSUMED_SIZE;
- break;
- }
-
- if (current_type == AS_EXPLICIT)
- break;
-
- gfc_error ("Bad array specification for an explicitly "
- "shaped array at %C");
-
- goto cleanup;
-
- case AS_ASSUMED_SHAPE:
- if ((current_type == AS_ASSUMED_SHAPE)
- || (current_type == AS_DEFERRED))
- break;
-
- gfc_error ("Bad array specification for assumed shape "
- "array at %C");
- goto cleanup;
-
- case AS_DEFERRED:
- if (current_type == AS_DEFERRED)
- break;
-
- if (current_type == AS_ASSUMED_SHAPE)
- {
- as->cotype = AS_ASSUMED_SHAPE;
- break;
- }
-
- gfc_error ("Bad specification for deferred shape array at %C");
- goto cleanup;
-
- case AS_ASSUMED_SIZE:
- gfc_error ("Bad specification for assumed size array at %C");
- goto cleanup;
-
- case AS_ASSUMED_RANK:
- gcc_unreachable ();
- }
-
- if (gfc_match_char (']') == MATCH_YES)
- break;
-
- if (gfc_match_char (',') != MATCH_YES)
- {
- gfc_error ("Expected another dimension in array declaration at %C");
- goto cleanup;
- }
-
- if (as->rank + as->corank >= GFC_MAX_DIMENSIONS)
- {
- gfc_error ("Array specification at %C has more than %d "
- "dimensions", GFC_MAX_DIMENSIONS);
- goto cleanup;
- }
- }
-
- if (current_type == AS_EXPLICIT)
- {
- gfc_error ("Upper bound of last coarray dimension must be %<*%> at %C");
- goto cleanup;
- }
-
- if (as->cotype == AS_ASSUMED_SIZE)
- as->cotype = AS_EXPLICIT;
-
- if (as->rank == 0)
- as->type = as->cotype;
-
-done:
- if (as->rank == 0 && as->corank == 0)
- {
- *asp = NULL;
- gfc_free_array_spec (as);
- return MATCH_NO;
- }
-
- /* If a lower bounds of an assumed shape array is blank, put in one. */
- if (as->type == AS_ASSUMED_SHAPE)
- {
- for (i = 0; i < as->rank + as->corank; i++)
- {
- if (as->lower[i] == NULL)
- as->lower[i] = gfc_get_int_expr (gfc_default_integer_kind, NULL, 1);
- }
- }
-
- *asp = as;
-
- return MATCH_YES;
-
-cleanup:
- /* Something went wrong. */
- gfc_free_array_spec (as);
- return MATCH_ERROR;
-}
-
-/* Given a symbol and an array specification, modify the symbol to
- have that array specification. The error locus is needed in case
- something goes wrong. On failure, the caller must free the spec. */
-
-bool
-gfc_set_array_spec (gfc_symbol *sym, gfc_array_spec *as, locus *error_loc)
-{
- int i;
- symbol_attribute *attr;
-
- if (as == NULL)
- return true;
-
- /* If the symbol corresponds to a submodule module procedure the array spec is
- already set, so do not attempt to set it again here. */
- attr = &sym->attr;
- if (gfc_submodule_procedure(attr))
- return true;
-
- if (as->rank
- && !gfc_add_dimension (&sym->attr, sym->name, error_loc))
- return false;
-
- if (as->corank
- && !gfc_add_codimension (&sym->attr, sym->name, error_loc))
- return false;
-
- if (sym->as == NULL)
- {
- sym->as = as;
- return true;
- }
-
- if ((sym->as->type == AS_ASSUMED_RANK && as->corank)
- || (as->type == AS_ASSUMED_RANK && sym->as->corank))
- {
- gfc_error ("The assumed-rank array %qs at %L shall not have a "
- "codimension", sym->name, error_loc);
- return false;
- }
-
- /* Check F2018:C822. */
- if (sym->as->rank + sym->as->corank > GFC_MAX_DIMENSIONS)
- goto too_many;
-
- if (as->corank)
- {
- sym->as->cotype = as->cotype;
- sym->as->corank = as->corank;
- /* Check F2018:C822. */
- if (sym->as->rank + sym->as->corank > GFC_MAX_DIMENSIONS)
- goto too_many;
-
- for (i = 0; i < as->corank; i++)
- {
- sym->as->lower[sym->as->rank + i] = as->lower[i];
- sym->as->upper[sym->as->rank + i] = as->upper[i];
- }
- }
- else
- {
- /* The "sym" has no rank (checked via gfc_add_dimension). Thus
- the dimension is added - but first the codimensions (if existing
- need to be shifted to make space for the dimension. */
- gcc_assert (as->corank == 0 && sym->as->rank == 0);
-
- sym->as->rank = as->rank;
- sym->as->type = as->type;
- sym->as->cray_pointee = as->cray_pointee;
- sym->as->cp_was_assumed = as->cp_was_assumed;
-
- /* Check F2018:C822. */
- if (sym->as->rank + sym->as->corank > GFC_MAX_DIMENSIONS)
- goto too_many;
-
- for (i = sym->as->corank - 1; i >= 0; i--)
- {
- sym->as->lower[as->rank + i] = sym->as->lower[i];
- sym->as->upper[as->rank + i] = sym->as->upper[i];
- }
- for (i = 0; i < as->rank; i++)
- {
- sym->as->lower[i] = as->lower[i];
- sym->as->upper[i] = as->upper[i];
- }
- }
-
- free (as);
- return true;
-
-too_many:
-
- gfc_error ("rank + corank of %qs exceeds %d at %C", sym->name,
- GFC_MAX_DIMENSIONS);
- return false;
-}
-
-
-/* Copy an array specification. */
-
-gfc_array_spec *
-gfc_copy_array_spec (gfc_array_spec *src)
-{
- gfc_array_spec *dest;
- int i;
-
- if (src == NULL)
- return NULL;
-
- dest = gfc_get_array_spec ();
-
- *dest = *src;
-
- for (i = 0; i < dest->rank + dest->corank; i++)
- {
- dest->lower[i] = gfc_copy_expr (dest->lower[i]);
- dest->upper[i] = gfc_copy_expr (dest->upper[i]);
- }
-
- return dest;
-}
-
-
-/* Returns nonzero if the two expressions are equal. Only handles integer
- constants. */
-
-static int
-compare_bounds (gfc_expr *bound1, gfc_expr *bound2)
-{
- if (bound1 == NULL || bound2 == NULL
- || bound1->expr_type != EXPR_CONSTANT
- || bound2->expr_type != EXPR_CONSTANT
- || bound1->ts.type != BT_INTEGER
- || bound2->ts.type != BT_INTEGER)
- gfc_internal_error ("gfc_compare_array_spec(): Array spec clobbered");
-
- if (mpz_cmp (bound1->value.integer, bound2->value.integer) == 0)
- return 1;
- else
- return 0;
-}
-
-
-/* Compares two array specifications. They must be constant or deferred
- shape. */
-
-int
-gfc_compare_array_spec (gfc_array_spec *as1, gfc_array_spec *as2)
-{
- int i;
-
- if (as1 == NULL && as2 == NULL)
- return 1;
-
- if (as1 == NULL || as2 == NULL)
- return 0;
-
- if (as1->rank != as2->rank)
- return 0;
-
- if (as1->corank != as2->corank)
- return 0;
-
- if (as1->rank == 0)
- return 1;
-
- if (as1->type != as2->type)
- return 0;
-
- if (as1->type == AS_EXPLICIT)
- for (i = 0; i < as1->rank + as1->corank; i++)
- {
- if (compare_bounds (as1->lower[i], as2->lower[i]) == 0)
- return 0;
-
- if (compare_bounds (as1->upper[i], as2->upper[i]) == 0)
- return 0;
- }
-
- return 1;
-}
-
-
-/****************** Array constructor functions ******************/
-
-
-/* Given an expression node that might be an array constructor and a
- symbol, make sure that no iterators in this or child constructors
- use the symbol as an implied-DO iterator. Returns nonzero if a
- duplicate was found. */
-
-static int
-check_duplicate_iterator (gfc_constructor_base base, gfc_symbol *master)
-{
- gfc_constructor *c;
- gfc_expr *e;
-
- for (c = gfc_constructor_first (base); c; c = gfc_constructor_next (c))
- {
- e = c->expr;
-
- if (e->expr_type == EXPR_ARRAY
- && check_duplicate_iterator (e->value.constructor, master))
- return 1;
-
- if (c->iterator == NULL)
- continue;
-
- if (c->iterator->var->symtree->n.sym == master)
- {
- gfc_error ("DO-iterator %qs at %L is inside iterator of the "
- "same name", master->name, &c->where);
-
- return 1;
- }
- }
-
- return 0;
-}
-
-
-/* Forward declaration because these functions are mutually recursive. */
-static match match_array_cons_element (gfc_constructor_base *);
-
-/* Match a list of array elements. */
-
-static match
-match_array_list (gfc_constructor_base *result)
-{
- gfc_constructor_base head;
- gfc_constructor *p;
- gfc_iterator iter;
- locus old_loc;
- gfc_expr *e;
- match m;
- int n;
-
- old_loc = gfc_current_locus;
-
- if (gfc_match_char ('(') == MATCH_NO)
- return MATCH_NO;
-
- memset (&iter, '\0', sizeof (gfc_iterator));
- head = NULL;
-
- m = match_array_cons_element (&head);
- if (m != MATCH_YES)
- goto cleanup;
-
- if (gfc_match_char (',') != MATCH_YES)
- {
- m = MATCH_NO;
- goto cleanup;
- }
-
- for (n = 1;; n++)
- {
- m = gfc_match_iterator (&iter, 0);
- if (m == MATCH_YES)
- break;
- if (m == MATCH_ERROR)
- goto cleanup;
-
- m = match_array_cons_element (&head);
- if (m == MATCH_ERROR)
- goto cleanup;
- if (m == MATCH_NO)
- {
- if (n > 2)
- goto syntax;
- m = MATCH_NO;
- goto cleanup; /* Could be a complex constant */
- }
-
- if (gfc_match_char (',') != MATCH_YES)
- {
- if (n > 2)
- goto syntax;
- m = MATCH_NO;
- goto cleanup;
- }
- }
-
- if (gfc_match_char (')') != MATCH_YES)
- goto syntax;
-
- if (check_duplicate_iterator (head, iter.var->symtree->n.sym))
- {
- m = MATCH_ERROR;
- goto cleanup;
- }
-
- e = gfc_get_array_expr (BT_UNKNOWN, 0, &old_loc);
- e->value.constructor = head;
-
- p = gfc_constructor_append_expr (result, e, &gfc_current_locus);
- p->iterator = gfc_get_iterator ();
- *p->iterator = iter;
-
- return MATCH_YES;
-
-syntax:
- gfc_error ("Syntax error in array constructor at %C");
- m = MATCH_ERROR;
-
-cleanup:
- gfc_constructor_free (head);
- gfc_free_iterator (&iter, 0);
- gfc_current_locus = old_loc;
- return m;
-}
-
-
-/* Match a single element of an array constructor, which can be a
- single expression or a list of elements. */
-
-static match
-match_array_cons_element (gfc_constructor_base *result)
-{
- gfc_expr *expr;
- match m;
-
- m = match_array_list (result);
- if (m != MATCH_NO)
- return m;
-
- m = gfc_match_expr (&expr);
- if (m != MATCH_YES)
- return m;
-
- if (expr->ts.type == BT_BOZ)
- {
- gfc_error ("BOZ literal constant at %L cannot appear in an "
- "array constructor", &expr->where);
- goto done;
- }
-
- if (expr->expr_type == EXPR_FUNCTION
- && expr->ts.type == BT_UNKNOWN
- && strcmp(expr->symtree->name, "null") == 0)
- {
- gfc_error ("NULL() at %C cannot appear in an array constructor");
- goto done;
- }
-
- gfc_constructor_append_expr (result, expr, &gfc_current_locus);
- return MATCH_YES;
-
-done:
- gfc_free_expr (expr);
- return MATCH_ERROR;
-}
-
-
-/* Convert components of an array constructor to the type in ts. */
-
-static match
-walk_array_constructor (gfc_typespec *ts, gfc_constructor_base head)
-{
- gfc_constructor *c;
- gfc_expr *e;
- match m;
-
- for (c = gfc_constructor_first (head); c; c = gfc_constructor_next (c))
- {
- e = c->expr;
- if (e->expr_type == EXPR_ARRAY && e->ts.type == BT_UNKNOWN
- && !e->ref && e->value.constructor)
- {
- m = walk_array_constructor (ts, e->value.constructor);
- if (m == MATCH_ERROR)
- return m;
- }
- else if (!gfc_convert_type_warn (e, ts, 1, 1, true)
- && e->ts.type != BT_UNKNOWN)
- return MATCH_ERROR;
- }
- return MATCH_YES;
-}
-
-/* Match an array constructor. */
-
-match
-gfc_match_array_constructor (gfc_expr **result)
-{
- gfc_constructor *c;
- gfc_constructor_base head;
- gfc_expr *expr;
- gfc_typespec ts;
- locus where;
- match m;
- const char *end_delim;
- bool seen_ts;
-
- head = NULL;
- seen_ts = false;
-
- if (gfc_match (" (/") == MATCH_NO)
- {
- if (gfc_match (" [") == MATCH_NO)
- return MATCH_NO;
- else
- {
- if (!gfc_notify_std (GFC_STD_F2003, "[...] "
- "style array constructors at %C"))
- return MATCH_ERROR;
- end_delim = " ]";
- }
- }
- else
- end_delim = " /)";
-
- where = gfc_current_locus;
-
- /* Try to match an optional "type-spec ::" */
- gfc_clear_ts (&ts);
- m = gfc_match_type_spec (&ts);
- if (m == MATCH_YES)
- {
- seen_ts = (gfc_match (" ::") == MATCH_YES);
-
- if (seen_ts)
- {
- if (!gfc_notify_std (GFC_STD_F2003, "Array constructor "
- "including type specification at %C"))
- goto cleanup;
-
- if (ts.deferred)
- {
- gfc_error ("Type-spec at %L cannot contain a deferred "
- "type parameter", &where);
- goto cleanup;
- }
-
- if (ts.type == BT_CHARACTER
- && ts.u.cl && !ts.u.cl->length && !ts.u.cl->length_from_typespec)
- {
- gfc_error ("Type-spec at %L cannot contain an asterisk for a "
- "type parameter", &where);
- goto cleanup;
- }
- }
- }
- else if (m == MATCH_ERROR)
- goto cleanup;
-
- if (!seen_ts)
- gfc_current_locus = where;
-
- if (gfc_match (end_delim) == MATCH_YES)
- {
- if (seen_ts)
- goto done;
- else
- {
- gfc_error ("Empty array constructor at %C is not allowed");
- goto cleanup;
- }
- }
-
- for (;;)
- {
- m = match_array_cons_element (&head);
- if (m == MATCH_ERROR)
- goto cleanup;
- if (m == MATCH_NO)
- goto syntax;
-
- if (gfc_match_char (',') == MATCH_NO)
- break;
- }
-
- if (gfc_match (end_delim) == MATCH_NO)
- goto syntax;
-
-done:
- /* Size must be calculated at resolution time. */
- if (seen_ts)
- {
- expr = gfc_get_array_expr (ts.type, ts.kind, &where);
- expr->ts = ts;
-
- /* If the typespec is CHARACTER, check that array elements can
- be converted. See PR fortran/67803. */
- if (ts.type == BT_CHARACTER)
- {
- c = gfc_constructor_first (head);
- for (; c; c = gfc_constructor_next (c))
- {
- if (gfc_numeric_ts (&c->expr->ts)
- || c->expr->ts.type == BT_LOGICAL)
- {
- gfc_error ("Incompatible typespec for array element at %L",
- &c->expr->where);
- return MATCH_ERROR;
- }
-
- /* Special case null(). */
- if (c->expr->expr_type == EXPR_FUNCTION
- && c->expr->ts.type == BT_UNKNOWN
- && strcmp (c->expr->symtree->name, "null") == 0)
- {
- gfc_error ("Incompatible typespec for array element at %L",
- &c->expr->where);
- return MATCH_ERROR;
- }
- }
- }
-
- /* Walk the constructor, and if possible, do type conversion for
- numeric types. */
- if (gfc_numeric_ts (&ts))
- {
- m = walk_array_constructor (&ts, head);
- if (m == MATCH_ERROR)
- return m;
- }
- }
- else
- expr = gfc_get_array_expr (BT_UNKNOWN, 0, &where);
-
- expr->value.constructor = head;
- if (expr->ts.u.cl)
- expr->ts.u.cl->length_from_typespec = seen_ts;
-
- *result = expr;
-
- return MATCH_YES;
-
-syntax:
- gfc_error ("Syntax error in array constructor at %C");
-
-cleanup:
- gfc_constructor_free (head);
- return MATCH_ERROR;
-}
-
-
-
-/************** Check array constructors for correctness **************/
-
-/* Given an expression, compare it's type with the type of the current
- constructor. Returns nonzero if an error was issued. The
- cons_state variable keeps track of whether the type of the
- constructor being read or resolved is known to be good, bad or just
- starting out. */
-
-static gfc_typespec constructor_ts;
-static enum
-{ CONS_START, CONS_GOOD, CONS_BAD }
-cons_state;
-
-static int
-check_element_type (gfc_expr *expr, bool convert)
-{
- if (cons_state == CONS_BAD)
- return 0; /* Suppress further errors */
-
- if (cons_state == CONS_START)
- {
- if (expr->ts.type == BT_UNKNOWN)
- cons_state = CONS_BAD;
- else
- {
- cons_state = CONS_GOOD;
- constructor_ts = expr->ts;
- }
-
- return 0;
- }
-
- if (gfc_compare_types (&constructor_ts, &expr->ts))
- return 0;
-
- if (convert)
- return gfc_convert_type_warn (expr, &constructor_ts, 1, 1, true) ? 0 : 1;
-
- gfc_error ("Element in %s array constructor at %L is %s",
- gfc_typename (&constructor_ts), &expr->where,
- gfc_typename (expr));
-
- cons_state = CONS_BAD;
- return 1;
-}
-
-
-/* Recursive work function for gfc_check_constructor_type(). */
-
-static bool
-check_constructor_type (gfc_constructor_base base, bool convert)
-{
- gfc_constructor *c;
- gfc_expr *e;
-
- for (c = gfc_constructor_first (base); c; c = gfc_constructor_next (c))
- {
- e = c->expr;
-
- if (e->expr_type == EXPR_ARRAY)
- {
- if (!check_constructor_type (e->value.constructor, convert))
- return false;
-
- continue;
- }
-
- if (check_element_type (e, convert))
- return false;
- }
-
- return true;
-}
-
-
-/* Check that all elements of an array constructor are the same type.
- On false, an error has been generated. */
-
-bool
-gfc_check_constructor_type (gfc_expr *e)
-{
- bool t;
-
- if (e->ts.type != BT_UNKNOWN)
- {
- cons_state = CONS_GOOD;
- constructor_ts = e->ts;
- }
- else
- {
- cons_state = CONS_START;
- gfc_clear_ts (&constructor_ts);
- }
-
- /* If e->ts.type != BT_UNKNOWN, the array constructor included a
- typespec, and we will now convert the values on the fly. */
- t = check_constructor_type (e->value.constructor, e->ts.type != BT_UNKNOWN);
- if (t && e->ts.type == BT_UNKNOWN)
- e->ts = constructor_ts;
-
- return t;
-}
-
-
-
-typedef struct cons_stack
-{
- gfc_iterator *iterator;
- struct cons_stack *previous;
-}
-cons_stack;
-
-static cons_stack *base;
-
-static bool check_constructor (gfc_constructor_base, bool (*) (gfc_expr *));
-
-/* Check an EXPR_VARIABLE expression in a constructor to make sure
- that that variable is an iteration variable. */
-
-bool
-gfc_check_iter_variable (gfc_expr *expr)
-{
- gfc_symbol *sym;
- cons_stack *c;
-
- sym = expr->symtree->n.sym;
-
- for (c = base; c && c->iterator; c = c->previous)
- if (sym == c->iterator->var->symtree->n.sym)
- return true;
-
- return false;
-}
-
-
-/* Recursive work function for gfc_check_constructor(). This amounts
- to calling the check function for each expression in the
- constructor, giving variables with the names of iterators a pass. */
-
-static bool
-check_constructor (gfc_constructor_base ctor, bool (*check_function) (gfc_expr *))
-{
- cons_stack element;
- gfc_expr *e;
- bool t;
- gfc_constructor *c;
-
- for (c = gfc_constructor_first (ctor); c; c = gfc_constructor_next (c))
- {
- e = c->expr;
-
- if (!e)
- continue;
-
- if (e->expr_type != EXPR_ARRAY)
- {
- if (!(*check_function)(e))
- return false;
- continue;
- }
-
- element.previous = base;
- element.iterator = c->iterator;
-
- base = &element;
- t = check_constructor (e->value.constructor, check_function);
- base = element.previous;
-
- if (!t)
- return false;
- }
-
- /* Nothing went wrong, so all OK. */
- return true;
-}
-
-
-/* Checks a constructor to see if it is a particular kind of
- expression -- specification, restricted, or initialization as
- determined by the check_function. */
-
-bool
-gfc_check_constructor (gfc_expr *expr, bool (*check_function) (gfc_expr *))
-{
- cons_stack *base_save;
- bool t;
-
- base_save = base;
- base = NULL;
-
- t = check_constructor (expr->value.constructor, check_function);
- base = base_save;
-
- return t;
-}
-
-
-
-/**************** Simplification of array constructors ****************/
-
-iterator_stack *iter_stack;
-
-typedef struct
-{
- gfc_constructor_base base;
- int extract_count, extract_n;
- gfc_expr *extracted;
- mpz_t *count;
-
- mpz_t *offset;
- gfc_component *component;
- mpz_t *repeat;
-
- bool (*expand_work_function) (gfc_expr *);
-}
-expand_info;
-
-static expand_info current_expand;
-
-static bool expand_constructor (gfc_constructor_base);
-
-
-/* Work function that counts the number of elements present in a
- constructor. */
-
-static bool
-count_elements (gfc_expr *e)
-{
- mpz_t result;
-
- if (e->rank == 0)
- mpz_add_ui (*current_expand.count, *current_expand.count, 1);
- else
- {
- if (!gfc_array_size (e, &result))
- {
- gfc_free_expr (e);
- return false;
- }
-
- mpz_add (*current_expand.count, *current_expand.count, result);
- mpz_clear (result);
- }
-
- gfc_free_expr (e);
- return true;
-}
-
-
-/* Work function that extracts a particular element from an array
- constructor, freeing the rest. */
-
-static bool
-extract_element (gfc_expr *e)
-{
- if (e->rank != 0)
- { /* Something unextractable */
- gfc_free_expr (e);
- return false;
- }
-
- if (current_expand.extract_count == current_expand.extract_n)
- current_expand.extracted = e;
- else
- gfc_free_expr (e);
-
- current_expand.extract_count++;
-
- return true;
-}
-
-
-/* Work function that constructs a new constructor out of the old one,
- stringing new elements together. */
-
-static bool
-expand (gfc_expr *e)
-{
- gfc_constructor *c = gfc_constructor_append_expr (&current_expand.base,
- e, &e->where);
-
- c->n.component = current_expand.component;
- return true;
-}
-
-
-/* Given an initialization expression that is a variable reference,
- substitute the current value of the iteration variable. */
-
-void
-gfc_simplify_iterator_var (gfc_expr *e)
-{
- iterator_stack *p;
-
- for (p = iter_stack; p; p = p->prev)
- if (e->symtree == p->variable)
- break;
-
- if (p == NULL)
- return; /* Variable not found */
-
- gfc_replace_expr (e, gfc_get_int_expr (gfc_default_integer_kind, NULL, 0));
-
- mpz_set (e->value.integer, p->value);
-
- return;
-}
-
-
-/* Expand an expression with that is inside of a constructor,
- recursing into other constructors if present. */
-
-static bool
-expand_expr (gfc_expr *e)
-{
- if (e->expr_type == EXPR_ARRAY)
- return expand_constructor (e->value.constructor);
-
- e = gfc_copy_expr (e);
-
- if (!gfc_simplify_expr (e, 1))
- {
- gfc_free_expr (e);
- return false;
- }
-
- return current_expand.expand_work_function (e);
-}
-
-
-static bool
-expand_iterator (gfc_constructor *c)
-{
- gfc_expr *start, *end, *step;
- iterator_stack frame;
- mpz_t trip;
- bool t;
-
- end = step = NULL;
-
- t = false;
-
- mpz_init (trip);
- mpz_init (frame.value);
- frame.prev = NULL;
-
- start = gfc_copy_expr (c->iterator->start);
- if (!gfc_simplify_expr (start, 1))
- goto cleanup;
-
- if (start->expr_type != EXPR_CONSTANT || start->ts.type != BT_INTEGER)
- goto cleanup;
-
- end = gfc_copy_expr (c->iterator->end);
- if (!gfc_simplify_expr (end, 1))
- goto cleanup;
-
- if (end->expr_type != EXPR_CONSTANT || end->ts.type != BT_INTEGER)
- goto cleanup;
-
- step = gfc_copy_expr (c->iterator->step);
- if (!gfc_simplify_expr (step, 1))
- goto cleanup;
-
- if (step->expr_type != EXPR_CONSTANT || step->ts.type != BT_INTEGER)
- goto cleanup;
-
- if (mpz_sgn (step->value.integer) == 0)
- {
- gfc_error ("Iterator step at %L cannot be zero", &step->where);
- goto cleanup;
- }
-
- /* Calculate the trip count of the loop. */
- mpz_sub (trip, end->value.integer, start->value.integer);
- mpz_add (trip, trip, step->value.integer);
- mpz_tdiv_q (trip, trip, step->value.integer);
-
- mpz_set (frame.value, start->value.integer);
-
- frame.prev = iter_stack;
- frame.variable = c->iterator->var->symtree;
- iter_stack = &frame;
-
- while (mpz_sgn (trip) > 0)
- {
- if (!expand_expr (c->expr))
- goto cleanup;
-
- mpz_add (frame.value, frame.value, step->value.integer);
- mpz_sub_ui (trip, trip, 1);
- }
-
- t = true;
-
-cleanup:
- gfc_free_expr (start);
- gfc_free_expr (end);
- gfc_free_expr (step);
-
- mpz_clear (trip);
- mpz_clear (frame.value);
-
- iter_stack = frame.prev;
-
- return t;
-}
-
-/* Variables for noticing if all constructors are empty, and
- if any of them had a type. */
-
-static bool empty_constructor;
-static gfc_typespec empty_ts;
-
-/* Expand a constructor into constant constructors without any
- iterators, calling the work function for each of the expanded
- expressions. The work function needs to either save or free the
- passed expression. */
-
-static bool
-expand_constructor (gfc_constructor_base base)
-{
- gfc_constructor *c;
- gfc_expr *e;
-
- for (c = gfc_constructor_first (base); c; c = gfc_constructor_next(c))
- {
- if (c->iterator != NULL)
- {
- if (!expand_iterator (c))
- return false;
- continue;
- }
-
- e = c->expr;
-
- if (e == NULL)
- return false;
-
- if (empty_constructor)
- empty_ts = e->ts;
-
- /* Simplify constant array expression/section within constructor. */
- if (e->expr_type == EXPR_VARIABLE && e->rank > 0 && e->ref
- && e->symtree && e->symtree->n.sym
- && e->symtree->n.sym->attr.flavor == FL_PARAMETER)
- gfc_simplify_expr (e, 0);
-
- if (e->expr_type == EXPR_ARRAY)
- {
- if (!expand_constructor (e->value.constructor))
- return false;
-
- continue;
- }
-
- empty_constructor = false;
- e = gfc_copy_expr (e);
- if (!gfc_simplify_expr (e, 1))
- {
- gfc_free_expr (e);
- return false;
- }
- e->from_constructor = 1;
- current_expand.offset = &c->offset;
- current_expand.repeat = &c->repeat;
- current_expand.component = c->n.component;
- if (!current_expand.expand_work_function(e))
- return false;
- }
- return true;
-}
-
-
-/* Given an array expression and an element number (starting at zero),
- return a pointer to the array element. NULL is returned if the
- size of the array has been exceeded. The expression node returned
- remains a part of the array and should not be freed. Access is not
- efficient at all, but this is another place where things do not
- have to be particularly fast. */
-
-static gfc_expr *
-gfc_get_array_element (gfc_expr *array, int element)
-{
- expand_info expand_save;
- gfc_expr *e;
- bool rc;
-
- expand_save = current_expand;
- current_expand.extract_n = element;
- current_expand.expand_work_function = extract_element;
- current_expand.extracted = NULL;
- current_expand.extract_count = 0;
-
- iter_stack = NULL;
-
- rc = expand_constructor (array->value.constructor);
- e = current_expand.extracted;
- current_expand = expand_save;
-
- if (!rc)
- return NULL;
-
- return e;
-}
-
-
-/* Top level subroutine for expanding constructors. We only expand
- constructor if they are small enough. */
-
-bool
-gfc_expand_constructor (gfc_expr *e, bool fatal)
-{
- expand_info expand_save;
- gfc_expr *f;
- bool rc;
-
- /* If we can successfully get an array element at the max array size then
- the array is too big to expand, so we just return. */
- f = gfc_get_array_element (e, flag_max_array_constructor);
- if (f != NULL)
- {
- gfc_free_expr (f);
- if (fatal)
- {
- gfc_error ("The number of elements in the array constructor "
- "at %L requires an increase of the allowed %d "
- "upper limit. See %<-fmax-array-constructor%> "
- "option", &e->where, flag_max_array_constructor);
- return false;
- }
- return true;
- }
-
- /* We now know the array is not too big so go ahead and try to expand it. */
- expand_save = current_expand;
- current_expand.base = NULL;
-
- iter_stack = NULL;
-
- empty_constructor = true;
- gfc_clear_ts (&empty_ts);
- current_expand.expand_work_function = expand;
-
- if (!expand_constructor (e->value.constructor))
- {
- gfc_constructor_free (current_expand.base);
- rc = false;
- goto done;
- }
-
- /* If we don't have an explicit constructor type, and there
- were only empty constructors, then take the type from
- them. */
-
- if (constructor_ts.type == BT_UNKNOWN && empty_constructor)
- e->ts = empty_ts;
-
- gfc_constructor_free (e->value.constructor);
- e->value.constructor = current_expand.base;
-
- rc = true;
-
-done:
- current_expand = expand_save;
-
- return rc;
-}
-
-
-/* Work function for checking that an element of a constructor is a
- constant, after removal of any iteration variables. We return
- false if not so. */
-
-static bool
-is_constant_element (gfc_expr *e)
-{
- int rv;
-
- rv = gfc_is_constant_expr (e);
- gfc_free_expr (e);
-
- return rv ? true : false;
-}
-
-
-/* Given an array constructor, determine if the constructor is
- constant or not by expanding it and making sure that all elements
- are constants. This is a bit of a hack since something like (/ (i,
- i=1,100000000) /) will take a while as* opposed to a more clever
- function that traverses the expression tree. FIXME. */
-
-int
-gfc_constant_ac (gfc_expr *e)
-{
- expand_info expand_save;
- bool rc;
-
- iter_stack = NULL;
- expand_save = current_expand;
- current_expand.expand_work_function = is_constant_element;
-
- rc = expand_constructor (e->value.constructor);
-
- current_expand = expand_save;
- if (!rc)
- return 0;
-
- return 1;
-}
-
-
-/* Returns nonzero if an array constructor has been completely
- expanded (no iterators) and zero if iterators are present. */
-
-int
-gfc_expanded_ac (gfc_expr *e)
-{
- gfc_constructor *c;
-
- if (e->expr_type == EXPR_ARRAY)
- for (c = gfc_constructor_first (e->value.constructor);
- c; c = gfc_constructor_next (c))
- if (c->iterator != NULL || !gfc_expanded_ac (c->expr))
- return 0;
-
- return 1;
-}
-
-
-/*************** Type resolution of array constructors ***************/
-
-
-/* The symbol expr_is_sought_symbol_ref will try to find. */
-static const gfc_symbol *sought_symbol = NULL;
-
-
-/* Tells whether the expression E is a variable reference to the symbol
- in the static variable SOUGHT_SYMBOL, and sets the locus pointer WHERE
- accordingly.
- To be used with gfc_expr_walker: if a reference is found we don't need
- to look further so we return 1 to skip any further walk. */
-
-static int
-expr_is_sought_symbol_ref (gfc_expr **e, int *walk_subtrees ATTRIBUTE_UNUSED,
- void *where)
-{
- gfc_expr *expr = *e;
- locus *sym_loc = (locus *)where;
-
- if (expr->expr_type == EXPR_VARIABLE
- && expr->symtree->n.sym == sought_symbol)
- {
- *sym_loc = expr->where;
- return 1;
- }
-
- return 0;
-}
-
-
-/* Tells whether the expression EXPR contains a reference to the symbol
- SYM and in that case sets the position SYM_LOC where the reference is. */
-
-static bool
-find_symbol_in_expr (gfc_symbol *sym, gfc_expr *expr, locus *sym_loc)
-{
- int ret;
-
- sought_symbol = sym;
- ret = gfc_expr_walker (&expr, &expr_is_sought_symbol_ref, sym_loc);
- sought_symbol = NULL;
- return ret;
-}
-
-
-/* Recursive array list resolution function. All of the elements must
- be of the same type. */
-
-static bool
-resolve_array_list (gfc_constructor_base base)
-{
- bool t;
- gfc_constructor *c;
- gfc_iterator *iter;
-
- t = true;
-
- for (c = gfc_constructor_first (base); c; c = gfc_constructor_next (c))
- {
- iter = c->iterator;
- if (iter != NULL)
- {
- gfc_symbol *iter_var;
- locus iter_var_loc;
-
- if (!gfc_resolve_iterator (iter, false, true))
- t = false;
-
- /* Check for bounds referencing the iterator variable. */
- gcc_assert (iter->var->expr_type == EXPR_VARIABLE);
- iter_var = iter->var->symtree->n.sym;
- if (find_symbol_in_expr (iter_var, iter->start, &iter_var_loc))
- {
- if (!gfc_notify_std (GFC_STD_LEGACY, "AC-IMPLIED-DO initial "
- "expression references control variable "
- "at %L", &iter_var_loc))
- t = false;
- }
- if (find_symbol_in_expr (iter_var, iter->end, &iter_var_loc))
- {
- if (!gfc_notify_std (GFC_STD_LEGACY, "AC-IMPLIED-DO final "
- "expression references control variable "
- "at %L", &iter_var_loc))
- t = false;
- }
- if (find_symbol_in_expr (iter_var, iter->step, &iter_var_loc))
- {
- if (!gfc_notify_std (GFC_STD_LEGACY, "AC-IMPLIED-DO step "
- "expression references control variable "
- "at %L", &iter_var_loc))
- t = false;
- }
- }
-
- if (!gfc_resolve_expr (c->expr))
- t = false;
-
- if (UNLIMITED_POLY (c->expr))
- {
- gfc_error ("Array constructor value at %L shall not be unlimited "
- "polymorphic [F2008: C4106]", &c->expr->where);
- t = false;
- }
- }
-
- return t;
-}
-
-/* Resolve character array constructor. If it has a specified constant character
- length, pad/truncate the elements here; if the length is not specified and
- all elements are of compile-time known length, emit an error as this is
- invalid. */
-
-bool
-gfc_resolve_character_array_constructor (gfc_expr *expr)
-{
- gfc_constructor *p;
- HOST_WIDE_INT found_length;
-
- gcc_assert (expr->expr_type == EXPR_ARRAY);
- gcc_assert (expr->ts.type == BT_CHARACTER);
-
- if (expr->ts.u.cl == NULL)
- {
- for (p = gfc_constructor_first (expr->value.constructor);
- p; p = gfc_constructor_next (p))
- if (p->expr->ts.u.cl != NULL)
- {
- /* Ensure that if there is a char_len around that it is
- used; otherwise the middle-end confuses them! */
- expr->ts.u.cl = p->expr->ts.u.cl;
- goto got_charlen;
- }
-
- expr->ts.u.cl = gfc_new_charlen (gfc_current_ns, NULL);
- }
-
-got_charlen:
-
- /* Early exit for zero size arrays. */
- if (expr->shape)
- {
- mpz_t size;
- HOST_WIDE_INT arraysize;
-
- gfc_array_size (expr, &size);
- arraysize = mpz_get_ui (size);
- mpz_clear (size);
-
- if (arraysize == 0)
- return true;
- }
-
- found_length = -1;
-
- if (expr->ts.u.cl->length == NULL)
- {
- /* Check that all constant string elements have the same length until
- we reach the end or find a variable-length one. */
-
- for (p = gfc_constructor_first (expr->value.constructor);
- p; p = gfc_constructor_next (p))
- {
- HOST_WIDE_INT current_length = -1;
- gfc_ref *ref;
- for (ref = p->expr->ref; ref; ref = ref->next)
- if (ref->type == REF_SUBSTRING
- && ref->u.ss.start
- && ref->u.ss.start->expr_type == EXPR_CONSTANT
- && ref->u.ss.end
- && ref->u.ss.end->expr_type == EXPR_CONSTANT)
- break;
-
- if (p->expr->expr_type == EXPR_CONSTANT)
- current_length = p->expr->value.character.length;
- else if (ref)
- current_length = gfc_mpz_get_hwi (ref->u.ss.end->value.integer)
- - gfc_mpz_get_hwi (ref->u.ss.start->value.integer) + 1;
- else if (p->expr->ts.u.cl && p->expr->ts.u.cl->length
- && p->expr->ts.u.cl->length->expr_type == EXPR_CONSTANT)
- current_length = gfc_mpz_get_hwi (p->expr->ts.u.cl->length->value.integer);
- else
- return true;
-
- if (current_length < 0)
- current_length = 0;
-
- if (found_length == -1)
- found_length = current_length;
- else if (found_length != current_length)
- {
- gfc_error ("Different CHARACTER lengths (%ld/%ld) in array"
- " constructor at %L", (long) found_length,
- (long) current_length, &p->expr->where);
- return false;
- }
-
- gcc_assert (found_length == current_length);
- }
-
- gcc_assert (found_length != -1);
-
- /* Update the character length of the array constructor. */
- expr->ts.u.cl->length = gfc_get_int_expr (gfc_charlen_int_kind,
- NULL, found_length);
- }
- else
- {
- /* We've got a character length specified. It should be an integer,
- otherwise an error is signalled elsewhere. */
- gcc_assert (expr->ts.u.cl->length);
-
- /* If we've got a constant character length, pad according to this.
- gfc_extract_int does check for BT_INTEGER and EXPR_CONSTANT and sets
- max_length only if they pass. */
- gfc_extract_hwi (expr->ts.u.cl->length, &found_length);
-
- /* Now pad/truncate the elements accordingly to the specified character
- length. This is ok inside this conditional, as in the case above
- (without typespec) all elements are verified to have the same length
- anyway. */
- if (found_length != -1)
- for (p = gfc_constructor_first (expr->value.constructor);
- p; p = gfc_constructor_next (p))
- if (p->expr->expr_type == EXPR_CONSTANT)
- {
- gfc_expr *cl = NULL;
- HOST_WIDE_INT current_length = -1;
- bool has_ts;
-
- if (p->expr->ts.u.cl && p->expr->ts.u.cl->length)
- {
- cl = p->expr->ts.u.cl->length;
- gfc_extract_hwi (cl, &current_length);
- }
-
- /* If gfc_extract_int above set current_length, we implicitly
- know the type is BT_INTEGER and it's EXPR_CONSTANT. */
-
- has_ts = expr->ts.u.cl->length_from_typespec;
-
- if (! cl
- || (current_length != -1 && current_length != found_length))
- gfc_set_constant_character_len (found_length, p->expr,
- has_ts ? -1 : found_length);
- }
- }
-
- return true;
-}
-
-
-/* Resolve all of the expressions in an array list. */
-
-bool
-gfc_resolve_array_constructor (gfc_expr *expr)
-{
- bool t;
-
- t = resolve_array_list (expr->value.constructor);
- if (t)
- t = gfc_check_constructor_type (expr);
-
- /* gfc_resolve_character_array_constructor is called in gfc_resolve_expr after
- the call to this function, so we don't need to call it here; if it was
- called twice, an error message there would be duplicated. */
-
- return t;
-}
-
-
-/* Copy an iterator structure. */
-
-gfc_iterator *
-gfc_copy_iterator (gfc_iterator *src)
-{
- gfc_iterator *dest;
-
- if (src == NULL)
- return NULL;
-
- dest = gfc_get_iterator ();
-
- dest->var = gfc_copy_expr (src->var);
- dest->start = gfc_copy_expr (src->start);
- dest->end = gfc_copy_expr (src->end);
- dest->step = gfc_copy_expr (src->step);
- dest->unroll = src->unroll;
- dest->ivdep = src->ivdep;
- dest->vector = src->vector;
- dest->novector = src->novector;
-
- return dest;
-}
-
-
-/********* Subroutines for determining the size of an array *********/
-
-/* These are needed just to accommodate RESHAPE(). There are no
- diagnostics here, we just return false if something goes wrong. */
-
-
-/* Get the size of single dimension of an array specification. The
- array is guaranteed to be one dimensional. */
-
-bool
-spec_dimen_size (gfc_array_spec *as, int dimen, mpz_t *result)
-{
- if (as == NULL)
- return false;
-
- if (dimen < 0 || dimen > as->rank - 1)
- gfc_internal_error ("spec_dimen_size(): Bad dimension");
-
- if (as->type != AS_EXPLICIT
- || !as->lower[dimen]
- || !as->upper[dimen])
- return false;
-
- if (as->lower[dimen]->expr_type != EXPR_CONSTANT
- || as->upper[dimen]->expr_type != EXPR_CONSTANT
- || as->lower[dimen]->ts.type != BT_INTEGER
- || as->upper[dimen]->ts.type != BT_INTEGER)
- return false;
-
- mpz_init (*result);
-
- mpz_sub (*result, as->upper[dimen]->value.integer,
- as->lower[dimen]->value.integer);
-
- mpz_add_ui (*result, *result, 1);
-
- if (mpz_cmp_si (*result, 0) < 0)
- mpz_set_si (*result, 0);
-
- return true;
-}
-
-
-bool
-spec_size (gfc_array_spec *as, mpz_t *result)
-{
- mpz_t size;
- int d;
-
- if (!as || as->type == AS_ASSUMED_RANK)
- return false;
-
- mpz_init_set_ui (*result, 1);
-
- for (d = 0; d < as->rank; d++)
- {
- if (!spec_dimen_size (as, d, &size))
- {
- mpz_clear (*result);
- return false;
- }
-
- mpz_mul (*result, *result, size);
- mpz_clear (size);
- }
-
- return true;
-}
-
-
-/* Get the number of elements in an array section. Optionally, also supply
- the end value. */
-
-bool
-gfc_ref_dimen_size (gfc_array_ref *ar, int dimen, mpz_t *result, mpz_t *end)
-{
- mpz_t upper, lower, stride;
- mpz_t diff;
- bool t;
- gfc_expr *stride_expr = NULL;
-
- if (dimen < 0 || ar == NULL)
- gfc_internal_error ("gfc_ref_dimen_size(): Bad dimension");
-
- if (dimen > ar->dimen - 1)
- {
- gfc_error ("Bad array dimension at %L", &ar->c_where[dimen]);
- return false;
- }
-
- switch (ar->dimen_type[dimen])
- {
- case DIMEN_ELEMENT:
- mpz_init (*result);
- mpz_set_ui (*result, 1);
- t = true;
- break;
-
- case DIMEN_VECTOR:
- t = gfc_array_size (ar->start[dimen], result); /* Recurse! */
- break;
-
- case DIMEN_RANGE:
-
- mpz_init (stride);
-
- if (ar->stride[dimen] == NULL)
- mpz_set_ui (stride, 1);
- else
- {
- stride_expr = gfc_copy_expr(ar->stride[dimen]);
-
- if (!gfc_simplify_expr (stride_expr, 1)
- || stride_expr->expr_type != EXPR_CONSTANT
- || mpz_cmp_ui (stride_expr->value.integer, 0) == 0)
- {
- gfc_free_expr (stride_expr);
- mpz_clear (stride);
- return false;
- }
- mpz_set (stride, stride_expr->value.integer);
- gfc_free_expr(stride_expr);
- }
-
- /* Calculate the number of elements via gfc_dep_differce, but only if
- start and end are both supplied in the reference or the array spec.
- This is to guard against strange but valid code like
-
- subroutine foo(a,n)
- real a(1:n)
- n = 3
- print *,size(a(n-1:))
-
- where the user changes the value of a variable. If we have to
- determine end as well, we cannot do this using gfc_dep_difference.
- Fall back to the constants-only code then. */
-
- if (end == NULL)
- {
- bool use_dep;
-
- use_dep = gfc_dep_difference (ar->end[dimen], ar->start[dimen],
- &diff);
- if (!use_dep && ar->end[dimen] == NULL && ar->start[dimen] == NULL)
- use_dep = gfc_dep_difference (ar->as->upper[dimen],
- ar->as->lower[dimen], &diff);
-
- if (use_dep)
- {
- mpz_init (*result);
- mpz_add (*result, diff, stride);
- mpz_div (*result, *result, stride);
- if (mpz_cmp_ui (*result, 0) < 0)
- mpz_set_ui (*result, 0);
-
- mpz_clear (stride);
- mpz_clear (diff);
- return true;
- }
-
- }
-
- /* Constant-only code here, which covers more cases
- like a(:4) etc. */
- mpz_init (upper);
- mpz_init (lower);
- t = false;
-
- if (ar->start[dimen] == NULL)
- {
- if (ar->as->lower[dimen] == NULL
- || ar->as->lower[dimen]->expr_type != EXPR_CONSTANT
- || ar->as->lower[dimen]->ts.type != BT_INTEGER)
- goto cleanup;
- mpz_set (lower, ar->as->lower[dimen]->value.integer);
- }
- else
- {
- if (ar->start[dimen]->expr_type != EXPR_CONSTANT)
- goto cleanup;
- mpz_set (lower, ar->start[dimen]->value.integer);
- }
-
- if (ar->end[dimen] == NULL)
- {
- if (ar->as->upper[dimen] == NULL
- || ar->as->upper[dimen]->expr_type != EXPR_CONSTANT
- || ar->as->upper[dimen]->ts.type != BT_INTEGER)
- goto cleanup;
- mpz_set (upper, ar->as->upper[dimen]->value.integer);
- }
- else
- {
- if (ar->end[dimen]->expr_type != EXPR_CONSTANT)
- goto cleanup;
- mpz_set (upper, ar->end[dimen]->value.integer);
- }
-
- mpz_init (*result);
- mpz_sub (*result, upper, lower);
- mpz_add (*result, *result, stride);
- mpz_div (*result, *result, stride);
-
- /* Zero stride caught earlier. */
- if (mpz_cmp_ui (*result, 0) < 0)
- mpz_set_ui (*result, 0);
- t = true;
-
- if (end)
- {
- mpz_init (*end);
-
- mpz_sub_ui (*end, *result, 1UL);
- mpz_mul (*end, *end, stride);
- mpz_add (*end, *end, lower);
- }
-
- cleanup:
- mpz_clear (upper);
- mpz_clear (lower);
- mpz_clear (stride);
- return t;
-
- default:
- gfc_internal_error ("gfc_ref_dimen_size(): Bad dimen_type");
- }
-
- return t;
-}
-
-
-static bool
-ref_size (gfc_array_ref *ar, mpz_t *result)
-{
- mpz_t size;
- int d;
-
- mpz_init_set_ui (*result, 1);
-
- for (d = 0; d < ar->dimen; d++)
- {
- if (!gfc_ref_dimen_size (ar, d, &size, NULL))
- {
- mpz_clear (*result);
- return false;
- }
-
- mpz_mul (*result, *result, size);
- mpz_clear (size);
- }
-
- return true;
-}
-
-
-/* Given an array expression and a dimension, figure out how many
- elements it has along that dimension. Returns true if we were
- able to return a result in the 'result' variable, false
- otherwise. */
-
-bool
-gfc_array_dimen_size (gfc_expr *array, int dimen, mpz_t *result)
-{
- gfc_ref *ref;
- int i;
-
- gcc_assert (array != NULL);
-
- if (array->ts.type == BT_CLASS)
- return false;
-
- if (array->rank == -1)
- return false;
-
- if (dimen < 0 || dimen > array->rank - 1)
- gfc_internal_error ("gfc_array_dimen_size(): Bad dimension");
-
- switch (array->expr_type)
- {
- case EXPR_VARIABLE:
- case EXPR_FUNCTION:
- for (ref = array->ref; ref; ref = ref->next)
- {
- if (ref->type != REF_ARRAY)
- continue;
-
- if (ref->u.ar.type == AR_FULL)
- return spec_dimen_size (ref->u.ar.as, dimen, result);
-
- if (ref->u.ar.type == AR_SECTION)
- {
- for (i = 0; dimen >= 0; i++)
- if (ref->u.ar.dimen_type[i] != DIMEN_ELEMENT)
- dimen--;
-
- return gfc_ref_dimen_size (&ref->u.ar, i - 1, result, NULL);
- }
- }
-
- if (array->shape)
- {
- mpz_init_set (*result, array->shape[dimen]);
- return true;
- }
-
- if (array->symtree->n.sym->attr.generic
- && array->value.function.esym != NULL)
- {
- if (!spec_dimen_size (array->value.function.esym->as, dimen, result))
- return false;
- }
- else if (!spec_dimen_size (array->symtree->n.sym->as, dimen, result))
- return false;
-
- break;
-
- case EXPR_ARRAY:
- if (array->shape == NULL) {
- /* Expressions with rank > 1 should have "shape" properly set */
- if ( array->rank != 1 )
- gfc_internal_error ("gfc_array_dimen_size(): Bad EXPR_ARRAY expr");
- return gfc_array_size(array, result);
- }
-
- /* Fall through */
- default:
- if (array->shape == NULL)
- return false;
-
- mpz_init_set (*result, array->shape[dimen]);
-
- break;
- }
-
- return true;
-}
-
-
-/* Given an array expression, figure out how many elements are in the
- array. Returns true if this is possible, and sets the 'result'
- variable. Otherwise returns false. */
-
-bool
-gfc_array_size (gfc_expr *array, mpz_t *result)
-{
- expand_info expand_save;
- gfc_ref *ref;
- int i;
- bool t;
-
- if (array->ts.type == BT_CLASS)
- return false;
-
- switch (array->expr_type)
- {
- case EXPR_ARRAY:
- gfc_push_suppress_errors ();
-
- expand_save = current_expand;
-
- current_expand.count = result;
- mpz_init_set_ui (*result, 0);
-
- current_expand.expand_work_function = count_elements;
- iter_stack = NULL;
-
- t = expand_constructor (array->value.constructor);
-
- gfc_pop_suppress_errors ();
-
- if (!t)
- mpz_clear (*result);
- current_expand = expand_save;
- return t;
-
- case EXPR_VARIABLE:
- for (ref = array->ref; ref; ref = ref->next)
- {
- if (ref->type != REF_ARRAY)
- continue;
-
- if (ref->u.ar.type == AR_FULL)
- return spec_size (ref->u.ar.as, result);
-
- if (ref->u.ar.type == AR_SECTION)
- return ref_size (&ref->u.ar, result);
- }
-
- return spec_size (array->symtree->n.sym->as, result);
-
-
- default:
- if (array->rank == 0 || array->shape == NULL)
- return false;
-
- mpz_init_set_ui (*result, 1);
-
- for (i = 0; i < array->rank; i++)
- mpz_mul (*result, *result, array->shape[i]);
-
- break;
- }
-
- return true;
-}
-
-
-/* Given an array reference, return the shape of the reference in an
- array of mpz_t integers. */
-
-bool
-gfc_array_ref_shape (gfc_array_ref *ar, mpz_t *shape)
-{
- int d;
- int i;
-
- d = 0;
-
- switch (ar->type)
- {
- case AR_FULL:
- for (; d < ar->as->rank; d++)
- if (!spec_dimen_size (ar->as, d, &shape[d]))
- goto cleanup;
-
- return true;
-
- case AR_SECTION:
- for (i = 0; i < ar->dimen; i++)
- {
- if (ar->dimen_type[i] != DIMEN_ELEMENT)
- {
- if (!gfc_ref_dimen_size (ar, i, &shape[d], NULL))
- goto cleanup;
- d++;
- }
- }
-
- return true;
-
- default:
- break;
- }
-
-cleanup:
- gfc_clear_shape (shape, d);
- return false;
-}
-
-
-/* Given an array expression, find the array reference structure that
- characterizes the reference. */
-
-gfc_array_ref *
-gfc_find_array_ref (gfc_expr *e, bool allow_null)
-{
- gfc_ref *ref;
-
- for (ref = e->ref; ref; ref = ref->next)
- if (ref->type == REF_ARRAY
- && (ref->u.ar.type == AR_FULL || ref->u.ar.type == AR_SECTION))
- break;
-
- if (ref == NULL)
- {
- if (allow_null)
- return NULL;
- else
- gfc_internal_error ("gfc_find_array_ref(): No ref found");
- }
-
- return &ref->u.ar;
-}
-
-
-/* Find out if an array shape is known at compile time. */
-
-bool
-gfc_is_compile_time_shape (gfc_array_spec *as)
-{
- if (as->type != AS_EXPLICIT)
- return false;
-
- for (int i = 0; i < as->rank; i++)
- if (!gfc_is_constant_expr (as->lower[i])
- || !gfc_is_constant_expr (as->upper[i]))
- return false;
-
- return true;
-}