Set array capacity/shared immediately after alloc
If auto-compaction is enabled, then we have to set the capacity/shared immediately after allocating a heap array. If compaction runs before capacity/shared is set then it could cause the array to be re-embedded, which can cause crashes.
This commit is contained in:
parent
6dd1a5f532
commit
9a6c3355c5
10
array.c
10
array.c
@ -837,11 +837,11 @@ ary_new(VALUE klass, long capa)
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
ary = ary_alloc_heap(klass);
|
ary = ary_alloc_heap(klass);
|
||||||
|
ARY_SET_CAPA(ary, capa);
|
||||||
assert(!ARY_EMBED_P(ary));
|
assert(!ARY_EMBED_P(ary));
|
||||||
|
|
||||||
ptr = ary_heap_alloc(ary, capa);
|
ptr = ary_heap_alloc(ary, capa);
|
||||||
ARY_SET_PTR(ary, ptr);
|
ARY_SET_PTR(ary, ptr);
|
||||||
ARY_SET_CAPA(ary, capa);
|
|
||||||
ARY_SET_HEAP_LEN(ary, 0);
|
ARY_SET_HEAP_LEN(ary, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -945,11 +945,11 @@ ec_ary_new(rb_execution_context_t *ec, VALUE klass, long capa)
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
ary = ec_ary_alloc_heap(ec, klass);
|
ary = ec_ary_alloc_heap(ec, klass);
|
||||||
|
ARY_SET_CAPA(ary, capa);
|
||||||
assert(!ARY_EMBED_P(ary));
|
assert(!ARY_EMBED_P(ary));
|
||||||
|
|
||||||
ptr = ary_heap_alloc(ary, capa);
|
ptr = ary_heap_alloc(ary, capa);
|
||||||
ARY_SET_PTR(ary, ptr);
|
ARY_SET_PTR(ary, ptr);
|
||||||
ARY_SET_CAPA(ary, capa);
|
|
||||||
ARY_SET_HEAP_LEN(ary, 0);
|
ARY_SET_HEAP_LEN(ary, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1056,6 +1056,7 @@ ary_make_shared(VALUE ary)
|
|||||||
/* Shared roots cannot be embedded because the reference count
|
/* Shared roots cannot be embedded because the reference count
|
||||||
* (refcnt) is stored in as.heap.aux.capa. */
|
* (refcnt) is stored in as.heap.aux.capa. */
|
||||||
VALUE shared = ary_alloc_heap(0);
|
VALUE shared = ary_alloc_heap(0);
|
||||||
|
FL_SET_SHARED_ROOT(shared);
|
||||||
|
|
||||||
if (ARY_EMBED_P(ary)) {
|
if (ARY_EMBED_P(ary)) {
|
||||||
/* Cannot use ary_heap_alloc because we don't want to allocate
|
/* Cannot use ary_heap_alloc because we don't want to allocate
|
||||||
@ -1074,7 +1075,6 @@ ary_make_shared(VALUE ary)
|
|||||||
|
|
||||||
ARY_SET_LEN(shared, capa);
|
ARY_SET_LEN(shared, capa);
|
||||||
ary_mem_clear(shared, len, capa - len);
|
ary_mem_clear(shared, len, capa - len);
|
||||||
FL_SET_SHARED_ROOT(shared);
|
|
||||||
ARY_SET_SHARED_ROOT_REFCNT(shared, 1);
|
ARY_SET_SHARED_ROOT_REFCNT(shared, 1);
|
||||||
FL_SET_SHARED(ary);
|
FL_SET_SHARED(ary);
|
||||||
RB_DEBUG_COUNTER_INC(obj_ary_shared_create);
|
RB_DEBUG_COUNTER_INC(obj_ary_shared_create);
|
||||||
@ -1348,11 +1348,11 @@ ary_make_partial(VALUE ary, VALUE klass, long offset, long len)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
VALUE shared = ary_make_shared(ary);
|
||||||
|
|
||||||
VALUE result = ary_alloc_heap(klass);
|
VALUE result = ary_alloc_heap(klass);
|
||||||
assert(!ARY_EMBED_P(result));
|
assert(!ARY_EMBED_P(result));
|
||||||
|
|
||||||
VALUE shared = ary_make_shared(ary);
|
|
||||||
|
|
||||||
ARY_SET_PTR(result, RARRAY_CONST_PTR_TRANSIENT(ary));
|
ARY_SET_PTR(result, RARRAY_CONST_PTR_TRANSIENT(ary));
|
||||||
ARY_SET_LEN(result, RARRAY_LEN(ary));
|
ARY_SET_LEN(result, RARRAY_LEN(ary));
|
||||||
rb_ary_set_shared(result, shared);
|
rb_ary_set_shared(result, shared);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user