aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorThomas Koenig <tkoenig@gcc.gnu.org>2019-01-15 22:18:55 +0000
committerThomas Koenig <tkoenig@gcc.gnu.org>2019-01-15 22:18:55 +0000
commit0335cc372c9849c61b9a5583655820d88ecf829e (patch)
tree422262bd5ef07f5e981b8ccb2e3429ce59dd8e15
parent6a0c8e77f289a5c2e1b1ad0f2c8a5c5105f599a6 (diff)
downloadgcc-0335cc372c9849c61b9a5583655820d88ecf829e.zip
gcc-0335cc372c9849c61b9a5583655820d88ecf829e.tar.gz
gcc-0335cc372c9849c61b9a5583655820d88ecf829e.tar.bz2
re PR fortran/43072 (unneeded temporary (s=s+f(a)))
2019-01-15 Thomas Koenig <tkoenig@gcc.gnu.org> PR fortran/43072 * resolve.c (resolve_array_ref): Add equal_length argument; set it if the length of the substring equals that of the orignal variable. (resolve_ref): Remove the substring if it is equal in length to the original variable, unless it is an EXPR_SUBSTRING). 2019-01-15 Thomas Koenig <tkoenig@gcc.gnu.org> PR fortran/43072 * gfortran.dg/actual_array_substr_3.f90: New test. From-SVN: r267953
-rw-r--r--gcc/fortran/ChangeLog9
-rw-r--r--gcc/fortran/resolve.c28
-rw-r--r--gcc/testsuite/ChangeLog5
-rw-r--r--gcc/testsuite/gfortran.dg/actual_array_substr_3.f9011
4 files changed, 49 insertions, 4 deletions
diff --git a/gcc/fortran/ChangeLog b/gcc/fortran/ChangeLog
index 76a840d..574f50a 100644
--- a/gcc/fortran/ChangeLog
+++ b/gcc/fortran/ChangeLog
@@ -1,3 +1,12 @@
+2019-01-15 Thomas Koenig <tkoenig@gcc.gnu.org>
+
+ PR fortran/43072
+ * resolve.c (resolve_array_ref): Add equal_length argument; set it
+ if the length of the substring equals that of the orignal
+ variable.
+ (resolve_ref): Remove the substring if it is equal in length to
+ the original variable, unless it is an EXPR_SUBSTRING).
+
2019-01-15 Steven G. Kargl <kargl@gcc.gnu.org>
PR fortran/81849
diff --git a/gcc/fortran/resolve.c b/gcc/fortran/resolve.c
index 7fbfa69..b1c9292 100644
--- a/gcc/fortran/resolve.c
+++ b/gcc/fortran/resolve.c
@@ -4873,7 +4873,7 @@ resolve_array_ref (gfc_array_ref *ar)
static bool
-resolve_substring (gfc_ref *ref)
+resolve_substring (gfc_ref *ref, bool *equal_length)
{
int k = gfc_validate_kind (BT_INTEGER, gfc_charlen_int_kind, false);
@@ -4944,6 +4944,13 @@ resolve_substring (gfc_ref *ref)
&ref->u.ss.end->where);
return false;
}
+ /* If the substring has the same length as the original
+ variable, the reference itself can be deleted. */
+
+ if (ref->u.ss.length != NULL
+ && compare_bound (ref->u.ss.end, ref->u.ss.length->length) == CMP_EQ
+ && compare_bound_int (ref->u.ss.start, 1) == CMP_EQ)
+ *equal_length = true;
}
return true;
@@ -5037,7 +5044,8 @@ static bool
resolve_ref (gfc_expr *expr)
{
int current_part_dimension, n_components, seen_part_dimension;
- gfc_ref *ref;
+ gfc_ref *ref, **prev;
+ bool equal_length;
for (ref = expr->ref; ref; ref = ref->next)
if (ref->type == REF_ARRAY && ref->u.ar.as == NULL)
@@ -5046,7 +5054,8 @@ resolve_ref (gfc_expr *expr)
break;
}
- for (ref = expr->ref; ref; ref = ref->next)
+
+ for (ref = expr->ref, prev = &expr->ref; ref; prev = &ref->next, ref = ref->next)
switch (ref->type)
{
case REF_ARRAY:
@@ -5059,8 +5068,19 @@ resolve_ref (gfc_expr *expr)
break;
case REF_SUBSTRING:
- if (!resolve_substring (ref))
+ equal_length = false;
+ if (!resolve_substring (ref, &equal_length))
return false;
+
+ if (expr->expr_type != EXPR_SUBSTRING && equal_length)
+ {
+ /* Remove the reference and move the charlen, if any. */
+ *prev = ref->next;
+ ref->next = NULL;
+ expr->ts.u.cl = ref->u.ss.length;
+ ref->u.ss.length = NULL;
+ gfc_free_ref_list (ref);
+ }
break;
}
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index 56265c8..9c9299e 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,3 +1,8 @@
+2019-01-15 Thomas Koenig <tkoenig@gcc.gnu.org>
+
+ PR fortran/43072
+ * gfortran.dg/actual_array_substr_3.f90: New test.
+
2019-01-15 Steven G. Kargl <kargl@gcc.gnu.org>
PR fortran/81849
diff --git a/gcc/testsuite/gfortran.dg/actual_array_substr_3.f90 b/gcc/testsuite/gfortran.dg/actual_array_substr_3.f90
new file mode 100644
index 0000000..30d8edf
--- /dev/null
+++ b/gcc/testsuite/gfortran.dg/actual_array_substr_3.f90
@@ -0,0 +1,11 @@
+! { dg-do compile }
+! { dg-additional-options "-fdump-tree-original" }
+! PR 43072 - no temporary needed because the substring
+! is of equal length to the string.
+subroutine foo2
+ implicit none
+ external foo
+ character(len=20) :: str(2) = '1234567890'
+ call foo(str(:)(1:20))
+end
+! { dg-final { scan-tree-dump-not "memmove" "original" } }