Ruby 3.5.0dev (2025-04-03 revision 1dddc6c78b5f6dc6ae18ee04ebe44abfce3b0433)
array.c (1dddc6c78b5f6dc6ae18ee04ebe44abfce3b0433)
1/**********************************************************************
2
3 array.c -
4
5 $Author$
6 created at: Fri Aug 6 09:46:12 JST 1993
7
8 Copyright (C) 1993-2007 Yukihiro Matsumoto
9 Copyright (C) 2000 Network Applied Communication Laboratory, Inc.
10 Copyright (C) 2000 Information-technology Promotion Agency, Japan
11
12**********************************************************************/
13
14#include "debug_counter.h"
15#include "id.h"
16#include "internal.h"
17#include "internal/array.h"
18#include "internal/compar.h"
19#include "internal/enum.h"
20#include "internal/gc.h"
21#include "internal/hash.h"
22#include "internal/numeric.h"
23#include "internal/object.h"
24#include "internal/proc.h"
25#include "internal/rational.h"
26#include "internal/vm.h"
27#include "probes.h"
28#include "ruby/encoding.h"
29#include "ruby/st.h"
30#include "ruby/util.h"
31#include "vm_core.h"
32#include "builtin.h"
33
34#if !ARRAY_DEBUG
35# undef NDEBUG
36# define NDEBUG
37#endif
38#include "ruby_assert.h"
39
41VALUE rb_cArray_empty_frozen;
42
43/* Flags of RArray
44 *
45 * 0: RARRAY_SHARED_FLAG (equal to ELTS_SHARED)
46 * The array is shared. The buffer this array points to is owned by
47 * another array (the shared root).
48 * 1: RARRAY_EMBED_FLAG
49 * The array is embedded (its contents follow the header, rather than
50 * being on a separately allocated buffer).
51 * 3-9: RARRAY_EMBED_LEN
52 * The length of the array when RARRAY_EMBED_FLAG is set.
53 * 12: RARRAY_SHARED_ROOT_FLAG
54 * The array is a shared root that does reference counting. The buffer
55 * this array points to is owned by this array but may be pointed to
56 * by other arrays.
57 * Note: Frozen arrays may be a shared root without this flag being
58 * set. Frozen arrays do not have reference counting because
59 * they cannot be modified. Not updating the reference count
60 * improves copy-on-write performance. Their reference count is
61 * assumed to be infinity.
62 * 14: RARRAY_PTR_IN_USE_FLAG
63 * The buffer of the array is in use. This is only used during
64 * debugging.
65 */
66
67/* for OPTIMIZED_CMP: */
68#define id_cmp idCmp
69
70#define ARY_DEFAULT_SIZE 16
71#define ARY_MAX_SIZE (LONG_MAX / (int)sizeof(VALUE))
72#define SMALL_ARRAY_LEN 16
73
75static int
76should_be_T_ARRAY(VALUE ary)
77{
78 return RB_TYPE_P(ary, T_ARRAY);
79}
80
81#define ARY_HEAP_PTR(a) (RUBY_ASSERT(!ARY_EMBED_P(a)), RARRAY(a)->as.heap.ptr)
82#define ARY_HEAP_LEN(a) (RUBY_ASSERT(!ARY_EMBED_P(a)), RARRAY(a)->as.heap.len)
83#define ARY_HEAP_CAPA(a) (RUBY_ASSERT(!ARY_EMBED_P(a)), RUBY_ASSERT(!ARY_SHARED_ROOT_P(a)), \
84 RARRAY(a)->as.heap.aux.capa)
85
86#define ARY_EMBED_PTR(a) (RUBY_ASSERT(ARY_EMBED_P(a)), RARRAY(a)->as.ary)
87#define ARY_EMBED_LEN(a) \
88 (RUBY_ASSERT(ARY_EMBED_P(a)), \
89 (long)((RBASIC(a)->flags >> RARRAY_EMBED_LEN_SHIFT) & \
90 (RARRAY_EMBED_LEN_MASK >> RARRAY_EMBED_LEN_SHIFT)))
91#define ARY_HEAP_SIZE(a) (RUBY_ASSERT(!ARY_EMBED_P(a)), RUBY_ASSERT(ARY_OWNS_HEAP_P(a)), ARY_CAPA(a) * sizeof(VALUE))
92
93#define ARY_OWNS_HEAP_P(a) (RUBY_ASSERT(should_be_T_ARRAY((VALUE)(a))), \
94 !FL_TEST_RAW((a), RARRAY_SHARED_FLAG|RARRAY_EMBED_FLAG))
95
96#define FL_SET_EMBED(a) do { \
97 RUBY_ASSERT(!ARY_SHARED_P(a)); \
98 FL_SET((a), RARRAY_EMBED_FLAG); \
99 ary_verify(a); \
100} while (0)
101
102#define FL_UNSET_EMBED(ary) FL_UNSET((ary), RARRAY_EMBED_FLAG|RARRAY_EMBED_LEN_MASK)
103#define FL_SET_SHARED(ary) do { \
104 RUBY_ASSERT(!ARY_EMBED_P(ary)); \
105 FL_SET((ary), RARRAY_SHARED_FLAG); \
106} while (0)
107#define FL_UNSET_SHARED(ary) FL_UNSET((ary), RARRAY_SHARED_FLAG)
108
109#define ARY_SET_PTR(ary, p) do { \
110 RUBY_ASSERT(!ARY_EMBED_P(ary)); \
111 RUBY_ASSERT(!OBJ_FROZEN(ary)); \
112 RARRAY(ary)->as.heap.ptr = (p); \
113} while (0)
114#define ARY_SET_EMBED_LEN(ary, n) do { \
115 long tmp_n = (n); \
116 RUBY_ASSERT(ARY_EMBED_P(ary)); \
117 RBASIC(ary)->flags &= ~RARRAY_EMBED_LEN_MASK; \
118 RBASIC(ary)->flags |= (tmp_n) << RARRAY_EMBED_LEN_SHIFT; \
119} while (0)
120#define ARY_SET_HEAP_LEN(ary, n) do { \
121 RUBY_ASSERT(!ARY_EMBED_P(ary)); \
122 RARRAY(ary)->as.heap.len = (n); \
123} while (0)
124#define ARY_SET_LEN(ary, n) do { \
125 if (ARY_EMBED_P(ary)) { \
126 ARY_SET_EMBED_LEN((ary), (n)); \
127 } \
128 else { \
129 ARY_SET_HEAP_LEN((ary), (n)); \
130 } \
131 RUBY_ASSERT(RARRAY_LEN(ary) == (n)); \
132} while (0)
133#define ARY_INCREASE_PTR(ary, n) do { \
134 RUBY_ASSERT(!ARY_EMBED_P(ary)); \
135 RUBY_ASSERT(!OBJ_FROZEN(ary)); \
136 RARRAY(ary)->as.heap.ptr += (n); \
137} while (0)
138#define ARY_INCREASE_LEN(ary, n) do { \
139 RUBY_ASSERT(!OBJ_FROZEN(ary)); \
140 if (ARY_EMBED_P(ary)) { \
141 ARY_SET_EMBED_LEN((ary), RARRAY_LEN(ary)+(n)); \
142 } \
143 else { \
144 RARRAY(ary)->as.heap.len += (n); \
145 } \
146} while (0)
147
148#define ARY_CAPA(ary) (ARY_EMBED_P(ary) ? ary_embed_capa(ary) : \
149 ARY_SHARED_ROOT_P(ary) ? RARRAY_LEN(ary) : ARY_HEAP_CAPA(ary))
150#define ARY_SET_CAPA(ary, n) do { \
151 RUBY_ASSERT(!ARY_EMBED_P(ary)); \
152 RUBY_ASSERT(!ARY_SHARED_P(ary)); \
153 RUBY_ASSERT(!OBJ_FROZEN(ary)); \
154 RARRAY(ary)->as.heap.aux.capa = (n); \
155} while (0)
156
157#define ARY_SHARED_ROOT_OCCUPIED(ary) (!OBJ_FROZEN(ary) && ARY_SHARED_ROOT_REFCNT(ary) == 1)
158#define ARY_SET_SHARED_ROOT_REFCNT(ary, value) do { \
159 RUBY_ASSERT(ARY_SHARED_ROOT_P(ary)); \
160 RUBY_ASSERT(!OBJ_FROZEN(ary)); \
161 RUBY_ASSERT((value) >= 0); \
162 RARRAY(ary)->as.heap.aux.capa = (value); \
163} while (0)
164#define FL_SET_SHARED_ROOT(ary) do { \
165 RUBY_ASSERT(!OBJ_FROZEN(ary)); \
166 RUBY_ASSERT(!ARY_EMBED_P(ary)); \
167 FL_SET((ary), RARRAY_SHARED_ROOT_FLAG); \
168} while (0)
169
170static inline void
171ARY_SET(VALUE a, long i, VALUE v)
172{
173 RUBY_ASSERT(!ARY_SHARED_P(a));
175
176 RARRAY_ASET(a, i, v);
177}
178#undef RARRAY_ASET
179
180static long
181ary_embed_capa(VALUE ary)
182{
183 size_t size = rb_gc_obj_slot_size(ary) - offsetof(struct RArray, as.ary);
184 RUBY_ASSERT(size % sizeof(VALUE) == 0);
185 return size / sizeof(VALUE);
186}
187
188static size_t
189ary_embed_size(long capa)
190{
191 return offsetof(struct RArray, as.ary) + (sizeof(VALUE) * capa);
192}
193
194static bool
195ary_embeddable_p(long capa)
196{
197 return rb_gc_size_allocatable_p(ary_embed_size(capa));
198}
199
200bool
201rb_ary_embeddable_p(VALUE ary)
202{
203 /* An array cannot be turned embeddable when the array is:
204 * - Shared root: other objects may point to the buffer of this array
205 * so we cannot make it embedded.
206 * - Frozen: this array may also be a shared root without the shared root
207 * flag.
208 * - Shared: we don't want to re-embed an array that points to a shared
209 * root (to save memory).
210 */
211 return !(ARY_SHARED_ROOT_P(ary) || OBJ_FROZEN(ary) || ARY_SHARED_P(ary));
212}
213
214size_t
215rb_ary_size_as_embedded(VALUE ary)
216{
217 size_t real_size;
218
219 if (ARY_EMBED_P(ary)) {
220 real_size = ary_embed_size(ARY_EMBED_LEN(ary));
221 }
222 else if (rb_ary_embeddable_p(ary)) {
223 real_size = ary_embed_size(ARY_HEAP_CAPA(ary));
224 }
225 else {
226 real_size = sizeof(struct RArray);
227 }
228 return real_size;
229}
230
231
232#if ARRAY_DEBUG
233#define ary_verify(ary) ary_verify_(ary, __FILE__, __LINE__)
234
235static VALUE
236ary_verify_(VALUE ary, const char *file, int line)
237{
239
240 if (ARY_SHARED_P(ary)) {
241 VALUE root = ARY_SHARED_ROOT(ary);
242 const VALUE *ptr = ARY_HEAP_PTR(ary);
243 const VALUE *root_ptr = RARRAY_CONST_PTR(root);
244 long len = ARY_HEAP_LEN(ary), root_len = RARRAY_LEN(root);
245 RUBY_ASSERT(ARY_SHARED_ROOT_P(root) || OBJ_FROZEN(root));
246 RUBY_ASSERT(root_ptr <= ptr && ptr + len <= root_ptr + root_len);
247 ary_verify(root);
248 }
249 else if (ARY_EMBED_P(ary)) {
250 RUBY_ASSERT(!ARY_SHARED_P(ary));
251 RUBY_ASSERT(RARRAY_LEN(ary) <= ary_embed_capa(ary));
252 }
253 else {
254 const VALUE *ptr = RARRAY_CONST_PTR(ary);
255 long i, len = RARRAY_LEN(ary);
256 volatile VALUE v;
257 if (len > 1) len = 1; /* check only HEAD */
258 for (i=0; i<len; i++) {
259 v = ptr[i]; /* access check */
260 }
261 v = v;
262 }
263
264 return ary;
265}
266#else
267#define ary_verify(ary) ((void)0)
268#endif
269
270VALUE *
271rb_ary_ptr_use_start(VALUE ary)
272{
273#if ARRAY_DEBUG
274 FL_SET_RAW(ary, RARRAY_PTR_IN_USE_FLAG);
275#endif
276 return (VALUE *)RARRAY_CONST_PTR(ary);
277}
278
279void
280rb_ary_ptr_use_end(VALUE ary)
281{
282#if ARRAY_DEBUG
283 FL_UNSET_RAW(ary, RARRAY_PTR_IN_USE_FLAG);
284#endif
285}
286
287void
288rb_mem_clear(VALUE *mem, long size)
289{
290 while (size--) {
291 *mem++ = Qnil;
292 }
293}
294
295static void
296ary_mem_clear(VALUE ary, long beg, long size)
297{
299 rb_mem_clear(ptr + beg, size);
300 });
301}
302
303static inline void
304memfill(register VALUE *mem, register long size, register VALUE val)
305{
306 while (size--) {
307 *mem++ = val;
308 }
309}
310
311static void
312ary_memfill(VALUE ary, long beg, long size, VALUE val)
313{
315 memfill(ptr + beg, size, val);
317 });
318}
319
320static void
321ary_memcpy0(VALUE ary, long beg, long argc, const VALUE *argv, VALUE buff_owner_ary)
322{
323 RUBY_ASSERT(!ARY_SHARED_P(buff_owner_ary));
324
325 if (argc > (int)(128/sizeof(VALUE)) /* is magic number (cache line size) */) {
326 rb_gc_writebarrier_remember(buff_owner_ary);
328 MEMCPY(ptr+beg, argv, VALUE, argc);
329 });
330 }
331 else {
332 int i;
334 for (i=0; i<argc; i++) {
335 RB_OBJ_WRITE(buff_owner_ary, &ptr[i+beg], argv[i]);
336 }
337 });
338 }
339}
340
341static void
342ary_memcpy(VALUE ary, long beg, long argc, const VALUE *argv)
343{
344 ary_memcpy0(ary, beg, argc, argv, ary);
345}
346
347static VALUE *
348ary_heap_alloc_buffer(size_t capa)
349{
350 return ALLOC_N(VALUE, capa);
351}
352
353static void
354ary_heap_free_ptr(VALUE ary, const VALUE *ptr, long size)
355{
356 ruby_sized_xfree((void *)ptr, size);
357}
358
359static void
360ary_heap_free(VALUE ary)
361{
362 ary_heap_free_ptr(ary, ARY_HEAP_PTR(ary), ARY_HEAP_SIZE(ary));
363}
364
365static size_t
366ary_heap_realloc(VALUE ary, size_t new_capa)
367{
369 SIZED_REALLOC_N(RARRAY(ary)->as.heap.ptr, VALUE, new_capa, ARY_HEAP_CAPA(ary));
370 ary_verify(ary);
371
372 return new_capa;
373}
374
375void
376rb_ary_make_embedded(VALUE ary)
377{
378 RUBY_ASSERT(rb_ary_embeddable_p(ary));
379 if (!ARY_EMBED_P(ary)) {
380 const VALUE *buf = ARY_HEAP_PTR(ary);
381 long len = ARY_HEAP_LEN(ary);
382
383 FL_SET_EMBED(ary);
384 ARY_SET_EMBED_LEN(ary, len);
385
386 MEMCPY((void *)ARY_EMBED_PTR(ary), (void *)buf, VALUE, len);
387
388 ary_heap_free_ptr(ary, buf, len * sizeof(VALUE));
389 }
390}
391
392static void
393ary_resize_capa(VALUE ary, long capacity)
394{
395 RUBY_ASSERT(RARRAY_LEN(ary) <= capacity);
397 RUBY_ASSERT(!ARY_SHARED_P(ary));
398
399 if (capacity > ary_embed_capa(ary)) {
400 size_t new_capa = capacity;
401 if (ARY_EMBED_P(ary)) {
402 long len = ARY_EMBED_LEN(ary);
403 VALUE *ptr = ary_heap_alloc_buffer(capacity);
404
405 MEMCPY(ptr, ARY_EMBED_PTR(ary), VALUE, len);
406 FL_UNSET_EMBED(ary);
407 ARY_SET_PTR(ary, ptr);
408 ARY_SET_HEAP_LEN(ary, len);
409 }
410 else {
411 new_capa = ary_heap_realloc(ary, capacity);
412 }
413 ARY_SET_CAPA(ary, new_capa);
414 }
415 else {
416 if (!ARY_EMBED_P(ary)) {
417 long len = ARY_HEAP_LEN(ary);
418 long old_capa = ARY_HEAP_CAPA(ary);
419 const VALUE *ptr = ARY_HEAP_PTR(ary);
420
421 if (len > capacity) len = capacity;
422 MEMCPY((VALUE *)RARRAY(ary)->as.ary, ptr, VALUE, len);
423 ary_heap_free_ptr(ary, ptr, old_capa);
424
425 FL_SET_EMBED(ary);
426 ARY_SET_LEN(ary, len);
427 }
428 }
429
430 ary_verify(ary);
431}
432
433static inline void
434ary_shrink_capa(VALUE ary)
435{
436 long capacity = ARY_HEAP_LEN(ary);
437 long old_capa = ARY_HEAP_CAPA(ary);
438 RUBY_ASSERT(!ARY_SHARED_P(ary));
439 RUBY_ASSERT(old_capa >= capacity);
440 if (old_capa > capacity) {
441 size_t new_capa = ary_heap_realloc(ary, capacity);
442 ARY_SET_CAPA(ary, new_capa);
443 }
444
445 ary_verify(ary);
446}
447
448static void
449ary_double_capa(VALUE ary, long min)
450{
451 long new_capa = ARY_CAPA(ary) / 2;
452
453 if (new_capa < ARY_DEFAULT_SIZE) {
454 new_capa = ARY_DEFAULT_SIZE;
455 }
456 if (new_capa >= ARY_MAX_SIZE - min) {
457 new_capa = (ARY_MAX_SIZE - min) / 2;
458 }
459 new_capa += min;
460 ary_resize_capa(ary, new_capa);
461
462 ary_verify(ary);
463}
464
465static void
466rb_ary_decrement_share(VALUE shared_root)
467{
468 if (!OBJ_FROZEN(shared_root)) {
469 long num = ARY_SHARED_ROOT_REFCNT(shared_root);
470 ARY_SET_SHARED_ROOT_REFCNT(shared_root, num - 1);
471 }
472}
473
474static void
475rb_ary_unshare(VALUE ary)
476{
477 VALUE shared_root = ARY_SHARED_ROOT(ary);
478 rb_ary_decrement_share(shared_root);
479 FL_UNSET_SHARED(ary);
480}
481
482static void
483rb_ary_reset(VALUE ary)
484{
485 if (ARY_OWNS_HEAP_P(ary)) {
486 ary_heap_free(ary);
487 }
488 else if (ARY_SHARED_P(ary)) {
489 rb_ary_unshare(ary);
490 }
491
492 FL_SET_EMBED(ary);
493 ARY_SET_EMBED_LEN(ary, 0);
494}
495
496static VALUE
497rb_ary_increment_share(VALUE shared_root)
498{
499 if (!OBJ_FROZEN(shared_root)) {
500 long num = ARY_SHARED_ROOT_REFCNT(shared_root);
501 RUBY_ASSERT(num >= 0);
502 ARY_SET_SHARED_ROOT_REFCNT(shared_root, num + 1);
503 }
504 return shared_root;
505}
506
507static void
508rb_ary_set_shared(VALUE ary, VALUE shared_root)
509{
510 RUBY_ASSERT(!ARY_EMBED_P(ary));
512 RUBY_ASSERT(ARY_SHARED_ROOT_P(shared_root) || OBJ_FROZEN(shared_root));
513
514 rb_ary_increment_share(shared_root);
515 FL_SET_SHARED(ary);
516 RB_OBJ_WRITE(ary, &RARRAY(ary)->as.heap.aux.shared_root, shared_root);
517
518 RB_DEBUG_COUNTER_INC(obj_ary_shared_create);
519}
520
521static inline void
522rb_ary_modify_check(VALUE ary)
523{
524 rb_check_frozen(ary);
525 ary_verify(ary);
526}
527
528void
529rb_ary_cancel_sharing(VALUE ary)
530{
531 if (ARY_SHARED_P(ary)) {
532 long shared_len, len = RARRAY_LEN(ary);
533 VALUE shared_root = ARY_SHARED_ROOT(ary);
534
535 ary_verify(shared_root);
536
537 if (len <= ary_embed_capa(ary)) {
538 const VALUE *ptr = ARY_HEAP_PTR(ary);
539 FL_UNSET_SHARED(ary);
540 FL_SET_EMBED(ary);
541 MEMCPY((VALUE *)ARY_EMBED_PTR(ary), ptr, VALUE, len);
542 rb_ary_decrement_share(shared_root);
543 ARY_SET_EMBED_LEN(ary, len);
544 }
545 else if (ARY_SHARED_ROOT_OCCUPIED(shared_root) && len > ((shared_len = RARRAY_LEN(shared_root))>>1)) {
547 FL_UNSET_SHARED(ary);
548 ARY_SET_PTR(ary, RARRAY_CONST_PTR(shared_root));
549 ARY_SET_CAPA(ary, shared_len);
551 MEMMOVE(ptr, ptr+shift, VALUE, len);
552 });
553 FL_SET_EMBED(shared_root);
554 rb_ary_decrement_share(shared_root);
555 }
556 else {
557 VALUE *ptr = ary_heap_alloc_buffer(len);
558 MEMCPY(ptr, ARY_HEAP_PTR(ary), VALUE, len);
559 rb_ary_unshare(ary);
560 ARY_SET_CAPA(ary, len);
561 ARY_SET_PTR(ary, ptr);
562 }
563
564 rb_gc_writebarrier_remember(ary);
565 }
566 ary_verify(ary);
567}
568
569void
570rb_ary_modify(VALUE ary)
571{
572 rb_ary_modify_check(ary);
573 rb_ary_cancel_sharing(ary);
574}
575
576static VALUE
577ary_ensure_room_for_push(VALUE ary, long add_len)
578{
579 long old_len = RARRAY_LEN(ary);
580 long new_len = old_len + add_len;
581 long capa;
582
583 if (old_len > ARY_MAX_SIZE - add_len) {
584 rb_raise(rb_eIndexError, "index %ld too big", new_len);
585 }
586 if (ARY_SHARED_P(ary)) {
587 if (new_len > ary_embed_capa(ary)) {
588 VALUE shared_root = ARY_SHARED_ROOT(ary);
589 if (ARY_SHARED_ROOT_OCCUPIED(shared_root)) {
590 if (ARY_HEAP_PTR(ary) - RARRAY_CONST_PTR(shared_root) + new_len <= RARRAY_LEN(shared_root)) {
591 rb_ary_modify_check(ary);
592
593 ary_verify(ary);
594 ary_verify(shared_root);
595 return shared_root;
596 }
597 else {
598 /* if array is shared, then it is likely it participate in push/shift pattern */
599 rb_ary_modify(ary);
600 capa = ARY_CAPA(ary);
601 if (new_len > capa - (capa >> 6)) {
602 ary_double_capa(ary, new_len);
603 }
604 ary_verify(ary);
605 return ary;
606 }
607 }
608 }
609 ary_verify(ary);
610 rb_ary_modify(ary);
611 }
612 else {
613 rb_ary_modify_check(ary);
614 }
615 capa = ARY_CAPA(ary);
616 if (new_len > capa) {
617 ary_double_capa(ary, new_len);
618 }
619
620 ary_verify(ary);
621 return ary;
622}
623
624/*
625 * call-seq:
626 * freeze -> self
627 *
628 * Freezes +self+ (if not already frozen); returns +self+:
629 *
630 * a = []
631 * a.frozen? # => false
632 * a.freeze
633 * a.frozen? # => true
634 *
635 * No further changes may be made to +self+;
636 * raises FrozenError if a change is attempted.
637 *
638 * Related: Kernel#frozen?.
639 */
640
641VALUE
642rb_ary_freeze(VALUE ary)
643{
645
646 if (OBJ_FROZEN(ary)) return ary;
647
648 if (!ARY_EMBED_P(ary) && !ARY_SHARED_P(ary) && !ARY_SHARED_ROOT_P(ary)) {
649 ary_shrink_capa(ary);
650 }
651
652 return rb_obj_freeze(ary);
653}
654
655/* This can be used to take a snapshot of an array (with
656 e.g. rb_ary_replace) and check later whether the array has been
657 modified from the snapshot. The snapshot is cheap, though if
658 something does modify the array it will pay the cost of copying
659 it. If Array#pop or Array#shift has been called, the array will
660 be still shared with the snapshot, but the array length will
661 differ. */
662VALUE
663rb_ary_shared_with_p(VALUE ary1, VALUE ary2)
664{
665 if (!ARY_EMBED_P(ary1) && ARY_SHARED_P(ary1) &&
666 !ARY_EMBED_P(ary2) && ARY_SHARED_P(ary2) &&
667 ARY_SHARED_ROOT(ary1) == ARY_SHARED_ROOT(ary2) &&
668 ARY_HEAP_LEN(ary1) == ARY_HEAP_LEN(ary2)) {
669 return Qtrue;
670 }
671 return Qfalse;
672}
673
674static VALUE
675ary_alloc_embed(VALUE klass, long capa)
676{
677 size_t size = ary_embed_size(capa);
678 RUBY_ASSERT(rb_gc_size_allocatable_p(size));
679 NEWOBJ_OF(ary, struct RArray, klass,
680 T_ARRAY | RARRAY_EMBED_FLAG | (RGENGC_WB_PROTECTED_ARRAY ? FL_WB_PROTECTED : 0),
681 size, 0);
682 /* Created array is:
683 * FL_SET_EMBED((VALUE)ary);
684 * ARY_SET_EMBED_LEN((VALUE)ary, 0);
685 */
686 return (VALUE)ary;
687}
688
689static VALUE
690ary_alloc_heap(VALUE klass)
691{
692 NEWOBJ_OF(ary, struct RArray, klass,
694 sizeof(struct RArray), 0);
695 return (VALUE)ary;
696}
697
698static VALUE
699empty_ary_alloc(VALUE klass)
700{
701 RUBY_DTRACE_CREATE_HOOK(ARRAY, 0);
702 return ary_alloc_embed(klass, 0);
703}
704
705static VALUE
706ary_new(VALUE klass, long capa)
707{
708 VALUE ary;
709
710 if (capa < 0) {
711 rb_raise(rb_eArgError, "negative array size (or size too big)");
712 }
713 if (capa > ARY_MAX_SIZE) {
714 rb_raise(rb_eArgError, "array size too big");
715 }
716
717 RUBY_DTRACE_CREATE_HOOK(ARRAY, capa);
718
719 if (ary_embeddable_p(capa)) {
720 ary = ary_alloc_embed(klass, capa);
721 }
722 else {
723 ary = ary_alloc_heap(klass);
724 ARY_SET_CAPA(ary, capa);
725 RUBY_ASSERT(!ARY_EMBED_P(ary));
726
727 ARY_SET_PTR(ary, ary_heap_alloc_buffer(capa));
728 ARY_SET_HEAP_LEN(ary, 0);
729 }
730
731 return ary;
732}
733
734VALUE
735rb_ary_new_capa(long capa)
736{
737 return ary_new(rb_cArray, capa);
738}
739
740VALUE
741rb_ary_new(void)
742{
743 return rb_ary_new_capa(0);
744}
745
747(rb_ary_new_from_args)(long n, ...)
748{
749 va_list ar;
750 VALUE ary;
751 long i;
752
753 ary = rb_ary_new2(n);
754
755 va_start(ar, n);
756 for (i=0; i<n; i++) {
757 ARY_SET(ary, i, va_arg(ar, VALUE));
758 }
759 va_end(ar);
760
761 ARY_SET_LEN(ary, n);
762 return ary;
763}
764
765VALUE
766rb_ary_tmp_new_from_values(VALUE klass, long n, const VALUE *elts)
767{
768 VALUE ary;
769
770 ary = ary_new(klass, n);
771 if (n > 0 && elts) {
772 ary_memcpy(ary, 0, n, elts);
773 ARY_SET_LEN(ary, n);
774 }
775
776 return ary;
777}
778
779VALUE
780rb_ary_new_from_values(long n, const VALUE *elts)
781{
782 return rb_ary_tmp_new_from_values(rb_cArray, n, elts);
783}
784
785static VALUE
786ec_ary_alloc_embed(rb_execution_context_t *ec, VALUE klass, long capa)
787{
788 size_t size = ary_embed_size(capa);
789 RUBY_ASSERT(rb_gc_size_allocatable_p(size));
790 NEWOBJ_OF(ary, struct RArray, klass,
791 T_ARRAY | RARRAY_EMBED_FLAG | (RGENGC_WB_PROTECTED_ARRAY ? FL_WB_PROTECTED : 0),
792 size, ec);
793 /* Created array is:
794 * FL_SET_EMBED((VALUE)ary);
795 * ARY_SET_EMBED_LEN((VALUE)ary, 0);
796 */
797 return (VALUE)ary;
798}
799
800static VALUE
801ec_ary_alloc_heap(rb_execution_context_t *ec, VALUE klass)
802{
803 NEWOBJ_OF(ary, struct RArray, klass,
805 sizeof(struct RArray), ec);
806 return (VALUE)ary;
807}
808
809static VALUE
810ec_ary_new(rb_execution_context_t *ec, VALUE klass, long capa)
811{
812 VALUE ary;
813
814 if (capa < 0) {
815 rb_raise(rb_eArgError, "negative array size (or size too big)");
816 }
817 if (capa > ARY_MAX_SIZE) {
818 rb_raise(rb_eArgError, "array size too big");
819 }
820
821 RUBY_DTRACE_CREATE_HOOK(ARRAY, capa);
822
823 if (ary_embeddable_p(capa)) {
824 ary = ec_ary_alloc_embed(ec, klass, capa);
825 }
826 else {
827 ary = ec_ary_alloc_heap(ec, klass);
828 ARY_SET_CAPA(ary, capa);
829 RUBY_ASSERT(!ARY_EMBED_P(ary));
830
831 ARY_SET_PTR(ary, ary_heap_alloc_buffer(capa));
832 ARY_SET_HEAP_LEN(ary, 0);
833 }
834
835 return ary;
836}
837
838VALUE
839rb_ec_ary_new_from_values(rb_execution_context_t *ec, long n, const VALUE *elts)
840{
841 VALUE ary;
842
843 ary = ec_ary_new(ec, rb_cArray, n);
844 if (n > 0 && elts) {
845 ary_memcpy(ary, 0, n, elts);
846 ARY_SET_LEN(ary, n);
847 }
848
849 return ary;
850}
851
852VALUE
853rb_ary_hidden_new(long capa)
854{
855 VALUE ary = ary_new(0, capa);
856 return ary;
857}
858
859VALUE
860rb_ary_hidden_new_fill(long capa)
861{
862 VALUE ary = rb_ary_hidden_new(capa);
863 ary_memfill(ary, 0, capa, Qnil);
864 ARY_SET_LEN(ary, capa);
865 return ary;
866}
867
868void
869rb_ary_free(VALUE ary)
870{
871 if (ARY_OWNS_HEAP_P(ary)) {
872 if (USE_DEBUG_COUNTER &&
873 !ARY_SHARED_ROOT_P(ary) &&
874 ARY_HEAP_CAPA(ary) > RARRAY_LEN(ary)) {
875 RB_DEBUG_COUNTER_INC(obj_ary_extracapa);
876 }
877
878 RB_DEBUG_COUNTER_INC(obj_ary_ptr);
879 ary_heap_free(ary);
880 }
881 else {
882 RB_DEBUG_COUNTER_INC(obj_ary_embed);
883 }
884
885 if (ARY_SHARED_P(ary)) {
886 RB_DEBUG_COUNTER_INC(obj_ary_shared);
887 }
888 if (ARY_SHARED_ROOT_P(ary) && ARY_SHARED_ROOT_OCCUPIED(ary)) {
889 RB_DEBUG_COUNTER_INC(obj_ary_shared_root_occupied);
890 }
891}
892
893static VALUE fake_ary_flags;
894
895static VALUE
896init_fake_ary_flags(void)
897{
898 struct RArray fake_ary = {0};
899 fake_ary.basic.flags = T_ARRAY;
900 VALUE ary = (VALUE)&fake_ary;
901 rb_ary_freeze(ary);
902 return fake_ary.basic.flags;
903}
904
905VALUE
906rb_setup_fake_ary(struct RArray *fake_ary, const VALUE *list, long len)
907{
908 fake_ary->basic.flags = fake_ary_flags;
909 RBASIC_CLEAR_CLASS((VALUE)fake_ary);
910
911 // bypass frozen checks
912 fake_ary->as.heap.ptr = list;
913 fake_ary->as.heap.len = len;
914 fake_ary->as.heap.aux.capa = len;
915 return (VALUE)fake_ary;
916}
917
918size_t
919rb_ary_memsize(VALUE ary)
920{
921 if (ARY_OWNS_HEAP_P(ary)) {
922 return ARY_CAPA(ary) * sizeof(VALUE);
923 }
924 else {
925 return 0;
926 }
927}
928
929static VALUE
930ary_make_shared(VALUE ary)
931{
932 ary_verify(ary);
933
934 if (ARY_SHARED_P(ary)) {
935 return ARY_SHARED_ROOT(ary);
936 }
937 else if (ARY_SHARED_ROOT_P(ary)) {
938 return ary;
939 }
940 else if (OBJ_FROZEN(ary)) {
941 return ary;
942 }
943 else {
944 long capa = ARY_CAPA(ary);
945 long len = RARRAY_LEN(ary);
946
947 /* Shared roots cannot be embedded because the reference count
948 * (refcnt) is stored in as.heap.aux.capa. */
949 VALUE shared = ary_alloc_heap(0);
950 FL_SET_SHARED_ROOT(shared);
951
952 if (ARY_EMBED_P(ary)) {
953 VALUE *ptr = ary_heap_alloc_buffer(capa);
954 ARY_SET_PTR(shared, ptr);
955 ary_memcpy(shared, 0, len, RARRAY_CONST_PTR(ary));
956
957 FL_UNSET_EMBED(ary);
958 ARY_SET_HEAP_LEN(ary, len);
959 ARY_SET_PTR(ary, ptr);
960 }
961 else {
962 ARY_SET_PTR(shared, RARRAY_CONST_PTR(ary));
963 }
964
965 ARY_SET_LEN(shared, capa);
966 ary_mem_clear(shared, len, capa - len);
967 rb_ary_set_shared(ary, shared);
968
969 ary_verify(shared);
970 ary_verify(ary);
971
972 return shared;
973 }
974}
975
976static VALUE
977ary_make_substitution(VALUE ary)
978{
979 long len = RARRAY_LEN(ary);
980
981 if (ary_embeddable_p(len)) {
982 VALUE subst = rb_ary_new_capa(len);
983 RUBY_ASSERT(ARY_EMBED_P(subst));
984
985 ary_memcpy(subst, 0, len, RARRAY_CONST_PTR(ary));
986 ARY_SET_EMBED_LEN(subst, len);
987 return subst;
988 }
989 else {
990 return rb_ary_increment_share(ary_make_shared(ary));
991 }
992}
993
994VALUE
995rb_assoc_new(VALUE car, VALUE cdr)
996{
997 return rb_ary_new3(2, car, cdr);
998}
999
1000VALUE
1001rb_to_array_type(VALUE ary)
1002{
1003 return rb_convert_type_with_id(ary, T_ARRAY, "Array", idTo_ary);
1004}
1005#define to_ary rb_to_array_type
1006
1007VALUE
1008rb_check_array_type(VALUE ary)
1009{
1010 return rb_check_convert_type_with_id(ary, T_ARRAY, "Array", idTo_ary);
1011}
1012
1013VALUE
1014rb_check_to_array(VALUE ary)
1015{
1016 return rb_check_convert_type_with_id(ary, T_ARRAY, "Array", idTo_a);
1017}
1018
1019VALUE
1020rb_to_array(VALUE ary)
1021{
1022 return rb_convert_type_with_id(ary, T_ARRAY, "Array", idTo_a);
1023}
1024
1025/*
1026 * call-seq:
1027 * Array.try_convert(object) -> object, new_array, or nil
1028 *
1029 * Attempts to return an array, based on the given +object+.
1030 *
1031 * If +object+ is an array, returns +object+.
1032 *
1033 * Otherwise if +object+ responds to <tt>:to_ary</tt>.
1034 * calls <tt>object.to_ary</tt>:
1035 * if the return value is an array or +nil+, returns that value;
1036 * if not, raises TypeError.
1037 *
1038 * Otherwise returns +nil+.
1039 *
1040 * Related: see {Methods for Creating an Array}[rdoc-ref:Array@Methods+for+Creating+an+Array].
1041 */
1042
1043static VALUE
1044rb_ary_s_try_convert(VALUE dummy, VALUE ary)
1045{
1046 return rb_check_array_type(ary);
1047}
1048
1049/* :nodoc: */
1050static VALUE
1051rb_ary_s_new(int argc, VALUE *argv, VALUE klass)
1052{
1053 VALUE ary;
1054
1055 if (klass == rb_cArray) {
1056 long size = 0;
1057 if (argc > 0 && FIXNUM_P(argv[0])) {
1058 size = FIX2LONG(argv[0]);
1059 if (size < 0) size = 0;
1060 }
1061
1062 ary = ary_new(klass, size);
1063
1064 rb_obj_call_init_kw(ary, argc, argv, RB_PASS_CALLED_KEYWORDS);
1065 }
1066 else {
1067 ary = rb_class_new_instance_pass_kw(argc, argv, klass);
1068 }
1069
1070 return ary;
1071}
1072
1073/*
1074 * call-seq:
1075 * Array.new -> new_empty_array
1076 * Array.new(array) -> new_array
1077 * Array.new(size, default_value = nil) -> new_array
1078 * Array.new(size = 0) {|index| ... } -> new_array
1079 *
1080 * Returns a new array.
1081 *
1082 * With no block and no argument given, returns a new empty array:
1083 *
1084 * Array.new # => []
1085 *
1086 * With no block and array argument given, returns a new array with the same elements:
1087 *
1088 * Array.new([:foo, 'bar', 2]) # => [:foo, "bar", 2]
1089 *
1090 * With no block and integer argument given, returns a new array containing
1091 * that many instances of the given +default_value+:
1092 *
1093 * Array.new(0) # => []
1094 * Array.new(3) # => [nil, nil, nil]
1095 * Array.new(2, 3) # => [3, 3]
1096 *
1097 * With a block given, returns an array of the given +size+;
1098 * calls the block with each +index+ in the range <tt>(0...size)</tt>;
1099 * the element at that +index+ in the returned array is the blocks return value:
1100 *
1101 * Array.new(3) {|index| "Element #{index}" } # => ["Element 0", "Element 1", "Element 2"]
1102 *
1103 * A common pitfall for new Rubyists is providing an expression as +default_value+:
1104 *
1105 * array = Array.new(2, {})
1106 * array # => [{}, {}]
1107 * array[0][:a] = 1
1108 * array # => [{a: 1}, {a: 1}], as array[0] and array[1] are same object
1109 *
1110 * If you want the elements of the array to be distinct, you should pass a block:
1111 *
1112 * array = Array.new(2) { {} }
1113 * array # => [{}, {}]
1114 * array[0][:a] = 1
1115 * array # => [{a: 1}, {}], as array[0] and array[1] are different objects
1116 *
1117 * Raises TypeError if the first argument is not either an array
1118 * or an {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects]).
1119 * Raises ArgumentError if the first argument is a negative integer.
1120 *
1121 * Related: see {Methods for Creating an Array}[rdoc-ref:Array@Methods+for+Creating+an+Array].
1122 */
1123
1124static VALUE
1125rb_ary_initialize(int argc, VALUE *argv, VALUE ary)
1126{
1127 long len;
1128 VALUE size, val;
1129
1130 rb_ary_modify(ary);
1131 if (argc == 0) {
1132 rb_ary_reset(ary);
1133 RUBY_ASSERT(ARY_EMBED_P(ary));
1134 RUBY_ASSERT(ARY_EMBED_LEN(ary) == 0);
1135 if (rb_block_given_p()) {
1136 rb_warning("given block not used");
1137 }
1138 return ary;
1139 }
1140 rb_scan_args(argc, argv, "02", &size, &val);
1141 if (argc == 1 && !FIXNUM_P(size)) {
1142 val = rb_check_array_type(size);
1143 if (!NIL_P(val)) {
1144 rb_ary_replace(ary, val);
1145 return ary;
1146 }
1147 }
1148
1149 len = NUM2LONG(size);
1150 /* NUM2LONG() may call size.to_int, ary can be frozen, modified, etc */
1151 if (len < 0) {
1152 rb_raise(rb_eArgError, "negative array size");
1153 }
1154 if (len > ARY_MAX_SIZE) {
1155 rb_raise(rb_eArgError, "array size too big");
1156 }
1157 /* recheck after argument conversion */
1158 rb_ary_modify(ary);
1159 ary_resize_capa(ary, len);
1160 if (rb_block_given_p()) {
1161 long i;
1162
1163 if (argc == 2) {
1164 rb_warn("block supersedes default value argument");
1165 }
1166 for (i=0; i<len; i++) {
1167 rb_ary_store(ary, i, rb_yield(LONG2NUM(i)));
1168 ARY_SET_LEN(ary, i + 1);
1169 }
1170 }
1171 else {
1172 ary_memfill(ary, 0, len, val);
1173 ARY_SET_LEN(ary, len);
1174 }
1175 return ary;
1176}
1177
1178/*
1179 * Returns a new array, populated with the given objects:
1180 *
1181 * Array[1, 'a', /^A/] # => [1, "a", /^A/]
1182 * Array[] # => []
1183 * Array.[](1, 'a', /^A/) # => [1, "a", /^A/]
1184 *
1185 * Related: see {Methods for Creating an Array}[rdoc-ref:Array@Methods+for+Creating+an+Array].
1186 */
1187
1188static VALUE
1189rb_ary_s_create(int argc, VALUE *argv, VALUE klass)
1190{
1191 VALUE ary = ary_new(klass, argc);
1192 if (argc > 0 && argv) {
1193 ary_memcpy(ary, 0, argc, argv);
1194 ARY_SET_LEN(ary, argc);
1195 }
1196
1197 return ary;
1198}
1199
1200void
1201rb_ary_store(VALUE ary, long idx, VALUE val)
1202{
1203 long len = RARRAY_LEN(ary);
1204
1205 if (idx < 0) {
1206 idx += len;
1207 if (idx < 0) {
1208 rb_raise(rb_eIndexError, "index %ld too small for array; minimum: %ld",
1209 idx - len, -len);
1210 }
1211 }
1212 else if (idx >= ARY_MAX_SIZE) {
1213 rb_raise(rb_eIndexError, "index %ld too big", idx);
1214 }
1215
1216 rb_ary_modify(ary);
1217 if (idx >= ARY_CAPA(ary)) {
1218 ary_double_capa(ary, idx);
1219 }
1220 if (idx > len) {
1221 ary_mem_clear(ary, len, idx - len + 1);
1222 }
1223
1224 if (idx >= len) {
1225 ARY_SET_LEN(ary, idx + 1);
1226 }
1227 ARY_SET(ary, idx, val);
1228}
1229
1230static VALUE
1231ary_make_partial(VALUE ary, VALUE klass, long offset, long len)
1232{
1233 RUBY_ASSERT(offset >= 0);
1234 RUBY_ASSERT(len >= 0);
1235 RUBY_ASSERT(offset+len <= RARRAY_LEN(ary));
1236
1237 VALUE result = ary_alloc_heap(klass);
1238 size_t embed_capa = ary_embed_capa(result);
1239 if ((size_t)len <= embed_capa) {
1240 FL_SET_EMBED(result);
1241 ary_memcpy(result, 0, len, RARRAY_CONST_PTR(ary) + offset);
1242 ARY_SET_EMBED_LEN(result, len);
1243 }
1244 else {
1245 VALUE shared = ary_make_shared(ary);
1246
1247 /* The ary_make_shared call may allocate, which can trigger a GC
1248 * compaction. This can cause the array to be embedded because it has
1249 * a length of 0. */
1250 FL_UNSET_EMBED(result);
1251
1252 ARY_SET_PTR(result, RARRAY_CONST_PTR(ary));
1253 ARY_SET_LEN(result, RARRAY_LEN(ary));
1254 rb_ary_set_shared(result, shared);
1255
1256 ARY_INCREASE_PTR(result, offset);
1257 ARY_SET_LEN(result, len);
1258
1259 ary_verify(shared);
1260 }
1261
1262 ary_verify(result);
1263 return result;
1264}
1265
1266static VALUE
1267ary_make_partial_step(VALUE ary, VALUE klass, long offset, long len, long step)
1268{
1269 RUBY_ASSERT(offset >= 0);
1270 RUBY_ASSERT(len >= 0);
1271 RUBY_ASSERT(offset+len <= RARRAY_LEN(ary));
1272 RUBY_ASSERT(step != 0);
1273
1274 const long orig_len = len;
1275
1276 if (step > 0 && step >= len) {
1277 VALUE result = ary_new(klass, 1);
1278 VALUE *ptr = (VALUE *)ARY_EMBED_PTR(result);
1279 const VALUE *values = RARRAY_CONST_PTR(ary);
1280
1281 RB_OBJ_WRITE(result, ptr, values[offset]);
1282 ARY_SET_EMBED_LEN(result, 1);
1283 return result;
1284 }
1285 else if (step < 0 && step < -len) {
1286 step = -len;
1287 }
1288
1289 long ustep = (step < 0) ? -step : step;
1290 len = roomof(len, ustep);
1291
1292 long i;
1293 long j = offset + ((step > 0) ? 0 : (orig_len - 1));
1294
1295 VALUE result = ary_new(klass, len);
1296 if (ARY_EMBED_P(result)) {
1297 VALUE *ptr = (VALUE *)ARY_EMBED_PTR(result);
1298 const VALUE *values = RARRAY_CONST_PTR(ary);
1299
1300 for (i = 0; i < len; ++i) {
1301 RB_OBJ_WRITE(result, ptr+i, values[j]);
1302 j += step;
1303 }
1304 ARY_SET_EMBED_LEN(result, len);
1305 }
1306 else {
1307 const VALUE *values = RARRAY_CONST_PTR(ary);
1308
1309 RARRAY_PTR_USE(result, ptr, {
1310 for (i = 0; i < len; ++i) {
1311 RB_OBJ_WRITE(result, ptr+i, values[j]);
1312 j += step;
1313 }
1314 });
1315 ARY_SET_LEN(result, len);
1316 }
1317
1318 return result;
1319}
1320
1321static VALUE
1322ary_make_shared_copy(VALUE ary)
1323{
1324 return ary_make_partial(ary, rb_cArray, 0, RARRAY_LEN(ary));
1325}
1326
1327enum ary_take_pos_flags
1328{
1329 ARY_TAKE_FIRST = 0,
1330 ARY_TAKE_LAST = 1
1331};
1332
1333static VALUE
1334ary_take_first_or_last_n(VALUE ary, long n, enum ary_take_pos_flags last)
1335{
1336 long len = RARRAY_LEN(ary);
1337 long offset = 0;
1338
1339 if (n > len) {
1340 n = len;
1341 }
1342 else if (n < 0) {
1343 rb_raise(rb_eArgError, "negative array size");
1344 }
1345 if (last) {
1346 offset = len - n;
1347 }
1348 return ary_make_partial(ary, rb_cArray, offset, n);
1349}
1350
1351static VALUE
1352ary_take_first_or_last(int argc, const VALUE *argv, VALUE ary, enum ary_take_pos_flags last)
1353{
1354 argc = rb_check_arity(argc, 0, 1);
1355 /* the case optional argument is omitted should be handled in
1356 * callers of this function. if another arity case is added,
1357 * this arity check needs to rewrite. */
1358 RUBY_ASSERT_ALWAYS(argc == 1);
1359 return ary_take_first_or_last_n(ary, NUM2LONG(argv[0]), last);
1360}
1361
1362/*
1363 * call-seq:
1364 * self << object -> self
1365 *
1366 * Appends +object+ as the last element in +self+; returns +self+:
1367 *
1368 * [:foo, 'bar', 2] << :baz # => [:foo, "bar", 2, :baz]
1369 *
1370 * Appends +object+ as a single element, even if it is another array:
1371 *
1372 * [:foo, 'bar', 2] << [3, 4] # => [:foo, "bar", 2, [3, 4]]
1373 *
1374 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
1375 */
1376
1377VALUE
1378rb_ary_push(VALUE ary, VALUE item)
1379{
1380 long idx = RARRAY_LEN((ary_verify(ary), ary));
1381 VALUE target_ary = ary_ensure_room_for_push(ary, 1);
1383 RB_OBJ_WRITE(target_ary, &ptr[idx], item);
1384 });
1385 ARY_SET_LEN(ary, idx + 1);
1386 ary_verify(ary);
1387 return ary;
1388}
1389
1390VALUE
1391rb_ary_cat(VALUE ary, const VALUE *argv, long len)
1392{
1393 long oldlen = RARRAY_LEN(ary);
1394 VALUE target_ary = ary_ensure_room_for_push(ary, len);
1395 ary_memcpy0(ary, oldlen, len, argv, target_ary);
1396 ARY_SET_LEN(ary, oldlen + len);
1397 return ary;
1398}
1399
1400/*
1401 * call-seq:
1402 * push(*objects) -> self
1403 * append(*objects) -> self
1404 *
1405 * Appends each argument in +objects+ to +self+; returns +self+:
1406 *
1407 * a = [:foo, 'bar', 2] # => [:foo, "bar", 2]
1408 * a.push(:baz, :bat) # => [:foo, "bar", 2, :baz, :bat]
1409 *
1410 * Appends each argument as a single element, even if it is another array:
1411 *
1412 * a = [:foo, 'bar', 2] # => [:foo, "bar", 2]
1413 a.push([:baz, :bat], [:bam, :bad]) # => [:foo, "bar", 2, [:baz, :bat], [:bam, :bad]]
1414 *
1415 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
1416 */
1417
1418static VALUE
1419rb_ary_push_m(int argc, VALUE *argv, VALUE ary)
1420{
1421 return rb_ary_cat(ary, argv, argc);
1422}
1423
1424VALUE
1425rb_ary_pop(VALUE ary)
1426{
1427 long n;
1428 rb_ary_modify_check(ary);
1429 n = RARRAY_LEN(ary);
1430 if (n == 0) return Qnil;
1431 if (ARY_OWNS_HEAP_P(ary) &&
1432 n * 3 < ARY_CAPA(ary) &&
1433 ARY_CAPA(ary) > ARY_DEFAULT_SIZE)
1434 {
1435 ary_resize_capa(ary, n * 2);
1436 }
1437 --n;
1438 ARY_SET_LEN(ary, n);
1439 ary_verify(ary);
1440 return RARRAY_AREF(ary, n);
1441}
1442
1443/*
1444 * call-seq:
1445 * pop -> object or nil
1446 * pop(count) -> new_array
1447 *
1448 * Removes and returns trailing elements of +self+.
1449 *
1450 * With no argument given, removes and returns the last element, if available;
1451 * otherwise returns +nil+:
1452 *
1453 * a = [:foo, 'bar', 2]
1454 * a.pop # => 2
1455 * a # => [:foo, "bar"]
1456 * [].pop # => nil
1457 *
1458 * With non-negative integer argument +count+ given,
1459 * returns a new array containing the trailing +count+ elements of +self+, as available:
1460 *
1461 * a = [:foo, 'bar', 2]
1462 * a.pop(2) # => ["bar", 2]
1463 * a # => [:foo]
1464 *
1465 * a = [:foo, 'bar', 2]
1466 * a.pop(50) # => [:foo, "bar", 2]
1467 * a # => []
1468 *
1469 * Related: Array#push;
1470 * see also {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
1471 */
1472
1473static VALUE
1474rb_ary_pop_m(int argc, VALUE *argv, VALUE ary)
1475{
1476 VALUE result;
1477
1478 if (argc == 0) {
1479 return rb_ary_pop(ary);
1480 }
1481
1482 rb_ary_modify_check(ary);
1483 result = ary_take_first_or_last(argc, argv, ary, ARY_TAKE_LAST);
1484 ARY_INCREASE_LEN(ary, -RARRAY_LEN(result));
1485 ary_verify(ary);
1486 return result;
1487}
1488
1489VALUE
1490rb_ary_shift(VALUE ary)
1491{
1492 VALUE top;
1493 long len = RARRAY_LEN(ary);
1494
1495 if (len == 0) {
1496 rb_ary_modify_check(ary);
1497 return Qnil;
1498 }
1499
1500 top = RARRAY_AREF(ary, 0);
1501
1502 rb_ary_behead(ary, 1);
1503
1504 return top;
1505}
1506
1507/*
1508 * call-seq:
1509 * shift -> object or nil
1510 * shift(count) -> new_array or nil
1511 *
1512 * Removes and returns leading elements from +self+.
1513 *
1514 * With no argument, removes and returns one element, if available,
1515 * or +nil+ otherwise:
1516 *
1517 * a = [0, 1, 2, 3]
1518 * a.shift # => 0
1519 * a # => [1, 2, 3]
1520 * [].shift # => nil
1521 *
1522 * With non-negative numeric argument +count+ given,
1523 * removes and returns the first +count+ elements:
1524 *
1525 * a = [0, 1, 2, 3]
1526 * a.shift(2) # => [0, 1]
1527 * a # => [2, 3]
1528 * a.shift(1.1) # => [2]
1529 * a # => [3]
1530 * a.shift(0) # => []
1531 * a # => [3]
1532 *
1533 * If +count+ is large,
1534 * removes and returns all elements:
1535 *
1536 * a = [0, 1, 2, 3]
1537 * a.shift(50) # => [0, 1, 2, 3]
1538 * a # => []
1539 *
1540 * If +self+ is empty, returns a new empty array.
1541 *
1542 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
1543 */
1544
1545static VALUE
1546rb_ary_shift_m(int argc, VALUE *argv, VALUE ary)
1547{
1548 VALUE result;
1549 long n;
1550
1551 if (argc == 0) {
1552 return rb_ary_shift(ary);
1553 }
1554
1555 rb_ary_modify_check(ary);
1556 result = ary_take_first_or_last(argc, argv, ary, ARY_TAKE_FIRST);
1557 n = RARRAY_LEN(result);
1558 rb_ary_behead(ary,n);
1559
1560 return result;
1561}
1562
1563VALUE
1564rb_ary_behead(VALUE ary, long n)
1565{
1566 if (n <= 0) {
1567 return ary;
1568 }
1569
1570 rb_ary_modify_check(ary);
1571
1572 if (!ARY_SHARED_P(ary)) {
1573 if (ARY_EMBED_P(ary) || RARRAY_LEN(ary) < ARY_DEFAULT_SIZE) {
1575 MEMMOVE(ptr, ptr + n, VALUE, RARRAY_LEN(ary) - n);
1576 }); /* WB: no new reference */
1577 ARY_INCREASE_LEN(ary, -n);
1578 ary_verify(ary);
1579 return ary;
1580 }
1581
1582 ary_mem_clear(ary, 0, n);
1583 ary_make_shared(ary);
1584 }
1585 else if (ARY_SHARED_ROOT_OCCUPIED(ARY_SHARED_ROOT(ary))) {
1586 ary_mem_clear(ary, 0, n);
1587 }
1588
1589 ARY_INCREASE_PTR(ary, n);
1590 ARY_INCREASE_LEN(ary, -n);
1591 ary_verify(ary);
1592
1593 return ary;
1594}
1595
1596static VALUE
1597make_room_for_unshift(VALUE ary, const VALUE *head, VALUE *sharedp, int argc, long capa, long len)
1598{
1599 if (head - sharedp < argc) {
1600 long room = capa - len - argc;
1601
1602 room -= room >> 4;
1603 MEMMOVE((VALUE *)sharedp + argc + room, head, VALUE, len);
1604 head = sharedp + argc + room;
1605 }
1606 ARY_SET_PTR(ary, head - argc);
1607 RUBY_ASSERT(ARY_SHARED_ROOT_OCCUPIED(ARY_SHARED_ROOT(ary)));
1608
1609 ary_verify(ary);
1610 return ARY_SHARED_ROOT(ary);
1611}
1612
1613static VALUE
1614ary_modify_for_unshift(VALUE ary, int argc)
1615{
1616 long len = RARRAY_LEN(ary);
1617 long new_len = len + argc;
1618 long capa;
1619 const VALUE *head, *sharedp;
1620
1621 rb_ary_modify(ary);
1622 capa = ARY_CAPA(ary);
1623 if (capa - (capa >> 6) <= new_len) {
1624 ary_double_capa(ary, new_len);
1625 }
1626
1627 /* use shared array for big "queues" */
1628 if (new_len > ARY_DEFAULT_SIZE * 4 && !ARY_EMBED_P(ary)) {
1629 ary_verify(ary);
1630
1631 /* make a room for unshifted items */
1632 capa = ARY_CAPA(ary);
1633 ary_make_shared(ary);
1634
1635 head = sharedp = RARRAY_CONST_PTR(ary);
1636 return make_room_for_unshift(ary, head, (void *)sharedp, argc, capa, len);
1637 }
1638 else {
1639 /* sliding items */
1641 MEMMOVE(ptr + argc, ptr, VALUE, len);
1642 });
1643
1644 ary_verify(ary);
1645 return ary;
1646 }
1647}
1648
1649static VALUE
1650ary_ensure_room_for_unshift(VALUE ary, int argc)
1651{
1652 long len = RARRAY_LEN(ary);
1653 long new_len = len + argc;
1654
1655 if (len > ARY_MAX_SIZE - argc) {
1656 rb_raise(rb_eIndexError, "index %ld too big", new_len);
1657 }
1658 else if (! ARY_SHARED_P(ary)) {
1659 return ary_modify_for_unshift(ary, argc);
1660 }
1661 else {
1662 VALUE shared_root = ARY_SHARED_ROOT(ary);
1663 long capa = RARRAY_LEN(shared_root);
1664
1665 if (! ARY_SHARED_ROOT_OCCUPIED(shared_root)) {
1666 return ary_modify_for_unshift(ary, argc);
1667 }
1668 else if (new_len > capa) {
1669 return ary_modify_for_unshift(ary, argc);
1670 }
1671 else {
1672 const VALUE * head = RARRAY_CONST_PTR(ary);
1673 void *sharedp = (void *)RARRAY_CONST_PTR(shared_root);
1674
1675 rb_ary_modify_check(ary);
1676 return make_room_for_unshift(ary, head, sharedp, argc, capa, len);
1677 }
1678 }
1679}
1680
1681/*
1682 * call-seq:
1683 * unshift(*objects) -> self
1684 * prepend(*objects) -> self
1685 *
1686 * Prepends the given +objects+ to +self+:
1687 *
1688 * a = [:foo, 'bar', 2]
1689 * a.unshift(:bam, :bat) # => [:bam, :bat, :foo, "bar", 2]
1690 *
1691 * Related: Array#shift;
1692 * see also {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
1693 */
1694
1695VALUE
1696rb_ary_unshift_m(int argc, VALUE *argv, VALUE ary)
1697{
1698 long len = RARRAY_LEN(ary);
1699 VALUE target_ary;
1700
1701 if (argc == 0) {
1702 rb_ary_modify_check(ary);
1703 return ary;
1704 }
1705
1706 target_ary = ary_ensure_room_for_unshift(ary, argc);
1707 ary_memcpy0(ary, 0, argc, argv, target_ary);
1708 ARY_SET_LEN(ary, len + argc);
1709 return ary;
1710}
1711
1712VALUE
1713rb_ary_unshift(VALUE ary, VALUE item)
1714{
1715 return rb_ary_unshift_m(1, &item, ary);
1716}
1717
1718/* faster version - use this if you don't need to treat negative offset */
1719static inline VALUE
1720rb_ary_elt(VALUE ary, long offset)
1721{
1722 long len = RARRAY_LEN(ary);
1723 if (len == 0) return Qnil;
1724 if (offset < 0 || len <= offset) {
1725 return Qnil;
1726 }
1727 return RARRAY_AREF(ary, offset);
1728}
1729
1730VALUE
1731rb_ary_entry(VALUE ary, long offset)
1732{
1733 return rb_ary_entry_internal(ary, offset);
1734}
1735
1736VALUE
1737rb_ary_subseq_step(VALUE ary, long beg, long len, long step)
1738{
1739 VALUE klass;
1740 long alen = RARRAY_LEN(ary);
1741
1742 if (beg > alen) return Qnil;
1743 if (beg < 0 || len < 0) return Qnil;
1744
1745 if (alen < len || alen < beg + len) {
1746 len = alen - beg;
1747 }
1748 klass = rb_cArray;
1749 if (len == 0) return ary_new(klass, 0);
1750 if (step == 0)
1751 rb_raise(rb_eArgError, "slice step cannot be zero");
1752 if (step == 1)
1753 return ary_make_partial(ary, klass, beg, len);
1754 else
1755 return ary_make_partial_step(ary, klass, beg, len, step);
1756}
1757
1758VALUE
1759rb_ary_subseq(VALUE ary, long beg, long len)
1760{
1761 return rb_ary_subseq_step(ary, beg, len, 1);
1762}
1763
1764static VALUE rb_ary_aref2(VALUE ary, VALUE b, VALUE e);
1765
1766/*
1767 * call-seq:
1768 * self[index] -> object or nil
1769 * self[start, length] -> object or nil
1770 * self[range] -> object or nil
1771 * self[aseq] -> object or nil
1772 * slice(index) -> object or nil
1773 * slice(start, length) -> object or nil
1774 * slice(range) -> object or nil
1775 * slice(aseq) -> object or nil
1776 *
1777 * Returns elements from +self+; does not modify +self+.
1778 *
1779 * In brief:
1780 *
1781 * a = [:foo, 'bar', 2]
1782 *
1783 * # Single argument index: returns one element.
1784 * a[0] # => :foo # Zero-based index.
1785 * a[-1] # => 2 # Negative index counts backwards from end.
1786 *
1787 * # Arguments start and length: returns an array.
1788 * a[1, 2] # => ["bar", 2]
1789 * a[-2, 2] # => ["bar", 2] # Negative start counts backwards from end.
1790 *
1791 * # Single argument range: returns an array.
1792 * a[0..1] # => [:foo, "bar"]
1793 * a[0..-2] # => [:foo, "bar"] # Negative range-begin counts backwards from end.
1794 * a[-2..2] # => ["bar", 2] # Negative range-end counts backwards from end.
1795 *
1796 * When a single integer argument +index+ is given, returns the element at offset +index+:
1797 *
1798 * a = [:foo, 'bar', 2]
1799 * a[0] # => :foo
1800 * a[2] # => 2
1801 * a # => [:foo, "bar", 2]
1802 *
1803 * If +index+ is negative, counts backwards from the end of +self+:
1804 *
1805 * a = [:foo, 'bar', 2]
1806 * a[-1] # => 2
1807 * a[-2] # => "bar"
1808 *
1809 * If +index+ is out of range, returns +nil+.
1810 *
1811 * When two Integer arguments +start+ and +length+ are given,
1812 * returns a new array of size +length+ containing successive elements beginning at offset +start+:
1813 *
1814 * a = [:foo, 'bar', 2]
1815 * a[0, 2] # => [:foo, "bar"]
1816 * a[1, 2] # => ["bar", 2]
1817 *
1818 * If <tt>start + length</tt> is greater than <tt>self.length</tt>,
1819 * returns all elements from offset +start+ to the end:
1820 *
1821 * a = [:foo, 'bar', 2]
1822 * a[0, 4] # => [:foo, "bar", 2]
1823 * a[1, 3] # => ["bar", 2]
1824 * a[2, 2] # => [2]
1825 *
1826 * If <tt>start == self.size</tt> and <tt>length >= 0</tt>,
1827 * returns a new empty array.
1828 *
1829 * If +length+ is negative, returns +nil+.
1830 *
1831 * When a single Range argument +range+ is given,
1832 * treats <tt>range.min</tt> as +start+ above
1833 * and <tt>range.size</tt> as +length+ above:
1834 *
1835 * a = [:foo, 'bar', 2]
1836 * a[0..1] # => [:foo, "bar"]
1837 * a[1..2] # => ["bar", 2]
1838 *
1839 * Special case: If <tt>range.start == a.size</tt>, returns a new empty array.
1840 *
1841 * If <tt>range.end</tt> is negative, calculates the end index from the end:
1842 *
1843 * a = [:foo, 'bar', 2]
1844 * a[0..-1] # => [:foo, "bar", 2]
1845 * a[0..-2] # => [:foo, "bar"]
1846 * a[0..-3] # => [:foo]
1847 *
1848 * If <tt>range.start</tt> is negative, calculates the start index from the end:
1849 *
1850 * a = [:foo, 'bar', 2]
1851 * a[-1..2] # => [2]
1852 * a[-2..2] # => ["bar", 2]
1853 * a[-3..2] # => [:foo, "bar", 2]
1854 *
1855 * If <tt>range.start</tt> is larger than the array size, returns +nil+.
1856 *
1857 * a = [:foo, 'bar', 2]
1858 * a[4..1] # => nil
1859 * a[4..0] # => nil
1860 * a[4..-1] # => nil
1861 *
1862 * When a single Enumerator::ArithmeticSequence argument +aseq+ is given,
1863 * returns an array of elements corresponding to the indexes produced by
1864 * the sequence.
1865 *
1866 * a = ['--', 'data1', '--', 'data2', '--', 'data3']
1867 * a[(1..).step(2)] # => ["data1", "data2", "data3"]
1868 *
1869 * Unlike slicing with range, if the start or the end of the arithmetic sequence
1870 * is larger than array size, throws RangeError.
1871 *
1872 * a = ['--', 'data1', '--', 'data2', '--', 'data3']
1873 * a[(1..11).step(2)]
1874 * # RangeError (((1..11).step(2)) out of range)
1875 * a[(7..).step(2)]
1876 * # RangeError (((7..).step(2)) out of range)
1877 *
1878 * If given a single argument, and its type is not one of the listed, tries to
1879 * convert it to Integer, and raises if it is impossible:
1880 *
1881 * a = [:foo, 'bar', 2]
1882 * # Raises TypeError (no implicit conversion of Symbol into Integer):
1883 * a[:foo]
1884 *
1885 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
1886 */
1887
1888VALUE
1889rb_ary_aref(int argc, const VALUE *argv, VALUE ary)
1890{
1891 rb_check_arity(argc, 1, 2);
1892 if (argc == 2) {
1893 return rb_ary_aref2(ary, argv[0], argv[1]);
1894 }
1895 return rb_ary_aref1(ary, argv[0]);
1896}
1897
1898static VALUE
1899rb_ary_aref2(VALUE ary, VALUE b, VALUE e)
1900{
1901 long beg = NUM2LONG(b);
1902 long len = NUM2LONG(e);
1903 if (beg < 0) {
1904 beg += RARRAY_LEN(ary);
1905 }
1906 return rb_ary_subseq(ary, beg, len);
1907}
1908
1909VALUE
1910rb_ary_aref1(VALUE ary, VALUE arg)
1911{
1912 long beg, len, step;
1913
1914 /* special case - speeding up */
1915 if (FIXNUM_P(arg)) {
1916 return rb_ary_entry(ary, FIX2LONG(arg));
1917 }
1918 /* check if idx is Range or ArithmeticSequence */
1919 switch (rb_arithmetic_sequence_beg_len_step(arg, &beg, &len, &step, RARRAY_LEN(ary), 0)) {
1920 case Qfalse:
1921 break;
1922 case Qnil:
1923 return Qnil;
1924 default:
1925 return rb_ary_subseq_step(ary, beg, len, step);
1926 }
1927
1928 return rb_ary_entry(ary, NUM2LONG(arg));
1929}
1930
1931/*
1932 * call-seq:
1933 * at(index) -> object or nil
1934 *
1935 * Returns the element of +self+ specified by the given +index+
1936 * or +nil+ if there is no such element;
1937 * +index+ must be an
1938 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects].
1939 *
1940 * For non-negative +index+, returns the element of +self+ at offset +index+:
1941 *
1942 * a = [:foo, 'bar', 2]
1943 * a.at(0) # => :foo
1944 * a.at(2) # => 2
1945 * a.at(2.0) # => 2
1946 *
1947 * For negative +index+, counts backwards from the end of +self+:
1948 *
1949 * a.at(-2) # => "bar"
1950 *
1951 * Related: Array#[];
1952 * see also {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
1953 */
1954
1955VALUE
1956rb_ary_at(VALUE ary, VALUE pos)
1957{
1958 return rb_ary_entry(ary, NUM2LONG(pos));
1959}
1960
1961#if 0
1962static VALUE
1963rb_ary_first(int argc, VALUE *argv, VALUE ary)
1964{
1965 if (argc == 0) {
1966 if (RARRAY_LEN(ary) == 0) return Qnil;
1967 return RARRAY_AREF(ary, 0);
1968 }
1969 else {
1970 return ary_take_first_or_last(argc, argv, ary, ARY_TAKE_FIRST);
1971 }
1972}
1973#endif
1974
1975static VALUE
1976ary_first(VALUE self)
1977{
1978 return (RARRAY_LEN(self) == 0) ? Qnil : RARRAY_AREF(self, 0);
1979}
1980
1981static VALUE
1982ary_last(VALUE self)
1983{
1984 long len = RARRAY_LEN(self);
1985 return (len == 0) ? Qnil : RARRAY_AREF(self, len-1);
1986}
1987
1988VALUE
1989rb_ary_last(int argc, const VALUE *argv, VALUE ary) // used by parse.y
1990{
1991 if (argc == 0) {
1992 return ary_last(ary);
1993 }
1994 else {
1995 return ary_take_first_or_last(argc, argv, ary, ARY_TAKE_LAST);
1996 }
1997}
1998
1999/*
2000 * call-seq:
2001 * fetch(index) -> element
2002 * fetch(index, default_value) -> element or default_value
2003 * fetch(index) {|index| ... } -> element or block_return_value
2004 *
2005 * Returns the element of +self+ at offset +index+ if +index+ is in range; +index+ must be an
2006 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects].
2007 *
2008 * With the single argument +index+ and no block,
2009 * returns the element at offset +index+:
2010 *
2011 * a = [:foo, 'bar', 2]
2012 * a.fetch(1) # => "bar"
2013 * a.fetch(1.1) # => "bar"
2014 *
2015 * If +index+ is negative, counts from the end of the array:
2016 *
2017 * a = [:foo, 'bar', 2]
2018 * a.fetch(-1) # => 2
2019 * a.fetch(-2) # => "bar"
2020 *
2021 * With arguments +index+ and +default_value+ (which may be any object) and no block,
2022 * returns +default_value+ if +index+ is out-of-range:
2023 *
2024 * a = [:foo, 'bar', 2]
2025 * a.fetch(1, nil) # => "bar"
2026 * a.fetch(3, :foo) # => :foo
2027 *
2028 * With argument +index+ and a block,
2029 * returns the element at offset +index+ if index is in range
2030 * (and the block is not called); otherwise calls the block with index and returns its return value:
2031 *
2032 * a = [:foo, 'bar', 2]
2033 * a.fetch(1) {|index| raise 'Cannot happen' } # => "bar"
2034 * a.fetch(50) {|index| "Value for #{index}" } # => "Value for 50"
2035 *
2036 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
2037 */
2038
2039static VALUE
2040rb_ary_fetch(int argc, VALUE *argv, VALUE ary)
2041{
2042 VALUE pos, ifnone;
2043 long block_given;
2044 long idx;
2045
2046 rb_scan_args(argc, argv, "11", &pos, &ifnone);
2047 block_given = rb_block_given_p();
2048 if (block_given && argc == 2) {
2049 rb_warn("block supersedes default value argument");
2050 }
2051 idx = NUM2LONG(pos);
2052
2053 if (idx < 0) {
2054 idx += RARRAY_LEN(ary);
2055 }
2056 if (idx < 0 || RARRAY_LEN(ary) <= idx) {
2057 if (block_given) return rb_yield(pos);
2058 if (argc == 1) {
2059 rb_raise(rb_eIndexError, "index %ld outside of array bounds: %ld...%ld",
2060 idx - (idx < 0 ? RARRAY_LEN(ary) : 0), -RARRAY_LEN(ary), RARRAY_LEN(ary));
2061 }
2062 return ifnone;
2063 }
2064 return RARRAY_AREF(ary, idx);
2065}
2066
2067/*
2068 * call-seq:
2069 * find_index(object) -> integer or nil
2070 * find_index {|element| ... } -> integer or nil
2071 * find_index -> new_enumerator
2072 * index(object) -> integer or nil
2073 * index {|element| ... } -> integer or nil
2074 * index -> new_enumerator
2075 *
2076 * Returns the zero-based integer index of a specified element, or +nil+.
2077 *
2078 * With only argument +object+ given,
2079 * returns the index of the first element +element+
2080 * for which <tt>object == element</tt>:
2081 *
2082 * a = [:foo, 'bar', 2, 'bar']
2083 * a.index('bar') # => 1
2084 *
2085 * Returns +nil+ if no such element found.
2086 *
2087 * With only a block given,
2088 * calls the block with each successive element;
2089 * returns the index of the first element for which the block returns a truthy value:
2090 *
2091 * a = [:foo, 'bar', 2, 'bar']
2092 * a.index {|element| element == 'bar' } # => 1
2093 *
2094 * Returns +nil+ if the block never returns a truthy value.
2095 *
2096 * With neither an argument nor a block given, returns a new Enumerator.
2097 *
2098 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
2099 */
2100
2101static VALUE
2102rb_ary_index(int argc, VALUE *argv, VALUE ary)
2103{
2104 VALUE val;
2105 long i;
2106
2107 if (argc == 0) {
2108 RETURN_ENUMERATOR(ary, 0, 0);
2109 for (i=0; i<RARRAY_LEN(ary); i++) {
2110 if (RTEST(rb_yield(RARRAY_AREF(ary, i)))) {
2111 return LONG2NUM(i);
2112 }
2113 }
2114 return Qnil;
2115 }
2116 rb_check_arity(argc, 0, 1);
2117 val = argv[0];
2118 if (rb_block_given_p())
2119 rb_warn("given block not used");
2120 for (i=0; i<RARRAY_LEN(ary); i++) {
2121 VALUE e = RARRAY_AREF(ary, i);
2122 if (rb_equal(e, val)) {
2123 return LONG2NUM(i);
2124 }
2125 }
2126 return Qnil;
2127}
2128
2129/*
2130 * call-seq:
2131 * rindex(object) -> integer or nil
2132 * rindex {|element| ... } -> integer or nil
2133 * rindex -> new_enumerator
2134 *
2135 * Returns the index of the last element for which <tt>object == element</tt>.
2136 *
2137 * With argument +object+ given, returns the index of the last such element found:
2138 *
2139 * a = [:foo, 'bar', 2, 'bar']
2140 * a.rindex('bar') # => 3
2141 *
2142 * Returns +nil+ if no such object found.
2143 *
2144 * With a block given, calls the block with each successive element;
2145 * returns the index of the last element for which the block returns a truthy value:
2146 *
2147 * a = [:foo, 'bar', 2, 'bar']
2148 * a.rindex {|element| element == 'bar' } # => 3
2149 *
2150 * Returns +nil+ if the block never returns a truthy value.
2151 *
2152 * When neither an argument nor a block is given, returns a new Enumerator.
2153 *
2154 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
2155 */
2156
2157static VALUE
2158rb_ary_rindex(int argc, VALUE *argv, VALUE ary)
2159{
2160 VALUE val;
2161 long i = RARRAY_LEN(ary), len;
2162
2163 if (argc == 0) {
2164 RETURN_ENUMERATOR(ary, 0, 0);
2165 while (i--) {
2166 if (RTEST(rb_yield(RARRAY_AREF(ary, i))))
2167 return LONG2NUM(i);
2168 if (i > (len = RARRAY_LEN(ary))) {
2169 i = len;
2170 }
2171 }
2172 return Qnil;
2173 }
2174 rb_check_arity(argc, 0, 1);
2175 val = argv[0];
2176 if (rb_block_given_p())
2177 rb_warn("given block not used");
2178 while (i--) {
2179 VALUE e = RARRAY_AREF(ary, i);
2180 if (rb_equal(e, val)) {
2181 return LONG2NUM(i);
2182 }
2183 if (i > RARRAY_LEN(ary)) {
2184 break;
2185 }
2186 }
2187 return Qnil;
2188}
2189
2190VALUE
2191rb_ary_to_ary(VALUE obj)
2192{
2193 VALUE tmp = rb_check_array_type(obj);
2194
2195 if (!NIL_P(tmp)) return tmp;
2196 return rb_ary_new3(1, obj);
2197}
2198
2199static void
2200rb_ary_splice(VALUE ary, long beg, long len, const VALUE *rptr, long rlen)
2201{
2202 long olen;
2203 long rofs;
2204
2205 if (len < 0) rb_raise(rb_eIndexError, "negative length (%ld)", len);
2206 olen = RARRAY_LEN(ary);
2207 if (beg < 0) {
2208 beg += olen;
2209 if (beg < 0) {
2210 rb_raise(rb_eIndexError, "index %ld too small for array; minimum: %ld",
2211 beg - olen, -olen);
2212 }
2213 }
2214 if (olen < len || olen < beg + len) {
2215 len = olen - beg;
2216 }
2217
2218 {
2219 const VALUE *optr = RARRAY_CONST_PTR(ary);
2220 rofs = (rptr >= optr && rptr < optr + olen) ? rptr - optr : -1;
2221 }
2222
2223 if (beg >= olen) {
2224 VALUE target_ary;
2225 if (beg > ARY_MAX_SIZE - rlen) {
2226 rb_raise(rb_eIndexError, "index %ld too big", beg);
2227 }
2228 target_ary = ary_ensure_room_for_push(ary, rlen-len); /* len is 0 or negative */
2229 len = beg + rlen;
2230 ary_mem_clear(ary, olen, beg - olen);
2231 if (rlen > 0) {
2232 if (rofs != -1) rptr = RARRAY_CONST_PTR(ary) + rofs;
2233 ary_memcpy0(ary, beg, rlen, rptr, target_ary);
2234 }
2235 ARY_SET_LEN(ary, len);
2236 }
2237 else {
2238 long alen;
2239
2240 if (olen - len > ARY_MAX_SIZE - rlen) {
2241 rb_raise(rb_eIndexError, "index %ld too big", olen + rlen - len);
2242 }
2243 rb_ary_modify(ary);
2244 alen = olen + rlen - len;
2245 if (alen >= ARY_CAPA(ary)) {
2246 ary_double_capa(ary, alen);
2247 }
2248
2249 if (len != rlen) {
2251 MEMMOVE(ptr + beg + rlen, ptr + beg + len,
2252 VALUE, olen - (beg + len)));
2253 ARY_SET_LEN(ary, alen);
2254 }
2255 if (rlen > 0) {
2256 if (rofs == -1) {
2257 rb_gc_writebarrier_remember(ary);
2258 }
2259 else {
2260 /* In this case, we're copying from a region in this array, so
2261 * we don't need to fire the write barrier. */
2262 rptr = RARRAY_CONST_PTR(ary) + rofs;
2263 }
2264
2265 /* do not use RARRAY_PTR() because it can causes GC.
2266 * ary can contain T_NONE object because it is not cleared.
2267 */
2269 MEMMOVE(ptr + beg, rptr, VALUE, rlen));
2270 }
2271 }
2272}
2273
2274void
2275rb_ary_set_len(VALUE ary, long len)
2276{
2277 long capa;
2278
2279 rb_ary_modify_check(ary);
2280 if (ARY_SHARED_P(ary)) {
2281 rb_raise(rb_eRuntimeError, "can't set length of shared ");
2282 }
2283 if (len > (capa = (long)ARY_CAPA(ary))) {
2284 rb_bug("probable buffer overflow: %ld for %ld", len, capa);
2285 }
2286 ARY_SET_LEN(ary, len);
2287}
2288
2289VALUE
2290rb_ary_resize(VALUE ary, long len)
2291{
2292 long olen;
2293
2294 rb_ary_modify(ary);
2295 olen = RARRAY_LEN(ary);
2296 if (len == olen) return ary;
2297 if (len > ARY_MAX_SIZE) {
2298 rb_raise(rb_eIndexError, "index %ld too big", len);
2299 }
2300 if (len > olen) {
2301 if (len > ARY_CAPA(ary)) {
2302 ary_double_capa(ary, len);
2303 }
2304 ary_mem_clear(ary, olen, len - olen);
2305 ARY_SET_LEN(ary, len);
2306 }
2307 else if (ARY_EMBED_P(ary)) {
2308 ARY_SET_EMBED_LEN(ary, len);
2309 }
2310 else if (len <= ary_embed_capa(ary)) {
2311 const VALUE *ptr = ARY_HEAP_PTR(ary);
2312 long ptr_capa = ARY_HEAP_SIZE(ary);
2313 bool is_malloc_ptr = !ARY_SHARED_P(ary);
2314
2315 FL_SET_EMBED(ary);
2316
2317 MEMCPY((VALUE *)ARY_EMBED_PTR(ary), ptr, VALUE, len); /* WB: no new reference */
2318 ARY_SET_EMBED_LEN(ary, len);
2319
2320 if (is_malloc_ptr) ruby_sized_xfree((void *)ptr, ptr_capa);
2321 }
2322 else {
2323 if (olen > len + ARY_DEFAULT_SIZE) {
2324 size_t new_capa = ary_heap_realloc(ary, len);
2325 ARY_SET_CAPA(ary, new_capa);
2326 }
2327 ARY_SET_HEAP_LEN(ary, len);
2328 }
2329 ary_verify(ary);
2330 return ary;
2331}
2332
2333static VALUE
2334ary_aset_by_rb_ary_store(VALUE ary, long key, VALUE val)
2335{
2336 rb_ary_store(ary, key, val);
2337 return val;
2338}
2339
2340static VALUE
2341ary_aset_by_rb_ary_splice(VALUE ary, long beg, long len, VALUE val)
2342{
2343 VALUE rpl = rb_ary_to_ary(val);
2344 rb_ary_splice(ary, beg, len, RARRAY_CONST_PTR(rpl), RARRAY_LEN(rpl));
2345 RB_GC_GUARD(rpl);
2346 return val;
2347}
2348
2349/*
2350 * call-seq:
2351 * self[index] = object -> object
2352 * self[start, length] = object -> object
2353 * self[range] = object -> object
2354 *
2355 * Assigns elements in +self+, based on the given +object+; returns +object+.
2356 *
2357 * In brief:
2358 *
2359 * a_orig = [:foo, 'bar', 2]
2360 *
2361 * # With argument index.
2362 * a = a_orig.dup
2363 * a[0] = 'foo' # => "foo"
2364 * a # => ["foo", "bar", 2]
2365 * a = a_orig.dup
2366 * a[7] = 'foo' # => "foo"
2367 * a # => [:foo, "bar", 2, nil, nil, nil, nil, "foo"]
2368 *
2369 * # With arguments start and length.
2370 * a = a_orig.dup
2371 * a[0, 2] = 'foo' # => "foo"
2372 * a # => ["foo", 2]
2373 * a = a_orig.dup
2374 * a[6, 50] = 'foo' # => "foo"
2375 * a # => [:foo, "bar", 2, nil, nil, nil, "foo"]
2376 *
2377 * # With argument range.
2378 * a = a_orig.dup
2379 * a[0..1] = 'foo' # => "foo"
2380 * a # => ["foo", 2]
2381 * a = a_orig.dup
2382 * a[6..50] = 'foo' # => "foo"
2383 * a # => [:foo, "bar", 2, nil, nil, nil, "foo"]
2384 *
2385 * When Integer argument +index+ is given, assigns +object+ to an element in +self+.
2386 *
2387 * If +index+ is non-negative, assigns +object+ the element at offset +index+:
2388 *
2389 * a = [:foo, 'bar', 2]
2390 * a[0] = 'foo' # => "foo"
2391 * a # => ["foo", "bar", 2]
2392 *
2393 * If +index+ is greater than <tt>self.length</tt>, extends the array:
2394 *
2395 * a = [:foo, 'bar', 2]
2396 * a[7] = 'foo' # => "foo"
2397 * a # => [:foo, "bar", 2, nil, nil, nil, nil, "foo"]
2398 *
2399 * If +index+ is negative, counts backwards from the end of the array:
2400 *
2401 * a = [:foo, 'bar', 2]
2402 * a[-1] = 'two' # => "two"
2403 * a # => [:foo, "bar", "two"]
2404 *
2405 * When Integer arguments +start+ and +length+ are given and +object+ is not an array,
2406 * removes <tt>length - 1</tt> elements beginning at offset +start+,
2407 * and assigns +object+ at offset +start+:
2408 *
2409 * a = [:foo, 'bar', 2]
2410 * a[0, 2] = 'foo' # => "foo"
2411 * a # => ["foo", 2]
2412 *
2413 * If +start+ is negative, counts backwards from the end of the array:
2414 *
2415 * a = [:foo, 'bar', 2]
2416 * a[-2, 2] = 'foo' # => "foo"
2417 * a # => [:foo, "foo"]
2418 *
2419 * If +start+ is non-negative and outside the array (<tt> >= self.size</tt>),
2420 * extends the array with +nil+, assigns +object+ at offset +start+,
2421 * and ignores +length+:
2422 *
2423 * a = [:foo, 'bar', 2]
2424 * a[6, 50] = 'foo' # => "foo"
2425 * a # => [:foo, "bar", 2, nil, nil, nil, "foo"]
2426 *
2427 * If +length+ is zero, shifts elements at and following offset +start+
2428 * and assigns +object+ at offset +start+:
2429 *
2430 * a = [:foo, 'bar', 2]
2431 * a[1, 0] = 'foo' # => "foo"
2432 * a # => [:foo, "foo", "bar", 2]
2433 *
2434 * If +length+ is too large for the existing array, does not extend the array:
2435 *
2436 * a = [:foo, 'bar', 2]
2437 * a[1, 5] = 'foo' # => "foo"
2438 * a # => [:foo, "foo"]
2439 *
2440 * When Range argument +range+ is given and +object+ is not an array,
2441 * removes <tt>length - 1</tt> elements beginning at offset +start+,
2442 * and assigns +object+ at offset +start+:
2443 *
2444 * a = [:foo, 'bar', 2]
2445 * a[0..1] = 'foo' # => "foo"
2446 * a # => ["foo", 2]
2447 *
2448 * if <tt>range.begin</tt> is negative, counts backwards from the end of the array:
2449 *
2450 * a = [:foo, 'bar', 2]
2451 * a[-2..2] = 'foo' # => "foo"
2452 * a # => [:foo, "foo"]
2453 *
2454 * If the array length is less than <tt>range.begin</tt>,
2455 * extends the array with +nil+, assigns +object+ at offset <tt>range.begin</tt>,
2456 * and ignores +length+:
2457 *
2458 * a = [:foo, 'bar', 2]
2459 * a[6..50] = 'foo' # => "foo"
2460 * a # => [:foo, "bar", 2, nil, nil, nil, "foo"]
2461 *
2462 * If <tt>range.end</tt> is zero, shifts elements at and following offset +start+
2463 * and assigns +object+ at offset +start+:
2464 *
2465 * a = [:foo, 'bar', 2]
2466 * a[1..0] = 'foo' # => "foo"
2467 * a # => [:foo, "foo", "bar", 2]
2468 *
2469 * If <tt>range.end</tt> is negative, assigns +object+ at offset +start+,
2470 * retains <tt>range.end.abs -1</tt> elements past that, and removes those beyond:
2471 *
2472 * a = [:foo, 'bar', 2]
2473 * a[1..-1] = 'foo' # => "foo"
2474 * a # => [:foo, "foo"]
2475 * a = [:foo, 'bar', 2]
2476 * a[1..-2] = 'foo' # => "foo"
2477 * a # => [:foo, "foo", 2]
2478 * a = [:foo, 'bar', 2]
2479 * a[1..-3] = 'foo' # => "foo"
2480 * a # => [:foo, "foo", "bar", 2]
2481 * a = [:foo, 'bar', 2]
2482 *
2483 * If <tt>range.end</tt> is too large for the existing array,
2484 * replaces array elements, but does not extend the array with +nil+ values:
2485 *
2486 * a = [:foo, 'bar', 2]
2487 * a[1..5] = 'foo' # => "foo"
2488 * a # => [:foo, "foo"]
2489 *
2490 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
2491 */
2492
2493static VALUE
2494rb_ary_aset(int argc, VALUE *argv, VALUE ary)
2495{
2496 long offset, beg, len;
2497
2498 rb_check_arity(argc, 2, 3);
2499 rb_ary_modify_check(ary);
2500 if (argc == 3) {
2501 beg = NUM2LONG(argv[0]);
2502 len = NUM2LONG(argv[1]);
2503 return ary_aset_by_rb_ary_splice(ary, beg, len, argv[2]);
2504 }
2505 if (FIXNUM_P(argv[0])) {
2506 offset = FIX2LONG(argv[0]);
2507 return ary_aset_by_rb_ary_store(ary, offset, argv[1]);
2508 }
2509 if (rb_range_beg_len(argv[0], &beg, &len, RARRAY_LEN(ary), 1)) {
2510 /* check if idx is Range */
2511 return ary_aset_by_rb_ary_splice(ary, beg, len, argv[1]);
2512 }
2513
2514 offset = NUM2LONG(argv[0]);
2515 return ary_aset_by_rb_ary_store(ary, offset, argv[1]);
2516}
2517
2518/*
2519 * call-seq:
2520 * insert(index, *objects) -> self
2521 *
2522 * Inserts the given +objects+ as elements of +self+;
2523 * returns +self+.
2524 *
2525 * When +index+ is non-negative, inserts +objects+
2526 * _before_ the element at offset +index+:
2527 *
2528 * a = ['a', 'b', 'c'] # => ["a", "b", "c"]
2529 * a.insert(1, :x, :y, :z) # => ["a", :x, :y, :z, "b", "c"]
2530 *
2531 * Extends the array if +index+ is beyond the array (<tt>index >= self.size</tt>):
2532 *
2533 * a = ['a', 'b', 'c'] # => ["a", "b", "c"]
2534 * a.insert(5, :x, :y, :z) # => ["a", "b", "c", nil, nil, :x, :y, :z]
2535 *
2536 * When +index+ is negative, inserts +objects+
2537 * _after_ the element at offset <tt>index + self.size</tt>:
2538 *
2539 * a = ['a', 'b', 'c'] # => ["a", "b", "c"]
2540 * a.insert(-2, :x, :y, :z) # => ["a", "b", :x, :y, :z, "c"]
2541 *
2542 * With no +objects+ given, does nothing:
2543 *
2544 * a = ['a', 'b', 'c'] # => ["a", "b", "c"]
2545 * a.insert(1) # => ["a", "b", "c"]
2546 * a.insert(50) # => ["a", "b", "c"]
2547 * a.insert(-50) # => ["a", "b", "c"]
2548 *
2549 * Raises IndexError if +objects+ are given and +index+ is negative and out of range.
2550 *
2551 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
2552 */
2553
2554static VALUE
2555rb_ary_insert(int argc, VALUE *argv, VALUE ary)
2556{
2557 long pos;
2558
2560 rb_ary_modify_check(ary);
2561 pos = NUM2LONG(argv[0]);
2562 if (argc == 1) return ary;
2563 if (pos == -1) {
2564 pos = RARRAY_LEN(ary);
2565 }
2566 else if (pos < 0) {
2567 long minpos = -RARRAY_LEN(ary) - 1;
2568 if (pos < minpos) {
2569 rb_raise(rb_eIndexError, "index %ld too small for array; minimum: %ld",
2570 pos, minpos);
2571 }
2572 pos++;
2573 }
2574 rb_ary_splice(ary, pos, 0, argv + 1, argc - 1);
2575 return ary;
2576}
2577
2578static VALUE
2579rb_ary_length(VALUE ary);
2580
2581static VALUE
2582ary_enum_length(VALUE ary, VALUE args, VALUE eobj)
2583{
2584 return rb_ary_length(ary);
2585}
2586
2587// Primitive to avoid a race condition in Array#each.
2588// Return `true` and write `value` and `index` if the element exists.
2589static VALUE
2590ary_fetch_next(VALUE self, VALUE *index, VALUE *value)
2591{
2592 long i = NUM2LONG(*index);
2593 if (i >= RARRAY_LEN(self)) {
2594 return Qfalse;
2595 }
2596 *value = RARRAY_AREF(self, i);
2597 *index = LONG2NUM(i + 1);
2598 return Qtrue;
2599}
2600
2601/*
2602 * call-seq:
2603 * each {|element| ... } -> self
2604 * each -> new_enumerator
2605 *
2606 * With a block given, iterates over the elements of +self+,
2607 * passing each element to the block;
2608 * returns +self+:
2609 *
2610 * a = [:foo, 'bar', 2]
2611 * a.each {|element| puts "#{element.class} #{element}" }
2612 *
2613 * Output:
2614 *
2615 * Symbol foo
2616 * String bar
2617 * Integer 2
2618 *
2619 * Allows the array to be modified during iteration:
2620 *
2621 * a = [:foo, 'bar', 2]
2622 * a.each {|element| puts element; a.clear if element.to_s.start_with?('b') }
2623 *
2624 * Output:
2625 *
2626 * foo
2627 * bar
2628 *
2629 * With no block given, returns a new Enumerator.
2630 *
2631 * Related: see {Methods for Iterating}[rdoc-ref:Array@Methods+for+Iterating].
2632 */
2633
2634VALUE
2635rb_ary_each(VALUE ary)
2636{
2637 long i;
2638 ary_verify(ary);
2639 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
2640 for (i=0; i<RARRAY_LEN(ary); i++) {
2642 }
2643 return ary;
2644}
2645
2646/*
2647 * call-seq:
2648 * each_index {|index| ... } -> self
2649 * each_index -> new_enumerator
2650 *
2651 * With a block given, iterates over the elements of +self+,
2652 * passing each <i>array index</i> to the block;
2653 * returns +self+:
2654 *
2655 * a = [:foo, 'bar', 2]
2656 * a.each_index {|index| puts "#{index} #{a[index]}" }
2657 *
2658 * Output:
2659 *
2660 * 0 foo
2661 * 1 bar
2662 * 2 2
2663 *
2664 * Allows the array to be modified during iteration:
2665 *
2666 * a = [:foo, 'bar', 2]
2667 * a.each_index {|index| puts index; a.clear if index > 0 }
2668 * a # => []
2669 *
2670 * Output:
2671 *
2672 * 0
2673 * 1
2674 *
2675 * With no block given, returns a new Enumerator.
2676 *
2677 * Related: see {Methods for Iterating}[rdoc-ref:Array@Methods+for+Iterating].
2678 */
2679
2680static VALUE
2681rb_ary_each_index(VALUE ary)
2682{
2683 long i;
2684 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
2685
2686 for (i=0; i<RARRAY_LEN(ary); i++) {
2687 rb_yield(LONG2NUM(i));
2688 }
2689 return ary;
2690}
2691
2692/*
2693 * call-seq:
2694 * reverse_each {|element| ... } -> self
2695 * reverse_each -> Enumerator
2696 *
2697 * When a block given, iterates backwards over the elements of +self+,
2698 * passing, in reverse order, each element to the block;
2699 * returns +self+:
2700 *
2701 * a = []
2702 * [0, 1, 2].reverse_each {|element| a.push(element) }
2703 * a # => [2, 1, 0]
2704 *
2705 * Allows the array to be modified during iteration:
2706 *
2707 * a = ['a', 'b', 'c']
2708 * a.reverse_each {|element| a.clear if element.start_with?('b') }
2709 * a # => []
2710 *
2711 * When no block given, returns a new Enumerator.
2712 *
2713 * Related: see {Methods for Iterating}[rdoc-ref:Array@Methods+for+Iterating].
2714 */
2715
2716static VALUE
2717rb_ary_reverse_each(VALUE ary)
2718{
2719 long len;
2720
2721 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
2722 len = RARRAY_LEN(ary);
2723 while (len--) {
2724 long nlen;
2726 nlen = RARRAY_LEN(ary);
2727 if (nlen < len) {
2728 len = nlen;
2729 }
2730 }
2731 return ary;
2732}
2733
2734/*
2735 * call-seq:
2736 * length -> integer
2737 * size -> integer
2738 *
2739 * Returns the count of elements in +self+:
2740 *
2741 * [0, 1, 2].length # => 3
2742 * [].length # => 0
2743 *
2744 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
2745 */
2746
2747static VALUE
2748rb_ary_length(VALUE ary)
2749{
2750 long len = RARRAY_LEN(ary);
2751 return LONG2NUM(len);
2752}
2753
2754/*
2755 * call-seq:
2756 * empty? -> true or false
2757 *
2758 * Returns +true+ if the count of elements in +self+ is zero,
2759 * +false+ otherwise.
2760 *
2761 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
2762 */
2763
2764static VALUE
2765rb_ary_empty_p(VALUE ary)
2766{
2767 return RBOOL(RARRAY_LEN(ary) == 0);
2768}
2769
2770VALUE
2771rb_ary_dup(VALUE ary)
2772{
2773 long len = RARRAY_LEN(ary);
2774 VALUE dup = rb_ary_new2(len);
2775 ary_memcpy(dup, 0, len, RARRAY_CONST_PTR(ary));
2776 ARY_SET_LEN(dup, len);
2777
2778 ary_verify(ary);
2779 ary_verify(dup);
2780 return dup;
2781}
2782
2783VALUE
2784rb_ary_resurrect(VALUE ary)
2785{
2786 return ary_make_partial(ary, rb_cArray, 0, RARRAY_LEN(ary));
2787}
2788
2789extern VALUE rb_output_fs;
2790
2791static void ary_join_1(VALUE obj, VALUE ary, VALUE sep, long i, VALUE result, int *first);
2792
2793static VALUE
2794recursive_join(VALUE obj, VALUE argp, int recur)
2795{
2796 VALUE *arg = (VALUE *)argp;
2797 VALUE ary = arg[0];
2798 VALUE sep = arg[1];
2799 VALUE result = arg[2];
2800 int *first = (int *)arg[3];
2801
2802 if (recur) {
2803 rb_raise(rb_eArgError, "recursive array join");
2804 }
2805 else {
2806 ary_join_1(obj, ary, sep, 0, result, first);
2807 }
2808 return Qnil;
2809}
2810
2811static long
2812ary_join_0(VALUE ary, VALUE sep, long max, VALUE result)
2813{
2814 long i;
2815 VALUE val;
2816
2817 if (max > 0) rb_enc_copy(result, RARRAY_AREF(ary, 0));
2818 for (i=0; i<max; i++) {
2819 val = RARRAY_AREF(ary, i);
2820 if (!RB_TYPE_P(val, T_STRING)) break;
2821 if (i > 0 && !NIL_P(sep))
2822 rb_str_buf_append(result, sep);
2823 rb_str_buf_append(result, val);
2824 }
2825 return i;
2826}
2827
2828static void
2829ary_join_1_str(VALUE dst, VALUE src, int *first)
2830{
2831 rb_str_buf_append(dst, src);
2832 if (*first) {
2833 rb_enc_copy(dst, src);
2834 *first = FALSE;
2835 }
2836}
2837
2838static void
2839ary_join_1_ary(VALUE obj, VALUE ary, VALUE sep, VALUE result, VALUE val, int *first)
2840{
2841 if (val == ary) {
2842 rb_raise(rb_eArgError, "recursive array join");
2843 }
2844 else {
2845 VALUE args[4];
2846
2847 *first = FALSE;
2848 args[0] = val;
2849 args[1] = sep;
2850 args[2] = result;
2851 args[3] = (VALUE)first;
2852 rb_exec_recursive(recursive_join, obj, (VALUE)args);
2853 }
2854}
2855
2856static void
2857ary_join_1(VALUE obj, VALUE ary, VALUE sep, long i, VALUE result, int *first)
2858{
2859 VALUE val, tmp;
2860
2861 for (; i<RARRAY_LEN(ary); i++) {
2862 if (i > 0 && !NIL_P(sep))
2863 rb_str_buf_append(result, sep);
2864
2865 val = RARRAY_AREF(ary, i);
2866 if (RB_TYPE_P(val, T_STRING)) {
2867 ary_join_1_str(result, val, first);
2868 }
2869 else if (RB_TYPE_P(val, T_ARRAY)) {
2870 ary_join_1_ary(val, ary, sep, result, val, first);
2871 }
2872 else if (!NIL_P(tmp = rb_check_string_type(val))) {
2873 ary_join_1_str(result, tmp, first);
2874 }
2875 else if (!NIL_P(tmp = rb_check_array_type(val))) {
2876 ary_join_1_ary(val, ary, sep, result, tmp, first);
2877 }
2878 else {
2879 ary_join_1_str(result, rb_obj_as_string(val), first);
2880 }
2881 }
2882}
2883
2884VALUE
2885rb_ary_join(VALUE ary, VALUE sep)
2886{
2887 long len = 1, i;
2888 VALUE val, tmp, result;
2889
2890 if (RARRAY_LEN(ary) == 0) return rb_usascii_str_new(0, 0);
2891
2892 if (!NIL_P(sep)) {
2893 StringValue(sep);
2894 len += RSTRING_LEN(sep) * (RARRAY_LEN(ary) - 1);
2895 }
2896 for (i=0; i<RARRAY_LEN(ary); i++) {
2897 val = RARRAY_AREF(ary, i);
2898 tmp = rb_check_string_type(val);
2899
2900 if (NIL_P(tmp) || tmp != val) {
2901 int first;
2902 long n = RARRAY_LEN(ary);
2903 if (i > n) i = n;
2904 result = rb_str_buf_new(len + (n-i)*10);
2905 rb_enc_associate(result, rb_usascii_encoding());
2906 i = ary_join_0(ary, sep, i, result);
2907 first = i == 0;
2908 ary_join_1(ary, ary, sep, i, result, &first);
2909 return result;
2910 }
2911
2912 len += RSTRING_LEN(tmp);
2913 }
2914
2915 result = rb_str_new(0, len);
2916 rb_str_set_len(result, 0);
2917
2918 ary_join_0(ary, sep, RARRAY_LEN(ary), result);
2919
2920 return result;
2921}
2922
2923/*
2924 * call-seq:
2925 * join(separator = $,) -> new_string
2926 *
2927 * Returns the new string formed by joining the converted elements of +self+;
2928 * for each element +element+:
2929 *
2930 * - Converts recursively using <tt>element.join(separator)</tt>
2931 * if +element+ is a <tt>kind_of?(Array)</tt>.
2932 * - Otherwise, converts using <tt>element.to_s</tt>.
2933 *
2934 * With no argument given, joins using the output field separator, <tt>$,</tt>:
2935 *
2936 * a = [:foo, 'bar', 2]
2937 * $, # => nil
2938 * a.join # => "foobar2"
2939 *
2940 * With string argument +separator+ given, joins using that separator:
2941 *
2942 * a = [:foo, 'bar', 2]
2943 * a.join("\n") # => "foo\nbar\n2"
2944 *
2945 * Joins recursively for nested arrays:
2946 *
2947 * a = [:foo, [:bar, [:baz, :bat]]]
2948 * a.join # => "foobarbazbat"
2949 *
2950 * Related: see {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
2951 */
2952static VALUE
2953rb_ary_join_m(int argc, VALUE *argv, VALUE ary)
2954{
2955 VALUE sep;
2956
2957 if (rb_check_arity(argc, 0, 1) == 0 || NIL_P(sep = argv[0])) {
2958 sep = rb_output_fs;
2959 if (!NIL_P(sep)) {
2960 rb_category_warn(RB_WARN_CATEGORY_DEPRECATED, "$, is set to non-nil value");
2961 }
2962 }
2963
2964 return rb_ary_join(ary, sep);
2965}
2966
2967static VALUE
2968inspect_ary(VALUE ary, VALUE dummy, int recur)
2969{
2970 long i;
2971 VALUE s, str;
2972
2973 if (recur) return rb_usascii_str_new_cstr("[...]");
2974 str = rb_str_buf_new2("[");
2975 for (i=0; i<RARRAY_LEN(ary); i++) {
2976 s = rb_inspect(RARRAY_AREF(ary, i));
2977 if (i > 0) rb_str_buf_cat2(str, ", ");
2978 else rb_enc_copy(str, s);
2979 rb_str_buf_append(str, s);
2980 }
2981 rb_str_buf_cat2(str, "]");
2982 return str;
2983}
2984
2985/*
2986 * call-seq:
2987 * inspect -> new_string
2988 * to_s -> new_string
2989 *
2990 * Returns the new string formed by calling method <tt>#inspect</tt>
2991 * on each array element:
2992 *
2993 * a = [:foo, 'bar', 2]
2994 * a.inspect # => "[:foo, \"bar\", 2]"
2995 *
2996 * Related: see {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
2997 */
2998
2999static VALUE
3000rb_ary_inspect(VALUE ary)
3001{
3002 if (RARRAY_LEN(ary) == 0) return rb_usascii_str_new2("[]");
3003 return rb_exec_recursive(inspect_ary, ary, 0);
3004}
3005
3006VALUE
3007rb_ary_to_s(VALUE ary)
3008{
3009 return rb_ary_inspect(ary);
3010}
3011
3012/*
3013 * call-seq:
3014 * to_a -> self or new_array
3015 *
3016 * When +self+ is an instance of \Array, returns +self+.
3017 *
3018 * Otherwise, returns a new array containing the elements of +self+:
3019 *
3020 * class MyArray < Array; end
3021 * my_a = MyArray.new(['foo', 'bar', 'two'])
3022 * a = my_a.to_a
3023 * a # => ["foo", "bar", "two"]
3024 * a.class # => Array # Not MyArray.
3025 *
3026 * Related: see {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
3027 */
3028
3029static VALUE
3030rb_ary_to_a(VALUE ary)
3031{
3032 if (rb_obj_class(ary) != rb_cArray) {
3034 rb_ary_replace(dup, ary);
3035 return dup;
3036 }
3037 return ary;
3038}
3039
3040/*
3041 * call-seq:
3042 * to_h -> new_hash
3043 * to_h {|element| ... } -> new_hash
3044 *
3045 * Returns a new hash formed from +self+.
3046 *
3047 * With no block given, each element of +self+ must be a 2-element sub-array;
3048 * forms each sub-array into a key-value pair in the new hash:
3049 *
3050 * a = [['foo', 'zero'], ['bar', 'one'], ['baz', 'two']]
3051 * a.to_h # => {"foo"=>"zero", "bar"=>"one", "baz"=>"two"}
3052 * [].to_h # => {}
3053 *
3054 * With a block given, the block must return a 2-element array;
3055 * calls the block with each element of +self+;
3056 * forms each returned array into a key-value pair in the returned hash:
3057 *
3058 * a = ['foo', :bar, 1, [2, 3], {baz: 4}]
3059 * a.to_h {|element| [element, element.class] }
3060 * # => {"foo"=>String, :bar=>Symbol, 1=>Integer, [2, 3]=>Array, {:baz=>4}=>Hash}
3061 *
3062 * Related: see {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
3063 */
3064
3065static VALUE
3066rb_ary_to_h(VALUE ary)
3067{
3068 long i;
3069 VALUE hash = rb_hash_new_with_size(RARRAY_LEN(ary));
3070 int block_given = rb_block_given_p();
3071
3072 for (i=0; i<RARRAY_LEN(ary); i++) {
3073 const VALUE e = rb_ary_elt(ary, i);
3074 const VALUE elt = block_given ? rb_yield_force_blockarg(e) : e;
3075 const VALUE key_value_pair = rb_check_array_type(elt);
3076 if (NIL_P(key_value_pair)) {
3077 rb_raise(rb_eTypeError, "wrong element type %"PRIsVALUE" at %ld (expected array)",
3078 rb_obj_class(elt), i);
3079 }
3080 if (RARRAY_LEN(key_value_pair) != 2) {
3081 rb_raise(rb_eArgError, "wrong array length at %ld (expected 2, was %ld)",
3082 i, RARRAY_LEN(key_value_pair));
3083 }
3084 rb_hash_aset(hash, RARRAY_AREF(key_value_pair, 0), RARRAY_AREF(key_value_pair, 1));
3085 }
3086 return hash;
3087}
3088
3089/*
3090 * call-seq:
3091 * to_ary -> self
3092 *
3093 * Returns +self+.
3094 */
3095
3096static VALUE
3097rb_ary_to_ary_m(VALUE ary)
3098{
3099 return ary;
3100}
3101
3102static void
3103ary_reverse(VALUE *p1, VALUE *p2)
3104{
3105 while (p1 < p2) {
3106 VALUE tmp = *p1;
3107 *p1++ = *p2;
3108 *p2-- = tmp;
3109 }
3110}
3111
3112VALUE
3113rb_ary_reverse(VALUE ary)
3114{
3115 VALUE *p2;
3116 long len = RARRAY_LEN(ary);
3117
3118 rb_ary_modify(ary);
3119 if (len > 1) {
3120 RARRAY_PTR_USE(ary, p1, {
3121 p2 = p1 + len - 1; /* points last item */
3122 ary_reverse(p1, p2);
3123 }); /* WB: no new reference */
3124 }
3125 return ary;
3126}
3127
3128/*
3129 * call-seq:
3130 * reverse! -> self
3131 *
3132 * Reverses the order of the elements of +self+;
3133 * returns +self+:
3134 *
3135 * a = [0, 1, 2]
3136 * a.reverse! # => [2, 1, 0]
3137 * a # => [2, 1, 0]
3138 *
3139 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
3140 */
3141
3142static VALUE
3143rb_ary_reverse_bang(VALUE ary)
3144{
3145 return rb_ary_reverse(ary);
3146}
3147
3148/*
3149 * call-seq:
3150 * reverse -> new_array
3151 *
3152 * Returns a new array containing the elements of +self+ in reverse order:
3153 *
3154 * [0, 1, 2].reverse # => [2, 1, 0]
3155 *
3156 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
3157 */
3158
3159static VALUE
3160rb_ary_reverse_m(VALUE ary)
3161{
3162 long len = RARRAY_LEN(ary);
3163 VALUE dup = rb_ary_new2(len);
3164
3165 if (len > 0) {
3166 const VALUE *p1 = RARRAY_CONST_PTR(ary);
3167 VALUE *p2 = (VALUE *)RARRAY_CONST_PTR(dup) + len - 1;
3168 do *p2-- = *p1++; while (--len > 0);
3169 }
3170 ARY_SET_LEN(dup, RARRAY_LEN(ary));
3171 return dup;
3172}
3173
3174static inline long
3175rotate_count(long cnt, long len)
3176{
3177 return (cnt < 0) ? (len - (~cnt % len) - 1) : (cnt % len);
3178}
3179
3180static void
3181ary_rotate_ptr(VALUE *ptr, long len, long cnt)
3182{
3183 if (cnt == 1) {
3184 VALUE tmp = *ptr;
3185 memmove(ptr, ptr + 1, sizeof(VALUE)*(len - 1));
3186 *(ptr + len - 1) = tmp;
3187 }
3188 else if (cnt == len - 1) {
3189 VALUE tmp = *(ptr + len - 1);
3190 memmove(ptr + 1, ptr, sizeof(VALUE)*(len - 1));
3191 *ptr = tmp;
3192 }
3193 else {
3194 --len;
3195 if (cnt < len) ary_reverse(ptr + cnt, ptr + len);
3196 if (--cnt > 0) ary_reverse(ptr, ptr + cnt);
3197 if (len > 0) ary_reverse(ptr, ptr + len);
3198 }
3199}
3200
3201VALUE
3202rb_ary_rotate(VALUE ary, long cnt)
3203{
3204 rb_ary_modify(ary);
3205
3206 if (cnt != 0) {
3207 long len = RARRAY_LEN(ary);
3208 if (len > 1 && (cnt = rotate_count(cnt, len)) > 0) {
3209 RARRAY_PTR_USE(ary, ptr, ary_rotate_ptr(ptr, len, cnt));
3210 return ary;
3211 }
3212 }
3213 return Qnil;
3214}
3215
3216/*
3217 * call-seq:
3218 * rotate!(count = 1) -> self
3219 *
3220 * Rotates +self+ in place by moving elements from one end to the other; returns +self+.
3221 *
3222 * With non-negative numeric +count+,
3223 * rotates +count+ elements from the beginning to the end:
3224 *
3225 * [0, 1, 2, 3].rotate!(2) # => [2, 3, 0, 1]
3226 [0, 1, 2, 3].rotate!(2.1) # => [2, 3, 0, 1]
3227 *
3228 * If +count+ is large, uses <tt>count % array.size</tt> as the count:
3229 *
3230 * [0, 1, 2, 3].rotate!(21) # => [1, 2, 3, 0]
3231 *
3232 * If +count+ is zero, rotates no elements:
3233 *
3234 * [0, 1, 2, 3].rotate!(0) # => [0, 1, 2, 3]
3235 *
3236 * With a negative numeric +count+, rotates in the opposite direction,
3237 * from end to beginning:
3238 *
3239 * [0, 1, 2, 3].rotate!(-1) # => [3, 0, 1, 2]
3240 *
3241 * If +count+ is small (far from zero), uses <tt>count % array.size</tt> as the count:
3242 *
3243 * [0, 1, 2, 3].rotate!(-21) # => [3, 0, 1, 2]
3244 *
3245 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
3246 */
3247
3248static VALUE
3249rb_ary_rotate_bang(int argc, VALUE *argv, VALUE ary)
3250{
3251 long n = (rb_check_arity(argc, 0, 1) ? NUM2LONG(argv[0]) : 1);
3252 rb_ary_rotate(ary, n);
3253 return ary;
3254}
3255
3256/*
3257 * call-seq:
3258 * rotate(count = 1) -> new_array
3259 *
3260 * Returns a new array formed from +self+ with elements
3261 * rotated from one end to the other.
3262 *
3263 * With non-negative numeric +count+,
3264 * rotates elements from the beginning to the end:
3265 *
3266 * [0, 1, 2, 3].rotate(2) # => [2, 3, 0, 1]
3267 * [0, 1, 2, 3].rotate(2.1) # => [2, 3, 0, 1]
3268 *
3269 * If +count+ is large, uses <tt>count % array.size</tt> as the count:
3270 *
3271 * [0, 1, 2, 3].rotate(22) # => [2, 3, 0, 1]
3272 *
3273 * With a +count+ of zero, rotates no elements:
3274 *
3275 * [0, 1, 2, 3].rotate(0) # => [0, 1, 2, 3]
3276 *
3277 * With negative numeric +count+, rotates in the opposite direction,
3278 * from the end to the beginning:
3279 *
3280 * [0, 1, 2, 3].rotate(-1) # => [3, 0, 1, 2]
3281 *
3282 * If +count+ is small (far from zero), uses <tt>count % array.size</tt> as the count:
3283 *
3284 * [0, 1, 2, 3].rotate(-21) # => [3, 0, 1, 2]
3285 *
3286 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
3287 */
3288
3289static VALUE
3290rb_ary_rotate_m(int argc, VALUE *argv, VALUE ary)
3291{
3292 VALUE rotated;
3293 const VALUE *ptr;
3294 long len;
3295 long cnt = (rb_check_arity(argc, 0, 1) ? NUM2LONG(argv[0]) : 1);
3296
3297 len = RARRAY_LEN(ary);
3298 rotated = rb_ary_new2(len);
3299 if (len > 0) {
3300 cnt = rotate_count(cnt, len);
3302 len -= cnt;
3303 ary_memcpy(rotated, 0, len, ptr + cnt);
3304 ary_memcpy(rotated, len, cnt, ptr);
3305 }
3306 ARY_SET_LEN(rotated, RARRAY_LEN(ary));
3307 return rotated;
3308}
3309
3311 VALUE ary;
3312 VALUE receiver;
3313};
3314
3315static VALUE
3316sort_reentered(VALUE ary)
3317{
3318 if (RBASIC(ary)->klass) {
3319 rb_raise(rb_eRuntimeError, "sort reentered");
3320 }
3321 return Qnil;
3322}
3323
3324static void
3325sort_returned(struct ary_sort_data *data)
3326{
3327 if (rb_obj_frozen_p(data->receiver)) {
3328 rb_raise(rb_eFrozenError, "array frozen during sort");
3329 }
3330 sort_reentered(data->ary);
3331}
3332
3333static int
3334sort_1(const void *ap, const void *bp, void *dummy)
3335{
3336 struct ary_sort_data *data = dummy;
3337 VALUE retval = sort_reentered(data->ary);
3338 VALUE a = *(const VALUE *)ap, b = *(const VALUE *)bp;
3339 VALUE args[2];
3340 int n;
3341
3342 args[0] = a;
3343 args[1] = b;
3344 retval = rb_yield_values2(2, args);
3345 n = rb_cmpint(retval, a, b);
3346 sort_returned(data);
3347 return n;
3348}
3349
3350static int
3351sort_2(const void *ap, const void *bp, void *dummy)
3352{
3353 struct ary_sort_data *data = dummy;
3354 VALUE retval = sort_reentered(data->ary);
3355 VALUE a = *(const VALUE *)ap, b = *(const VALUE *)bp;
3356 int n;
3357
3358 if (FIXNUM_P(a) && FIXNUM_P(b) && CMP_OPTIMIZABLE(INTEGER)) {
3359 if ((long)a > (long)b) return 1;
3360 if ((long)a < (long)b) return -1;
3361 return 0;
3362 }
3363 if (STRING_P(a) && STRING_P(b) && CMP_OPTIMIZABLE(STRING)) {
3364 return rb_str_cmp(a, b);
3365 }
3366 if (RB_FLOAT_TYPE_P(a) && CMP_OPTIMIZABLE(FLOAT)) {
3367 return rb_float_cmp(a, b);
3368 }
3369
3370 retval = rb_funcallv(a, id_cmp, 1, &b);
3371 n = rb_cmpint(retval, a, b);
3372 sort_returned(data);
3373
3374 return n;
3375}
3376
3377/*
3378 * call-seq:
3379 * sort! -> self
3380 * sort! {|a, b| ... } -> self
3381 *
3382 * Like Array#sort, but returns +self+ with its elements sorted in place.
3383 *
3384 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
3385 */
3386
3387VALUE
3388rb_ary_sort_bang(VALUE ary)
3389{
3390 rb_ary_modify(ary);
3391 RUBY_ASSERT(!ARY_SHARED_P(ary));
3392 if (RARRAY_LEN(ary) > 1) {
3393 VALUE tmp = ary_make_substitution(ary); /* only ary refers tmp */
3394 struct ary_sort_data data;
3395 long len = RARRAY_LEN(ary);
3396 RBASIC_CLEAR_CLASS(tmp);
3397 data.ary = tmp;
3398 data.receiver = ary;
3399 RARRAY_PTR_USE(tmp, ptr, {
3400 ruby_qsort(ptr, len, sizeof(VALUE),
3401 rb_block_given_p()?sort_1:sort_2, &data);
3402 }); /* WB: no new reference */
3403 rb_ary_modify(ary);
3404 if (ARY_EMBED_P(tmp)) {
3405 if (ARY_SHARED_P(ary)) { /* ary might be destructively operated in the given block */
3406 rb_ary_unshare(ary);
3407 FL_SET_EMBED(ary);
3408 }
3409 if (ARY_EMBED_LEN(tmp) > ARY_CAPA(ary)) {
3410 ary_resize_capa(ary, ARY_EMBED_LEN(tmp));
3411 }
3412 ary_memcpy(ary, 0, ARY_EMBED_LEN(tmp), ARY_EMBED_PTR(tmp));
3413 ARY_SET_LEN(ary, ARY_EMBED_LEN(tmp));
3414 }
3415 else {
3416 if (!ARY_EMBED_P(ary) && ARY_HEAP_PTR(ary) == ARY_HEAP_PTR(tmp)) {
3417 FL_UNSET_SHARED(ary);
3418 ARY_SET_CAPA(ary, RARRAY_LEN(tmp));
3419 }
3420 else {
3421 RUBY_ASSERT(!ARY_SHARED_P(tmp));
3422 if (ARY_EMBED_P(ary)) {
3423 FL_UNSET_EMBED(ary);
3424 }
3425 else if (ARY_SHARED_P(ary)) {
3426 /* ary might be destructively operated in the given block */
3427 rb_ary_unshare(ary);
3428 }
3429 else {
3430 ary_heap_free(ary);
3431 }
3432 ARY_SET_PTR(ary, ARY_HEAP_PTR(tmp));
3433 ARY_SET_HEAP_LEN(ary, len);
3434 ARY_SET_CAPA(ary, ARY_HEAP_LEN(tmp));
3435 }
3436 /* tmp was lost ownership for the ptr */
3437 FL_UNSET(tmp, FL_FREEZE);
3438 FL_SET_EMBED(tmp);
3439 ARY_SET_EMBED_LEN(tmp, 0);
3440 FL_SET(tmp, FL_FREEZE);
3441 }
3442 /* tmp will be GC'ed. */
3443 RBASIC_SET_CLASS_RAW(tmp, rb_cArray); /* rb_cArray must be marked */
3444 }
3445 ary_verify(ary);
3446 return ary;
3447}
3448
3449/*
3450 * call-seq:
3451 * sort -> new_array
3452 * sort {|a, b| ... } -> new_array
3453 *
3454 * Returns a new array containing the elements of +self+, sorted.
3455 *
3456 * With no block given, compares elements using operator <tt>#<=></tt>
3457 * (see Object#<=>):
3458 *
3459 * [0, 2, 3, 1].sort # => [0, 1, 2, 3]
3460 *
3461 * With a block given, calls the block with each combination of pairs of elements from +self+;
3462 * for each pair +a+ and +b+, the block should return a numeric:
3463 *
3464 * - Negative when +b+ is to follow +a+.
3465 * - Zero when +a+ and +b+ are equivalent.
3466 * - Positive when +a+ is to follow +b+.
3467 *
3468 * Example:
3469 *
3470 * a = [3, 2, 0, 1]
3471 * a.sort {|a, b| a <=> b } # => [0, 1, 2, 3]
3472 * a.sort {|a, b| b <=> a } # => [3, 2, 1, 0]
3473 *
3474 * When the block returns zero, the order for +a+ and +b+ is indeterminate,
3475 * and may be unstable.
3476 *
3477 * See an example in Numeric#nonzero? for the idiom to sort more
3478 * complex structure.
3479 *
3480 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
3481 */
3482
3483VALUE
3484rb_ary_sort(VALUE ary)
3485{
3486 ary = rb_ary_dup(ary);
3487 rb_ary_sort_bang(ary);
3488 return ary;
3489}
3490
3491static VALUE rb_ary_bsearch_index(VALUE ary);
3492
3493/*
3494 * call-seq:
3495 * bsearch {|element| ... } -> found_element or nil
3496 * bsearch -> new_enumerator
3497 *
3498 * Returns the element from +self+ found by a binary search,
3499 * or +nil+ if the search found no suitable element.
3500 *
3501 * See {Binary Searching}[rdoc-ref:bsearch.rdoc].
3502 *
3503 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
3504 */
3505
3506static VALUE
3507rb_ary_bsearch(VALUE ary)
3508{
3509 VALUE index_result = rb_ary_bsearch_index(ary);
3510
3511 if (FIXNUM_P(index_result)) {
3512 return rb_ary_entry(ary, FIX2LONG(index_result));
3513 }
3514 return index_result;
3515}
3516
3517/*
3518 * call-seq:
3519 * bsearch_index {|element| ... } -> integer or nil
3520 * bsearch_index -> new_enumerator
3521 *
3522 * Returns the integer index of the element from +self+ found by a binary search,
3523 * or +nil+ if the search found no suitable element.
3524 *
3525 * See {Binary Searching}[rdoc-ref:bsearch.rdoc].
3526 *
3527 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
3528 */
3529
3530static VALUE
3531rb_ary_bsearch_index(VALUE ary)
3532{
3533 long low = 0, high = RARRAY_LEN(ary), mid;
3534 int smaller = 0, satisfied = 0;
3535 VALUE v, val;
3536
3537 RETURN_ENUMERATOR(ary, 0, 0);
3538 while (low < high) {
3539 mid = low + ((high - low) / 2);
3540 val = rb_ary_entry(ary, mid);
3541 v = rb_yield(val);
3542 if (FIXNUM_P(v)) {
3543 if (v == INT2FIX(0)) return INT2FIX(mid);
3544 smaller = (SIGNED_VALUE)v < 0; /* Fixnum preserves its sign-bit */
3545 }
3546 else if (v == Qtrue) {
3547 satisfied = 1;
3548 smaller = 1;
3549 }
3550 else if (!RTEST(v)) {
3551 smaller = 0;
3552 }
3553 else if (rb_obj_is_kind_of(v, rb_cNumeric)) {
3554 const VALUE zero = INT2FIX(0);
3555 switch (rb_cmpint(rb_funcallv(v, id_cmp, 1, &zero), v, zero)) {
3556 case 0: return INT2FIX(mid);
3557 case 1: smaller = 0; break;
3558 case -1: smaller = 1;
3559 }
3560 }
3561 else {
3562 rb_raise(rb_eTypeError, "wrong argument type %"PRIsVALUE
3563 " (must be numeric, true, false or nil)",
3564 rb_obj_class(v));
3565 }
3566 if (smaller) {
3567 high = mid;
3568 }
3569 else {
3570 low = mid + 1;
3571 }
3572 }
3573 if (!satisfied) return Qnil;
3574 return INT2FIX(low);
3575}
3576
3577
3578static VALUE
3579sort_by_i(RB_BLOCK_CALL_FUNC_ARGLIST(i, dummy))
3580{
3581 return rb_yield(i);
3582}
3583
3584/*
3585 * call-seq:
3586 * sort_by! {|element| ... } -> self
3587 * sort_by! -> new_enumerator
3588 *
3589 * With a block given, sorts the elements of +self+ in place;
3590 * returns self.
3591 *
3592 * Calls the block with each successive element;
3593 * sorts elements based on the values returned from the block:
3594 *
3595 * a = ['aaaa', 'bbb', 'cc', 'd']
3596 * a.sort_by! {|element| element.size }
3597 * a # => ["d", "cc", "bbb", "aaaa"]
3598 *
3599 * For duplicate values returned by the block, the ordering is indeterminate, and may be unstable.
3600 *
3601 * With no block given, returns a new Enumerator.
3602 *
3603 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
3604 */
3605
3606static VALUE
3607rb_ary_sort_by_bang(VALUE ary)
3608{
3609 VALUE sorted;
3610
3611 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
3612 rb_ary_modify(ary);
3613 if (RARRAY_LEN(ary) > 1) {
3614 sorted = rb_block_call(ary, rb_intern("sort_by"), 0, 0, sort_by_i, 0);
3615 rb_ary_replace(ary, sorted);
3616 }
3617 return ary;
3618}
3619
3620
3621/*
3622 * call-seq:
3623 * collect {|element| ... } -> new_array
3624 * collect -> new_enumerator
3625 * map {|element| ... } -> new_array
3626 * map -> new_enumerator
3627 *
3628 * With a block given, calls the block with each element of +self+;
3629 * returns a new array whose elements are the return values from the block:
3630 *
3631 * a = [:foo, 'bar', 2]
3632 * a1 = a.map {|element| element.class }
3633 * a1 # => [Symbol, String, Integer]
3634 *
3635 * With no block given, returns a new Enumerator.
3636 *
3637 * Related: #collect!;
3638 * see also {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
3639 */
3640
3641static VALUE
3642rb_ary_collect(VALUE ary)
3643{
3644 long i;
3645 VALUE collect;
3646
3647 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
3648 collect = rb_ary_new2(RARRAY_LEN(ary));
3649 for (i = 0; i < RARRAY_LEN(ary); i++) {
3650 rb_ary_push(collect, rb_yield(RARRAY_AREF(ary, i)));
3651 }
3652 return collect;
3653}
3654
3655
3656/*
3657 * call-seq:
3658 * collect! {|element| ... } -> new_array
3659 * collect! -> new_enumerator
3660 * map! {|element| ... } -> new_array
3661 * map! -> new_enumerator
3662 *
3663 * With a block given, calls the block with each element of +self+
3664 * and replaces the element with the block's return value;
3665 * returns +self+:
3666 *
3667 * a = [:foo, 'bar', 2]
3668 * a.map! { |element| element.class } # => [Symbol, String, Integer]
3669 *
3670 * With no block given, returns a new Enumerator.
3671 *
3672 * Related: #collect;
3673 * see also {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
3674 */
3675
3676static VALUE
3677rb_ary_collect_bang(VALUE ary)
3678{
3679 long i;
3680
3681 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
3682 rb_ary_modify(ary);
3683 for (i = 0; i < RARRAY_LEN(ary); i++) {
3684 rb_ary_store(ary, i, rb_yield(RARRAY_AREF(ary, i)));
3685 }
3686 return ary;
3687}
3688
3689VALUE
3690rb_get_values_at(VALUE obj, long olen, int argc, const VALUE *argv, VALUE (*func) (VALUE, long))
3691{
3692 VALUE result = rb_ary_new2(argc);
3693 long beg, len, i, j;
3694
3695 for (i=0; i<argc; i++) {
3696 if (FIXNUM_P(argv[i])) {
3697 rb_ary_push(result, (*func)(obj, FIX2LONG(argv[i])));
3698 continue;
3699 }
3700 /* check if idx is Range */
3701 if (rb_range_beg_len(argv[i], &beg, &len, olen, 1)) {
3702 long end = olen < beg+len ? olen : beg+len;
3703 for (j = beg; j < end; j++) {
3704 rb_ary_push(result, (*func)(obj, j));
3705 }
3706 if (beg + len > j)
3707 rb_ary_resize(result, RARRAY_LEN(result) + (beg + len) - j);
3708 continue;
3709 }
3710 rb_ary_push(result, (*func)(obj, NUM2LONG(argv[i])));
3711 }
3712 return result;
3713}
3714
3715static VALUE
3716append_values_at_single(VALUE result, VALUE ary, long olen, VALUE idx)
3717{
3718 long beg, len;
3719 if (FIXNUM_P(idx)) {
3720 beg = FIX2LONG(idx);
3721 }
3722 /* check if idx is Range */
3723 else if (rb_range_beg_len(idx, &beg, &len, olen, 1)) {
3724 if (len > 0) {
3725 const VALUE *const src = RARRAY_CONST_PTR(ary);
3726 const long end = beg + len;
3727 const long prevlen = RARRAY_LEN(result);
3728 if (beg < olen) {
3729 rb_ary_cat(result, src + beg, end > olen ? olen-beg : len);
3730 }
3731 if (end > olen) {
3732 rb_ary_store(result, prevlen + len - 1, Qnil);
3733 }
3734 }
3735 return result;
3736 }
3737 else {
3738 beg = NUM2LONG(idx);
3739 }
3740 return rb_ary_push(result, rb_ary_entry(ary, beg));
3741}
3742
3743/*
3744 * call-seq:
3745 * values_at(*specifiers) -> new_array
3746 *
3747 * Returns elements from +self+ in a new array; does not modify +self+.
3748 *
3749 * The objects included in the returned array are the elements of +self+
3750 * selected by the given +specifiers+,
3751 * each of which must be a numeric index or a Range.
3752 *
3753 * In brief:
3754 *
3755 * a = ['a', 'b', 'c', 'd']
3756 *
3757 * # Index specifiers.
3758 * a.values_at(2, 0, 2, 0) # => ["c", "a", "c", "a"] # May repeat.
3759 * a.values_at(-4, -3, -2, -1) # => ["a", "b", "c", "d"] # Counts backwards if negative.
3760 * a.values_at(-50, 50) # => [nil, nil] # Outside of self.
3761 *
3762 * # Range specifiers.
3763 * a.values_at(1..3) # => ["b", "c", "d"] # From range.begin to range.end.
3764 * a.values_at(1...3) # => ["b", "c"] # End excluded.
3765 * a.values_at(3..1) # => [] # No such elements.
3766 *
3767 * a.values_at(-3..3) # => ["b", "c", "d"] # Negative range.begin counts backwards.
3768 * a.values_at(-50..3) # Raises RangeError.
3769 *
3770 * a.values_at(1..-2) # => ["b", "c"] # Negative range.end counts backwards.
3771 * a.values_at(1..-50) # => [] # No such elements.
3772 *
3773 * # Mixture of specifiers.
3774 * a.values_at(2..3, 3, 0..1, 0) # => ["c", "d", "d", "a", "b", "a"]
3775 *
3776 * With no +specifiers+ given, returns a new empty array:
3777 *
3778 * a = ['a', 'b', 'c', 'd']
3779 * a.values_at # => []
3780 *
3781 * For each numeric specifier +index+, includes an element:
3782 *
3783 * - For each non-negative numeric specifier +index+ that is in-range (less than <tt>self.size</tt>),
3784 * includes the element at offset +index+:
3785 *
3786 * a.values_at(0, 2) # => ["a", "c"]
3787 * a.values_at(0.1, 2.9) # => ["a", "c"]
3788 *
3789 * - For each negative numeric +index+ that is in-range (greater than or equal to <tt>- self.size</tt>),
3790 * counts backwards from the end of +self+:
3791 *
3792 * a.values_at(-1, -4) # => ["d", "a"]
3793 *
3794 * The given indexes may be in any order, and may repeat:
3795 *
3796 * a.values_at(2, 0, 1, 0, 2) # => ["c", "a", "b", "a", "c"]
3797 *
3798 * For each +index+ that is out-of-range, includes +nil+:
3799 *
3800 * a.values_at(4, -5) # => [nil, nil]
3801 *
3802 * For each Range specifier +range+, includes elements
3803 * according to <tt>range.begin</tt> and <tt>range.end</tt>:
3804 *
3805 * - If both <tt>range.begin</tt> and <tt>range.end</tt>
3806 * are non-negative and in-range (less than <tt>self.size</tt>),
3807 * includes elements from index <tt>range.begin</tt>
3808 * through <tt>range.end - 1</tt> (if <tt>range.exclude_end?</tt>),
3809 * or through <tt>range.end</tt> (otherwise):
3810 *
3811 * a.values_at(1..2) # => ["b", "c"]
3812 * a.values_at(1...2) # => ["b"]
3813 *
3814 * - If <tt>range.begin</tt> is negative and in-range (greater than or equal to <tt>- self.size</tt>),
3815 * counts backwards from the end of +self+:
3816 *
3817 * a.values_at(-2..3) # => ["c", "d"]
3818 *
3819 * - If <tt>range.begin</tt> is negative and out-of-range, raises an exception:
3820 *
3821 * a.values_at(-5..3) # Raises RangeError.
3822 *
3823 * - If <tt>range.end</tt> is positive and out-of-range,
3824 * extends the returned array with +nil+ elements:
3825 *
3826 * a.values_at(1..5) # => ["b", "c", "d", nil, nil]
3827 *
3828 * - If <tt>range.end</tt> is negative and in-range,
3829 * counts backwards from the end of +self+:
3830 *
3831 * a.values_at(1..-2) # => ["b", "c"]
3832 *
3833 * - If <tt>range.end</tt> is negative and out-of-range,
3834 * returns an empty array:
3835 *
3836 * a.values_at(1..-5) # => []
3837 *
3838 * The given ranges may be in any order and may repeat:
3839 *
3840 * a.values_at(2..3, 0..1, 2..3) # => ["c", "d", "a", "b", "c", "d"]
3841 *
3842 * The given specifiers may be any mixture of indexes and ranges:
3843 *
3844 * a.values_at(3, 1..2, 0, 2..3) # => ["d", "b", "c", "a", "c", "d"]
3845 *
3846 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
3847 */
3848
3849static VALUE
3850rb_ary_values_at(int argc, VALUE *argv, VALUE ary)
3851{
3852 long i, olen = RARRAY_LEN(ary);
3853 VALUE result = rb_ary_new_capa(argc);
3854 for (i = 0; i < argc; ++i) {
3855 append_values_at_single(result, ary, olen, argv[i]);
3856 }
3857 RB_GC_GUARD(ary);
3858 return result;
3859}
3860
3861
3862/*
3863 * call-seq:
3864 * select {|element| ... } -> new_array
3865 * select -> new_enumerator
3866 * filter {|element| ... } -> new_array
3867 * filter -> new_enumerator
3868 *
3869 * With a block given, calls the block with each element of +self+;
3870 * returns a new array containing those elements of +self+
3871 * for which the block returns a truthy value:
3872 *
3873 * a = [:foo, 'bar', 2, :bam]
3874 * a.select {|element| element.to_s.start_with?('b') }
3875 * # => ["bar", :bam]
3876 *
3877 * With no block given, returns a new Enumerator.
3878 *
3879 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
3880 */
3881
3882static VALUE
3883rb_ary_select(VALUE ary)
3884{
3885 VALUE result;
3886 long i;
3887
3888 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
3889 result = rb_ary_new2(RARRAY_LEN(ary));
3890 for (i = 0; i < RARRAY_LEN(ary); i++) {
3891 if (RTEST(rb_yield(RARRAY_AREF(ary, i)))) {
3892 rb_ary_push(result, rb_ary_elt(ary, i));
3893 }
3894 }
3895 return result;
3896}
3897
3899 VALUE ary;
3900 long len[2];
3901};
3902
3903static VALUE
3904select_bang_i(VALUE a)
3905{
3906 volatile struct select_bang_arg *arg = (void *)a;
3907 VALUE ary = arg->ary;
3908 long i1, i2;
3909
3910 for (i1 = i2 = 0; i1 < RARRAY_LEN(ary); arg->len[0] = ++i1) {
3911 VALUE v = RARRAY_AREF(ary, i1);
3912 if (!RTEST(rb_yield(v))) continue;
3913 if (i1 != i2) {
3914 rb_ary_store(ary, i2, v);
3915 }
3916 arg->len[1] = ++i2;
3917 }
3918 return (i1 == i2) ? Qnil : ary;
3919}
3920
3921static VALUE
3922select_bang_ensure(VALUE a)
3923{
3924 volatile struct select_bang_arg *arg = (void *)a;
3925 VALUE ary = arg->ary;
3926 long len = RARRAY_LEN(ary);
3927 long i1 = arg->len[0], i2 = arg->len[1];
3928
3929 if (i2 < len && i2 < i1) {
3930 long tail = 0;
3931 rb_ary_modify(ary);
3932 if (i1 < len) {
3933 tail = len - i1;
3934 RARRAY_PTR_USE(ary, ptr, {
3935 MEMMOVE(ptr + i2, ptr + i1, VALUE, tail);
3936 });
3937 }
3938 ARY_SET_LEN(ary, i2 + tail);
3939 }
3940 return ary;
3941}
3942
3943/*
3944 * call-seq:
3945 * select! {|element| ... } -> self or nil
3946 * select! -> new_enumerator
3947 * filter! {|element| ... } -> self or nil
3948 * filter! -> new_enumerator
3949 *
3950 * With a block given, calls the block with each element of +self+;
3951 * removes from +self+ those elements for which the block returns +false+ or +nil+.
3952 *
3953 * Returns +self+ if any elements were removed:
3954 *
3955 * a = [:foo, 'bar', 2, :bam]
3956 * a.select! {|element| element.to_s.start_with?('b') } # => ["bar", :bam]
3957 *
3958 * Returns +nil+ if no elements were removed.
3959 *
3960 * With no block given, returns a new Enumerator.
3961 *
3962 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
3963 */
3964
3965static VALUE
3966rb_ary_select_bang(VALUE ary)
3967{
3968 struct select_bang_arg args;
3969
3970 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
3971 rb_ary_modify(ary);
3972
3973 args.ary = ary;
3974 args.len[0] = args.len[1] = 0;
3975 return rb_ensure(select_bang_i, (VALUE)&args, select_bang_ensure, (VALUE)&args);
3976}
3977
3978/*
3979 * call-seq:
3980 * keep_if {|element| ... } -> self
3981 * keep_if -> new_enumerator
3982 *
3983 * With a block given, calls the block with each element of +self+;
3984 * removes the element from +self+ if the block does not return a truthy value:
3985 *
3986 * a = [:foo, 'bar', 2, :bam]
3987 * a.keep_if {|element| element.to_s.start_with?('b') } # => ["bar", :bam]
3988 *
3989 * With no block given, returns a new Enumerator.
3990 *
3991 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
3992 */
3993
3994static VALUE
3995rb_ary_keep_if(VALUE ary)
3996{
3997 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
3998 rb_ary_select_bang(ary);
3999 return ary;
4000}
4001
4002static void
4003ary_resize_smaller(VALUE ary, long len)
4004{
4005 rb_ary_modify(ary);
4006 if (RARRAY_LEN(ary) > len) {
4007 ARY_SET_LEN(ary, len);
4008 if (len * 2 < ARY_CAPA(ary) &&
4009 ARY_CAPA(ary) > ARY_DEFAULT_SIZE) {
4010 ary_resize_capa(ary, len * 2);
4011 }
4012 }
4013}
4014
4015/*
4016 * call-seq:
4017 * delete(object) -> last_removed_object
4018 * delete(object) {|element| ... } -> last_removed_object or block_return
4019 *
4020 * Removes zero or more elements from +self+.
4021 *
4022 * With no block given,
4023 * removes from +self+ each element +ele+ such that <tt>ele == object</tt>;
4024 * returns the last removed element:
4025 *
4026 * a = [0, 1, 2, 2.0]
4027 * a.delete(2) # => 2.0
4028 * a # => [0, 1]
4029 *
4030 * Returns +nil+ if no elements removed:
4031 *
4032 * a.delete(2) # => nil
4033 *
4034 * With a block given,
4035 * removes from +self+ each element +ele+ such that <tt>ele == object</tt>.
4036 *
4037 * If any such elements are found, ignores the block
4038 * and returns the last removed element:
4039 *
4040 * a = [0, 1, 2, 2.0]
4041 * a.delete(2) {|element| fail 'Cannot happen' } # => 2.0
4042 * a # => [0, 1]
4043 *
4044 * If no such element is found, returns the block's return value:
4045 *
4046 * a.delete(2) {|element| "Element #{element} not found." }
4047 * # => "Element 2 not found."
4048 *
4049 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
4050 */
4051
4052VALUE
4053rb_ary_delete(VALUE ary, VALUE item)
4054{
4055 VALUE v = item;
4056 long i1, i2;
4057
4058 for (i1 = i2 = 0; i1 < RARRAY_LEN(ary); i1++) {
4059 VALUE e = RARRAY_AREF(ary, i1);
4060
4061 if (rb_equal(e, item)) {
4062 v = e;
4063 continue;
4064 }
4065 if (i1 != i2) {
4066 rb_ary_store(ary, i2, e);
4067 }
4068 i2++;
4069 }
4070 if (RARRAY_LEN(ary) == i2) {
4071 if (rb_block_given_p()) {
4072 return rb_yield(item);
4073 }
4074 return Qnil;
4075 }
4076
4077 ary_resize_smaller(ary, i2);
4078
4079 ary_verify(ary);
4080 return v;
4081}
4082
4083void
4084rb_ary_delete_same(VALUE ary, VALUE item)
4085{
4086 long i1, i2;
4087
4088 for (i1 = i2 = 0; i1 < RARRAY_LEN(ary); i1++) {
4089 VALUE e = RARRAY_AREF(ary, i1);
4090
4091 if (e == item) {
4092 continue;
4093 }
4094 if (i1 != i2) {
4095 rb_ary_store(ary, i2, e);
4096 }
4097 i2++;
4098 }
4099 if (RARRAY_LEN(ary) == i2) {
4100 return;
4101 }
4102
4103 ary_resize_smaller(ary, i2);
4104}
4105
4106VALUE
4107rb_ary_delete_at(VALUE ary, long pos)
4108{
4109 long len = RARRAY_LEN(ary);
4110 VALUE del;
4111
4112 if (pos >= len) return Qnil;
4113 if (pos < 0) {
4114 pos += len;
4115 if (pos < 0) return Qnil;
4116 }
4117
4118 rb_ary_modify(ary);
4119 del = RARRAY_AREF(ary, pos);
4120 RARRAY_PTR_USE(ary, ptr, {
4121 MEMMOVE(ptr+pos, ptr+pos+1, VALUE, len-pos-1);
4122 });
4123 ARY_INCREASE_LEN(ary, -1);
4124 ary_verify(ary);
4125 return del;
4126}
4127
4128/*
4129 * call-seq:
4130 * delete_at(index) -> removed_object or nil
4131 *
4132 * Removes the element of +self+ at the given +index+, which must be an
4133 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects].
4134 *
4135 * When +index+ is non-negative, deletes the element at offset +index+:
4136 *
4137 * a = [:foo, 'bar', 2]
4138 * a.delete_at(1) # => "bar"
4139 * a # => [:foo, 2]
4140 *
4141 * When +index+ is negative, counts backward from the end of the array:
4142 *
4143 * a = [:foo, 'bar', 2]
4144 * a.delete_at(-2) # => "bar"
4145 * a # => [:foo, 2]
4146 *
4147 * When +index+ is out of range, returns +nil+.
4148 *
4149 * a = [:foo, 'bar', 2]
4150 * a.delete_at(3) # => nil
4151 * a.delete_at(-4) # => nil
4152 *
4153 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
4154 */
4155
4156static VALUE
4157rb_ary_delete_at_m(VALUE ary, VALUE pos)
4158{
4159 return rb_ary_delete_at(ary, NUM2LONG(pos));
4160}
4161
4162static VALUE
4163ary_slice_bang_by_rb_ary_splice(VALUE ary, long pos, long len)
4164{
4165 const long orig_len = RARRAY_LEN(ary);
4166
4167 if (len < 0) {
4168 return Qnil;
4169 }
4170 else if (pos < -orig_len) {
4171 return Qnil;
4172 }
4173 else if (pos < 0) {
4174 pos += orig_len;
4175 }
4176 else if (orig_len < pos) {
4177 return Qnil;
4178 }
4179 if (orig_len < pos + len) {
4180 len = orig_len - pos;
4181 }
4182 if (len == 0) {
4183 return rb_ary_new2(0);
4184 }
4185 else {
4186 VALUE arg2 = rb_ary_new4(len, RARRAY_CONST_PTR(ary)+pos);
4187 rb_ary_splice(ary, pos, len, 0, 0);
4188 return arg2;
4189 }
4190}
4191
4192/*
4193 * call-seq:
4194 * slice!(index) -> object or nil
4195 * slice!(start, length) -> new_array or nil
4196 * slice!(range) -> new_array or nil
4197 *
4198 * Removes and returns elements from +self+.
4199 *
4200 * With numeric argument +index+ given,
4201 * removes and returns the element at offset +index+:
4202 *
4203 * a = ['a', 'b', 'c', 'd']
4204 * a.slice!(2) # => "c"
4205 * a # => ["a", "b", "d"]
4206 * a.slice!(2.1) # => "d"
4207 * a # => ["a", "b"]
4208 *
4209 * If +index+ is negative, counts backwards from the end of +self+:
4210 *
4211 * a = ['a', 'b', 'c', 'd']
4212 * a.slice!(-2) # => "c"
4213 * a # => ["a", "b", "d"]
4214 *
4215 * If +index+ is out of range, returns +nil+.
4216 *
4217 * With numeric arguments +start+ and +length+ given,
4218 * removes +length+ elements from +self+ beginning at zero-based offset +start+;
4219 * returns the removed objects in a new array:
4220 *
4221 * a = ['a', 'b', 'c', 'd']
4222 * a.slice!(1, 2) # => ["b", "c"]
4223 * a # => ["a", "d"]
4224 * a.slice!(0.1, 1.1) # => ["a"]
4225 * a # => ["d"]
4226 *
4227 * If +start+ is negative, counts backwards from the end of +self+:
4228 *
4229 * a = ['a', 'b', 'c', 'd']
4230 * a.slice!(-2, 1) # => ["c"]
4231 * a # => ["a", "b", "d"]
4232 *
4233 * If +start+ is out-of-range, returns +nil+:
4234 *
4235 * a = ['a', 'b', 'c', 'd']
4236 * a.slice!(5, 1) # => nil
4237 * a.slice!(-5, 1) # => nil
4238 *
4239 * If <tt>start + length</tt> exceeds the array size,
4240 * removes and returns all elements from offset +start+ to the end:
4241 *
4242 * a = ['a', 'b', 'c', 'd']
4243 * a.slice!(2, 50) # => ["c", "d"]
4244 * a # => ["a", "b"]
4245 *
4246 * If <tt>start == a.size</tt> and +length+ is non-negative,
4247 * returns a new empty array.
4248 *
4249 * If +length+ is negative, returns +nil+.
4250 *
4251 * With Range argument +range+ given,
4252 * treats <tt>range.min</tt> as +start+ (as above)
4253 * and <tt>range.size</tt> as +length+ (as above):
4254 *
4255 * a = ['a', 'b', 'c', 'd']
4256 * a.slice!(1..2) # => ["b", "c"]
4257 * a # => ["a", "d"]
4258 *
4259 * If <tt>range.start == a.size</tt>, returns a new empty array:
4260 *
4261 * a = ['a', 'b', 'c', 'd']
4262 * a.slice!(4..5) # => []
4263 *
4264 * If <tt>range.start</tt> is larger than the array size, returns +nil+:
4265 *
4266 * a = ['a', 'b', 'c', 'd']
4267 a.slice!(5..6) # => nil
4268 *
4269 * If <tt>range.start</tt> is negative,
4270 * calculates the start index by counting backwards from the end of +self+:
4271 *
4272 * a = ['a', 'b', 'c', 'd']
4273 * a.slice!(-2..2) # => ["c"]
4274 *
4275 * If <tt>range.end</tt> is negative,
4276 * calculates the end index by counting backwards from the end of +self+:
4277 *
4278 * a = ['a', 'b', 'c', 'd']
4279 * a.slice!(0..-2) # => ["a", "b", "c"]
4280 *
4281 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
4282 */
4283
4284static VALUE
4285rb_ary_slice_bang(int argc, VALUE *argv, VALUE ary)
4286{
4287 VALUE arg1;
4288 long pos, len;
4289
4290 rb_ary_modify_check(ary);
4291 rb_check_arity(argc, 1, 2);
4292 arg1 = argv[0];
4293
4294 if (argc == 2) {
4295 pos = NUM2LONG(argv[0]);
4296 len = NUM2LONG(argv[1]);
4297 return ary_slice_bang_by_rb_ary_splice(ary, pos, len);
4298 }
4299
4300 if (!FIXNUM_P(arg1)) {
4301 switch (rb_range_beg_len(arg1, &pos, &len, RARRAY_LEN(ary), 0)) {
4302 case Qtrue:
4303 /* valid range */
4304 return ary_slice_bang_by_rb_ary_splice(ary, pos, len);
4305 case Qnil:
4306 /* invalid range */
4307 return Qnil;
4308 default:
4309 /* not a range */
4310 break;
4311 }
4312 }
4313
4314 return rb_ary_delete_at(ary, NUM2LONG(arg1));
4315}
4316
4317static VALUE
4318ary_reject(VALUE orig, VALUE result)
4319{
4320 long i;
4321
4322 for (i = 0; i < RARRAY_LEN(orig); i++) {
4323 VALUE v = RARRAY_AREF(orig, i);
4324
4325 if (!RTEST(rb_yield(v))) {
4326 rb_ary_push(result, v);
4327 }
4328 }
4329 return result;
4330}
4331
4332static VALUE
4333reject_bang_i(VALUE a)
4334{
4335 volatile struct select_bang_arg *arg = (void *)a;
4336 VALUE ary = arg->ary;
4337 long i1, i2;
4338
4339 for (i1 = i2 = 0; i1 < RARRAY_LEN(ary); arg->len[0] = ++i1) {
4340 VALUE v = RARRAY_AREF(ary, i1);
4341 if (RTEST(rb_yield(v))) continue;
4342 if (i1 != i2) {
4343 rb_ary_store(ary, i2, v);
4344 }
4345 arg->len[1] = ++i2;
4346 }
4347 return (i1 == i2) ? Qnil : ary;
4348}
4349
4350static VALUE
4351ary_reject_bang(VALUE ary)
4352{
4353 struct select_bang_arg args;
4354 rb_ary_modify_check(ary);
4355 args.ary = ary;
4356 args.len[0] = args.len[1] = 0;
4357 return rb_ensure(reject_bang_i, (VALUE)&args, select_bang_ensure, (VALUE)&args);
4358}
4359
4360/*
4361 * call-seq:
4362 * reject! {|element| ... } -> self or nil
4363 * reject! -> new_enumerator
4364 *
4365 * With a block given, calls the block with each element of +self+;
4366 * removes each element for which the block returns a truthy value.
4367 *
4368 * Returns +self+ if any elements removed:
4369 *
4370 * a = [:foo, 'bar', 2, 'bat']
4371 * a.reject! {|element| element.to_s.start_with?('b') } # => [:foo, 2]
4372 *
4373 * Returns +nil+ if no elements removed.
4374 *
4375 * With no block given, returns a new Enumerator.
4376 *
4377 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
4378 */
4379
4380static VALUE
4381rb_ary_reject_bang(VALUE ary)
4382{
4383 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
4384 rb_ary_modify(ary);
4385 return ary_reject_bang(ary);
4386}
4387
4388/*
4389 * call-seq:
4390 * reject {|element| ... } -> new_array
4391 * reject -> new_enumerator
4392 *
4393 * With a block given, returns a new array whose elements are all those from +self+
4394 * for which the block returns +false+ or +nil+:
4395 *
4396 * a = [:foo, 'bar', 2, 'bat']
4397 * a1 = a.reject {|element| element.to_s.start_with?('b') }
4398 * a1 # => [:foo, 2]
4399 *
4400 * With no block given, returns a new Enumerator.
4401 *
4402 * Related: {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
4403 */
4404
4405static VALUE
4406rb_ary_reject(VALUE ary)
4407{
4408 VALUE rejected_ary;
4409
4410 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
4411 rejected_ary = rb_ary_new();
4412 ary_reject(ary, rejected_ary);
4413 return rejected_ary;
4414}
4415
4416/*
4417 * call-seq:
4418 * delete_if {|element| ... } -> self
4419 * delete_if -> new_numerator
4420 *
4421 * With a block given, calls the block with each element of +self+;
4422 * removes the element if the block returns a truthy value;
4423 * returns +self+:
4424 *
4425 * a = [:foo, 'bar', 2, 'bat']
4426 * a.delete_if {|element| element.to_s.start_with?('b') } # => [:foo, 2]
4427 *
4428 * With no block given, returns a new Enumerator.
4429 *
4430 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
4431 */
4432
4433static VALUE
4434rb_ary_delete_if(VALUE ary)
4435{
4436 ary_verify(ary);
4437 RETURN_SIZED_ENUMERATOR(ary, 0, 0, ary_enum_length);
4438 ary_reject_bang(ary);
4439 return ary;
4440}
4441
4442static VALUE
4443take_i(RB_BLOCK_CALL_FUNC_ARGLIST(val, cbarg))
4444{
4445 VALUE *args = (VALUE *)cbarg;
4446 if (argc > 1) val = rb_ary_new4(argc, argv);
4447 rb_ary_push(args[0], val);
4448 if (--args[1] == 0) rb_iter_break();
4449 return Qnil;
4450}
4451
4452static VALUE
4453take_items(VALUE obj, long n)
4454{
4455 VALUE result = rb_check_array_type(obj);
4456 VALUE args[2];
4457
4458 if (n == 0) return result;
4459 if (!NIL_P(result)) return rb_ary_subseq(result, 0, n);
4460 result = rb_ary_new2(n);
4461 args[0] = result; args[1] = (VALUE)n;
4462 if (UNDEF_P(rb_check_block_call(obj, idEach, 0, 0, take_i, (VALUE)args)))
4463 rb_raise(rb_eTypeError, "wrong argument type %"PRIsVALUE" (must respond to :each)",
4464 rb_obj_class(obj));
4465 return result;
4466}
4467
4468
4469/*
4470 * call-seq:
4471 * zip(*other_arrays) -> new_array
4472 * zip(*other_arrays) {|sub_array| ... } -> nil
4473 *
4474 * With no block given, combines +self+ with the collection of +other_arrays+;
4475 * returns a new array of sub-arrays:
4476 *
4477 * [0, 1].zip(['zero', 'one'], [:zero, :one])
4478 * # => [[0, "zero", :zero], [1, "one", :one]]
4479 *
4480 * Returned:
4481 *
4482 * - The outer array is of size <tt>self.size</tt>.
4483 * - Each sub-array is of size <tt>other_arrays.size + 1</tt>.
4484 * - The _nth_ sub-array contains (in order):
4485 *
4486 * - The _nth_ element of +self+.
4487 * - The _nth_ element of each of the other arrays, as available.
4488 *
4489 * Example:
4490 *
4491 * a = [0, 1]
4492 * zipped = a.zip(['zero', 'one'], [:zero, :one])
4493 * # => [[0, "zero", :zero], [1, "one", :one]]
4494 * zipped.size # => 2 # Same size as a.
4495 * zipped.first.size # => 3 # Size of other arrays plus 1.
4496 *
4497 * When the other arrays are all the same size as +self+,
4498 * the returned sub-arrays are a rearrangement containing exactly elements of all the arrays
4499 * (including +self+), with no omissions or additions:
4500 *
4501 * a = [:a0, :a1, :a2, :a3]
4502 * b = [:b0, :b1, :b2, :b3]
4503 * c = [:c0, :c1, :c2, :c3]
4504 * d = a.zip(b, c)
4505 * pp d
4506 * # =>
4507 * [[:a0, :b0, :c0],
4508 * [:a1, :b1, :c1],
4509 * [:a2, :b2, :c2],
4510 * [:a3, :b3, :c3]]
4511 *
4512 * When one of the other arrays is smaller than +self+,
4513 * pads the corresponding sub-array with +nil+ elements:
4514 *
4515 * a = [:a0, :a1, :a2, :a3]
4516 * b = [:b0, :b1, :b2]
4517 * c = [:c0, :c1]
4518 * d = a.zip(b, c)
4519 * pp d
4520 * # =>
4521 * [[:a0, :b0, :c0],
4522 * [:a1, :b1, :c1],
4523 * [:a2, :b2, nil],
4524 * [:a3, nil, nil]]
4525 *
4526 * When one of the other arrays is larger than +self+,
4527 * _ignores_ its trailing elements:
4528 *
4529 * a = [:a0, :a1, :a2, :a3]
4530 * b = [:b0, :b1, :b2, :b3, :b4]
4531 * c = [:c0, :c1, :c2, :c3, :c4, :c5]
4532 * d = a.zip(b, c)
4533 * pp d
4534 * # =>
4535 * [[:a0, :b0, :c0],
4536 * [:a1, :b1, :c1],
4537 * [:a2, :b2, :c2],
4538 * [:a3, :b3, :c3]]
4539 *
4540 * With a block given, calls the block with each of the other arrays;
4541 * returns +nil+:
4542 *
4543 * d = []
4544 * a = [:a0, :a1, :a2, :a3]
4545 * b = [:b0, :b1, :b2, :b3]
4546 * c = [:c0, :c1, :c2, :c3]
4547 * a.zip(b, c) {|sub_array| d.push(sub_array.reverse) } # => nil
4548 * pp d
4549 * # =>
4550 * [[:c0, :b0, :a0],
4551 * [:c1, :b1, :a1],
4552 * [:c2, :b2, :a2],
4553 * [:c3, :b3, :a3]]
4554 *
4555 * For an *object* in *other_arrays* that is not actually an array,
4556 * forms the the "other array" as <tt>object.to_ary</tt>, if defined,
4557 * or as <tt>object.each.to_a</tt> otherwise.
4558 *
4559 * Related: see {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
4560 */
4561
4562static VALUE
4563rb_ary_zip(int argc, VALUE *argv, VALUE ary)
4564{
4565 int i, j;
4566 long len = RARRAY_LEN(ary);
4567 VALUE result = Qnil;
4568
4569 for (i=0; i<argc; i++) {
4570 argv[i] = take_items(argv[i], len);
4571 }
4572
4573 if (rb_block_given_p()) {
4574 int arity = rb_block_arity();
4575
4576 if (arity > 1) {
4577 VALUE work, *tmp;
4578
4579 tmp = ALLOCV_N(VALUE, work, argc+1);
4580
4581 for (i=0; i<RARRAY_LEN(ary); i++) {
4582 tmp[0] = RARRAY_AREF(ary, i);
4583 for (j=0; j<argc; j++) {
4584 tmp[j+1] = rb_ary_elt(argv[j], i);
4585 }
4586 rb_yield_values2(argc+1, tmp);
4587 }
4588
4589 if (work) ALLOCV_END(work);
4590 }
4591 else {
4592 for (i=0; i<RARRAY_LEN(ary); i++) {
4593 VALUE tmp = rb_ary_new2(argc+1);
4594
4595 rb_ary_push(tmp, RARRAY_AREF(ary, i));
4596 for (j=0; j<argc; j++) {
4597 rb_ary_push(tmp, rb_ary_elt(argv[j], i));
4598 }
4599 rb_yield(tmp);
4600 }
4601 }
4602 }
4603 else {
4604 result = rb_ary_new_capa(len);
4605
4606 for (i=0; i<len; i++) {
4607 VALUE tmp = rb_ary_new_capa(argc+1);
4608
4609 rb_ary_push(tmp, RARRAY_AREF(ary, i));
4610 for (j=0; j<argc; j++) {
4611 rb_ary_push(tmp, rb_ary_elt(argv[j], i));
4612 }
4613 rb_ary_push(result, tmp);
4614 }
4615 }
4616
4617 return result;
4618}
4619
4620/*
4621 * call-seq:
4622 * transpose -> new_array
4623 *
4624 * Returns a new array that is +self+
4625 * as a {transposed matrix}[https://en.wikipedia.org/wiki/Transpose]:
4626 *
4627 * a = [[:a0, :a1], [:b0, :b1], [:c0, :c1]]
4628 * a.transpose # => [[:a0, :b0, :c0], [:a1, :b1, :c1]]
4629 *
4630 * The elements of +self+ must all be the same size.
4631 *
4632 * Related: see {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
4633 */
4634
4635static VALUE
4636rb_ary_transpose(VALUE ary)
4637{
4638 long elen = -1, alen, i, j;
4639 VALUE tmp, result = 0;
4640
4641 alen = RARRAY_LEN(ary);
4642 if (alen == 0) return rb_ary_dup(ary);
4643 for (i=0; i<alen; i++) {
4644 tmp = to_ary(rb_ary_elt(ary, i));
4645 if (elen < 0) { /* first element */
4646 elen = RARRAY_LEN(tmp);
4647 result = rb_ary_new2(elen);
4648 for (j=0; j<elen; j++) {
4649 rb_ary_store(result, j, rb_ary_new2(alen));
4650 }
4651 }
4652 else if (elen != RARRAY_LEN(tmp)) {
4653 rb_raise(rb_eIndexError, "element size differs (%ld should be %ld)",
4654 RARRAY_LEN(tmp), elen);
4655 }
4656 for (j=0; j<elen; j++) {
4657 rb_ary_store(rb_ary_elt(result, j), i, rb_ary_elt(tmp, j));
4658 }
4659 }
4660 return result;
4661}
4662
4663/*
4664 * call-seq:
4665 * initialize_copy(other_array) -> self
4666 * replace(other_array) -> self
4667 *
4668 * Replaces the elements of +self+ with the elements of +other_array+, which must be an
4669 * {array-convertible object}[rdoc-ref:implicit_conversion.rdoc@Array-Convertible+Objects];
4670 * returns +self+:
4671 *
4672 * a = ['a', 'b', 'c'] # => ["a", "b", "c"]
4673 * a.replace(['d', 'e']) # => ["d", "e"]
4674 *
4675 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
4676 */
4677
4678VALUE
4679rb_ary_replace(VALUE copy, VALUE orig)
4680{
4681 rb_ary_modify_check(copy);
4682 orig = to_ary(orig);
4683 if (copy == orig) return copy;
4684
4685 rb_ary_reset(copy);
4686
4687 /* orig has enough space to embed the contents of orig. */
4688 if (RARRAY_LEN(orig) <= ary_embed_capa(copy)) {
4689 RUBY_ASSERT(ARY_EMBED_P(copy));
4690 ary_memcpy(copy, 0, RARRAY_LEN(orig), RARRAY_CONST_PTR(orig));
4691 ARY_SET_EMBED_LEN(copy, RARRAY_LEN(orig));
4692 }
4693 /* orig is embedded but copy does not have enough space to embed the
4694 * contents of orig. */
4695 else if (ARY_EMBED_P(orig)) {
4696 long len = ARY_EMBED_LEN(orig);
4697 VALUE *ptr = ary_heap_alloc_buffer(len);
4698
4699 FL_UNSET_EMBED(copy);
4700 ARY_SET_PTR(copy, ptr);
4701 ARY_SET_LEN(copy, len);
4702 ARY_SET_CAPA(copy, len);
4703
4704 // No allocation and exception expected that could leave `copy` in a
4705 // bad state from the edits above.
4706 ary_memcpy(copy, 0, len, RARRAY_CONST_PTR(orig));
4707 }
4708 /* Otherwise, orig is on heap and copy does not have enough space to embed
4709 * the contents of orig. */
4710 else {
4711 VALUE shared_root = ary_make_shared(orig);
4712 FL_UNSET_EMBED(copy);
4713 ARY_SET_PTR(copy, ARY_HEAP_PTR(orig));
4714 ARY_SET_LEN(copy, ARY_HEAP_LEN(orig));
4715 rb_ary_set_shared(copy, shared_root);
4716 }
4717 ary_verify(copy);
4718 return copy;
4719}
4720
4721/*
4722 * call-seq:
4723 * clear -> self
4724 *
4725 * Removes all elements from +self+; returns +self+:
4726 *
4727 * a = [:foo, 'bar', 2]
4728 * a.clear # => []
4729 *
4730 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
4731 */
4732
4733VALUE
4734rb_ary_clear(VALUE ary)
4735{
4736 rb_ary_modify_check(ary);
4737 if (ARY_SHARED_P(ary)) {
4738 rb_ary_unshare(ary);
4739 FL_SET_EMBED(ary);
4740 ARY_SET_EMBED_LEN(ary, 0);
4741 }
4742 else {
4743 ARY_SET_LEN(ary, 0);
4744 if (ARY_DEFAULT_SIZE * 2 < ARY_CAPA(ary)) {
4745 ary_resize_capa(ary, ARY_DEFAULT_SIZE * 2);
4746 }
4747 }
4748 ary_verify(ary);
4749 return ary;
4750}
4751
4752/*
4753 * call-seq:
4754 * fill(object, start = nil, count = nil) -> new_array
4755 * fill(object, range) -> new_array
4756 * fill(start = nil, count = nil) {|element| ... } -> new_array
4757 * fill(range) {|element| ... } -> new_array
4758 *
4759 * Replaces selected elements in +self+;
4760 * may add elements to +self+;
4761 * always returns +self+ (never a new array).
4762 *
4763 * In brief:
4764 *
4765 * # Non-negative start.
4766 * ['a', 'b', 'c', 'd'].fill('-', 1, 2) # => ["a", "-", "-", "d"]
4767 * ['a', 'b', 'c', 'd'].fill(1, 2) {|e| e.to_s } # => ["a", "1", "2", "d"]
4768 *
4769 * # Extends with specified values if necessary.
4770 * ['a', 'b', 'c', 'd'].fill('-', 3, 2) # => ["a", "b", "c", "-", "-"]
4771 * ['a', 'b', 'c', 'd'].fill(3, 2) {|e| e.to_s } # => ["a", "b", "c", "3", "4"]
4772 *
4773 * # Fills with nils if necessary.
4774 * ['a', 'b', 'c', 'd'].fill('-', 6, 2) # => ["a", "b", "c", "d", nil, nil, "-", "-"]
4775 * ['a', 'b', 'c', 'd'].fill(6, 2) {|e| e.to_s } # => ["a", "b", "c", "d", nil, nil, "6", "7"]
4776 *
4777 * # For negative start, counts backwards from the end.
4778 * ['a', 'b', 'c', 'd'].fill('-', -3, 3) # => ["a", "-", "-", "-"]
4779 * ['a', 'b', 'c', 'd'].fill(-3, 3) {|e| e.to_s } # => ["a", "1", "2", "3"]
4780 *
4781 * # Range.
4782 * ['a', 'b', 'c', 'd'].fill('-', 1..2) # => ["a", "-", "-", "d"]
4783 * ['a', 'b', 'c', 'd'].fill(1..2) {|e| e.to_s } # => ["a", "1", "2", "d"]
4784 *
4785 * When arguments +start+ and +count+ are given,
4786 * they select the elements of +self+ to be replaced;
4787 * each must be an
4788 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects]
4789 * (or +nil+):
4790 *
4791 * - +start+ specifies the zero-based offset of the first element to be replaced;
4792 * +nil+ means zero.
4793 * - +count+ is the number of consecutive elements to be replaced;
4794 * +nil+ means "all the rest."
4795 *
4796 * With argument +object+ given,
4797 * that one object is used for all replacements:
4798 *
4799 * o = Object.new # => #<Object:0x0000014e7bff7600>
4800 * a = ['a', 'b', 'c', 'd'] # => ["a", "b", "c", "d"]
4801 * a.fill(o, 1, 2)
4802 * # => ["a", #<Object:0x0000014e7bff7600>, #<Object:0x0000014e7bff7600>, "d"]
4803 *
4804 * With a block given, the block is called once for each element to be replaced;
4805 * the value passed to the block is the _index_ of the element to be replaced
4806 * (not the element itself);
4807 * the block's return value replaces the element:
4808 *
4809 * a = ['a', 'b', 'c', 'd'] # => ["a", "b", "c", "d"]
4810 * a.fill(1, 2) {|element| element.to_s } # => ["a", "1", "2", "d"]
4811 *
4812 * For arguments +start+ and +count+:
4813 *
4814 * - If +start+ is non-negative,
4815 * replaces +count+ elements beginning at offset +start+:
4816 *
4817 * ['a', 'b', 'c', 'd'].fill('-', 0, 2) # => ["-", "-", "c", "d"]
4818 * ['a', 'b', 'c', 'd'].fill('-', 1, 2) # => ["a", "-", "-", "d"]
4819 * ['a', 'b', 'c', 'd'].fill('-', 2, 2) # => ["a", "b", "-", "-"]
4820 *
4821 * ['a', 'b', 'c', 'd'].fill(0, 2) {|e| e.to_s } # => ["0", "1", "c", "d"]
4822 * ['a', 'b', 'c', 'd'].fill(1, 2) {|e| e.to_s } # => ["a", "1", "2", "d"]
4823 * ['a', 'b', 'c', 'd'].fill(2, 2) {|e| e.to_s } # => ["a", "b", "2", "3"]
4824 *
4825 * Extends +self+ if necessary:
4826 *
4827 * ['a', 'b', 'c', 'd'].fill('-', 3, 2) # => ["a", "b", "c", "-", "-"]
4828 * ['a', 'b', 'c', 'd'].fill('-', 4, 2) # => ["a", "b", "c", "d", "-", "-"]
4829 *
4830 * ['a', 'b', 'c', 'd'].fill(3, 2) {|e| e.to_s } # => ["a", "b", "c", "3", "4"]
4831 * ['a', 'b', 'c', 'd'].fill(4, 2) {|e| e.to_s } # => ["a", "b", "c", "d", "4", "5"]
4832 *
4833 * Fills with +nil+ if necessary:
4834 *
4835 * ['a', 'b', 'c', 'd'].fill('-', 5, 2) # => ["a", "b", "c", "d", nil, "-", "-"]
4836 * ['a', 'b', 'c', 'd'].fill('-', 6, 2) # => ["a", "b", "c", "d", nil, nil, "-", "-"]
4837 *
4838 * ['a', 'b', 'c', 'd'].fill(5, 2) {|e| e.to_s } # => ["a", "b", "c", "d", nil, "5", "6"]
4839 * ['a', 'b', 'c', 'd'].fill(6, 2) {|e| e.to_s } # => ["a", "b", "c", "d", nil, nil, "6", "7"]
4840 *
4841 * Does nothing if +count+ is non-positive:
4842 *
4843 * ['a', 'b', 'c', 'd'].fill('-', 2, 0) # => ["a", "b", "c", "d"]
4844 * ['a', 'b', 'c', 'd'].fill('-', 2, -100) # => ["a", "b", "c", "d"]
4845 * ['a', 'b', 'c', 'd'].fill('-', 6, -100) # => ["a", "b", "c", "d"]
4846 *
4847 * ['a', 'b', 'c', 'd'].fill(2, 0) {|e| fail 'Cannot happen' } # => ["a", "b", "c", "d"]
4848 * ['a', 'b', 'c', 'd'].fill(2, -100) {|e| fail 'Cannot happen' } # => ["a", "b", "c", "d"]
4849 * ['a', 'b', 'c', 'd'].fill(6, -100) {|e| fail 'Cannot happen' } # => ["a", "b", "c", "d"]
4850 *
4851 * - If +start+ is negative, counts backwards from the end of +self+:
4852 *
4853 * ['a', 'b', 'c', 'd'].fill('-', -4, 3) # => ["-", "-", "-", "d"]
4854 * ['a', 'b', 'c', 'd'].fill('-', -3, 3) # => ["a", "-", "-", "-"]
4855 *
4856 * ['a', 'b', 'c', 'd'].fill(-4, 3) {|e| e.to_s } # => ["0", "1", "2", "d"]
4857 * ['a', 'b', 'c', 'd'].fill(-3, 3) {|e| e.to_s } # => ["a", "1", "2", "3"]
4858 *
4859 * Extends +self+ if necessary:
4860 *
4861 * ['a', 'b', 'c', 'd'].fill('-', -2, 3) # => ["a", "b", "-", "-", "-"]
4862 * ['a', 'b', 'c', 'd'].fill('-', -1, 3) # => ["a", "b", "c", "-", "-", "-"]
4863 *
4864 * ['a', 'b', 'c', 'd'].fill(-2, 3) {|e| e.to_s } # => ["a", "b", "2", "3", "4"]
4865 * ['a', 'b', 'c', 'd'].fill(-1, 3) {|e| e.to_s } # => ["a", "b", "c", "3", "4", "5"]
4866 *
4867 * Starts at the beginning of +self+ if +start+ is negative and out-of-range:
4868 *
4869 * ['a', 'b', 'c', 'd'].fill('-', -5, 2) # => ["-", "-", "c", "d"]
4870 * ['a', 'b', 'c', 'd'].fill('-', -6, 2) # => ["-", "-", "c", "d"]
4871 *
4872 * ['a', 'b', 'c', 'd'].fill(-5, 2) {|e| e.to_s } # => ["0", "1", "c", "d"]
4873 * ['a', 'b', 'c', 'd'].fill(-6, 2) {|e| e.to_s } # => ["0", "1", "c", "d"]
4874 *
4875 * Does nothing if +count+ is non-positive:
4876 *
4877 * ['a', 'b', 'c', 'd'].fill('-', -2, 0) # => ["a", "b", "c", "d"]
4878 * ['a', 'b', 'c', 'd'].fill('-', -2, -1) # => ["a", "b", "c", "d"]
4879 *
4880 * ['a', 'b', 'c', 'd'].fill(-2, 0) {|e| fail 'Cannot happen' } # => ["a", "b", "c", "d"]
4881 * ['a', 'b', 'c', 'd'].fill(-2, -1) {|e| fail 'Cannot happen' } # => ["a", "b", "c", "d"]
4882 *
4883 * When argument +range+ is given,
4884 * it must be a Range object whose members are numeric;
4885 * its +begin+ and +end+ values determine the elements of +self+
4886 * to be replaced:
4887 *
4888 * - If both +begin+ and +end+ are positive, they specify the first and last elements
4889 * to be replaced:
4890 *
4891 * ['a', 'b', 'c', 'd'].fill('-', 1..2) # => ["a", "-", "-", "d"]
4892 * ['a', 'b', 'c', 'd'].fill(1..2) {|e| e.to_s } # => ["a", "1", "2", "d"]
4893 *
4894 * If +end+ is smaller than +begin+, replaces no elements:
4895 *
4896 * ['a', 'b', 'c', 'd'].fill('-', 2..1) # => ["a", "b", "c", "d"]
4897 * ['a', 'b', 'c', 'd'].fill(2..1) {|e| e.to_s } # => ["a", "b", "c", "d"]
4898 *
4899 * - If either is negative (or both are negative), counts backwards from the end of +self+:
4900 *
4901 * ['a', 'b', 'c', 'd'].fill('-', -3..2) # => ["a", "-", "-", "d"]
4902 * ['a', 'b', 'c', 'd'].fill('-', 1..-2) # => ["a", "-", "-", "d"]
4903 * ['a', 'b', 'c', 'd'].fill('-', -3..-2) # => ["a", "-", "-", "d"]
4904 *
4905 * ['a', 'b', 'c', 'd'].fill(-3..2) {|e| e.to_s } # => ["a", "1", "2", "d"]
4906 * ['a', 'b', 'c', 'd'].fill(1..-2) {|e| e.to_s } # => ["a", "1", "2", "d"]
4907 * ['a', 'b', 'c', 'd'].fill(-3..-2) {|e| e.to_s } # => ["a", "1", "2", "d"]
4908 *
4909 * - If the +end+ value is excluded (see Range#exclude_end?), omits the last replacement:
4910 *
4911 * ['a', 'b', 'c', 'd'].fill('-', 1...2) # => ["a", "-", "c", "d"]
4912 * ['a', 'b', 'c', 'd'].fill('-', 1...-2) # => ["a", "-", "c", "d"]
4913 *
4914 * ['a', 'b', 'c', 'd'].fill(1...2) {|e| e.to_s } # => ["a", "1", "c", "d"]
4915 * ['a', 'b', 'c', 'd'].fill(1...-2) {|e| e.to_s } # => ["a", "1", "c", "d"]
4916 *
4917 * - If the range is endless (see {Endless Ranges}[rdoc-ref:Range@Endless+Ranges]),
4918 * replaces elements to the end of +self+:
4919 *
4920 * ['a', 'b', 'c', 'd'].fill('-', 1..) # => ["a", "-", "-", "-"]
4921 * ['a', 'b', 'c', 'd'].fill(1..) {|e| e.to_s } # => ["a", "1", "2", "3"]
4922 *
4923 * - If the range is beginless (see {Beginless Ranges}[rdoc-ref:Range@Beginless+Ranges]),
4924 * replaces elements from the beginning of +self+:
4925 *
4926 * ['a', 'b', 'c', 'd'].fill('-', ..2) # => ["-", "-", "-", "d"]
4927 * ['a', 'b', 'c', 'd'].fill(..2) {|e| e.to_s } # => ["0", "1", "2", "d"]
4928 *
4929 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
4930 */
4931
4932static VALUE
4933rb_ary_fill(int argc, VALUE *argv, VALUE ary)
4934{
4935 VALUE item = Qundef, arg1, arg2;
4936 long beg = 0, end = 0, len = 0;
4937
4938 if (rb_block_given_p()) {
4939 rb_scan_args(argc, argv, "02", &arg1, &arg2);
4940 argc += 1; /* hackish */
4941 }
4942 else {
4943 rb_scan_args(argc, argv, "12", &item, &arg1, &arg2);
4944 }
4945 switch (argc) {
4946 case 1:
4947 beg = 0;
4948 len = RARRAY_LEN(ary);
4949 break;
4950 case 2:
4951 if (rb_range_beg_len(arg1, &beg, &len, RARRAY_LEN(ary), 1)) {
4952 break;
4953 }
4954 /* fall through */
4955 case 3:
4956 beg = NIL_P(arg1) ? 0 : NUM2LONG(arg1);
4957 if (beg < 0) {
4958 beg = RARRAY_LEN(ary) + beg;
4959 if (beg < 0) beg = 0;
4960 }
4961 len = NIL_P(arg2) ? RARRAY_LEN(ary) - beg : NUM2LONG(arg2);
4962 break;
4963 }
4964 rb_ary_modify(ary);
4965 if (len < 0) {
4966 return ary;
4967 }
4968 if (beg >= ARY_MAX_SIZE || len > ARY_MAX_SIZE - beg) {
4969 rb_raise(rb_eArgError, "argument too big");
4970 }
4971 end = beg + len;
4972 if (RARRAY_LEN(ary) < end) {
4973 if (end >= ARY_CAPA(ary)) {
4974 ary_resize_capa(ary, end);
4975 }
4976 ary_mem_clear(ary, RARRAY_LEN(ary), end - RARRAY_LEN(ary));
4977 ARY_SET_LEN(ary, end);
4978 }
4979
4980 if (UNDEF_P(item)) {
4981 VALUE v;
4982 long i;
4983
4984 for (i=beg; i<end; i++) {
4985 v = rb_yield(LONG2NUM(i));
4986 if (i>=RARRAY_LEN(ary)) break;
4987 ARY_SET(ary, i, v);
4988 }
4989 }
4990 else {
4991 ary_memfill(ary, beg, len, item);
4992 }
4993 return ary;
4994}
4995
4996/*
4997 * call-seq:
4998 * self + other_array -> new_array
4999 *
5000 * Returns a new array containing all elements of +self+
5001 * followed by all elements of +other_array+:
5002 *
5003 * a = [0, 1] + [2, 3]
5004 * a # => [0, 1, 2, 3]
5005 *
5006 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
5007 */
5008
5009VALUE
5010rb_ary_plus(VALUE x, VALUE y)
5011{
5012 VALUE z;
5013 long len, xlen, ylen;
5014
5015 y = to_ary(y);
5016 xlen = RARRAY_LEN(x);
5017 ylen = RARRAY_LEN(y);
5018 len = xlen + ylen;
5019 z = rb_ary_new2(len);
5020
5021 ary_memcpy(z, 0, xlen, RARRAY_CONST_PTR(x));
5022 ary_memcpy(z, xlen, ylen, RARRAY_CONST_PTR(y));
5023 ARY_SET_LEN(z, len);
5024 return z;
5025}
5026
5027static VALUE
5028ary_append(VALUE x, VALUE y)
5029{
5030 long n = RARRAY_LEN(y);
5031 if (n > 0) {
5032 rb_ary_splice(x, RARRAY_LEN(x), 0, RARRAY_CONST_PTR(y), n);
5033 }
5034 RB_GC_GUARD(y);
5035 return x;
5036}
5037
5038/*
5039 * call-seq:
5040 * concat(*other_arrays) -> self
5041 *
5042 * Adds to +self+ all elements from each array in +other_arrays+; returns +self+:
5043 *
5044 * a = [0, 1]
5045 * a.concat(['two', 'three'], [:four, :five], a)
5046 * # => [0, 1, "two", "three", :four, :five, 0, 1]
5047 *
5048 * Related: see {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
5049 */
5050
5051static VALUE
5052rb_ary_concat_multi(int argc, VALUE *argv, VALUE ary)
5053{
5054 rb_ary_modify_check(ary);
5055
5056 if (argc == 1) {
5057 rb_ary_concat(ary, argv[0]);
5058 }
5059 else if (argc > 1) {
5060 int i;
5061 VALUE args = rb_ary_hidden_new(argc);
5062 for (i = 0; i < argc; i++) {
5063 rb_ary_concat(args, argv[i]);
5064 }
5065 ary_append(ary, args);
5066 }
5067
5068 ary_verify(ary);
5069 return ary;
5070}
5071
5072VALUE
5073rb_ary_concat(VALUE x, VALUE y)
5074{
5075 return ary_append(x, to_ary(y));
5076}
5077
5078/*
5079 * call-seq:
5080 * self * n -> new_array
5081 * self * string_separator -> new_string
5082 *
5083 * When non-negative integer argument +n+ is given,
5084 * returns a new array built by concatenating +n+ copies of +self+:
5085 *
5086 * a = ['x', 'y']
5087 * a * 3 # => ["x", "y", "x", "y", "x", "y"]
5088 *
5089 * When string argument +string_separator+ is given,
5090 * equivalent to <tt>self.join(string_separator)</tt>:
5091 *
5092 * [0, [0, 1], {foo: 0}] * ', ' # => "0, 0, 1, {foo: 0}"
5093 *
5094 */
5095
5096static VALUE
5097rb_ary_times(VALUE ary, VALUE times)
5098{
5099 VALUE ary2, tmp;
5100 const VALUE *ptr;
5101 long t, len;
5102
5103 tmp = rb_check_string_type(times);
5104 if (!NIL_P(tmp)) {
5105 return rb_ary_join(ary, tmp);
5106 }
5107
5108 len = NUM2LONG(times);
5109 if (len == 0) {
5110 ary2 = ary_new(rb_cArray, 0);
5111 goto out;
5112 }
5113 if (len < 0) {
5114 rb_raise(rb_eArgError, "negative argument");
5115 }
5116 if (ARY_MAX_SIZE/len < RARRAY_LEN(ary)) {
5117 rb_raise(rb_eArgError, "argument too big");
5118 }
5119 len *= RARRAY_LEN(ary);
5120
5121 ary2 = ary_new(rb_cArray, len);
5122 ARY_SET_LEN(ary2, len);
5123
5124 ptr = RARRAY_CONST_PTR(ary);
5125 t = RARRAY_LEN(ary);
5126 if (0 < t) {
5127 ary_memcpy(ary2, 0, t, ptr);
5128 while (t <= len/2) {
5129 ary_memcpy(ary2, t, t, RARRAY_CONST_PTR(ary2));
5130 t *= 2;
5131 }
5132 if (t < len) {
5133 ary_memcpy(ary2, t, len-t, RARRAY_CONST_PTR(ary2));
5134 }
5135 }
5136 out:
5137 return ary2;
5138}
5139
5140/*
5141 * call-seq:
5142 * assoc(object) -> found_array or nil
5143 *
5144 * Returns the first element +ele+ in +self+ such that +ele+ is an array
5145 * and <tt>ele[0] == object</tt>:
5146 *
5147 * a = [{foo: 0}, [2, 4], [4, 5, 6], [4, 5]]
5148 * a.assoc(4) # => [4, 5, 6]
5149 *
5150 * Returns +nil+ if no such element is found.
5151 *
5152 * Related: Array#rassoc;
5153 * see also {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
5154 */
5155
5156VALUE
5157rb_ary_assoc(VALUE ary, VALUE key)
5158{
5159 long i;
5160 VALUE v;
5161
5162 for (i = 0; i < RARRAY_LEN(ary); ++i) {
5163 v = rb_check_array_type(RARRAY_AREF(ary, i));
5164 if (!NIL_P(v) && RARRAY_LEN(v) > 0 &&
5165 rb_equal(RARRAY_AREF(v, 0), key))
5166 return v;
5167 }
5168 return Qnil;
5169}
5170
5171/*
5172 * call-seq:
5173 * rassoc(object) -> found_array or nil
5174 *
5175 * Returns the first element +ele+ in +self+ such that +ele+ is an array
5176 * and <tt>ele[1] == object</tt>:
5177 *
5178 * a = [{foo: 0}, [2, 4], [4, 5, 6], [4, 5]]
5179 * a.rassoc(4) # => [2, 4]
5180 * a.rassoc(5) # => [4, 5, 6]
5181 *
5182 * Returns +nil+ if no such element is found.
5183 *
5184 * Related: Array#assoc;
5185 * see also {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
5186 */
5187
5188VALUE
5189rb_ary_rassoc(VALUE ary, VALUE value)
5190{
5191 long i;
5192 VALUE v;
5193
5194 for (i = 0; i < RARRAY_LEN(ary); ++i) {
5195 v = rb_check_array_type(RARRAY_AREF(ary, i));
5196 if (RB_TYPE_P(v, T_ARRAY) &&
5197 RARRAY_LEN(v) > 1 &&
5198 rb_equal(RARRAY_AREF(v, 1), value))
5199 return v;
5200 }
5201 return Qnil;
5202}
5203
5204static VALUE
5205recursive_equal(VALUE ary1, VALUE ary2, int recur)
5206{
5207 long i, len1;
5208 const VALUE *p1, *p2;
5209
5210 if (recur) return Qtrue; /* Subtle! */
5211
5212 /* rb_equal() can evacuate ptrs */
5213 p1 = RARRAY_CONST_PTR(ary1);
5214 p2 = RARRAY_CONST_PTR(ary2);
5215 len1 = RARRAY_LEN(ary1);
5216
5217 for (i = 0; i < len1; i++) {
5218 if (*p1 != *p2) {
5219 if (rb_equal(*p1, *p2)) {
5220 len1 = RARRAY_LEN(ary1);
5221 if (len1 != RARRAY_LEN(ary2))
5222 return Qfalse;
5223 if (len1 < i)
5224 return Qtrue;
5225 p1 = RARRAY_CONST_PTR(ary1) + i;
5226 p2 = RARRAY_CONST_PTR(ary2) + i;
5227 }
5228 else {
5229 return Qfalse;
5230 }
5231 }
5232 p1++;
5233 p2++;
5234 }
5235 return Qtrue;
5236}
5237
5238/*
5239 * call-seq:
5240 * self == other_array -> true or false
5241 *
5242 * Returns whether both:
5243 *
5244 * - +self+ and +other_array+ are the same size.
5245 * - Their corresponding elements are the same;
5246 * that is, for each index +i+ in <tt>(0...self.size)</tt>,
5247 * <tt>self[i] == other_array[i]</tt>.
5248 *
5249 * Examples:
5250 *
5251 * [:foo, 'bar', 2] == [:foo, 'bar', 2] # => true
5252 * [:foo, 'bar', 2] == [:foo, 'bar', 2.0] # => true
5253 * [:foo, 'bar', 2] == [:foo, 'bar'] # => false # Different sizes.
5254 * [:foo, 'bar', 2] == [:foo, 'bar', 3] # => false # Different elements.
5255 *
5256 * This method is different from method Array#eql?,
5257 * which compares elements using <tt>Object#eql?</tt>.
5258 *
5259 * Related: see {Methods for Comparing}[rdoc-ref:Array@Methods+for+Comparing].
5260 */
5261
5262static VALUE
5263rb_ary_equal(VALUE ary1, VALUE ary2)
5264{
5265 if (ary1 == ary2) return Qtrue;
5266 if (!RB_TYPE_P(ary2, T_ARRAY)) {
5267 if (!rb_respond_to(ary2, idTo_ary)) {
5268 return Qfalse;
5269 }
5270 return rb_equal(ary2, ary1);
5271 }
5272 if (RARRAY_LEN(ary1) != RARRAY_LEN(ary2)) return Qfalse;
5273 if (RARRAY_CONST_PTR(ary1) == RARRAY_CONST_PTR(ary2)) return Qtrue;
5274 return rb_exec_recursive_paired(recursive_equal, ary1, ary2, ary2);
5275}
5276
5277static VALUE
5278recursive_eql(VALUE ary1, VALUE ary2, int recur)
5279{
5280 long i;
5281
5282 if (recur) return Qtrue; /* Subtle! */
5283 for (i=0; i<RARRAY_LEN(ary1); i++) {
5284 if (!rb_eql(rb_ary_elt(ary1, i), rb_ary_elt(ary2, i)))
5285 return Qfalse;
5286 }
5287 return Qtrue;
5288}
5289
5290/*
5291 * call-seq:
5292 * eql?(other_array) -> true or false
5293 *
5294 * Returns +true+ if +self+ and +other_array+ are the same size,
5295 * and if, for each index +i+ in +self+, <tt>self[i].eql?(other_array[i])</tt>:
5296 *
5297 * a0 = [:foo, 'bar', 2]
5298 * a1 = [:foo, 'bar', 2]
5299 * a1.eql?(a0) # => true
5300 *
5301 * Otherwise, returns +false+.
5302 *
5303 * This method is different from method Array#==,
5304 * which compares using method <tt>Object#==</tt>.
5305 *
5306 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
5307 */
5308
5309static VALUE
5310rb_ary_eql(VALUE ary1, VALUE ary2)
5311{
5312 if (ary1 == ary2) return Qtrue;
5313 if (!RB_TYPE_P(ary2, T_ARRAY)) return Qfalse;
5314 if (RARRAY_LEN(ary1) != RARRAY_LEN(ary2)) return Qfalse;
5315 if (RARRAY_CONST_PTR(ary1) == RARRAY_CONST_PTR(ary2)) return Qtrue;
5316 return rb_exec_recursive_paired(recursive_eql, ary1, ary2, ary2);
5317}
5318
5319VALUE
5320rb_ary_hash_values(long len, const VALUE *elements)
5321{
5322 long i;
5323 st_index_t h;
5324 VALUE n;
5325
5326 h = rb_hash_start(len);
5327 h = rb_hash_uint(h, (st_index_t)rb_ary_hash_values);
5328 for (i=0; i<len; i++) {
5329 n = rb_hash(elements[i]);
5330 h = rb_hash_uint(h, NUM2LONG(n));
5331 }
5332 h = rb_hash_end(h);
5333 return ST2FIX(h);
5334}
5335
5336/*
5337 * call-seq:
5338 * hash -> integer
5339 *
5340 * Returns the integer hash value for +self+.
5341 *
5342 * Two arrays with the same content will have the same hash value
5343 * (and will compare using eql?):
5344 *
5345 * ['a', 'b'].hash == ['a', 'b'].hash # => true
5346 * ['a', 'b'].hash == ['a', 'c'].hash # => false
5347 * ['a', 'b'].hash == ['a'].hash # => false
5348 *
5349 */
5350
5351static VALUE
5352rb_ary_hash(VALUE ary)
5353{
5354 return rb_ary_hash_values(RARRAY_LEN(ary), RARRAY_CONST_PTR(ary));
5355}
5356
5357/*
5358 * call-seq:
5359 * include?(object) -> true or false
5360 *
5361 * Returns whether for some element +element+ in +self+,
5362 * <tt>object == element</tt>:
5363 *
5364 * [0, 1, 2].include?(2) # => true
5365 * [0, 1, 2].include?(2.0) # => true
5366 * [0, 1, 2].include?(2.1) # => false
5367 *
5368 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
5369 */
5370
5371VALUE
5372rb_ary_includes(VALUE ary, VALUE item)
5373{
5374 long i;
5375 VALUE e;
5376
5377 for (i=0; i<RARRAY_LEN(ary); i++) {
5378 e = RARRAY_AREF(ary, i);
5379 if (rb_equal(e, item)) {
5380 return Qtrue;
5381 }
5382 }
5383 return Qfalse;
5384}
5385
5386static VALUE
5387rb_ary_includes_by_eql(VALUE ary, VALUE item)
5388{
5389 long i;
5390 VALUE e;
5391
5392 for (i=0; i<RARRAY_LEN(ary); i++) {
5393 e = RARRAY_AREF(ary, i);
5394 if (rb_eql(item, e)) {
5395 return Qtrue;
5396 }
5397 }
5398 return Qfalse;
5399}
5400
5401static VALUE
5402recursive_cmp(VALUE ary1, VALUE ary2, int recur)
5403{
5404 long i, len;
5405
5406 if (recur) return Qundef; /* Subtle! */
5407 len = RARRAY_LEN(ary1);
5408 if (len > RARRAY_LEN(ary2)) {
5409 len = RARRAY_LEN(ary2);
5410 }
5411 for (i=0; i<len; i++) {
5412 VALUE e1 = rb_ary_elt(ary1, i), e2 = rb_ary_elt(ary2, i);
5413 VALUE v = rb_funcallv(e1, id_cmp, 1, &e2);
5414 if (v != INT2FIX(0)) {
5415 return v;
5416 }
5417 }
5418 return Qundef;
5419}
5420
5421/*
5422 * call-seq:
5423 * self <=> other_array -> -1, 0, or 1
5424 *
5425 * Returns -1, 0, or 1 as +self+ is determined
5426 * to be less than, equal to, or greater than +other_array+.
5427 *
5428 * Iterates over each index +i+ in <tt>(0...self.size)</tt>:
5429 *
5430 * - Computes <tt>result[i]</tt> as <tt>self[i] <=> other_array[i]</tt>.
5431 * - Immediately returns 1 if <tt>result[i]</tt> is 1:
5432 *
5433 * [0, 1, 2] <=> [0, 0, 2] # => 1
5434 *
5435 * - Immediately returns -1 if <tt>result[i]</tt> is -1:
5436 *
5437 * [0, 1, 2] <=> [0, 2, 2] # => -1
5438 *
5439 * - Continues if <tt>result[i]</tt> is 0.
5440 *
5441 * When every +result+ is 0,
5442 * returns <tt>self.size <=> other_array.size</tt>
5443 * (see Integer#<=>):
5444 *
5445 * [0, 1, 2] <=> [0, 1] # => 1
5446 * [0, 1, 2] <=> [0, 1, 2] # => 0
5447 * [0, 1, 2] <=> [0, 1, 2, 3] # => -1
5448 *
5449 * Note that when +other_array+ is larger than +self+,
5450 * its trailing elements do not affect the result:
5451 *
5452 * [0, 1, 2] <=> [0, 1, 2, -3] # => -1
5453 * [0, 1, 2] <=> [0, 1, 2, 0] # => -1
5454 * [0, 1, 2] <=> [0, 1, 2, 3] # => -1
5455 *
5456 * Related: see {Methods for Comparing}[rdoc-ref:Array@Methods+for+Comparing].
5457 */
5458
5459VALUE
5460rb_ary_cmp(VALUE ary1, VALUE ary2)
5461{
5462 long len;
5463 VALUE v;
5464
5465 ary2 = rb_check_array_type(ary2);
5466 if (NIL_P(ary2)) return Qnil;
5467 if (ary1 == ary2) return INT2FIX(0);
5468 v = rb_exec_recursive_paired(recursive_cmp, ary1, ary2, ary2);
5469 if (!UNDEF_P(v)) return v;
5470 len = RARRAY_LEN(ary1) - RARRAY_LEN(ary2);
5471 if (len == 0) return INT2FIX(0);
5472 if (len > 0) return INT2FIX(1);
5473 return INT2FIX(-1);
5474}
5475
5476static VALUE
5477ary_add_hash(VALUE hash, VALUE ary)
5478{
5479 long i;
5480
5481 for (i=0; i<RARRAY_LEN(ary); i++) {
5482 VALUE elt = RARRAY_AREF(ary, i);
5483 rb_hash_add_new_element(hash, elt, elt);
5484 }
5485 return hash;
5486}
5487
5488static inline VALUE
5489ary_tmp_hash_new(VALUE ary)
5490{
5491 long size = RARRAY_LEN(ary);
5492 VALUE hash = rb_hash_new_with_size(size);
5493
5494 RBASIC_CLEAR_CLASS(hash);
5495 return hash;
5496}
5497
5498static VALUE
5499ary_make_hash(VALUE ary)
5500{
5501 VALUE hash = ary_tmp_hash_new(ary);
5502 return ary_add_hash(hash, ary);
5503}
5504
5505static VALUE
5506ary_add_hash_by(VALUE hash, VALUE ary)
5507{
5508 long i;
5509
5510 for (i = 0; i < RARRAY_LEN(ary); ++i) {
5511 VALUE v = rb_ary_elt(ary, i), k = rb_yield(v);
5512 rb_hash_add_new_element(hash, k, v);
5513 }
5514 return hash;
5515}
5516
5517static VALUE
5518ary_make_hash_by(VALUE ary)
5519{
5520 VALUE hash = ary_tmp_hash_new(ary);
5521 return ary_add_hash_by(hash, ary);
5522}
5523
5524/*
5525 * call-seq:
5526 * self - other_array -> new_array
5527 *
5528 * Returns a new array containing only those elements of +self+
5529 * that are not found in +other_array+;
5530 * the order from +self+ is preserved:
5531 *
5532 * [0, 1, 1, 2, 1, 1, 3, 1, 1] - [1] # => [0, 2, 3]
5533 * [0, 1, 1, 2, 1, 1, 3, 1, 1] - [3, 2, 0, :foo] # => [1, 1, 1, 1, 1, 1]
5534 * [0, 1, 2] - [:foo] # => [0, 1, 2]
5535 *
5536 * Element are compared using method <tt>#eql?</tt>
5537 * (as defined in each element of +self+).
5538 *
5539 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
5540 */
5541
5542VALUE
5543rb_ary_diff(VALUE ary1, VALUE ary2)
5544{
5545 VALUE ary3;
5546 VALUE hash;
5547 long i;
5548
5549 ary2 = to_ary(ary2);
5550 if (RARRAY_LEN(ary2) == 0) { return ary_make_shared_copy(ary1); }
5551 ary3 = rb_ary_new();
5552
5553 if (RARRAY_LEN(ary1) <= SMALL_ARRAY_LEN || RARRAY_LEN(ary2) <= SMALL_ARRAY_LEN) {
5554 for (i=0; i<RARRAY_LEN(ary1); i++) {
5555 VALUE elt = rb_ary_elt(ary1, i);
5556 if (rb_ary_includes_by_eql(ary2, elt)) continue;
5557 rb_ary_push(ary3, elt);
5558 }
5559 return ary3;
5560 }
5561
5562 hash = ary_make_hash(ary2);
5563 for (i=0; i<RARRAY_LEN(ary1); i++) {
5564 if (rb_hash_stlike_lookup(hash, RARRAY_AREF(ary1, i), NULL)) continue;
5565 rb_ary_push(ary3, rb_ary_elt(ary1, i));
5566 }
5567
5568 return ary3;
5569}
5570
5571/*
5572 * call-seq:
5573 * difference(*other_arrays = []) -> new_array
5574 *
5575 * Returns a new array containing only those elements from +self+
5576 * that are not found in any of the given +other_arrays+;
5577 * items are compared using <tt>eql?</tt>; order from +self+ is preserved:
5578 *
5579 * [0, 1, 1, 2, 1, 1, 3, 1, 1].difference([1]) # => [0, 2, 3]
5580 * [0, 1, 2, 3].difference([3, 0], [1, 3]) # => [2]
5581 * [0, 1, 2].difference([4]) # => [0, 1, 2]
5582 * [0, 1, 2].difference # => [0, 1, 2]
5583 *
5584 * Returns a copy of +self+ if no arguments are given.
5585 *
5586 * Related: Array#-;
5587 * see also {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
5588 */
5589
5590static VALUE
5591rb_ary_difference_multi(int argc, VALUE *argv, VALUE ary)
5592{
5593 VALUE ary_diff;
5594 long i, length;
5595 volatile VALUE t0;
5596 bool *is_hash = ALLOCV_N(bool, t0, argc);
5597 ary_diff = rb_ary_new();
5598 length = RARRAY_LEN(ary);
5599
5600 for (i = 0; i < argc; i++) {
5601 argv[i] = to_ary(argv[i]);
5602 is_hash[i] = (length > SMALL_ARRAY_LEN && RARRAY_LEN(argv[i]) > SMALL_ARRAY_LEN);
5603 if (is_hash[i]) argv[i] = ary_make_hash(argv[i]);
5604 }
5605
5606 for (i = 0; i < RARRAY_LEN(ary); i++) {
5607 int j;
5608 VALUE elt = rb_ary_elt(ary, i);
5609 for (j = 0; j < argc; j++) {
5610 if (is_hash[j]) {
5611 if (rb_hash_stlike_lookup(argv[j], RARRAY_AREF(ary, i), NULL))
5612 break;
5613 }
5614 else {
5615 if (rb_ary_includes_by_eql(argv[j], elt)) break;
5616 }
5617 }
5618 if (j == argc) rb_ary_push(ary_diff, elt);
5619 }
5620
5621 ALLOCV_END(t0);
5622
5623 return ary_diff;
5624}
5625
5626
5627/*
5628 * call-seq:
5629 * self & other_array -> new_array
5630 *
5631 * Returns a new array containing the _intersection_ of +self+ and +other_array+;
5632 * that is, containing those elements found in both +self+ and +other_array+:
5633 *
5634 * [0, 1, 2, 3] & [1, 2] # => [1, 2]
5635 *
5636 * Omits duplicates:
5637 *
5638 * [0, 1, 1, 0] & [0, 1] # => [0, 1]
5639 *
5640 * Preserves order from +self+:
5641 *
5642 * [0, 1, 2] & [3, 2, 1, 0] # => [0, 1, 2]
5643 *
5644 * Identifies common elements using method <tt>#eql?</tt>
5645 * (as defined in each element of +self+).
5646 *
5647 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
5648 */
5649
5650
5651static VALUE
5652rb_ary_and(VALUE ary1, VALUE ary2)
5653{
5654 VALUE hash, ary3, v;
5655 st_data_t vv;
5656 long i;
5657
5658 ary2 = to_ary(ary2);
5659 ary3 = rb_ary_new();
5660 if (RARRAY_LEN(ary1) == 0 || RARRAY_LEN(ary2) == 0) return ary3;
5661
5662 if (RARRAY_LEN(ary1) <= SMALL_ARRAY_LEN && RARRAY_LEN(ary2) <= SMALL_ARRAY_LEN) {
5663 for (i=0; i<RARRAY_LEN(ary1); i++) {
5664 v = RARRAY_AREF(ary1, i);
5665 if (!rb_ary_includes_by_eql(ary2, v)) continue;
5666 if (rb_ary_includes_by_eql(ary3, v)) continue;
5667 rb_ary_push(ary3, v);
5668 }
5669 return ary3;
5670 }
5671
5672 hash = ary_make_hash(ary2);
5673
5674 for (i=0; i<RARRAY_LEN(ary1); i++) {
5675 v = RARRAY_AREF(ary1, i);
5676 vv = (st_data_t)v;
5677 if (rb_hash_stlike_delete(hash, &vv, 0)) {
5678 rb_ary_push(ary3, v);
5679 }
5680 }
5681
5682 return ary3;
5683}
5684
5685/*
5686 * call-seq:
5687 * intersection(*other_arrays) -> new_array
5688 *
5689 * Returns a new array containing each element in +self+ that is +#eql?+
5690 * to at least one element in each of the given +other_arrays+;
5691 * duplicates are omitted:
5692 *
5693 * [0, 0, 1, 1, 2, 3].intersection([0, 1, 2], [0, 1, 3]) # => [0, 1]
5694 *
5695 * Each element must correctly implement method <tt>#hash</tt>.
5696 *
5697 * Order from +self+ is preserved:
5698 *
5699 * [0, 1, 2].intersection([2, 1, 0]) # => [0, 1, 2]
5700 *
5701 * Returns a copy of +self+ if no arguments are given.
5702 *
5703 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
5704 */
5705
5706static VALUE
5707rb_ary_intersection_multi(int argc, VALUE *argv, VALUE ary)
5708{
5709 VALUE result = rb_ary_dup(ary);
5710 int i;
5711
5712 for (i = 0; i < argc; i++) {
5713 result = rb_ary_and(result, argv[i]);
5714 }
5715
5716 return result;
5717}
5718
5719static int
5720ary_hash_orset(st_data_t *key, st_data_t *value, st_data_t arg, int existing)
5721{
5722 if (existing) return ST_STOP;
5723 *key = *value = (VALUE)arg;
5724 return ST_CONTINUE;
5725}
5726
5727static void
5728rb_ary_union(VALUE ary_union, VALUE ary)
5729{
5730 long i;
5731 for (i = 0; i < RARRAY_LEN(ary); i++) {
5732 VALUE elt = rb_ary_elt(ary, i);
5733 if (rb_ary_includes_by_eql(ary_union, elt)) continue;
5734 rb_ary_push(ary_union, elt);
5735 }
5736}
5737
5738static void
5739rb_ary_union_hash(VALUE hash, VALUE ary2)
5740{
5741 long i;
5742 for (i = 0; i < RARRAY_LEN(ary2); i++) {
5743 VALUE elt = RARRAY_AREF(ary2, i);
5744 if (!rb_hash_stlike_update(hash, (st_data_t)elt, ary_hash_orset, (st_data_t)elt)) {
5745 RB_OBJ_WRITTEN(hash, Qundef, elt);
5746 }
5747 }
5748}
5749
5750/*
5751 * call-seq:
5752 * self | other_array -> new_array
5753 *
5754 * Returns the union of +self+ and +other_array+;
5755 * duplicates are removed; order is preserved;
5756 * items are compared using <tt>eql?</tt>:
5757 *
5758 * [0, 1] | [2, 3] # => [0, 1, 2, 3]
5759 * [0, 1, 1] | [2, 2, 3] # => [0, 1, 2, 3]
5760 * [0, 1, 2] | [3, 2, 1, 0] # => [0, 1, 2, 3]
5761 *
5762 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
5763 */
5764
5765static VALUE
5766rb_ary_or(VALUE ary1, VALUE ary2)
5767{
5768 VALUE hash;
5769
5770 ary2 = to_ary(ary2);
5771 if (RARRAY_LEN(ary1) + RARRAY_LEN(ary2) <= SMALL_ARRAY_LEN) {
5772 VALUE ary3 = rb_ary_new();
5773 rb_ary_union(ary3, ary1);
5774 rb_ary_union(ary3, ary2);
5775 return ary3;
5776 }
5777
5778 hash = ary_make_hash(ary1);
5779 rb_ary_union_hash(hash, ary2);
5780
5781 return rb_hash_values(hash);
5782}
5783
5784/*
5785 * call-seq:
5786 * union(*other_arrays) -> new_array
5787 *
5788 * Returns a new array that is the union of the elements of +self+
5789 * and all given arrays +other_arrays+;
5790 * items are compared using <tt>eql?</tt>:
5791 *
5792 * [0, 1, 2, 3].union([4, 5], [6, 7]) # => [0, 1, 2, 3, 4, 5, 6, 7]
5793 *
5794 * Removes duplicates (preserving the first found):
5795 *
5796 * [0, 1, 1].union([2, 1], [3, 1]) # => [0, 1, 2, 3]
5797 *
5798 * Preserves order (preserving the position of the first found):
5799 *
5800 * [3, 2, 1, 0].union([5, 3], [4, 2]) # => [3, 2, 1, 0, 5, 4]
5801 *
5802 * With no arguments given, returns a copy of +self+.
5803 *
5804 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
5805 */
5806
5807static VALUE
5808rb_ary_union_multi(int argc, VALUE *argv, VALUE ary)
5809{
5810 int i;
5811 long sum;
5812 VALUE hash;
5813
5814 sum = RARRAY_LEN(ary);
5815 for (i = 0; i < argc; i++) {
5816 argv[i] = to_ary(argv[i]);
5817 sum += RARRAY_LEN(argv[i]);
5818 }
5819
5820 if (sum <= SMALL_ARRAY_LEN) {
5821 VALUE ary_union = rb_ary_new();
5822
5823 rb_ary_union(ary_union, ary);
5824 for (i = 0; i < argc; i++) rb_ary_union(ary_union, argv[i]);
5825
5826 return ary_union;
5827 }
5828
5829 hash = ary_make_hash(ary);
5830 for (i = 0; i < argc; i++) rb_ary_union_hash(hash, argv[i]);
5831
5832 return rb_hash_values(hash);
5833}
5834
5835/*
5836 * call-seq:
5837 * intersect?(other_array) -> true or false
5838 *
5839 * Returns whether +other_array+ has at least one element that is +#eql?+ to some element of +self+:
5840 *
5841 * [1, 2, 3].intersect?([3, 4, 5]) # => true
5842 * [1, 2, 3].intersect?([4, 5, 6]) # => false
5843 *
5844 * Each element must correctly implement method <tt>#hash</tt>.
5845 *
5846 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
5847 */
5848
5849static VALUE
5850rb_ary_intersect_p(VALUE ary1, VALUE ary2)
5851{
5852 VALUE hash, v, result, shorter, longer;
5853 st_data_t vv;
5854 long i;
5855
5856 ary2 = to_ary(ary2);
5857 if (RARRAY_LEN(ary1) == 0 || RARRAY_LEN(ary2) == 0) return Qfalse;
5858
5859 if (RARRAY_LEN(ary1) <= SMALL_ARRAY_LEN && RARRAY_LEN(ary2) <= SMALL_ARRAY_LEN) {
5860 for (i=0; i<RARRAY_LEN(ary1); i++) {
5861 v = RARRAY_AREF(ary1, i);
5862 if (rb_ary_includes_by_eql(ary2, v)) return Qtrue;
5863 }
5864 return Qfalse;
5865 }
5866
5867 shorter = ary1;
5868 longer = ary2;
5869 if (RARRAY_LEN(ary1) > RARRAY_LEN(ary2)) {
5870 longer = ary1;
5871 shorter = ary2;
5872 }
5873
5874 hash = ary_make_hash(shorter);
5875 result = Qfalse;
5876
5877 for (i=0; i<RARRAY_LEN(longer); i++) {
5878 v = RARRAY_AREF(longer, i);
5879 vv = (st_data_t)v;
5880 if (rb_hash_stlike_lookup(hash, vv, 0)) {
5881 result = Qtrue;
5882 break;
5883 }
5884 }
5885
5886 return result;
5887}
5888
5889static VALUE
5890ary_max_generic(VALUE ary, long i, VALUE vmax)
5891{
5892 RUBY_ASSERT(i > 0 && i < RARRAY_LEN(ary));
5893
5894 VALUE v;
5895 for (; i < RARRAY_LEN(ary); ++i) {
5896 v = RARRAY_AREF(ary, i);
5897
5898 if (rb_cmpint(rb_funcallv(vmax, id_cmp, 1, &v), vmax, v) < 0) {
5899 vmax = v;
5900 }
5901 }
5902
5903 return vmax;
5904}
5905
5906static VALUE
5907ary_max_opt_fixnum(VALUE ary, long i, VALUE vmax)
5908{
5909 const long n = RARRAY_LEN(ary);
5910 RUBY_ASSERT(i > 0 && i < n);
5911 RUBY_ASSERT(FIXNUM_P(vmax));
5912
5913 VALUE v;
5914 for (; i < n; ++i) {
5915 v = RARRAY_AREF(ary, i);
5916
5917 if (FIXNUM_P(v)) {
5918 if ((long)vmax < (long)v) {
5919 vmax = v;
5920 }
5921 }
5922 else {
5923 return ary_max_generic(ary, i, vmax);
5924 }
5925 }
5926
5927 return vmax;
5928}
5929
5930static VALUE
5931ary_max_opt_float(VALUE ary, long i, VALUE vmax)
5932{
5933 const long n = RARRAY_LEN(ary);
5934 RUBY_ASSERT(i > 0 && i < n);
5936
5937 VALUE v;
5938 for (; i < n; ++i) {
5939 v = RARRAY_AREF(ary, i);
5940
5941 if (RB_FLOAT_TYPE_P(v)) {
5942 if (rb_float_cmp(vmax, v) < 0) {
5943 vmax = v;
5944 }
5945 }
5946 else {
5947 return ary_max_generic(ary, i, vmax);
5948 }
5949 }
5950
5951 return vmax;
5952}
5953
5954static VALUE
5955ary_max_opt_string(VALUE ary, long i, VALUE vmax)
5956{
5957 const long n = RARRAY_LEN(ary);
5958 RUBY_ASSERT(i > 0 && i < n);
5959 RUBY_ASSERT(STRING_P(vmax));
5960
5961 VALUE v;
5962 for (; i < n; ++i) {
5963 v = RARRAY_AREF(ary, i);
5964
5965 if (STRING_P(v)) {
5966 if (rb_str_cmp(vmax, v) < 0) {
5967 vmax = v;
5968 }
5969 }
5970 else {
5971 return ary_max_generic(ary, i, vmax);
5972 }
5973 }
5974
5975 return vmax;
5976}
5977
5978/*
5979 * call-seq:
5980 * max -> element
5981 * max(count) -> new_array
5982 * max {|a, b| ... } -> element
5983 * max(count) {|a, b| ... } -> new_array
5984 *
5985 * Returns one of the following:
5986 *
5987 * - The maximum-valued element from +self+.
5988 * - A new array of maximum-valued elements from +self+.
5989 *
5990 * Does not modify +self+.
5991 *
5992 * With no block given, each element in +self+ must respond to method <tt>#<=></tt>
5993 * with a numeric.
5994 *
5995 * With no argument and no block, returns the element in +self+
5996 * having the maximum value per method <tt>#<=></tt>:
5997 *
5998 * [1, 0, 3, 2].max # => 3
5999 *
6000 * With non-negative numeric argument +count+ and no block,
6001 * returns a new array with at most +count+ elements,
6002 * in descending order, per method <tt>#<=></tt>:
6003 *
6004 * [1, 0, 3, 2].max(3) # => [3, 2, 1]
6005 * [1, 0, 3, 2].max(3.0) # => [3, 2, 1]
6006 * [1, 0, 3, 2].max(9) # => [3, 2, 1, 0]
6007 * [1, 0, 3, 2].max(0) # => []
6008 *
6009 * With a block given, the block must return a numeric.
6010 *
6011 * With a block and no argument, calls the block <tt>self.size - 1</tt> times to compare elements;
6012 * returns the element having the maximum value per the block:
6013 *
6014 * ['0', '', '000', '00'].max {|a, b| a.size <=> b.size }
6015 * # => "000"
6016 *
6017 * With non-negative numeric argument +count+ and a block,
6018 * returns a new array with at most +count+ elements,
6019 * in descending order, per the block:
6020 *
6021 * ['0', '', '000', '00'].max(2) {|a, b| a.size <=> b.size }
6022 * # => ["000", "00"]
6023 *
6024 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
6025 */
6026static VALUE
6027rb_ary_max(int argc, VALUE *argv, VALUE ary)
6028{
6029 VALUE result = Qundef, v;
6030 VALUE num;
6031 long i;
6032
6033 if (rb_check_arity(argc, 0, 1) && !NIL_P(num = argv[0]))
6034 return rb_nmin_run(ary, num, 0, 1, 1);
6035
6036 const long n = RARRAY_LEN(ary);
6037 if (rb_block_given_p()) {
6038 for (i = 0; i < RARRAY_LEN(ary); i++) {
6039 v = RARRAY_AREF(ary, i);
6040 if (UNDEF_P(result) || rb_cmpint(rb_yield_values(2, v, result), v, result) > 0) {
6041 result = v;
6042 }
6043 }
6044 }
6045 else if (n > 0) {
6046 result = RARRAY_AREF(ary, 0);
6047 if (n > 1) {
6048 if (FIXNUM_P(result) && CMP_OPTIMIZABLE(INTEGER)) {
6049 return ary_max_opt_fixnum(ary, 1, result);
6050 }
6051 else if (STRING_P(result) && CMP_OPTIMIZABLE(STRING)) {
6052 return ary_max_opt_string(ary, 1, result);
6053 }
6054 else if (RB_FLOAT_TYPE_P(result) && CMP_OPTIMIZABLE(FLOAT)) {
6055 return ary_max_opt_float(ary, 1, result);
6056 }
6057 else {
6058 return ary_max_generic(ary, 1, result);
6059 }
6060 }
6061 }
6062 if (UNDEF_P(result)) return Qnil;
6063 return result;
6064}
6065
6066static VALUE
6067ary_min_generic(VALUE ary, long i, VALUE vmin)
6068{
6069 RUBY_ASSERT(i > 0 && i < RARRAY_LEN(ary));
6070
6071 VALUE v;
6072 for (; i < RARRAY_LEN(ary); ++i) {
6073 v = RARRAY_AREF(ary, i);
6074
6075 if (rb_cmpint(rb_funcallv(vmin, id_cmp, 1, &v), vmin, v) > 0) {
6076 vmin = v;
6077 }
6078 }
6079
6080 return vmin;
6081}
6082
6083static VALUE
6084ary_min_opt_fixnum(VALUE ary, long i, VALUE vmin)
6085{
6086 const long n = RARRAY_LEN(ary);
6087 RUBY_ASSERT(i > 0 && i < n);
6088 RUBY_ASSERT(FIXNUM_P(vmin));
6089
6090 VALUE a;
6091 for (; i < n; ++i) {
6092 a = RARRAY_AREF(ary, i);
6093
6094 if (FIXNUM_P(a)) {
6095 if ((long)vmin > (long)a) {
6096 vmin = a;
6097 }
6098 }
6099 else {
6100 return ary_min_generic(ary, i, vmin);
6101 }
6102 }
6103
6104 return vmin;
6105}
6106
6107static VALUE
6108ary_min_opt_float(VALUE ary, long i, VALUE vmin)
6109{
6110 const long n = RARRAY_LEN(ary);
6111 RUBY_ASSERT(i > 0 && i < n);
6113
6114 VALUE a;
6115 for (; i < n; ++i) {
6116 a = RARRAY_AREF(ary, i);
6117
6118 if (RB_FLOAT_TYPE_P(a)) {
6119 if (rb_float_cmp(vmin, a) > 0) {
6120 vmin = a;
6121 }
6122 }
6123 else {
6124 return ary_min_generic(ary, i, vmin);
6125 }
6126 }
6127
6128 return vmin;
6129}
6130
6131static VALUE
6132ary_min_opt_string(VALUE ary, long i, VALUE vmin)
6133{
6134 const long n = RARRAY_LEN(ary);
6135 RUBY_ASSERT(i > 0 && i < n);
6136 RUBY_ASSERT(STRING_P(vmin));
6137
6138 VALUE a;
6139 for (; i < n; ++i) {
6140 a = RARRAY_AREF(ary, i);
6141
6142 if (STRING_P(a)) {
6143 if (rb_str_cmp(vmin, a) > 0) {
6144 vmin = a;
6145 }
6146 }
6147 else {
6148 return ary_min_generic(ary, i, vmin);
6149 }
6150 }
6151
6152 return vmin;
6153}
6154
6155/*
6156 * call-seq:
6157 * min -> element
6158 * min(count) -> new_array
6159 * min {|a, b| ... } -> element
6160 * min(count) {|a, b| ... } -> new_array
6161 *
6162 * Returns one of the following:
6163 *
6164 * - The minimum-valued element from +self+.
6165 * - A new array of minimum-valued elements from +self+.
6166 *
6167 * Does not modify +self+.
6168 *
6169 * With no block given, each element in +self+ must respond to method <tt>#<=></tt>
6170 * with a numeric.
6171 *
6172 * With no argument and no block, returns the element in +self+
6173 * having the minimum value per method <tt>#<=></tt>:
6174 *
6175 * [1, 0, 3, 2].min # => 0
6176 *
6177 * With non-negative numeric argument +count+ and no block,
6178 * returns a new array with at most +count+ elements,
6179 * in ascending order, per method <tt>#<=></tt>:
6180 *
6181 * [1, 0, 3, 2].min(3) # => [0, 1, 2]
6182 * [1, 0, 3, 2].min(3.0) # => [0, 1, 2]
6183 * [1, 0, 3, 2].min(9) # => [0, 1, 2, 3]
6184 * [1, 0, 3, 2].min(0) # => []
6185 *
6186 * With a block given, the block must return a numeric.
6187 *
6188 * With a block and no argument, calls the block <tt>self.size - 1</tt> times to compare elements;
6189 * returns the element having the minimum value per the block:
6190 *
6191 * ['0', '', '000', '00'].min {|a, b| a.size <=> b.size }
6192 * # => ""
6193 *
6194 * With non-negative numeric argument +count+ and a block,
6195 * returns a new array with at most +count+ elements,
6196 * in ascending order, per the block:
6197 *
6198 * ['0', '', '000', '00'].min(2) {|a, b| a.size <=> b.size }
6199 * # => ["", "0"]
6200 *
6201 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
6202 */
6203static VALUE
6204rb_ary_min(int argc, VALUE *argv, VALUE ary)
6205{
6206 VALUE result = Qundef, v;
6207 VALUE num;
6208 long i;
6209
6210 if (rb_check_arity(argc, 0, 1) && !NIL_P(num = argv[0]))
6211 return rb_nmin_run(ary, num, 0, 0, 1);
6212
6213 const long n = RARRAY_LEN(ary);
6214 if (rb_block_given_p()) {
6215 for (i = 0; i < RARRAY_LEN(ary); i++) {
6216 v = RARRAY_AREF(ary, i);
6217 if (UNDEF_P(result) || rb_cmpint(rb_yield_values(2, v, result), v, result) < 0) {
6218 result = v;
6219 }
6220 }
6221 }
6222 else if (n > 0) {
6223 result = RARRAY_AREF(ary, 0);
6224 if (n > 1) {
6225 if (FIXNUM_P(result) && CMP_OPTIMIZABLE(INTEGER)) {
6226 return ary_min_opt_fixnum(ary, 1, result);
6227 }
6228 else if (STRING_P(result) && CMP_OPTIMIZABLE(STRING)) {
6229 return ary_min_opt_string(ary, 1, result);
6230 }
6231 else if (RB_FLOAT_TYPE_P(result) && CMP_OPTIMIZABLE(FLOAT)) {
6232 return ary_min_opt_float(ary, 1, result);
6233 }
6234 else {
6235 return ary_min_generic(ary, 1, result);
6236 }
6237 }
6238 }
6239 if (UNDEF_P(result)) return Qnil;
6240 return result;
6241}
6242
6243/*
6244 * call-seq:
6245 * minmax -> array
6246 * minmax {|a, b| ... } -> array
6247 *
6248 * Returns a 2-element array containing the minimum-valued and maximum-valued
6249 * elements from +self+;
6250 * does not modify +self+.
6251 *
6252 * With no block given, the minimum and maximum values are determined using method <tt>#<=></tt>:
6253 *
6254 * [1, 0, 3, 2].minmax # => [0, 3]
6255 *
6256 * With a block given, the block must return a numeric;
6257 * the block is called <tt>self.size - 1</tt> times to compare elements;
6258 * returns the elements having the minimum and maximum values per the block:
6259 *
6260 * ['0', '', '000', '00'].minmax {|a, b| a.size <=> b.size }
6261 * # => ["", "000"]
6262 *
6263 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
6264 */
6265static VALUE
6266rb_ary_minmax(VALUE ary)
6267{
6268 if (rb_block_given_p()) {
6269 return rb_call_super(0, NULL);
6270 }
6271 return rb_assoc_new(rb_ary_min(0, 0, ary), rb_ary_max(0, 0, ary));
6272}
6273
6274static int
6275push_value(st_data_t key, st_data_t val, st_data_t ary)
6276{
6277 rb_ary_push((VALUE)ary, (VALUE)val);
6278 return ST_CONTINUE;
6279}
6280
6281/*
6282 * call-seq:
6283 * uniq! -> self or nil
6284 * uniq! {|element| ... } -> self or nil
6285 *
6286 * Removes duplicate elements from +self+, the first occurrence always being retained;
6287 * returns +self+ if any elements removed, +nil+ otherwise.
6288 *
6289 * With no block given, identifies and removes elements using method <tt>eql?</tt>
6290 * to compare elements:
6291 *
6292 * a = [0, 0, 1, 1, 2, 2]
6293 * a.uniq! # => [0, 1, 2]
6294 * a.uniq! # => nil
6295 *
6296 * With a block given, calls the block for each element;
6297 * identifies and omits "duplicate" elements using method <tt>eql?</tt>
6298 * to compare <i>block return values</i>;
6299 * that is, an element is a duplicate if its block return value
6300 * is the same as that of a previous element:
6301 *
6302 * a = ['a', 'aa', 'aaa', 'b', 'bb', 'bbb']
6303 * a.uniq! {|element| element.size } # => ["a", "aa", "aaa"]
6304 * a.uniq! {|element| element.size } # => nil
6305 *
6306 * Related: see {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
6307 */
6308static VALUE
6309rb_ary_uniq_bang(VALUE ary)
6310{
6311 VALUE hash;
6312 long hash_size;
6313
6314 rb_ary_modify_check(ary);
6315 if (RARRAY_LEN(ary) <= 1)
6316 return Qnil;
6317 if (rb_block_given_p())
6318 hash = ary_make_hash_by(ary);
6319 else
6320 hash = ary_make_hash(ary);
6321
6322 hash_size = RHASH_SIZE(hash);
6323 if (RARRAY_LEN(ary) == hash_size) {
6324 return Qnil;
6325 }
6326 rb_ary_modify_check(ary);
6327 ARY_SET_LEN(ary, 0);
6328 if (ARY_SHARED_P(ary)) {
6329 rb_ary_unshare(ary);
6330 FL_SET_EMBED(ary);
6331 }
6332 ary_resize_capa(ary, hash_size);
6333 rb_hash_foreach(hash, push_value, ary);
6334
6335 return ary;
6336}
6337
6338/*
6339 * call-seq:
6340 * uniq -> new_array
6341 * uniq {|element| ... } -> new_array
6342 *
6343 * Returns a new array containing those elements from +self+ that are not duplicates,
6344 * the first occurrence always being retained.
6345 *
6346 * With no block given, identifies and omits duplicate elements using method <tt>eql?</tt>
6347 * to compare elements:
6348 *
6349 * a = [0, 0, 1, 1, 2, 2]
6350 * a.uniq # => [0, 1, 2]
6351 *
6352 * With a block given, calls the block for each element;
6353 * identifies and omits "duplicate" elements using method <tt>eql?</tt>
6354 * to compare <i>block return values</i>;
6355 * that is, an element is a duplicate if its block return value
6356 * is the same as that of a previous element:
6357 *
6358 * a = ['a', 'aa', 'aaa', 'b', 'bb', 'bbb']
6359 * a.uniq {|element| element.size } # => ["a", "aa", "aaa"]
6360 *
6361 * Related: {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
6362 */
6363
6364static VALUE
6365rb_ary_uniq(VALUE ary)
6366{
6367 VALUE hash, uniq;
6368
6369 if (RARRAY_LEN(ary) <= 1) {
6370 hash = 0;
6371 uniq = rb_ary_dup(ary);
6372 }
6373 else if (rb_block_given_p()) {
6374 hash = ary_make_hash_by(ary);
6375 uniq = rb_hash_values(hash);
6376 }
6377 else {
6378 hash = ary_make_hash(ary);
6379 uniq = rb_hash_values(hash);
6380 }
6381
6382 return uniq;
6383}
6384
6385/*
6386 * call-seq:
6387 * compact! -> self or nil
6388 *
6389 * Removes all +nil+ elements from +self+;
6390 * Returns +self+ if any elements are removed, +nil+ otherwise:
6391 *
6392 * a = [nil, 0, nil, false, nil, '', nil, [], nil, {}]
6393 * a.compact! # => [0, false, "", [], {}]
6394 * a # => [0, false, "", [], {}]
6395 * a.compact! # => nil
6396 *
6397 * Related: Array#compact;
6398 * see also {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
6399 */
6400
6401static VALUE
6402rb_ary_compact_bang(VALUE ary)
6403{
6404 VALUE *p, *t, *end;
6405 long n;
6406
6407 rb_ary_modify(ary);
6408 p = t = (VALUE *)RARRAY_CONST_PTR(ary); /* WB: no new reference */
6409 end = p + RARRAY_LEN(ary);
6410
6411 while (t < end) {
6412 if (NIL_P(*t)) t++;
6413 else *p++ = *t++;
6414 }
6415 n = p - RARRAY_CONST_PTR(ary);
6416 if (RARRAY_LEN(ary) == n) {
6417 return Qnil;
6418 }
6419 ary_resize_smaller(ary, n);
6420
6421 return ary;
6422}
6423
6424/*
6425 * call-seq:
6426 * compact -> new_array
6427 *
6428 * Returns a new array containing only the non-+nil+ elements from +self+;
6429 * element order is preserved:
6430 *
6431 * a = [nil, 0, nil, false, nil, '', nil, [], nil, {}]
6432 * a.compact # => [0, false, "", [], {}]
6433 *
6434 * Related: Array#compact!;
6435 * see also {Methods for Deleting}[rdoc-ref:Array@Methods+for+Deleting].
6436 */
6437
6438static VALUE
6439rb_ary_compact(VALUE ary)
6440{
6441 ary = rb_ary_dup(ary);
6442 rb_ary_compact_bang(ary);
6443 return ary;
6444}
6445
6446/*
6447 * call-seq:
6448 * count -> integer
6449 * count(object) -> integer
6450 * count {|element| ... } -> integer
6451 *
6452 * Returns a count of specified elements.
6453 *
6454 * With no argument and no block, returns the count of all elements:
6455 *
6456 * [0, :one, 'two', 3, 3.0].count # => 5
6457 *
6458 * With argument +object+ given, returns the count of elements <tt>==</tt> to +object+:
6459 *
6460 * [0, :one, 'two', 3, 3.0].count(3) # => 2
6461 *
6462 * With no argument and a block given, calls the block with each element;
6463 * returns the count of elements for which the block returns a truthy value:
6464 *
6465 * [0, 1, 2, 3].count {|element| element > 1 } # => 2
6466 *
6467 * With argument +object+ and a block given, issues a warning, ignores the block,
6468 * and returns the count of elements <tt>==</tt> to +object+.
6469 *
6470 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
6471 */
6472
6473static VALUE
6474rb_ary_count(int argc, VALUE *argv, VALUE ary)
6475{
6476 long i, n = 0;
6477
6478 if (rb_check_arity(argc, 0, 1) == 0) {
6479 VALUE v;
6480
6481 if (!rb_block_given_p())
6482 return LONG2NUM(RARRAY_LEN(ary));
6483
6484 for (i = 0; i < RARRAY_LEN(ary); i++) {
6485 v = RARRAY_AREF(ary, i);
6486 if (RTEST(rb_yield(v))) n++;
6487 }
6488 }
6489 else {
6490 VALUE obj = argv[0];
6491
6492 if (rb_block_given_p()) {
6493 rb_warn("given block not used");
6494 }
6495 for (i = 0; i < RARRAY_LEN(ary); i++) {
6496 if (rb_equal(RARRAY_AREF(ary, i), obj)) n++;
6497 }
6498 }
6499
6500 return LONG2NUM(n);
6501}
6502
6503static VALUE
6504flatten(VALUE ary, int level)
6505{
6506 long i;
6507 VALUE stack, result, tmp = 0, elt;
6508 VALUE memo = Qfalse;
6509
6510 for (i = 0; i < RARRAY_LEN(ary); i++) {
6511 elt = RARRAY_AREF(ary, i);
6512 tmp = rb_check_array_type(elt);
6513 if (!NIL_P(tmp)) {
6514 break;
6515 }
6516 }
6517 if (i == RARRAY_LEN(ary)) {
6518 return ary;
6519 }
6520
6521 result = ary_new(0, RARRAY_LEN(ary));
6522 ary_memcpy(result, 0, i, RARRAY_CONST_PTR(ary));
6523 ARY_SET_LEN(result, i);
6524
6525 stack = ary_new(0, ARY_DEFAULT_SIZE);
6526 rb_ary_push(stack, ary);
6527 rb_ary_push(stack, LONG2NUM(i + 1));
6528
6529 if (level < 0) {
6530 memo = rb_obj_hide(rb_ident_hash_new());
6531 rb_hash_aset(memo, ary, Qtrue);
6532 rb_hash_aset(memo, tmp, Qtrue);
6533 }
6534
6535 ary = tmp;
6536 i = 0;
6537
6538 while (1) {
6539 while (i < RARRAY_LEN(ary)) {
6540 elt = RARRAY_AREF(ary, i++);
6541 if (level >= 0 && RARRAY_LEN(stack) / 2 >= level) {
6542 rb_ary_push(result, elt);
6543 continue;
6544 }
6545 tmp = rb_check_array_type(elt);
6546 if (RBASIC(result)->klass) {
6547 if (RTEST(memo)) {
6548 rb_hash_clear(memo);
6549 }
6550 rb_raise(rb_eRuntimeError, "flatten reentered");
6551 }
6552 if (NIL_P(tmp)) {
6553 rb_ary_push(result, elt);
6554 }
6555 else {
6556 if (memo) {
6557 if (rb_hash_aref(memo, tmp) == Qtrue) {
6558 rb_hash_clear(memo);
6559 rb_raise(rb_eArgError, "tried to flatten recursive array");
6560 }
6561 rb_hash_aset(memo, tmp, Qtrue);
6562 }
6563 rb_ary_push(stack, ary);
6564 rb_ary_push(stack, LONG2NUM(i));
6565 ary = tmp;
6566 i = 0;
6567 }
6568 }
6569 if (RARRAY_LEN(stack) == 0) {
6570 break;
6571 }
6572 if (memo) {
6573 rb_hash_delete(memo, ary);
6574 }
6575 tmp = rb_ary_pop(stack);
6576 i = NUM2LONG(tmp);
6577 ary = rb_ary_pop(stack);
6578 }
6579
6580 if (memo) {
6581 rb_hash_clear(memo);
6582 }
6583
6584 RBASIC_SET_CLASS(result, rb_cArray);
6585 return result;
6586}
6587
6588/*
6589 * call-seq:
6590 * flatten!(depth = nil) -> self or nil
6591 *
6592 * Returns +self+ as a recursively flattening of +self+ to +depth+ levels of recursion;
6593 * +depth+ must be an
6594 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects],
6595 * or +nil+.
6596 * At each level of recursion:
6597 *
6598 * - Each element that is an array is "flattened"
6599 * (that is, replaced by its individual array elements).
6600 * - Each element that is not an array is unchanged
6601 * (even if the element is an object that has instance method +flatten+).
6602 *
6603 * Returns +nil+ if no elements were flattened.
6604 *
6605 * With non-negative integer argument +depth+, flattens recursively through +depth+ levels:
6606 *
6607 * a = [ 0, [ 1, [2, 3], 4 ], 5, {foo: 0}, Set.new([6, 7]) ]
6608 * a # => [0, [1, [2, 3], 4], 5, {:foo=>0}, #<Set: {6, 7}>]
6609 * a.dup.flatten!(1) # => [0, 1, [2, 3], 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6610 * a.dup.flatten!(1.1) # => [0, 1, [2, 3], 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6611 * a.dup.flatten!(2) # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6612 * a.dup.flatten!(3) # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6613 *
6614 * With +nil+ or negative argument +depth+, flattens all levels:
6615 *
6616 * a.dup.flatten! # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6617 * a.dup.flatten!(-1) # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6618 *
6619 * Related: Array#flatten;
6620 * see also {Methods for Assigning}[rdoc-ref:Array@Methods+for+Assigning].
6621 */
6622
6623static VALUE
6624rb_ary_flatten_bang(int argc, VALUE *argv, VALUE ary)
6625{
6626 int mod = 0, level = -1;
6627 VALUE result, lv;
6628
6629 lv = (rb_check_arity(argc, 0, 1) ? argv[0] : Qnil);
6630 rb_ary_modify_check(ary);
6631 if (!NIL_P(lv)) level = NUM2INT(lv);
6632 if (level == 0) return Qnil;
6633
6634 result = flatten(ary, level);
6635 if (result == ary) {
6636 return Qnil;
6637 }
6638 if (!(mod = ARY_EMBED_P(result))) rb_ary_freeze(result);
6639 rb_ary_replace(ary, result);
6640 if (mod) ARY_SET_EMBED_LEN(result, 0);
6641
6642 return ary;
6643}
6644
6645/*
6646 * call-seq:
6647 * flatten(depth = nil) -> new_array
6648 *
6649 * Returns a new array that is a recursive flattening of +self+
6650 * to +depth+ levels of recursion;
6651 * +depth+ must be an
6652 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects]
6653 * or +nil+.
6654 * At each level of recursion:
6655 *
6656 * - Each element that is an array is "flattened"
6657 * (that is, replaced by its individual array elements).
6658 * - Each element that is not an array is unchanged
6659 * (even if the element is an object that has instance method +flatten+).
6660 *
6661 * With non-negative integer argument +depth+, flattens recursively through +depth+ levels:
6662 *
6663 * a = [ 0, [ 1, [2, 3], 4 ], 5, {foo: 0}, Set.new([6, 7]) ]
6664 * a # => [0, [1, [2, 3], 4], 5, {:foo=>0}, #<Set: {6, 7}>]
6665 * a.flatten(0) # => [0, [1, [2, 3], 4], 5, {:foo=>0}, #<Set: {6, 7}>]
6666 * a.flatten(1 ) # => [0, 1, [2, 3], 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6667 * a.flatten(1.1) # => [0, 1, [2, 3], 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6668 * a.flatten(2) # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6669 * a.flatten(3) # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6670 *
6671 * With +nil+ or negative +depth+, flattens all levels.
6672 *
6673 * a.flatten # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6674 * a.flatten(-1) # => [0, 1, 2, 3, 4, 5, {:foo=>0}, #<Set: {6, 7}>]
6675 *
6676 * Related: Array#flatten!;
6677 * see also {Methods for Converting}[rdoc-ref:Array@Methods+for+Converting].
6678 */
6679
6680static VALUE
6681rb_ary_flatten(int argc, VALUE *argv, VALUE ary)
6682{
6683 int level = -1;
6684 VALUE result;
6685
6686 if (rb_check_arity(argc, 0, 1) && !NIL_P(argv[0])) {
6687 level = NUM2INT(argv[0]);
6688 if (level == 0) return ary_make_shared_copy(ary);
6689 }
6690
6691 result = flatten(ary, level);
6692 if (result == ary) {
6693 result = ary_make_shared_copy(ary);
6694 }
6695
6696 return result;
6697}
6698
6699#define RAND_UPTO(max) (long)rb_random_ulong_limited((randgen), (max)-1)
6700
6701static VALUE
6702rb_ary_shuffle_bang(rb_execution_context_t *ec, VALUE ary, VALUE randgen)
6703{
6704 long i, len;
6705
6706 rb_ary_modify(ary);
6707 i = len = RARRAY_LEN(ary);
6708 RARRAY_PTR_USE(ary, ptr, {
6709 while (i > 1) {
6710 long j = RAND_UPTO(i);
6711 VALUE tmp;
6712 if (len != RARRAY_LEN(ary) || ptr != RARRAY_CONST_PTR(ary)) {
6713 rb_raise(rb_eRuntimeError, "modified during shuffle");
6714 }
6715 tmp = ptr[--i];
6716 ptr[i] = ptr[j];
6717 ptr[j] = tmp;
6718 }
6719 }); /* WB: no new reference */
6720 return ary;
6721}
6722
6723static VALUE
6724rb_ary_shuffle(rb_execution_context_t *ec, VALUE ary, VALUE randgen)
6725{
6726 ary = rb_ary_dup(ary);
6727 rb_ary_shuffle_bang(ec, ary, randgen);
6728 return ary;
6729}
6730
6731static const rb_data_type_t ary_sample_memo_type = {
6732 .wrap_struct_name = "ary_sample_memo",
6733 .function = {
6734 .dfree = (RUBY_DATA_FUNC)st_free_table,
6735 },
6736 .flags = RUBY_TYPED_WB_PROTECTED | RUBY_TYPED_FREE_IMMEDIATELY
6737};
6738
6739static VALUE
6740ary_sample(rb_execution_context_t *ec, VALUE ary, VALUE randgen, VALUE nv, VALUE to_array)
6741{
6742 VALUE result;
6743 long n, len, i, j, k, idx[10];
6744 long rnds[numberof(idx)];
6745 long memo_threshold;
6746
6747 len = RARRAY_LEN(ary);
6748 if (!to_array) {
6749 if (len < 2)
6750 i = 0;
6751 else
6752 i = RAND_UPTO(len);
6753
6754 return rb_ary_elt(ary, i);
6755 }
6756 n = NUM2LONG(nv);
6757 if (n < 0) rb_raise(rb_eArgError, "negative sample number");
6758 if (n > len) n = len;
6759 if (n <= numberof(idx)) {
6760 for (i = 0; i < n; ++i) {
6761 rnds[i] = RAND_UPTO(len - i);
6762 }
6763 }
6764 k = len;
6765 len = RARRAY_LEN(ary);
6766 if (len < k && n <= numberof(idx)) {
6767 for (i = 0; i < n; ++i) {
6768 if (rnds[i] >= len) return rb_ary_new_capa(0);
6769 }
6770 }
6771 if (n > len) n = len;
6772 switch (n) {
6773 case 0:
6774 return rb_ary_new_capa(0);
6775 case 1:
6776 i = rnds[0];
6777 return rb_ary_new_from_args(1, RARRAY_AREF(ary, i));
6778 case 2:
6779 i = rnds[0];
6780 j = rnds[1];
6781 if (j >= i) j++;
6782 return rb_ary_new_from_args(2, RARRAY_AREF(ary, i), RARRAY_AREF(ary, j));
6783 case 3:
6784 i = rnds[0];
6785 j = rnds[1];
6786 k = rnds[2];
6787 {
6788 long l = j, g = i;
6789 if (j >= i) l = i, g = ++j;
6790 if (k >= l && (++k >= g)) ++k;
6791 }
6792 return rb_ary_new_from_args(3, RARRAY_AREF(ary, i), RARRAY_AREF(ary, j), RARRAY_AREF(ary, k));
6793 }
6794 memo_threshold =
6795 len < 2560 ? len / 128 :
6796 len < 5120 ? len / 64 :
6797 len < 10240 ? len / 32 :
6798 len / 16;
6799 if (n <= numberof(idx)) {
6800 long sorted[numberof(idx)];
6801 sorted[0] = idx[0] = rnds[0];
6802 for (i=1; i<n; i++) {
6803 k = rnds[i];
6804 for (j = 0; j < i; ++j) {
6805 if (k < sorted[j]) break;
6806 ++k;
6807 }
6808 memmove(&sorted[j+1], &sorted[j], sizeof(sorted[0])*(i-j));
6809 sorted[j] = idx[i] = k;
6810 }
6811 result = rb_ary_new_capa(n);
6812 RARRAY_PTR_USE(result, ptr_result, {
6813 for (i=0; i<n; i++) {
6814 ptr_result[i] = RARRAY_AREF(ary, idx[i]);
6815 }
6816 });
6817 }
6818 else if (n <= memo_threshold / 2) {
6819 long max_idx = 0;
6820 VALUE vmemo = TypedData_Wrap_Struct(0, &ary_sample_memo_type, 0);
6821 st_table *memo = st_init_numtable_with_size(n);
6822 RTYPEDDATA_DATA(vmemo) = memo;
6823 result = rb_ary_new_capa(n);
6824 RARRAY_PTR_USE(result, ptr_result, {
6825 for (i=0; i<n; i++) {
6826 long r = RAND_UPTO(len-i) + i;
6827 ptr_result[i] = r;
6828 if (r > max_idx) max_idx = r;
6829 }
6830 len = RARRAY_LEN(ary);
6831 if (len <= max_idx) n = 0;
6832 else if (n > len) n = len;
6833 RARRAY_PTR_USE(ary, ptr_ary, {
6834 for (i=0; i<n; i++) {
6835 long j2 = j = ptr_result[i];
6836 long i2 = i;
6837 st_data_t value;
6838 if (st_lookup(memo, (st_data_t)i, &value)) i2 = (long)value;
6839 if (st_lookup(memo, (st_data_t)j, &value)) j2 = (long)value;
6840 st_insert(memo, (st_data_t)j, (st_data_t)i2);
6841 ptr_result[i] = ptr_ary[j2];
6842 }
6843 });
6844 });
6845 RTYPEDDATA_DATA(vmemo) = 0;
6846 st_free_table(memo);
6847 RB_GC_GUARD(vmemo);
6848 }
6849 else {
6850 result = rb_ary_dup(ary);
6851 RBASIC_CLEAR_CLASS(result);
6852 RB_GC_GUARD(ary);
6853 RARRAY_PTR_USE(result, ptr_result, {
6854 for (i=0; i<n; i++) {
6855 j = RAND_UPTO(len-i) + i;
6856 nv = ptr_result[j];
6857 ptr_result[j] = ptr_result[i];
6858 ptr_result[i] = nv;
6859 }
6860 });
6861 RBASIC_SET_CLASS_RAW(result, rb_cArray);
6862 }
6863 ARY_SET_LEN(result, n);
6864
6865 return result;
6866}
6867
6868static VALUE
6869ary_sized_alloc(rb_execution_context_t *ec, VALUE self)
6870{
6871 return rb_ary_new2(RARRAY_LEN(self));
6872}
6873
6874static VALUE
6875ary_sample0(rb_execution_context_t *ec, VALUE ary)
6876{
6877 return ary_sample(ec, ary, rb_cRandom, Qfalse, Qfalse);
6878}
6879
6880static VALUE
6881rb_ary_cycle_size(VALUE self, VALUE args, VALUE eobj)
6882{
6883 long mul;
6884 VALUE n = Qnil;
6885 if (args && (RARRAY_LEN(args) > 0)) {
6886 n = RARRAY_AREF(args, 0);
6887 }
6888 if (RARRAY_LEN(self) == 0) return INT2FIX(0);
6889 if (NIL_P(n)) return DBL2NUM(HUGE_VAL);
6890 mul = NUM2LONG(n);
6891 if (mul <= 0) return INT2FIX(0);
6892 n = LONG2FIX(mul);
6893 return rb_fix_mul_fix(rb_ary_length(self), n);
6894}
6895
6896/*
6897 * call-seq:
6898 * cycle(count = nil) {|element| ... } -> nil
6899 * cycle(count = nil) -> new_enumerator
6900 *
6901 * With a block given, may call the block, depending on the value of argument +count+;
6902 * +count+ must be an
6903 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects],
6904 * or +nil+.
6905 *
6906 * When +count+ is positive,
6907 * calls the block with each element, then does so repeatedly,
6908 * until it has done so +count+ times; returns +nil+:
6909 *
6910 * output = []
6911 * [0, 1].cycle(2) {|element| output.push(element) } # => nil
6912 * output # => [0, 1, 0, 1]
6913 *
6914 * When +count+ is zero or negative, does not call the block:
6915 *
6916 * [0, 1].cycle(0) {|element| fail 'Cannot happen' } # => nil
6917 * [0, 1].cycle(-1) {|element| fail 'Cannot happen' } # => nil
6918 *
6919 * When +count+ is +nil+, cycles forever:
6920 *
6921 * # Prints 0 and 1 forever.
6922 * [0, 1].cycle {|element| puts element }
6923 * [0, 1].cycle(nil) {|element| puts element }
6924 *
6925 * With no block given, returns a new Enumerator.
6926 *
6927 * Related: see {Methods for Iterating}[rdoc-ref:Array@Methods+for+Iterating].
6928 */
6929static VALUE
6930rb_ary_cycle(int argc, VALUE *argv, VALUE ary)
6931{
6932 long n, i;
6933
6934 rb_check_arity(argc, 0, 1);
6935
6936 RETURN_SIZED_ENUMERATOR(ary, argc, argv, rb_ary_cycle_size);
6937 if (argc == 0 || NIL_P(argv[0])) {
6938 n = -1;
6939 }
6940 else {
6941 n = NUM2LONG(argv[0]);
6942 if (n <= 0) return Qnil;
6943 }
6944
6945 while (RARRAY_LEN(ary) > 0 && (n < 0 || 0 < n--)) {
6946 for (i=0; i<RARRAY_LEN(ary); i++) {
6947 rb_yield(RARRAY_AREF(ary, i));
6948 }
6949 }
6950 return Qnil;
6951}
6952
6953/*
6954 * Build a ruby array of the corresponding values and yield it to the
6955 * associated block.
6956 * Return the class of +values+ for reentry check.
6957 */
6958static int
6959yield_indexed_values(const VALUE values, const long r, const long *const p)
6960{
6961 const VALUE result = rb_ary_new2(r);
6962 long i;
6963
6964 for (i = 0; i < r; i++) ARY_SET(result, i, RARRAY_AREF(values, p[i]));
6965 ARY_SET_LEN(result, r);
6966 rb_yield(result);
6967 return !RBASIC(values)->klass;
6968}
6969
6970/*
6971 * Compute permutations of +r+ elements of the set <code>[0..n-1]</code>.
6972 *
6973 * When we have a complete permutation of array indices, copy the values
6974 * at those indices into a new array and yield that array.
6975 *
6976 * n: the size of the set
6977 * r: the number of elements in each permutation
6978 * p: the array (of size r) that we're filling in
6979 * used: an array of booleans: whether a given index is already used
6980 * values: the Ruby array that holds the actual values to permute
6981 */
6982static void
6983permute0(const long n, const long r, long *const p, char *const used, const VALUE values)
6984{
6985 long i = 0, index = 0;
6986
6987 for (;;) {
6988 const char *const unused = memchr(&used[i], 0, n-i);
6989 if (!unused) {
6990 if (!index) break;
6991 i = p[--index]; /* pop index */
6992 used[i++] = 0; /* index unused */
6993 }
6994 else {
6995 i = unused - used;
6996 p[index] = i;
6997 used[i] = 1; /* mark index used */
6998 ++index;
6999 if (index < r-1) { /* if not done yet */
7000 p[index] = i = 0;
7001 continue;
7002 }
7003 for (i = 0; i < n; ++i) {
7004 if (used[i]) continue;
7005 p[index] = i;
7006 if (!yield_indexed_values(values, r, p)) {
7007 rb_raise(rb_eRuntimeError, "permute reentered");
7008 }
7009 }
7010 i = p[--index]; /* pop index */
7011 used[i] = 0; /* index unused */
7012 p[index] = ++i;
7013 }
7014 }
7015}
7016
7017/*
7018 * Returns the product of from, from-1, ..., from - how_many + 1.
7019 * https://en.wikipedia.org/wiki/Pochhammer_symbol
7020 */
7021static VALUE
7022descending_factorial(long from, long how_many)
7023{
7024 VALUE cnt;
7025 if (how_many > 0) {
7026 cnt = LONG2FIX(from);
7027 while (--how_many > 0) {
7028 long v = --from;
7029 cnt = rb_int_mul(cnt, LONG2FIX(v));
7030 }
7031 }
7032 else {
7033 cnt = LONG2FIX(how_many == 0);
7034 }
7035 return cnt;
7036}
7037
7038static VALUE
7039binomial_coefficient(long comb, long size)
7040{
7041 VALUE r;
7042 long i;
7043 if (comb > size-comb) {
7044 comb = size-comb;
7045 }
7046 if (comb < 0) {
7047 return LONG2FIX(0);
7048 }
7049 else if (comb == 0) {
7050 return LONG2FIX(1);
7051 }
7052 r = LONG2FIX(size);
7053 for (i = 1; i < comb; ++i) {
7054 r = rb_int_mul(r, LONG2FIX(size - i));
7055 r = rb_int_idiv(r, LONG2FIX(i + 1));
7056 }
7057 return r;
7058}
7059
7060static VALUE
7061rb_ary_permutation_size(VALUE ary, VALUE args, VALUE eobj)
7062{
7063 long n = RARRAY_LEN(ary);
7064 long k = (args && (RARRAY_LEN(args) > 0)) ? NUM2LONG(RARRAY_AREF(args, 0)) : n;
7065
7066 return descending_factorial(n, k);
7067}
7068
7069/*
7070 * call-seq:
7071 * permutation(count = self.size) {|permutation| ... } -> self
7072 * permutation(count = self.size) -> new_enumerator
7073 *
7074 * Iterates over permutations of the elements of +self+;
7075 * the order of permutations is indeterminate.
7076 *
7077 * With a block and an in-range positive integer argument +count+ (<tt>0 < count <= self.size</tt>) given,
7078 * calls the block with each permutation of +self+ of size +count+;
7079 * returns +self+:
7080 *
7081 * a = [0, 1, 2]
7082 * perms = []
7083 * a.permutation(1) {|perm| perms.push(perm) }
7084 * perms # => [[0], [1], [2]]
7085 *
7086 * perms = []
7087 * a.permutation(2) {|perm| perms.push(perm) }
7088 * perms # => [[0, 1], [0, 2], [1, 0], [1, 2], [2, 0], [2, 1]]
7089 *
7090 * perms = []
7091 * a.permutation(3) {|perm| perms.push(perm) }
7092 * perms # => [[0, 1, 2], [0, 2, 1], [1, 0, 2], [1, 2, 0], [2, 0, 1], [2, 1, 0]]
7093 *
7094 * When +count+ is zero, calls the block once with a new empty array:
7095 *
7096 * perms = []
7097 * a.permutation(0) {|perm| perms.push(perm) }
7098 * perms # => [[]]
7099 *
7100 * When +count+ is out of range (negative or larger than <tt>self.size</tt>),
7101 * does not call the block:
7102 *
7103 * a.permutation(-1) {|permutation| fail 'Cannot happen' }
7104 * a.permutation(4) {|permutation| fail 'Cannot happen' }
7105 *
7106 * With no block given, returns a new Enumerator.
7107 *
7108 * Related: {Methods for Iterating}[rdoc-ref:Array@Methods+for+Iterating].
7109 */
7110
7111static VALUE
7112rb_ary_permutation(int argc, VALUE *argv, VALUE ary)
7113{
7114 long r, n, i;
7115
7116 n = RARRAY_LEN(ary); /* Array length */
7117 RETURN_SIZED_ENUMERATOR(ary, argc, argv, rb_ary_permutation_size); /* Return enumerator if no block */
7118 r = n;
7119 if (rb_check_arity(argc, 0, 1) && !NIL_P(argv[0]))
7120 r = NUM2LONG(argv[0]); /* Permutation size from argument */
7121
7122 if (r < 0 || n < r) {
7123 /* no permutations: yield nothing */
7124 }
7125 else if (r == 0) { /* exactly one permutation: the zero-length array */
7127 }
7128 else if (r == 1) { /* this is a special, easy case */
7129 for (i = 0; i < RARRAY_LEN(ary); i++) {
7130 rb_yield(rb_ary_new3(1, RARRAY_AREF(ary, i)));
7131 }
7132 }
7133 else { /* this is the general case */
7134 volatile VALUE t0;
7135 long *p = ALLOCV_N(long, t0, r+roomof(n, sizeof(long)));
7136 char *used = (char*)(p + r);
7137 VALUE ary0 = ary_make_shared_copy(ary); /* private defensive copy of ary */
7138 RBASIC_CLEAR_CLASS(ary0);
7139
7140 MEMZERO(used, char, n); /* initialize array */
7141
7142 permute0(n, r, p, used, ary0); /* compute and yield permutations */
7143 ALLOCV_END(t0);
7144 RBASIC_SET_CLASS_RAW(ary0, rb_cArray);
7145 }
7146 return ary;
7147}
7148
7149static void
7150combinate0(const long len, const long n, long *const stack, const VALUE values)
7151{
7152 long lev = 0;
7153
7154 MEMZERO(stack+1, long, n);
7155 stack[0] = -1;
7156 for (;;) {
7157 for (lev++; lev < n; lev++) {
7158 stack[lev+1] = stack[lev]+1;
7159 }
7160 if (!yield_indexed_values(values, n, stack+1)) {
7161 rb_raise(rb_eRuntimeError, "combination reentered");
7162 }
7163 do {
7164 if (lev == 0) return;
7165 stack[lev--]++;
7166 } while (stack[lev+1]+n == len+lev+1);
7167 }
7168}
7169
7170static VALUE
7171rb_ary_combination_size(VALUE ary, VALUE args, VALUE eobj)
7172{
7173 long n = RARRAY_LEN(ary);
7174 long k = NUM2LONG(RARRAY_AREF(args, 0));
7175
7176 return binomial_coefficient(k, n);
7177}
7178
7179/*
7180 * call-seq:
7181 * combination(count) {|element| ... } -> self
7182 * combination(count) -> new_enumerator
7183 *
7184 * When a block and a positive
7185 * {integer-convertible object}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects]
7186 * argument +count+ (<tt>0 < count <= self.size</tt>)
7187 * are given, calls the block with each combination of +self+ of size +count+;
7188 * returns +self+:
7189 *
7190 * a = %w[a b c] # => ["a", "b", "c"]
7191 * a.combination(2) {|combination| p combination } # => ["a", "b", "c"]
7192 *
7193 * Output:
7194 *
7195 * ["a", "b"]
7196 * ["a", "c"]
7197 * ["b", "c"]
7198 *
7199 * The order of the yielded combinations is not guaranteed.
7200 *
7201 * When +count+ is zero, calls the block once with a new empty array:
7202 *
7203 * a.combination(0) {|combination| p combination }
7204 * [].combination(0) {|combination| p combination }
7205 *
7206 * Output:
7207 *
7208 * []
7209 * []
7210 *
7211 * When +count+ is negative or larger than +self.size+ and +self+ is non-empty,
7212 * does not call the block:
7213 *
7214 * a.combination(-1) {|combination| fail 'Cannot happen' } # => ["a", "b", "c"]
7215 * a.combination(4) {|combination| fail 'Cannot happen' } # => ["a", "b", "c"]
7216 *
7217 * With no block given, returns a new Enumerator.
7218 *
7219 * Related: Array#permutation;
7220 * see also {Methods for Iterating}[rdoc-ref:Array@Methods+for+Iterating].
7221 */
7222
7223static VALUE
7224rb_ary_combination(VALUE ary, VALUE num)
7225{
7226 long i, n, len;
7227
7228 n = NUM2LONG(num);
7229 RETURN_SIZED_ENUMERATOR(ary, 1, &num, rb_ary_combination_size);
7230 len = RARRAY_LEN(ary);
7231 if (n < 0 || len < n) {
7232 /* yield nothing */
7233 }
7234 else if (n == 0) {
7236 }
7237 else if (n == 1) {
7238 for (i = 0; i < RARRAY_LEN(ary); i++) {
7239 rb_yield(rb_ary_new3(1, RARRAY_AREF(ary, i)));
7240 }
7241 }
7242 else {
7243 VALUE ary0 = ary_make_shared_copy(ary); /* private defensive copy of ary */
7244 volatile VALUE t0;
7245 long *stack = ALLOCV_N(long, t0, n+1);
7246
7247 RBASIC_CLEAR_CLASS(ary0);
7248 combinate0(len, n, stack, ary0);
7249 ALLOCV_END(t0);
7250 RBASIC_SET_CLASS_RAW(ary0, rb_cArray);
7251 }
7252 return ary;
7253}
7254
7255/*
7256 * Compute repeated permutations of +r+ elements of the set
7257 * <code>[0..n-1]</code>.
7258 *
7259 * When we have a complete repeated permutation of array indices, copy the
7260 * values at those indices into a new array and yield that array.
7261 *
7262 * n: the size of the set
7263 * r: the number of elements in each permutation
7264 * p: the array (of size r) that we're filling in
7265 * values: the Ruby array that holds the actual values to permute
7266 */
7267static void
7268rpermute0(const long n, const long r, long *const p, const VALUE values)
7269{
7270 long i = 0, index = 0;
7271
7272 p[index] = i;
7273 for (;;) {
7274 if (++index < r-1) {
7275 p[index] = i = 0;
7276 continue;
7277 }
7278 for (i = 0; i < n; ++i) {
7279 p[index] = i;
7280 if (!yield_indexed_values(values, r, p)) {
7281 rb_raise(rb_eRuntimeError, "repeated permute reentered");
7282 }
7283 }
7284 do {
7285 if (index <= 0) return;
7286 } while ((i = ++p[--index]) >= n);
7287 }
7288}
7289
7290static VALUE
7291rb_ary_repeated_permutation_size(VALUE ary, VALUE args, VALUE eobj)
7292{
7293 long n = RARRAY_LEN(ary);
7294 long k = NUM2LONG(RARRAY_AREF(args, 0));
7295
7296 if (k < 0) {
7297 return LONG2FIX(0);
7298 }
7299 if (n <= 0) {
7300 return LONG2FIX(!k);
7301 }
7302 return rb_int_positive_pow(n, (unsigned long)k);
7303}
7304
7305/*
7306 * call-seq:
7307 * repeated_permutation(size) {|permutation| ... } -> self
7308 * repeated_permutation(size) -> new_enumerator
7309 *
7310 * With a block given, calls the block with each repeated permutation of length +size+
7311 * of the elements of +self+;
7312 * each permutation is an array;
7313 * returns +self+. The order of the permutations is indeterminate.
7314 *
7315 * If a positive integer argument +size+ is given,
7316 * calls the block with each +size+-tuple repeated permutation of the elements of +self+.
7317 * The number of permutations is <tt>self.size**size</tt>.
7318 *
7319 * Examples:
7320 *
7321 * - +size+ is 1:
7322 *
7323 * p = []
7324 * [0, 1, 2].repeated_permutation(1) {|permutation| p.push(permutation) }
7325 * p # => [[0], [1], [2]]
7326 *
7327 * - +size+ is 2:
7328 *
7329 * p = []
7330 * [0, 1, 2].repeated_permutation(2) {|permutation| p.push(permutation) }
7331 * p # => [[0, 0], [0, 1], [0, 2], [1, 0], [1, 1], [1, 2], [2, 0], [2, 1], [2, 2]]
7332 *
7333 * If +size+ is zero, calls the block once with an empty array.
7334 *
7335 * If +size+ is negative, does not call the block:
7336 *
7337 * [0, 1, 2].repeated_permutation(-1) {|permutation| fail 'Cannot happen' }
7338 *
7339 * With no block given, returns a new Enumerator.
7340 *
7341 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
7342 */
7343static VALUE
7344rb_ary_repeated_permutation(VALUE ary, VALUE num)
7345{
7346 long r, n, i;
7347
7348 n = RARRAY_LEN(ary); /* Array length */
7349 RETURN_SIZED_ENUMERATOR(ary, 1, &num, rb_ary_repeated_permutation_size); /* Return Enumerator if no block */
7350 r = NUM2LONG(num); /* Permutation size from argument */
7351
7352 if (r < 0) {
7353 /* no permutations: yield nothing */
7354 }
7355 else if (r == 0) { /* exactly one permutation: the zero-length array */
7357 }
7358 else if (r == 1) { /* this is a special, easy case */
7359 for (i = 0; i < RARRAY_LEN(ary); i++) {
7360 rb_yield(rb_ary_new3(1, RARRAY_AREF(ary, i)));
7361 }
7362 }
7363 else { /* this is the general case */
7364 volatile VALUE t0;
7365 long *p = ALLOCV_N(long, t0, r);
7366 VALUE ary0 = ary_make_shared_copy(ary); /* private defensive copy of ary */
7367 RBASIC_CLEAR_CLASS(ary0);
7368
7369 rpermute0(n, r, p, ary0); /* compute and yield repeated permutations */
7370 ALLOCV_END(t0);
7371 RBASIC_SET_CLASS_RAW(ary0, rb_cArray);
7372 }
7373 return ary;
7374}
7375
7376static void
7377rcombinate0(const long n, const long r, long *const p, const long rest, const VALUE values)
7378{
7379 long i = 0, index = 0;
7380
7381 p[index] = i;
7382 for (;;) {
7383 if (++index < r-1) {
7384 p[index] = i;
7385 continue;
7386 }
7387 for (; i < n; ++i) {
7388 p[index] = i;
7389 if (!yield_indexed_values(values, r, p)) {
7390 rb_raise(rb_eRuntimeError, "repeated combination reentered");
7391 }
7392 }
7393 do {
7394 if (index <= 0) return;
7395 } while ((i = ++p[--index]) >= n);
7396 }
7397}
7398
7399static VALUE
7400rb_ary_repeated_combination_size(VALUE ary, VALUE args, VALUE eobj)
7401{
7402 long n = RARRAY_LEN(ary);
7403 long k = NUM2LONG(RARRAY_AREF(args, 0));
7404 if (k == 0) {
7405 return LONG2FIX(1);
7406 }
7407 return binomial_coefficient(k, n + k - 1);
7408}
7409
7410/*
7411 * call-seq:
7412 * repeated_combination(size) {|combination| ... } -> self
7413 * repeated_combination(size) -> new_enumerator
7414 *
7415 * With a block given, calls the block with each repeated combination of length +size+
7416 * of the elements of +self+;
7417 * each combination is an array;
7418 * returns +self+. The order of the combinations is indeterminate.
7419 *
7420 * If a positive integer argument +size+ is given,
7421 * calls the block with each +size+-tuple repeated combination of the elements of +self+.
7422 * The number of combinations is <tt>(size+1)(size+2)/2</tt>.
7423 *
7424 * Examples:
7425 *
7426 * - +size+ is 1:
7427 *
7428 * c = []
7429 * [0, 1, 2].repeated_combination(1) {|combination| c.push(combination) }
7430 * c # => [[0], [1], [2]]
7431 *
7432 * - +size+ is 2:
7433 *
7434 * c = []
7435 * [0, 1, 2].repeated_combination(2) {|combination| c.push(combination) }
7436 * c # => [[0, 0], [0, 1], [0, 2], [1, 1], [1, 2], [2, 2]]
7437 *
7438 * If +size+ is zero, calls the block once with an empty array.
7439 *
7440 * If +size+ is negative, does not call the block:
7441 *
7442 * [0, 1, 2].repeated_combination(-1) {|combination| fail 'Cannot happen' }
7443 *
7444 * With no block given, returns a new Enumerator.
7445 *
7446 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
7447 */
7448
7449static VALUE
7450rb_ary_repeated_combination(VALUE ary, VALUE num)
7451{
7452 long n, i, len;
7453
7454 n = NUM2LONG(num); /* Combination size from argument */
7455 RETURN_SIZED_ENUMERATOR(ary, 1, &num, rb_ary_repeated_combination_size); /* Return enumerator if no block */
7456 len = RARRAY_LEN(ary);
7457 if (n < 0) {
7458 /* yield nothing */
7459 }
7460 else if (n == 0) {
7462 }
7463 else if (n == 1) {
7464 for (i = 0; i < RARRAY_LEN(ary); i++) {
7465 rb_yield(rb_ary_new3(1, RARRAY_AREF(ary, i)));
7466 }
7467 }
7468 else if (len == 0) {
7469 /* yield nothing */
7470 }
7471 else {
7472 volatile VALUE t0;
7473 long *p = ALLOCV_N(long, t0, n);
7474 VALUE ary0 = ary_make_shared_copy(ary); /* private defensive copy of ary */
7475 RBASIC_CLEAR_CLASS(ary0);
7476
7477 rcombinate0(len, n, p, n, ary0); /* compute and yield repeated combinations */
7478 ALLOCV_END(t0);
7479 RBASIC_SET_CLASS_RAW(ary0, rb_cArray);
7480 }
7481 return ary;
7482}
7483
7484/*
7485 * call-seq:
7486 * product(*other_arrays) -> new_array
7487 * product(*other_arrays) {|combination| ... } -> self
7488 *
7489 * Computes all combinations of elements from all the arrays,
7490 * including both +self+ and +other_arrays+:
7491 *
7492 * - The number of combinations is the product of the sizes of all the arrays,
7493 * including both +self+ and +other_arrays+.
7494 * - The order of the returned combinations is indeterminate.
7495 *
7496 * With no block given, returns the combinations as an array of arrays:
7497 *
7498 * p = [0, 1].product([2, 3])
7499 * # => [[0, 2], [0, 3], [1, 2], [1, 3]]
7500 * p.size # => 4
7501 * p = [0, 1].product([2, 3], [4, 5])
7502 * # => [[0, 2, 4], [0, 2, 5], [0, 3, 4], [0, 3, 5], [1, 2, 4], [1, 2, 5], [1, 3, 4], [1, 3,...
7503 * p.size # => 8
7504 *
7505 * If +self+ or any argument is empty, returns an empty array:
7506 *
7507 * [].product([2, 3], [4, 5]) # => []
7508 * [0, 1].product([2, 3], []) # => []
7509 *
7510 * If no argument is given, returns an array of 1-element arrays,
7511 * each containing an element of +self+:
7512 *
7513 * a.product # => [[0], [1], [2]]
7514 *
7515 * With a block given, calls the block with each combination; returns +self+:
7516 *
7517 * p = []
7518 * [0, 1].product([2, 3]) {|combination| p.push(combination) }
7519 * p # => [[0, 2], [0, 3], [1, 2], [1, 3]]
7520 *
7521 * If +self+ or any argument is empty, does not call the block:
7522 *
7523 * [].product([2, 3], [4, 5]) {|combination| fail 'Cannot happen' }
7524 * # => []
7525 * [0, 1].product([2, 3], []) {|combination| fail 'Cannot happen' }
7526 * # => [0, 1]
7527 *
7528 * If no argument is given, calls the block with each element of +self+ as a 1-element array:
7529 *
7530 * p = []
7531 * [0, 1].product {|combination| p.push(combination) }
7532 * p # => [[0], [1]]
7533 *
7534 * Related: see {Methods for Combining}[rdoc-ref:Array@Methods+for+Combining].
7535 */
7536
7537static VALUE
7538rb_ary_product(int argc, VALUE *argv, VALUE ary)
7539{
7540 int n = argc+1; /* How many arrays we're operating on */
7541 volatile VALUE t0 = rb_ary_hidden_new(n);
7542 volatile VALUE t1 = Qundef;
7543 VALUE *arrays = RARRAY_PTR(t0); /* The arrays we're computing the product of */
7544 int *counters = ALLOCV_N(int, t1, n); /* The current position in each one */
7545 VALUE result = Qnil; /* The array we'll be returning, when no block given */
7546 long i,j;
7547 long resultlen = 1;
7548
7549 RBASIC_CLEAR_CLASS(t0);
7550
7551 /* initialize the arrays of arrays */
7552 ARY_SET_LEN(t0, n);
7553 arrays[0] = ary;
7554 for (i = 1; i < n; i++) arrays[i] = Qnil;
7555 for (i = 1; i < n; i++) arrays[i] = to_ary(argv[i-1]);
7556
7557 /* initialize the counters for the arrays */
7558 for (i = 0; i < n; i++) counters[i] = 0;
7559
7560 /* Otherwise, allocate and fill in an array of results */
7561 if (rb_block_given_p()) {
7562 /* Make defensive copies of arrays; exit if any is empty */
7563 for (i = 0; i < n; i++) {
7564 if (RARRAY_LEN(arrays[i]) == 0) goto done;
7565 arrays[i] = ary_make_shared_copy(arrays[i]);
7566 }
7567 }
7568 else {
7569 /* Compute the length of the result array; return [] if any is empty */
7570 for (i = 0; i < n; i++) {
7571 long k = RARRAY_LEN(arrays[i]);
7572 if (k == 0) {
7573 result = rb_ary_new2(0);
7574 goto done;
7575 }
7576 if (MUL_OVERFLOW_LONG_P(resultlen, k))
7577 rb_raise(rb_eRangeError, "too big to product");
7578 resultlen *= k;
7579 }
7580 result = rb_ary_new2(resultlen);
7581 }
7582 for (;;) {
7583 int m;
7584 /* fill in one subarray */
7585 VALUE subarray = rb_ary_new2(n);
7586 for (j = 0; j < n; j++) {
7587 rb_ary_push(subarray, rb_ary_entry(arrays[j], counters[j]));
7588 }
7589
7590 /* put it on the result array */
7591 if (NIL_P(result)) {
7592 FL_SET(t0, RARRAY_SHARED_ROOT_FLAG);
7593 rb_yield(subarray);
7594 if (!FL_TEST(t0, RARRAY_SHARED_ROOT_FLAG)) {
7595 rb_raise(rb_eRuntimeError, "product reentered");
7596 }
7597 else {
7598 FL_UNSET(t0, RARRAY_SHARED_ROOT_FLAG);
7599 }
7600 }
7601 else {
7602 rb_ary_push(result, subarray);
7603 }
7604
7605 /*
7606 * Increment the last counter. If it overflows, reset to 0
7607 * and increment the one before it.
7608 */
7609 m = n-1;
7610 counters[m]++;
7611 while (counters[m] == RARRAY_LEN(arrays[m])) {
7612 counters[m] = 0;
7613 /* If the first counter overflows, we are done */
7614 if (--m < 0) goto done;
7615 counters[m]++;
7616 }
7617 }
7618
7619done:
7620 ALLOCV_END(t1);
7621
7622 return NIL_P(result) ? ary : result;
7623}
7624
7625/*
7626 * call-seq:
7627 * take(count) -> new_array
7628 *
7629 * Returns a new array containing the first +count+ element of +self+
7630 * (as available);
7631 * +count+ must be a non-negative numeric;
7632 * does not modify +self+:
7633 *
7634 * a = ['a', 'b', 'c', 'd']
7635 * a.take(2) # => ["a", "b"]
7636 * a.take(2.1) # => ["a", "b"]
7637 * a.take(50) # => ["a", "b", "c", "d"]
7638 * a.take(0) # => []
7639 *
7640 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
7641 */
7642
7643static VALUE
7644rb_ary_take(VALUE obj, VALUE n)
7645{
7646 long len = NUM2LONG(n);
7647 if (len < 0) {
7648 rb_raise(rb_eArgError, "attempt to take negative size");
7649 }
7650 return rb_ary_subseq(obj, 0, len);
7651}
7652
7653/*
7654 * call-seq:
7655 * take_while {|element| ... } -> new_array
7656 * take_while -> new_enumerator
7657 *
7658 * With a block given, calls the block with each successive element of +self+;
7659 * stops iterating if the block returns +false+ or +nil+;
7660 * returns a new array containing those elements for which the block returned a truthy value:
7661 *
7662 * a = [0, 1, 2, 3, 4, 5]
7663 * a.take_while {|element| element < 3 } # => [0, 1, 2]
7664 * a.take_while {|element| true } # => [0, 1, 2, 3, 4, 5]
7665 * a.take_while {|element| false } # => []
7666 *
7667 * With no block given, returns a new Enumerator.
7668 *
7669 * Does not modify +self+.
7670 *
7671 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
7672 */
7673
7674static VALUE
7675rb_ary_take_while(VALUE ary)
7676{
7677 long i;
7678
7679 RETURN_ENUMERATOR(ary, 0, 0);
7680 for (i = 0; i < RARRAY_LEN(ary); i++) {
7681 if (!RTEST(rb_yield(RARRAY_AREF(ary, i)))) break;
7682 }
7683 return rb_ary_take(ary, LONG2FIX(i));
7684}
7685
7686/*
7687 * call-seq:
7688 * drop(count) -> new_array
7689 *
7690 * Returns a new array containing all but the first +count+ element of +self+,
7691 * where +count+ is a non-negative integer;
7692 * does not modify +self+.
7693 *
7694 * Examples:
7695 *
7696 * a = [0, 1, 2, 3, 4, 5]
7697 * a.drop(0) # => [0, 1, 2, 3, 4, 5]
7698 * a.drop(1) # => [1, 2, 3, 4, 5]
7699 * a.drop(2) # => [2, 3, 4, 5]
7700 * a.drop(9) # => []
7701 *
7702 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
7703 */
7704
7705static VALUE
7706rb_ary_drop(VALUE ary, VALUE n)
7707{
7708 VALUE result;
7709 long pos = NUM2LONG(n);
7710 if (pos < 0) {
7711 rb_raise(rb_eArgError, "attempt to drop negative size");
7712 }
7713
7714 result = rb_ary_subseq(ary, pos, RARRAY_LEN(ary));
7715 if (NIL_P(result)) result = rb_ary_new();
7716 return result;
7717}
7718
7719/*
7720 * call-seq:
7721 * drop_while {|element| ... } -> new_array
7722 * drop_while -> new_enumerator
7723 *
7724 * With a block given, calls the block with each successive element of +self+;
7725 * stops if the block returns +false+ or +nil+;
7726 * returns a new array _omitting_ those elements for which the block returned a truthy value;
7727 * does not modify +self+:
7728 *
7729 * a = [0, 1, 2, 3, 4, 5]
7730 * a.drop_while {|element| element < 3 } # => [3, 4, 5]
7731 *
7732 * With no block given, returns a new Enumerator.
7733 *
7734 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
7735 */
7736
7737static VALUE
7738rb_ary_drop_while(VALUE ary)
7739{
7740 long i;
7741
7742 RETURN_ENUMERATOR(ary, 0, 0);
7743 for (i = 0; i < RARRAY_LEN(ary); i++) {
7744 if (!RTEST(rb_yield(RARRAY_AREF(ary, i)))) break;
7745 }
7746 return rb_ary_drop(ary, LONG2FIX(i));
7747}
7748
7749/*
7750 * call-seq:
7751 * any? -> true or false
7752 * any?(object) -> true or false
7753 * any? {|element| ... } -> true or false
7754 *
7755 * Returns whether for any element of +self+, a given criterion is satisfied.
7756 *
7757 * With no block and no argument, returns whether any element of +self+ is truthy:
7758 *
7759 * [nil, false, []].any? # => true # Array object is truthy.
7760 * [nil, false, {}].any? # => true # Hash object is truthy.
7761 * [nil, false, ''].any? # => true # String object is truthy.
7762 * [nil, false].any? # => false # Nil and false are not truthy.
7763 *
7764 * With argument +object+ given,
7765 * returns whether <tt>object === ele</tt> for any element +ele+ in +self+:
7766 *
7767 * [nil, false, 0].any?(0) # => true
7768 * [nil, false, 1].any?(0) # => false
7769 * [nil, false, 'food'].any?(/foo/) # => true
7770 * [nil, false, 'food'].any?(/bar/) # => false
7771 *
7772 * With a block given,
7773 * calls the block with each element in +self+;
7774 * returns whether the block returns any truthy value:
7775 *
7776 * [0, 1, 2].any? {|ele| ele < 1 } # => true
7777 * [0, 1, 2].any? {|ele| ele < 0 } # => false
7778 *
7779 * With both a block and argument +object+ given,
7780 * ignores the block and uses +object+ as above.
7781 *
7782 * <b>Special case</b>: returns +false+ if +self+ is empty
7783 * (regardless of any given argument or block).
7784 *
7785 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
7786 */
7787
7788static VALUE
7789rb_ary_any_p(int argc, VALUE *argv, VALUE ary)
7790{
7791 long i, len = RARRAY_LEN(ary);
7792
7793 rb_check_arity(argc, 0, 1);
7794 if (!len) return Qfalse;
7795 if (argc) {
7796 if (rb_block_given_p()) {
7797 rb_warn("given block not used");
7798 }
7799 for (i = 0; i < RARRAY_LEN(ary); ++i) {
7800 if (RTEST(rb_funcall(argv[0], idEqq, 1, RARRAY_AREF(ary, i)))) return Qtrue;
7801 }
7802 }
7803 else if (!rb_block_given_p()) {
7804 for (i = 0; i < len; ++i) {
7805 if (RTEST(RARRAY_AREF(ary, i))) return Qtrue;
7806 }
7807 }
7808 else {
7809 for (i = 0; i < RARRAY_LEN(ary); ++i) {
7810 if (RTEST(rb_yield(RARRAY_AREF(ary, i)))) return Qtrue;
7811 }
7812 }
7813 return Qfalse;
7814}
7815
7816/*
7817 * call-seq:
7818 * all? -> true or false
7819 * all?(object) -> true or false
7820 * all? {|element| ... } -> true or false
7821 *
7822 * Returns whether for every element of +self+,
7823 * a given criterion is satisfied.
7824 *
7825 * With no block and no argument,
7826 * returns whether every element of +self+ is truthy:
7827 *
7828 * [[], {}, '', 0, 0.0, Object.new].all? # => true # All truthy objects.
7829 * [[], {}, '', 0, 0.0, nil].all? # => false # nil is not truthy.
7830 * [[], {}, '', 0, 0.0, false].all? # => false # false is not truthy.
7831 *
7832 * With argument +object+ given, returns whether <tt>object === ele</tt>
7833 * for every element +ele+ in +self+:
7834 *
7835 * [0, 0, 0].all?(0) # => true
7836 * [0, 1, 2].all?(1) # => false
7837 * ['food', 'fool', 'foot'].all?(/foo/) # => true
7838 * ['food', 'drink'].all?(/foo/) # => false
7839 *
7840 * With a block given, calls the block with each element in +self+;
7841 * returns whether the block returns only truthy values:
7842 *
7843 * [0, 1, 2].all? { |ele| ele < 3 } # => true
7844 * [0, 1, 2].all? { |ele| ele < 2 } # => false
7845 *
7846 * With both a block and argument +object+ given,
7847 * ignores the block and uses +object+ as above.
7848 *
7849 * <b>Special case</b>: returns +true+ if +self+ is empty
7850 * (regardless of any given argument or block).
7851 *
7852 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
7853 */
7854
7855static VALUE
7856rb_ary_all_p(int argc, VALUE *argv, VALUE ary)
7857{
7858 long i, len = RARRAY_LEN(ary);
7859
7860 rb_check_arity(argc, 0, 1);
7861 if (!len) return Qtrue;
7862 if (argc) {
7863 if (rb_block_given_p()) {
7864 rb_warn("given block not used");
7865 }
7866 for (i = 0; i < RARRAY_LEN(ary); ++i) {
7867 if (!RTEST(rb_funcall(argv[0], idEqq, 1, RARRAY_AREF(ary, i)))) return Qfalse;
7868 }
7869 }
7870 else if (!rb_block_given_p()) {
7871 for (i = 0; i < len; ++i) {
7872 if (!RTEST(RARRAY_AREF(ary, i))) return Qfalse;
7873 }
7874 }
7875 else {
7876 for (i = 0; i < RARRAY_LEN(ary); ++i) {
7877 if (!RTEST(rb_yield(RARRAY_AREF(ary, i)))) return Qfalse;
7878 }
7879 }
7880 return Qtrue;
7881}
7882
7883/*
7884 * call-seq:
7885 * none? -> true or false
7886 * none?(object) -> true or false
7887 * none? {|element| ... } -> true or false
7888 *
7889 * Returns +true+ if no element of +self+ meets a given criterion, +false+ otherwise.
7890 *
7891 * With no block given and no argument, returns +true+ if +self+ has no truthy elements,
7892 * +false+ otherwise:
7893 *
7894 * [nil, false].none? # => true
7895 * [nil, 0, false].none? # => false
7896 * [].none? # => true
7897 *
7898 * With argument +object+ given, returns +false+ if for any element +element+,
7899 * <tt>object === element</tt>; +true+ otherwise:
7900 *
7901 * ['food', 'drink'].none?(/bar/) # => true
7902 * ['food', 'drink'].none?(/foo/) # => false
7903 * [].none?(/foo/) # => true
7904 * [0, 1, 2].none?(3) # => true
7905 * [0, 1, 2].none?(1) # => false
7906 *
7907 * With a block given, calls the block with each element in +self+;
7908 * returns +true+ if the block returns no truthy value, +false+ otherwise:
7909 *
7910 * [0, 1, 2].none? {|element| element > 3 } # => true
7911 * [0, 1, 2].none? {|element| element > 1 } # => false
7912 *
7913 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
7914 */
7915
7916static VALUE
7917rb_ary_none_p(int argc, VALUE *argv, VALUE ary)
7918{
7919 long i, len = RARRAY_LEN(ary);
7920
7921 rb_check_arity(argc, 0, 1);
7922 if (!len) return Qtrue;
7923 if (argc) {
7924 if (rb_block_given_p()) {
7925 rb_warn("given block not used");
7926 }
7927 for (i = 0; i < RARRAY_LEN(ary); ++i) {
7928 if (RTEST(rb_funcall(argv[0], idEqq, 1, RARRAY_AREF(ary, i)))) return Qfalse;
7929 }
7930 }
7931 else if (!rb_block_given_p()) {
7932 for (i = 0; i < len; ++i) {
7933 if (RTEST(RARRAY_AREF(ary, i))) return Qfalse;
7934 }
7935 }
7936 else {
7937 for (i = 0; i < RARRAY_LEN(ary); ++i) {
7938 if (RTEST(rb_yield(RARRAY_AREF(ary, i)))) return Qfalse;
7939 }
7940 }
7941 return Qtrue;
7942}
7943
7944/*
7945 * call-seq:
7946 * one? -> true or false
7947 * one? {|element| ... } -> true or false
7948 * one?(object) -> true or false
7949 *
7950 * Returns +true+ if exactly one element of +self+ meets a given criterion.
7951 *
7952 * With no block given and no argument, returns +true+ if +self+ has exactly one truthy element,
7953 * +false+ otherwise:
7954 *
7955 * [nil, 0].one? # => true
7956 * [0, 0].one? # => false
7957 * [nil, nil].one? # => false
7958 * [].one? # => false
7959 *
7960 * With a block given, calls the block with each element in +self+;
7961 * returns +true+ if the block a truthy value for exactly one element, +false+ otherwise:
7962 *
7963 * [0, 1, 2].one? {|element| element > 0 } # => false
7964 * [0, 1, 2].one? {|element| element > 1 } # => true
7965 * [0, 1, 2].one? {|element| element > 2 } # => false
7966 *
7967 * With argument +object+ given, returns +true+ if for exactly one element +element+, <tt>object === element</tt>;
7968 * +false+ otherwise:
7969 *
7970 * [0, 1, 2].one?(0) # => true
7971 * [0, 0, 1].one?(0) # => false
7972 * [1, 1, 2].one?(0) # => false
7973 * ['food', 'drink'].one?(/bar/) # => false
7974 * ['food', 'drink'].one?(/foo/) # => true
7975 * [].one?(/foo/) # => false
7976 *
7977 * Related: see {Methods for Querying}[rdoc-ref:Array@Methods+for+Querying].
7978 */
7979
7980static VALUE
7981rb_ary_one_p(int argc, VALUE *argv, VALUE ary)
7982{
7983 long i, len = RARRAY_LEN(ary);
7984 VALUE result = Qfalse;
7985
7986 rb_check_arity(argc, 0, 1);
7987 if (!len) return Qfalse;
7988 if (argc) {
7989 if (rb_block_given_p()) {
7990 rb_warn("given block not used");
7991 }
7992 for (i = 0; i < RARRAY_LEN(ary); ++i) {
7993 if (RTEST(rb_funcall(argv[0], idEqq, 1, RARRAY_AREF(ary, i)))) {
7994 if (result) return Qfalse;
7995 result = Qtrue;
7996 }
7997 }
7998 }
7999 else if (!rb_block_given_p()) {
8000 for (i = 0; i < len; ++i) {
8001 if (RTEST(RARRAY_AREF(ary, i))) {
8002 if (result) return Qfalse;
8003 result = Qtrue;
8004 }
8005 }
8006 }
8007 else {
8008 for (i = 0; i < RARRAY_LEN(ary); ++i) {
8009 if (RTEST(rb_yield(RARRAY_AREF(ary, i)))) {
8010 if (result) return Qfalse;
8011 result = Qtrue;
8012 }
8013 }
8014 }
8015 return result;
8016}
8017
8018/*
8019 * call-seq:
8020 * dig(index, *identifiers) -> object
8021 *
8022 * Finds and returns the object in nested object
8023 * specified by +index+ and +identifiers+;
8024 * the nested objects may be instances of various classes.
8025 * See {Dig Methods}[rdoc-ref:dig_methods.rdoc].
8026 *
8027 * Examples:
8028 *
8029 * a = [:foo, [:bar, :baz, [:bat, :bam]]]
8030 * a.dig(1) # => [:bar, :baz, [:bat, :bam]]
8031 * a.dig(1, 2) # => [:bat, :bam]
8032 * a.dig(1, 2, 0) # => :bat
8033 * a.dig(1, 2, 3) # => nil
8034 *
8035 * Related: see {Methods for Fetching}[rdoc-ref:Array@Methods+for+Fetching].
8036 */
8037
8038static VALUE
8039rb_ary_dig(int argc, VALUE *argv, VALUE self)
8040{
8042 self = rb_ary_at(self, *argv);
8043 if (!--argc) return self;
8044 ++argv;
8045 return rb_obj_dig(argc, argv, self, Qnil);
8046}
8047
8048static inline VALUE
8049finish_exact_sum(long n, VALUE r, VALUE v, int z)
8050{
8051 if (n != 0)
8052 v = rb_fix_plus(LONG2FIX(n), v);
8053 if (!UNDEF_P(r)) {
8054 v = rb_rational_plus(r, v);
8055 }
8056 else if (!n && z) {
8057 v = rb_fix_plus(LONG2FIX(0), v);
8058 }
8059 return v;
8060}
8061
8062/*
8063 * call-seq:
8064 * sum(init = 0) -> object
8065 * sum(init = 0) {|element| ... } -> object
8066 *
8067 * With no block given, returns the sum of +init+ and all elements of +self+;
8068 * for array +array+ and value +init+, equivalent to:
8069 *
8070 * sum = init
8071 * array.each {|element| sum += element }
8072 * sum
8073 *
8074 * For example, <tt>[e0, e1, e2].sum</tt> returns <tt>init + e0 + e1 + e2</tt>.
8075 *
8076 * Examples:
8077 *
8078 * [0, 1, 2, 3].sum # => 6
8079 * [0, 1, 2, 3].sum(100) # => 106
8080 * ['abc', 'def', 'ghi'].sum('jkl') # => "jklabcdefghi"
8081 * [[:foo, :bar], ['foo', 'bar']].sum([2, 3])
8082 * # => [2, 3, :foo, :bar, "foo", "bar"]
8083 *
8084 * The +init+ value and elements need not be numeric, but must all be <tt>+</tt>-compatible:
8085 *
8086 * # Raises TypeError: Array can't be coerced into Integer.
8087 * [[:foo, :bar], ['foo', 'bar']].sum(2)
8088 *
8089 * With a block given, calls the block with each element of +self+;
8090 * the block's return value (instead of the element itself) is used as the addend:
8091 *
8092 * ['zero', 1, :two].sum('Coerced and concatenated: ') {|element| element.to_s }
8093 * # => "Coerced and concatenated: zero1two"
8094 *
8095 * Notes:
8096 *
8097 * - Array#join and Array#flatten may be faster than Array#sum
8098 * for an array of strings or an array of arrays.
8099 * - Array#sum method may not respect method redefinition of "+" methods such as Integer#+.
8100 *
8101 */
8102
8103static VALUE
8104rb_ary_sum(int argc, VALUE *argv, VALUE ary)
8105{
8106 VALUE e, v, r;
8107 long i, n;
8108 int block_given;
8109
8110 v = (rb_check_arity(argc, 0, 1) ? argv[0] : LONG2FIX(0));
8111
8112 block_given = rb_block_given_p();
8113
8114 if (RARRAY_LEN(ary) == 0)
8115 return v;
8116
8117 n = 0;
8118 r = Qundef;
8119
8120 if (!FIXNUM_P(v) && !RB_BIGNUM_TYPE_P(v) && !RB_TYPE_P(v, T_RATIONAL)) {
8121 i = 0;
8122 goto init_is_a_value;
8123 }
8124
8125 for (i = 0; i < RARRAY_LEN(ary); i++) {
8126 e = RARRAY_AREF(ary, i);
8127 if (block_given)
8128 e = rb_yield(e);
8129 if (FIXNUM_P(e)) {
8130 n += FIX2LONG(e); /* should not overflow long type */
8131 if (!FIXABLE(n)) {
8132 v = rb_big_plus(LONG2NUM(n), v);
8133 n = 0;
8134 }
8135 }
8136 else if (RB_BIGNUM_TYPE_P(e))
8137 v = rb_big_plus(e, v);
8138 else if (RB_TYPE_P(e, T_RATIONAL)) {
8139 if (UNDEF_P(r))
8140 r = e;
8141 else
8142 r = rb_rational_plus(r, e);
8143 }
8144 else
8145 goto not_exact;
8146 }
8147 v = finish_exact_sum(n, r, v, argc!=0);
8148 return v;
8149
8150 not_exact:
8151 v = finish_exact_sum(n, r, v, i!=0);
8152
8153 if (RB_FLOAT_TYPE_P(e)) {
8154 /*
8155 * Kahan-Babuska balancing compensated summation algorithm
8156 * See https://link.springer.com/article/10.1007/s00607-005-0139-x
8157 */
8158 double f, c;
8159 double x, t;
8160
8161 f = NUM2DBL(v);
8162 c = 0.0;
8163 goto has_float_value;
8164 for (; i < RARRAY_LEN(ary); i++) {
8165 e = RARRAY_AREF(ary, i);
8166 if (block_given)
8167 e = rb_yield(e);
8168 if (RB_FLOAT_TYPE_P(e))
8169 has_float_value:
8170 x = RFLOAT_VALUE(e);
8171 else if (FIXNUM_P(e))
8172 x = FIX2LONG(e);
8173 else if (RB_BIGNUM_TYPE_P(e))
8174 x = rb_big2dbl(e);
8175 else if (RB_TYPE_P(e, T_RATIONAL))
8176 x = rb_num2dbl(e);
8177 else
8178 goto not_float;
8179
8180 if (isnan(f)) continue;
8181 if (isnan(x)) {
8182 f = x;
8183 continue;
8184 }
8185 if (isinf(x)) {
8186 if (isinf(f) && signbit(x) != signbit(f))
8187 f = NAN;
8188 else
8189 f = x;
8190 continue;
8191 }
8192 if (isinf(f)) continue;
8193
8194 t = f + x;
8195 if (fabs(f) >= fabs(x))
8196 c += ((f - t) + x);
8197 else
8198 c += ((x - t) + f);
8199 f = t;
8200 }
8201 f += c;
8202 return DBL2NUM(f);
8203
8204 not_float:
8205 v = DBL2NUM(f);
8206 }
8207
8208 goto has_some_value;
8209 init_is_a_value:
8210 for (; i < RARRAY_LEN(ary); i++) {
8211 e = RARRAY_AREF(ary, i);
8212 if (block_given)
8213 e = rb_yield(e);
8214 has_some_value:
8215 v = rb_funcall(v, idPLUS, 1, e);
8216 }
8217 return v;
8218}
8219
8220/* :nodoc: */
8221static VALUE
8222rb_ary_deconstruct(VALUE ary)
8223{
8224 return ary;
8225}
8226
8227/*
8228 * An \Array object is an ordered, integer-indexed collection of objects,
8229 * called _elements_;
8230 * the object represents
8231 * an {array data structure}[https://en.wikipedia.org/wiki/Array_(data_structure)].
8232 *
8233 * An element may be any object (even another array);
8234 * elements may be any mixture of objects of different types.
8235 *
8236 * Important data structures that use arrays include:
8237 *
8238 * - {Coordinate vector}[https://en.wikipedia.org/wiki/Coordinate_vector].
8239 * - {Matrix}[https://en.wikipedia.org/wiki/Matrix_(mathematics)].
8240 * - {Heap}[https://en.wikipedia.org/wiki/Heap_(data_structure)].
8241 * - {Hash table}[https://en.wikipedia.org/wiki/Hash_table].
8242 * - {Deque (double-ended queue)}[https://en.wikipedia.org/wiki/Double-ended_queue].
8243 * - {Queue}[https://en.wikipedia.org/wiki/Queue_(abstract_data_type)].
8244 * - {Stack}[https://en.wikipedia.org/wiki/Stack_(abstract_data_type)].
8245 *
8246 * There are also array-like data structures:
8247 *
8248 * - {Associative array}[https://en.wikipedia.org/wiki/Associative_array] (see Hash).
8249 * - {Directory}[https://en.wikipedia.org/wiki/Directory_(computing)] (see Dir).
8250 * - {Environment}[https://en.wikipedia.org/wiki/Environment_variable] (see ENV).
8251 * - {Set}[https://en.wikipedia.org/wiki/Set_(abstract_data_type)] (see Set).
8252 * - {String}[https://en.wikipedia.org/wiki/String_(computer_science)] (see String).
8253 *
8254 * == \Array Indexes
8255 *
8256 * \Array indexing starts at 0, as in C or Java.
8257 *
8258 * A non-negative index is an offset from the first element:
8259 *
8260 * - Index 0 indicates the first element.
8261 * - Index 1 indicates the second element.
8262 * - ...
8263 *
8264 * A negative index is an offset, backwards, from the end of the array:
8265 *
8266 * - Index -1 indicates the last element.
8267 * - Index -2 indicates the next-to-last element.
8268 * - ...
8269 *
8270 *
8271 * === In-Range and Out-of-Range Indexes
8272 *
8273 * A non-negative index is <i>in range</i> if and only if it is smaller than
8274 * the size of the array. For a 3-element array:
8275 *
8276 * - Indexes 0 through 2 are in range.
8277 * - Index 3 is out of range.
8278 *
8279 * A negative index is <i>in range</i> if and only if its absolute value is
8280 * not larger than the size of the array. For a 3-element array:
8281 *
8282 * - Indexes -1 through -3 are in range.
8283 * - Index -4 is out of range.
8284 *
8285 * === Effective Index
8286 *
8287 * Although the effective index into an array is always an integer,
8288 * some methods (both within class \Array and elsewhere)
8289 * accept one or more non-integer arguments that are
8290 * {integer-convertible objects}[rdoc-ref:implicit_conversion.rdoc@Integer-Convertible+Objects].
8291 *
8292 * == Creating Arrays
8293 *
8294 * You can create an \Array object explicitly with:
8295 *
8296 * - An {array literal}[rdoc-ref:syntax/literals.rdoc@Array+Literals]:
8297 *
8298 * [1, 'one', :one, [2, 'two', :two]]
8299 *
8300 * - A {%w or %W string-array Literal}[rdoc-ref:syntax/literals.rdoc@25w+and+-25W-3A+String-Array+Literals]:
8301 *
8302 * %w[foo bar baz] # => ["foo", "bar", "baz"]
8303 * %w[1 % *] # => ["1", "%", "*"]
8304 *
8305 * - A {%i or %I symbol-array Literal}[rdoc-ref:syntax/literals.rdoc@25i+and+-25I-3A+Symbol-Array+Literals]:
8306 *
8307 * %i[foo bar baz] # => [:foo, :bar, :baz]
8308 * %i[1 % *] # => [:"1", :%, :*]
8309 *
8310 * - Method Kernel#Array:
8311 *
8312 * Array(["a", "b"]) # => ["a", "b"]
8313 * Array(1..5) # => [1, 2, 3, 4, 5]
8314 * Array(key: :value) # => [[:key, :value]]
8315 * Array(nil) # => []
8316 * Array(1) # => [1]
8317 * Array({:a => "a", :b => "b"}) # => [[:a, "a"], [:b, "b"]]
8318 *
8319 * - Method Array.new:
8320 *
8321 * Array.new # => []
8322 * Array.new(3) # => [nil, nil, nil]
8323 * Array.new(4) {Hash.new} # => [{}, {}, {}, {}]
8324 * Array.new(3, true) # => [true, true, true]
8325 *
8326 * Note that the last example above populates the array
8327 * with references to the same object.
8328 * This is recommended only in cases where that object is a natively immutable object
8329 * such as a symbol, a numeric, +nil+, +true+, or +false+.
8330 *
8331 * Another way to create an array with various objects, using a block;
8332 * this usage is safe for mutable objects such as hashes, strings or
8333 * other arrays:
8334 *
8335 * Array.new(4) {|i| i.to_s } # => ["0", "1", "2", "3"]
8336 *
8337 * Here is a way to create a multi-dimensional array:
8338 *
8339 * Array.new(3) {Array.new(3)}
8340 * # => [[nil, nil, nil], [nil, nil, nil], [nil, nil, nil]]
8341 *
8342 * A number of Ruby methods, both in the core and in the standard library,
8343 * provide instance method +to_a+, which converts an object to an array.
8344 *
8345 * - ARGF#to_a
8346 * - Array#to_a
8347 * - Enumerable#to_a
8348 * - Hash#to_a
8349 * - MatchData#to_a
8350 * - NilClass#to_a
8351 * - OptionParser#to_a
8352 * - Range#to_a
8353 * - Set#to_a
8354 * - Struct#to_a
8355 * - Time#to_a
8356 * - Benchmark::Tms#to_a
8357 * - CSV::Table#to_a
8358 * - Enumerator::Lazy#to_a
8359 * - Gem::List#to_a
8360 * - Gem::NameTuple#to_a
8361 * - Gem::Platform#to_a
8362 * - Gem::RequestSet::Lockfile::Tokenizer#to_a
8363 * - Gem::SourceList#to_a
8364 * - OpenSSL::X509::Extension#to_a
8365 * - OpenSSL::X509::Name#to_a
8366 * - Racc::ISet#to_a
8367 * - Rinda::RingFinger#to_a
8368 * - Ripper::Lexer::Elem#to_a
8369 * - RubyVM::InstructionSequence#to_a
8370 * - YAML::DBM#to_a
8371 *
8372 * == Example Usage
8373 *
8374 * In addition to the methods it mixes in through the Enumerable module,
8375 * class \Array has proprietary methods for accessing, searching and otherwise
8376 * manipulating arrays.
8377 *
8378 * Some of the more common ones are illustrated below.
8379 *
8380 * == Accessing Elements
8381 *
8382 * Elements in an array can be retrieved using the Array#[] method. It can
8383 * take a single integer argument (a numeric index), a pair of arguments
8384 * (start and length) or a range. Negative indices start counting from the end,
8385 * with -1 being the last element.
8386 *
8387 * arr = [1, 2, 3, 4, 5, 6]
8388 * arr[2] #=> 3
8389 * arr[100] #=> nil
8390 * arr[-3] #=> 4
8391 * arr[2, 3] #=> [3, 4, 5]
8392 * arr[1..4] #=> [2, 3, 4, 5]
8393 * arr[1..-3] #=> [2, 3, 4]
8394 *
8395 * Another way to access a particular array element is by using the #at method
8396 *
8397 * arr.at(0) #=> 1
8398 *
8399 * The #slice method works in an identical manner to Array#[].
8400 *
8401 * To raise an error for indices outside of the array bounds or else to
8402 * provide a default value when that happens, you can use #fetch.
8403 *
8404 * arr = ['a', 'b', 'c', 'd', 'e', 'f']
8405 * arr.fetch(100) #=> IndexError: index 100 outside of array bounds: -6...6
8406 * arr.fetch(100, "oops") #=> "oops"
8407 *
8408 * The special methods #first and #last will return the first and last
8409 * elements of an array, respectively.
8410 *
8411 * arr.first #=> 1
8412 * arr.last #=> 6
8413 *
8414 * To return the first +n+ elements of an array, use #take
8415 *
8416 * arr.take(3) #=> [1, 2, 3]
8417 *
8418 * #drop does the opposite of #take, by returning the elements after +n+
8419 * elements have been dropped:
8420 *
8421 * arr.drop(3) #=> [4, 5, 6]
8422 *
8423 * == Obtaining Information about an \Array
8424 *
8425 * An array keeps track of its own length at all times. To query an array
8426 * about the number of elements it contains, use #length, #count or #size.
8427 *
8428 * browsers = ['Chrome', 'Firefox', 'Safari', 'Opera', 'IE']
8429 * browsers.length #=> 5
8430 * browsers.count #=> 5
8431 *
8432 * To check whether an array contains any elements at all
8433 *
8434 * browsers.empty? #=> false
8435 *
8436 * To check whether a particular item is included in the array
8437 *
8438 * browsers.include?('Konqueror') #=> false
8439 *
8440 * == Adding Items to an \Array
8441 *
8442 * Items can be added to the end of an array by using either #push or #<<
8443 *
8444 * arr = [1, 2, 3, 4]
8445 * arr.push(5) #=> [1, 2, 3, 4, 5]
8446 * arr << 6 #=> [1, 2, 3, 4, 5, 6]
8447 *
8448 * #unshift will add a new item to the beginning of an array.
8449 *
8450 * arr.unshift(0) #=> [0, 1, 2, 3, 4, 5, 6]
8451 *
8452 * With #insert you can add a new element to an array at any position.
8453 *
8454 * arr.insert(3, 'apple') #=> [0, 1, 2, 'apple', 3, 4, 5, 6]
8455 *
8456 * Using the #insert method, you can also insert multiple values at once:
8457 *
8458 * arr.insert(3, 'orange', 'pear', 'grapefruit')
8459 * #=> [0, 1, 2, "orange", "pear", "grapefruit", "apple", 3, 4, 5, 6]
8460 *
8461 * == Removing Items from an \Array
8462 *
8463 * The method #pop removes the last element in an array and returns it:
8464 *
8465 * arr = [1, 2, 3, 4, 5, 6]
8466 * arr.pop #=> 6
8467 * arr #=> [1, 2, 3, 4, 5]
8468 *
8469 * To retrieve and at the same time remove the first item, use #shift:
8470 *
8471 * arr.shift #=> 1
8472 * arr #=> [2, 3, 4, 5]
8473 *
8474 * To delete an element at a particular index:
8475 *
8476 * arr.delete_at(2) #=> 4
8477 * arr #=> [2, 3, 5]
8478 *
8479 * To delete a particular element anywhere in an array, use #delete:
8480 *
8481 * arr = [1, 2, 2, 3]
8482 * arr.delete(2) #=> 2
8483 * arr #=> [1,3]
8484 *
8485 * A useful method if you need to remove +nil+ values from an array is
8486 * #compact:
8487 *
8488 * arr = ['foo', 0, nil, 'bar', 7, 'baz', nil]
8489 * arr.compact #=> ['foo', 0, 'bar', 7, 'baz']
8490 * arr #=> ['foo', 0, nil, 'bar', 7, 'baz', nil]
8491 * arr.compact! #=> ['foo', 0, 'bar', 7, 'baz']
8492 * arr #=> ['foo', 0, 'bar', 7, 'baz']
8493 *
8494 * Another common need is to remove duplicate elements from an array.
8495 *
8496 * It has the non-destructive #uniq, and destructive method #uniq!
8497 *
8498 * arr = [2, 5, 6, 556, 6, 6, 8, 9, 0, 123, 556]
8499 * arr.uniq #=> [2, 5, 6, 556, 8, 9, 0, 123]
8500 *
8501 * == Iterating over an \Array
8502 *
8503 * Like all classes that include the Enumerable module, class \Array has an each
8504 * method, which defines what elements should be iterated over and how. In
8505 * case of Array#each, all elements in +self+ are yielded to
8506 * the supplied block in sequence.
8507 *
8508 * Note that this operation leaves the array unchanged.
8509 *
8510 * arr = [1, 2, 3, 4, 5]
8511 * arr.each {|a| print a -= 10, " "}
8512 * # prints: -9 -8 -7 -6 -5
8513 * #=> [1, 2, 3, 4, 5]
8514 *
8515 * Another sometimes useful iterator is #reverse_each which will iterate over
8516 * the elements in the array in reverse order.
8517 *
8518 * words = %w[first second third fourth fifth sixth]
8519 * str = ""
8520 * words.reverse_each {|word| str += "#{word} "}
8521 * p str #=> "sixth fifth fourth third second first "
8522 *
8523 * The #map method can be used to create a new array based on the original
8524 * array, but with the values modified by the supplied block:
8525 *
8526 * arr.map {|a| 2*a} #=> [2, 4, 6, 8, 10]
8527 * arr #=> [1, 2, 3, 4, 5]
8528 * arr.map! {|a| a**2} #=> [1, 4, 9, 16, 25]
8529 * arr #=> [1, 4, 9, 16, 25]
8530 *
8531 *
8532 * == Selecting Items from an \Array
8533 *
8534 * Elements can be selected from an array according to criteria defined in a
8535 * block. The selection can happen in a destructive or a non-destructive
8536 * manner. While the destructive operations will modify the array they were
8537 * called on, the non-destructive methods usually return a new array with the
8538 * selected elements, but leave the original array unchanged.
8539 *
8540 * === Non-destructive Selection
8541 *
8542 * arr = [1, 2, 3, 4, 5, 6]
8543 * arr.select {|a| a > 3} #=> [4, 5, 6]
8544 * arr.reject {|a| a < 3} #=> [3, 4, 5, 6]
8545 * arr.drop_while {|a| a < 4} #=> [4, 5, 6]
8546 * arr #=> [1, 2, 3, 4, 5, 6]
8547 *
8548 * === Destructive Selection
8549 *
8550 * #select! and #reject! are the corresponding destructive methods to #select
8551 * and #reject
8552 *
8553 * Similar to #select vs. #reject, #delete_if and #keep_if have the exact
8554 * opposite result when supplied with the same block:
8555 *
8556 * arr.delete_if {|a| a < 4} #=> [4, 5, 6]
8557 * arr #=> [4, 5, 6]
8558 *
8559 * arr = [1, 2, 3, 4, 5, 6]
8560 * arr.keep_if {|a| a < 4} #=> [1, 2, 3]
8561 * arr #=> [1, 2, 3]
8562 *
8563 * == What's Here
8564 *
8565 * First, what's elsewhere. Class \Array:
8566 *
8567 * - Inherits from {class Object}[rdoc-ref:Object@What-27s+Here].
8568 * - Includes {module Enumerable}[rdoc-ref:Enumerable@What-27s+Here],
8569 * which provides dozens of additional methods.
8570 *
8571 * Here, class \Array provides methods that are useful for:
8572 *
8573 * - {Creating an Array}[rdoc-ref:Array@Methods+for+Creating+an+Array]
8574 * - {Querying}[rdoc-ref:Array@Methods+for+Querying]
8575 * - {Comparing}[rdoc-ref:Array@Methods+for+Comparing]
8576 * - {Fetching}[rdoc-ref:Array@Methods+for+Fetching]
8577 * - {Assigning}[rdoc-ref:Array@Methods+for+Assigning]
8578 * - {Deleting}[rdoc-ref:Array@Methods+for+Deleting]
8579 * - {Combining}[rdoc-ref:Array@Methods+for+Combining]
8580 * - {Iterating}[rdoc-ref:Array@Methods+for+Iterating]
8581 * - {Converting}[rdoc-ref:Array@Methods+for+Converting]
8582 * - {And more....}[rdoc-ref:Array@Other+Methods]
8583 *
8584 * === Methods for Creating an \Array
8585 *
8586 * - ::[]: Returns a new array populated with given objects.
8587 * - ::new: Returns a new array.
8588 * - ::try_convert: Returns a new array created from a given object.
8589 *
8590 * See also {Creating Arrays}[rdoc-ref:Array@Creating+Arrays].
8591 *
8592 * === Methods for Querying
8593 *
8594 * - #all?: Returns whether all elements meet a given criterion.
8595 * - #any?: Returns whether any element meets a given criterion.
8596 * - #count: Returns the count of elements that meet a given criterion.
8597 * - #empty?: Returns whether there are no elements.
8598 * - #find_index (aliased as #index): Returns the index of the first element that meets a given criterion.
8599 * - #hash: Returns the integer hash code.
8600 * - #include?: Returns whether any element <tt>==</tt> a given object.
8601 * - #length (aliased as #size): Returns the count of elements.
8602 * - #none?: Returns whether no element <tt>==</tt> a given object.
8603 * - #one?: Returns whether exactly one element <tt>==</tt> a given object.
8604 * - #rindex: Returns the index of the last element that meets a given criterion.
8605 *
8606 * === Methods for Comparing
8607 *
8608 * - #<=>: Returns -1, 0, or 1, as +self+ is less than, equal to, or greater than a given object.
8609 * - #==: Returns whether each element in +self+ is <tt>==</tt> to the corresponding element in a given object.
8610 * - #eql?: Returns whether each element in +self+ is <tt>eql?</tt> to the corresponding element in a given object.
8611
8612 * === Methods for Fetching
8613 *
8614 * These methods do not modify +self+.
8615 *
8616 * - #[] (aliased as #slice): Returns consecutive elements as determined by a given argument.
8617 * - #assoc: Returns the first element that is an array whose first element <tt>==</tt> a given object.
8618 * - #at: Returns the element at a given offset.
8619 * - #bsearch: Returns an element selected via a binary search as determined by a given block.
8620 * - #bsearch_index: Returns the index of an element selected via a binary search as determined by a given block.
8621 * - #compact: Returns an array containing all non-+nil+ elements.
8622 * - #dig: Returns the object in nested objects that is specified by a given index and additional arguments.
8623 * - #drop: Returns trailing elements as determined by a given index.
8624 * - #drop_while: Returns trailing elements as determined by a given block.
8625 * - #fetch: Returns the element at a given offset.
8626 * - #fetch_values: Returns elements at given offsets.
8627 * - #first: Returns one or more leading elements.
8628 * - #last: Returns one or more trailing elements.
8629 * - #max: Returns one or more maximum-valued elements, as determined by <tt>#<=></tt> or a given block.
8630 * - #min: Returns one or more minimum-valued elements, as determined by <tt>#<=></tt> or a given block.
8631 * - #minmax: Returns the minimum-valued and maximum-valued elements, as determined by <tt>#<=></tt> or a given block.
8632 * - #rassoc: Returns the first element that is an array whose second element <tt>==</tt> a given object.
8633 * - #reject: Returns an array containing elements not rejected by a given block.
8634 * - #reverse: Returns all elements in reverse order.
8635 * - #rotate: Returns all elements with some rotated from one end to the other.
8636 * - #sample: Returns one or more random elements.
8637 * - #select (aliased as #filter): Returns an array containing elements selected by a given block.
8638 * - #shuffle: Returns elements in a random order.
8639 * - #sort: Returns all elements in an order determined by <tt>#<=></tt> or a given block.
8640 * - #take: Returns leading elements as determined by a given index.
8641 * - #take_while: Returns leading elements as determined by a given block.
8642 * - #uniq: Returns an array containing non-duplicate elements.
8643 * - #values_at: Returns the elements at given offsets.
8644 *
8645 * === Methods for Assigning
8646 *
8647 * These methods add, replace, or reorder elements in +self+.
8648 *
8649 * - #<<: Appends an element.
8650 * - #[]=: Assigns specified elements with a given object.
8651 * - #concat: Appends all elements from given arrays.
8652 * - #fill: Replaces specified elements with specified objects.
8653 * - #flatten!: Replaces each nested array in +self+ with the elements from that array.
8654 * - #initialize_copy (aliased as #replace): Replaces the content of +self+ with the content of a given array.
8655 * - #insert: Inserts given objects at a given offset; does not replace elements.
8656 * - #push (aliased as #append): Appends elements.
8657 * - #reverse!: Replaces +self+ with its elements reversed.
8658 * - #rotate!: Replaces +self+ with its elements rotated.
8659 * - #shuffle!: Replaces +self+ with its elements in random order.
8660 * - #sort!: Replaces +self+ with its elements sorted, as determined by <tt>#<=></tt> or a given block.
8661 * - #sort_by!: Replaces +self+ with its elements sorted, as determined by a given block.
8662 * - #unshift (aliased as #prepend): Prepends leading elements.
8663 *
8664 * === Methods for Deleting
8665 *
8666 * Each of these methods removes elements from +self+:
8667 *
8668 * - #clear: Removes all elements.
8669 * - #compact!: Removes all +nil+ elements.
8670 * - #delete: Removes elements equal to a given object.
8671 * - #delete_at: Removes the element at a given offset.
8672 * - #delete_if: Removes elements specified by a given block.
8673 * - #keep_if: Removes elements not specified by a given block.
8674 * - #pop: Removes and returns the last element.
8675 * - #reject!: Removes elements specified by a given block.
8676 * - #select! (aliased as #filter!): Removes elements not specified by a given block.
8677 * - #shift: Removes and returns the first element.
8678 * - #slice!: Removes and returns a sequence of elements.
8679 * - #uniq!: Removes duplicates.
8680 *
8681 * === Methods for Combining
8682 *
8683 * - #&: Returns an array containing elements found both in +self+ and a given array.
8684 * - #+: Returns an array containing all elements of +self+ followed by all elements of a given array.
8685 * - #-: Returns an array containing all elements of +self+ that are not found in a given array.
8686 * - #|: Returns an array containing all element of +self+ and all elements of a given array, duplicates removed.
8687 * - #difference: Returns an array containing all elements of +self+ that are not found in any of the given arrays..
8688 * - #intersection: Returns an array containing elements found both in +self+ and in each given array.
8689 * - #product: Returns or yields all combinations of elements from +self+ and given arrays.
8690 * - #reverse: Returns an array containing all elements of +self+ in reverse order.
8691 * - #union: Returns an array containing all elements of +self+ and all elements of given arrays, duplicates removed.
8692 *
8693 * === Methods for Iterating
8694 *
8695 * - #combination: Calls a given block with combinations of elements of +self+; a combination does not use the same element more than once.
8696 * - #cycle: Calls a given block with each element, then does so again, for a specified number of times, or forever.
8697 * - #each: Passes each element to a given block.
8698 * - #each_index: Passes each element index to a given block.
8699 * - #permutation: Calls a given block with permutations of elements of +self+; a permutation does not use the same element more than once.
8700 * - #repeated_combination: Calls a given block with combinations of elements of +self+; a combination may use the same element more than once.
8701 * - #repeated_permutation: Calls a given block with permutations of elements of +self+; a permutation may use the same element more than once.
8702 * - #reverse_each: Passes each element, in reverse order, to a given block.
8703 *
8704 * === Methods for Converting
8705 *
8706 * - #collect (aliased as #map): Returns an array containing the block return-value for each element.
8707 * - #collect! (aliased as #map!): Replaces each element with a block return-value.
8708 * - #flatten: Returns an array that is a recursive flattening of +self+.
8709 * - #inspect (aliased as #to_s): Returns a new String containing the elements.
8710 * - #join: Returns a newsString containing the elements joined by the field separator.
8711 * - #to_a: Returns +self+ or a new array containing all elements.
8712 * - #to_ary: Returns +self+.
8713 * - #to_h: Returns a new hash formed from the elements.
8714 * - #transpose: Transposes +self+, which must be an array of arrays.
8715 * - #zip: Returns a new array of arrays containing +self+ and given arrays.
8716 *
8717 * === Other Methods
8718 *
8719 * - #*: Returns one of the following:
8720 *
8721 * - With integer argument +n+, a new array that is the concatenation
8722 * of +n+ copies of +self+.
8723 * - With string argument +field_separator+, a new string that is equivalent to
8724 * <tt>join(field_separator)</tt>.
8725 *
8726 * - #pack: Packs the elements into a binary sequence.
8727 * - #sum: Returns a sum of elements according to either <tt>+</tt> or a given block.
8728 */
8729
8730void
8731Init_Array(void)
8732{
8733 fake_ary_flags = init_fake_ary_flags();
8734
8735 rb_cArray = rb_define_class("Array", rb_cObject);
8737
8738 rb_define_alloc_func(rb_cArray, empty_ary_alloc);
8739 rb_define_singleton_method(rb_cArray, "new", rb_ary_s_new, -1);
8740 rb_define_singleton_method(rb_cArray, "[]", rb_ary_s_create, -1);
8741 rb_define_singleton_method(rb_cArray, "try_convert", rb_ary_s_try_convert, 1);
8742 rb_define_method(rb_cArray, "initialize", rb_ary_initialize, -1);
8743 rb_define_method(rb_cArray, "initialize_copy", rb_ary_replace, 1);
8744
8745 rb_define_method(rb_cArray, "inspect", rb_ary_inspect, 0);
8746 rb_define_alias(rb_cArray, "to_s", "inspect");
8747 rb_define_method(rb_cArray, "to_a", rb_ary_to_a, 0);
8748 rb_define_method(rb_cArray, "to_h", rb_ary_to_h, 0);
8749 rb_define_method(rb_cArray, "to_ary", rb_ary_to_ary_m, 0);
8750
8751 rb_define_method(rb_cArray, "==", rb_ary_equal, 1);
8752 rb_define_method(rb_cArray, "eql?", rb_ary_eql, 1);
8753 rb_define_method(rb_cArray, "hash", rb_ary_hash, 0);
8754
8755 rb_define_method(rb_cArray, "[]", rb_ary_aref, -1);
8756 rb_define_method(rb_cArray, "[]=", rb_ary_aset, -1);
8757 rb_define_method(rb_cArray, "at", rb_ary_at, 1);
8758 rb_define_method(rb_cArray, "fetch", rb_ary_fetch, -1);
8759 rb_define_method(rb_cArray, "concat", rb_ary_concat_multi, -1);
8760 rb_define_method(rb_cArray, "union", rb_ary_union_multi, -1);
8761 rb_define_method(rb_cArray, "difference", rb_ary_difference_multi, -1);
8762 rb_define_method(rb_cArray, "intersection", rb_ary_intersection_multi, -1);
8763 rb_define_method(rb_cArray, "intersect?", rb_ary_intersect_p, 1);
8764 rb_define_method(rb_cArray, "<<", rb_ary_push, 1);
8765 rb_define_method(rb_cArray, "push", rb_ary_push_m, -1);
8766 rb_define_alias(rb_cArray, "append", "push");
8767 rb_define_method(rb_cArray, "pop", rb_ary_pop_m, -1);
8768 rb_define_method(rb_cArray, "shift", rb_ary_shift_m, -1);
8769 rb_define_method(rb_cArray, "unshift", rb_ary_unshift_m, -1);
8770 rb_define_alias(rb_cArray, "prepend", "unshift");
8771 rb_define_method(rb_cArray, "insert", rb_ary_insert, -1);
8772 rb_define_method(rb_cArray, "each", rb_ary_each, 0);
8773 rb_define_method(rb_cArray, "each_index", rb_ary_each_index, 0);
8774 rb_define_method(rb_cArray, "reverse_each", rb_ary_reverse_each, 0);
8775 rb_define_method(rb_cArray, "length", rb_ary_length, 0);
8776 rb_define_method(rb_cArray, "size", rb_ary_length, 0);
8777 rb_define_method(rb_cArray, "empty?", rb_ary_empty_p, 0);
8778 rb_define_method(rb_cArray, "find_index", rb_ary_index, -1);
8779 rb_define_method(rb_cArray, "index", rb_ary_index, -1);
8780 rb_define_method(rb_cArray, "rindex", rb_ary_rindex, -1);
8781 rb_define_method(rb_cArray, "join", rb_ary_join_m, -1);
8782 rb_define_method(rb_cArray, "reverse", rb_ary_reverse_m, 0);
8783 rb_define_method(rb_cArray, "reverse!", rb_ary_reverse_bang, 0);
8784 rb_define_method(rb_cArray, "rotate", rb_ary_rotate_m, -1);
8785 rb_define_method(rb_cArray, "rotate!", rb_ary_rotate_bang, -1);
8786 rb_define_method(rb_cArray, "sort", rb_ary_sort, 0);
8787 rb_define_method(rb_cArray, "sort!", rb_ary_sort_bang, 0);
8788 rb_define_method(rb_cArray, "sort_by!", rb_ary_sort_by_bang, 0);
8789 rb_define_method(rb_cArray, "collect", rb_ary_collect, 0);
8790 rb_define_method(rb_cArray, "collect!", rb_ary_collect_bang, 0);
8791 rb_define_method(rb_cArray, "map", rb_ary_collect, 0);
8792 rb_define_method(rb_cArray, "map!", rb_ary_collect_bang, 0);
8793 rb_define_method(rb_cArray, "select", rb_ary_select, 0);
8794 rb_define_method(rb_cArray, "select!", rb_ary_select_bang, 0);
8795 rb_define_method(rb_cArray, "filter", rb_ary_select, 0);
8796 rb_define_method(rb_cArray, "filter!", rb_ary_select_bang, 0);
8797 rb_define_method(rb_cArray, "keep_if", rb_ary_keep_if, 0);
8798 rb_define_method(rb_cArray, "values_at", rb_ary_values_at, -1);
8799 rb_define_method(rb_cArray, "delete", rb_ary_delete, 1);
8800 rb_define_method(rb_cArray, "delete_at", rb_ary_delete_at_m, 1);
8801 rb_define_method(rb_cArray, "delete_if", rb_ary_delete_if, 0);
8802 rb_define_method(rb_cArray, "reject", rb_ary_reject, 0);
8803 rb_define_method(rb_cArray, "reject!", rb_ary_reject_bang, 0);
8804 rb_define_method(rb_cArray, "zip", rb_ary_zip, -1);
8805 rb_define_method(rb_cArray, "transpose", rb_ary_transpose, 0);
8806 rb_define_method(rb_cArray, "replace", rb_ary_replace, 1);
8807 rb_define_method(rb_cArray, "clear", rb_ary_clear, 0);
8808 rb_define_method(rb_cArray, "fill", rb_ary_fill, -1);
8809 rb_define_method(rb_cArray, "include?", rb_ary_includes, 1);
8810 rb_define_method(rb_cArray, "<=>", rb_ary_cmp, 1);
8811
8812 rb_define_method(rb_cArray, "slice", rb_ary_aref, -1);
8813 rb_define_method(rb_cArray, "slice!", rb_ary_slice_bang, -1);
8814
8815 rb_define_method(rb_cArray, "assoc", rb_ary_assoc, 1);
8816 rb_define_method(rb_cArray, "rassoc", rb_ary_rassoc, 1);
8817
8818 rb_define_method(rb_cArray, "+", rb_ary_plus, 1);
8819 rb_define_method(rb_cArray, "*", rb_ary_times, 1);
8820
8821 rb_define_method(rb_cArray, "-", rb_ary_diff, 1);
8822 rb_define_method(rb_cArray, "&", rb_ary_and, 1);
8823 rb_define_method(rb_cArray, "|", rb_ary_or, 1);
8824
8825 rb_define_method(rb_cArray, "max", rb_ary_max, -1);
8826 rb_define_method(rb_cArray, "min", rb_ary_min, -1);
8827 rb_define_method(rb_cArray, "minmax", rb_ary_minmax, 0);
8828
8829 rb_define_method(rb_cArray, "uniq", rb_ary_uniq, 0);
8830 rb_define_method(rb_cArray, "uniq!", rb_ary_uniq_bang, 0);
8831 rb_define_method(rb_cArray, "compact", rb_ary_compact, 0);
8832 rb_define_method(rb_cArray, "compact!", rb_ary_compact_bang, 0);
8833 rb_define_method(rb_cArray, "flatten", rb_ary_flatten, -1);
8834 rb_define_method(rb_cArray, "flatten!", rb_ary_flatten_bang, -1);
8835 rb_define_method(rb_cArray, "count", rb_ary_count, -1);
8836 rb_define_method(rb_cArray, "cycle", rb_ary_cycle, -1);
8837 rb_define_method(rb_cArray, "permutation", rb_ary_permutation, -1);
8838 rb_define_method(rb_cArray, "combination", rb_ary_combination, 1);
8839 rb_define_method(rb_cArray, "repeated_permutation", rb_ary_repeated_permutation, 1);
8840 rb_define_method(rb_cArray, "repeated_combination", rb_ary_repeated_combination, 1);
8841 rb_define_method(rb_cArray, "product", rb_ary_product, -1);
8842
8843 rb_define_method(rb_cArray, "take", rb_ary_take, 1);
8844 rb_define_method(rb_cArray, "take_while", rb_ary_take_while, 0);
8845 rb_define_method(rb_cArray, "drop", rb_ary_drop, 1);
8846 rb_define_method(rb_cArray, "drop_while", rb_ary_drop_while, 0);
8847 rb_define_method(rb_cArray, "bsearch", rb_ary_bsearch, 0);
8848 rb_define_method(rb_cArray, "bsearch_index", rb_ary_bsearch_index, 0);
8849 rb_define_method(rb_cArray, "any?", rb_ary_any_p, -1);
8850 rb_define_method(rb_cArray, "all?", rb_ary_all_p, -1);
8851 rb_define_method(rb_cArray, "none?", rb_ary_none_p, -1);
8852 rb_define_method(rb_cArray, "one?", rb_ary_one_p, -1);
8853 rb_define_method(rb_cArray, "dig", rb_ary_dig, -1);
8854 rb_define_method(rb_cArray, "sum", rb_ary_sum, -1);
8855 rb_define_method(rb_cArray, "freeze", rb_ary_freeze, 0);
8856
8857 rb_define_method(rb_cArray, "deconstruct", rb_ary_deconstruct, 0);
8858
8859 rb_cArray_empty_frozen = rb_ary_freeze(rb_ary_new());
8860 rb_vm_register_global_object(rb_cArray_empty_frozen);
8861}
8862
8863#include "array.rbinc"
#define RUBY_ASSERT_ALWAYS(expr,...)
A variant of RUBY_ASSERT that does not interface with RUBY_DEBUG.
Definition assert.h:199
#define RUBY_ASSERT(...)
Asserts that the given expression is truthy if and only if RUBY_DEBUG is truthy.
Definition assert.h:219
#define rb_define_method(klass, mid, func, arity)
Defines klass#mid.
#define rb_define_singleton_method(klass, mid, func, arity)
Defines klass.mid.
void rb_include_module(VALUE klass, VALUE module)
Includes a module to a class.
Definition class.c:1190
VALUE rb_define_class(const char *name, VALUE super)
Defines a top-level class.
Definition class.c:980
void rb_define_alias(VALUE klass, const char *name1, const char *name2)
Defines an alias of a method.
Definition class.c:2348
int rb_scan_args(int argc, const VALUE *argv, const char *fmt,...)
Retrieves argument from argc and argv to given VALUE references according to the format string.
Definition class.c:2638
int rb_block_given_p(void)
Determines if the current method is given a block.
Definition eval.c:936
#define FL_UNSET_RAW
Old name of RB_FL_UNSET_RAW.
Definition fl_type.h:134
#define rb_str_buf_cat2
Old name of rb_usascii_str_new_cstr.
Definition string.h:1682
#define RFLOAT_VALUE
Old name of rb_float_value.
Definition double.h:28
#define T_STRING
Old name of RUBY_T_STRING.
Definition value_type.h:78
#define Qundef
Old name of RUBY_Qundef.
#define INT2FIX
Old name of RB_INT2FIX.
Definition long.h:48
#define OBJ_FROZEN
Old name of RB_OBJ_FROZEN.
Definition fl_type.h:137
#define rb_str_buf_new2
Old name of rb_str_buf_new_cstr.
Definition string.h:1679
#define rb_ary_new4
Old name of rb_ary_new_from_values.
Definition array.h:659
#define FIXABLE
Old name of RB_FIXABLE.
Definition fixnum.h:25
#define LONG2FIX
Old name of RB_INT2FIX.
Definition long.h:49
#define T_RATIONAL
Old name of RUBY_T_RATIONAL.
Definition value_type.h:76
#define ALLOC_N
Old name of RB_ALLOC_N.
Definition memory.h:399
#define NUM2DBL
Old name of rb_num2dbl.
Definition double.h:27
#define FL_SET
Old name of RB_FL_SET.
Definition fl_type.h:129
#define rb_ary_new3
Old name of rb_ary_new_from_args.
Definition array.h:658
#define LONG2NUM
Old name of RB_LONG2NUM.
Definition long.h:50
#define rb_usascii_str_new2
Old name of rb_usascii_str_new_cstr.
Definition string.h:1680
#define Qtrue
Old name of RUBY_Qtrue.
#define ST2FIX
Old name of RB_ST2FIX.
Definition st_data_t.h:33
#define NUM2INT
Old name of RB_NUM2INT.
Definition int.h:44
#define Qnil
Old name of RUBY_Qnil.
#define Qfalse
Old name of RUBY_Qfalse.
#define FIX2LONG
Old name of RB_FIX2LONG.
Definition long.h:46
#define T_ARRAY
Old name of RUBY_T_ARRAY.
Definition value_type.h:56
#define NIL_P
Old name of RB_NIL_P.
#define ALLOCV_N
Old name of RB_ALLOCV_N.
Definition memory.h:405
#define FL_WB_PROTECTED
Old name of RUBY_FL_WB_PROTECTED.
Definition fl_type.h:59
#define DBL2NUM
Old name of rb_float_new.
Definition double.h:29
#define FL_TEST
Old name of RB_FL_TEST.
Definition fl_type.h:131
#define FL_FREEZE
Old name of RUBY_FL_FREEZE.
Definition fl_type.h:67
#define NUM2LONG
Old name of RB_NUM2LONG.
Definition long.h:51
#define FL_UNSET
Old name of RB_FL_UNSET.
Definition fl_type.h:133
#define FIXNUM_P
Old name of RB_FIXNUM_P.
#define rb_ary_new2
Old name of rb_ary_new_capa.
Definition array.h:657
#define FL_SET_RAW
Old name of RB_FL_SET_RAW.
Definition fl_type.h:130
#define ALLOCV_END
Old name of RB_ALLOCV_END.
Definition memory.h:406
void rb_category_warn(rb_warning_category_t category, const char *fmt,...)
Identical to rb_category_warning(), except it reports unless $VERBOSE is nil.
Definition error.c:476
void rb_iter_break(void)
Breaks from a block.
Definition vm.c:2086
VALUE rb_eFrozenError
FrozenError exception.
Definition error.c:1429
VALUE rb_eRangeError
RangeError exception.
Definition error.c:1434
VALUE rb_eTypeError
TypeError exception.
Definition error.c:1430
VALUE rb_eRuntimeError
RuntimeError exception.
Definition error.c:1428
void rb_warn(const char *fmt,...)
Identical to rb_warning(), except it reports unless $VERBOSE is nil.
Definition error.c:466
VALUE rb_eIndexError
IndexError exception.
Definition error.c:1432
void rb_warning(const char *fmt,...)
Issues a warning.
Definition error.c:497
@ RB_WARN_CATEGORY_DEPRECATED
Warning is for deprecated features.
Definition error.h:48
VALUE rb_cArray
Array class.
Definition array.c:40
VALUE rb_mEnumerable
Enumerable module.
Definition enum.c:27
VALUE rb_obj_hide(VALUE obj)
Make the object invisible from Ruby code.
Definition object.c:104
VALUE rb_class_new_instance_pass_kw(int argc, const VALUE *argv, VALUE klass)
Identical to rb_class_new_instance(), except it passes the passed keywords if any to the #initialize ...
Definition object.c:2115
VALUE rb_obj_frozen_p(VALUE obj)
Just calls RB_OBJ_FROZEN() inside.
Definition object.c:1272
int rb_eql(VALUE lhs, VALUE rhs)
Checks for equality of the passed objects, in terms of Object#eql?.
Definition object.c:192
VALUE rb_cNumeric
Numeric class.
Definition numeric.c:196
VALUE rb_cRandom
Random class.
Definition random.c:236
VALUE rb_obj_class(VALUE obj)
Queries the class of an object.
Definition object.c:247
VALUE rb_inspect(VALUE obj)
Generates a human-readable textual representation of the given object.
Definition object.c:680
double rb_num2dbl(VALUE num)
Converts an instance of rb_cNumeric into C's double.
Definition object.c:3715
VALUE rb_equal(VALUE lhs, VALUE rhs)
This function is an optimised version of calling #==.
Definition object.c:179
VALUE rb_obj_is_kind_of(VALUE obj, VALUE klass)
Queries if the given object is an instance (of possibly descendants) of the given class.
Definition object.c:865
VALUE rb_obj_freeze(VALUE obj)
Just calls rb_obj_freeze_inline() inside.
Definition object.c:1260
#define RB_OBJ_WRITTEN(old, oldv, young)
Identical to RB_OBJ_WRITE(), except it doesn't write any values, but only a WB declaration.
Definition gc.h:615
#define RB_OBJ_WRITE(old, slot, young)
Declaration of a "back" pointer.
Definition gc.h:603
Encoding relates APIs.
VALUE rb_funcall(VALUE recv, ID mid, int n,...)
Calls a method.
Definition vm_eval.c:1099
VALUE rb_call_super(int argc, const VALUE *argv)
This resembles ruby's super.
Definition vm_eval.c:362
#define RGENGC_WB_PROTECTED_ARRAY
This is a compile-time flag to enable/disable write barrier for struct RArray.
Definition gc.h:446
Defines RBIMPL_HAS_BUILTIN.
#define RETURN_SIZED_ENUMERATOR(obj, argc, argv, size_fn)
This roughly resembles return enum_for(__callee__) unless block_given?.
Definition enumerator.h:206
#define RETURN_ENUMERATOR(obj, argc, argv)
Identical to RETURN_SIZED_ENUMERATOR(), except its size is unknown.
Definition enumerator.h:239
#define UNLIMITED_ARGUMENTS
This macro is used in conjunction with rb_check_arity().
Definition error.h:35
static int rb_check_arity(int argc, int min, int max)
Ensures that the passed integer is in the passed range.
Definition error.h:284
VALUE rb_int_positive_pow(long x, unsigned long y)
Raises the passed x to the power of y.
Definition numeric.c:4559
VALUE rb_range_beg_len(VALUE range, long *begp, long *lenp, long len, int err)
Deconstructs a numerical range.
Definition range.c:1926
#define rb_hash_uint(h, i)
Just another name of st_hash_uint.
Definition string.h:942
#define rb_hash_end(h)
Just another name of st_hash_end.
Definition string.h:945
#define rb_str_new(str, len)
Allocates an instance of rb_cString.
Definition string.h:1498
#define rb_usascii_str_new(str, len)
Identical to rb_str_new, except it generates a string of "US ASCII" encoding.
Definition string.h:1532
#define rb_usascii_str_new_cstr(str)
Identical to rb_str_new_cstr, except it generates a string of "US ASCII" encoding.
Definition string.h:1567
VALUE rb_str_buf_append(VALUE dst, VALUE src)
Identical to rb_str_cat_cstr(), except it takes Ruby's string instead of C's.
Definition string.c:3661
void rb_str_set_len(VALUE str, long len)
Overwrites the length of the string.
Definition string.c:3287
st_index_t rb_hash_start(st_index_t i)
Starts a series of hashing.
Definition random.c:1746
int rb_str_cmp(VALUE lhs, VALUE rhs)
Compares two strings, as in strcmp(3).
Definition string.c:4120
VALUE rb_check_string_type(VALUE obj)
Try converting an object to its stringised representation using its to_str method,...
Definition string.c:2867
VALUE rb_str_buf_new(long capa)
Allocates a "string buffer".
Definition string.c:1660
VALUE rb_obj_as_string(VALUE obj)
Try converting an object to its stringised representation using its to_s method, if any.
Definition string.c:1792
VALUE rb_exec_recursive(VALUE(*f)(VALUE g, VALUE h, int r), VALUE g, VALUE h)
"Recursion" API entry point.
VALUE rb_exec_recursive_paired(VALUE(*f)(VALUE g, VALUE h, int r), VALUE g, VALUE p, VALUE h)
Identical to rb_exec_recursive(), except it checks for the recursion on the ordered pair of { g,...
int rb_respond_to(VALUE obj, ID mid)
Queries if the object responds to the method.
Definition vm_method.c:2956
void rb_define_alloc_func(VALUE klass, rb_alloc_func_t func)
Sets the allocator function of a class.
int capa
Designed capacity of the buffer.
Definition io.h:11
int len
Length of the buffer.
Definition io.h:8
void ruby_qsort(void *, const size_t, const size_t, int(*)(const void *, const void *, void *), void *)
Reentrant implementation of quick sort.
#define RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg)
Shim for block function parameters.
Definition iterator.h:58
VALUE rb_yield_values(int n,...)
Identical to rb_yield(), except it takes variadic number of parameters and pass them to the block.
Definition vm_eval.c:1366
VALUE rb_yield_values2(int n, const VALUE *argv)
Identical to rb_yield_values(), except it takes the parameters as a C array instead of variadic argum...
Definition vm_eval.c:1388
VALUE rb_yield(VALUE val)
Yields the block.
Definition vm_eval.c:1354
#define RBIMPL_ATTR_MAYBE_UNUSED()
Wraps (or simulates) [[maybe_unused]]
#define MEMCPY(p1, p2, type, n)
Handy macro to call memcpy.
Definition memory.h:372
#define MEMZERO(p, type, n)
Handy macro to erase a region of memory.
Definition memory.h:360
#define RB_GC_GUARD(v)
Prevents premature destruction of local objects.
Definition memory.h:167
#define MEMMOVE(p1, p2, type, n)
Handy macro to call memmove.
Definition memory.h:384
VALUE rb_block_call(VALUE q, ID w, int e, const VALUE *r, type *t, VALUE y)
Call a method with a block.
void rb_hash_foreach(VALUE q, int_type *w, VALUE e)
Iteration over the given hash.
VALUE rb_ensure(type *q, VALUE w, type *e, VALUE r)
An equivalent of ensure clause.
#define RARRAY_LEN
Just another name of rb_array_len.
Definition rarray.h:51
#define RARRAY(obj)
Convenient casting macro.
Definition rarray.h:44
static void RARRAY_ASET(VALUE ary, long i, VALUE v)
Assigns an object in an array.
Definition rarray.h:386
#define RARRAY_PTR_USE(ary, ptr_name, expr)
Declares a section of code where raw pointers are used.
Definition rarray.h:348
static VALUE * RARRAY_PTR(VALUE ary)
Wild use of a C pointer.
Definition rarray.h:366
#define RARRAY_AREF(a, i)
Definition rarray.h:403
#define RARRAY_CONST_PTR
Just another name of rb_array_const_ptr.
Definition rarray.h:52
#define RBASIC(obj)
Convenient casting macro.
Definition rbasic.h:40
void(* RUBY_DATA_FUNC)(void *)
This is the type of callbacks registered to RData.
Definition rdata.h:104
#define RHASH_SIZE(h)
Queries the size of the hash.
Definition rhash.h:69
#define StringValue(v)
Ensures that the parameter object is a String.
Definition rstring.h:66
#define RTYPEDDATA_DATA(v)
Convenient getter macro.
Definition rtypeddata.h:102
#define TypedData_Wrap_Struct(klass, data_type, sval)
Converts sval, a pointer to your struct, into a Ruby object.
Definition rtypeddata.h:449
#define RB_PASS_CALLED_KEYWORDS
Pass keywords if current method is called with keywords, useful for argument delegation.
Definition scan_args.h:78
#define RTEST
This is an old name of RB_TEST.
Ruby's array.
Definition rarray.h:128
union RArray::@47 as
Array's specific fields.
struct RBasic basic
Basic part, including flags and class.
Definition rarray.h:131
struct RArray::@47::@48 heap
Arrays that use separated memory region for elements use this pattern.
const VALUE shared_root
Parent of the array.
Definition rarray.h:166
const VALUE ary[1]
Embedded elements.
Definition rarray.h:188
long capa
Capacity of *ptr.
Definition rarray.h:153
union RArray::@47::@48::@49 aux
Auxiliary info.
long len
Number of elements of the array.
Definition rarray.h:143
const VALUE * ptr
Pointer to the C array that holds the elements of the array.
Definition rarray.h:175
VALUE flags
Per-object flags.
Definition rbasic.h:75
This is the struct that holds necessary info for a struct.
Definition rtypeddata.h:200
const char * wrap_struct_name
Name of structs of this kind.
Definition rtypeddata.h:207
Definition st.h:79
intptr_t SIGNED_VALUE
A signed integer type that has the same width with VALUE.
Definition value.h:63
uintptr_t VALUE
Type that represents a Ruby object.
Definition value.h:40
static bool RB_FLOAT_TYPE_P(VALUE obj)
Queries if the object is an instance of rb_cFloat.
Definition value_type.h:264
static bool RB_TYPE_P(VALUE obj, enum ruby_value_type t)
Queries if the given object is of given type.
Definition value_type.h:376