Skip to content

Commit ec475ab

Browse files
author
normal
committed
proc.c (rb_proc_alloc): inline and move to vm.c
* proc.c (rb_proc_alloc): inline and move to vm.c (rb_proc_wrap): new wrapper function used by rb_proc_alloc (proc_dup): simplify alloc + copy + wrap operation [ruby-core:64994] * vm.c (rb_proc_alloc): new inline function (rb_vm_make_proc): call rb_proc_alloc * vm_core.h: remove rb_proc_alloc, add rb_proc_wrap * benchmark/bm_vm2_newlambda.rb: short test to show difference First we allocate and populate an rb_proc_t struct inline to avoid unnecessary zeroing of the large struct. Inlining speeds up callers as this takes many parameters to ensure correctness. We then call the new rb_proc_wrap function to create the object. rb_proc_wrap - wraps a rb_proc_t pointer as a Ruby object, but we only use it inside rb_proc_alloc. We must call this before the compiler may clobber VALUE parameters passed to rb_proc_alloc. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@47562 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
1 parent afa512d commit ec475ab

File tree

5 files changed

+59
-25
lines changed

5 files changed

+59
-25
lines changed

ChangeLog

+14
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,17 @@
1+
Sat Sep 13 05:52:15 2014 Eric Wong <[email protected]>
2+
3+
* proc.c (rb_proc_alloc): inline and move to vm.c
4+
(rb_proc_wrap): new wrapper function used by rb_proc_alloc
5+
(proc_dup): simplify alloc + copy + wrap operation
6+
[ruby-core:64994]
7+
8+
* vm.c (rb_proc_alloc): new inline function
9+
(rb_vm_make_proc): call rb_proc_alloc
10+
11+
* vm_core.h: remove rb_proc_alloc, add rb_proc_wrap
12+
13+
* benchmark/bm_vm2_newlambda.rb: short test to show difference
14+
115
Sat Sep 13 04:40:04 2014 Eric Wong <[email protected]>
216

317
* process.c (Init_process): subclass Thread as Process::Waiter

benchmark/bm_vm2_newlambda.rb

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
i = 0
2+
while i<6_000_000 # benchmark loop 2
3+
i += 1
4+
lambda {}
5+
end

proc.c

+11-13
Original file line numberDiff line numberDiff line change
@@ -84,10 +84,11 @@ static const rb_data_type_t proc_data_type = {
8484
};
8585

8686
VALUE
87-
rb_proc_alloc(VALUE klass)
87+
rb_proc_wrap(VALUE klass, rb_proc_t *proc)
8888
{
89-
rb_proc_t *proc;
90-
return TypedData_Make_Struct(klass, rb_proc_t, &proc_data_type, proc);
89+
proc->block.proc = TypedData_Wrap_Struct(klass, &proc_data_type, proc);
90+
91+
return proc->block.proc;
9192
}
9293

9394
VALUE
@@ -105,17 +106,14 @@ rb_obj_is_proc(VALUE proc)
105106
static VALUE
106107
proc_dup(VALUE self)
107108
{
108-
VALUE procval = rb_proc_alloc(rb_cProc);
109-
rb_proc_t *src, *dst;
109+
VALUE procval;
110+
rb_proc_t *src;
111+
rb_proc_t *dst = ALLOC(rb_proc_t);
112+
110113
GetProcPtr(self, src);
111-
GetProcPtr(procval, dst);
112-
113-
dst->block = src->block;
114-
dst->block.proc = procval;
115-
dst->blockprocval = src->blockprocval;
116-
dst->envval = src->envval;
117-
dst->safe_level = src->safe_level;
118-
dst->is_lambda = src->is_lambda;
114+
*dst = *src;
115+
procval = rb_proc_wrap(rb_cProc, dst);
116+
RB_GC_GUARD(self); /* for: body = proc_dup(body) */
119117

120118
return procval;
121119
}

vm.c

+28-11
Original file line numberDiff line numberDiff line change
@@ -651,11 +651,35 @@ vm_make_proc_from_block(rb_thread_t *th, rb_block_t *block)
651651
return block->proc;
652652
}
653653

654+
static inline VALUE
655+
rb_proc_alloc(VALUE klass, const rb_block_t *block,
656+
VALUE envval, VALUE blockprocval,
657+
int8_t safe_level, int8_t is_from_method, int8_t is_lambda)
658+
{
659+
VALUE procval;
660+
rb_proc_t *proc = ALLOC(rb_proc_t);
661+
662+
proc->block = *block;
663+
proc->safe_level = safe_level;
664+
proc->is_from_method = is_from_method;
665+
proc->is_lambda = is_lambda;
666+
667+
procval = rb_proc_wrap(klass, proc);
668+
669+
/*
670+
* ensure VALUEs are markable here as rb_proc_wrap may trigger allocation
671+
* and clobber envval + blockprocval
672+
*/
673+
proc->envval = envval;
674+
proc->blockprocval = blockprocval;
675+
676+
return procval;
677+
}
678+
654679
VALUE
655680
rb_vm_make_proc(rb_thread_t *th, const rb_block_t *block, VALUE klass)
656681
{
657682
VALUE procval, envval, blockprocval = 0;
658-
rb_proc_t *proc;
659683
rb_control_frame_t *cfp = RUBY_VM_GET_CFP_FROM_BLOCK_PTR(block);
660684

661685
if (block->proc) {
@@ -667,16 +691,9 @@ rb_vm_make_proc(rb_thread_t *th, const rb_block_t *block, VALUE klass)
667691
if (PROCDEBUG) {
668692
check_env_value(envval);
669693
}
670-
procval = rb_proc_alloc(klass);
671-
GetProcPtr(procval, proc);
672-
proc->blockprocval = blockprocval;
673-
proc->block.self = block->self;
674-
proc->block.klass = block->klass;
675-
proc->block.ep = block->ep;
676-
proc->block.iseq = block->iseq;
677-
proc->block.proc = procval;
678-
proc->envval = envval;
679-
proc->safe_level = th->safe_level;
694+
695+
procval = rb_proc_alloc(klass, block, envval, blockprocval,
696+
th->safe_level, 0, 0);
680697

681698
if (VMDEBUG) {
682699
if (th->stack < block->ep && block->ep < th->stack + th->stack_size) {

vm_core.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -884,7 +884,7 @@ rb_block_t *rb_vm_control_frame_block_ptr(rb_control_frame_t *cfp);
884884

885885
/* VM related object allocate functions */
886886
VALUE rb_thread_alloc(VALUE klass);
887-
VALUE rb_proc_alloc(VALUE klass);
887+
VALUE rb_proc_wrap(VALUE klass, rb_proc_t *); /* may use with rb_proc_alloc */
888888

889889
/* for debug */
890890
extern void rb_vmdebug_stack_dump_raw(rb_thread_t *, rb_control_frame_t *);

0 commit comments

Comments
 (0)