xref: /netbsd-src/external/gpl3/gcc.old/dist/libgfortran/generated/iany_i4.c (revision 4c3eb207d36f67d31994830c0a694161fc1ca39b)
1627f7eb2Smrg /* Implementation of the IANY intrinsic
2*4c3eb207Smrg    Copyright (C) 2010-2020 Free Software Foundation, Inc.
3627f7eb2Smrg    Contributed by Tobias Burnus <burnus@net-b.de>
4627f7eb2Smrg 
5627f7eb2Smrg This file is part of the GNU Fortran runtime library (libgfortran).
6627f7eb2Smrg 
7627f7eb2Smrg Libgfortran is free software; you can redistribute it and/or
8627f7eb2Smrg modify it under the terms of the GNU General Public
9627f7eb2Smrg License as published by the Free Software Foundation; either
10627f7eb2Smrg version 3 of the License, or (at your option) any later version.
11627f7eb2Smrg 
12627f7eb2Smrg Libgfortran is distributed in the hope that it will be useful,
13627f7eb2Smrg but WITHOUT ANY WARRANTY; without even the implied warranty of
14627f7eb2Smrg MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15627f7eb2Smrg GNU General Public License for more details.
16627f7eb2Smrg 
17627f7eb2Smrg Under Section 7 of GPL version 3, you are granted additional
18627f7eb2Smrg permissions described in the GCC Runtime Library Exception, version
19627f7eb2Smrg 3.1, as published by the Free Software Foundation.
20627f7eb2Smrg 
21627f7eb2Smrg You should have received a copy of the GNU General Public License and
22627f7eb2Smrg a copy of the GCC Runtime Library Exception along with this program;
23627f7eb2Smrg see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see
24627f7eb2Smrg <http://www.gnu.org/licenses/>.  */
25627f7eb2Smrg 
26627f7eb2Smrg #include "libgfortran.h"
27627f7eb2Smrg 
28627f7eb2Smrg 
29627f7eb2Smrg #if defined (HAVE_GFC_INTEGER_4) && defined (HAVE_GFC_INTEGER_4)
30627f7eb2Smrg 
31627f7eb2Smrg 
32627f7eb2Smrg extern void iany_i4 (gfc_array_i4 * const restrict,
33627f7eb2Smrg 	gfc_array_i4 * const restrict, const index_type * const restrict);
34627f7eb2Smrg export_proto(iany_i4);
35627f7eb2Smrg 
36627f7eb2Smrg void
iany_i4(gfc_array_i4 * const restrict retarray,gfc_array_i4 * const restrict array,const index_type * const restrict pdim)37627f7eb2Smrg iany_i4 (gfc_array_i4 * const restrict retarray,
38627f7eb2Smrg 	gfc_array_i4 * const restrict array,
39627f7eb2Smrg 	const index_type * const restrict pdim)
40627f7eb2Smrg {
41627f7eb2Smrg   index_type count[GFC_MAX_DIMENSIONS];
42627f7eb2Smrg   index_type extent[GFC_MAX_DIMENSIONS];
43627f7eb2Smrg   index_type sstride[GFC_MAX_DIMENSIONS];
44627f7eb2Smrg   index_type dstride[GFC_MAX_DIMENSIONS];
45627f7eb2Smrg   const GFC_INTEGER_4 * restrict base;
46627f7eb2Smrg   GFC_INTEGER_4 * restrict dest;
47627f7eb2Smrg   index_type rank;
48627f7eb2Smrg   index_type n;
49627f7eb2Smrg   index_type len;
50627f7eb2Smrg   index_type delta;
51627f7eb2Smrg   index_type dim;
52627f7eb2Smrg   int continue_loop;
53627f7eb2Smrg 
54627f7eb2Smrg   /* Make dim zero based to avoid confusion.  */
55627f7eb2Smrg   rank = GFC_DESCRIPTOR_RANK (array) - 1;
56627f7eb2Smrg   dim = (*pdim) - 1;
57627f7eb2Smrg 
58627f7eb2Smrg   if (unlikely (dim < 0 || dim > rank))
59627f7eb2Smrg     {
60627f7eb2Smrg       runtime_error ("Dim argument incorrect in IANY intrinsic: "
61627f7eb2Smrg  		     "is %ld, should be between 1 and %ld",
62627f7eb2Smrg 		     (long int) dim + 1, (long int) rank + 1);
63627f7eb2Smrg     }
64627f7eb2Smrg 
65627f7eb2Smrg   len = GFC_DESCRIPTOR_EXTENT(array,dim);
66627f7eb2Smrg   if (len < 0)
67627f7eb2Smrg     len = 0;
68627f7eb2Smrg   delta = GFC_DESCRIPTOR_STRIDE(array,dim);
69627f7eb2Smrg 
70627f7eb2Smrg   for (n = 0; n < dim; n++)
71627f7eb2Smrg     {
72627f7eb2Smrg       sstride[n] = GFC_DESCRIPTOR_STRIDE(array,n);
73627f7eb2Smrg       extent[n] = GFC_DESCRIPTOR_EXTENT(array,n);
74627f7eb2Smrg 
75627f7eb2Smrg       if (extent[n] < 0)
76627f7eb2Smrg 	extent[n] = 0;
77627f7eb2Smrg     }
78627f7eb2Smrg   for (n = dim; n < rank; n++)
79627f7eb2Smrg     {
80627f7eb2Smrg       sstride[n] = GFC_DESCRIPTOR_STRIDE(array, n + 1);
81627f7eb2Smrg       extent[n] = GFC_DESCRIPTOR_EXTENT(array, n + 1);
82627f7eb2Smrg 
83627f7eb2Smrg       if (extent[n] < 0)
84627f7eb2Smrg 	extent[n] = 0;
85627f7eb2Smrg     }
86627f7eb2Smrg 
87627f7eb2Smrg   if (retarray->base_addr == NULL)
88627f7eb2Smrg     {
89627f7eb2Smrg       size_t alloc_size, str;
90627f7eb2Smrg 
91627f7eb2Smrg       for (n = 0; n < rank; n++)
92627f7eb2Smrg 	{
93627f7eb2Smrg 	  if (n == 0)
94627f7eb2Smrg 	    str = 1;
95627f7eb2Smrg 	  else
96627f7eb2Smrg 	    str = GFC_DESCRIPTOR_STRIDE(retarray,n-1) * extent[n-1];
97627f7eb2Smrg 
98627f7eb2Smrg 	  GFC_DIMENSION_SET(retarray->dim[n], 0, extent[n] - 1, str);
99627f7eb2Smrg 
100627f7eb2Smrg 	}
101627f7eb2Smrg 
102627f7eb2Smrg       retarray->offset = 0;
103627f7eb2Smrg       retarray->dtype.rank = rank;
104627f7eb2Smrg 
105627f7eb2Smrg       alloc_size = GFC_DESCRIPTOR_STRIDE(retarray,rank-1) * extent[rank-1];
106627f7eb2Smrg 
107627f7eb2Smrg       retarray->base_addr = xmallocarray (alloc_size, sizeof (GFC_INTEGER_4));
108627f7eb2Smrg       if (alloc_size == 0)
109627f7eb2Smrg 	{
110627f7eb2Smrg 	  /* Make sure we have a zero-sized array.  */
111627f7eb2Smrg 	  GFC_DIMENSION_SET(retarray->dim[0], 0, -1, 1);
112627f7eb2Smrg 	  return;
113627f7eb2Smrg 
114627f7eb2Smrg 	}
115627f7eb2Smrg     }
116627f7eb2Smrg   else
117627f7eb2Smrg     {
118627f7eb2Smrg       if (rank != GFC_DESCRIPTOR_RANK (retarray))
119627f7eb2Smrg 	runtime_error ("rank of return array incorrect in"
120627f7eb2Smrg 		       " IANY intrinsic: is %ld, should be %ld",
121627f7eb2Smrg 		       (long int) (GFC_DESCRIPTOR_RANK (retarray)),
122627f7eb2Smrg 		       (long int) rank);
123627f7eb2Smrg 
124627f7eb2Smrg       if (unlikely (compile_options.bounds_check))
125627f7eb2Smrg 	bounds_ifunction_return ((array_t *) retarray, extent,
126627f7eb2Smrg 				 "return value", "IANY");
127627f7eb2Smrg     }
128627f7eb2Smrg 
129627f7eb2Smrg   for (n = 0; n < rank; n++)
130627f7eb2Smrg     {
131627f7eb2Smrg       count[n] = 0;
132627f7eb2Smrg       dstride[n] = GFC_DESCRIPTOR_STRIDE(retarray,n);
133627f7eb2Smrg       if (extent[n] <= 0)
134627f7eb2Smrg 	return;
135627f7eb2Smrg     }
136627f7eb2Smrg 
137627f7eb2Smrg   base = array->base_addr;
138627f7eb2Smrg   dest = retarray->base_addr;
139627f7eb2Smrg 
140627f7eb2Smrg   continue_loop = 1;
141627f7eb2Smrg   while (continue_loop)
142627f7eb2Smrg     {
143627f7eb2Smrg       const GFC_INTEGER_4 * restrict src;
144627f7eb2Smrg       GFC_INTEGER_4 result;
145627f7eb2Smrg       src = base;
146627f7eb2Smrg       {
147627f7eb2Smrg 
148627f7eb2Smrg   result = 0;
149627f7eb2Smrg 	if (len <= 0)
150627f7eb2Smrg 	  *dest = 0;
151627f7eb2Smrg 	else
152627f7eb2Smrg 	  {
153627f7eb2Smrg #if ! defined HAVE_BACK_ARG
154627f7eb2Smrg 	    for (n = 0; n < len; n++, src += delta)
155627f7eb2Smrg 	      {
156627f7eb2Smrg #endif
157627f7eb2Smrg 
158627f7eb2Smrg   result |= *src;
159627f7eb2Smrg 	      }
160627f7eb2Smrg 
161627f7eb2Smrg 	    *dest = result;
162627f7eb2Smrg 	  }
163627f7eb2Smrg       }
164627f7eb2Smrg       /* Advance to the next element.  */
165627f7eb2Smrg       count[0]++;
166627f7eb2Smrg       base += sstride[0];
167627f7eb2Smrg       dest += dstride[0];
168627f7eb2Smrg       n = 0;
169627f7eb2Smrg       while (count[n] == extent[n])
170627f7eb2Smrg 	{
171627f7eb2Smrg 	  /* When we get to the end of a dimension, reset it and increment
172627f7eb2Smrg 	     the next dimension.  */
173627f7eb2Smrg 	  count[n] = 0;
174627f7eb2Smrg 	  /* We could precalculate these products, but this is a less
175627f7eb2Smrg 	     frequently used path so probably not worth it.  */
176627f7eb2Smrg 	  base -= sstride[n] * extent[n];
177627f7eb2Smrg 	  dest -= dstride[n] * extent[n];
178627f7eb2Smrg 	  n++;
179627f7eb2Smrg 	  if (n >= rank)
180627f7eb2Smrg 	    {
181627f7eb2Smrg 	      /* Break out of the loop.  */
182627f7eb2Smrg 	      continue_loop = 0;
183627f7eb2Smrg 	      break;
184627f7eb2Smrg 	    }
185627f7eb2Smrg 	  else
186627f7eb2Smrg 	    {
187627f7eb2Smrg 	      count[n]++;
188627f7eb2Smrg 	      base += sstride[n];
189627f7eb2Smrg 	      dest += dstride[n];
190627f7eb2Smrg 	    }
191627f7eb2Smrg 	}
192627f7eb2Smrg     }
193627f7eb2Smrg }
194627f7eb2Smrg 
195627f7eb2Smrg 
196627f7eb2Smrg extern void miany_i4 (gfc_array_i4 * const restrict,
197627f7eb2Smrg 	gfc_array_i4 * const restrict, const index_type * const restrict,
198627f7eb2Smrg 	gfc_array_l1 * const restrict);
199627f7eb2Smrg export_proto(miany_i4);
200627f7eb2Smrg 
201627f7eb2Smrg void
miany_i4(gfc_array_i4 * const restrict retarray,gfc_array_i4 * const restrict array,const index_type * const restrict pdim,gfc_array_l1 * const restrict mask)202627f7eb2Smrg miany_i4 (gfc_array_i4 * const restrict retarray,
203627f7eb2Smrg 	gfc_array_i4 * const restrict array,
204627f7eb2Smrg 	const index_type * const restrict pdim,
205627f7eb2Smrg 	gfc_array_l1 * const restrict mask)
206627f7eb2Smrg {
207627f7eb2Smrg   index_type count[GFC_MAX_DIMENSIONS];
208627f7eb2Smrg   index_type extent[GFC_MAX_DIMENSIONS];
209627f7eb2Smrg   index_type sstride[GFC_MAX_DIMENSIONS];
210627f7eb2Smrg   index_type dstride[GFC_MAX_DIMENSIONS];
211627f7eb2Smrg   index_type mstride[GFC_MAX_DIMENSIONS];
212627f7eb2Smrg   GFC_INTEGER_4 * restrict dest;
213627f7eb2Smrg   const GFC_INTEGER_4 * restrict base;
214627f7eb2Smrg   const GFC_LOGICAL_1 * restrict mbase;
215627f7eb2Smrg   index_type rank;
216627f7eb2Smrg   index_type dim;
217627f7eb2Smrg   index_type n;
218627f7eb2Smrg   index_type len;
219627f7eb2Smrg   index_type delta;
220627f7eb2Smrg   index_type mdelta;
221627f7eb2Smrg   int mask_kind;
222627f7eb2Smrg 
223627f7eb2Smrg   if (mask == NULL)
224627f7eb2Smrg     {
225627f7eb2Smrg #ifdef HAVE_BACK_ARG
226627f7eb2Smrg       iany_i4 (retarray, array, pdim, back);
227627f7eb2Smrg #else
228627f7eb2Smrg       iany_i4 (retarray, array, pdim);
229627f7eb2Smrg #endif
230627f7eb2Smrg       return;
231627f7eb2Smrg     }
232627f7eb2Smrg 
233627f7eb2Smrg   dim = (*pdim) - 1;
234627f7eb2Smrg   rank = GFC_DESCRIPTOR_RANK (array) - 1;
235627f7eb2Smrg 
236627f7eb2Smrg 
237627f7eb2Smrg   if (unlikely (dim < 0 || dim > rank))
238627f7eb2Smrg     {
239627f7eb2Smrg       runtime_error ("Dim argument incorrect in IANY intrinsic: "
240627f7eb2Smrg  		     "is %ld, should be between 1 and %ld",
241627f7eb2Smrg 		     (long int) dim + 1, (long int) rank + 1);
242627f7eb2Smrg     }
243627f7eb2Smrg 
244627f7eb2Smrg   len = GFC_DESCRIPTOR_EXTENT(array,dim);
245627f7eb2Smrg   if (len <= 0)
246627f7eb2Smrg     return;
247627f7eb2Smrg 
248627f7eb2Smrg   mbase = mask->base_addr;
249627f7eb2Smrg 
250627f7eb2Smrg   mask_kind = GFC_DESCRIPTOR_SIZE (mask);
251627f7eb2Smrg 
252627f7eb2Smrg   if (mask_kind == 1 || mask_kind == 2 || mask_kind == 4 || mask_kind == 8
253627f7eb2Smrg #ifdef HAVE_GFC_LOGICAL_16
254627f7eb2Smrg       || mask_kind == 16
255627f7eb2Smrg #endif
256627f7eb2Smrg       )
257627f7eb2Smrg     mbase = GFOR_POINTER_TO_L1 (mbase, mask_kind);
258627f7eb2Smrg   else
259627f7eb2Smrg     runtime_error ("Funny sized logical array");
260627f7eb2Smrg 
261627f7eb2Smrg   delta = GFC_DESCRIPTOR_STRIDE(array,dim);
262627f7eb2Smrg   mdelta = GFC_DESCRIPTOR_STRIDE_BYTES(mask,dim);
263627f7eb2Smrg 
264627f7eb2Smrg   for (n = 0; n < dim; n++)
265627f7eb2Smrg     {
266627f7eb2Smrg       sstride[n] = GFC_DESCRIPTOR_STRIDE(array,n);
267627f7eb2Smrg       mstride[n] = GFC_DESCRIPTOR_STRIDE_BYTES(mask,n);
268627f7eb2Smrg       extent[n] = GFC_DESCRIPTOR_EXTENT(array,n);
269627f7eb2Smrg 
270627f7eb2Smrg       if (extent[n] < 0)
271627f7eb2Smrg 	extent[n] = 0;
272627f7eb2Smrg 
273627f7eb2Smrg     }
274627f7eb2Smrg   for (n = dim; n < rank; n++)
275627f7eb2Smrg     {
276627f7eb2Smrg       sstride[n] = GFC_DESCRIPTOR_STRIDE(array,n + 1);
277627f7eb2Smrg       mstride[n] = GFC_DESCRIPTOR_STRIDE_BYTES(mask, n + 1);
278627f7eb2Smrg       extent[n] = GFC_DESCRIPTOR_EXTENT(array, n + 1);
279627f7eb2Smrg 
280627f7eb2Smrg       if (extent[n] < 0)
281627f7eb2Smrg 	extent[n] = 0;
282627f7eb2Smrg     }
283627f7eb2Smrg 
284627f7eb2Smrg   if (retarray->base_addr == NULL)
285627f7eb2Smrg     {
286627f7eb2Smrg       size_t alloc_size, str;
287627f7eb2Smrg 
288627f7eb2Smrg       for (n = 0; n < rank; n++)
289627f7eb2Smrg 	{
290627f7eb2Smrg 	  if (n == 0)
291627f7eb2Smrg 	    str = 1;
292627f7eb2Smrg 	  else
293627f7eb2Smrg 	    str= GFC_DESCRIPTOR_STRIDE(retarray,n-1) * extent[n-1];
294627f7eb2Smrg 
295627f7eb2Smrg 	  GFC_DIMENSION_SET(retarray->dim[n], 0, extent[n] - 1, str);
296627f7eb2Smrg 
297627f7eb2Smrg 	}
298627f7eb2Smrg 
299627f7eb2Smrg       alloc_size = GFC_DESCRIPTOR_STRIDE(retarray,rank-1) * extent[rank-1];
300627f7eb2Smrg 
301627f7eb2Smrg       retarray->offset = 0;
302627f7eb2Smrg       retarray->dtype.rank = rank;
303627f7eb2Smrg 
304627f7eb2Smrg       if (alloc_size == 0)
305627f7eb2Smrg 	{
306627f7eb2Smrg 	  /* Make sure we have a zero-sized array.  */
307627f7eb2Smrg 	  GFC_DIMENSION_SET(retarray->dim[0], 0, -1, 1);
308627f7eb2Smrg 	  return;
309627f7eb2Smrg 	}
310627f7eb2Smrg       else
311627f7eb2Smrg 	retarray->base_addr = xmallocarray (alloc_size, sizeof (GFC_INTEGER_4));
312627f7eb2Smrg 
313627f7eb2Smrg     }
314627f7eb2Smrg   else
315627f7eb2Smrg     {
316627f7eb2Smrg       if (rank != GFC_DESCRIPTOR_RANK (retarray))
317627f7eb2Smrg 	runtime_error ("rank of return array incorrect in IANY intrinsic");
318627f7eb2Smrg 
319627f7eb2Smrg       if (unlikely (compile_options.bounds_check))
320627f7eb2Smrg 	{
321627f7eb2Smrg 	  bounds_ifunction_return ((array_t *) retarray, extent,
322627f7eb2Smrg 				   "return value", "IANY");
323627f7eb2Smrg 	  bounds_equal_extents ((array_t *) mask, (array_t *) array,
324627f7eb2Smrg 	  			"MASK argument", "IANY");
325627f7eb2Smrg 	}
326627f7eb2Smrg     }
327627f7eb2Smrg 
328627f7eb2Smrg   for (n = 0; n < rank; n++)
329627f7eb2Smrg     {
330627f7eb2Smrg       count[n] = 0;
331627f7eb2Smrg       dstride[n] = GFC_DESCRIPTOR_STRIDE(retarray,n);
332627f7eb2Smrg       if (extent[n] <= 0)
333627f7eb2Smrg 	return;
334627f7eb2Smrg     }
335627f7eb2Smrg 
336627f7eb2Smrg   dest = retarray->base_addr;
337627f7eb2Smrg   base = array->base_addr;
338627f7eb2Smrg 
339627f7eb2Smrg   while (base)
340627f7eb2Smrg     {
341627f7eb2Smrg       const GFC_INTEGER_4 * restrict src;
342627f7eb2Smrg       const GFC_LOGICAL_1 * restrict msrc;
343627f7eb2Smrg       GFC_INTEGER_4 result;
344627f7eb2Smrg       src = base;
345627f7eb2Smrg       msrc = mbase;
346627f7eb2Smrg       {
347627f7eb2Smrg 
348627f7eb2Smrg   result = 0;
349627f7eb2Smrg 	for (n = 0; n < len; n++, src += delta, msrc += mdelta)
350627f7eb2Smrg 	  {
351627f7eb2Smrg 
352627f7eb2Smrg   if (*msrc)
353627f7eb2Smrg     result |= *src;
354627f7eb2Smrg 	  }
355627f7eb2Smrg 	*dest = result;
356627f7eb2Smrg       }
357627f7eb2Smrg       /* Advance to the next element.  */
358627f7eb2Smrg       count[0]++;
359627f7eb2Smrg       base += sstride[0];
360627f7eb2Smrg       mbase += mstride[0];
361627f7eb2Smrg       dest += dstride[0];
362627f7eb2Smrg       n = 0;
363627f7eb2Smrg       while (count[n] == extent[n])
364627f7eb2Smrg 	{
365627f7eb2Smrg 	  /* When we get to the end of a dimension, reset it and increment
366627f7eb2Smrg 	     the next dimension.  */
367627f7eb2Smrg 	  count[n] = 0;
368627f7eb2Smrg 	  /* We could precalculate these products, but this is a less
369627f7eb2Smrg 	     frequently used path so probably not worth it.  */
370627f7eb2Smrg 	  base -= sstride[n] * extent[n];
371627f7eb2Smrg 	  mbase -= mstride[n] * extent[n];
372627f7eb2Smrg 	  dest -= dstride[n] * extent[n];
373627f7eb2Smrg 	  n++;
374627f7eb2Smrg 	  if (n >= rank)
375627f7eb2Smrg 	    {
376627f7eb2Smrg 	      /* Break out of the loop.  */
377627f7eb2Smrg 	      base = NULL;
378627f7eb2Smrg 	      break;
379627f7eb2Smrg 	    }
380627f7eb2Smrg 	  else
381627f7eb2Smrg 	    {
382627f7eb2Smrg 	      count[n]++;
383627f7eb2Smrg 	      base += sstride[n];
384627f7eb2Smrg 	      mbase += mstride[n];
385627f7eb2Smrg 	      dest += dstride[n];
386627f7eb2Smrg 	    }
387627f7eb2Smrg 	}
388627f7eb2Smrg     }
389627f7eb2Smrg }
390627f7eb2Smrg 
391627f7eb2Smrg 
392627f7eb2Smrg extern void siany_i4 (gfc_array_i4 * const restrict,
393627f7eb2Smrg 	gfc_array_i4 * const restrict, const index_type * const restrict,
394627f7eb2Smrg 	GFC_LOGICAL_4 *);
395627f7eb2Smrg export_proto(siany_i4);
396627f7eb2Smrg 
397627f7eb2Smrg void
siany_i4(gfc_array_i4 * const restrict retarray,gfc_array_i4 * const restrict array,const index_type * const restrict pdim,GFC_LOGICAL_4 * mask)398627f7eb2Smrg siany_i4 (gfc_array_i4 * const restrict retarray,
399627f7eb2Smrg 	gfc_array_i4 * const restrict array,
400627f7eb2Smrg 	const index_type * const restrict pdim,
401627f7eb2Smrg 	GFC_LOGICAL_4 * mask)
402627f7eb2Smrg {
403627f7eb2Smrg   index_type count[GFC_MAX_DIMENSIONS];
404627f7eb2Smrg   index_type extent[GFC_MAX_DIMENSIONS];
405627f7eb2Smrg   index_type dstride[GFC_MAX_DIMENSIONS];
406627f7eb2Smrg   GFC_INTEGER_4 * restrict dest;
407627f7eb2Smrg   index_type rank;
408627f7eb2Smrg   index_type n;
409627f7eb2Smrg   index_type dim;
410627f7eb2Smrg 
411627f7eb2Smrg 
412627f7eb2Smrg   if (mask == NULL || *mask)
413627f7eb2Smrg     {
414627f7eb2Smrg #ifdef HAVE_BACK_ARG
415627f7eb2Smrg       iany_i4 (retarray, array, pdim, back);
416627f7eb2Smrg #else
417627f7eb2Smrg       iany_i4 (retarray, array, pdim);
418627f7eb2Smrg #endif
419627f7eb2Smrg       return;
420627f7eb2Smrg     }
421627f7eb2Smrg   /* Make dim zero based to avoid confusion.  */
422627f7eb2Smrg   dim = (*pdim) - 1;
423627f7eb2Smrg   rank = GFC_DESCRIPTOR_RANK (array) - 1;
424627f7eb2Smrg 
425627f7eb2Smrg   if (unlikely (dim < 0 || dim > rank))
426627f7eb2Smrg     {
427627f7eb2Smrg       runtime_error ("Dim argument incorrect in IANY intrinsic: "
428627f7eb2Smrg  		     "is %ld, should be between 1 and %ld",
429627f7eb2Smrg 		     (long int) dim + 1, (long int) rank + 1);
430627f7eb2Smrg     }
431627f7eb2Smrg 
432627f7eb2Smrg   for (n = 0; n < dim; n++)
433627f7eb2Smrg     {
434627f7eb2Smrg       extent[n] = GFC_DESCRIPTOR_EXTENT(array,n);
435627f7eb2Smrg 
436627f7eb2Smrg       if (extent[n] <= 0)
437627f7eb2Smrg 	extent[n] = 0;
438627f7eb2Smrg     }
439627f7eb2Smrg 
440627f7eb2Smrg   for (n = dim; n < rank; n++)
441627f7eb2Smrg     {
442627f7eb2Smrg       extent[n] =
443627f7eb2Smrg 	GFC_DESCRIPTOR_EXTENT(array,n + 1);
444627f7eb2Smrg 
445627f7eb2Smrg       if (extent[n] <= 0)
446627f7eb2Smrg 	extent[n] = 0;
447627f7eb2Smrg     }
448627f7eb2Smrg 
449627f7eb2Smrg   if (retarray->base_addr == NULL)
450627f7eb2Smrg     {
451627f7eb2Smrg       size_t alloc_size, str;
452627f7eb2Smrg 
453627f7eb2Smrg       for (n = 0; n < rank; n++)
454627f7eb2Smrg 	{
455627f7eb2Smrg 	  if (n == 0)
456627f7eb2Smrg 	    str = 1;
457627f7eb2Smrg 	  else
458627f7eb2Smrg 	    str = GFC_DESCRIPTOR_STRIDE(retarray,n-1) * extent[n-1];
459627f7eb2Smrg 
460627f7eb2Smrg 	  GFC_DIMENSION_SET(retarray->dim[n], 0, extent[n] - 1, str);
461627f7eb2Smrg 
462627f7eb2Smrg 	}
463627f7eb2Smrg 
464627f7eb2Smrg       retarray->offset = 0;
465627f7eb2Smrg       retarray->dtype.rank = rank;
466627f7eb2Smrg 
467627f7eb2Smrg       alloc_size = GFC_DESCRIPTOR_STRIDE(retarray,rank-1) * extent[rank-1];
468627f7eb2Smrg 
469627f7eb2Smrg       if (alloc_size == 0)
470627f7eb2Smrg 	{
471627f7eb2Smrg 	  /* Make sure we have a zero-sized array.  */
472627f7eb2Smrg 	  GFC_DIMENSION_SET(retarray->dim[0], 0, -1, 1);
473627f7eb2Smrg 	  return;
474627f7eb2Smrg 	}
475627f7eb2Smrg       else
476627f7eb2Smrg 	retarray->base_addr = xmallocarray (alloc_size, sizeof (GFC_INTEGER_4));
477627f7eb2Smrg     }
478627f7eb2Smrg   else
479627f7eb2Smrg     {
480627f7eb2Smrg       if (rank != GFC_DESCRIPTOR_RANK (retarray))
481627f7eb2Smrg 	runtime_error ("rank of return array incorrect in"
482627f7eb2Smrg 		       " IANY intrinsic: is %ld, should be %ld",
483627f7eb2Smrg 		       (long int) (GFC_DESCRIPTOR_RANK (retarray)),
484627f7eb2Smrg 		       (long int) rank);
485627f7eb2Smrg 
486627f7eb2Smrg       if (unlikely (compile_options.bounds_check))
487627f7eb2Smrg 	{
488627f7eb2Smrg 	  for (n=0; n < rank; n++)
489627f7eb2Smrg 	    {
490627f7eb2Smrg 	      index_type ret_extent;
491627f7eb2Smrg 
492627f7eb2Smrg 	      ret_extent = GFC_DESCRIPTOR_EXTENT(retarray,n);
493627f7eb2Smrg 	      if (extent[n] != ret_extent)
494627f7eb2Smrg 		runtime_error ("Incorrect extent in return value of"
495627f7eb2Smrg 			       " IANY intrinsic in dimension %ld:"
496627f7eb2Smrg 			       " is %ld, should be %ld", (long int) n + 1,
497627f7eb2Smrg 			       (long int) ret_extent, (long int) extent[n]);
498627f7eb2Smrg 	    }
499627f7eb2Smrg 	}
500627f7eb2Smrg     }
501627f7eb2Smrg 
502627f7eb2Smrg   for (n = 0; n < rank; n++)
503627f7eb2Smrg     {
504627f7eb2Smrg       count[n] = 0;
505627f7eb2Smrg       dstride[n] = GFC_DESCRIPTOR_STRIDE(retarray,n);
506627f7eb2Smrg     }
507627f7eb2Smrg 
508627f7eb2Smrg   dest = retarray->base_addr;
509627f7eb2Smrg 
510627f7eb2Smrg   while(1)
511627f7eb2Smrg     {
512627f7eb2Smrg       *dest = 0;
513627f7eb2Smrg       count[0]++;
514627f7eb2Smrg       dest += dstride[0];
515627f7eb2Smrg       n = 0;
516627f7eb2Smrg       while (count[n] == extent[n])
517627f7eb2Smrg 	{
518627f7eb2Smrg 	  /* When we get to the end of a dimension, reset it and increment
519627f7eb2Smrg 	     the next dimension.  */
520627f7eb2Smrg 	  count[n] = 0;
521627f7eb2Smrg 	  /* We could precalculate these products, but this is a less
522627f7eb2Smrg 	     frequently used path so probably not worth it.  */
523627f7eb2Smrg 	  dest -= dstride[n] * extent[n];
524627f7eb2Smrg 	  n++;
525627f7eb2Smrg 	  if (n >= rank)
526627f7eb2Smrg 	    return;
527627f7eb2Smrg 	  else
528627f7eb2Smrg 	    {
529627f7eb2Smrg 	      count[n]++;
530627f7eb2Smrg 	      dest += dstride[n];
531627f7eb2Smrg 	    }
532627f7eb2Smrg       	}
533627f7eb2Smrg     }
534627f7eb2Smrg }
535627f7eb2Smrg 
536627f7eb2Smrg #endif
537