1 /* Copyright (C) 1995,1996,1997,1998,2000,2001,2002,2003,2004, 2005, 2006, 2009, 2010, 2011 Free Software Foundation, Inc.
3 * This library is free software; you can redistribute it and/or
4 * modify it under the terms of the GNU Lesser General Public License
5 * as published by the Free Software Foundation; either version 3 of
6 * the License, or (at your option) any later version.
8 * This library is distributed in the hope that it will be useful, but
9 * WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * Lesser General Public License for more details.
13 * You should have received a copy of the GNU Lesser General Public
14 * License along with this library; if not, write to the Free Software
15 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
30 #include "libguile/_scm.h"
31 #include "libguile/__scm.h"
32 #include "libguile/eq.h"
33 #include "libguile/chars.h"
34 #include "libguile/eval.h"
35 #include "libguile/fports.h"
36 #include "libguile/feature.h"
37 #include "libguile/root.h"
38 #include "libguile/strings.h"
39 #include "libguile/srfi-13.h"
40 #include "libguile/srfi-4.h"
41 #include "libguile/vectors.h"
42 #include "libguile/bitvectors.h"
43 #include "libguile/bytevectors.h"
44 #include "libguile/list.h"
45 #include "libguile/dynwind.h"
46 #include "libguile/read.h"
48 #include "libguile/validate.h"
49 #include "libguile/arrays.h"
50 #include "libguile/array-map.h"
51 #include "libguile/generalized-vectors.h"
52 #include "libguile/generalized-arrays.h"
53 #include "libguile/uniform.h"
56 #define SCM_SET_ARRAY_CONTIGUOUS_FLAG(x) \
57 (SCM_SET_CELL_WORD_0 ((x), SCM_CELL_WORD_0 (x) | (SCM_I_ARRAY_FLAG_CONTIGUOUS << 16)))
58 #define SCM_CLR_ARRAY_CONTIGUOUS_FLAG(x) \
59 (SCM_SET_CELL_WORD_0 ((x), SCM_CELL_WORD_0 (x) & ~(SCM_I_ARRAY_FLAG_CONTIGUOUS << 16)))
62 SCM_DEFINE (scm_shared_array_root
, "shared-array-root", 1, 0, 0,
64 "Return the root vector of a shared array.")
65 #define FUNC_NAME s_scm_shared_array_root
67 if (SCM_I_ARRAYP (ra
))
68 return SCM_I_ARRAY_V (ra
);
69 else if (scm_is_generalized_vector (ra
))
71 scm_wrong_type_arg_msg (FUNC_NAME
, SCM_ARG1
, ra
, "array");
76 SCM_DEFINE (scm_shared_array_offset
, "shared-array-offset", 1, 0, 0,
78 "Return the root vector index of the first element in the array.")
79 #define FUNC_NAME s_scm_shared_array_offset
81 scm_t_array_handle handle
;
84 scm_array_get_handle (ra
, &handle
);
85 res
= scm_from_size_t (handle
.base
);
86 scm_array_handle_release (&handle
);
92 SCM_DEFINE (scm_shared_array_increments
, "shared-array-increments", 1, 0, 0,
94 "For each dimension, return the distance between elements in the root vector.")
95 #define FUNC_NAME s_scm_shared_array_increments
97 scm_t_array_handle handle
;
102 scm_array_get_handle (ra
, &handle
);
103 k
= scm_array_handle_rank (&handle
);
104 s
= scm_array_handle_dims (&handle
);
106 res
= scm_cons (scm_from_ssize_t (s
[k
].inc
), res
);
107 scm_array_handle_release (&handle
);
113 scm_i_make_array (int ndim
)
116 ra
= scm_cell (((scm_t_bits
) ndim
<< 17) + scm_tc7_array
,
117 (scm_t_bits
) scm_gc_malloc (sizeof (scm_i_t_array
) +
118 ndim
* sizeof (scm_t_array_dim
),
120 SCM_I_ARRAY_V (ra
) = SCM_BOOL_F
;
124 static char s_bad_spec
[] = "Bad scm_array dimension";
127 /* Increments will still need to be set. */
130 scm_i_shap2ra (SCM args
)
134 int ndim
= scm_ilength (args
);
136 scm_misc_error (NULL
, s_bad_spec
, SCM_EOL
);
138 ra
= scm_i_make_array (ndim
);
139 SCM_I_ARRAY_BASE (ra
) = 0;
140 s
= SCM_I_ARRAY_DIMS (ra
);
141 for (; !scm_is_null (args
); s
++, args
= SCM_CDR (args
))
143 spec
= SCM_CAR (args
);
144 if (scm_is_integer (spec
))
146 if (scm_to_long (spec
) < 0)
147 scm_misc_error (NULL
, s_bad_spec
, SCM_EOL
);
149 s
->ubnd
= scm_to_long (spec
) - 1;
154 if (!scm_is_pair (spec
) || !scm_is_integer (SCM_CAR (spec
)))
155 scm_misc_error (NULL
, s_bad_spec
, SCM_EOL
);
156 s
->lbnd
= scm_to_long (SCM_CAR (spec
));
158 if (!scm_is_pair (sp
)
159 || !scm_is_integer (SCM_CAR (sp
))
160 || !scm_is_null (SCM_CDR (sp
)))
161 scm_misc_error (NULL
, s_bad_spec
, SCM_EOL
);
162 s
->ubnd
= scm_to_long (SCM_CAR (sp
));
169 SCM_DEFINE (scm_make_typed_array
, "make-typed-array", 2, 0, 1,
170 (SCM type
, SCM fill
, SCM bounds
),
171 "Create and return an array of type @var{type}.")
172 #define FUNC_NAME s_scm_make_typed_array
178 ra
= scm_i_shap2ra (bounds
);
179 SCM_SET_ARRAY_CONTIGUOUS_FLAG (ra
);
180 s
= SCM_I_ARRAY_DIMS (ra
);
181 k
= SCM_I_ARRAY_NDIM (ra
);
186 SCM_ASSERT_RANGE (1, bounds
, s
[k
].lbnd
<= s
[k
].ubnd
+ 1);
187 rlen
= (s
[k
].ubnd
- s
[k
].lbnd
+ 1) * s
[k
].inc
;
190 if (scm_is_eq (fill
, SCM_UNSPECIFIED
))
191 fill
= SCM_UNDEFINED
;
194 scm_make_generalized_vector (type
, scm_from_size_t (rlen
), fill
);
196 if (1 == SCM_I_ARRAY_NDIM (ra
) && 0 == SCM_I_ARRAY_BASE (ra
))
197 if (s
->ubnd
< s
->lbnd
|| (0 == s
->lbnd
&& 1 == s
->inc
))
198 return SCM_I_ARRAY_V (ra
);
204 scm_from_contiguous_typed_array (SCM type
, SCM bounds
, const void *bytes
,
206 #define FUNC_NAME "scm_from_contiguous_typed_array"
211 scm_t_array_handle h
;
215 ra
= scm_i_shap2ra (bounds
);
216 SCM_SET_ARRAY_CONTIGUOUS_FLAG (ra
);
217 s
= SCM_I_ARRAY_DIMS (ra
);
218 k
= SCM_I_ARRAY_NDIM (ra
);
223 SCM_ASSERT_RANGE (1, bounds
, s
[k
].lbnd
<= s
[k
].ubnd
+ 1);
224 rlen
= (s
[k
].ubnd
- s
[k
].lbnd
+ 1) * s
[k
].inc
;
227 scm_make_generalized_vector (type
, scm_from_size_t (rlen
), SCM_UNDEFINED
);
230 scm_array_get_handle (ra
, &h
);
231 elts
= h
.writable_elements
;
232 sz
= scm_array_handle_uniform_element_bit_size (&h
);
233 scm_array_handle_release (&h
);
235 if (sz
>= 8 && ((sz
% 8) == 0))
237 if (byte_len
% (sz
/ 8))
238 SCM_MISC_ERROR ("byte length not a multiple of the unit size", SCM_EOL
);
239 if (byte_len
/ (sz
/ 8) != rlen
)
240 SCM_MISC_ERROR ("byte length and dimensions do not match", SCM_EOL
);
244 /* byte_len ?= ceil (rlen * sz / 8) */
245 if (byte_len
!= (rlen
* sz
+ 7) / 8)
246 SCM_MISC_ERROR ("byte length and dimensions do not match", SCM_EOL
);
249 /* an internal guile error, really */
250 SCM_MISC_ERROR ("uniform elements larger than 8 bits must fill whole bytes", SCM_EOL
);
252 memcpy (elts
, bytes
, byte_len
);
254 if (1 == SCM_I_ARRAY_NDIM (ra
) && 0 == SCM_I_ARRAY_BASE (ra
))
255 if (s
->ubnd
< s
->lbnd
|| (0 == s
->lbnd
&& 1 == s
->inc
))
256 return SCM_I_ARRAY_V (ra
);
262 scm_from_contiguous_array (SCM bounds
, const SCM
*elts
, size_t len
)
263 #define FUNC_NAME "scm_from_contiguous_array"
268 scm_t_array_handle h
;
270 ra
= scm_i_shap2ra (bounds
);
271 SCM_SET_ARRAY_CONTIGUOUS_FLAG (ra
);
272 s
= SCM_I_ARRAY_DIMS (ra
);
273 k
= SCM_I_ARRAY_NDIM (ra
);
278 SCM_ASSERT_RANGE (1, bounds
, s
[k
].lbnd
<= s
[k
].ubnd
+ 1);
279 rlen
= (s
[k
].ubnd
- s
[k
].lbnd
+ 1) * s
[k
].inc
;
282 SCM_MISC_ERROR ("element length and dimensions do not match", SCM_EOL
);
284 SCM_I_ARRAY_V (ra
) = scm_c_make_vector (rlen
, SCM_UNDEFINED
);
285 scm_array_get_handle (ra
, &h
);
286 memcpy (h
.writable_elements
, elts
, rlen
* sizeof(SCM
));
287 scm_array_handle_release (&h
);
289 if (1 == SCM_I_ARRAY_NDIM (ra
) && 0 == SCM_I_ARRAY_BASE (ra
))
290 if (s
->ubnd
< s
->lbnd
|| (0 == s
->lbnd
&& 1 == s
->inc
))
291 return SCM_I_ARRAY_V (ra
);
296 SCM_DEFINE (scm_make_array
, "make-array", 1, 0, 1,
297 (SCM fill
, SCM bounds
),
298 "Create and return an array.")
299 #define FUNC_NAME s_scm_make_array
301 return scm_make_typed_array (SCM_BOOL_T
, fill
, bounds
);
306 scm_i_ra_set_contp (SCM ra
)
308 size_t k
= SCM_I_ARRAY_NDIM (ra
);
311 long inc
= SCM_I_ARRAY_DIMS (ra
)[k
- 1].inc
;
314 if (inc
!= SCM_I_ARRAY_DIMS (ra
)[k
].inc
)
316 SCM_CLR_ARRAY_CONTIGUOUS_FLAG (ra
);
319 inc
*= (SCM_I_ARRAY_DIMS (ra
)[k
].ubnd
320 - SCM_I_ARRAY_DIMS (ra
)[k
].lbnd
+ 1);
323 SCM_SET_ARRAY_CONTIGUOUS_FLAG (ra
);
327 SCM_DEFINE (scm_make_shared_array
, "make-shared-array", 2, 0, 1,
328 (SCM oldra
, SCM mapfunc
, SCM dims
),
329 "@code{make-shared-array} can be used to create shared subarrays of other\n"
330 "arrays. The @var{mapper} is a function that translates coordinates in\n"
331 "the new array into coordinates in the old array. A @var{mapper} must be\n"
332 "linear, and its range must stay within the bounds of the old array, but\n"
333 "it can be otherwise arbitrary. A simple example:\n"
335 "(define fred (make-array #f 8 8))\n"
336 "(define freds-diagonal\n"
337 " (make-shared-array fred (lambda (i) (list i i)) 8))\n"
338 "(array-set! freds-diagonal 'foo 3)\n"
339 "(array-ref fred 3 3) @result{} foo\n"
340 "(define freds-center\n"
341 " (make-shared-array fred (lambda (i j) (list (+ 3 i) (+ 3 j))) 2 2))\n"
342 "(array-ref freds-center 0 0) @result{} foo\n"
344 #define FUNC_NAME s_scm_make_shared_array
346 scm_t_array_handle old_handle
;
352 long old_base
, old_min
, new_min
, old_max
, new_max
;
355 SCM_VALIDATE_REST_ARGUMENT (dims
);
356 SCM_VALIDATE_PROC (2, mapfunc
);
357 ra
= scm_i_shap2ra (dims
);
359 scm_array_get_handle (oldra
, &old_handle
);
361 if (SCM_I_ARRAYP (oldra
))
363 SCM_I_ARRAY_V (ra
) = SCM_I_ARRAY_V (oldra
);
364 old_base
= old_min
= old_max
= SCM_I_ARRAY_BASE (oldra
);
365 s
= scm_array_handle_dims (&old_handle
);
366 k
= scm_array_handle_rank (&old_handle
);
370 old_max
+= (s
[k
].ubnd
- s
[k
].lbnd
) * s
[k
].inc
;
372 old_min
+= (s
[k
].ubnd
- s
[k
].lbnd
) * s
[k
].inc
;
377 SCM_I_ARRAY_V (ra
) = oldra
;
378 old_base
= old_min
= 0;
379 old_max
= scm_c_generalized_vector_length (oldra
) - 1;
383 s
= SCM_I_ARRAY_DIMS (ra
);
384 for (k
= 0; k
< SCM_I_ARRAY_NDIM (ra
); k
++)
386 inds
= scm_cons (scm_from_long (s
[k
].lbnd
), inds
);
387 if (s
[k
].ubnd
< s
[k
].lbnd
)
389 if (1 == SCM_I_ARRAY_NDIM (ra
))
390 ra
= scm_make_generalized_vector (scm_array_type (ra
),
391 SCM_INUM0
, SCM_UNDEFINED
);
394 scm_make_generalized_vector (scm_array_type (ra
),
395 SCM_INUM0
, SCM_UNDEFINED
);
396 scm_array_handle_release (&old_handle
);
401 imap
= scm_apply_0 (mapfunc
, scm_reverse (inds
));
402 i
= scm_array_handle_pos (&old_handle
, imap
);
403 SCM_I_ARRAY_BASE (ra
) = new_min
= new_max
= i
+ old_base
;
405 k
= SCM_I_ARRAY_NDIM (ra
);
408 if (s
[k
].ubnd
> s
[k
].lbnd
)
410 SCM_SETCAR (indptr
, scm_sum (SCM_CAR (indptr
), scm_from_int (1)));
411 imap
= scm_apply_0 (mapfunc
, scm_reverse (inds
));
412 s
[k
].inc
= scm_array_handle_pos (&old_handle
, imap
) - i
;
415 new_max
+= (s
[k
].ubnd
- s
[k
].lbnd
) * s
[k
].inc
;
417 new_min
+= (s
[k
].ubnd
- s
[k
].lbnd
) * s
[k
].inc
;
420 s
[k
].inc
= new_max
- new_min
+ 1; /* contiguous by default */
421 indptr
= SCM_CDR (indptr
);
424 scm_array_handle_release (&old_handle
);
426 if (old_min
> new_min
|| old_max
< new_max
)
427 SCM_MISC_ERROR ("mapping out of range", SCM_EOL
);
428 if (1 == SCM_I_ARRAY_NDIM (ra
) && 0 == SCM_I_ARRAY_BASE (ra
))
430 SCM v
= SCM_I_ARRAY_V (ra
);
431 size_t length
= scm_c_generalized_vector_length (v
);
432 if (1 == s
->inc
&& 0 == s
->lbnd
&& length
== 1 + s
->ubnd
)
434 if (s
->ubnd
< s
->lbnd
)
435 return scm_make_generalized_vector (scm_array_type (ra
), SCM_INUM0
,
438 scm_i_ra_set_contp (ra
);
444 /* args are RA . DIMS */
445 SCM_DEFINE (scm_transpose_array
, "transpose-array", 1, 0, 1,
447 "Return an array sharing contents with @var{array}, but with\n"
448 "dimensions arranged in a different order. There must be one\n"
449 "@var{dim} argument for each dimension of @var{array}.\n"
450 "@var{dim0}, @var{dim1}, @dots{} should be integers between 0\n"
451 "and the rank of the array to be returned. Each integer in that\n"
452 "range must appear at least once in the argument list.\n"
454 "The values of @var{dim0}, @var{dim1}, @dots{} correspond to\n"
455 "dimensions in the array to be returned, their positions in the\n"
456 "argument list to dimensions of @var{array}. Several @var{dim}s\n"
457 "may have the same value, in which case the returned array will\n"
458 "have smaller rank than @var{array}.\n"
461 "(transpose-array '#2((a b) (c d)) 1 0) @result{} #2((a c) (b d))\n"
462 "(transpose-array '#2((a b) (c d)) 0 0) @result{} #1(a d)\n"
463 "(transpose-array '#3(((a b c) (d e f)) ((1 2 3) (4 5 6))) 1 1 0) @result{}\n"
464 " #2((a 4) (b 5) (c 6))\n"
466 #define FUNC_NAME s_scm_transpose_array
469 scm_t_array_dim
*s
, *r
;
472 SCM_VALIDATE_REST_ARGUMENT (args
);
473 SCM_ASSERT (SCM_NIMP (ra
), ra
, SCM_ARG1
, FUNC_NAME
);
475 if (scm_is_generalized_vector (ra
))
477 /* Make sure that we are called with a single zero as
480 if (scm_is_null (args
) || !scm_is_null (SCM_CDR (args
)))
481 SCM_WRONG_NUM_ARGS ();
482 SCM_VALIDATE_INT_COPY (SCM_ARG2
, SCM_CAR (args
), i
);
483 SCM_ASSERT_RANGE (SCM_ARG2
, SCM_CAR (args
), i
== 0);
487 if (SCM_I_ARRAYP (ra
))
489 vargs
= scm_vector (args
);
490 if (SCM_SIMPLE_VECTOR_LENGTH (vargs
) != SCM_I_ARRAY_NDIM (ra
))
491 SCM_WRONG_NUM_ARGS ();
493 for (k
= 0; k
< SCM_I_ARRAY_NDIM (ra
); k
++)
495 i
= scm_to_signed_integer (SCM_SIMPLE_VECTOR_REF (vargs
, k
),
496 0, SCM_I_ARRAY_NDIM(ra
));
501 res
= scm_i_make_array (ndim
);
502 SCM_I_ARRAY_V (res
) = SCM_I_ARRAY_V (ra
);
503 SCM_I_ARRAY_BASE (res
) = SCM_I_ARRAY_BASE (ra
);
506 SCM_I_ARRAY_DIMS (res
)[k
].lbnd
= 0;
507 SCM_I_ARRAY_DIMS (res
)[k
].ubnd
= -1;
509 for (k
= SCM_I_ARRAY_NDIM (ra
); k
--;)
511 i
= scm_to_int (SCM_SIMPLE_VECTOR_REF (vargs
, k
));
512 s
= &(SCM_I_ARRAY_DIMS (ra
)[k
]);
513 r
= &(SCM_I_ARRAY_DIMS (res
)[i
]);
514 if (r
->ubnd
< r
->lbnd
)
523 if (r
->ubnd
> s
->ubnd
)
525 if (r
->lbnd
< s
->lbnd
)
527 SCM_I_ARRAY_BASE (res
) += (s
->lbnd
- r
->lbnd
) * r
->inc
;
534 SCM_MISC_ERROR ("bad argument list", SCM_EOL
);
535 scm_i_ra_set_contp (res
);
539 scm_wrong_type_arg_msg (NULL
, 0, ra
, "array");
543 /* attempts to unroll an array into a one-dimensional array.
544 returns the unrolled array or #f if it can't be done. */
545 /* if strict is not SCM_UNDEFINED, return #f if returned array
546 wouldn't have contiguous elements. */
547 SCM_DEFINE (scm_array_contents
, "array-contents", 1, 1, 0,
548 (SCM ra
, SCM strict
),
549 "If @var{array} may be @dfn{unrolled} into a one dimensional shared array\n"
550 "without changing their order (last subscript changing fastest), then\n"
551 "@code{array-contents} returns that shared array, otherwise it returns\n"
552 "@code{#f}. All arrays made by @var{make-array} and\n"
553 "@var{make-uniform-array} may be unrolled, some arrays made by\n"
554 "@var{make-shared-array} may not be.\n\n"
555 "If the optional argument @var{strict} is provided, a shared array will\n"
556 "be returned only if its elements are stored internally contiguous in\n"
558 #define FUNC_NAME s_scm_array_contents
562 if (scm_is_generalized_vector (ra
))
565 if (SCM_I_ARRAYP (ra
))
567 size_t k
, ndim
= SCM_I_ARRAY_NDIM (ra
), len
= 1;
568 if (!SCM_I_ARRAYP (ra
) || !SCM_I_ARRAY_CONTP (ra
))
570 for (k
= 0; k
< ndim
; k
++)
571 len
*= SCM_I_ARRAY_DIMS (ra
)[k
].ubnd
- SCM_I_ARRAY_DIMS (ra
)[k
].lbnd
+ 1;
572 if (!SCM_UNBNDP (strict
) && scm_is_true (strict
))
574 if (ndim
&& (1 != SCM_I_ARRAY_DIMS (ra
)[ndim
- 1].inc
))
576 if (scm_is_bitvector (SCM_I_ARRAY_V (ra
)))
578 if (len
!= scm_c_bitvector_length (SCM_I_ARRAY_V (ra
)) ||
579 SCM_I_ARRAY_BASE (ra
) % SCM_LONG_BIT
||
586 SCM v
= SCM_I_ARRAY_V (ra
);
587 size_t length
= scm_c_generalized_vector_length (v
);
588 if ((len
== length
) && 0 == SCM_I_ARRAY_BASE (ra
) && SCM_I_ARRAY_DIMS (ra
)->inc
)
592 sra
= scm_i_make_array (1);
593 SCM_I_ARRAY_DIMS (sra
)->lbnd
= 0;
594 SCM_I_ARRAY_DIMS (sra
)->ubnd
= len
- 1;
595 SCM_I_ARRAY_V (sra
) = SCM_I_ARRAY_V (ra
);
596 SCM_I_ARRAY_BASE (sra
) = SCM_I_ARRAY_BASE (ra
);
597 SCM_I_ARRAY_DIMS (sra
)->inc
= (ndim
? SCM_I_ARRAY_DIMS (ra
)[ndim
- 1].inc
: 1);
601 scm_wrong_type_arg_msg (NULL
, 0, ra
, "array");
607 list_to_array (SCM lst
, scm_t_array_handle
*handle
, ssize_t pos
, size_t k
)
609 if (k
== scm_array_handle_rank (handle
))
610 scm_array_handle_set (handle
, pos
, lst
);
613 scm_t_array_dim
*dim
= scm_array_handle_dims (handle
) + k
;
614 ssize_t inc
= dim
->inc
;
615 size_t len
= 1 + dim
->ubnd
- dim
->lbnd
, n
;
619 while (n
> 0 && scm_is_pair (lst
))
621 list_to_array (SCM_CAR (lst
), handle
, pos
, k
+ 1);
627 errmsg
= "too few elements for array dimension ~a, need ~a";
628 if (!scm_is_null (lst
))
629 errmsg
= "too many elements for array dimension ~a, want ~a";
631 scm_misc_error (NULL
, errmsg
, scm_list_2 (scm_from_ulong (k
),
632 scm_from_size_t (len
)));
637 SCM_DEFINE (scm_list_to_typed_array
, "list->typed-array", 3, 0, 0,
638 (SCM type
, SCM shape
, SCM lst
),
639 "Return an array of the type @var{type}\n"
640 "with elements the same as those of @var{lst}.\n"
642 "The argument @var{shape} determines the number of dimensions\n"
643 "of the array and their shape. It is either an exact integer,\n"
645 "number of dimensions directly, or a list whose length\n"
646 "specifies the number of dimensions and each element specified\n"
647 "the lower and optionally the upper bound of the corresponding\n"
649 "When the element is list of two elements, these elements\n"
650 "give the lower and upper bounds. When it is an exact\n"
651 "integer, it gives only the lower bound.")
652 #define FUNC_NAME s_scm_list_to_typed_array
656 scm_t_array_handle handle
;
659 if (scm_is_integer (shape
))
661 size_t k
= scm_to_size_t (shape
);
665 shape
= scm_cons (scm_length (row
), shape
);
666 if (k
> 0 && !scm_is_null (row
))
672 SCM shape_spec
= shape
;
676 SCM spec
= scm_car (shape_spec
);
677 if (scm_is_pair (spec
))
678 shape
= scm_cons (spec
, shape
);
680 shape
= scm_cons (scm_list_2 (spec
,
681 scm_sum (scm_sum (spec
,
685 shape_spec
= scm_cdr (shape_spec
);
686 if (scm_is_pair (shape_spec
))
688 if (!scm_is_null (row
))
696 ra
= scm_make_typed_array (type
, SCM_UNSPECIFIED
,
697 scm_reverse_x (shape
, SCM_EOL
));
699 scm_array_get_handle (ra
, &handle
);
700 list_to_array (lst
, &handle
, 0, 0);
701 scm_array_handle_release (&handle
);
707 SCM_DEFINE (scm_list_to_array
, "list->array", 2, 0, 0,
709 "Return an array with elements the same as those of @var{lst}.")
710 #define FUNC_NAME s_scm_list_to_array
712 return scm_list_to_typed_array (SCM_BOOL_T
, ndim
, lst
);
716 /* Print dimension DIM of ARRAY.
720 scm_i_print_array_dimension (scm_t_array_handle
*h
, int dim
, int pos
,
721 SCM port
, scm_print_state
*pstate
)
724 scm_iprin1 (scm_array_handle_ref (h
, pos
), port
, pstate
);
728 scm_putc ('(', port
);
729 for (i
= h
->dims
[dim
].lbnd
; i
<= h
->dims
[dim
].ubnd
;
730 i
++, pos
+= h
->dims
[dim
].inc
)
732 scm_i_print_array_dimension (h
, dim
+1, pos
, port
, pstate
);
733 if (i
< h
->dims
[dim
].ubnd
)
734 scm_putc (' ', port
);
736 scm_putc (')', port
);
745 scm_i_print_array (SCM array
, SCM port
, scm_print_state
*pstate
)
747 scm_t_array_handle h
;
749 int print_lbnds
= 0, zero_size
= 0, print_lens
= 0;
751 scm_array_get_handle (array
, &h
);
753 scm_putc ('#', port
);
754 if (h
.ndims
!= 1 || h
.dims
[0].lbnd
!= 0)
755 scm_intprint (h
.ndims
, 10, port
);
756 if (h
.element_type
!= SCM_ARRAY_ELEMENT_TYPE_SCM
)
757 scm_write (scm_array_handle_element_type (&h
), port
);
759 for (i
= 0; i
< h
.ndims
; i
++)
761 if (h
.dims
[i
].lbnd
!= 0)
763 if (h
.dims
[i
].ubnd
- h
.dims
[i
].lbnd
+ 1 == 0)
769 if (print_lbnds
|| print_lens
)
770 for (i
= 0; i
< h
.ndims
; i
++)
774 scm_putc ('@', port
);
775 scm_intprint (h
.dims
[i
].lbnd
, 10, port
);
779 scm_putc (':', port
);
780 scm_intprint (h
.dims
[i
].ubnd
- h
.dims
[i
].lbnd
+ 1,
787 /* Rank zero arrays, which are really just scalars, are printed
788 specially. The consequent way would be to print them as
792 where OBJ is the printed representation of the scalar, but we
793 print them instead as
797 to make them look less strange.
799 Just printing them as
803 would be correct in a way as well, but zero rank arrays are
804 not really the same as Scheme values since they are boxed and
805 can be modified with array-set!, say.
807 scm_putc ('(', port
);
808 scm_i_print_array_dimension (&h
, 0, 0, port
, pstate
);
809 scm_putc (')', port
);
813 return scm_i_print_array_dimension (&h
, 0, 0, port
, pstate
);
816 /* Read an array. This function can also read vectors and uniform
817 vectors. Also, the conflict between '#f' and '#f32' and '#f64' is
820 C is the first character read after the '#'.
824 read_decimal_integer (SCM port
, int c
, ssize_t
*resp
)
836 while ('0' <= c
&& c
<= '9')
838 res
= 10*res
+ c
-'0';
849 scm_i_read_array (SCM port
, int c
)
852 scm_t_wchar tag_buf
[8];
855 SCM tag
, shape
= SCM_BOOL_F
, elements
;
857 /* XXX - shortcut for ordinary vectors. Shouldn't be necessary but
858 the array code can not deal with zero-length dimensions yet, and
859 we want to allow zero-length vectors, of course.
863 scm_ungetc (c
, port
);
864 return scm_vector (scm_read (port
));
867 /* Disambiguate between '#f' and uniform floating point vectors.
872 if (c
!= '3' && c
!= '6')
875 scm_ungetc (c
, port
);
881 goto continue_reading_tag
;
887 c
= read_decimal_integer (port
, c
, &rank
);
889 scm_i_input_error (NULL
, port
, "array rank must be non-negative",
895 continue_reading_tag
:
896 while (c
!= EOF
&& c
!= '(' && c
!= '@' && c
!= ':'
897 && tag_len
< sizeof tag_buf
/ sizeof tag_buf
[0])
899 tag_buf
[tag_len
++] = c
;
906 tag
= scm_string_to_symbol (scm_from_utf32_stringn (tag_buf
, tag_len
));
907 if (tag_len
== sizeof tag_buf
/ sizeof tag_buf
[0])
908 scm_i_input_error (NULL
, port
, "invalid array tag, starting with: ~a",
914 if (c
== '@' || c
== ':')
920 ssize_t lbnd
= 0, len
= 0;
926 c
= read_decimal_integer (port
, c
, &lbnd
);
929 s
= scm_from_ssize_t (lbnd
);
934 c
= read_decimal_integer (port
, c
, &len
);
936 scm_i_input_error (NULL
, port
,
937 "array length must be non-negative",
940 s
= scm_list_2 (s
, scm_from_ssize_t (lbnd
+len
-1));
943 shape
= scm_cons (s
, shape
);
944 } while (c
== '@' || c
== ':');
946 shape
= scm_reverse_x (shape
, SCM_EOL
);
949 /* Read nested lists of elements.
952 scm_i_input_error (NULL
, port
,
953 "missing '(' in vector or array literal",
955 scm_ungetc (c
, port
);
956 elements
= scm_read (port
);
958 if (scm_is_false (shape
))
959 shape
= scm_from_ssize_t (rank
);
960 else if (scm_ilength (shape
) != rank
)
963 "the number of shape specifications must match the array rank",
966 /* Handle special print syntax of rank zero arrays; see
967 scm_i_print_array for a rationale.
971 if (!scm_is_pair (elements
))
972 scm_i_input_error (NULL
, port
,
973 "too few elements in array literal, need 1",
975 if (!scm_is_null (SCM_CDR (elements
)))
976 scm_i_input_error (NULL
, port
,
977 "too many elements in array literal, want 1",
979 elements
= SCM_CAR (elements
);
984 return scm_list_to_typed_array (tag
, shape
, elements
);
989 array_handle_ref (scm_t_array_handle
*h
, size_t pos
)
991 return scm_c_generalized_vector_ref (SCM_I_ARRAY_V (h
->array
), pos
);
995 array_handle_set (scm_t_array_handle
*h
, size_t pos
, SCM val
)
997 scm_c_generalized_vector_set_x (SCM_I_ARRAY_V (h
->array
), pos
, val
);
1000 /* FIXME: should be handle for vect? maybe not, because of dims */
1002 array_get_handle (SCM array
, scm_t_array_handle
*h
)
1004 scm_t_array_handle vh
;
1005 scm_array_get_handle (SCM_I_ARRAY_V (array
), &vh
);
1006 h
->element_type
= vh
.element_type
;
1007 h
->elements
= vh
.elements
;
1008 h
->writable_elements
= vh
.writable_elements
;
1009 scm_array_handle_release (&vh
);
1011 h
->dims
= SCM_I_ARRAY_DIMS (array
);
1012 h
->ndims
= SCM_I_ARRAY_NDIM (array
);
1013 h
->base
= SCM_I_ARRAY_BASE (array
);
1016 SCM_ARRAY_IMPLEMENTATION (scm_tc7_array
,
1018 array_handle_ref
, array_handle_set
,
1024 scm_add_feature ("array");
1026 #include "libguile/arrays.x"