* array.c (sort_1): remove ary_sort_check(). in-place sort keep
original elements even when it's modified. * array.c (sort_2): ditto. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@15206 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
1ca6577e0c
commit
ffe425ecaa
@ -1,4 +1,4 @@
|
|||||||
Thu Jan 24 15:15:30 2008 Yukihiro Matsumoto <matz@ruby-lang.org>
|
Thu Jan 24 15:44:42 2008 Yukihiro Matsumoto <matz@ruby-lang.org>
|
||||||
|
|
||||||
* array.c (collect_bang_i): use rb_ary_store() to avoid potential
|
* array.c (collect_bang_i): use rb_ary_store() to avoid potential
|
||||||
memory corruption. a patch from Yusuke Endoh <mame@tsg.ne.jp>
|
memory corruption. a patch from Yusuke Endoh <mame@tsg.ne.jp>
|
||||||
@ -6,6 +6,11 @@ Thu Jan 24 15:15:30 2008 Yukihiro Matsumoto <matz@ruby-lang.org>
|
|||||||
|
|
||||||
* array.c (ITERATE): remove unnecessary macro.
|
* array.c (ITERATE): remove unnecessary macro.
|
||||||
|
|
||||||
|
* array.c (sort_1): remove ary_sort_check(). in-place sort keep
|
||||||
|
original elements even when it's modified.
|
||||||
|
|
||||||
|
* array.c (sort_2): ditto.
|
||||||
|
|
||||||
Thu Jan 24 15:09:40 2008 Tanaka Akira <akr@fsij.org>
|
Thu Jan 24 15:09:40 2008 Tanaka Akira <akr@fsij.org>
|
||||||
|
|
||||||
* time.c (make_time_t): revert round trip test. [ruby-dev:33058]
|
* time.c (make_time_t): revert round trip test. [ruby-dev:33058]
|
||||||
|
55
array.c
55
array.c
@ -37,7 +37,6 @@ memfill(register VALUE *mem, register long size, register VALUE val)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#define ARY_SORTLOCK FL_USER3
|
|
||||||
#define ARY_SHARED_P(a) FL_TEST(a, ELTS_SHARED)
|
#define ARY_SHARED_P(a) FL_TEST(a, ELTS_SHARED)
|
||||||
|
|
||||||
#define ARY_SET_LEN(ary, n) do { \
|
#define ARY_SET_LEN(ary, n) do { \
|
||||||
@ -54,8 +53,6 @@ static inline void
|
|||||||
rb_ary_modify_check(VALUE ary)
|
rb_ary_modify_check(VALUE ary)
|
||||||
{
|
{
|
||||||
if (OBJ_FROZEN(ary)) rb_error_frozen("array");
|
if (OBJ_FROZEN(ary)) rb_error_frozen("array");
|
||||||
if (FL_TEST(ary, ARY_SORTLOCK))
|
|
||||||
rb_raise(rb_eRuntimeError, "can't modify array during sort");
|
|
||||||
if (!OBJ_TAINTED(ary) && rb_safe_level() >= 4)
|
if (!OBJ_TAINTED(ary) && rb_safe_level() >= 4)
|
||||||
rb_raise(rb_eSecurityError, "Insecure: can't modify array");
|
rb_raise(rb_eSecurityError, "Insecure: can't modify array");
|
||||||
}
|
}
|
||||||
@ -93,7 +90,6 @@ static VALUE
|
|||||||
rb_ary_frozen_p(VALUE ary)
|
rb_ary_frozen_p(VALUE ary)
|
||||||
{
|
{
|
||||||
if (OBJ_FROZEN(ary)) return Qtrue;
|
if (OBJ_FROZEN(ary)) return Qtrue;
|
||||||
if (FL_TEST(ary, ARY_SORTLOCK)) return Qtrue;
|
|
||||||
return Qfalse;
|
return Qfalse;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1436,34 +1432,19 @@ rb_ary_reverse_m(VALUE ary)
|
|||||||
return rb_ary_reverse(rb_ary_dup(ary));
|
return rb_ary_reverse(rb_ary_dup(ary));
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ary_sort_data {
|
|
||||||
VALUE ary;
|
|
||||||
VALUE *ptr;
|
|
||||||
long len;
|
|
||||||
};
|
|
||||||
|
|
||||||
static void
|
|
||||||
ary_sort_check(struct ary_sort_data *data)
|
|
||||||
{
|
|
||||||
if (RARRAY_PTR(data->ary) != data->ptr || RARRAY_LEN(data->ary) != data->len) {
|
|
||||||
rb_raise(rb_eRuntimeError, "array modified during sort");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
sort_1(const void *ap, const void *bp, void *data)
|
sort_1(const void *ap, const void *bp, void *dummy)
|
||||||
{
|
{
|
||||||
VALUE a = *(const VALUE *)ap, b = *(const VALUE *)bp;
|
VALUE a = *(const VALUE *)ap, b = *(const VALUE *)bp;
|
||||||
VALUE retval = rb_yield_values(2, a, b);
|
VALUE retval = rb_yield_values(2, a, b);
|
||||||
int n;
|
int n;
|
||||||
|
|
||||||
n = rb_cmpint(retval, a, b);
|
n = rb_cmpint(retval, a, b);
|
||||||
ary_sort_check((struct ary_sort_data *)data);
|
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
sort_2(const void *ap, const void *bp, void *data)
|
sort_2(const void *ap, const void *bp, void *dummy)
|
||||||
{
|
{
|
||||||
VALUE retval;
|
VALUE retval;
|
||||||
VALUE a = *(const VALUE *)ap, b = *(const VALUE *)bp;
|
VALUE a = *(const VALUE *)ap, b = *(const VALUE *)bp;
|
||||||
@ -1480,30 +1461,10 @@ sort_2(const void *ap, const void *bp, void *data)
|
|||||||
|
|
||||||
retval = rb_funcall(a, id_cmp, 1, b);
|
retval = rb_funcall(a, id_cmp, 1, b);
|
||||||
n = rb_cmpint(retval, a, b);
|
n = rb_cmpint(retval, a, b);
|
||||||
ary_sort_check((struct ary_sort_data *)data);
|
|
||||||
|
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
static VALUE
|
|
||||||
sort_i(VALUE ary)
|
|
||||||
{
|
|
||||||
struct ary_sort_data data;
|
|
||||||
|
|
||||||
data.ary = ary;
|
|
||||||
data.ptr = RARRAY_PTR(ary); data.len = RARRAY_LEN(ary);
|
|
||||||
ruby_qsort(RARRAY_PTR(ary), RARRAY_LEN(ary), sizeof(VALUE),
|
|
||||||
rb_block_given_p()?sort_1:sort_2, &data);
|
|
||||||
return ary;
|
|
||||||
}
|
|
||||||
|
|
||||||
static VALUE
|
|
||||||
sort_unlock(VALUE ary)
|
|
||||||
{
|
|
||||||
FL_UNSET(ary, ARY_SORTLOCK);
|
|
||||||
return ary;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* call-seq:
|
* call-seq:
|
||||||
* array.sort! -> array
|
* array.sort! -> array
|
||||||
@ -1525,8 +1486,16 @@ rb_ary_sort_bang(VALUE ary)
|
|||||||
{
|
{
|
||||||
rb_ary_modify(ary);
|
rb_ary_modify(ary);
|
||||||
if (RARRAY_LEN(ary) > 1) {
|
if (RARRAY_LEN(ary) > 1) {
|
||||||
FL_SET(ary, ARY_SORTLOCK); /* prohibit modification during sort */
|
VALUE tmp = ary_make_shared(ary);
|
||||||
rb_ensure(sort_i, ary, sort_unlock, ary);
|
|
||||||
|
RBASIC(tmp)->klass = 0;
|
||||||
|
ruby_qsort(RARRAY_PTR(tmp), RARRAY_LEN(tmp), sizeof(VALUE),
|
||||||
|
rb_block_given_p()?sort_1:sort_2, 0);
|
||||||
|
RARRAY(ary)->ptr = RARRAY(tmp)->ptr;
|
||||||
|
RARRAY(ary)->len = RARRAY(tmp)->len;
|
||||||
|
RARRAY(ary)->aux.capa = RARRAY(tmp)->aux.capa;
|
||||||
|
FL_UNSET(ary, ELTS_SHARED);
|
||||||
|
rb_gc_force_recycle(tmp);
|
||||||
}
|
}
|
||||||
return ary;
|
return ary;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user