re PR fortran/45648 (Unnecessary temporary for transpose calls as actual argument.)

2010-09-21  Mikael Morin  <mikael@gcc.gnu.org>

	PR fortran/45648
	* trans-array.c (gfc_conv_expr_descriptor): Calculate dim out of n and
	info->dim.

	PR fortran/45648
	* trans-array.c (gfc_conv_expr_descriptor): Unset full if we are
	accessing dimensions in reversed order. 

	PR fortran/45648
	* trans-array.c (gfc_conv_expr_descriptor): Special case noncopying
	intrinsic function call. 

	* trans-array.c (gfc_conv_expr_descriptor): Remove ss lookup.
	Update asserts accordingly.

	PR fortran/45648
	* trans.h (gfc_se): New field force_tmp. 
	* trans-expr.c (gfc_conv_procedure_call): Check for argument alias
	and set parmse.force_tmp if some alias is found. 
	* trans-array.c (gfc_conv_expr_descriptor): Force a temporary creation
	if se->force_tmp is set. 

2010-09-21  Mikael Morin  <mikael@gcc.gnu.org>

	PR fortran/45648
	* gfortran.dg/inline_transpose_1.f90: Update temporary's locations
	and counts. Add non-elemental function call check.

	PR fortran/45648
	* gfortran.dg/inline_transpose_1.f90: Add function calls with aliasing
	arguments checks. Update temporary counts.
	* gfortran.dg/transpose_optimization_1.f90: New.

From-SVN: r164494
This commit is contained in:
Mikael Morin 2010-09-21 19:04:09 +00:00
parent 5e68c77aff
commit 0b4f2770ff
7 changed files with 284 additions and 53 deletions

View File

@ -1,3 +1,27 @@
2010-09-21 Mikael Morin <mikael@gcc.gnu.org>
PR fortran/45648
* trans-array.c (gfc_conv_expr_descriptor): Calculate dim out of n and
info->dim.
PR fortran/45648
* trans-array.c (gfc_conv_expr_descriptor): Unset full if we are
accessing dimensions in reversed order.
PR fortran/45648
* trans-array.c (gfc_conv_expr_descriptor): Special case noncopying
intrinsic function call.
* trans-array.c (gfc_conv_expr_descriptor): Remove ss lookup.
Update asserts accordingly.
PR fortran/45648
* trans.h (gfc_se): New field force_tmp.
* trans-expr.c (gfc_conv_procedure_call): Check for argument alias
and set parmse.force_tmp if some alias is found.
* trans-array.c (gfc_conv_expr_descriptor): Force a temporary creation
if se->force_tmp is set.
2010-09-20 Janus Weil <janus@gcc.gnu.org> 2010-09-20 Janus Weil <janus@gcc.gnu.org>
PR fortran/45438 PR fortran/45438

View File

