4 ** See Copyright Notice in mruby.h
8 #include <mruby/class.h>
9 #include <mruby/proc.h>
10 #include <mruby/opcode.h>
11 #include <mruby/data.h>
12 #include <mruby/array.h>
13 #include <mruby/hash.h>
14 #include <mruby/internal.h>
15 #include <mruby/presym.h>
17 static const mrb_code call_iseq
[] = {
21 static const mrb_irep call_irep
= {
25 MRB_ISEQ_NO_FREE
| MRB_IREP_NO_FREE
, /* flags */
31 NULL
, /* debug_info */
39 static const struct RProc call_proc
= {
40 NULL
, NULL
, MRB_TT_PROC
, MRB_GC_RED
, MRB_FL_OBJ_IS_FROZEN
| MRB_PROC_SCOPE
| MRB_PROC_STRICT
,
41 { &call_irep
}, NULL
, { NULL
}
45 mrb_proc_new(mrb_state
*mrb
, const mrb_irep
*irep
)
48 mrb_callinfo
*ci
= mrb
->c
->ci
;
50 p
= MRB_OBJ_ALLOC(mrb
, MRB_TT_PROC
, mrb
->proc_class
);
52 struct RClass
*tc
= NULL
;
55 tc
= MRB_PROC_TARGET_CLASS(ci
->proc
);
58 tc
= mrb_vm_ci_target_class(ci
);
61 p
->e
.target_class
= tc
;
64 mrb_irep_incref(mrb
, (mrb_irep
*)irep
);
72 mrb_env_new(mrb_state
*mrb
, struct mrb_context
*c
, mrb_callinfo
*ci
, int nstacks
, mrb_value
*stack
, struct RClass
*tc
)
79 e
= MRB_OBJ_ALLOC(mrb
, MRB_TT_ENV
, NULL
);
81 MRB_ENV_SET_LEN(e
, nstacks
);
82 bidx
+= (n
== 15) ? 1 : n
;
83 bidx
+= (nk
== 15) ? 1 : (2*nk
);
84 MRB_ENV_SET_BIDX(e
, bidx
);
93 closure_setup(mrb_state
*mrb
, struct RProc
*p
)
95 mrb_callinfo
*ci
= mrb
->c
->ci
;
96 const struct RProc
*up
= p
->upper
;
97 struct REnv
*e
= NULL
;
99 mrb_assert(ci
!= NULL
);
100 if ((e
= mrb_vm_ci_env(ci
)) != NULL
) {
101 /* do nothing, because e is assigned already */
104 struct RClass
*tc
= ci
->u
.target_class
;
106 if (MRB_PROC_ALIAS_P(up
)) { /* alias */
109 e
= mrb_env_new(mrb
, mrb
->c
, ci
, up
->body
.irep
->nlocals
, ci
->stack
, tc
);
111 if (MRB_PROC_ENV_P(up
) && MRB_PROC_ENV(up
)->cxt
== NULL
) {
112 e
->mid
= MRB_PROC_ENV(up
)->mid
;
117 p
->flags
|= MRB_PROC_ENVSET
;
118 mrb_field_write_barrier(mrb
, (struct RBasic
*)p
, (struct RBasic
*)e
);
123 mrb_closure_new(mrb_state
*mrb
, const mrb_irep
*irep
)
125 struct RProc
*p
= mrb_proc_new(mrb
, irep
);
127 closure_setup(mrb
, p
);
131 MRB_API
struct RProc
*
132 mrb_proc_new_cfunc(mrb_state
*mrb
, mrb_func_t func
)
136 p
= MRB_OBJ_ALLOC(mrb
, MRB_TT_PROC
, mrb
->proc_class
);
138 p
->flags
|= MRB_PROC_CFUNC_FL
;
140 p
->e
.target_class
= 0;
145 MRB_API
struct RProc
*
146 mrb_proc_new_cfunc_with_env(mrb_state
*mrb
, mrb_func_t func
, mrb_int argc
, const mrb_value
*argv
)
148 struct RProc
*p
= mrb_proc_new_cfunc(mrb
, func
);
152 p
->e
.env
= e
= mrb_env_new(mrb
, mrb
->c
, mrb
->c
->ci
, 0, NULL
, NULL
);
153 p
->flags
|= MRB_PROC_ENVSET
;
154 mrb_field_write_barrier(mrb
, (struct RBasic
*)p
, (struct RBasic
*)e
);
157 e
->stack
= (mrb_value
*)mrb_malloc(mrb
, sizeof(mrb_value
) * argc
);
158 MRB_ENV_SET_LEN(e
, argc
);
161 for (i
= 0; i
< argc
; i
++) {
162 e
->stack
[i
] = argv
[i
];
166 for (i
= 0; i
< argc
; i
++) {
167 SET_NIL_VALUE(e
->stack
[i
]);
173 MRB_API
struct RProc
*
174 mrb_closure_new_cfunc(mrb_state
*mrb
, mrb_func_t func
, int nlocals
)
176 return mrb_proc_new_cfunc_with_env(mrb
, func
, nlocals
, NULL
);
180 mrb_proc_cfunc_env_get(mrb_state
*mrb
, mrb_int idx
)
182 const struct RProc
*p
= mrb
->c
->ci
->proc
;
185 if (!p
|| !MRB_PROC_CFUNC_P(p
)) {
186 mrb_raise(mrb
, E_TYPE_ERROR
, "Can't get cfunc env from non-cfunc proc");
190 mrb_raise(mrb
, E_TYPE_ERROR
, "Can't get cfunc env from cfunc Proc without REnv");
192 if (idx
< 0 || MRB_ENV_LEN(e
) <= idx
) {
193 mrb_raisef(mrb
, E_INDEX_ERROR
, "Env index out of range: %i (expected: 0 <= index < %i)",
194 idx
, MRB_ENV_LEN(e
));
197 return e
->stack
[idx
];
201 mrb_proc_get_self(mrb_state
*mrb
, struct RProc
*p
, struct RClass
**target_class_p
)
203 if (MRB_PROC_CFUNC_P(p
)) {
204 *target_class_p
= mrb
->object_class
;
205 return mrb_nil_value();
208 struct REnv
*e
= p
->e
.env
;
210 if (!e
|| e
->tt
!= MRB_TT_ENV
) {
211 *target_class_p
= mrb
->object_class
;
212 return mrb_top_self(mrb
);
214 else if (MRB_ENV_LEN(e
) < 1) {
215 mrb_raise(mrb
, E_ARGUMENT_ERROR
, "self is lost (probably ran out of memory when the block became independent)");
218 *target_class_p
= e
->c
;
224 mrb_proc_copy(mrb_state
*mrb
, struct RProc
*a
, struct RProc
*b
)
227 /* already initialized proc */
230 if (!MRB_PROC_CFUNC_P(b
) && b
->body
.irep
) {
231 mrb_irep_incref(mrb
, (mrb_irep
*)b
->body
.irep
);
237 /* a->e.target_class = a->e.target_class; */
241 mrb_proc_s_new(mrb_state
*mrb
, mrb_value proc_class
)
247 /* Calling Proc.new without a block is not implemented yet */
248 mrb_get_args(mrb
, "&!", &blk
);
249 p
= MRB_OBJ_ALLOC(mrb
, MRB_TT_PROC
, mrb_class_ptr(proc_class
));
250 mrb_proc_copy(mrb
, p
, mrb_proc_ptr(blk
));
251 proc
= mrb_obj_value(p
);
252 mrb_funcall_with_block(mrb
, proc
, MRB_SYM(initialize
), 0, NULL
, proc
);
253 if (!MRB_PROC_STRICT_P(p
) &&
254 mrb
->c
->ci
> mrb
->c
->cibase
&& MRB_PROC_ENV(p
) == mrb
->c
->ci
[-1].u
.env
) {
255 p
->flags
|= MRB_PROC_ORPHAN
;
261 check_proc(mrb_state
*mrb
, mrb_value proc
)
263 if (!mrb_proc_p(proc
)) {
264 mrb_raise(mrb
, E_ARGUMENT_ERROR
, "not a proc");
269 mrb_proc_init_copy(mrb_state
*mrb
, mrb_value self
)
271 mrb_value proc
= mrb_get_arg1(mrb
);
273 check_proc(mrb
, proc
);
274 mrb_proc_copy(mrb
, mrb_proc_ptr(self
), mrb_proc_ptr(proc
));
279 proc_arity(mrb_state
*mrb
, mrb_value self
)
281 return mrb_int_value(mrb
, mrb_proc_arity(mrb_proc_ptr(self
)));
285 mrb_proc_eql(mrb_state
*mrb
, mrb_value self
, mrb_value other
)
287 if (mrb_type(self
) != MRB_TT_PROC
) return FALSE
;
288 if (mrb_type(other
) != MRB_TT_PROC
) return FALSE
;
290 struct RProc
*p1
= mrb_proc_ptr(self
);
291 struct RProc
*p2
= mrb_proc_ptr(other
);
292 if (MRB_PROC_CFUNC_P(p1
)) {
293 if (!MRB_PROC_CFUNC_P(p1
)) return FALSE
;
294 if (p1
->body
.func
!= p2
->body
.func
) return FALSE
;
296 else if (MRB_PROC_CFUNC_P(p2
)) return FALSE
;
297 else if (p1
->body
.irep
!= p2
->body
.irep
) return FALSE
;
302 proc_eql(mrb_state
*mrb
, mrb_value self
)
304 return mrb_bool_value(mrb_proc_eql(mrb
, self
, mrb_get_arg1(mrb
)));
308 proc_hash(mrb_state
*mrb
, mrb_value self
)
310 struct RProc
*p
= mrb_proc_ptr(self
);
311 return mrb_int_value(mrb
, (mrb_int
)(((intptr_t)p
->body
.irep
)^MRB_TT_PROC
));
318 * lambda { |...| block } -> a_proc
320 * Equivalent to <code>Proc.new</code>, except the resulting Proc objects
321 * check the number of parameters passed when called.
324 proc_lambda(mrb_state
*mrb
, mrb_value self
)
329 mrb_get_args(mrb
, "&", &blk
);
330 if (mrb_nil_p(blk
)) {
331 mrb_raise(mrb
, E_ARGUMENT_ERROR
, "tried to create Proc object without a block");
333 check_proc(mrb
, blk
);
334 p
= mrb_proc_ptr(blk
);
335 if (!MRB_PROC_STRICT_P(p
)) {
336 struct RProc
*p2
= MRB_OBJ_ALLOC(mrb
, MRB_TT_PROC
, p
->c
);
337 mrb_proc_copy(mrb
, p2
, p
);
338 p2
->flags
|= MRB_PROC_STRICT
;
339 return mrb_obj_value(p2
);
345 mrb_proc_arity(const struct RProc
*p
)
347 const mrb_irep
*irep
;
350 int ma
, op
, ra
, pa
, arity
;
352 if (MRB_PROC_CFUNC_P(p
)) {
353 /* TODO cfunc aspec not implemented yet */
363 /* arity is depend on OP_ENTER */
364 if (*pc
!= OP_ENTER
) {
368 aspec
= PEEK_W(pc
+1);
369 ma
= MRB_ASPEC_REQ(aspec
);
370 op
= MRB_ASPEC_OPT(aspec
);
371 ra
= MRB_ASPEC_REST(aspec
);
372 pa
= MRB_ASPEC_POST(aspec
);
373 arity
= ra
|| (MRB_PROC_STRICT_P(p
) && op
) ? -(ma
+ pa
+ 1) : ma
+ pa
;
379 mrb_proc_local_variables(mrb_state
*mrb
, const struct RProc
*proc
)
381 const mrb_irep
*irep
;
385 if (proc
== NULL
|| MRB_PROC_CFUNC_P(proc
)) {
386 return mrb_ary_new(mrb
);
388 vars
= mrb_hash_new(mrb
);
390 if (MRB_PROC_CFUNC_P(proc
)) break;
391 irep
= proc
->body
.irep
;
393 for (i
= 0; i
+ 1 < irep
->nlocals
; i
++) {
395 mrb_sym sym
= irep
->lv
[i
];
396 const char *name
= mrb_sym_name(mrb
, sym
);
401 mrb_hash_set(mrb
, vars
, mrb_symbol_value(sym
), mrb_true_value());
407 if (MRB_PROC_SCOPE_P(proc
)) break;
411 return mrb_hash_keys(mrb
, vars
);
415 mrb_proc_get_caller(mrb_state
*mrb
, struct REnv
**envp
)
417 struct mrb_context
*c
= mrb
->c
;
418 mrb_callinfo
*ci
= (c
->ci
> c
->cibase
) ? c
->ci
- 1 : c
->cibase
;
419 const struct RProc
*proc
= ci
->proc
;
421 if (!proc
|| MRB_PROC_CFUNC_P(proc
)) {
422 if (envp
) *envp
= NULL
;
425 struct REnv
*e
= mrb_vm_ci_env(ci
);
428 int nstacks
= proc
->body
.irep
->nlocals
;
429 e
= mrb_env_new(mrb
, c
, ci
, nstacks
, ci
->stack
, mrb_vm_ci_target_class(ci
));
438 #define IREP_LVAR_MERGE_DEFAULT 50
439 #define IREP_LVAR_MERGE_MINIMUM 8
440 #define IREP_LVAR_MERGE_MAXIMUM 240
442 #ifdef MRB_IREP_LVAR_MERGE_LIMIT
443 # define IREP_LVAR_MERGE_LIMIT \
444 ((MRB_IREP_LVAR_MERGE_LIMIT) < IREP_LVAR_MERGE_MINIMUM ? IREP_LVAR_MERGE_MINIMUM : \
445 (MRB_IREP_LVAR_MERGE_LIMIT) > IREP_LVAR_MERGE_MAXIMUM ? IREP_LVAR_MERGE_MAXIMUM : \
446 (MRB_IREP_LVAR_MERGE_LIMIT))
448 # define IREP_LVAR_MERGE_LIMIT IREP_LVAR_MERGE_DEFAULT
452 mrb_proc_merge_lvar(mrb_state
*mrb
, mrb_irep
*irep
, struct REnv
*env
, int num
, const mrb_sym
*lv
, const mrb_value
*stack
)
454 mrb_assert(!(irep
->flags
& MRB_IREP_NO_FREE
));
456 if ((irep
->nlocals
+ num
) > IREP_LVAR_MERGE_LIMIT
) {
457 mrb_raise(mrb
, E_RUNTIME_ERROR
, "too many local variables for binding (mruby limitation)");
461 mrb_raise(mrb
, E_RUNTIME_ERROR
, "unavailable local variable names");
464 irep
->lv
= (mrb_sym
*)mrb_realloc(mrb
, (mrb_sym
*)irep
->lv
, sizeof(mrb_sym
) * (irep
->nlocals
- 1 /* self */ + num
));
465 env
->stack
= (mrb_value
*)mrb_realloc(mrb
, env
->stack
, sizeof(mrb_value
) * (irep
->nlocals
+ num
));
467 mrb_sym
*destlv
= (mrb_sym
*)irep
->lv
+ irep
->nlocals
- 1 /* self */;
468 mrb_value
*destst
= env
->stack
+ irep
->nlocals
;
469 memmove(destlv
, lv
, sizeof(mrb_sym
) * num
);
471 memmove(destst
, stack
, sizeof(mrb_value
) * num
);
472 for (int i
= 0; i
< num
; i
++) {
473 if (!mrb_immediate_p(stack
[i
])) {
474 mrb_field_write_barrier(mrb
, (struct RBasic
*)env
, (struct RBasic
*)mrb_obj_ptr(stack
[i
]));
479 for (int i
= num
; i
> 0; i
--, destst
++) {
480 *destst
= mrb_nil_value();
483 irep
->nlocals
+= num
;
484 irep
->nregs
= irep
->nlocals
;
485 MRB_ENV_SET_LEN(env
, irep
->nlocals
);
489 mrb_init_proc(mrb_state
*mrb
)
492 struct RClass
*pc
= mrb
->proc_class
= mrb_define_class_id(mrb
, MRB_SYM(Proc
), mrb
->object_class
); /* 15.2.17 */
494 MRB_SET_INSTANCE_TT(pc
, MRB_TT_PROC
);
495 MRB_UNDEF_ALLOCATOR(pc
);
496 mrb_define_class_method_id(mrb
, pc
, MRB_SYM(new), mrb_proc_s_new
, MRB_ARGS_NONE()|MRB_ARGS_BLOCK());
497 mrb_define_method_id(mrb
, pc
, MRB_SYM(initialize_copy
), mrb_proc_init_copy
, MRB_ARGS_REQ(1));
498 mrb_define_method_id(mrb
, pc
, MRB_SYM(arity
), proc_arity
, MRB_ARGS_NONE()); /* 15.2.17.4.2 */
499 mrb_define_method_id(mrb
, pc
, MRB_OPSYM(eq
), proc_eql
, MRB_ARGS_REQ(1));
500 mrb_define_method_id(mrb
, pc
, MRB_SYM_Q(eql
), proc_eql
, MRB_ARGS_REQ(1));
501 mrb_define_method_id(mrb
, pc
, MRB_SYM(hash
), proc_hash
, MRB_ARGS_NONE()); /* 15.2.17.4.2 */
503 MRB_METHOD_FROM_PROC(m
, &call_proc
);
504 mrb_define_method_raw(mrb
, pc
, MRB_SYM(call
), m
); /* 15.2.17.4.3 */
505 mrb_define_method_raw(mrb
, pc
, MRB_OPSYM(aref
), m
); /* 15.2.17.4.1 */
507 mrb_define_method(mrb
, mrb
->kernel_module
, "lambda", proc_lambda
, MRB_ARGS_NONE()|MRB_ARGS_BLOCK()); /* 15.3.1.3.27 */