This is the mail archive of the
fortran@gcc.gnu.org
mailing list for the GNU Fortran project.
Re: [patch, fortran] PR fortran/36928 - array dependencies
- From: Thomas Koenig <tkoenig at netcologne dot de>
- To: Tobias Burnus <burnus at net-b dot de>
- Cc: gfortran <fortran at gcc dot gnu dot org>, gcc patches <gcc-patches at gcc dot gnu dot org>
- Date: Mon, 31 May 2010 22:26:06 +0200
- Subject: Re: [patch, fortran] PR fortran/36928 - array dependencies
- References: <1275220993.4579.37.camel@linux-fd1f.site> <4C029C03.80307@net-b.de>
Tobias Burnus wrote:
> You should add a short comment on the algorithm thus one can quickly
> check it, e.g.
> "No dependency if reminder of (l_start - r_start)/gcd(l_stride,
> r_stride)) is nonzero."
> At least it took me a short while to translate the GMP functions into
> the string above.
>
> (Additionally, I felt better, if you would add a run-test and include
> also some checks that the temporary is still generated in some commen
> cases. I am always worrying that on introduces bugs in such cases.)
I introduced some correctness checks in an additional test case (see
below).
> > Regression-tested on x86_64-unknown-linux-gnu. OK for trunk?
> >
> OK with considering the nit above.
Waiting for Emacs...
Sende fortran/ChangeLog
Sende fortran/dependency.c
Sende testsuite/ChangeLog
Hinzufügen testsuite/gfortran.dg/array_assignment_1.F90
Hinzufügen testsuite/gfortran.dg/dependency_27.f90
Übertrage Daten .....
Revision 160085 übertragen.
Here's what I committed.
Thanks for the review!
Thomas
2010-05-31 Thomas Koenig <tkoenig@gcc.gnu.org>
PR fortran/36928
* dependency.c (gfc_check_section_vs_section): Check
for interleaving array assignments without conflicts.
2010-05-31 Thomas Koenig <tkoenig@gcc.gnu.org>
PR fortran/36928
* gfortran.dg/dependency_27.f90: New test.
* gfortran.dg/array_assign_1.F90: New test.
! { dg-do compile }
! { dg-options "-Warray-temporaries" }
! PR 36928 - optimize array interleaving array temporaries
program main
real, dimension(20) :: a
read (10) a
a(2:10:2) = a (1:9:2)
write (11) a
read (10) a
a(2:10:4) = a(1:5:2)
write (11) a
read (10) a
a(2:10:4) = a(5:1:-2)
write (11) a
end program main
! { dg-do run }
! { dg-options "-ffree-line-length-none" }
! Test that different array assignments work even when interleaving,
! reversing etc. Make sure the results from assignment with constants
! as array triples and runtime array triples (where we always create
! a temporary) match.
#define TST(b,c,d,e,f,g,r) a=init; a(b:c:d) = a(e:f:g); \
write(unit=line ,fmt="(9I1)") a;\
if (line /= r) call abort ; \
call mytst(b,c,d,e,f,g,r);
program main
implicit none
integer :: i
integer, parameter :: n=9
integer, dimension(n) :: a
character(len=n) :: line
integer, dimension(n), parameter :: init = (/(i,i=1,n)/)
TST(2,n,2,1,n-1,2,'113355779')
TST(3,9,3,2,6,2,'122454786');
TST(1,8,2,3,9,2,'325476989');
TST(1,6,1,4,9,1,'456789789');
TST(9,5,-1,1,5,1,'123454321');
TST(9,5,-2,1,5,2,'123456381');
TST(5,9,2,5,1,-2,'123456381');
TST(1,6,1,2,7,1,'234567789');
TST(2,7,1,1,6,1,'112345689');
end program main
subroutine mytst(b,c,d,e,f,g,r)
integer,intent(in) :: b,c,d,e,f,g
character(len=9), intent(in) :: r
character(len=9) :: line
integer, dimension(9) :: a
a = (/(i,i=1,9)/)
a(b:c:d) = a(e:f:g)
write (unit=line,fmt='(9I1)') a
if (line /= r) call abort
end subroutine mytst
Index: fortran/dependency.c
===================================================================
--- fortran/dependency.c (Revision 159995)
+++ fortran/dependency.c (Arbeitskopie)
@@ -999,6 +999,42 @@
return GFC_DEP_EQUAL;
}
+ /* Handle cases like x:y:2 vs. x+1:z:4 as GFC_DEP_NODEP.
+ There is no dependency if the remainder of
+ (l_start - r_start) / gcd(l_stride, r_stride) is
+ nonzero.
+ TODO:
+ - Handle cases where x is an expression.
+ - Cases like a(1:4:2) = a(2:3) are still not handled.
+ */
+
+#define IS_CONSTANT_INTEGER(a) ((a) && ((a)->expr_type == EXPR_CONSTANT) \
+ && (a)->ts.type == BT_INTEGER)
+
+ if (IS_CONSTANT_INTEGER(l_start) && IS_CONSTANT_INTEGER(r_start)
+ && IS_CONSTANT_INTEGER(l_stride) && IS_CONSTANT_INTEGER(r_stride))
+ {
+ mpz_t gcd, tmp;
+ int result;
+
+ mpz_init (gcd);
+ mpz_init (tmp);
+
+ mpz_gcd (gcd, l_stride->value.integer, r_stride->value.integer);
+ mpz_sub (tmp, l_start->value.integer, r_start->value.integer);
+
+ mpz_fdiv_r (tmp, tmp, gcd);
+ result = mpz_cmp_si (tmp, 0L);
+
+ mpz_clear (gcd);
+ mpz_clear (tmp);
+
+ if (result != 0)
+ return GFC_DEP_NODEP;
+ }
+
+#undef IS_CONSTANT_INTEGER
+
/* Check for forward dependencies x:y vs. x+1:z. */
if (l_dir == 1 && r_dir == 1
&& l_start && r_start && gfc_dep_compare_expr (l_start, r_start) == -1