diff options
author | Paul Thomas <pault@gcc.gnu.org> | 2007-11-27 20:47:55 +0000 |
---|---|---|
committer | Paul Thomas <pault@gcc.gnu.org> | 2007-11-27 20:47:55 +0000 |
commit | 908a22351801ee5f0f364d14a55ae38f546565b4 (patch) | |
tree | 02fd93e8d327aa79aa562c5f974746f1257b83be /gcc/fortran/expr.c | |
parent | 0e5a218b31eb720caa70b19439e26f658f151070 (diff) | |
download | gcc-908a22351801ee5f0f364d14a55ae38f546565b4.zip gcc-908a22351801ee5f0f364d14a55ae38f546565b4.tar.gz gcc-908a22351801ee5f0f364d14a55ae38f546565b4.tar.bz2 |
re PR fortran/29389 (Statement functions are not recognized as pure when they are)
2007-11-27 Paul Thomas <pault@gcc.gnu.org>
PR fortran/29389
*resolve.c (resolve_ordinary_assign): Use find_sym_in_expr to
test if a temporary should be written for a vector subscript
on the lhs.
PR fortran/33850
* restore.c (pure_stmt_function): Add prototype and new
function. Calls impure_stmt_fcn.
(pure_function): Call it.
(impure_stmt_fcn): New function.
* expr.c (gfc_traverse_expr): Call *func for all expression
types, not just variables. Add traversal of character lengths,
iterators and component character lengths and arrayspecs.
(expr_set_symbols_referenced): Return false if not a variable.
* trans-stmt.c (forall_replace, forall_restore): Ditto.
* resolve.c (forall_index): Ditto.
(sym_in_expr): New function.
(find_sym_in_expr): Rewrite to traverse expression calling
sym_in_expr.
*trans-decl.c (expr_decls): New function.
(generate_expr_decls): Rewrite to traverse expression calling
expr_decls.
*match.c (check_stmt_fcn): New function.
(recursive_stmt_fcn): Rewrite to traverse expression calling
check_stmt_fcn.
2007-11-27 Paul Thomas <pault@gcc.gnu.org>
PR fortran/29389
* gfortran.dg/stfunc_6.f90: New test.
PR fortran/33850
* gfortran.dg/assign_10.f90: New test.
From-SVN: r130472
Diffstat (limited to 'gcc/fortran/expr.c')
-rw-r--r-- | gcc/fortran/expr.c | 58 |
1 files changed, 49 insertions, 9 deletions
diff --git a/gcc/fortran/expr.c b/gcc/fortran/expr.c index 22df131..e33d97a 100644 --- a/gcc/fortran/expr.c +++ b/gcc/fortran/expr.c @@ -3010,14 +3010,18 @@ gfc_traverse_expr (gfc_expr *expr, gfc_symbol *sym, if (!expr) return false; - switch (expr->expr_type) - { - case EXPR_VARIABLE: - gcc_assert (expr->symtree->n.sym); + if ((*func) (expr, sym, &f)) + return true; - if ((*func) (expr, sym, &f)) - return true; + if (expr->ts.type == BT_CHARACTER + && expr->ts.cl + && expr->ts.cl->length + && expr->ts.cl->length->expr_type != EXPR_CONSTANT + && gfc_traverse_expr (expr->ts.cl->length, sym, func, f)) + return true; + switch (expr->expr_type) + { case EXPR_FUNCTION: for (args = expr->value.function.actual; args; args = args->next) { @@ -3026,6 +3030,7 @@ gfc_traverse_expr (gfc_expr *expr, gfc_symbol *sym, } break; + case EXPR_VARIABLE: case EXPR_CONSTANT: case EXPR_NULL: case EXPR_SUBSTRING: @@ -3034,7 +3039,21 @@ gfc_traverse_expr (gfc_expr *expr, gfc_symbol *sym, case EXPR_STRUCTURE: case EXPR_ARRAY: for (c = expr->value.constructor; c; c = c->next) - gfc_expr_set_symbols_referenced (c->expr); + { + if (gfc_traverse_expr (c->expr, sym, func, f)) + return true; + if (c->iterator) + { + if (gfc_traverse_expr (c->iterator->var, sym, func, f)) + return true; + if (gfc_traverse_expr (c->iterator->start, sym, func, f)) + return true; + if (gfc_traverse_expr (c->iterator->end, sym, func, f)) + return true; + if (gfc_traverse_expr (c->iterator->step, sym, func, f)) + return true; + } + } break; case EXPR_OP: @@ -3074,8 +3093,27 @@ gfc_traverse_expr (gfc_expr *expr, gfc_symbol *sym, return true; break; - case REF_COMPONENT: - break; + case REF_COMPONENT: + if (ref->u.c.component->ts.type == BT_CHARACTER + && ref->u.c.component->ts.cl + && ref->u.c.component->ts.cl->length + && ref->u.c.component->ts.cl->length->expr_type + != EXPR_CONSTANT + && gfc_traverse_expr (ref->u.c.component->ts.cl->length, + sym, func, f)) + return true; + + if (ref->u.c.component->as) + for (i = 0; i < ref->u.c.component->as->rank; i++) + { + if (gfc_traverse_expr (ref->u.c.component->as->lower[i], + sym, func, f)) + return true; + if (gfc_traverse_expr (ref->u.c.component->as->upper[i], + sym, func, f)) + return true; + } + break; default: gcc_unreachable (); @@ -3092,6 +3130,8 @@ expr_set_symbols_referenced (gfc_expr *expr, gfc_symbol *sym ATTRIBUTE_UNUSED, int *f ATTRIBUTE_UNUSED) { + if (expr->expr_type != EXPR_VARIABLE) + return false; gfc_set_sym_referenced (expr->symtree->n.sym); return false; } |