mruby-random: Random#rand(nil) should raise TypeError
[mruby.git] / src / array.c
blobf6051157c0f0dd0d1d4a5ae044bd6d180ea093a6
1 /*
2 ** array.c - Array class
3 **
4 ** See Copyright Notice in mruby.h
5 */
7 #include <mruby.h>
8 #include <mruby/array.h>
9 #include <mruby/class.h>
10 #include <mruby/string.h>
11 #include <mruby/range.h>
12 #include <mruby/proc.h>
13 #include <mruby/internal.h>
14 #include <mruby/presym.h>
15 #include "value_array.h"
17 #define ARY_DEFAULT_LEN 4
18 #define ARY_SHRINK_RATIO 5 /* must be larger than 2 */
19 #define ARY_C_MAX_SIZE (SIZE_MAX / sizeof(mrb_value))
20 #ifndef MRB_ARY_LENGTH_MAX
21 #define MRB_ARY_LENGTH_MAX 131072
22 #endif
23 #define ARY_MAX_SIZE ((mrb_int)((ARY_C_MAX_SIZE < (size_t)MRB_INT_MAX) ? ARY_C_MAX_SIZE : MRB_INT_MAX-1))
25 static void
26 ary_too_big(mrb_state *mrb)
28 mrb_raise(mrb, E_ARGUMENT_ERROR, "array size too big");
31 static inline void
32 ary_check_too_big(mrb_state *mrb, mrb_int a, mrb_int b)
34 if (a > ARY_MAX_SIZE - b || a < 0)
35 ary_too_big(mrb);
36 #if MRB_ARY_LENGTH_MAX != 0
37 if (a > MRB_ARY_LENGTH_MAX - b || a < 0)
38 ary_too_big(mrb);
39 #endif
42 static struct RArray*
43 ary_new_capa(mrb_state *mrb, mrb_int capa)
45 ary_check_too_big(mrb, capa, 0);
47 size_t blen = capa * sizeof(mrb_value);
48 struct RArray *a = MRB_OBJ_ALLOC(mrb, MRB_TT_ARRAY, mrb->array_class);
50 if (capa <= MRB_ARY_EMBED_LEN_MAX) {
51 ARY_SET_EMBED_LEN(a, 0);
53 else {
54 a->as.heap.ptr = (mrb_value *)mrb_malloc(mrb, blen);
55 a->as.heap.aux.capa = capa;
56 a->as.heap.len = 0;
59 return a;
62 MRB_API mrb_value
63 mrb_ary_new_capa(mrb_state *mrb, mrb_int capa)
65 struct RArray *a = ary_new_capa(mrb, capa);
66 return mrb_obj_value(a);
69 MRB_API mrb_value
70 mrb_ary_new(mrb_state *mrb)
72 return mrb_ary_new_capa(mrb, 0);
76 * To copy array, use this instead of memcpy because of portability
77 * * gcc on ARM may fail optimization of memcpy
78 * https://gcc.gnu.org/bugzilla/show_bug.cgi?id=56620
79 * * gcc on MIPS also fail
80 * https://gcc.gnu.org/bugzilla/show_bug.cgi?id=39755
81 * * memcpy doesn't exist on freestanding environment
83 * If you optimize for binary size, use memcpy instead of this at your own risk
84 * of above portability issue.
86 * See also https://togetter.com/li/462898 (Japanese)
88 static inline void
89 array_copy(mrb_value *dst, const mrb_value *src, mrb_int size)
91 for (mrb_int i = 0; i < size; i++) {
92 dst[i] = src[i];
96 static struct RArray*
97 ary_new_from_values(mrb_state *mrb, mrb_int size, const mrb_value *vals)
99 struct RArray *a = ary_new_capa(mrb, size);
101 array_copy(ARY_PTR(a), vals, size);
102 ARY_SET_LEN(a, size);
104 return a;
107 MRB_API mrb_value
108 mrb_ary_new_from_values(mrb_state *mrb, mrb_int size, const mrb_value *vals)
110 struct RArray *a = ary_new_from_values(mrb, size, vals);
111 return mrb_obj_value(a);
114 MRB_API mrb_value
115 mrb_assoc_new(mrb_state *mrb, mrb_value car, mrb_value cdr)
117 struct RArray *a = ary_new_capa(mrb, 2);
118 mrb_value *p = ARY_PTR(a);
120 p[0] = car;
121 p[1] = cdr;
122 ARY_SET_LEN(a, 2);
123 return mrb_obj_value(a);
126 static void
127 ary_fill_with_nil(mrb_value *ptr, mrb_int size)
129 mrb_value nil = mrb_nil_value();
131 while (size--) {
132 *ptr++ = nil;
136 #define ary_modify_check(mrb, a) mrb_check_frozen((mrb), (a))
138 static void
139 ary_modify(mrb_state *mrb, struct RArray *a)
141 ary_modify_check(mrb, a);
143 if (ARY_SHARED_P(a)) {
144 mrb_shared_array *shared = a->as.heap.aux.shared;
146 if (shared->refcnt == 1 && a->as.heap.ptr == shared->ptr) {
147 a->as.heap.ptr = shared->ptr;
148 a->as.heap.aux.capa = a->as.heap.len;
149 mrb_free(mrb, shared);
151 else {
152 mrb_value *p = a->as.heap.ptr;
153 mrb_value *ptr = (mrb_value*)mrb_malloc(mrb, a->as.heap.len * sizeof(mrb_value));
155 if (p) {
156 array_copy(ptr, p, a->as.heap.len);
158 a->as.heap.ptr = ptr;
159 a->as.heap.aux.capa = a->as.heap.len;
160 mrb_ary_decref(mrb, shared);
162 ARY_UNSET_SHARED_FLAG(a);
166 MRB_API void
167 mrb_ary_modify(mrb_state *mrb, struct RArray* a)
169 mrb_write_barrier(mrb, (struct RBasic*)a);
170 ary_modify(mrb, a);
173 static void
174 ary_make_shared(mrb_state *mrb, struct RArray *a)
176 if (!ARY_SHARED_P(a) && !ARY_EMBED_P(a)) {
177 mrb_shared_array *shared = (mrb_shared_array*)mrb_malloc(mrb, sizeof(mrb_shared_array));
178 mrb_value *ptr = a->as.heap.ptr;
179 mrb_int len = a->as.heap.len;
181 shared->refcnt = 1;
182 if (a->as.heap.aux.capa > len) {
183 a->as.heap.ptr = shared->ptr = (mrb_value*)mrb_realloc(mrb, ptr, sizeof(mrb_value)*len+1);
185 else {
186 shared->ptr = ptr;
188 shared->len = len;
189 a->as.heap.aux.shared = shared;
190 ARY_SET_SHARED_FLAG(a);
194 static void
195 ary_expand_capa(mrb_state *mrb, struct RArray *a, mrb_int len)
197 mrb_int capa = ARY_CAPA(a);
199 ary_check_too_big(mrb, len, 0);
200 if (capa < ARY_DEFAULT_LEN) {
201 capa = ARY_DEFAULT_LEN;
203 while (capa < len) {
204 if (capa <= ARY_MAX_SIZE / 2) {
205 capa *= 2;
207 else {
208 capa = len;
211 if (capa > ARY_MAX_SIZE) {
212 ary_too_big(mrb);
215 if (ARY_EMBED_P(a)) {
216 mrb_value *ptr = ARY_EMBED_PTR(a);
217 mrb_int slen = ARY_EMBED_LEN(a);
218 mrb_value *expanded_ptr = (mrb_value*)mrb_malloc(mrb, sizeof(mrb_value)*capa);
220 ARY_UNSET_EMBED_FLAG(a);
221 array_copy(expanded_ptr, ptr, slen);
222 a->as.heap.len = slen;
223 a->as.heap.aux.capa = capa;
224 a->as.heap.ptr = expanded_ptr;
226 else if (capa > a->as.heap.aux.capa) {
227 mrb_value *expanded_ptr = (mrb_value*)mrb_realloc(mrb, a->as.heap.ptr, sizeof(mrb_value)*capa);
229 a->as.heap.aux.capa = capa;
230 a->as.heap.ptr = expanded_ptr;
234 static void
235 ary_shrink_capa(mrb_state *mrb, struct RArray *a)
237 if (ARY_EMBED_P(a)) return;
239 mrb_int capa = a->as.heap.aux.capa;
241 if (capa < ARY_DEFAULT_LEN * 2) return;
242 if (capa <= a->as.heap.len * ARY_SHRINK_RATIO) return;
244 do {
245 capa /= 2;
246 if (capa < ARY_DEFAULT_LEN) {
247 capa = ARY_DEFAULT_LEN;
248 break;
250 } while (capa > a->as.heap.len * ARY_SHRINK_RATIO);
252 if (capa > a->as.heap.len && capa < a->as.heap.aux.capa) {
253 a->as.heap.aux.capa = capa;
254 a->as.heap.ptr = (mrb_value*)mrb_realloc(mrb, a->as.heap.ptr, sizeof(mrb_value)*capa);
258 MRB_API mrb_value
259 mrb_ary_resize(mrb_state *mrb, mrb_value ary, mrb_int new_len)
261 struct RArray *a = mrb_ary_ptr(ary);
263 ary_modify(mrb, a);
264 mrb_int old_len = RARRAY_LEN(ary);
265 if (old_len != new_len) {
266 if (new_len < old_len) {
267 ary_shrink_capa(mrb, a);
269 else {
270 ary_expand_capa(mrb, a, new_len);
271 ary_fill_with_nil(ARY_PTR(a) + old_len, new_len - old_len);
273 ARY_SET_LEN(a, new_len);
276 return ary;
279 static mrb_value
280 mrb_ary_s_create(mrb_state *mrb, mrb_value klass)
282 const mrb_value *vals;
283 mrb_int len;
285 mrb_get_args(mrb, "*!", &vals, &len);
286 mrb_value ary = mrb_ary_new_from_values(mrb, len, vals);
287 struct RArray *a = mrb_ary_ptr(ary);
288 a->c = mrb_class_ptr(klass);
290 return ary;
293 static void ary_replace(mrb_state*, struct RArray*, struct RArray*);
295 static mrb_value
296 mrb_ary_init(mrb_state *mrb, mrb_value ary)
298 mrb_value ss = mrb_fixnum_value(0);
299 mrb_value obj = mrb_nil_value();
300 mrb_value blk = mrb_nil_value();
302 mrb_get_args(mrb, "|oo&", &ss, &obj, &blk);
304 if (mrb_array_p(ss) && mrb_nil_p(obj) && mrb_nil_p(blk)) {
305 ary_replace(mrb, mrb_ary_ptr(ary), mrb_ary_ptr(ss));
306 return ary;
309 mrb_int size = mrb_as_int(mrb, ss);
310 struct RArray *a = mrb_ary_ptr(ary);
312 if (ARY_CAPA(a) < size) {
313 ary_expand_capa(mrb, a, size);
316 int ai = mrb_gc_arena_save(mrb);
317 for (mrb_int i=0; i<size; i++) {
318 mrb_value val;
319 if (mrb_nil_p(blk)) {
320 val = obj;
322 else {
323 val = mrb_yield(mrb, blk, mrb_fixnum_value(i));
325 mrb_ary_set(mrb, ary, i, val);
326 mrb_gc_arena_restore(mrb, ai); // for mrb_funcall
328 return ary;
331 static void
332 ary_concat(mrb_state *mrb, struct RArray *a, struct RArray *a2)
334 mrb_int len = ARY_LEN(a);
336 if (len == 0) {
337 ary_replace(mrb, a, a2);
338 return;
341 mrb_int len2 = ARY_LEN(a2);
342 ary_check_too_big(mrb, len2, len);
343 ary_modify(mrb, a);
345 mrb_int newlen = len + len2;
346 if (ARY_CAPA(a) < newlen) {
347 ary_expand_capa(mrb, a, newlen);
349 array_copy(ARY_PTR(a)+len, ARY_PTR(a2), len2);
350 mrb_write_barrier(mrb, (struct RBasic*)a);
351 ARY_SET_LEN(a, newlen);
354 MRB_API void
355 mrb_ary_concat(mrb_state *mrb, mrb_value self, mrb_value other)
357 struct RArray *a2 = mrb_ary_ptr(other);
359 ary_concat(mrb, mrb_ary_ptr(self), a2);
363 * call-seq:
364 * array.concat(*other_arrays) -> self
366 * Adds to +array+ all elements from each \Array in +other_arrays+; returns +self+:
368 * a = [0, 1]
369 * a.concat([2, 3], [4, 5]) # => [0, 1, 2, 3, 4, 5]
372 static mrb_value
373 mrb_ary_concat_m(mrb_state *mrb, mrb_value self)
375 mrb_value *args;
376 mrb_int len;
378 mrb_get_args(mrb, "*!", &args, &len);
379 for (int i=0; i<len; i++) {
380 mrb_ensure_array_type(mrb, args[i]);
382 for (int i=0; i<len; i++) {
383 mrb_ary_concat(mrb, self, args[i]);
385 return self;
388 static mrb_value
389 mrb_ary_plus(mrb_state *mrb, mrb_value self)
391 struct RArray *a1 = mrb_ary_ptr(self);
392 const mrb_value *ptr;
393 mrb_int blen;
395 mrb_get_args(mrb, "a", &ptr, &blen);
396 ary_check_too_big(mrb, ARY_LEN(a1), blen);
397 mrb_int len1 = ARY_LEN(a1);
398 struct RArray *a2 = ary_new_capa(mrb, len1 + blen);
399 array_copy(ARY_PTR(a2), ARY_PTR(a1), len1);
400 array_copy(ARY_PTR(a2) + len1, ptr, blen);
401 ARY_SET_LEN(a2, len1+blen);
403 return mrb_obj_value(a2);
406 #define ARY_REPLACE_SHARED_MIN 20
408 static void
409 ary_replace(mrb_state *mrb, struct RArray *a, struct RArray *b)
411 mrb_int len = ARY_LEN(b);
413 ary_modify_check(mrb, a);
414 if (a == b) return;
415 if (ARY_SHARED_P(a)) {
416 mrb_ary_decref(mrb, a->as.heap.aux.shared);
417 a->as.heap.aux.capa = 0;
418 a->as.heap.len = 0;
419 a->as.heap.ptr = NULL;
420 ARY_UNSET_SHARED_FLAG(a);
422 if (ARY_SHARED_P(b)) {
423 shared_b:
424 if (ARY_EMBED_P(a)) {
425 ARY_UNSET_EMBED_FLAG(a);
427 else {
428 mrb_free(mrb, a->as.heap.ptr);
430 a->as.heap.ptr = b->as.heap.ptr;
431 a->as.heap.len = len;
432 a->as.heap.aux.shared = b->as.heap.aux.shared;
433 a->as.heap.aux.shared->refcnt++;
434 ARY_SET_SHARED_FLAG(a);
435 mrb_write_barrier(mrb, (struct RBasic*)a);
436 return;
438 if (!mrb_frozen_p(b) && len > ARY_REPLACE_SHARED_MIN) {
439 ary_make_shared(mrb, b);
440 goto shared_b;
442 if (ARY_CAPA(a) < len)
443 ary_expand_capa(mrb, a, len);
444 array_copy(ARY_PTR(a), ARY_PTR(b), len);
445 mrb_write_barrier(mrb, (struct RBasic*)a);
446 ARY_SET_LEN(a, len);
449 MRB_API void
450 mrb_ary_replace(mrb_state *mrb, mrb_value self, mrb_value other)
452 struct RArray *a1 = mrb_ary_ptr(self);
453 struct RArray *a2 = mrb_ary_ptr(other);
455 if (a1 != a2) {
456 ary_replace(mrb, a1, a2);
460 static mrb_value
461 mrb_ary_replace_m(mrb_state *mrb, mrb_value self)
463 mrb_value other;
465 mrb_get_args(mrb, "A", &other);
466 mrb_ary_replace(mrb, self, other);
468 return self;
471 static mrb_value
472 mrb_ary_times(mrb_state *mrb, mrb_value self)
474 struct RArray *a1 = mrb_ary_ptr(self);
476 mrb_value arg = mrb_get_arg1(mrb);
477 mrb_value tmp = mrb_check_string_type(mrb, arg);
478 if (!mrb_nil_p(tmp)) {
479 return mrb_ary_join(mrb, self, tmp);
482 mrb_int times = mrb_as_int(mrb, arg);
483 if (times < 0) {
484 mrb_raise(mrb, E_ARGUMENT_ERROR, "negative argument");
486 if (times == 0) return mrb_ary_new(mrb);
487 if (ARY_MAX_SIZE / times < ARY_LEN(a1)) {
488 ary_too_big(mrb);
491 mrb_int len1 = ARY_LEN(a1);
492 struct RArray *a2 = ary_new_capa(mrb, len1 * times);
493 ARY_SET_LEN(a2, len1 * times);
495 mrb_value *ptr = ARY_PTR(a2);
496 while (times--) {
497 array_copy(ptr, ARY_PTR(a1), len1);
498 ptr += len1;
501 return mrb_obj_value(a2);
504 static mrb_value
505 mrb_ary_reverse_bang(mrb_state *mrb, mrb_value self)
507 struct RArray *a = mrb_ary_ptr(self);
508 mrb_int len = ARY_LEN(a);
510 if (len > 1) {
511 ary_modify(mrb, a);
513 mrb_value *p1 = ARY_PTR(a);
514 mrb_value *p2 = p1 + len - 1;
516 while (p1 < p2) {
517 mrb_value tmp = *p1;
518 *p1++ = *p2;
519 *p2-- = tmp;
522 return self;
525 static mrb_value
526 mrb_ary_reverse(mrb_state *mrb, mrb_value self)
528 struct RArray *a = mrb_ary_ptr(self), *b = ary_new_capa(mrb, ARY_LEN(a));
529 mrb_int len = ARY_LEN(a);
531 if (len > 0) {
532 mrb_value *p1 = ARY_PTR(a);
533 mrb_value *e = p1 + len;
534 mrb_value *p2 = ARY_PTR(b) + len - 1;
535 while (p1 < e) {
536 *p2-- = *p1++;
538 ARY_SET_LEN(b, len);
540 return mrb_obj_value(b);
543 MRB_API void
544 mrb_ary_push(mrb_state *mrb, mrb_value ary, mrb_value elem)
546 struct RArray *a = mrb_ary_ptr(ary);
547 mrb_int len = ARY_LEN(a);
549 ary_modify(mrb, a);
550 if (len == ARY_CAPA(a))
551 ary_expand_capa(mrb, a, len + 1);
552 ARY_PTR(a)[len] = elem;
553 ARY_SET_LEN(a, len+1);
554 mrb_field_write_barrier_value(mrb, (struct RBasic*)a, elem);
557 static mrb_value
558 mrb_ary_push_m(mrb_state *mrb, mrb_value self)
560 mrb_int argc = mrb_get_argc(mrb);
561 if (argc == 1) {
562 mrb_ary_push(mrb, self, mrb_get_argv(mrb)[0]);
563 return self;
565 struct RArray *a = mrb_ary_ptr(self);
566 mrb_int len = ARY_LEN(a);
567 mrb_int len2 = len + argc;
568 ary_modify(mrb, a);
569 if (ARY_CAPA(a) < len2) {
570 ary_expand_capa(mrb, a, len2);
572 const mrb_value *argv = mrb_get_argv(mrb);
573 array_copy(ARY_PTR(a)+len, argv, argc);
574 ARY_SET_LEN(a, len2);
575 while (argc--) {
576 mrb_field_write_barrier_value(mrb, (struct RBasic*)a, *argv);
577 argv++;
579 return self;
582 MRB_API mrb_value
583 mrb_ary_pop(mrb_state *mrb, mrb_value ary)
585 struct RArray *a = mrb_ary_ptr(ary);
586 mrb_int len = ARY_LEN(a);
588 ary_modify_check(mrb, a);
589 if (len == 0) return mrb_nil_value();
590 ARY_SET_LEN(a, len-1);
591 return ARY_PTR(a)[len-1];
594 #define ARY_SHIFT_SHARED_MIN 10
596 MRB_API mrb_value
597 mrb_ary_shift(mrb_state *mrb, mrb_value self)
599 struct RArray *a = mrb_ary_ptr(self);
600 mrb_int len = ARY_LEN(a);
602 ary_modify_check(mrb, a);
603 if (len == 0) return mrb_nil_value();
604 if (ARY_SHARED_P(a)) {
605 L_SHIFT:
606 a->as.heap.ptr++;
607 a->as.heap.len--;
608 return a->as.heap.ptr[-1];
610 else if (len > ARY_SHIFT_SHARED_MIN) {
611 ary_make_shared(mrb, a);
612 goto L_SHIFT;
614 else {
615 mrb_value *ptr = ARY_PTR(a);
616 mrb_int size = len;
617 mrb_value val = *ptr;
619 while (--size) {
620 *ptr = *(ptr+1);
621 ptr++;
623 ARY_SET_LEN(a, len-1);
624 return val;
628 static mrb_value
629 mrb_ary_shift_m(mrb_state *mrb, mrb_value self)
632 if (mrb_get_argc(mrb) == 0) {
633 return mrb_ary_shift(mrb, self);
636 mrb_int n = mrb_as_int(mrb, mrb_get_arg1(mrb));
637 struct RArray *a = mrb_ary_ptr(self);
638 mrb_int len = ARY_LEN(a);
640 ary_modify_check(mrb, a);
641 if (len == 0 || n == 0) return mrb_ary_new(mrb);
642 if (n < 0) mrb_raise(mrb, E_ARGUMENT_ERROR, "negative array shift");
643 if (n > len) n = len;
644 mrb_value val = mrb_ary_new_from_values(mrb, n, ARY_PTR(a));
645 if (ARY_SHARED_P(a)) {
646 L_SHIFT:
647 a->as.heap.ptr+=n;
648 a->as.heap.len-=n;
649 return val;
651 if (len > ARY_SHIFT_SHARED_MIN) {
652 ary_make_shared(mrb, a);
653 goto L_SHIFT;
655 else if (len == n) {
656 ARY_SET_LEN(a, 0);
658 else {
659 mrb_value *ptr = ARY_PTR(a);
660 mrb_int size = len-n;
662 while (size--) {
663 *ptr = *(ptr+n);
664 ptr++;
666 ARY_SET_LEN(a, len-n);
668 return val;
671 /* self = [1,2,3]
672 item = 0
673 self.unshift item
674 p self #=> [0, 1, 2, 3] */
675 MRB_API mrb_value
676 mrb_ary_unshift(mrb_state *mrb, mrb_value self, mrb_value item)
678 struct RArray *a = mrb_ary_ptr(self);
679 mrb_int len = ARY_LEN(a);
681 if (ARY_SHARED_P(a)
682 && a->as.heap.aux.shared->refcnt == 1 /* shared only referenced from this array */
683 && a->as.heap.ptr - a->as.heap.aux.shared->ptr >= 1) /* there's room for unshifted item */ {
684 a->as.heap.ptr--;
685 a->as.heap.ptr[0] = item;
687 else {
688 mrb_value *ptr;
690 ary_modify(mrb, a);
691 if (ARY_CAPA(a) < len + 1)
692 ary_expand_capa(mrb, a, len + 1);
693 ptr = ARY_PTR(a);
694 value_move(ptr + 1, ptr, len);
695 ptr[0] = item;
697 ARY_SET_LEN(a, len+1);
698 mrb_field_write_barrier_value(mrb, (struct RBasic*)a, item);
700 return self;
704 * call-seq:
705 * array.unshift(*objects) -> self
707 * Prepends the given +objects+ to +self+:
709 * a = [:foo, 'bar', 2]
710 * a.unshift(:bam, :bat) # => [:bam, :bat, :foo, "bar", 2]
712 * Array#prepend is an alias for Array#unshift.
714 * Related: #push, #pop, #shift.
717 static mrb_value
718 mrb_ary_unshift_m(mrb_state *mrb, mrb_value self)
720 struct RArray *a = mrb_ary_ptr(self);
721 mrb_value *ptr;
723 mrb_int alen = mrb_get_argc(mrb);
725 if (alen == 0) {
726 ary_modify_check(mrb, a);
727 return self;
729 const mrb_value *vals = mrb_get_argv(mrb);
730 mrb_int len = ARY_LEN(a);
731 if (alen > ARY_MAX_SIZE - len) {
732 ary_too_big(mrb);
734 if (ARY_SHARED_P(a)
735 && a->as.heap.aux.shared->refcnt == 1 /* shared only referenced from this array */
736 && a->as.heap.ptr - a->as.heap.aux.shared->ptr >= alen) /* there's room for unshifted item */ {
737 ary_modify_check(mrb, a);
738 a->as.heap.ptr -= alen;
739 ptr = a->as.heap.ptr;
741 else {
742 mrb_bool same = vals == ARY_PTR(a);
743 ary_modify(mrb, a);
744 if (ARY_CAPA(a) < len + alen)
745 ary_expand_capa(mrb, a, len + alen);
746 ptr = ARY_PTR(a);
747 value_move(ptr + alen, ptr, len);
748 if (same) vals = ptr;
750 array_copy(ptr, vals, alen);
751 ARY_SET_LEN(a, len+alen);
752 while (alen--) {
753 mrb_field_write_barrier_value(mrb, (struct RBasic*)a, vals[alen]);
756 return self;
759 MRB_API void
760 mrb_ary_set(mrb_state *mrb, mrb_value ary, mrb_int n, mrb_value val)
762 struct RArray *a = mrb_ary_ptr(ary);
763 mrb_int len = ARY_LEN(a);
765 ary_modify(mrb, a);
766 /* range check */
767 if (n < 0) {
768 n += len;
769 if (n < 0) {
770 mrb_raisef(mrb, E_INDEX_ERROR, "index %i out of array", n - len);
773 if (n >= ARY_MAX_SIZE) {
774 mrb_raise(mrb, E_INDEX_ERROR, "index too big");
776 if (len <= n) {
777 if (ARY_CAPA(a) <= n)
778 ary_expand_capa(mrb, a, n + 1);
779 ary_fill_with_nil(ARY_PTR(a) + len, n + 1 - len);
780 ARY_SET_LEN(a, n+1);
783 ARY_PTR(a)[n] = val;
784 mrb_field_write_barrier_value(mrb, (struct RBasic*)a, val);
787 static struct RArray*
788 ary_dup(mrb_state *mrb, struct RArray *a)
790 return ary_new_from_values(mrb, ARY_LEN(a), ARY_PTR(a));
793 MRB_API mrb_value
794 mrb_ary_splice(mrb_state *mrb, mrb_value ary, mrb_int head, mrb_int len, mrb_value rpl)
796 struct RArray *a = mrb_ary_ptr(ary);
797 mrb_int alen = ARY_LEN(a);
798 const mrb_value *argv;
799 mrb_int argc;
800 mrb_int tail;
802 ary_modify(mrb, a);
804 /* len check */
805 if (len < 0) mrb_raisef(mrb, E_INDEX_ERROR, "negative length (%i)", len);
807 /* range check */
808 if (head < 0) {
809 head += alen;
810 if (head < 0) goto out_of_range;
812 if (head > ARY_MAX_SIZE - len) {
813 out_of_range:
814 mrb_raisef(mrb, E_INDEX_ERROR, "index %i is out of array", head);
816 tail = head + len;
817 if (alen < len || alen < tail) {
818 len = alen - head;
819 tail = head + len;
822 /* size check */
823 if (mrb_array_p(rpl)) {
824 argc = RARRAY_LEN(rpl);
825 argv = RARRAY_PTR(rpl);
826 if (argv == ARY_PTR(a)) {
827 struct RArray *r;
829 if (argc > 32767) {
830 mrb_raise(mrb, E_ARGUMENT_ERROR, "too big recursive splice");
832 r = ary_dup(mrb, a);
833 argv = ARY_PTR(r);
836 else if (mrb_undef_p(rpl)) {
837 argc = 0;
838 argv = NULL;
840 else {
841 argc = 1;
842 argv = &rpl;
844 if (head >= alen) {
845 if (head > ARY_MAX_SIZE - argc) goto out_of_range;
846 len = head + argc;
847 if (len > ARY_CAPA(a)) {
848 ary_expand_capa(mrb, a, len);
850 ary_fill_with_nil(ARY_PTR(a) + alen, head - alen);
851 if (argc > 0) {
852 array_copy(ARY_PTR(a) + head, argv, argc);
854 ARY_SET_LEN(a, len);
856 else {
858 if (alen - len > ARY_MAX_SIZE - argc) {
859 head = alen + argc - len;
860 goto out_of_range;
862 mrb_int newlen = alen + argc - len;
863 if (newlen > ARY_CAPA(a)) {
864 ary_expand_capa(mrb, a, newlen);
867 if (len != argc) {
868 mrb_value *ptr = ARY_PTR(a);
869 value_move(ptr + head + argc, ptr + tail, alen - tail);
870 ARY_SET_LEN(a, newlen);
872 if (argc > 0) {
873 value_move(ARY_PTR(a) + head, argv, argc);
876 mrb_write_barrier(mrb, (struct RBasic*)a);
877 return ary;
880 void
881 mrb_ary_decref(mrb_state *mrb, mrb_shared_array *shared)
883 shared->refcnt--;
884 if (shared->refcnt == 0) {
885 mrb_free(mrb, shared->ptr);
886 mrb_free(mrb, shared);
890 static mrb_value
891 ary_subseq(mrb_state *mrb, struct RArray *a, mrb_int beg, mrb_int len)
893 struct RArray *b;
895 if (!ARY_SHARED_P(a) && len <= ARY_SHIFT_SHARED_MIN) {
896 return mrb_ary_new_from_values(mrb, len, ARY_PTR(a)+beg);
898 ary_make_shared(mrb, a);
899 b = MRB_OBJ_ALLOC(mrb, MRB_TT_ARRAY, mrb->array_class);
900 b->as.heap.ptr = a->as.heap.ptr + beg;
901 b->as.heap.len = len;
902 b->as.heap.aux.shared = a->as.heap.aux.shared;
903 b->as.heap.aux.shared->refcnt++;
904 ARY_SET_SHARED_FLAG(b);
906 return mrb_obj_value(b);
909 mrb_value
910 mrb_ary_subseq(mrb_state *mrb, mrb_value ary, mrb_int beg, mrb_int len)
912 struct RArray *a = mrb_ary_ptr(ary);
913 return ary_subseq(mrb, a, beg, len);
916 static mrb_int
917 aget_index(mrb_state *mrb, mrb_value index)
919 if (mrb_integer_p(index)) {
920 return mrb_integer(index);
922 #ifndef MRB_NO_FLOAT
923 else if (mrb_float_p(index)) {
924 return (mrb_int)mrb_float(index);
926 #endif
927 else {
928 mrb_int i, argc;
929 const mrb_value *argv;
931 mrb_get_args(mrb, "i*!", &i, &argv, &argc);
932 return i;
937 * call-seq:
938 * ary[index] -> obj or nil
939 * ary[start, length] -> new_ary or nil
940 * ary[range] -> new_ary or nil
941 * ary.slice(index) -> obj or nil
942 * ary.slice(start, length) -> new_ary or nil
943 * ary.slice(range) -> new_ary or nil
945 * Element Reference --- Returns the element at +index+, or returns a
946 * subarray starting at the +start+ index and continuing for +length+
947 * elements, or returns a subarray specified by +range+ of indices.
949 * Negative indices count backward from the end of the array (-1 is the last
950 * element). For +start+ and +range+ cases the starting index is just before
951 * an element. Additionally, an empty array is returned when the starting
952 * index for an element range is at the end of the array.
954 * Returns +nil+ if the index (or starting index) are out of range.
956 * a = [ "a", "b", "c", "d", "e" ]
957 * a[1] => "b"
958 * a[1,2] => ["b", "c"]
959 * a[1..-2] => ["b", "c", "d"]
963 static mrb_value
964 mrb_ary_aget(mrb_state *mrb, mrb_value self)
966 struct RArray *a = mrb_ary_ptr(self);
967 mrb_int i, len;
968 mrb_value index;
970 if (mrb_get_argc(mrb) == 1) {
971 index = mrb_get_arg1(mrb);
972 switch (mrb_type(index)) {
973 /* a[n..m] */
974 case MRB_TT_RANGE:
975 if (mrb_range_beg_len(mrb, index, &i, &len, ARY_LEN(a), TRUE) == MRB_RANGE_OK) {
976 return ary_subseq(mrb, a, i, len);
978 else {
979 return mrb_nil_value();
981 case MRB_TT_INTEGER:
982 return mrb_ary_ref(mrb, self, mrb_integer(index));
983 default:
984 return mrb_ary_ref(mrb, self, aget_index(mrb, index));
988 mrb_get_args(mrb, "oi", &index, &len);
989 i = aget_index(mrb, index);
990 mrb_int alen = ARY_LEN(a);
991 if (i < 0) i += alen;
992 if (i < 0 || alen < i) return mrb_nil_value();
993 if (len < 0) return mrb_nil_value();
994 if (alen == i) return mrb_ary_new(mrb);
995 if (len > alen - i) len = alen - i;
997 return ary_subseq(mrb, a, i, len);
1001 * call-seq:
1002 * ary[index] = obj -> obj
1003 * ary[start, length] = obj or other_ary or nil -> obj or other_ary or nil
1004 * ary[range] = obj or other_ary or nil -> obj or other_ary or nil
1006 * Element Assignment --- Sets the element at +index+, or replaces a subarray
1007 * from the +start+ index for +length+ elements, or replaces a subarray
1008 * specified by the +range+ of indices.
1010 * If indices are greater than the current capacity of the array, the array
1011 * grows automatically. Elements are inserted into the array at +start+ if
1012 * +length+ is zero.
1014 * Negative indices will count backward from the end of the array. For
1015 * +start+ and +range+ cases the starting index is just before an element.
1017 * An IndexError is raised if a negative index points past the beginning of
1018 * the array.
1020 * See also Array#push, and Array#unshift.
1022 * a = Array.new
1023 * a[4] = "4"; #=> [nil, nil, nil, nil, "4"]
1024 * a[0, 3] = [ 'a', 'b', 'c' ] #=> ["a", "b", "c", nil, "4"]
1025 * a[1..2] = [ 1, 2 ] #=> ["a", 1, 2, nil, "4"]
1026 * a[0, 2] = "?" #=> ["?", 2, nil, "4"]
1027 * a[0..2] = "A" #=> ["A", "4"]
1028 * a[-1] = "Z" #=> ["A", "Z"]
1029 * a[1..-1] = nil #=> ["A", nil]
1030 * a[1..-1] = [] #=> ["A"]
1031 * a[0, 0] = [ 1, 2 ] #=> [1, 2, "A"]
1032 * a[3, 0] = "B" #=> [1, 2, "A", "B"]
1035 static mrb_value
1036 mrb_ary_aset(mrb_state *mrb, mrb_value self)
1038 mrb_value v1, v2, v3;
1040 if (mrb_get_argc(mrb) == 2) {
1041 mrb_int i, len;
1042 const mrb_value *vs = mrb_get_argv(mrb);
1043 v1 = vs[0]; v2 = vs[1];
1045 /* a[n..m] = v */
1046 switch (mrb_range_beg_len(mrb, v1, &i, &len, RARRAY_LEN(self), FALSE)) {
1047 case MRB_RANGE_TYPE_MISMATCH:
1048 mrb_ary_set(mrb, self, aget_index(mrb, v1), v2);
1049 break;
1050 case MRB_RANGE_OK:
1051 mrb_ary_splice(mrb, self, i, len, v2);
1052 break;
1053 case MRB_RANGE_OUT:
1054 mrb_raisef(mrb, E_RANGE_ERROR, "%v out of range", v1);
1055 break;
1057 return v2;
1060 mrb_get_args(mrb, "ooo", &v1, &v2, &v3);
1061 /* a[n,m] = v */
1062 mrb_ary_splice(mrb, self, aget_index(mrb, v1), aget_index(mrb, v2), v3);
1063 return v3;
1066 mrb_value
1067 mrb_ary_delete_at(mrb_state *mrb, mrb_value self)
1069 struct RArray *a = mrb_ary_ptr(self);
1071 mrb_int index = mrb_as_int(mrb, mrb_get_arg1(mrb));
1072 mrb_int alen = ARY_LEN(a);
1073 if (index < 0) index += alen;
1074 if (index < 0 || alen <= index) return mrb_nil_value();
1076 ary_modify(mrb, a);
1077 mrb_value *ptr = ARY_PTR(a);
1078 mrb_value val = ptr[index];
1080 ptr += index;
1081 mrb_int len = alen - index;
1082 while (--len) {
1083 *ptr = *(ptr+1);
1084 ptr++;
1086 ARY_SET_LEN(a, alen-1);
1088 ary_shrink_capa(mrb, a);
1090 return val;
1093 static mrb_value
1094 mrb_ary_first(mrb_state *mrb, mrb_value self)
1096 struct RArray *a = mrb_ary_ptr(self);
1097 mrb_int size;
1099 if (mrb_get_argc(mrb) == 0) {
1100 if (ARY_LEN(a) > 0) return ARY_PTR(a)[0];
1101 return mrb_nil_value();
1103 mrb_get_args(mrb, "|i", &size);
1104 if (size < 0) {
1105 mrb_raise(mrb, E_ARGUMENT_ERROR, "negative array size");
1108 mrb_int alen = ARY_LEN(a);
1109 if (size > alen) size = alen;
1110 if (ARY_SHARED_P(a)) {
1111 return ary_subseq(mrb, a, 0, size);
1113 return mrb_ary_new_from_values(mrb, size, ARY_PTR(a));
1116 static mrb_value
1117 mrb_ary_last(mrb_state *mrb, mrb_value self)
1119 struct RArray *a = mrb_ary_ptr(self);
1120 mrb_int alen = ARY_LEN(a);
1122 if (mrb_get_argc(mrb) == 0) {
1123 if (alen > 0) return ARY_PTR(a)[alen - 1];
1124 return mrb_nil_value();
1127 mrb_int size = mrb_integer(mrb_get_arg1(mrb));
1128 if (size < 0) {
1129 mrb_raise(mrb, E_ARGUMENT_ERROR, "negative array size");
1131 if (size > alen) size = alen;
1132 if (ARY_SHARED_P(a) || size > ARY_DEFAULT_LEN) {
1133 return ary_subseq(mrb, a, alen - size, size);
1135 return mrb_ary_new_from_values(mrb, size, ARY_PTR(a) + alen - size);
1139 * call-seq:
1140 * ary.index(val) -> int or nil
1141 * ary.index {|item| block } -> int or nil
1142 * array.index -> enumerator
1144 * Returns the _index_ of the first object in +ary+ such that the object is
1145 * <code>==</code> to +obj+.
1147 * If a block is given instead of an argument, returns the _index_ of the
1148 * first object for which the block returns +true+. Returns +nil+ if no
1149 * match is found.
1151 * ISO 15.2.12.5.14
1153 static mrb_value
1154 mrb_ary_index_m(mrb_state *mrb, mrb_value self)
1156 mrb_value obj, blk;
1158 if (mrb_get_args(mrb, "|o&", &obj, &blk) == 0 && mrb_nil_p(blk)) {
1159 return mrb_funcall_id(mrb, self, MRB_SYM(to_enum), 1, mrb_symbol_value(MRB_SYM(index)));
1162 if (mrb_nil_p(blk)) {
1163 for (mrb_int i = 0; i < RARRAY_LEN(self); i++) {
1164 if (mrb_equal(mrb, RARRAY_PTR(self)[i], obj)) {
1165 return mrb_int_value(mrb, i);
1169 else {
1170 for (mrb_int i = 0; i < RARRAY_LEN(self); i++) {
1171 mrb_value eq = mrb_yield(mrb, blk, RARRAY_PTR(self)[i]);
1172 if (mrb_test(eq)) {
1173 return mrb_int_value(mrb, i);
1177 return mrb_nil_value();
1181 * call-seq:
1182 * ary.rindex(val) -> int or nil
1183 * ary.rindex {|item| block } -> int or nil
1184 * array.rindex -> enumerator
1186 * Returns the _index_ of the first object in +ary+ such that the object is
1187 * <code>==</code> to +obj+.
1189 * If a block is given instead of an argument, returns the _index_ of the
1190 * first object for which the block returns +true+. Returns +nil+ if no
1191 * match is found.
1193 * ISO 15.2.12.5.26
1195 static mrb_value
1196 mrb_ary_rindex_m(mrb_state *mrb, mrb_value self)
1198 mrb_value obj, blk;
1200 if (mrb_get_args(mrb, "|o&", &obj, &blk) == 0 && mrb_nil_p(blk)) {
1201 return mrb_funcall_id(mrb, self, MRB_SYM(to_enum), 1, mrb_symbol_value(MRB_SYM(rindex)));
1204 for (mrb_int i = RARRAY_LEN(self) - 1; i >= 0; i--) {
1205 if (mrb_nil_p(blk)) {
1206 if (mrb_equal(mrb, RARRAY_PTR(self)[i], obj)) {
1207 return mrb_int_value(mrb, i);
1210 else {
1211 mrb_value eq = mrb_yield(mrb, blk, RARRAY_PTR(self)[i]);
1212 if (mrb_test(eq)) return mrb_int_value(mrb, i);
1214 mrb_int len = RARRAY_LEN(self);
1215 if (i > len) {
1216 i = len;
1219 return mrb_nil_value();
1222 MRB_API mrb_value
1223 mrb_ary_splat(mrb_state *mrb, mrb_value v)
1225 struct RArray *a;
1227 if (mrb_array_p(v)) {
1228 a = ary_dup(mrb, mrb_ary_ptr(v));
1229 return mrb_obj_value(a);
1232 if (!mrb_respond_to(mrb, v, MRB_SYM(to_a))) {
1233 return mrb_ary_new_from_values(mrb, 1, &v);
1236 mrb_value ary = mrb_funcall_argv(mrb, v, MRB_SYM(to_a), 0, NULL);
1237 if (mrb_nil_p(ary)) {
1238 return mrb_ary_new_from_values(mrb, 1, &v);
1240 mrb_ensure_array_type(mrb, ary);
1241 a = mrb_ary_ptr(ary);
1242 a = ary_dup(mrb, a);
1243 return mrb_obj_value(a);
1246 static mrb_value
1247 mrb_ary_size(mrb_state *mrb, mrb_value self)
1249 struct RArray *a = mrb_ary_ptr(self);
1251 return mrb_int_value(mrb, ARY_LEN(a));
1254 MRB_API mrb_value
1255 mrb_ary_clear(mrb_state *mrb, mrb_value self)
1257 struct RArray *a = mrb_ary_ptr(self);
1259 ary_modify(mrb, a);
1260 if (ARY_SHARED_P(a)) {
1261 mrb_ary_decref(mrb, a->as.heap.aux.shared);
1262 ARY_UNSET_SHARED_FLAG(a);
1264 else if (!ARY_EMBED_P(a)){
1265 mrb_free(mrb, a->as.heap.ptr);
1267 if (MRB_ARY_EMBED_LEN_MAX > 0) {
1268 ARY_SET_EMBED_LEN(a, 0);
1270 else {
1271 a->as.heap.ptr = NULL;
1272 a->as.heap.aux.capa = 0;
1273 ARY_SET_LEN(a, 0);
1275 return self;
1278 static mrb_value
1279 mrb_ary_empty_p(mrb_state *mrb, mrb_value self)
1281 struct RArray *a = mrb_ary_ptr(self);
1283 return mrb_bool_value(ARY_LEN(a) == 0);
1286 MRB_API mrb_value
1287 mrb_ary_entry(mrb_value ary, mrb_int n)
1289 struct RArray *a = mrb_ary_ptr(ary);
1290 mrb_int len = ARY_LEN(a);
1292 /* range check */
1293 if (n < 0) n += len;
1294 if (n < 0 || len <= n) return mrb_nil_value();
1296 return ARY_PTR(a)[n];
1299 static mrb_value
1300 join_ary(mrb_state *mrb, mrb_value ary, mrb_value sep, mrb_value list)
1302 /* check recursive */
1303 for (mrb_int i=0; i<RARRAY_LEN(list); i++) {
1304 if (mrb_obj_equal(mrb, ary, RARRAY_PTR(list)[i])) {
1305 mrb_raise(mrb, E_ARGUMENT_ERROR, "recursive array join");
1309 mrb_ary_push(mrb, list, ary);
1311 mrb_value result = mrb_str_new_capa(mrb, 64);
1313 for (mrb_int i=0; i<RARRAY_LEN(ary); i++) {
1314 if (i > 0 && !mrb_nil_p(sep)) {
1315 mrb_str_cat_str(mrb, result, sep);
1318 mrb_value val = RARRAY_PTR(ary)[i];
1320 switch (mrb_type(val)) {
1321 case MRB_TT_ARRAY:
1322 ary_join:
1323 val = join_ary(mrb, val, sep, list);
1324 /* fall through */
1326 case MRB_TT_STRING:
1327 str_join:
1328 mrb_str_cat_str(mrb, result, val);
1329 break;
1331 default:
1332 if (!mrb_immediate_p(val)) {
1333 mrb_value tmp = mrb_check_string_type(mrb, val);
1334 if (!mrb_nil_p(tmp)) {
1335 val = tmp;
1336 goto str_join;
1338 tmp = mrb_check_array_type(mrb, val);
1339 if (!mrb_nil_p(tmp)) {
1340 val = tmp;
1341 goto ary_join;
1344 val = mrb_obj_as_string(mrb, val);
1345 goto str_join;
1349 mrb_ary_pop(mrb, list);
1351 return result;
1354 MRB_API mrb_value
1355 mrb_ary_join(mrb_state *mrb, mrb_value ary, mrb_value sep)
1357 if (!mrb_nil_p(sep)) {
1358 sep = mrb_obj_as_string(mrb, sep);
1360 return join_ary(mrb, ary, sep, mrb_ary_new(mrb));
1364 * call-seq:
1365 * ary.join(sep="") -> str
1367 * Returns a string created by converting each element of the array to
1368 * a string, separated by <i>sep</i>.
1370 * [ "a", "b", "c" ].join #=> "abc"
1371 * [ "a", "b", "c" ].join("-") #=> "a-b-c"
1374 static mrb_value
1375 mrb_ary_join_m(mrb_state *mrb, mrb_value ary)
1377 mrb_value sep = mrb_nil_value();
1379 mrb_get_args(mrb, "|S!", &sep);
1380 return mrb_ary_join(mrb, ary, sep);
1384 * call-seq:
1385 * ary.to_s -> string
1386 * ary.inspect -> string
1388 * Return the contents of this array as a string.
1390 static mrb_value
1391 mrb_ary_to_s(mrb_state *mrb, mrb_value self)
1393 mrb->c->ci->mid = MRB_SYM(inspect);
1394 mrb_value ret = mrb_str_new_lit(mrb, "[");
1395 int ai = mrb_gc_arena_save(mrb);
1396 if (mrb_inspect_recursive_p(mrb, self)) {
1397 mrb_str_cat_lit(mrb, ret, "...]");
1398 return ret;
1400 for (mrb_int i=0; i<RARRAY_LEN(self); i++) {
1401 if (i>0) mrb_str_cat_lit(mrb, ret, ", ");
1402 mrb_str_cat_str(mrb, ret, mrb_inspect(mrb, RARRAY_PTR(self)[i]));
1403 mrb_gc_arena_restore(mrb, ai);
1405 mrb_str_cat_lit(mrb, ret, "]");
1407 return ret;
1410 /* check array equality: 1=equal,0=not_equal,-1=need_elements_check */
1411 static mrb_int
1412 ary_eq(mrb_state *mrb, mrb_value ary1, mrb_value ary2)
1414 if (mrb_obj_equal(mrb, ary1, ary2)) return 1;
1415 if (!mrb_array_p(ary2)) return 0;
1416 if (RARRAY_LEN(ary1) != RARRAY_LEN(ary2)) return 0;
1418 return -1;
1422 * call-seq:
1423 * array == other -> true or false
1425 * Equality---Two arrays are equal if they contain the same number
1426 * of elements and if each element is equal to (according to
1427 * Object.==) the corresponding element in the other array.
1430 static mrb_value
1431 mrb_ary_eq(mrb_state *mrb, mrb_value ary1)
1433 mrb_value ary2 = mrb_get_arg1(mrb);
1434 mrb_int n = ary_eq(mrb, ary1, ary2);
1436 if (n == 1) return mrb_true_value();
1437 if (n == 0) return mrb_false_value();
1439 int ai = mrb_gc_arena_save(mrb);
1440 for (mrb_int i=0; i<RARRAY_LEN(ary1); i++) {
1441 mrb_value eq = mrb_funcall_id(mrb, mrb_ary_entry(ary1, i), MRB_OPSYM(eq), 1, mrb_ary_entry(ary2, i));
1442 if (!mrb_test(eq)) return mrb_false_value();
1443 mrb_gc_arena_restore(mrb, ai);
1445 return mrb_true_value();
1449 * call-seq:
1450 * array.eql? other_array -> true or false
1452 * Returns <code>true</code> if +self+ and _other_ are the same object,
1453 * or are both arrays with the same content.
1456 static mrb_value
1457 mrb_ary_eql(mrb_state *mrb, mrb_value ary1)
1459 mrb_value ary2 = mrb_get_arg1(mrb);
1460 mrb_int n = ary_eq(mrb, ary1, ary2);
1462 if (n == 1) return mrb_true_value();
1463 if (n == 0) return mrb_false_value();
1465 int ai = mrb_gc_arena_save(mrb);
1466 for (mrb_int i=0; i<RARRAY_LEN(ary1); i++) {
1467 mrb_value eq = mrb_funcall_id(mrb, mrb_ary_entry(ary1, i), MRB_SYM_Q(eql), 1, mrb_ary_entry(ary2, i));
1468 if (!mrb_test(eq)) return mrb_false_value();
1469 mrb_gc_arena_restore(mrb, ai);
1471 return mrb_true_value();
1475 * call-seq:
1476 * array <=> other_array -> -1, 0, or 1
1478 * Comparison---Returns an integer (-1, 0, or +1)
1479 * if this array is less than, equal to, or greater than <i>other_ary</i>.
1480 * Each object in each array is compared (using <=>). If any value isn't
1481 * equal, then that inequality is the return value. If all the
1482 * values found are equal, then the return is based on a
1483 * comparison of the array lengths. Thus, two arrays are
1484 * "equal" according to <code>Array*<=></code> if and only if they have
1485 * the same length and the value of each element is equal to the
1486 * value of the corresponding element in the other array.
1488 static mrb_value
1489 mrb_ary_cmp(mrb_state *mrb, mrb_value ary1)
1491 mrb_value ary2 = mrb_get_arg1(mrb);
1493 if (mrb_obj_equal(mrb, ary1, ary2)) return mrb_fixnum_value(0);
1494 if (!mrb_array_p(ary2)) return mrb_nil_value();
1496 for (mrb_int i=0; i<RARRAY_LEN(ary1) && i<RARRAY_LEN(ary2); i++) {
1497 mrb_int n = mrb_cmp(mrb, RARRAY_PTR(ary1)[i], RARRAY_PTR(ary2)[i]);
1498 if (n == -2) return mrb_nil_value();
1499 if (n != 0) return mrb_fixnum_value(n);
1501 mrb_int len = RARRAY_LEN(ary1) - RARRAY_LEN(ary2);
1502 if (len == 0) return mrb_fixnum_value(0);
1503 else if (len > 0) return mrb_fixnum_value(1);
1504 else return mrb_fixnum_value(-1);
1507 /* internal method to convert multi-value to single value */
1508 static mrb_value
1509 mrb_ary_svalue(mrb_state *mrb, mrb_value ary)
1511 switch (RARRAY_LEN(ary)) {
1512 case 0:
1513 return mrb_nil_value();
1514 case 1:
1515 return RARRAY_PTR(ary)[0];
1516 default:
1517 return ary;
1522 * call-seq:
1523 * array.delete(obj) -> deleted_object
1524 * array.delete(obj) {|nosuch| ... } -> deleted_object or block_return
1526 * Removes zero or more elements from self; returns self.
1528 * When no block is given, removes from self each element e such
1529 * that e == obj; returns the last deleted element
1531 * Returns nil if no elements removed.
1533 * When a block is given, removes from self each element e such
1534 * that e == obj. If any such elements are found, ignores the block and
1535 * returns the last. Otherwise, returns the block's return value.
1537 static mrb_value
1538 mrb_ary_delete(mrb_state *mrb, mrb_value self)
1540 mrb_value obj, blk;
1542 mrb_get_args(mrb, "o&", &obj, &blk);
1544 struct RArray *ary = RARRAY(self);
1545 mrb_value ret = obj;
1546 int ai = mrb_gc_arena_save(mrb);
1547 mrb_int i = 0;
1548 mrb_int j = 0;
1549 for (; i < ARY_LEN(ary); i++) {
1550 mrb_value elem = ARY_PTR(ary)[i];
1552 if (mrb_equal(mrb, elem, obj)) {
1553 mrb_gc_arena_restore(mrb, ai);
1554 mrb_gc_protect(mrb, elem);
1555 ret = elem;
1556 continue;
1559 if (i != j) {
1560 if (j >= ARY_LEN(ary)) {
1561 // Since breaking here will further change the array length,
1562 // there is no choice but to raise an exception or return.
1563 mrb_raise(mrb, E_RUNTIME_ERROR, "array modified during delete");
1565 ary_modify(mrb, ary);
1566 ARY_PTR(ary)[j] = elem;
1569 j++;
1572 if (i == j) {
1573 if (mrb_nil_p(blk)) return mrb_nil_value();
1574 return mrb_yield(mrb, blk, obj);
1577 ARY_SET_LEN(ary, j);
1578 return ret;
1581 static mrb_noreturn void
1582 cmp_failed(mrb_state *mrb, mrb_int a, mrb_int b)
1584 mrb_raisef(mrb, E_ARGUMENT_ERROR, "comparison failed (element %d and %d)", a, b);
1587 static mrb_bool
1588 sort_cmp(mrb_state *mrb, mrb_value ary, mrb_value *p, mrb_int a, mrb_int b, mrb_value blk)
1590 mrb_int cmp;
1592 if (mrb_nil_p(blk)) {
1593 cmp = mrb_cmp(mrb, p[a], p[b]);
1594 if (cmp == -2) cmp_failed(mrb, a, b);
1596 else {
1597 mrb_value args[2] = {p[a], p[b]};
1598 mrb_value c = mrb_yield_argv(mrb, blk, 2, args);
1599 if (mrb_nil_p(c) || !mrb_fixnum_p(c)) {
1600 cmp_failed(mrb, a, b);
1602 cmp = mrb_fixnum(c);
1604 mrb_int size = RARRAY_LEN(ary);
1605 if (RARRAY_PTR(ary) != p || size < a || size < b) {
1606 mrb_raise(mrb, E_RUNTIME_ERROR, "array modified during sort");
1608 return cmp > 0;
1611 static void
1612 heapify(mrb_state *mrb, mrb_value ary, mrb_value *a, mrb_int index, mrb_int size, mrb_value blk)
1614 mrb_int max = index;
1615 mrb_int left_index = 2 * index + 1;
1616 mrb_int right_index = left_index + 1;
1617 if (left_index < size && sort_cmp(mrb, ary, a, left_index, max, blk)) {
1618 max = left_index;
1620 if (right_index < size && sort_cmp(mrb, ary, a, right_index, max, blk)) {
1621 max = right_index;
1623 if (max != index) {
1624 mrb_value tmp = a[max];
1625 a[max] = a[index];
1626 a[index] = tmp;
1627 heapify(mrb, ary, a, max, size, blk);
1632 * call-seq:
1633 * array.sort! -> self
1634 * array.sort! {|a, b| ... } -> self
1636 * Sort all elements and replace +self+ with these
1637 * elements.
1639 static mrb_value
1640 mrb_ary_sort_bang(mrb_state *mrb, mrb_value ary)
1642 mrb_value blk;
1644 mrb_int n = RARRAY_LEN(ary);
1645 if (n < 2) return ary;
1647 ary_modify(mrb, mrb_ary_ptr(ary));
1648 mrb_get_args(mrb, "&", &blk);
1650 mrb_value *a = RARRAY_PTR(ary);
1651 for (mrb_int i = n / 2 - 1; i > -1; i--) {
1652 heapify(mrb, ary, a, i, n, blk);
1654 for (mrb_int i = n - 1; i > 0; i--) {
1655 mrb_value tmp = a[0];
1656 a[0] = a[i];
1657 a[i] = tmp;
1658 heapify(mrb, ary, a, 0, i, blk);
1660 return ary;
1663 void
1664 mrb_init_array(mrb_state *mrb)
1666 struct RClass *a;
1668 mrb->array_class = a = mrb_define_class_id(mrb, MRB_SYM(Array), mrb->object_class); /* 15.2.12 */
1669 MRB_SET_INSTANCE_TT(a, MRB_TT_ARRAY);
1671 mrb_define_class_method_id(mrb, a, MRB_OPSYM(aref), mrb_ary_s_create, MRB_ARGS_ANY()); /* 15.2.12.4.1 */
1673 mrb_define_method_id(mrb, a, MRB_OPSYM(add), mrb_ary_plus, MRB_ARGS_REQ(1)); /* 15.2.12.5.1 */
1674 mrb_define_method_id(mrb, a, MRB_OPSYM(mul), mrb_ary_times, MRB_ARGS_REQ(1)); /* 15.2.12.5.2 */
1675 mrb_define_method_id(mrb, a, MRB_OPSYM(lshift), mrb_ary_push_m, MRB_ARGS_REQ(1)); /* 15.2.12.5.3 */
1676 mrb_define_method_id(mrb, a, MRB_OPSYM(aref), mrb_ary_aget, MRB_ARGS_ARG(1,1)); /* 15.2.12.5.4 */
1677 mrb_define_method_id(mrb, a, MRB_OPSYM(aset), mrb_ary_aset, MRB_ARGS_ARG(2,1)); /* 15.2.12.5.5 */
1678 mrb_define_method_id(mrb, a, MRB_SYM(clear), mrb_ary_clear, MRB_ARGS_NONE()); /* 15.2.12.5.6 */
1679 mrb_define_method_id(mrb, a, MRB_OPSYM(cmp), mrb_ary_cmp, MRB_ARGS_REQ(1));
1680 mrb_define_method_id(mrb, a, MRB_SYM(concat), mrb_ary_concat_m, MRB_ARGS_REQ(1)); /* 15.2.12.5.8 */
1681 mrb_define_method_id(mrb, a, MRB_SYM(delete), mrb_ary_delete, MRB_ARGS_REQ(1));
1682 mrb_define_method_id(mrb, a, MRB_SYM(delete_at), mrb_ary_delete_at, MRB_ARGS_REQ(1)); /* 15.2.12.5.9 */
1683 mrb_define_method_id(mrb, a, MRB_SYM_Q(empty), mrb_ary_empty_p, MRB_ARGS_NONE()); /* 15.2.12.5.12 */
1684 mrb_define_method_id(mrb, a, MRB_OPSYM(eq), mrb_ary_eq, MRB_ARGS_REQ(1));
1685 mrb_define_method_id(mrb, a, MRB_SYM_Q(eql), mrb_ary_eql, MRB_ARGS_REQ(1));
1686 mrb_define_method_id(mrb, a, MRB_SYM(first), mrb_ary_first, MRB_ARGS_OPT(1)); /* 15.2.12.5.13 */
1687 mrb_define_method_id(mrb, a, MRB_SYM(index), mrb_ary_index_m, MRB_ARGS_REQ(1)); /* 15.2.12.5.14 */
1688 mrb_define_method_id(mrb, a, MRB_SYM(initialize), mrb_ary_init, MRB_ARGS_OPT(2)); /* 15.2.12.5.15 */
1689 mrb_define_method_id(mrb, a, MRB_SYM(initialize_copy), mrb_ary_replace_m, MRB_ARGS_REQ(1)); /* 15.2.12.5.16 */
1690 mrb_define_method_id(mrb, a, MRB_SYM(join), mrb_ary_join_m, MRB_ARGS_OPT(1)); /* 15.2.12.5.17 */
1691 mrb_define_method_id(mrb, a, MRB_SYM(last), mrb_ary_last, MRB_ARGS_OPT(1)); /* 15.2.12.5.18 */
1692 mrb_define_method_id(mrb, a, MRB_SYM(length), mrb_ary_size, MRB_ARGS_NONE()); /* 15.2.12.5.19 */
1693 mrb_define_method_id(mrb, a, MRB_SYM(pop), mrb_ary_pop, MRB_ARGS_NONE()); /* 15.2.12.5.21 */
1694 mrb_define_method_id(mrb, a, MRB_SYM(push), mrb_ary_push_m, MRB_ARGS_ANY()); /* 15.2.12.5.22 */
1695 mrb_define_method_id(mrb, a, MRB_SYM(replace), mrb_ary_replace_m, MRB_ARGS_REQ(1)); /* 15.2.12.5.23 */
1696 mrb_define_method_id(mrb, a, MRB_SYM(reverse), mrb_ary_reverse, MRB_ARGS_NONE()); /* 15.2.12.5.24 */
1697 mrb_define_method_id(mrb, a, MRB_SYM_B(reverse), mrb_ary_reverse_bang, MRB_ARGS_NONE()); /* 15.2.12.5.25 */
1698 mrb_define_method_id(mrb, a, MRB_SYM(rindex), mrb_ary_rindex_m, MRB_ARGS_REQ(1)); /* 15.2.12.5.26 */
1699 mrb_define_method_id(mrb, a, MRB_SYM(shift), mrb_ary_shift_m, MRB_ARGS_OPT(1)); /* 15.2.12.5.27 */
1700 mrb_define_method_id(mrb, a, MRB_SYM(size), mrb_ary_size, MRB_ARGS_NONE()); /* 15.2.12.5.28 */
1701 mrb_define_method_id(mrb, a, MRB_SYM(slice), mrb_ary_aget, MRB_ARGS_ARG(1,1)); /* 15.2.12.5.29 */
1702 mrb_define_method_id(mrb, a, MRB_SYM(unshift), mrb_ary_unshift_m, MRB_ARGS_ANY()); /* 15.2.12.5.30 */
1703 mrb_define_method_id(mrb, a, MRB_SYM(to_s), mrb_ary_to_s, MRB_ARGS_NONE());
1704 mrb_define_method_id(mrb, a, MRB_SYM(inspect), mrb_ary_to_s, MRB_ARGS_NONE());
1705 mrb_define_method_id(mrb, a, MRB_SYM_B(sort), mrb_ary_sort_bang, MRB_ARGS_NONE());
1707 mrb_define_method_id(mrb, a, MRB_SYM(__svalue), mrb_ary_svalue, MRB_ARGS_NONE());