generalized-arrays.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411
  1. /* Copyright 1995-1998,2000-2006,2009-2010,2013-2014,2018
  2. Free Software Foundation, Inc.
  3. This file is part of Guile.
  4. Guile is free software: you can redistribute it and/or modify it
  5. under the terms of the GNU Lesser General Public License as published
  6. by the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. Guile is distributed in the hope that it will be useful, but WITHOUT
  9. ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10. FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
  11. License for more details.
  12. You should have received a copy of the GNU Lesser General Public
  13. License along with Guile. If not, see
  14. <https://www.gnu.org/licenses/>. */
  15. #ifdef HAVE_CONFIG_H
  16. # include <config.h>
  17. #endif
  18. #include <errno.h>
  19. #include <stdio.h>
  20. #include <string.h>
  21. #include "array-handle.h"
  22. #include "gsubr.h"
  23. #include "list.h"
  24. #include "numbers.h"
  25. #include "pairs.h"
  26. #include "generalized-arrays.h"
  27. SCM_INTERNAL SCM scm_i_array_ref (SCM v,
  28. SCM idx0, SCM idx1, SCM idxN);
  29. SCM_INTERNAL SCM scm_i_array_set_x (SCM v, SCM obj,
  30. SCM idx0, SCM idx1, SCM idxN);
  31. int
  32. scm_is_array (SCM obj)
  33. {
  34. if (!SCM_HEAP_OBJECT_P (obj))
  35. return 0;
  36. switch (SCM_TYP7 (obj))
  37. {
  38. case scm_tc7_string:
  39. case scm_tc7_vector:
  40. case scm_tc7_bitvector:
  41. case scm_tc7_bytevector:
  42. case scm_tc7_array:
  43. return 1;
  44. default:
  45. return 0;
  46. }
  47. }
  48. SCM_DEFINE (scm_array_p_2, "array?", 1, 0, 0,
  49. (SCM obj),
  50. "Return @code{#t} if the @var{obj} is an array, and @code{#f} if\n"
  51. "not.")
  52. #define FUNC_NAME s_scm_array_p_2
  53. {
  54. return scm_from_bool (scm_is_array (obj));
  55. }
  56. #undef FUNC_NAME
  57. /* The array type predicate, with an extra argument kept for backward
  58. compatibility. Note that we can't use `SCM_DEFINE' directly because there
  59. would be an argument count mismatch that would be caught by
  60. `snarf-check-and-output-texi.scm'. */
  61. SCM
  62. scm_array_p (SCM obj, SCM unused)
  63. {
  64. return scm_array_p_2 (obj);
  65. }
  66. int
  67. scm_is_typed_array (SCM obj, SCM type)
  68. {
  69. int ret = 0;
  70. if (scm_is_array (obj))
  71. {
  72. scm_t_array_handle h;
  73. scm_array_get_handle (obj, &h);
  74. ret = scm_is_eq (scm_array_handle_element_type (&h), type);
  75. scm_array_handle_release (&h);
  76. }
  77. return ret;
  78. }
  79. SCM_DEFINE (scm_typed_array_p, "typed-array?", 2, 0, 0,
  80. (SCM obj, SCM type),
  81. "Return @code{#t} if the @var{obj} is an array of type\n"
  82. "@var{type}, and @code{#f} if not.")
  83. #define FUNC_NAME s_scm_typed_array_p
  84. {
  85. return scm_from_bool (scm_is_typed_array (obj, type));
  86. }
  87. #undef FUNC_NAME
  88. size_t
  89. scm_c_array_length (SCM array)
  90. {
  91. scm_t_array_handle handle;
  92. size_t res;
  93. scm_array_get_handle (array, &handle);
  94. if (scm_array_handle_rank (&handle) < 1)
  95. {
  96. scm_array_handle_release (&handle);
  97. scm_wrong_type_arg_msg (NULL, 0, array, "array of nonzero rank");
  98. }
  99. res = handle.dims[0].ubnd - handle.dims[0].lbnd + 1;
  100. scm_array_handle_release (&handle);
  101. return res;
  102. }
  103. SCM_DEFINE (scm_array_length, "array-length", 1, 0, 0,
  104. (SCM array),
  105. "Return the length of an array: its first dimension.\n"
  106. "It is an error to ask for the length of an array of rank 0.")
  107. #define FUNC_NAME s_scm_array_length
  108. {
  109. return scm_from_size_t (scm_c_array_length (array));
  110. }
  111. #undef FUNC_NAME
  112. SCM_DEFINE (scm_array_dimensions, "array-dimensions", 1, 0, 0,
  113. (SCM ra),
  114. "@code{array-dimensions} is similar to @code{array-shape} but replaces\n"
  115. "elements with a @code{0} minimum with one greater than the maximum. So:\n"
  116. "@lisp\n"
  117. "(array-dimensions (make-array 'foo '(-1 3) 5)) @result{} ((-1 3) 5)\n"
  118. "@end lisp")
  119. #define FUNC_NAME s_scm_array_dimensions
  120. {
  121. scm_t_array_handle handle;
  122. scm_t_array_dim *s;
  123. SCM res = SCM_EOL;
  124. size_t k;
  125. scm_array_get_handle (ra, &handle);
  126. s = scm_array_handle_dims (&handle);
  127. k = scm_array_handle_rank (&handle);
  128. while (k--)
  129. res = scm_cons (s[k].lbnd
  130. ? scm_cons2 (scm_from_ssize_t (s[k].lbnd),
  131. scm_from_ssize_t (s[k].ubnd),
  132. SCM_EOL)
  133. : scm_from_ssize_t (1 + s[k].ubnd),
  134. res);
  135. scm_array_handle_release (&handle);
  136. return res;
  137. }
  138. #undef FUNC_NAME
  139. SCM_DEFINE (scm_array_type, "array-type", 1, 0, 0,
  140. (SCM ra),
  141. "")
  142. #define FUNC_NAME s_scm_array_type
  143. {
  144. scm_t_array_handle h;
  145. SCM type;
  146. scm_array_get_handle (ra, &h);
  147. type = scm_array_handle_element_type (&h);
  148. scm_array_handle_release (&h);
  149. return type;
  150. }
  151. #undef FUNC_NAME
  152. SCM_DEFINE (scm_array_type_code,
  153. "array-type-code", 1, 0, 0,
  154. (SCM array),
  155. "Return the type of the elements in @var{array},\n"
  156. "as an integer code.")
  157. #define FUNC_NAME s_scm_array_type_code
  158. {
  159. scm_t_array_handle h;
  160. scm_t_array_element_type element_type;
  161. scm_array_get_handle (array, &h);
  162. element_type = h.element_type;
  163. scm_array_handle_release (&h);
  164. return scm_from_uint16 (element_type);
  165. }
  166. #undef FUNC_NAME
  167. SCM_DEFINE (scm_array_in_bounds_p, "array-in-bounds?", 1, 0, 1,
  168. (SCM ra, SCM args),
  169. "Return @code{#t} if its arguments would be acceptable to\n"
  170. "@code{array-ref}.")
  171. #define FUNC_NAME s_scm_array_in_bounds_p
  172. {
  173. SCM res = SCM_BOOL_T;
  174. size_t k, ndim;
  175. scm_t_array_dim *s;
  176. scm_t_array_handle handle;
  177. SCM_VALIDATE_REST_ARGUMENT (args);
  178. scm_array_get_handle (ra, &handle);
  179. s = scm_array_handle_dims (&handle);
  180. ndim = scm_array_handle_rank (&handle);
  181. for (k = 0; k < ndim; k++)
  182. {
  183. long ind;
  184. if (!scm_is_pair (args))
  185. SCM_WRONG_NUM_ARGS ();
  186. ind = scm_to_long (SCM_CAR (args));
  187. args = SCM_CDR (args);
  188. if (ind < s[k].lbnd || ind > s[k].ubnd)
  189. {
  190. res = SCM_BOOL_F;
  191. /* We do not stop the checking after finding a violation
  192. since we want to validate the type-correctness and
  193. number of arguments in any case.
  194. */
  195. }
  196. }
  197. scm_array_handle_release (&handle);
  198. return res;
  199. }
  200. #undef FUNC_NAME
  201. SCM
  202. scm_c_array_ref_1 (SCM array, ssize_t idx0)
  203. {
  204. scm_t_array_handle handle;
  205. SCM res;
  206. scm_array_get_handle (array, &handle);
  207. res = scm_array_handle_ref (&handle, scm_array_handle_pos_1 (&handle, idx0));
  208. scm_array_handle_release (&handle);
  209. return res;
  210. }
  211. SCM
  212. scm_c_array_ref_2 (SCM array, ssize_t idx0, ssize_t idx1)
  213. {
  214. scm_t_array_handle handle;
  215. SCM res;
  216. scm_array_get_handle (array, &handle);
  217. res = scm_array_handle_ref (&handle, scm_array_handle_pos_2 (&handle, idx0, idx1));
  218. scm_array_handle_release (&handle);
  219. return res;
  220. }
  221. SCM
  222. scm_array_ref (SCM v, SCM args)
  223. {
  224. scm_t_array_handle handle;
  225. SCM res;
  226. scm_array_get_handle (v, &handle);
  227. res = scm_array_handle_ref (&handle, scm_array_handle_pos (&handle, args));
  228. scm_array_handle_release (&handle);
  229. return res;
  230. }
  231. void
  232. scm_c_array_set_1_x (SCM array, SCM obj, ssize_t idx0)
  233. {
  234. scm_t_array_handle handle;
  235. scm_array_get_handle (array, &handle);
  236. scm_array_handle_set (&handle, scm_array_handle_pos_1 (&handle, idx0),
  237. obj);
  238. scm_array_handle_release (&handle);
  239. }
  240. void
  241. scm_c_array_set_2_x (SCM array, SCM obj, ssize_t idx0, ssize_t idx1)
  242. {
  243. scm_t_array_handle handle;
  244. scm_array_get_handle (array, &handle);
  245. scm_array_handle_set (&handle, scm_array_handle_pos_2 (&handle, idx0, idx1),
  246. obj);
  247. scm_array_handle_release (&handle);
  248. }
  249. SCM
  250. scm_array_set_x (SCM v, SCM obj, SCM args)
  251. {
  252. scm_t_array_handle handle;
  253. scm_array_get_handle (v, &handle);
  254. scm_array_handle_set (&handle, scm_array_handle_pos (&handle, args), obj);
  255. scm_array_handle_release (&handle);
  256. return SCM_UNSPECIFIED;
  257. }
  258. SCM_DEFINE (scm_i_array_ref, "array-ref", 1, 2, 1,
  259. (SCM v, SCM idx0, SCM idx1, SCM idxN),
  260. "Return the element at the @code{(idx0, idx1, idxN...)}\n"
  261. "position in array @var{v}.")
  262. #define FUNC_NAME s_scm_i_array_ref
  263. {
  264. if (SCM_UNBNDP (idx0))
  265. return scm_array_ref (v, SCM_EOL);
  266. else if (SCM_UNBNDP (idx1))
  267. return scm_c_array_ref_1 (v, scm_to_ssize_t (idx0));
  268. else if (scm_is_null (idxN))
  269. return scm_c_array_ref_2 (v, scm_to_ssize_t (idx0), scm_to_ssize_t (idx1));
  270. else
  271. return scm_array_ref (v, scm_cons (idx0, scm_cons (idx1, idxN)));
  272. }
  273. #undef FUNC_NAME
  274. SCM_DEFINE (scm_i_array_set_x, "array-set!", 2, 2, 1,
  275. (SCM v, SCM obj, SCM idx0, SCM idx1, SCM idxN),
  276. "Set the element at the @code{(idx0, idx1, idxN...)} position\n"
  277. "in the array @var{v} to @var{obj}. The value returned by\n"
  278. "@code{array-set!} is unspecified.")
  279. #define FUNC_NAME s_scm_i_array_set_x
  280. {
  281. if (SCM_UNBNDP (idx0))
  282. scm_array_set_x (v, obj, SCM_EOL);
  283. else if (SCM_UNBNDP (idx1))
  284. scm_c_array_set_1_x (v, obj, scm_to_ssize_t (idx0));
  285. else if (scm_is_null (idxN))
  286. scm_c_array_set_2_x (v, obj, scm_to_ssize_t (idx0), scm_to_ssize_t (idx1));
  287. else
  288. scm_array_set_x (v, obj, scm_cons (idx0, scm_cons (idx1, idxN)));
  289. return SCM_UNSPECIFIED;
  290. }
  291. #undef FUNC_NAME
  292. static SCM
  293. array_to_list (scm_t_array_handle *h, size_t dim, unsigned long pos)
  294. {
  295. if (dim == scm_array_handle_rank (h))
  296. return scm_array_handle_ref (h, pos);
  297. else
  298. {
  299. SCM res = SCM_EOL;
  300. long inc;
  301. size_t i;
  302. i = h->dims[dim].ubnd - h->dims[dim].lbnd + 1;
  303. inc = h->dims[dim].inc;
  304. pos += (i - 1) * inc;
  305. for (; i > 0; i--, pos -= inc)
  306. res = scm_cons (array_to_list (h, dim + 1, pos), res);
  307. return res;
  308. }
  309. }
  310. SCM_DEFINE (scm_array_to_list, "array->list", 1, 0, 0,
  311. (SCM array),
  312. "Return a list representation of @var{array}.\n\n"
  313. "It is easiest to specify the behavior of this function by\n"
  314. "example:\n"
  315. "@example\n"
  316. "(array->list #0(a)) @result{} 1\n"
  317. "(array->list #1(a b)) @result{} (a b)\n"
  318. "(array->list #2((aa ab) (ba bb)) @result{} ((aa ab) (ba bb))\n"
  319. "@end example\n")
  320. #define FUNC_NAME s_scm_array_to_list
  321. {
  322. scm_t_array_handle h;
  323. SCM res;
  324. scm_array_get_handle (array, &h);
  325. res = array_to_list (&h, 0, 0);
  326. scm_array_handle_release (&h);
  327. return res;
  328. }
  329. #undef FUNC_NAME
  330. void
  331. scm_init_generalized_arrays ()
  332. {
  333. #include "generalized-arrays.x"
  334. }