diff options
author | Peter Zhu <peter@peterzhu.ca> | 2022-11-14 15:40:47 -0500 |
---|---|---|
committer | Peter Zhu <peter@peterzhu.ca> | 2022-11-14 15:40:47 -0500 |
commit | 9a6c3355c5c9541b9afb94d4ee770a7584c7ac15 (patch) | |
tree | b4b6895d708ff9e509ed3bf4edfed3d4f9c6fc65 /array.c | |
parent | 6dd1a5f532711a28ffe815455503f5a2b1f70d23 (diff) | |
download | ruby-9a6c3355c5c9541b9afb94d4ee770a7584c7ac15.tar.gz |
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.
Diffstat (limited to 'array.c')
-rw-r--r-- | array.c | 10 |
1 files changed, 5 insertions, 5 deletions
@@ -837,11 +837,11 @@ ary_new(VALUE klass, long capa) } else { ary = ary_alloc_heap(klass); + ARY_SET_CAPA(ary, capa); assert(!ARY_EMBED_P(ary)); ptr = ary_heap_alloc(ary, capa); ARY_SET_PTR(ary, ptr); - ARY_SET_CAPA(ary, capa); ARY_SET_HEAP_LEN(ary, 0); } @@ -945,11 +945,11 @@ ec_ary_new(rb_execution_context_t *ec, VALUE klass, long capa) } else { ary = ec_ary_alloc_heap(ec, klass); + ARY_SET_CAPA(ary, capa); assert(!ARY_EMBED_P(ary)); ptr = ary_heap_alloc(ary, capa); ARY_SET_PTR(ary, ptr); - ARY_SET_CAPA(ary, capa); ARY_SET_HEAP_LEN(ary, 0); } @@ -1056,6 +1056,7 @@ ary_make_shared(VALUE ary) /* Shared roots cannot be embedded because the reference count * (refcnt) is stored in as.heap.aux.capa. */ VALUE shared = ary_alloc_heap(0); + FL_SET_SHARED_ROOT(shared); if (ARY_EMBED_P(ary)) { /* 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_mem_clear(shared, len, capa - len); - FL_SET_SHARED_ROOT(shared); ARY_SET_SHARED_ROOT_REFCNT(shared, 1); FL_SET_SHARED(ary); 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; } else { + VALUE shared = ary_make_shared(ary); + VALUE result = ary_alloc_heap(klass); assert(!ARY_EMBED_P(result)); - VALUE shared = ary_make_shared(ary); - ARY_SET_PTR(result, RARRAY_CONST_PTR_TRANSIENT(ary)); ARY_SET_LEN(result, RARRAY_LEN(ary)); rb_ary_set_shared(result, shared); |