@ -5136,7 +5136,6 @@ get_array_charlen (gfc_expr *expr, gfc_se *se)
} }
/* Convert an array for passing as an actual argument. Expressions and /* Convert an array for passing as an actual argument. Expressions and
vector subscripts are evaluated and stored in a temporary, which is then vector subscripts are evaluated and stored in a temporary, which is then
passed. For whole arrays the descriptor is passed. For array sections passed. For whole arrays the descriptor is passed. For array sections
@ -5158,13 +5157,18 @@ get_array_charlen (gfc_expr *expr, gfc_se *se)
EXPR is the right-hand side of a pointer assignment and EXPR is the right-hand side of a pointer assignment and
se->expr is the descriptor for the previously-evaluated se->expr is the descriptor for the previously-evaluated
left-hand side. The function creates an assignment from left-hand side. The function creates an assignment from
EXPR to se->expr. */ EXPR to se->expr.
The se->force_tmp flag disables the non-copying descriptor optimization
that is used for transpose. It may be used in cases where there is an
alias between the transpose argument and another argument in the same
function call. */
void void
gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss) gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
{ {
gfc_loopinfo loop; gfc_loopinfo loop;
gfc_ss *secss;
gfc_ss_info *info; gfc_ss_info *info;
int need_tmp; int need_tmp;
int n; int n;
@ -5175,7 +5179,9 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
tree offset; tree offset;
int full; int full;
bool subref_array_target = false; bool subref_array_target = false;
gfc_expr *arg;
gcc_assert (ss != NULL);
gcc_assert (ss != gfc_ss_terminator); gcc_assert (ss != gfc_ss_terminator);
/* Special case things we know we can pass easily. */ /* Special case things we know we can pass easily. */
@ -5185,22 +5191,21 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
/* If we have a linear array section, we can pass it directly. /* If we have a linear array section, we can pass it directly.
Otherwise we need to copy it into a temporary. */ Otherwise we need to copy it into a temporary. */
/* Find the SS for the array section. */ gcc_assert (ss->type == GFC_SS_SECTION);
secss = ss; gcc_assert (ss->expr == expr);
while (secss != gfc_ss_terminator && secss->type != GFC_SS_SECTION) info = &ss->data.info;
secss = secss->next;
gcc_assert (secss != gfc_ss_terminator);
info = &secss->data.info;
/* Get the descriptor for the array. */ /* Get the descriptor for the array. */
gfc_conv_ss_descriptor (&se->pre, secss, 0); gfc_conv_ss_descriptor (&se->pre, ss, 0);
desc = info->descriptor; desc = info->descriptor;
subref_array_target = se->direct_byref && is_subref_array (expr); subref_array_target = se->direct_byref && is_subref_array (expr);
need_tmp = gfc_ref_needs_temporary_p (expr->ref) need_tmp = gfc_ref_needs_temporary_p (expr->ref)
&& !subref_array_target; && !subref_array_target;
if (se->force_tmp)
need_tmp = 1;
if (need_tmp) if (need_tmp)
full = 0; full = 0;
else if (GFC_ARRAY_TYPE_P (TREE_TYPE (desc))) else if (GFC_ARRAY_TYPE_P (TREE_TYPE (desc)))
@ -5215,6 +5220,14 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
else else
full = gfc_full_array_ref_p (info->ref, NULL); full = gfc_full_array_ref_p (info->ref, NULL);
if (full)
for (n = 0; n < info->dimen; n++)
if (info->dim[n] != n)
{
full = 0;
break;
}
if (full) if (full)
{ {
if (se->direct_byref && !se->byref_noassign) if (se->direct_byref && !se->byref_noassign)
@ -5245,30 +5258,45 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
break; break;
case EXPR_FUNCTION: case EXPR_FUNCTION:
/* We don't need to copy data in some cases. */
arg = gfc_get_noncopying_intrinsic_argument (expr);
if (arg)
{
/* This is a call to transpose... */
gcc_assert (expr->value.function.isym->id == GFC_ISYM_TRANSPOSE);
/* ... which has already been handled by the scalarizer, so
that we just need to get its argument's descriptor. */
gfc_conv_expr_descriptor (se, expr->value.function.actual->expr, ss);
return;
}
/* A transformational function return value will be a temporary /* A transformational function return value will be a temporary
array descriptor. We still need to go through the scalarizer array descriptor. We still need to go through the scalarizer
to create the descriptor. Elemental functions ar handled as to create the descriptor. Elemental functions ar handled as
arbitrary expressions, i.e. copy to a temporary. */ arbitrary expressions, i.e. copy to a temporary. */
secss = ss;
/* Look for the SS for this function. */
while (secss != gfc_ss_terminator
&& (secss->type != GFC_SS_FUNCTION || secss->expr != expr))
secss = secss->next;
if (se->direct_byref) if (se->direct_byref)
{ {
gcc_assert (secss != gfc_ss_terminator); gcc_assert (ss->type == GFC_SS_FUNCTION && ss->expr == expr);
/* For pointer assignments pass the descriptor directly. */ /* For pointer assignments pass the descriptor directly. */
se->ss = secss; if (se->ss == NULL)
se->ss = ss;
else
gcc_assert (se->ss == ss);
se->expr = gfc_build_addr_expr (NULL_TREE, se->expr); se->expr = gfc_build_addr_expr (NULL_TREE, se->expr);
gfc_conv_expr (se, expr); gfc_conv_expr (se, expr);
return; return;
} }
if (secss == gfc_ss_terminator) if (ss->expr != expr)
{ {
/* Elemental function. */ /* Elemental function. */
gcc_assert ((expr->value.function.esym != NULL
&& expr->value.function.esym->attr.elemental)
|| (expr->value.function.isym != NULL
&& expr->value.function.isym->elemental));
need_tmp = 1; need_tmp = 1;
if (expr->ts.type == BT_CHARACTER if (expr->ts.type == BT_CHARACTER
&& expr->ts.u.cl->length->expr_type != EXPR_CONSTANT) && expr->ts.u.cl->length->expr_type != EXPR_CONSTANT)
@ -5279,7 +5307,7 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
else else
{ {
/* Transformational function. */ /* Transformational function. */
info = &secss->data.info; info = &ss->data.info;
need_tmp = 0; need_tmp = 0;
} }
break; break;
@ -5292,12 +5320,10 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
{ {
need_tmp = 0; need_tmp = 0;
info = &ss->data.info; info = &ss->data.info;
secss = ss;
} }
else else
{ {
need_tmp = 1; need_tmp = 1;
secss = NULL;
info = NULL; info = NULL;
} }
break; break;
@ -5305,11 +5331,15 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
default: default:
/* Something complicated. Copy it into a temporary. */ /* Something complicated. Copy it into a temporary. */
need_tmp = 1; need_tmp = 1;
secss = NULL;
info = NULL; info = NULL;
break; break;
} }
/* If we are creating a temporary, we don't need to bother about aliases
anymore. */
if (need_tmp)
se->force_tmp = 0;
gfc_init_loopinfo (&loop); gfc_init_loopinfo (&loop);
/* Associate the SS with the loop. */ /* Associate the SS with the loop. */
@ -5421,7 +5451,6 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
se->string_length = gfc_get_expr_charlen (expr); se->string_length = gfc_get_expr_charlen (expr);
desc = info->descriptor; desc = info->descriptor;
gcc_assert (secss && secss != gfc_ss_terminator);
if (se->direct_byref && !se->byref_noassign) if (se->direct_byref && !se->byref_noassign)
{ {
/* For pointer assignments we fill in the destination. */ /* For pointer assignments we fill in the destination. */
@ -5439,12 +5468,11 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
} }
offset = gfc_index_zero_node; offset = gfc_index_zero_node;
dim = 0;
/* The following can be somewhat confusing. We have two /* The following can be somewhat confusing. We have two
descriptors, a new one and the original array. descriptors, a new one and the original array.
{parm, parmtype, dim} refer to the new one. {parm, parmtype, dim} refer to the new one.
{desc, type, n, secss, loop} refer to the original, which maybe {desc, type, n, loop} refer to the original, which maybe
a descriptorless array. a descriptorless array.
The bounds of the scalarization are the bounds of the section. The bounds of the scalarization are the bounds of the section.
We don't have to worry about numeric overflows when calculating We don't have to worry about numeric overflows when calculating
@ -5479,9 +5507,6 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
} }
else else
{ {
/* Check we haven't somehow got out of sync. */
gcc_assert (info->dim[dim] == n);
/* Evaluate and remember the start of the section. */ /* Evaluate and remember the start of the section. */
start = info->start[n]; start = info->start[n];
stride = gfc_evaluate_now (stride, &loop.pre); stride = gfc_evaluate_now (stride, &loop.pre);
@ -5506,6 +5531,14 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
if (info->ref) if (info->ref)
gcc_assert (info->ref->u.ar.dimen_type[n] == DIMEN_RANGE); gcc_assert (info->ref->u.ar.dimen_type[n] == DIMEN_RANGE);
/* look for the corresponding scalarizer dimension: dim. */
for (dim = 0; dim < ndim; dim++)
if (info->dim[dim] == n)
break;
/* loop exited early: the DIM being looked for has been found. */
gcc_assert (dim < ndim);
/* Set the new lower bound. */ /* Set the new lower bound. */
from = loop.from[dim]; from = loop.from[dim];
to = loop.to[dim]; to = loop.to[dim];
@ -5559,8 +5592,6 @@ gfc_conv_expr_descriptor (gfc_se * se, gfc_expr * expr, gfc_ss * ss)
/* Store the new stride. */ /* Store the new stride. */
gfc_conv_descriptor_stride_set (&loop.pre, parm, gfc_conv_descriptor_stride_set (&loop.pre, parm,
gfc_rank_cst[dim], stride); gfc_rank_cst[dim], stride);
dim++;
} }
if (se->data_not_needed) if (se->data_not_needed)

View File

@ -2770,7 +2770,7 @@ conv_isocbinding_procedure (gfc_se * se, gfc_symbol * sym,
int int
gfc_conv_procedure_call (gfc_se * se, gfc_symbol * sym, gfc_conv_procedure_call (gfc_se * se, gfc_symbol * sym,
gfc_actual_arglist * arg, gfc_expr * expr, gfc_actual_arglist * args, gfc_expr * expr,
VEC(tree,gc) *append_args) VEC(tree,gc) *append_args)
{ {
gfc_interface_mapping mapping; gfc_interface_mapping mapping;
@ -2789,6 +2789,7 @@ gfc_conv_procedure_call (gfc_se * se, gfc_symbol * sym,
VEC(tree,gc) *stringargs; VEC(tree,gc) *stringargs;
tree result = NULL; tree result = NULL;
gfc_formal_arglist *formal; gfc_formal_arglist *formal;
gfc_actual_arglist *arg;
int has_alternate_specifier = 0; int has_alternate_specifier = 0;
bool need_interface_mapping; bool need_interface_mapping;
bool callee_alloc; bool callee_alloc;
@ -2809,7 +2810,7 @@ gfc_conv_procedure_call (gfc_se * se, gfc_symbol * sym,
gfc_clear_ts (&ts); gfc_clear_ts (&ts);
if (sym->from_intmod == INTMOD_ISO_C_BINDING if (sym->from_intmod == INTMOD_ISO_C_BINDING
&& conv_isocbinding_procedure (se, sym, arg)) && conv_isocbinding_procedure (se, sym, args))
return 0; return 0;
gfc_is_proc_ptr_comp (expr, &comp); gfc_is_proc_ptr_comp (expr, &comp);
@ -2859,7 +2860,8 @@ gfc_conv_procedure_call (gfc_se * se, gfc_symbol * sym,
} }
/* Evaluate the arguments. */ /* Evaluate the arguments. */
for (; arg != NULL; arg = arg->next, formal = formal ? formal->next : NULL) for (arg = args; arg != NULL;
arg = arg->next, formal = formal ? formal->next : NULL)
{ {
e = arg->expr; e = arg->expr;
fsym = formal ? formal->sym : NULL; fsym = formal ? formal->sym : NULL;
@ -3040,6 +3042,24 @@ gfc_conv_procedure_call (gfc_se * se, gfc_symbol * sym,
else else
f = f || !sym->attr.always_explicit; f = f || !sym->attr.always_explicit;
/* If the argument is a function call that may not create
a temporary for the result, we have to check that we
can do it, i.e. that there is no alias between this
argument and another one. */
if (gfc_get_noncopying_intrinsic_argument (e) != NULL)
{
sym_intent intent;
if (fsym != NULL)
intent = fsym->attr.intent;
else
intent = INTENT_UNKNOWN;
if (gfc_check_fncall_dependency (e, intent, sym, args,
NOT_ELEMENTAL))
parmse.force_tmp = 1;
}
if (e->expr_type == EXPR_VARIABLE if (e->expr_type == EXPR_VARIABLE
&& is_subref_array (e)) && is_subref_array (e))
/* The actual argument is a component reference to an /* The actual argument is a component reference to an

View File

@ -81,6 +81,11 @@ typedef struct gfc_se
/* If set, gfc_conv_procedure_call does not put byref calls into se->pre. */ /* If set, gfc_conv_procedure_call does not put byref calls into se->pre. */
unsigned no_function_call:1; unsigned no_function_call:1;
/* If set, we will force the creation of a temporary. Useful to disable
non-copying procedure argument passing optimizations, when some function
args alias. */
unsigned force_tmp:1;
/* Scalarization parameters. */ /* Scalarization parameters. */
struct gfc_se *parent; struct gfc_se *parent;
struct gfc_ss *ss; struct gfc_ss *ss;

View File

@ -1,3 +1,14 @@
2010-09-21 Mikael Morin <mikael@gcc.gnu.org>
PR fortran/45648
* gfortran.dg/inline_transpose_1.f90: Update temporary's locations
and counts. Add non-elemental function call check.
PR fortran/45648
* gfortran.dg/inline_transpose_1.f90: Add function calls with aliasing
arguments checks. Update temporary counts.
* gfortran.dg/transpose_optimization_1.f90: New.
2010-09-21 Nicola Pero <nicola.pero@meta-innovation.com> 2010-09-21 Nicola Pero <nicola.pero@meta-innovation.com>
Merge from 'apple/trunk' branch on FSF servers. Merge from 'apple/trunk' branch on FSF servers.

View File

@ -61,10 +61,10 @@
if (u /= v) call abort if (u /= v) call abort
a = foo(transpose(c)) ! Unnecessary { dg-warning "Creating array temporary" } a = foo(transpose(c))
if (any(a /= p+1)) call abort if (any(a /= p+1)) call abort
write(u,*) foo(transpose(c)) ! 2 temps, should be 1 { dg-warning "Creating array temporary" } write(u,*) foo(transpose(c)) ! { dg-warning "Creating array temporary" }
write(v,*) p+1 write(v,*) p+1
if (u /= v) call abort if (u /= v) call abort
@ -77,10 +77,10 @@
if (u /= v) call abort if (u /= v) call abort
e = foo(transpose(e)) ! 2 temps, should be 1 { dg-warning "Creating array temporary" } e = foo(transpose(e)) ! { dg-warning "Creating array temporary" }
if (any(e /= 2*s+1)) call abort if (any(e /= 2*s+1)) call abort
write(u,*) transpose(foo(transpose(e))-1) ! 2 temps, should be 1 { dg-warning "Creating array temporary" } write(u,*) transpose(foo(transpose(e))-1) ! { dg-warning "Creating array temporary" }
write(v,*) 2*s+1 write(v,*) 2*s+1
if (u /= v) call abort if (u /= v) call abort
@ -141,28 +141,46 @@
if (u /= v) call abort if (u /= v) call abort
if (any(transpose(matmul(a,c)) /= matmul(transpose(c), transpose(a)))) call abort ! 4 temps, should be 2 { dg-warning "Creating array temporary" } if (any(transpose(matmul(a,c)) /= matmul(transpose(c), transpose(a)))) call abort ! 2 temps { dg-warning "Creating array temporary" }
write(u,*) transpose(matmul(a,c)) ! { dg-warning "Creating array temporary" } write(u,*) transpose(matmul(a,c)) ! { dg-warning "Creating array temporary" }
write(v,*) matmul(transpose(c), transpose(a)) ! 3 temps, should be 1 { dg-warning "Creating array temporary" } write(v,*) matmul(transpose(c), transpose(a)) ! { dg-warning "Creating array temporary" }
if (u /= v) call abort if (u /= v) call abort
if (any(transpose(matmul(e,a)) /= matmul(transpose(a), transpose(e)))) call abort ! 4 temps, should be 2 { dg-warning "Creating array temporary" } if (any(transpose(matmul(e,a)) /= matmul(transpose(a), transpose(e)))) call abort ! 2 temps { dg-warning "Creating array temporary" }
write(u,*) transpose(matmul(e,a)) ! { dg-warning "Creating array temporary" } write(u,*) transpose(matmul(e,a)) ! { dg-warning "Creating array temporary" }
write(v,*) matmul(transpose(a), transpose(e)) ! 3 temps, should be 1 { dg-warning "Creating array temporary" } write(v,*) matmul(transpose(a), transpose(e)) ! { dg-warning "Creating array temporary" }
if (u /= v) call abort if (u /= v) call abort
call baz (transpose(a)) ! Unnecessary { dg-warning "Creating array temporary" } call baz (transpose(a))
call toto (f, transpose (e))
if (any (f /= 4 * s + 12)) call abort
call toto (f, transpose (f)) ! { dg-warning "Creating array temporary" } call toto1 (a, transpose (c))
if (any (f /= 8 * r + 24)) call abort if (any (a /= 2 * p + 12)) call abort
call toto1 (e, transpose (e)) ! { dg-warning "Creating array temporary" }
if (any (e /= 4 * s + 12)) call abort
call toto2 (c, transpose (a))
if (any (c /= 2 * q + 13)) call abort
call toto2 (e, transpose(e)) ! { dg-warning "Creating array temporary" }
if (any (e /= 4 * r + 13)) call abort
call toto2 (e, transpose(transpose(e))) ! { dg-warning "Creating array temporary" }
if (any (e /= 4 * r + 14)) call abort
call toto3 (e, transpose(e))
if (any (e /= 4 * r + 14)) call abort
call titi (nx, e, transpose(e)) ! { dg-warning "Creating array temporary" }
if (any (e /= 4 * s + 17)) call abort
contains contains
@ -182,22 +200,38 @@
integer, intent(in) :: x(:,:) integer, intent(in) :: x(:,:)
end subroutine baz end subroutine baz
elemental subroutine toto (x, y) elemental subroutine toto1 (x, y)
integer, intent(out) :: x integer, intent(out) :: x
integer, intent(in) :: y integer, intent(in) :: y
x = y + y x = y + y
end subroutine toto end subroutine toto1
subroutine toto2 (x, y)
integer, dimension(:,:), intent(out) :: x
integer, dimension(:,:), intent(in) :: y
x = y + 1
end subroutine toto2
subroutine toto3 (x, y)
integer, dimension(:,:), intent(in) :: x, y
end subroutine toto3
end end
subroutine titi (n, x, y)
integer :: n, x(n,n), y(n,n)
x = y + 3
end subroutine titi
! No call to transpose ! No call to transpose
! { dg-final { scan-tree-dump-times "_gfortran_transpose" 0 "original" } } ! { dg-final { scan-tree-dump-times "_gfortran_transpose" 0 "original" } }
! !
! 34 temporaries ! 24 temporaries
! { dg-final { scan-tree-dump-times "struct\[^\\n\]*atmp" 34 "original" } } ! { dg-final { scan-tree-dump-times "struct\[^\\n\]*atmp" 24 "original" } }
! !
! 2 tests optimized out ! 2 tests optimized out
! { dg-final { scan-tree-dump-times "_gfortran_abort" 34 "original" } } ! { dg-final { scan-tree-dump-times "_gfortran_abort" 39 "original" } }
! { # Commented out as failing at -O0: dg-final { scan-tree-dump-times "_gfortran_abort" 32 "optimized" } } ! { # Commented out as failing at -O0: dg-final { scan-tree-dump-times "_gfortran_abort" 37 "optimized" } }
! !
! cleanup ! cleanup
! { dg-final { cleanup-tree-dump "original" } } ! { dg-final { cleanup-tree-dump "original" } }

View File

@ -0,0 +1,106 @@
! { dg-do compile }
! { dg-options "-Warray-temporaries -fdump-tree-original" }
!
! PR fortran/45648
! Non-copying descriptor transpose optimization (for function call args).
!
! Contributed by Richard Sandiford <richard@codesourcery.com>
module foo
interface
subroutine ext1 (a, b)
real, intent (in), dimension (:, :) :: a, b
end subroutine ext1
subroutine ext2 (a, b)
real, intent (in), dimension (:, :) :: a
real, intent (out), dimension (:, :) :: b
end subroutine ext2
subroutine ext3 (a, b)
real, dimension (:, :) :: a, b
end subroutine ext3
end interface
contains
! No temporary needed here.
subroutine test1 (n, a, b, c)
integer :: n
real, dimension (n, n) :: a, b, c
a = matmul (transpose (b), c)
end subroutine test1
! No temporary either, as we know the arguments to matmul are intent(in)
subroutine test2 (n, a, b)
integer :: n
real, dimension (n, n) :: a, b
a = matmul (transpose (b), b)
end subroutine test2
! No temporary needed.
subroutine test3 (n, a, b, c)
integer :: n
real, dimension (n, n) :: a, c
real, dimension (n+4, n+4) :: b
a = matmul (transpose (b (2:n+1, 3:n+2)), c)
end subroutine test3
! A temporary is needed for the result of either the transpose or matmul.
subroutine test4 (n, a, b)
integer :: n
real, dimension (n, n) :: a, b
a = matmul (transpose (a), b) ! { dg-warning "Creating array temporary" }
end subroutine test4
! The temporary is needed here since the second argument to imp1
! has unknown intent.
subroutine test5 (n, a)
integer :: n
real, dimension (n, n) :: a
call imp1 (transpose (a), a) ! { dg-warning "Creating array temporary" }
end subroutine test5
! No temporaries are needed here; imp1 can't modify either argument.
! We have to pack the arguments, however.
subroutine test6 (n, a, b)
integer :: n
real, dimension (n, n) :: a, b
call imp1 (transpose (a), transpose (b)) ! { dg-warning "Creating array temporary" }
end subroutine test6
! No temporaries are needed here; imp1 can't modify either argument.
! We don't have to pack the arguments.
subroutine test6_bis (n, a, b)
integer :: n
real, dimension (n, n) :: a, b
call ext3 (transpose (a), transpose (b))
end subroutine test6_bis
! No temporary is neede here; the second argument is intent(in).
subroutine test7 (n, a)
integer :: n
real, dimension (n, n) :: a
call ext1 (transpose (a), a)
end subroutine test7
! The temporary is needed here though.
subroutine test8 (n, a)
integer :: n
real, dimension (n, n) :: a
call ext2 (transpose (a), a) ! { dg-warning "Creating array temporary" }
end subroutine test8
! Silly, but we don't need any temporaries here.
subroutine test9 (n, a)
integer :: n
real, dimension (n, n) :: a
call ext1 (transpose (transpose (a)), a)
end subroutine test9
! The outer transpose needs a temporary; the inner one doesn't.
subroutine test10 (n, a)
integer :: n
real, dimension (n, n) :: a
call ext2 (transpose (transpose (a)), a) ! { dg-warning "Creating array temporary" }
end subroutine test10
end module foo
! { dg-final { scan-tree-dump-times "struct\[^\\n\]*atmp" 4 "original" } }
! { dg-final { cleanup-tree-dump "original" } }