Skip to content

Commit c8aaf31

Browse files
committed
* array.c (rb_ary_resize): new utility function. [ruby-dev:42912]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@30465 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
1 parent 697a45b commit c8aaf31

File tree

5 files changed

+90
-0
lines changed

5 files changed

+90
-0
lines changed

array.c

+45
Original file line numberDiff line numberDiff line change
@@ -1308,6 +1308,51 @@ rb_ary_splice(VALUE ary, long beg, long len, VALUE rpl)
13081308
}
13091309
}
13101310

1311+
/*!
1312+
* expands or shrinks \a ary to \a len elements.
1313+
* expanded region will be filled with Qnil.
1314+
* \param ary an arrray
1315+
* \param len new size
1316+
* \return \a ary
1317+
* \post the size of \a ary is \a len.
1318+
*/
1319+
VALUE
1320+
rb_ary_resize(VALUE ary, long len)
1321+
{
1322+
long olen;
1323+
1324+
rb_ary_modify(ary);
1325+
olen = RARRAY_LEN(ary);
1326+
if (len == olen) return ary;
1327+
if (len > ARY_MAX_SIZE) {
1328+
rb_raise(rb_eIndexError, "index %ld too big", len);
1329+
}
1330+
if (len > olen) {
1331+
if (len >= ARY_CAPA(ary)) {
1332+
ary_double_capa(ary, len);
1333+
}
1334+
rb_mem_clear(RARRAY_PTR(ary) + olen, len - olen);
1335+
ARY_SET_HEAP_LEN(ary, len);
1336+
}
1337+
else if (ARY_EMBED_P(ary)) {
1338+
ARY_SET_EMBED_LEN(ary, len);
1339+
}
1340+
else if (len <= RARRAY_EMBED_LEN_MAX) {
1341+
VALUE tmp[RARRAY_EMBED_LEN_MAX];
1342+
MEMCPY(tmp, ARY_HEAP_PTR(ary), VALUE, len);
1343+
ary_discard(ary);
1344+
MEMCPY(ARY_EMBED_PTR(ary), tmp, VALUE, len);
1345+
ARY_SET_EMBED_LEN(ary, len);
1346+
}
1347+
else {
1348+
if (olen > len + ARY_DEFAULT_SIZE) {
1349+
REALLOC_N(RARRAY(ary)->as.heap.ptr, VALUE, len);
1350+
}
1351+
ARY_SET_HEAP_LEN(ary, len);
1352+
}
1353+
return ary;
1354+
}
1355+
13111356
/*
13121357
* call-seq:
13131358
* ary[index] = obj -> obj

ext/-test-/array/resize/extconf.rb

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
create_makefile("-test-/array/resize")

ext/-test-/array/resize/resize.c

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
#include "ruby/ruby.h"
2+
3+
static VALUE
4+
ary_resize(VALUE ary, VALUE len)
5+
{
6+
rb_ary_resize(ary, NUM2LONG(len));
7+
return ary;
8+
}
9+
10+
void
11+
Init_resize(void)
12+
{
13+
rb_define_method(rb_cArray, "resize", ary_resize, 1);
14+
}

include/ruby/intern.h

+1
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,7 @@ VALUE rb_ary_includes(VALUE, VALUE);
8383
VALUE rb_ary_cmp(VALUE, VALUE);
8484
VALUE rb_ary_replace(VALUE copy, VALUE orig);
8585
VALUE rb_get_values_at(VALUE, long, int, VALUE*, VALUE(*)(VALUE,long));
86+
VALUE rb_ary_resize(VALUE ary, long len);
8687
/* bignum.c */
8788
VALUE rb_big_new(long, int);
8889
int rb_bigzero_p(VALUE x);

test/-ext-/array/test_resize.rb

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
require 'test/unit'
2+
require '-test-/array/resize'
3+
4+
class TestArray < Test::Unit::TestCase
5+
class TestResize < Test::Unit::TestCase
6+
def test_expand
7+
feature = '[ruby-dev:42912]'
8+
ary = [*1..10]
9+
ary.resize(10)
10+
assert_equal(10, ary.size, feature)
11+
assert_equal([*1..10], ary, feature)
12+
ary.resize(100)
13+
assert_equal(100, ary.size, feature)
14+
assert_equal([*1..10]+[nil]*90, ary, feature)
15+
ary.resize(20)
16+
assert_equal(20, ary.size, feature)
17+
assert_equal([*1..10]+[nil]*10, ary, feature)
18+
ary.resize(2)
19+
assert_equal(2, ary.size, feature)
20+
assert_equal([1,2], ary, feature)
21+
ary.resize(3)
22+
assert_equal(3, ary.size, feature)
23+
assert_equal([1,2,nil], ary, feature)
24+
ary.resize(10)
25+
assert_equal(10, ary.size, feature)
26+
assert_equal([1,2]+[nil]*8, ary, feature)
27+
end
28+
end
29+
end

0 commit comments

Comments
 (0)