mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-04-30 03:40:34 +02:00
On x86-64-MinGW the size of long is 4. As long is used for SCM_FIXNUM_BIT, that would mean incompatible .go files, and waste of cell space. So we would like to use long long, but the GMP interface uses long. To get around this, the x86-64-MinGW port now requires the use of mini-GMP. Mini-GMP has been changed to use intptr_t and uintptr_t. Likewise, "integers.{h,c}" and "numbers.{h,c}" now use intptr_t instead of scm_t_inum or long, and uintptr_t instead of unsigned long. * configure.ac: When x86_64-w64-mingw32, require mini-GMP. * libguile/mini-gmp.h: Use intptr_t instead of long, uintptr_t instead of unsigned long throughout. * libguile/mini-gmp.c: Likewise. * libguile/scm.h (SCM_INTPTR_T_BIT): New define. * libguile/numbers.h (SCM_FIXNUM_BIT): Use it. * libguile/numbers.c (L1, UL1): New macros. Use them thoughout instead of 1L, 1UL. (verify): Use SCM_INTPTR_T_BIT. (verify): Use SCM_INTPTR_T_MAX and SCM_INTPTR_T_MIN. (scm_from_inum): Remove macro. Use intptr_t and uintptr_t instead of scm_t_inum or long, and unsigned long. * libguile/numbers.h (scm_from_intptr, scm_from_uintptr, scm_to_intptr, scm_to_uintptr): New defines. * libguile/integers.h: Use intptr_t and uintptr_t instead of scm_t_inum and unsigned long. * libguile/integers.c (L1) : New macro. Use it thoughout instead of 1L. Use intptr_t and uintptr_t instead of long and unsigned long. (long_magnitude): Rename to... (intptr_t_magnitude): ...this. Use intptr_t, uintptr_t. (negative_long): Rename to... (negative_t_intptr): ...this. Use uintptr_t, INTPTR_MIN. (inum_magnitude): Use intptr_t. (ulong_to_bignum): Rename to... (uintptr_t_to_bignum): ...this. Use uintptr_t. (long_to_bignum): Rename to... (intptr_t_to_bignum): ...this. Use intptr_t. (long_to_scm): Rename to... (intptr_t_to_scm): ...this. Use intptr_to_bignum. (ulong_to_scm): Rename to... (uintptr_t_to_scm): ...this. Use uintptr_to_bignum. (long_sign): Rename to.. (intptr_t_sign): ...this. Use SCM_SIZEOF_INTPTR_T. (bignum_cmp_long): Rename to... (bignum_cmp_intptr_t): ...this. Use uintptr_t. * libguile/array-map.c (array_compare): Use uintptr_t instead of unsigned long and intptr_t instead of long. * libguile/arrays.c (make-shared-array): Use ssize_t instead of long. * libguile/bytevectors.c (is_signed_int32, is_unsigned_int32) [MINGW32 && __x86_64__]: Use ULL. (twos_complement): Use uintptr_t instead of unsigned long. * libguile/hash.c (JENKINS_LOOKUP3_HASHWORD2): Likewise. (narrow_string_hash, wide_string_hash, scm_i_string_hash, scm_i_locale_string_hash, scm_i_latin1_string_hash, scm_i_utf8_string_hash, scm_i_struct_hash, scm_raw_ihashq, scm_raw_ihash): Use and return uintptr_t instead of unsigned long. (scm_hashv, scm_hash): Use SCM_UINTPTR_T_MAX. * libguile/hash.h (scm_i_locale_string_hash, scm_i_latin1_string_hash, scm_i_utf8_string_hash): update prototypes. * libguile/scmsigs.c (sigaction): Use intptr_t instead of long. * libguile/strings.c (scm_i_make_symbol, (scm_i_c_make_symbol): Use uintptr_t instead of unsigned long. * libguile/strings.h (scm_i_make_symbol, (scm_i_c_make_symbol): Update declacations. * libguile/srfi-60.c: Use scm_to_uintptr, scm_from_intptr and variants throughout. * libguile/symbols.c (symbol-hash): Use scm_from_uintptr. Co-authored-by: Mike Gran <spk121@yahoo.com> Co-authored-by: Andy Wingo <wingo@pobox.com>
1244 lines
36 KiB
C
1244 lines
36 KiB
C
/* Copyright 1995-1998,2000-2006,2009-2015,2018, 2021
|
||
Free Software Foundation, Inc.
|
||
|
||
This file is part of Guile.
|
||
|
||
Guile is free software: you can redistribute it and/or modify it
|
||
under the terms of the GNU Lesser General Public License as published
|
||
by the Free Software Foundation, either version 3 of the License, or
|
||
(at your option) any later version.
|
||
|
||
Guile is distributed in the hope that it will be useful, but WITHOUT
|
||
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
|
||
License for more details.
|
||
|
||
You should have received a copy of the GNU Lesser General Public
|
||
License along with Guile. If not, see
|
||
<https://www.gnu.org/licenses/>. */
|
||
|
||
|
||
|
||
|
||
#ifdef HAVE_CONFIG_H
|
||
# include <config.h>
|
||
#endif
|
||
|
||
#include <stdio.h>
|
||
#include <errno.h>
|
||
#include <string.h>
|
||
|
||
#include "array-map.h"
|
||
#include "bitvectors.h"
|
||
#include "boolean.h"
|
||
#include "bytevectors.h"
|
||
#include "chars.h"
|
||
#include "dynwind.h"
|
||
#include "eq.h"
|
||
#include "error.h"
|
||
#include "eval.h"
|
||
#include "feature.h"
|
||
#include "fports.h"
|
||
#include "generalized-vectors.h"
|
||
#include "gsubr.h"
|
||
#include "list.h"
|
||
#include "modules.h"
|
||
#include "numbers.h"
|
||
#include "pairs.h"
|
||
#include "procs.h"
|
||
#include "read.h"
|
||
#include "srfi-13.h"
|
||
#include "srfi-4.h"
|
||
#include "strings.h"
|
||
#include "uniform.h"
|
||
#include "vectors.h"
|
||
|
||
#include "arrays.h"
|
||
|
||
SCM_INTERNAL SCM scm_i_array_ref (SCM v,
|
||
SCM idx0, SCM idx1, SCM idxN);
|
||
SCM_INTERNAL SCM scm_i_array_set_x (SCM v, SCM obj,
|
||
SCM idx0, SCM idx1, SCM idxN);
|
||
|
||
|
||
int
|
||
scm_is_array (SCM obj)
|
||
{
|
||
if (!SCM_HEAP_OBJECT_P (obj))
|
||
return 0;
|
||
|
||
switch (SCM_TYP7 (obj))
|
||
{
|
||
case scm_tc7_string:
|
||
case scm_tc7_vector:
|
||
case scm_tc7_bitvector:
|
||
case scm_tc7_bytevector:
|
||
case scm_tc7_array:
|
||
return 1;
|
||
default:
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
SCM_DEFINE (scm_array_p_2, "array?", 1, 0, 0,
|
||
(SCM obj),
|
||
"Return @code{#t} if the @var{obj} is an array, and @code{#f} if\n"
|
||
"not.")
|
||
#define FUNC_NAME s_scm_array_p_2
|
||
{
|
||
return scm_from_bool (scm_is_array (obj));
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
/* The array type predicate, with an extra argument kept for backward
|
||
compatibility. Note that we can't use `SCM_DEFINE' directly because there
|
||
would be an argument count mismatch that would be caught by
|
||
`snarf-check-and-output-texi.scm'. */
|
||
SCM
|
||
scm_array_p (SCM obj, SCM unused)
|
||
{
|
||
return scm_array_p_2 (obj);
|
||
}
|
||
|
||
int
|
||
scm_is_typed_array (SCM obj, SCM type)
|
||
{
|
||
int ret = 0;
|
||
if (scm_is_array (obj))
|
||
{
|
||
scm_t_array_handle h;
|
||
|
||
scm_array_get_handle (obj, &h);
|
||
ret = scm_is_eq (scm_array_handle_element_type (&h), type);
|
||
scm_array_handle_release (&h);
|
||
}
|
||
|
||
return ret;
|
||
}
|
||
|
||
SCM_DEFINE (scm_typed_array_p, "typed-array?", 2, 0, 0,
|
||
(SCM obj, SCM type),
|
||
"Return @code{#t} if the @var{obj} is an array of type\n"
|
||
"@var{type}, and @code{#f} if not.")
|
||
#define FUNC_NAME s_scm_typed_array_p
|
||
{
|
||
return scm_from_bool (scm_is_typed_array (obj, type));
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
size_t
|
||
scm_c_array_length (SCM array)
|
||
{
|
||
scm_t_array_handle handle;
|
||
size_t res;
|
||
|
||
scm_array_get_handle (array, &handle);
|
||
if (scm_array_handle_rank (&handle) < 1)
|
||
{
|
||
scm_array_handle_release (&handle);
|
||
scm_wrong_type_arg_msg (NULL, 0, array, "array of nonzero rank");
|
||
}
|
||
res = handle.dims[0].ubnd - handle.dims[0].lbnd + 1;
|
||
scm_array_handle_release (&handle);
|
||
|
||
return res;
|
||
}
|
||
|
||
SCM_DEFINE (scm_array_length, "array-length", 1, 0, 0,
|
||
(SCM array),
|
||
"Return the length of an array: its first dimension.\n"
|
||
"It is an error to ask for the length of an array of rank 0.")
|
||
#define FUNC_NAME s_scm_array_length
|
||
{
|
||
return scm_from_size_t (scm_c_array_length (array));
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_array_dimensions, "array-dimensions", 1, 0, 0,
|
||
(SCM ra),
|
||
"@code{array-dimensions} is similar to @code{array-shape} but replaces\n"
|
||
"elements with a @code{0} minimum with one greater than the maximum. So:\n"
|
||
"@lisp\n"
|
||
"(array-dimensions (make-array 'foo '(-1 3) 5)) @result{} ((-1 3) 5)\n"
|
||
"@end lisp")
|
||
#define FUNC_NAME s_scm_array_dimensions
|
||
{
|
||
scm_t_array_handle handle;
|
||
scm_t_array_dim *s;
|
||
SCM res = SCM_EOL;
|
||
size_t k;
|
||
|
||
scm_array_get_handle (ra, &handle);
|
||
s = scm_array_handle_dims (&handle);
|
||
k = scm_array_handle_rank (&handle);
|
||
|
||
while (k--)
|
||
res = scm_cons (s[k].lbnd
|
||
? scm_cons2 (scm_from_ssize_t (s[k].lbnd),
|
||
scm_from_ssize_t (s[k].ubnd),
|
||
SCM_EOL)
|
||
: scm_from_ssize_t (1 + s[k].ubnd),
|
||
res);
|
||
|
||
scm_array_handle_release (&handle);
|
||
return res;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
SCM_DEFINE (scm_array_type, "array-type", 1, 0, 0,
|
||
(SCM ra),
|
||
"")
|
||
#define FUNC_NAME s_scm_array_type
|
||
{
|
||
scm_t_array_handle h;
|
||
SCM type;
|
||
|
||
scm_array_get_handle (ra, &h);
|
||
type = scm_array_handle_element_type (&h);
|
||
scm_array_handle_release (&h);
|
||
|
||
return type;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
SCM_DEFINE (scm_array_type_code,
|
||
"array-type-code", 1, 0, 0,
|
||
(SCM array),
|
||
"Return the type of the elements in @var{array},\n"
|
||
"as an integer code.")
|
||
#define FUNC_NAME s_scm_array_type_code
|
||
{
|
||
scm_t_array_handle h;
|
||
scm_t_array_element_type element_type;
|
||
|
||
scm_array_get_handle (array, &h);
|
||
element_type = h.element_type;
|
||
scm_array_handle_release (&h);
|
||
|
||
return scm_from_uint16 (element_type);
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
SCM_DEFINE (scm_array_in_bounds_p, "array-in-bounds?", 1, 0, 1,
|
||
(SCM ra, SCM args),
|
||
"Return @code{#t} if its arguments would be acceptable to\n"
|
||
"@code{array-ref}.")
|
||
#define FUNC_NAME s_scm_array_in_bounds_p
|
||
{
|
||
SCM res = SCM_BOOL_T;
|
||
size_t k, ndim;
|
||
scm_t_array_dim *s;
|
||
scm_t_array_handle handle;
|
||
|
||
SCM_VALIDATE_REST_ARGUMENT (args);
|
||
|
||
scm_array_get_handle (ra, &handle);
|
||
s = scm_array_handle_dims (&handle);
|
||
ndim = scm_array_handle_rank (&handle);
|
||
|
||
for (k = 0; k < ndim; k++)
|
||
{
|
||
intptr_t ind;
|
||
|
||
if (!scm_is_pair (args))
|
||
SCM_WRONG_NUM_ARGS ();
|
||
ind = scm_to_intptr_t (SCM_CAR (args));
|
||
args = SCM_CDR (args);
|
||
|
||
if (ind < s[k].lbnd || ind > s[k].ubnd)
|
||
{
|
||
res = SCM_BOOL_F;
|
||
/* We do not stop the checking after finding a violation
|
||
since we want to validate the type-correctness and
|
||
number of arguments in any case.
|
||
*/
|
||
}
|
||
}
|
||
|
||
scm_array_handle_release (&handle);
|
||
return res;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM
|
||
scm_c_array_ref_1 (SCM array, ssize_t idx0)
|
||
{
|
||
scm_t_array_handle handle;
|
||
SCM res;
|
||
|
||
scm_array_get_handle (array, &handle);
|
||
res = scm_array_handle_ref (&handle, scm_array_handle_pos_1 (&handle, idx0));
|
||
scm_array_handle_release (&handle);
|
||
return res;
|
||
}
|
||
|
||
|
||
SCM
|
||
scm_c_array_ref_2 (SCM array, ssize_t idx0, ssize_t idx1)
|
||
{
|
||
scm_t_array_handle handle;
|
||
SCM res;
|
||
|
||
scm_array_get_handle (array, &handle);
|
||
res = scm_array_handle_ref (&handle, scm_array_handle_pos_2 (&handle, idx0, idx1));
|
||
scm_array_handle_release (&handle);
|
||
return res;
|
||
}
|
||
|
||
|
||
SCM
|
||
scm_array_ref (SCM v, SCM args)
|
||
{
|
||
scm_t_array_handle handle;
|
||
SCM res;
|
||
|
||
scm_array_get_handle (v, &handle);
|
||
res = scm_array_handle_ref (&handle, scm_array_handle_pos (&handle, args));
|
||
scm_array_handle_release (&handle);
|
||
return res;
|
||
}
|
||
|
||
|
||
void
|
||
scm_c_array_set_1_x (SCM array, SCM obj, ssize_t idx0)
|
||
{
|
||
scm_t_array_handle handle;
|
||
|
||
scm_array_get_handle (array, &handle);
|
||
scm_array_handle_set (&handle, scm_array_handle_pos_1 (&handle, idx0),
|
||
obj);
|
||
scm_array_handle_release (&handle);
|
||
}
|
||
|
||
|
||
void
|
||
scm_c_array_set_2_x (SCM array, SCM obj, ssize_t idx0, ssize_t idx1)
|
||
{
|
||
scm_t_array_handle handle;
|
||
|
||
scm_array_get_handle (array, &handle);
|
||
scm_array_handle_set (&handle, scm_array_handle_pos_2 (&handle, idx0, idx1),
|
||
obj);
|
||
scm_array_handle_release (&handle);
|
||
}
|
||
|
||
|
||
SCM
|
||
scm_array_set_x (SCM v, SCM obj, SCM args)
|
||
{
|
||
scm_t_array_handle handle;
|
||
|
||
scm_array_get_handle (v, &handle);
|
||
scm_array_handle_set (&handle, scm_array_handle_pos (&handle, args), obj);
|
||
scm_array_handle_release (&handle);
|
||
return SCM_UNSPECIFIED;
|
||
}
|
||
|
||
|
||
SCM_DEFINE (scm_i_array_ref, "array-ref", 1, 2, 1,
|
||
(SCM v, SCM idx0, SCM idx1, SCM idxN),
|
||
"Return the element at the @code{(idx0, idx1, idxN...)}\n"
|
||
"position in array @var{v}.")
|
||
#define FUNC_NAME s_scm_i_array_ref
|
||
{
|
||
if (SCM_UNBNDP (idx0))
|
||
return scm_array_ref (v, SCM_EOL);
|
||
else if (SCM_UNBNDP (idx1))
|
||
return scm_c_array_ref_1 (v, scm_to_ssize_t (idx0));
|
||
else if (scm_is_null (idxN))
|
||
return scm_c_array_ref_2 (v, scm_to_ssize_t (idx0), scm_to_ssize_t (idx1));
|
||
else
|
||
return scm_array_ref (v, scm_cons (idx0, scm_cons (idx1, idxN)));
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
SCM_DEFINE (scm_i_array_set_x, "array-set!", 2, 2, 1,
|
||
(SCM v, SCM obj, SCM idx0, SCM idx1, SCM idxN),
|
||
"Set the element at the @code{(idx0, idx1, idxN...)} position\n"
|
||
"in the array @var{v} to @var{obj}. The value returned by\n"
|
||
"@code{array-set!} is unspecified.")
|
||
#define FUNC_NAME s_scm_i_array_set_x
|
||
{
|
||
if (SCM_UNBNDP (idx0))
|
||
scm_array_set_x (v, obj, SCM_EOL);
|
||
else if (SCM_UNBNDP (idx1))
|
||
scm_c_array_set_1_x (v, obj, scm_to_ssize_t (idx0));
|
||
else if (scm_is_null (idxN))
|
||
scm_c_array_set_2_x (v, obj, scm_to_ssize_t (idx0), scm_to_ssize_t (idx1));
|
||
else
|
||
scm_array_set_x (v, obj, scm_cons (idx0, scm_cons (idx1, idxN)));
|
||
|
||
return SCM_UNSPECIFIED;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
static SCM
|
||
array_to_list (scm_t_array_handle *h, size_t dim, uintptr_t pos)
|
||
{
|
||
if (dim == scm_array_handle_rank (h))
|
||
return scm_array_handle_ref (h, pos);
|
||
else
|
||
{
|
||
SCM res = SCM_EOL;
|
||
intptr_t inc;
|
||
size_t i;
|
||
|
||
i = h->dims[dim].ubnd - h->dims[dim].lbnd + 1;
|
||
inc = h->dims[dim].inc;
|
||
pos += (i - 1) * inc;
|
||
|
||
for (; i > 0; i--, pos -= inc)
|
||
res = scm_cons (array_to_list (h, dim + 1, pos), res);
|
||
return res;
|
||
}
|
||
}
|
||
|
||
SCM_DEFINE (scm_array_to_list, "array->list", 1, 0, 0,
|
||
(SCM array),
|
||
"Return a list representation of @var{array}.\n\n"
|
||
"It is easiest to specify the behavior of this function by\n"
|
||
"example:\n"
|
||
"@example\n"
|
||
"(array->list #0(a)) @result{} 1\n"
|
||
"(array->list #1(a b)) @result{} (a b)\n"
|
||
"(array->list #2((aa ab) (ba bb)) @result{} ((aa ab) (ba bb))\n"
|
||
"@end example\n")
|
||
#define FUNC_NAME s_scm_array_to_list
|
||
{
|
||
scm_t_array_handle h;
|
||
SCM res;
|
||
|
||
scm_array_get_handle (array, &h);
|
||
res = array_to_list (&h, 0, 0);
|
||
scm_array_handle_release (&h);
|
||
|
||
return res;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
size_t
|
||
scm_c_array_rank (SCM array)
|
||
{
|
||
if (SCM_I_ARRAYP (array))
|
||
return SCM_I_ARRAY_NDIM (array);
|
||
else if (scm_is_array (array))
|
||
return 1;
|
||
else
|
||
scm_wrong_type_arg_msg ("array-rank", SCM_ARG1, array, "array");
|
||
}
|
||
|
||
SCM_DEFINE (scm_array_rank, "array-rank", 1, 0, 0,
|
||
(SCM array),
|
||
"Return the number of dimensions of the array @var{array.}\n")
|
||
#define FUNC_NAME s_scm_array_rank
|
||
{
|
||
return scm_from_size_t (scm_c_array_rank (array));
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_shared_array_root, "shared-array-root", 1, 0, 0,
|
||
(SCM ra),
|
||
"Return the root vector of a shared array.")
|
||
#define FUNC_NAME s_scm_shared_array_root
|
||
{
|
||
if (SCM_I_ARRAYP (ra))
|
||
return SCM_I_ARRAY_V (ra);
|
||
else if (scm_is_array (ra))
|
||
return ra;
|
||
else
|
||
scm_wrong_type_arg_msg (FUNC_NAME, SCM_ARG1, ra, "array");
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_shared_array_offset, "shared-array-offset", 1, 0, 0,
|
||
(SCM ra),
|
||
"Return the root vector index of the first element in the array.")
|
||
#define FUNC_NAME s_scm_shared_array_offset
|
||
{
|
||
if (SCM_I_ARRAYP (ra))
|
||
return scm_from_size_t (SCM_I_ARRAY_BASE (ra));
|
||
else if (scm_is_array (ra))
|
||
return scm_from_size_t (0);
|
||
else
|
||
scm_wrong_type_arg_msg (FUNC_NAME, SCM_ARG1, ra, "array");
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_shared_array_increments, "shared-array-increments", 1, 0, 0,
|
||
(SCM ra),
|
||
"For each dimension, return the distance between elements in the root vector.")
|
||
#define FUNC_NAME s_scm_shared_array_increments
|
||
{
|
||
if (SCM_I_ARRAYP (ra))
|
||
{
|
||
size_t k = SCM_I_ARRAY_NDIM (ra);
|
||
SCM res = SCM_EOL;
|
||
scm_t_array_dim *dims = SCM_I_ARRAY_DIMS (ra);
|
||
while (k--)
|
||
res = scm_cons (scm_from_ssize_t (dims[k].inc), res);
|
||
return res;
|
||
}
|
||
else if (scm_is_array (ra))
|
||
return scm_list_1 (scm_from_ssize_t (1));
|
||
else
|
||
scm_wrong_type_arg_msg (FUNC_NAME, SCM_ARG1, ra, "array");
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM
|
||
scm_i_make_array (int ndim)
|
||
{
|
||
SCM ra = scm_i_raw_array (ndim);
|
||
SCM_I_ARRAY_SET_V (ra, SCM_BOOL_F);
|
||
SCM_I_ARRAY_SET_BASE (ra, 0);
|
||
/* dimensions are unset */
|
||
return ra;
|
||
}
|
||
|
||
/* Increments will still need to be set. */
|
||
|
||
SCM
|
||
scm_i_shap2ra (SCM args)
|
||
{
|
||
scm_t_array_dim *s;
|
||
int ndim = scm_ilength (args);
|
||
if (ndim < 0)
|
||
scm_misc_error (NULL, "bad array bounds ~a", scm_list_1 (args));
|
||
|
||
SCM ra = scm_i_make_array (ndim);
|
||
SCM_I_ARRAY_SET_BASE (ra, 0);
|
||
s = SCM_I_ARRAY_DIMS (ra);
|
||
for (; !scm_is_null (args); s++, args = SCM_CDR (args))
|
||
{
|
||
SCM spec = SCM_CAR (args);
|
||
if (scm_is_integer (spec))
|
||
{
|
||
s->lbnd = 0;
|
||
s->ubnd = scm_to_ssize_t (spec);
|
||
if (s->ubnd < 0)
|
||
scm_misc_error (NULL, "negative array dimension ~a", scm_list_1 (spec));
|
||
--s->ubnd;
|
||
}
|
||
else
|
||
{
|
||
if (!scm_is_pair (spec) || !scm_is_integer (SCM_CAR (spec)))
|
||
scm_misc_error (NULL, "bad array bound ~a", scm_list_1 (spec));
|
||
s->lbnd = scm_to_ssize_t (SCM_CAR (spec));
|
||
SCM rest = SCM_CDR (spec);
|
||
if (!scm_is_pair (rest)
|
||
|| !scm_is_integer (SCM_CAR (rest))
|
||
|| !scm_is_null (SCM_CDR (rest)))
|
||
scm_misc_error (NULL, "bad array bound ~a", scm_list_1 (spec));
|
||
s->ubnd = scm_to_ssize_t (SCM_CAR (rest));
|
||
if (s->ubnd - s->lbnd < -1)
|
||
scm_misc_error (NULL, "bad array bound ~a", scm_list_1 (spec));
|
||
}
|
||
s->inc = 1;
|
||
}
|
||
return ra;
|
||
}
|
||
|
||
SCM_DEFINE (scm_make_typed_array, "make-typed-array", 2, 0, 1,
|
||
(SCM type, SCM fill, SCM bounds),
|
||
"Create and return an array of type @var{type}.")
|
||
#define FUNC_NAME s_scm_make_typed_array
|
||
{
|
||
size_t k, rlen = 1;
|
||
scm_t_array_dim *s;
|
||
SCM ra;
|
||
|
||
ra = scm_i_shap2ra (bounds);
|
||
s = SCM_I_ARRAY_DIMS (ra);
|
||
k = SCM_I_ARRAY_NDIM (ra);
|
||
|
||
while (k--)
|
||
{
|
||
s[k].inc = rlen;
|
||
SCM_ASSERT_RANGE (1, bounds, s[k].lbnd <= s[k].ubnd + 1);
|
||
rlen = (s[k].ubnd - s[k].lbnd + 1) * s[k].inc;
|
||
}
|
||
|
||
if (scm_is_eq (fill, SCM_UNSPECIFIED))
|
||
fill = SCM_UNDEFINED;
|
||
|
||
SCM_I_ARRAY_SET_V (ra, scm_make_generalized_vector (type, scm_from_size_t (rlen), fill));
|
||
|
||
if (1 == SCM_I_ARRAY_NDIM (ra) && 0 == SCM_I_ARRAY_BASE (ra))
|
||
if (0 == s->lbnd)
|
||
return SCM_I_ARRAY_V (ra);
|
||
|
||
return ra;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_make_array, "make-array", 1, 0, 1,
|
||
(SCM fill, SCM bounds),
|
||
"Create and return an array.")
|
||
#define FUNC_NAME s_scm_make_array
|
||
{
|
||
return scm_make_typed_array (SCM_BOOL_T, fill, bounds);
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_make_shared_array, "make-shared-array", 2, 0, 1,
|
||
(SCM oldra, SCM mapfunc, SCM dims),
|
||
"@code{make-shared-array} can be used to create shared subarrays\n"
|
||
"of other arrays. The @var{mapfunc} is a function that\n"
|
||
"translates coordinates in the new array into coordinates in the\n"
|
||
"old array. A @var{mapfunc} must be linear, and its range must\n"
|
||
"stay within the bounds of the old array, but it can be\n"
|
||
"otherwise arbitrary. A simple example:\n"
|
||
"@lisp\n"
|
||
"(define fred (make-array #f 8 8))\n"
|
||
"(define freds-diagonal\n"
|
||
" (make-shared-array fred (lambda (i) (list i i)) 8))\n"
|
||
"(array-set! freds-diagonal 'foo 3)\n"
|
||
"(array-ref fred 3 3) @result{} foo\n"
|
||
"(define freds-center\n"
|
||
" (make-shared-array fred (lambda (i j) (list (+ 3 i) (+ 3 j))) 2 2))\n"
|
||
"(array-ref freds-center 0 0) @result{} foo\n"
|
||
"@end lisp")
|
||
#define FUNC_NAME s_scm_make_shared_array
|
||
{
|
||
scm_t_array_handle old_handle;
|
||
SCM ra;
|
||
SCM inds, indptr;
|
||
SCM imap;
|
||
size_t k;
|
||
ssize_t i;
|
||
ssize_t old_base, old_min, new_min, old_max, new_max;
|
||
scm_t_array_dim *s;
|
||
|
||
SCM_VALIDATE_REST_ARGUMENT (dims);
|
||
SCM_VALIDATE_PROC (2, mapfunc);
|
||
ra = scm_i_shap2ra (dims);
|
||
|
||
scm_array_get_handle (oldra, &old_handle);
|
||
|
||
if (SCM_I_ARRAYP (oldra))
|
||
{
|
||
SCM_I_ARRAY_SET_V (ra, SCM_I_ARRAY_V (oldra));
|
||
old_base = old_min = old_max = SCM_I_ARRAY_BASE (oldra);
|
||
s = scm_array_handle_dims (&old_handle);
|
||
k = scm_array_handle_rank (&old_handle);
|
||
while (k--)
|
||
{
|
||
if (s[k].inc > 0)
|
||
old_max += (s[k].ubnd - s[k].lbnd) * s[k].inc;
|
||
else
|
||
old_min += (s[k].ubnd - s[k].lbnd) * s[k].inc;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
SCM_I_ARRAY_SET_V (ra, oldra);
|
||
old_base = old_min = 0;
|
||
old_max = scm_c_array_length (oldra) - 1;
|
||
}
|
||
|
||
inds = SCM_EOL;
|
||
s = SCM_I_ARRAY_DIMS (ra);
|
||
for (k = 0; k < SCM_I_ARRAY_NDIM (ra); k++)
|
||
{
|
||
inds = scm_cons (scm_from_ssize_t (s[k].lbnd), inds);
|
||
if (s[k].ubnd < s[k].lbnd)
|
||
{
|
||
if (1 == SCM_I_ARRAY_NDIM (ra))
|
||
ra = scm_make_generalized_vector (scm_array_type (ra),
|
||
SCM_INUM0, SCM_UNDEFINED);
|
||
else
|
||
SCM_I_ARRAY_SET_V (ra, scm_make_generalized_vector (scm_array_type (ra),
|
||
SCM_INUM0, SCM_UNDEFINED));
|
||
scm_array_handle_release (&old_handle);
|
||
return ra;
|
||
}
|
||
}
|
||
|
||
imap = scm_apply_0 (mapfunc, scm_reverse (inds));
|
||
i = scm_array_handle_pos (&old_handle, imap);
|
||
new_min = new_max = i + old_base;
|
||
SCM_I_ARRAY_SET_BASE (ra, new_min);
|
||
indptr = inds;
|
||
k = SCM_I_ARRAY_NDIM (ra);
|
||
while (k--)
|
||
{
|
||
if (s[k].ubnd > s[k].lbnd)
|
||
{
|
||
SCM_SETCAR (indptr, scm_sum (SCM_CAR (indptr), scm_from_int (1)));
|
||
imap = scm_apply_0 (mapfunc, scm_reverse (inds));
|
||
s[k].inc = scm_array_handle_pos (&old_handle, imap) - i;
|
||
i += s[k].inc;
|
||
if (s[k].inc > 0)
|
||
new_max += (s[k].ubnd - s[k].lbnd) * s[k].inc;
|
||
else
|
||
new_min += (s[k].ubnd - s[k].lbnd) * s[k].inc;
|
||
}
|
||
else
|
||
s[k].inc = new_max - new_min + 1; /* contiguous by default */
|
||
indptr = SCM_CDR (indptr);
|
||
}
|
||
|
||
scm_array_handle_release (&old_handle);
|
||
|
||
if (old_min > new_min || old_max < new_max)
|
||
scm_misc_error (FUNC_NAME, "mapping out of range", SCM_EOL);
|
||
if (1 == SCM_I_ARRAY_NDIM (ra) && 0 == SCM_I_ARRAY_BASE (ra))
|
||
{
|
||
SCM v = SCM_I_ARRAY_V (ra);
|
||
size_t length = scm_c_array_length (v);
|
||
if (1 == s->inc && 0 == s->lbnd && length == 1 + s->ubnd)
|
||
return v;
|
||
if (s->ubnd < s->lbnd)
|
||
return scm_make_generalized_vector (scm_array_type (ra), SCM_INUM0,
|
||
SCM_UNDEFINED);
|
||
}
|
||
return ra;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
static void
|
||
array_from_pos (scm_t_array_handle *handle, size_t *k, SCM *i, ssize_t *pos,
|
||
scm_t_array_dim **s)
|
||
{
|
||
*s = scm_array_handle_dims (handle);
|
||
*k = scm_array_handle_rank (handle);
|
||
for (; *k>0 && scm_is_pair (*i); --*k, ++*s, *i=scm_cdr (*i))
|
||
{
|
||
ssize_t ik = scm_to_ssize_t (scm_car (*i));
|
||
if (ik<(*s)->lbnd || ik>(*s)->ubnd)
|
||
{
|
||
s = NULL;
|
||
return;
|
||
}
|
||
*pos += (ik-(*s)->lbnd) * (*s)->inc;
|
||
}
|
||
}
|
||
|
||
static void
|
||
array_from_get_o (scm_t_array_handle *handle, size_t k, scm_t_array_dim *s, ssize_t pos,
|
||
SCM *o)
|
||
{
|
||
if (k==scm_array_handle_rank (handle))
|
||
*o = handle->array;
|
||
else
|
||
{
|
||
*o = scm_i_make_array (k);
|
||
SCM_I_ARRAY_SET_V (*o, handle->vector);
|
||
SCM_I_ARRAY_SET_BASE (*o, pos + handle->base);
|
||
scm_t_array_dim * os = SCM_I_ARRAY_DIMS (*o);
|
||
for (; k>0; --k, ++s, ++os)
|
||
{
|
||
os->ubnd = s->ubnd;
|
||
os->lbnd = s->lbnd;
|
||
os->inc = s->inc;
|
||
}
|
||
}
|
||
}
|
||
|
||
SCM_DEFINE (scm_array_slice, "array-slice", 1, 0, 1,
|
||
(SCM ra, SCM indices),
|
||
"Return the array slice @var{ra}[@var{indices} ..., ...]\n"
|
||
"The rank of @var{ra} must equal to the number of indices or larger.\n\n"
|
||
"See also @code{array-ref}, @code{array-cell-ref}, @code{array-cell-set!}.\n\n"
|
||
"@code{array-slice} may return a rank-0 array. For example:\n"
|
||
"@lisp\n"
|
||
"(array-slice #2((1 2 3) (4 5 6)) 1 1) @result{} #0(5)\n"
|
||
"(array-slice #2((1 2 3) (4 5 6)) 1) @result{} #(4 5 6)\n"
|
||
"(array-slice #2((1 2 3) (4 5 6))) @result{} #2((1 2 3) (4 5 6))\n"
|
||
"(array-slice #0(5) @result{} #0(5).\n"
|
||
"@end lisp")
|
||
#define FUNC_NAME s_scm_array_slice
|
||
{
|
||
scm_t_array_handle handle;
|
||
scm_array_get_handle (ra, &handle);
|
||
SCM i = indices;
|
||
size_t k;
|
||
ssize_t pos = 0;
|
||
scm_t_array_dim *s;
|
||
array_from_pos (&handle, &k, &i, &pos, &s);
|
||
if (!s)
|
||
{
|
||
scm_array_handle_release (&handle);
|
||
scm_misc_error (FUNC_NAME, "indices ~a out of range for array bounds ~a", scm_list_2 (indices, scm_array_dimensions (ra)));
|
||
}
|
||
SCM o;
|
||
if (scm_is_null (i))
|
||
array_from_get_o (&handle, k, s, pos, &o);
|
||
else
|
||
{
|
||
scm_array_handle_release (&handle);
|
||
scm_misc_error (FUNC_NAME, "too many indices ~a for rank ~a", scm_list_2 (indices, scm_array_rank (ra)));
|
||
}
|
||
scm_array_handle_release (&handle);
|
||
return o;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_array_cell_ref, "array-cell-ref", 1, 0, 1,
|
||
(SCM ra, SCM indices),
|
||
"Return the element at the @code{(@var{indices} ...)} position\n"
|
||
"in array @var{ra}, or the array slice @var{ra}[@var{indices} ..., ...]\n"
|
||
"if the rank of @var{ra} is larger than the number of indices.\n\n"
|
||
"See also @code{array-ref}, @code{array-slice}, @code{array-cell-set!}.\n\n"
|
||
"@code{array-cell-ref} never returns a rank 0 array. For example:\n"
|
||
"@lisp\n"
|
||
"(array-cell-ref #2((1 2 3) (4 5 6)) 1 1) @result{} 5\n"
|
||
"(array-cell-ref #2((1 2 3) (4 5 6)) 1) @result{} #(4 5 6)\n"
|
||
"(array-cell-ref #2((1 2 3) (4 5 6))) @result{} #2((1 2 3) (4 5 6))\n"
|
||
"(array-cell-ref #0(5) @result{} 5.\n"
|
||
"@end lisp")
|
||
#define FUNC_NAME s_scm_array_cell_ref
|
||
{
|
||
scm_t_array_handle handle;
|
||
scm_array_get_handle (ra, &handle);
|
||
SCM i = indices;
|
||
size_t k;
|
||
ssize_t pos = 0;
|
||
scm_t_array_dim *s;
|
||
array_from_pos (&handle, &k, &i, &pos, &s);
|
||
if (!s)
|
||
{
|
||
scm_array_handle_release (&handle);
|
||
scm_misc_error (FUNC_NAME, "indices ~a out of range for array bounds ~a", scm_list_2 (indices, scm_array_dimensions (ra)));
|
||
}
|
||
SCM o;
|
||
if (k>0)
|
||
array_from_get_o (&handle, k, s, pos, &o);
|
||
else if (scm_is_null(i))
|
||
o = scm_array_handle_ref (&handle, pos);
|
||
else
|
||
{
|
||
scm_array_handle_release (&handle);
|
||
scm_misc_error (FUNC_NAME, "too many indices ~a for rank ~a", scm_list_2 (indices, scm_array_rank (ra)));
|
||
}
|
||
scm_array_handle_release (&handle);
|
||
return o;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
SCM_DEFINE (scm_array_cell_set_x, "array-cell-set!", 2, 0, 1,
|
||
(SCM ra, SCM b, SCM indices),
|
||
"Set the array slice @var{ra}[@var{indices} ..., ...] to @var{b}\n."
|
||
"Equivalent to @code{(array-copy! @var{b} (apply array-cell-ref @var{ra} @var{indices}))}\n"
|
||
"if the number of indices is smaller than the rank of @var{ra}; otherwise\n"
|
||
"equivalent to @code{(apply array-set! @var{ra} @var{b} @var{indices})}.\n"
|
||
"This function returns the modified array @var{ra}.\n\n"
|
||
"See also @code{array-ref}, @code{array-cell-ref}, @code{array-slice}.\n\n"
|
||
"For example:\n"
|
||
"@lisp\n"
|
||
"(define A (list->array 2 '((1 2 3) (4 5 6))))\n"
|
||
"(array-cell-set! A #0(99) 1 1) @result{} #2((1 2 3) (4 #0(99) 6))\n"
|
||
"(array-cell-set! A 99 1 1) @result{} #2((1 2 3) (4 99 6))\n"
|
||
"(array-cell-set! A #(a b c) 0) @result{} #2((a b c) (4 99 6))\n"
|
||
"(array-cell-set! A #2((x y z) (9 8 7))) @result{} #2((x y z) (9 8 7))\n\n"
|
||
"(define B (make-array 0))\n"
|
||
"(array-cell-set! B 15) @result{} #0(15)\n"
|
||
"@end lisp")
|
||
#define FUNC_NAME s_scm_array_cell_set_x
|
||
{
|
||
scm_t_array_handle handle;
|
||
scm_array_get_handle (ra, &handle);
|
||
SCM i = indices;
|
||
size_t k;
|
||
ssize_t pos = 0;
|
||
scm_t_array_dim *s;
|
||
array_from_pos (&handle, &k, &i, &pos, &s);
|
||
if (!s)
|
||
{
|
||
scm_array_handle_release (&handle);
|
||
scm_misc_error (FUNC_NAME, "indices ~a out of range for array bounds ~a", scm_list_2 (indices, scm_array_dimensions (ra)));
|
||
}
|
||
if (k>0)
|
||
{
|
||
SCM o;
|
||
array_from_get_o(&handle, k, s, pos, &o);
|
||
scm_array_handle_release(&handle);
|
||
/* an error is still possible here if o and b don't match. */
|
||
/* FIXME copying like this wastes the handle, and the bounds matching
|
||
behavior of array-copy! is not strict. */
|
||
scm_array_copy_x(b, o);
|
||
}
|
||
else if (scm_is_null(i))
|
||
{
|
||
scm_array_handle_set (&handle, pos, b); /* ra may be non-ARRAYP */
|
||
scm_array_handle_release (&handle);
|
||
}
|
||
else
|
||
{
|
||
scm_array_handle_release (&handle);
|
||
scm_misc_error (FUNC_NAME, "too many indices ~a for rank ~a", scm_list_2 (indices, scm_array_rank (ra)));
|
||
}
|
||
return ra;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
#undef ARRAY_FROM_GET_O
|
||
|
||
|
||
/* args are RA . DIMS */
|
||
SCM_DEFINE (scm_transpose_array, "transpose-array", 1, 0, 1,
|
||
(SCM ra, SCM args),
|
||
"Return an array sharing contents with @var{ra}, but with\n"
|
||
"dimensions arranged in a different order. There must be one\n"
|
||
"@var{dim} argument for each dimension of @var{ra}.\n"
|
||
"@var{dim0}, @var{dim1}, @dots{} should be integers between 0\n"
|
||
"and the rank of the array to be returned. Each integer in that\n"
|
||
"range must appear at least once in the argument list.\n"
|
||
"\n"
|
||
"The values of @var{dim0}, @var{dim1}, @dots{} correspond to\n"
|
||
"dimensions in the array to be returned, their positions in the\n"
|
||
"argument list to dimensions of @var{ra}. Several @var{dim}s\n"
|
||
"may have the same value, in which case the returned array will\n"
|
||
"have smaller rank than @var{ra}.\n"
|
||
"\n"
|
||
"@lisp\n"
|
||
"(transpose-array '#2((a b) (c d)) 1 0) @result{} #2((a c) (b d))\n"
|
||
"(transpose-array '#2((a b) (c d)) 0 0) @result{} #1(a d)\n"
|
||
"(transpose-array '#3(((a b c) (d e f)) ((1 2 3) (4 5 6))) 1 1 0) @result{}\n"
|
||
" #2((a 4) (b 5) (c 6))\n"
|
||
"@end lisp")
|
||
#define FUNC_NAME s_scm_transpose_array
|
||
{
|
||
SCM res, vargs;
|
||
scm_t_array_dim *s, *r;
|
||
int ndim, i, k;
|
||
|
||
SCM_VALIDATE_REST_ARGUMENT (args);
|
||
SCM_ASSERT (SCM_HEAP_OBJECT_P (ra), ra, SCM_ARG1, FUNC_NAME);
|
||
|
||
switch (scm_c_array_rank (ra))
|
||
{
|
||
case 0:
|
||
if (!scm_is_null (args))
|
||
SCM_WRONG_NUM_ARGS ();
|
||
return ra;
|
||
case 1:
|
||
/* Make sure that we are called with a single zero as
|
||
arguments.
|
||
*/
|
||
if (scm_is_null (args) || !scm_is_null (SCM_CDR (args)))
|
||
SCM_WRONG_NUM_ARGS ();
|
||
SCM_VALIDATE_INT_COPY (SCM_ARG2, SCM_CAR (args), i);
|
||
SCM_ASSERT_RANGE (SCM_ARG2, SCM_CAR (args), i == 0);
|
||
return ra;
|
||
default:
|
||
vargs = scm_vector (args);
|
||
if (SCM_SIMPLE_VECTOR_LENGTH (vargs) != SCM_I_ARRAY_NDIM (ra))
|
||
SCM_WRONG_NUM_ARGS ();
|
||
ndim = 0;
|
||
for (k = 0; k < SCM_I_ARRAY_NDIM (ra); k++)
|
||
{
|
||
i = scm_to_signed_integer (SCM_SIMPLE_VECTOR_REF (vargs, k),
|
||
0, SCM_I_ARRAY_NDIM(ra));
|
||
if (ndim < i)
|
||
ndim = i;
|
||
}
|
||
ndim++;
|
||
res = scm_i_make_array (ndim);
|
||
SCM_I_ARRAY_SET_V (res, SCM_I_ARRAY_V (ra));
|
||
SCM_I_ARRAY_SET_BASE (res, SCM_I_ARRAY_BASE (ra));
|
||
for (k = ndim; k--;)
|
||
{
|
||
SCM_I_ARRAY_DIMS (res)[k].lbnd = 0;
|
||
SCM_I_ARRAY_DIMS (res)[k].ubnd = -1;
|
||
}
|
||
for (k = SCM_I_ARRAY_NDIM (ra); k--;)
|
||
{
|
||
i = scm_to_int (SCM_SIMPLE_VECTOR_REF (vargs, k));
|
||
s = &(SCM_I_ARRAY_DIMS (ra)[k]);
|
||
r = &(SCM_I_ARRAY_DIMS (res)[i]);
|
||
if (r->ubnd < r->lbnd)
|
||
{
|
||
r->lbnd = s->lbnd;
|
||
r->ubnd = s->ubnd;
|
||
r->inc = s->inc;
|
||
ndim--;
|
||
}
|
||
else
|
||
{
|
||
if (r->ubnd > s->ubnd)
|
||
r->ubnd = s->ubnd;
|
||
if (r->lbnd < s->lbnd)
|
||
{
|
||
SCM_I_ARRAY_SET_BASE (res, SCM_I_ARRAY_BASE (res) + (s->lbnd - r->lbnd) * r->inc);
|
||
r->lbnd = s->lbnd;
|
||
}
|
||
r->inc += s->inc;
|
||
}
|
||
}
|
||
if (ndim > 0)
|
||
SCM_MISC_ERROR ("bad argument list", SCM_EOL);
|
||
return res;
|
||
}
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
/* attempts to unroll an array into a one-dimensional array.
|
||
returns the unrolled array or #f if it can't be done. */
|
||
/* if strict is true, return #f if returned array
|
||
wouldn't have contiguous elements. */
|
||
SCM_DEFINE (scm_array_contents, "array-contents", 1, 1, 0,
|
||
(SCM ra, SCM strict),
|
||
"If @var{ra} may be @dfn{unrolled} into a one dimensional shared\n"
|
||
"array without changing their order (last subscript changing\n"
|
||
"fastest), then @code{array-contents} returns that shared array,\n"
|
||
"otherwise it returns @code{#f}. All arrays made by\n"
|
||
"@code{make-array} and @code{make-uniform-array} may be unrolled,\n"
|
||
"some arrays made by @code{make-shared-array} may not be. If\n"
|
||
"the optional argument @var{strict} is provided, a shared array\n"
|
||
"will be returned only if its elements are stored contiguously\n"
|
||
"in memory.")
|
||
#define FUNC_NAME s_scm_array_contents
|
||
{
|
||
if (SCM_I_ARRAYP (ra))
|
||
{
|
||
SCM v;
|
||
size_t ndim = SCM_I_ARRAY_NDIM (ra);
|
||
scm_t_array_dim *s = SCM_I_ARRAY_DIMS (ra);
|
||
size_t k = ndim;
|
||
size_t len = 1;
|
||
|
||
if (k)
|
||
{
|
||
ssize_t last_inc = s[k - 1].inc;
|
||
while (k--)
|
||
{
|
||
if (len*last_inc != s[k].inc)
|
||
return SCM_BOOL_F;
|
||
len *= (s[k].ubnd - s[k].lbnd + 1);
|
||
}
|
||
}
|
||
|
||
if (!SCM_UNBNDP (strict) && scm_is_true (strict))
|
||
{
|
||
if (ndim && (1 != s[ndim - 1].inc))
|
||
return SCM_BOOL_F;
|
||
if (scm_is_bitvector (SCM_I_ARRAY_V (ra))
|
||
&& (len != scm_c_bitvector_length (SCM_I_ARRAY_V (ra)) ||
|
||
SCM_I_ARRAY_BASE (ra) % SCM_INTPTR_T_BIT ||
|
||
len % SCM_INTPTR_T_BIT))
|
||
return SCM_BOOL_F;
|
||
}
|
||
|
||
v = SCM_I_ARRAY_V (ra);
|
||
if ((len == scm_c_array_length (v)) && (0 == SCM_I_ARRAY_BASE (ra)))
|
||
return v;
|
||
else
|
||
{
|
||
SCM sra = scm_i_make_array (1);
|
||
SCM_I_ARRAY_DIMS (sra)->lbnd = 0;
|
||
SCM_I_ARRAY_DIMS (sra)->ubnd = len - 1;
|
||
SCM_I_ARRAY_SET_V (sra, v);
|
||
SCM_I_ARRAY_SET_BASE (sra, SCM_I_ARRAY_BASE (ra));
|
||
SCM_I_ARRAY_DIMS (sra)->inc = (ndim ? SCM_I_ARRAY_DIMS (ra)[ndim - 1].inc : 1);
|
||
return sra;
|
||
}
|
||
}
|
||
else if (scm_is_array (ra))
|
||
return ra;
|
||
else
|
||
scm_wrong_type_arg_msg (NULL, 0, ra, "array");
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
|
||
static void
|
||
list_to_array (SCM lst, scm_t_array_handle *handle, ssize_t pos, size_t k)
|
||
{
|
||
if (k == scm_array_handle_rank (handle))
|
||
scm_array_handle_set (handle, pos, lst);
|
||
else
|
||
{
|
||
scm_t_array_dim *dim = scm_array_handle_dims (handle) + k;
|
||
ssize_t inc = dim->inc;
|
||
size_t len = 1 + dim->ubnd - dim->lbnd, n;
|
||
char *errmsg = NULL;
|
||
|
||
n = len;
|
||
while (n > 0 && scm_is_pair (lst))
|
||
{
|
||
list_to_array (SCM_CAR (lst), handle, pos, k + 1);
|
||
pos += inc;
|
||
lst = SCM_CDR (lst);
|
||
n -= 1;
|
||
}
|
||
if (n != 0)
|
||
errmsg = "too few elements for array dimension ~a, need ~a";
|
||
if (!scm_is_null (lst))
|
||
errmsg = "too many elements for array dimension ~a, want ~a";
|
||
if (errmsg)
|
||
scm_misc_error (NULL, errmsg, scm_list_2 (scm_from_size_t (k),
|
||
scm_from_size_t (len)));
|
||
}
|
||
}
|
||
|
||
|
||
SCM_DEFINE (scm_list_to_typed_array, "list->typed-array", 3, 0, 0,
|
||
(SCM type, SCM shape, SCM lst),
|
||
"Return an array of the type @var{type}\n"
|
||
"with elements the same as those of @var{lst}.\n"
|
||
"\n"
|
||
"The argument @var{shape} determines the number of dimensions\n"
|
||
"of the array and their shape. It is either an exact integer,\n"
|
||
"giving the\n"
|
||
"number of dimensions directly, or a list whose length\n"
|
||
"specifies the number of dimensions and each element specified\n"
|
||
"the lower and optionally the upper bound of the corresponding\n"
|
||
"dimension.\n"
|
||
"When the element is list of two elements, these elements\n"
|
||
"give the lower and upper bounds. When it is an exact\n"
|
||
"integer, it gives only the lower bound.")
|
||
#define FUNC_NAME s_scm_list_to_typed_array
|
||
{
|
||
SCM row;
|
||
SCM ra;
|
||
scm_t_array_handle handle;
|
||
|
||
row = lst;
|
||
if (scm_is_integer (shape))
|
||
{
|
||
size_t k = scm_to_size_t (shape);
|
||
shape = SCM_EOL;
|
||
while (k-- > 0)
|
||
{
|
||
shape = scm_cons (scm_length (row), shape);
|
||
if (k > 0 && !scm_is_null (row))
|
||
row = scm_car (row);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
SCM shape_spec = shape;
|
||
shape = SCM_EOL;
|
||
while (1)
|
||
{
|
||
SCM spec = scm_car (shape_spec);
|
||
if (scm_is_pair (spec))
|
||
shape = scm_cons (spec, shape);
|
||
else
|
||
shape = scm_cons (scm_list_2 (spec,
|
||
scm_sum (scm_sum (spec,
|
||
scm_length (row)),
|
||
scm_from_int (-1))),
|
||
shape);
|
||
shape_spec = scm_cdr (shape_spec);
|
||
if (scm_is_pair (shape_spec))
|
||
{
|
||
if (!scm_is_null (row))
|
||
row = scm_car (row);
|
||
}
|
||
else
|
||
break;
|
||
}
|
||
}
|
||
|
||
ra = scm_make_typed_array (type, SCM_UNSPECIFIED,
|
||
scm_reverse_x (shape, SCM_EOL));
|
||
|
||
scm_array_get_handle (ra, &handle);
|
||
list_to_array (lst, &handle, 0, 0);
|
||
scm_array_handle_release (&handle);
|
||
|
||
return ra;
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
SCM_DEFINE (scm_list_to_array, "list->array", 2, 0, 0,
|
||
(SCM ndim, SCM lst),
|
||
"Return an array with elements the same as those of @var{lst}.")
|
||
#define FUNC_NAME s_scm_list_to_array
|
||
{
|
||
return scm_list_to_typed_array (SCM_BOOL_T, ndim, lst);
|
||
}
|
||
#undef FUNC_NAME
|
||
|
||
/* Print dimension DIM of ARRAY.
|
||
*/
|
||
|
||
static int
|
||
scm_i_print_array_dimension (scm_t_array_handle *h, int dim, int pos,
|
||
SCM port, scm_print_state *pstate)
|
||
{
|
||
if (dim == h->ndims)
|
||
scm_iprin1 (scm_array_handle_ref (h, pos), port, pstate);
|
||
else
|
||
{
|
||
ssize_t i;
|
||
scm_putc ('(', port);
|
||
for (i = h->dims[dim].lbnd; i <= h->dims[dim].ubnd;
|
||
i++, pos += h->dims[dim].inc)
|
||
{
|
||
scm_i_print_array_dimension (h, dim+1, pos, port, pstate);
|
||
if (i < h->dims[dim].ubnd)
|
||
scm_putc (' ', port);
|
||
}
|
||
scm_putc (')', port);
|
||
}
|
||
return 1;
|
||
}
|
||
|
||
int
|
||
scm_i_print_array (SCM array, SCM port, scm_print_state *pstate)
|
||
{
|
||
scm_t_array_handle h;
|
||
int d;
|
||
|
||
scm_call_2 (scm_c_private_ref ("ice-9 arrays", "array-print-prefix"),
|
||
array, port);
|
||
|
||
scm_array_get_handle (array, &h);
|
||
|
||
if (h.ndims == 0)
|
||
{
|
||
/* Rank zero arrays, which are really just scalars, are printed
|
||
specially. The consequent way would be to print them as
|
||
|
||
#0 OBJ
|
||
|
||
where OBJ is the printed representation of the scalar, but we
|
||
print them instead as
|
||
|
||
#0(OBJ)
|
||
|
||
to make them look less strange.
|
||
|
||
Just printing them as
|
||
|
||
OBJ
|
||
|
||
would be correct in a way as well, but zero rank arrays are
|
||
not really the same as Scheme values since they are boxed and
|
||
can be modified with array-set!, say.
|
||
*/
|
||
scm_putc ('(', port);
|
||
scm_i_print_array_dimension (&h, 0, 0, port, pstate);
|
||
scm_putc (')', port);
|
||
d = 1;
|
||
}
|
||
else
|
||
d = scm_i_print_array_dimension (&h, 0, 0, port, pstate);
|
||
|
||
scm_array_handle_release (&h);
|
||
return d;
|
||
}
|
||
|
||
void
|
||
scm_init_arrays ()
|
||
{
|
||
scm_add_feature ("array");
|
||
#include "arrays.x"
|
||
}
|