14#include "internal/thread.h"
26static int vm_collect_local_variables_in_heap(
const VALUE *dfp,
const struct local_var_list *vars);
28static VALUE rb_eUncaughtThrow;
29static ID id_result, id_tag, id_value;
32static VALUE send_internal(
int argc,
const VALUE *argv,
VALUE recv, call_type scope);
36vm_argv_ruby_array(
VALUE *av,
const VALUE *argv,
int *flags,
int *argc,
int kw_splat)
38 *flags |= VM_CALL_ARGS_SPLAT;
49 *flags |= VM_CALL_KW_SPLAT;
50 av[1] = rb_hash_new();
61 return vm_call0_cc(ec, recv,
id, argc, argv, &cc, kw_splat);
68 rb_callable_method_entry_with_refinements(
CLASS_OF(recv),
id, NULL);
70 return rb_vm_call0(ec, recv,
id, argc, argv, me, kw_splat);
74 return rb_funcallv(recv,
id, argc, argv);
81 int flags = kw_splat ? VM_CALL_KW_SPLAT : 0;
85 if (UNLIKELY(vm_cc_cme(cc)->def->type == VM_METHOD_TYPE_ISEQ && argc > VM_ARGC_STACK_MAX)) {
86 use_argv = vm_argv_ruby_array(av, argv, &flags, &argc, kw_splat);
91 .ci = &VM_CI_ON_STACK(
id, flags, argc, NULL),
95 .block_handler = vm_passed_block_handler(ec),
101 return vm_call0_body(ec, &calling, use_argv);
107 calling->cc = &VM_CC_ON_STACK(
Qundef, vm_call_general, {{ 0 }}, cme);
108 return vm_call0_body(ec, calling, argv);
114 ID mid = vm_ci_mid(calling->cd->ci);
121 RUBY_VM_CHECK_INTS(ec);
122 return vm_call0_cme(ec, calling, argv, cme);
126 vm_passed_block_handler_set(ec, calling->block_handler);
127 return method_missing(ec, calling->recv, mid, calling->argc, argv, ex, calling->kw_splat);
137 int len = cfunc->argc;
138 VALUE recv = calling->recv;
139 int argc = calling->argc;
140 ID mid = vm_ci_mid(ci);
141 VALUE block_handler = calling->block_handler;
142 int frame_flags = VM_FRAME_MAGIC_CFUNC | VM_FRAME_FLAG_CFRAME | VM_ENV_FLAG_LOCAL;
144 if (calling->kw_splat) {
149 frame_flags |= VM_FRAME_FLAG_CFRAME_KW;
153 RUBY_DTRACE_CMETHOD_ENTRY_HOOK(ec, me->owner, me->def->original_id);
158 vm_push_frame(ec, 0, frame_flags, recv,
159 block_handler, (
VALUE)me,
160 0, reg_cfp->sp, 0, 0);
164 val = (*cfunc->invoker)(recv, argc, argv, cfunc->func);
166 CHECK_CFP_CONSISTENCY(
"vm_call0_cfunc_with_frame");
170 RUBY_DTRACE_CMETHOD_RETURN_HOOK(ec, me->owner, me->def->original_id);
178 return vm_call0_cfunc_with_frame(ec, calling, argv);
184 if (calling->kw_splat &&
204 switch (vm_cc_cme(cc)->def->type) {
205 case VM_METHOD_TYPE_ISEQ:
210 CHECK_VM_STACK_OVERFLOW(reg_cfp, calling->argc + 1);
211 vm_check_canary(ec, reg_cfp->sp);
213 *reg_cfp->sp++ = calling->recv;
214 for (i = 0; i < calling->argc; i++) {
215 *reg_cfp->sp++ = argv[i];
218 if (ISEQ_BODY(def_iseq_ptr(vm_cc_cme(cc)->def))->param.flags.forwardable) {
219 vm_call_iseq_fwd_setup(ec, reg_cfp, calling);
222 vm_call_iseq_setup(ec, reg_cfp, calling);
224 VM_ENV_FLAGS_SET(ec->cfp->ep, VM_FRAME_FLAG_FINISH);
227 case VM_METHOD_TYPE_NOTIMPLEMENTED:
228 case VM_METHOD_TYPE_CFUNC:
229 ret = vm_call0_cfunc(ec, calling, argv);
231 case VM_METHOD_TYPE_ATTRSET:
232 vm_call_check_arity(calling, 1, argv);
233 VM_CALL_METHOD_ATTR(ret,
234 rb_ivar_set(calling->recv, vm_cc_cme(cc)->def->body.attr.id, argv[0]),
237 case VM_METHOD_TYPE_IVAR:
238 vm_call_check_arity(calling, 0, argv);
239 VM_CALL_METHOD_ATTR(ret,
240 rb_attr_get(calling->recv, vm_cc_cme(cc)->def->body.attr.id),
243 case VM_METHOD_TYPE_BMETHOD:
244 ret = vm_call_bmethod_body(ec, calling, argv);
246 case VM_METHOD_TYPE_ZSUPER:
248 VALUE klass = RCLASS_ORIGIN(vm_cc_cme(cc)->defined_class);
249 return vm_call0_super(ec, calling, argv, klass, MISSING_SUPER);
251 case VM_METHOD_TYPE_REFINED:
255 if (cme->def->body.refined.orig_me) {
257 return vm_call0_cme(ec, calling, argv, orig_cme);
260 VALUE klass = cme->defined_class;
261 return vm_call0_super(ec, calling, argv, klass, 0);
263 case VM_METHOD_TYPE_ALIAS:
268 if (cme == orig_cme) rb_bug(
"same!!");
270 if (vm_cc_markable(cc)) {
271 return vm_call0_cme(ec, calling, argv, orig_cme);
278 case VM_METHOD_TYPE_MISSING:
280 vm_passed_block_handler_set(ec, calling->block_handler);
281 return method_missing(ec, calling->recv, vm_ci_mid(ci), calling->argc,
282 argv, MISSING_NOENTRY, calling->kw_splat);
284 case VM_METHOD_TYPE_OPTIMIZED:
285 switch (vm_cc_cme(cc)->def->body.optimized.type) {
286 case OPTIMIZED_METHOD_TYPE_SEND:
287 ret = send_internal(calling->argc, argv, calling->recv, calling->kw_splat ? CALL_FCALL_KW : CALL_FCALL);
289 case OPTIMIZED_METHOD_TYPE_CALL:
292 GetProcPtr(calling->recv, proc);
293 ret = rb_vm_invoke_proc(ec, proc, calling->argc, argv, calling->kw_splat, calling->block_handler);
296 case OPTIMIZED_METHOD_TYPE_STRUCT_AREF:
297 vm_call_check_arity(calling, 0, argv);
298 VM_CALL_METHOD_ATTR(ret,
299 vm_call_opt_struct_aref0(ec, calling),
302 case OPTIMIZED_METHOD_TYPE_STRUCT_ASET:
303 vm_call_check_arity(calling, 1, argv);
304 VM_CALL_METHOD_ATTR(ret,
305 vm_call_opt_struct_aset0(ec, calling, argv[0]),
309 rb_bug(
"vm_call0: unsupported optimized method type (%d)", vm_cc_cme(cc)->def->body.optimized.type);
312 case VM_METHOD_TYPE_UNDEF:
315 rb_bug(
"vm_call0: unsupported method type (%d)", vm_cc_cme(cc)->def->type);
319 RUBY_VM_CHECK_INTS(ec);
326 return rb_vm_call0(ec, recv,
id, argc, argv, me, kw_splat);
332 VALUE recv = ec->cfp->self;
338 if (VM_FRAME_RUBYFRAME_P(cfp)) {
339 rb_bug(
"vm_call_super: should not be reached");
342 klass = RCLASS_ORIGIN(me->defined_class);
344 id = me->def->original_id;
345 me = rb_callable_method_entry(klass,
id);
348 return method_missing(ec, recv,
id, argc, argv, MISSING_SUPER, kw_splat);
350 return rb_vm_call_kw(ec, recv,
id, argc, argv, me, kw_splat);
357 PASS_PASSED_BLOCK_HANDLER_EC(ec);
358 return vm_call_super(ec, argc, argv, kw_splat);
372 if (!ec || !(cfp = ec->cfp)) {
381 if (!rb_ec_raised_p(ec, RAISED_STACKOVERFLOW) &&
382 rb_ec_stack_check(ec)) {
383 rb_ec_raised_set(ec, RAISED_STACKOVERFLOW);
384 rb_ec_stack_overflow(ec, 0);
389rb_check_stack_overflow(
void)
391#ifndef RB_THREAD_LOCAL_SPECIFIER
392 if (!ruby_current_ec_key)
return;
395 if (ec) stack_check(ec);
398NORETURN(
static void uncallable_object(
VALUE recv,
ID mid));
405 return ((klass ^ box_value) >> 3) ^ (
VALUE)mid;
418 vm_search_method_slowpath0(vm->self, &cd, klass);
420 if (UNLIKELY(!vm->global_cc_cache_table_used)) {
421 vm->global_cc_cache_table_used =
true;
423 return vm->global_cc_cache_table[index] = cd.cc;
427scope_to_ci(call_type scope,
ID mid,
int argc,
struct rb_callinfo *ci)
435 flags |= VM_CALL_FCALL;
438 flags |= VM_CALL_VCALL;
441 flags |= VM_CALL_KWARG;
444 flags |= (VM_CALL_KWARG | VM_CALL_FCALL);
447 *ci = VM_CI_ON_STACK(mid, flags, argc, NULL);
453 VALUE klass, box_value;
454 const rb_box_t *box = rb_current_box();
458 if (UNLIKELY(!klass)) uncallable_object(recv, mid);
464 if (BOX_USER_P(box)) {
465 box_value = box->box_object;
471 unsigned int index = (
unsigned int)(gccct_hash(klass, box_value, mid) % VM_GLOBAL_CC_CACHE_TABLE_SIZE);
472 rb_vm_t *vm = rb_ec_vm_ptr(ec);
473 const struct rb_callcache *cc = vm->global_cc_cache_table[index];
476 if (LIKELY(vm_cc_class_check(cc, klass))) {
478 if (LIKELY(!METHOD_ENTRY_INVALIDATED(cme) &&
479 cme->called_id == mid)) {
481 VM_ASSERT(vm_cc_check_cme(cc, rb_callable_method_entry(klass, mid)));
482 RB_DEBUG_COUNTER_INC(gccct_hit);
489 RB_DEBUG_COUNTER_INC(gccct_null);
492 RB_DEBUG_COUNTER_INC(gccct_miss);
493 return gccct_method_search_slowpath(vm, klass, index, ci);
497rb_gccct_clear_table(
void)
500 if (vm->global_cc_cache_table_used) {
502 vm->global_cc_cache_table_used =
false;
526 call_type call_scope,
VALUE self)
528 enum method_missing_reason call_status;
529 call_type scope = call_scope;
546 scope_to_ci(scope, mid, argc, &ci);
548 const struct rb_callcache *cc = gccct_method_search(ec, recv, mid, &ci);
550 if (scope == CALL_PUBLIC) {
551 RB_DEBUG_COUNTER_INC(call0_public);
555 call_status = rb_method_call_status(ec, cme, scope, self);
557 if (UNLIKELY(call_status != MISSING_NONE)) {
558 return method_missing(ec, recv, mid, argc, argv, call_status, kw_splat);
560 else if (UNLIKELY(cc_cme != cme)) {
562 return rb_vm_call_kw(ec, recv, mid, argc, argv, cme, kw_splat);
566 RB_DEBUG_COUNTER_INC(call0_other);
567 call_status = rb_method_call_status(ec, cc ? vm_cc_cme(cc) : NULL, scope, self);
569 if (UNLIKELY(call_status != MISSING_NONE)) {
570 return method_missing(ec, recv, mid, argc, argv, call_status, kw_splat);
575 return vm_call0_cc(ec, recv, mid, argc, argv, cc, kw_splat);
584 unsigned int respond: 1;
585 unsigned int respond_to_missing: 1;
592check_funcall_exec(
VALUE v)
595 return call_method_entry(args->ec, args->defined_class,
596 args->recv, idMethodMissing,
597 args->cme, args->argc, args->argv, args->kw_splat);
604 int ret = args->respond;
606 switch (method_boundp(args->defined_class, args->mid,
607 BOUND_PRIVATE|BOUND_RESPONDS)) {
612 ret = args->respond_to_missing;
628 return vm_respond_to(ec, klass, recv, mid, TRUE);
634 return rb_method_call_status(ec, me, CALL_FCALL, ec->cfp->self) == MISSING_NONE;
644 ret = basic_obj_respond_to_missing(ec, klass, recv,
646 if (!
RTEST(ret))
return def;
647 args.respond = respond > 0;
648 args.respond_to_missing = !UNDEF_P(ret);
650 cme = callable_method_entry(klass, idMethodMissing, &args.defined_class);
652 if (cme && !METHOD_ENTRY_BASIC(cme)) {
655 new_args[0] =
ID2SYM(mid);
659 VM_ASSERT(argc == 0);
664 ec->method_missing_reason = MISSING_NOENTRY;
669 args.argc = argc + 1;
670 args.argv = new_args;
671 args.kw_splat = kw_splat;
673 check_funcall_failed, (
VALUE)&args,
680static VALUE rb_check_funcall_default_kw(
VALUE recv,
ID mid,
int argc,
const VALUE *argv,
VALUE def,
int kw_splat);
685 return rb_check_funcall_default_kw(recv, mid, argc, argv,
Qundef, kw_splat);
695rb_check_funcall_default_kw(
VALUE recv,
ID mid,
int argc,
const VALUE *argv,
VALUE def,
int kw_splat)
697 VM_ASSERT(ruby_thread_has_gvl_p());
702 int respond = check_funcall_respond_to(ec, klass, recv, mid);
707 me = rb_search_method_entry(recv, mid);
708 if (!check_funcall_callable(ec, me)) {
709 VALUE ret = check_funcall_missing(ec, klass, recv, mid, argc, argv,
710 respond, def, kw_splat);
711 if (UNDEF_P(ret)) ret = def;
715 return rb_vm_call_kw(ec, recv, mid, argc, argv, me, kw_splat);
719rb_check_funcall_default(
VALUE recv,
ID mid,
int argc,
const VALUE *argv,
VALUE def)
721 return rb_check_funcall_default_kw(recv, mid, argc, argv, def,
RB_NO_KEYWORDS);
725rb_check_funcall_with_hook_kw(
VALUE recv,
ID mid,
int argc,
const VALUE *argv,
726 rb_check_funcall_hook *hook,
VALUE arg,
int kw_splat)
731 int respond = check_funcall_respond_to(ec, klass, recv, mid);
734 (*hook)(FALSE, recv, mid, argc, argv, arg);
738 me = rb_search_method_entry(recv, mid);
739 if (!check_funcall_callable(ec, me)) {
740 VALUE ret = check_funcall_missing(ec, klass, recv, mid, argc, argv,
741 respond,
Qundef, kw_splat);
742 (*hook)(!UNDEF_P(ret), recv, mid, argc, argv, arg);
746 (*hook)(TRUE, recv, mid, argc, argv, arg);
747 return rb_vm_call_kw(ec, recv, mid, argc, argv, me, kw_splat);
753#define type_case(t) t: return #t
771 case type_case(
T_NIL);
789uncallable_object(
VALUE recv,
ID mid)
794 VALUE mname = rb_id2str(mid);
798 "method '%"PRIsVALUE
"' called on unexpected immediate object (%p)",
799 mname, (
void *)recv);
801 else if ((flags =
RBASIC(recv)->flags) == 0) {
803 "method '%"PRIsVALUE
"' called on terminated object (%p)",
804 mname, (
void *)recv);
808 "method '%"PRIsVALUE
"' called on broken T_?""?""?(0x%02x) object"
809 " (%p flags=0x%"PRIxVALUE
")",
810 mname,
type, (
void *)recv, flags);
814 "method '%"PRIsVALUE
"' called on hidden %s object"
815 " (%p flags=0x%"PRIxVALUE
")",
816 mname, typestr, (
void *)recv, flags);
820 "method '%"PRIsVALUE
"' called on unexpected %s object"
821 " (%p flags=0x%"PRIxVALUE
")",
822 mname, typestr, (
void *)recv, flags);
827rb_search_method_entry(
VALUE recv,
ID mid)
831 if (!klass) uncallable_object(recv, mid);
832 return rb_callable_method_entry(klass, mid);
835static inline enum method_missing_reason
838 if (UNLIKELY(UNDEFINED_METHOD_ENTRY_P(me))) {
841 else if (UNLIKELY(me->def->type == VM_METHOD_TYPE_REFINED)) {
842 me = rb_resolve_refined_method_callable(
Qnil, me);
843 if (UNDEFINED_METHOD_ENTRY_P(me))
goto undefined;
846 rb_method_visibility_t visi = METHOD_ENTRY_VISI(me);
849 if (UNLIKELY(visi != METHOD_VISI_PUBLIC)) {
850 if (me->def->original_id == idMethodMissing) {
853 else if (visi == METHOD_VISI_PRIVATE &&
854 scope == CALL_PUBLIC) {
855 return MISSING_PRIVATE;
858 else if (visi == METHOD_VISI_PROTECTED &&
859 scope == CALL_PUBLIC) {
861 VALUE defined_class = me->owner;
863 defined_class =
RBASIC(defined_class)->klass;
867 return MISSING_PROTECTED;
875 return scope == CALL_VCALL ? MISSING_VCALL : MISSING_NOENTRY;
891rb_call(
VALUE recv,
ID mid,
int argc,
const VALUE *argv, call_type scope)
894 return rb_call0(ec, recv, mid, argc, argv, scope, ec->cfp->self);
898 VALUE obj,
enum method_missing_reason call_status));
940rb_method_missing(
int argc,
const VALUE *argv,
VALUE obj)
943 raise_method_missing(ec, argc, argv, obj, ec->method_missing_reason);
949 int argc,
const VALUE *argv,
int priv)
951 VALUE name = argv[0];
954 format = rb_fstring_lit(
"undefined method '%1$s' for %3$s%4$s");
958 return rb_nomethod_err_new(format, obj, name, args, priv);
961 return rb_name_err_new(format, obj, name);
967 enum method_missing_reason last_call_status)
972 if (UNLIKELY(argc == 0)) {
973 rb_raise(rb_eArgError,
"no method name given");
975 else if (UNLIKELY(!
SYMBOL_P(argv[0]))) {
976 const VALUE e = rb_eArgError;
977 rb_raise(e,
"method name must be a Symbol but %"PRIsVALUE
" is given",
983 if (last_call_status & MISSING_PRIVATE) {
984 format = rb_fstring_lit(
"private method '%1$s' called for %3$s%4$s");
986 else if (last_call_status & MISSING_PROTECTED) {
987 format = rb_fstring_lit(
"protected method '%1$s' called for %3$s%4$s");
989 else if (last_call_status & MISSING_VCALL) {
990 format = rb_fstring_lit(
"undefined local variable or method '%1$s' for %3$s%4$s");
993 else if (last_call_status & MISSING_SUPER) {
994 format = rb_fstring_lit(
"super: no superclass method '%1$s' for %3$s%4$s");
998 exc = rb_make_no_method_exception(exc, format, obj, argc, argv,
999 last_call_status & (MISSING_FCALL|MISSING_VCALL));
1000 if (!(last_call_status & MISSING_MISSING)) {
1001 rb_vm_pop_cfunc_frame();
1009 VALUE obj,
int call_status)
1011 vm_passed_block_handler_set(ec, VM_BLOCK_HANDLER_NONE);
1012 raise_method_missing(ec, argc, argv, obj, call_status | MISSING_MISSING);
1018 VALUE *nargv, result, work, klass;
1019 VALUE block_handler = vm_passed_block_handler(ec);
1022 ec->method_missing_reason = call_status;
1024 if (
id == idMethodMissing) {
1033 VM_ASSERT(argc == 0);
1042 if (!klass)
goto missing;
1043 me = rb_callable_method_entry(klass, idMethodMissing);
1044 if (!me || METHOD_ENTRY_BASIC(me))
goto missing;
1045 vm_passed_block_handler_set(ec, block_handler);
1046 result = rb_vm_call_kw(ec, obj, idMethodMissing, argc, argv, me, kw_splat);
1050 raise_method_missing(ec, argc, argv, obj, call_status | MISSING_MISSING);
1055rb_funcallv_scope(
VALUE recv,
ID mid,
int argc,
const VALUE *argv, call_type scope)
1060 scope_to_ci(scope, mid, argc, &ci);
1062 const struct rb_callcache *cc = gccct_method_search(ec, recv, mid, &ci);
1063 VALUE self = ec->cfp->self;
1066 LIKELY(rb_method_call_status(ec, vm_cc_cme(cc), scope, self) == MISSING_NONE)) {
1068 return vm_call0_cc(ec, recv, mid, argc, argv, cc,
false);
1071 return rb_call0(ec, recv, mid, argc, argv, scope, self);
1081 VM_ASSERT(ruby_thread_has_gvl_p());
1083 return rb_funcallv_scope(recv, mid, argc, argv, CALL_FCALL);
1089 VM_ASSERT(ruby_thread_has_gvl_p());
1091 return rb_call(recv, mid, argc, argv, kw_splat ? CALL_FCALL_KW : CALL_FCALL);
1101 if (argc >= 0x100) {
1103 RBASIC_CLEAR_CLASS(args);
1112 return rb_funcallv(recv, mid, argc, argv);
1132 for (i = 0; i < n; i++) {
1133 argv[i] = va_arg(ar,
VALUE);
1140 return rb_funcallv(recv, mid, n, argv);
1154rb_check_funcall_basic_kw(
VALUE recv,
ID mid,
VALUE ancestor,
int argc,
const VALUE *argv,
int kw_splat)
1159 if (!klass)
return Qundef;
1161 cme = rb_callable_method_entry(klass, mid);
1162 if (cme && METHOD_ENTRY_BASIC(cme) &&
RBASIC_CLASS(cme->defined_class) == ancestor) {
1164 return rb_vm_call0(ec, recv, mid, argc, argv, cme, kw_splat);
1173 return rb_funcallv_scope(recv, mid, argc, argv, CALL_PUBLIC);
1179 return rb_call(recv, mid, argc, argv, kw_splat ? CALL_PUBLIC_KW : CALL_PUBLIC);
1185 PASS_PASSED_BLOCK_HANDLER();
1192 PASS_PASSED_BLOCK_HANDLER();
1193 return rb_call(recv, mid, argc, argv, kw_splat ? CALL_PUBLIC_KW : CALL_PUBLIC);
1199 if (!
NIL_P(passed_procval)) {
1200 vm_passed_block_handler_set(GET_EC(), passed_procval);
1209 if (!
NIL_P(passed_procval)) {
1210 vm_passed_block_handler_set(GET_EC(), passed_procval);
1213 return rb_call(recv, mid, argc, argv, kw_splat ? CALL_PUBLIC_KW : CALL_PUBLIC);
1220 if (RUBY_VM_CONTROL_FRAME_STACK_OVERFLOW_P(ec, prev_cfp))
return NULL;
1221 if (prev_cfp->sp + 1 != argv)
return NULL;
1222 return prev_cfp->sp + 1;
1226send_internal(
int argc,
const VALUE *argv,
VALUE recv, call_type scope)
1231 VALUE ret, vargv = 0;
1233 int public = scope == CALL_PUBLIC || scope == CALL_PUBLIC_KW;
1239 self = RUBY_VM_PREVIOUS_CONTROL_FRAME(ec->cfp)->self;
1243 rb_raise(rb_eArgError,
"no method name given");
1250 if (rb_method_basic_definition_p(
CLASS_OF(recv), idMethodMissing)) {
1257 VALUE *tmp_argv = current_vm_stack_arg(ec, argv);
1262 else if (argc > 1) {
1272 id = idMethodMissing;
1273 ec->method_missing_reason = MISSING_NOENTRY;
1278 PASS_PASSED_BLOCK_HANDLER_EC(ec);
1279 ret = rb_call0(ec, recv,
id, argc, argv, scope, self);
1285send_internal_kw(
int argc,
const VALUE *argv,
VALUE recv, call_type scope)
1290 scope = CALL_PUBLIC_KW;
1293 scope = CALL_FCALL_KW;
1299 return send_internal(argc, argv, recv, scope);
1331 return send_internal_kw(argc, argv, recv, CALL_FCALL);
1349rb_f_public_send(
int argc,
VALUE *argv,
VALUE recv)
1351 return send_internal_kw(argc, argv, recv, CALL_PUBLIC);
1357rb_yield_0_kw(
int argc,
const VALUE * argv,
int kw_splat)
1359 return vm_yield(GET_EC(), argc, argv, kw_splat);
1363rb_yield_0(
int argc,
const VALUE * argv)
1369rb_yield_1(
VALUE val)
1371 return rb_yield_0(1, &val);
1378 return rb_yield_0(0, NULL);
1381 return rb_yield_0(1, &val);
1396#undef rb_yield_values
1401 return rb_yield_0(0, 0);
1410 for (i=0; i<n; i++) {
1411 argv[i] = va_arg(args,
VALUE);
1415 return rb_yield_0(n, argv);
1422 return rb_yield_0(argc, argv);
1428 return rb_yield_0_kw(argc, argv, kw_splat);
1437 rb_raise(rb_eArgError,
"not an array");
1450 rb_raise(rb_eArgError,
"not an array");
1458rb_yield_force_blockarg(
VALUE values)
1460 return vm_yield_force_blockarg(GET_EC(), values);
1466 return vm_yield_with_block(GET_EC(), argc, argv,
1467 NIL_P(blockarg) ? VM_BLOCK_HANDLER_NONE : blockarg,
1478 const struct vm_ifunc *
const ifunc,
1481 enum ruby_tag_type state;
1486 state = EC_EXEC_TAG();
1490 VALUE block_handler;
1494 captured->code.ifunc = ifunc;
1495 block_handler = VM_BH_FROM_IFUNC_BLOCK(captured);
1498 block_handler = VM_CF_BLOCK_HANDLER(cfp);
1500 vm_passed_block_handler_set(ec, block_handler);
1502 retval = (*it_proc) (data1);
1504 else if (state == TAG_BREAK || state == TAG_RETRY) {
1508 if (cfp == escape_cfp) {
1509 rb_vm_rewind_cfp(ec, cfp);
1512 ec->tag->state = TAG_NONE;
1515 if (state == TAG_RETRY)
goto iter_retry;
1516 retval = THROW_DATA_VAL(err);
1519 SDR(); fprintf(stderr,
"%p, %p\n", (
void *)cfp, (
void *)escape_cfp);
1525 EC_JUMP_TAG(ec, state);
1534 return rb_iterate0(it_proc, data1,
1535 bl_proc ? rb_vm_ifunc_proc_new(bl_proc, (
void *)data2) : 0,
1548iterate_method(
VALUE obj)
1553 return rb_call(arg->obj, arg->mid, arg->argc, arg->argv, arg->kw_splat ? CALL_FCALL_KW : CALL_FCALL);
1575 arg.kw_splat = kw_splat;
1576 return rb_iterate_internal(iterate_method, (
VALUE)&arg, bl_proc, data2);
1592rb_block_call2(
VALUE obj,
ID mid,
int argc,
const VALUE *argv,
1601 arg.kw_splat = flags & 1;
1603 struct vm_ifunc *ifunc = rb_vm_ifunc_proc_new(bl_proc, (
void *)data2);
1604 if (flags & RB_BLOCK_NO_USE_PACKED_ARGS)
1605 ifunc->flags |= IFUNC_YIELD_OPTIMIZABLE;
1607 return rb_iterate0(iterate_method, (
VALUE)&arg, ifunc, GET_EC());
1611rb_lambda_call(
VALUE obj,
ID mid,
int argc,
const VALUE *argv,
1618 if (!bl_proc) rb_raise(rb_eArgError,
"NULL lambda function");
1624 block = rb_vm_ifunc_new(bl_proc, (
void *)data2, min_argc, max_argc);
1625 return rb_iterate0(iterate_method, (
VALUE)&arg, block, GET_EC());
1629iterate_check_method(
VALUE obj)
1638rb_check_block_call(
VALUE obj,
ID mid,
int argc,
const VALUE *argv,
1648 return rb_iterate_internal(iterate_check_method, (
VALUE)&arg, bl_proc, data2);
1654 return rb_call(obj, idEach, 0, 0, CALL_FCALL);
1659#define EVAL_LOCATION_MARK "eval at "
1660#define EVAL_LOCATION_MARK_LEN (int)rb_strlen_lit(EVAL_LOCATION_MARK)
1663get_eval_default_path(
void)
1665 int location_lineno;
1666 VALUE location_path = rb_source_location(&location_lineno);
1667 if (!
NIL_P(location_path)) {
1668 return rb_fstring(rb_sprintf(
"("EVAL_LOCATION_MARK
"%"PRIsVALUE
":%d)",
1669 location_path, location_lineno));
1672 if (!eval_default_path) {
1673 eval_default_path = rb_fstring_lit(
"(eval)");
1674 rb_vm_register_global_object(eval_default_path);
1676 return eval_default_path;
1680compute_isolated_depth_from_ep(
const VALUE *ep)
1684 if (VM_ENV_FLAGS(ep, VM_ENV_FLAG_ISOLATED))
return depth;
1685 if (VM_ENV_LOCAL_P(ep))
return 0;
1686 ep = VM_ENV_PREV_EP(ep);
1692compute_isolated_depth_from_block(
const struct rb_block *blk)
1694 return compute_isolated_depth_from_ep(vm_block_ep(blk));
1698pm_eval_make_iseq(
VALUE src,
VALUE fname,
int line,
1701 const rb_iseq_t *
const parent = vm_block_iseq(base_block);
1703 VALUE name = rb_fstring_lit(
"<compiled>");
1705 int coverage_enabled = ((rb_get_coverage_mode() & COVERAGE_TARGET_EVAL) != 0) ? 1 : 0;
1706 int isolated_depth = compute_isolated_depth_from_block(base_block);
1709 fname = rb_source_location(&line);
1712 if (!UNDEF_P(fname)) {
1713 if (!
NIL_P(fname)) fname = rb_fstring(fname);
1716 fname = get_eval_default_path();
1717 coverage_enabled = 0;
1721 pm_parse_result_init(&result);
1722 pm_options_line_set(result.
options, line);
1723 result.
node.coverage_enabled = coverage_enabled;
1726 int scopes_count = 0;
1729 }
while ((iseq = ISEQ_BODY(iseq)->parent_iseq));
1730 pm_options_scopes_init(result.
options, scopes_count + 1);
1739#define FORWARDING_POSITIONALS_CHR '*'
1740#define FORWARDING_POSITIONALS_STR "*"
1741#define FORWARDING_KEYWORDS_CHR ':'
1742#define FORWARDING_KEYWORDS_STR ":"
1743#define FORWARDING_BLOCK_CHR '&'
1744#define FORWARDING_BLOCK_STR "&"
1745#define FORWARDING_ALL_CHR '.'
1746#define FORWARDING_ALL_STR "."
1748 for (
int scopes_index = 0; scopes_index < scopes_count; scopes_index++) {
1750 int locals_count = ISEQ_BODY(iseq)->local_table_size;
1753 pm_options_scope_init(options_scope, locals_count);
1757 for (
int local_index = 0; local_index < locals_count; local_index++) {
1758 pm_string_t *scope_local = pm_options_scope_local_mut(options_scope, local_index);
1759 ID local = ISEQ_BODY(iseq)->local_table[local_index];
1762 VALUE name_obj = rb_id2str(local);
1763 const char *name = RSTRING_PTR(name_obj);
1764 size_t length = RSTRING_LEN(name_obj);
1768 if (length == 2 && name[0] ==
'_' && name[1] >=
'1' && name[1] <=
'9') {
1783 char *name_dup =
xmalloc(length);
1784 MEMCPY(name_dup, name,
char, length);
1788 pm_string_owned_init(scope_local, (uint8_t *) name_dup, length);
1790 else if (local == idMULT) {
1792 pm_string_constant_init(scope_local, FORWARDING_POSITIONALS_STR, 1);
1794 else if (local == idPow) {
1796 pm_string_constant_init(scope_local, FORWARDING_KEYWORDS_STR, 1);
1798 else if (local == idAnd) {
1800 pm_string_constant_init(scope_local, FORWARDING_BLOCK_STR, 1);
1802 else if (local == idDot3) {
1804 pm_string_constant_init(scope_local, FORWARDING_ALL_STR, 1);
1808 pm_options_scope_forwarding_set(options_scope, forwarding);
1809 iseq = ISEQ_BODY(iseq)->parent_iseq;
1819 pm_options_scope_init(pm_options_scope_mut(result.
options, scopes_count), 0);
1822 VALUE error = pm_parse_string(&result, src, fname, ruby_vm_keep_script_lines ? &script_lines : NULL);
1825 if (error !=
Qnil) {
1826 pm_parse_result_free(&result);
1836 for (
int scopes_index = 0; scopes_index < scopes_count; scopes_index++) {
1841 parent_scope->coverage_enabled = coverage_enabled;
1842 parent_scope->parser = result.
parser;
1843 int locals_count = ISEQ_BODY(iseq)->local_table_size;
1845 parent_scope->local_table_for_iseq_size = locals_count;
1846 pm_constant_id_list_init(&parent_scope->locals);
1848 for (
int local_index = 0; local_index < locals_count; local_index++) {
1849 const pm_string_t *scope_local = pm_options_scope_local(options_scope, local_index);
1852 const uint8_t *source = pm_string_source(scope_local);
1853 size_t length = pm_string_length(scope_local);
1858 case FORWARDING_POSITIONALS_CHR:
1859 constant_id = PM_CONSTANT_MULT;
1861 case FORWARDING_KEYWORDS_CHR:
1862 constant_id = PM_CONSTANT_POW;
1864 case FORWARDING_BLOCK_CHR:
1865 constant_id = PM_CONSTANT_AND;
1867 case FORWARDING_ALL_CHR:
1868 constant_id = PM_CONSTANT_DOT3;
1871 constant_id = pm_parser_constant_find(result.
parser, source, length);
1876 constant_id = pm_parser_constant_find(result.
parser, source, length);
1879 pm_index_lookup_table_insert(&parent_scope->
index_lookup_table, constant_id, local_index);
1882 pm_constant_id_list_append(result.
arena, &parent_scope->locals, constant_id);
1885 node->previous = parent_scope;
1886 node = parent_scope;
1887 iseq = ISEQ_BODY(iseq)->parent_iseq;
1890#undef FORWARDING_POSITIONALS_CHR
1891#undef FORWARDING_POSITIONALS_STR
1892#undef FORWARDING_KEYWORDS_CHR
1893#undef FORWARDING_KEYWORDS_STR
1894#undef FORWARDING_BLOCK_CHR
1895#undef FORWARDING_BLOCK_STR
1896#undef FORWARDING_ALL_CHR
1897#undef FORWARDING_ALL_STR
1900 iseq = pm_iseq_new_eval(&result.
node, name, fname,
Qnil, line, parent, isolated_depth, &error_state);
1905 pm_scope_node_destroy(prev);
1910 pm_parse_result_free(&result);
1915 rb_jump_tag(error_state);
1918 rb_exec_event_hook_script_compiled(GET_EC(), iseq, src);
1924eval_make_iseq(
VALUE src,
VALUE fname,
int line,
1927 if (rb_ruby_prism_p()) {
1928 return pm_eval_make_iseq(src, fname, line, base_block);
1930 const VALUE parser = rb_parser_new();
1931 const rb_iseq_t *
const parent = vm_block_iseq(base_block);
1936 int coverage_enabled = (rb_get_coverage_mode() & COVERAGE_TARGET_EVAL) != 0;
1937 int isolated_depth = compute_isolated_depth_from_block(base_block);
1940 fname = rb_source_location(&line);
1943 if (!UNDEF_P(fname)) {
1944 if (!
NIL_P(fname)) fname = rb_fstring(fname);
1947 fname = get_eval_default_path();
1948 coverage_enabled = FALSE;
1951 rb_parser_set_context(parser, parent, FALSE);
1952 if (ruby_vm_keep_script_lines) rb_parser_set_script_lines(parser);
1953 ast_value = rb_parser_compile_string_path(parser, fname, src, line);
1955 ast = rb_ruby_ast_data_get(ast_value);
1957 if (ast->body.root) {
1958 ast->body.coverage_enabled = coverage_enabled;
1959 iseq = rb_iseq_new_eval(ast_value,
1960 ISEQ_BODY(parent)->location.label,
1962 parent, isolated_depth);
1964 rb_ast_dispose(ast);
1968 VALUE disasm = rb_iseq_disasm(iseq);
1972 rb_exec_event_hook_script_compiled(GET_EC(), iseq, src);
1989 block.as.captured = *VM_CFP_TO_CAPTURED_BLOCK(cfp);
1990 block.as.captured.self = self;
1991 block.as.captured.code.iseq = CFP_ISEQ(cfp);
1992 block.type = block_type_iseq;
1996 rb_zjit_invalidate_no_ep_escape(CFP_ISEQ(cfp));
1998 iseq = eval_make_iseq(src, file, line, &block);
2004 if (!cref && block.as.captured.code.val) {
2005 rb_cref_t *orig_cref = vm_get_cref(vm_block_ep(&block));
2006 cref = vm_cref_dup(orig_cref);
2008 vm_set_eval_stack(ec, iseq, cref, &block);
2019 const rb_iseq_t *iseq = eval_make_iseq(src, file, line, &bind->block);
2024 vm_set_eval_stack(ec, iseq, NULL, &bind->block);
2027 if (ISEQ_BODY(iseq)->local_table_size > 0) {
2028 vm_bind_update_env(scope, bind, vm_make_env_object(ec, ec->cfp));
2071rb_f_eval(
int argc,
const VALUE *argv,
VALUE self)
2073 VALUE src, scope, vfile, vline;
2077 rb_scan_args(argc, argv,
"13", &src, &scope, &vfile, &vline);
2090 return eval_string_with_cref(self, src, NULL, file, line);
2092 return eval_string_with_scope(scope, src, file, line);
2097ruby_eval_string_from_file(
const char *str,
const char *filename)
2102 VALUE self = cfp ? cfp->self : rb_vm_top_self();
2103 return eval_string_with_cref(self,
rb_str_new2(str), NULL, file, 1);
2109 return ruby_eval_string_from_file(str,
"eval");
2113eval_string_protect(
VALUE str)
2121 return rb_protect(eval_string_protect, (
VALUE)str, pstate);
2131eval_string_wrap_protect(
VALUE data)
2134 rb_cref_t *cref = rb_vm_cref_new_toplevel();
2135 cref->klass_or_self = arg->klass;
2144 VALUE self = th->top_self;
2145 VALUE wrapper = th->top_wrapper;
2153 data.top_self = th->top_self;
2154 data.klass = th->top_wrapper;
2157 val = rb_protect(eval_string_wrap_protect, (
VALUE)&data, &state);
2159 th->top_self = self;
2160 th->top_wrapper = wrapper;
2165 else if (state != TAG_NONE) {
2166 EC_JUMP_TAG(th->ec, state);
2177 VALUE val = rb_eval_cmd_call_kw(cmd, argc, argv, kw_splat);
2183rb_eval_cmd_call_kw(
VALUE cmd,
int argc,
const VALUE *argv,
int kw_splat)
2185 enum ruby_tag_type state;
2190 if ((state = EC_EXEC_TAG()) == TAG_NONE) {
2195 val = eval_string_with_cref(rb_vm_top_self(), cmd, NULL, 0, 0);
2200 if (state) EC_JUMP_TAG(ec, state);
2207yield_under(
VALUE self,
int singleton,
int argc,
const VALUE *argv,
int kw_splat)
2211 VALUE block_handler = VM_CF_BLOCK_HANDLER(cfp);
2212 VALUE new_block_handler = 0;
2215 const VALUE *ep = NULL;
2217 int is_lambda = FALSE;
2219 if (block_handler != VM_BLOCK_HANDLER_NONE) {
2221 switch (vm_block_handler_type(block_handler)) {
2222 case block_handler_type_iseq:
2223 captured = VM_BH_TO_CAPT_BLOCK(block_handler);
2224 new_captured = *captured;
2225 new_block_handler = VM_BH_FROM_ISEQ_BLOCK(&new_captured);
2227 case block_handler_type_ifunc:
2228 captured = VM_BH_TO_CAPT_BLOCK(block_handler);
2229 new_captured = *captured;
2230 new_block_handler = VM_BH_FROM_IFUNC_BLOCK(&new_captured);
2232 case block_handler_type_proc:
2234 block_handler = vm_proc_to_block_handler(VM_BH_TO_PROC(block_handler));
2236 case block_handler_type_symbol:
2237 return rb_sym_proc_call(
SYM2ID(VM_BH_TO_SYMBOL(block_handler)),
2238 argc, argv, kw_splat,
2239 VM_BLOCK_HANDLER_NONE);
2242 new_captured.self = self;
2245 VM_FORCE_WRITE_SPECIAL_CONST(&VM_CF_LEP(ec->cfp)[VM_ENV_DATA_INDEX_SPECVAL], new_block_handler);
2249 cref = vm_cref_push(ec, self, ep, TRUE, singleton);
2251 return vm_yield_with_cref(ec, argc, argv, kw_splat, cref, is_lambda);
2255rb_yield_refine_block(
VALUE refinement,
VALUE refinements)
2258 VALUE block_handler = VM_CF_BLOCK_HANDLER(ec->cfp);
2260 if (vm_block_handler_type(block_handler) != block_handler_type_iseq) {
2261 rb_bug(
"rb_yield_refine_block: an iseq block is required");
2266 const VALUE *
const argv = &new_captured.self;
2267 VALUE new_block_handler = VM_BH_FROM_ISEQ_BLOCK(&new_captured);
2268 const VALUE *ep = captured->ep;
2269 rb_cref_t *cref = vm_cref_push(ec, refinement, ep, TRUE, FALSE);
2270 CREF_REFINEMENTS_SET(cref, refinements);
2271 VM_FORCE_WRITE_SPECIAL_CONST(&VM_CF_LEP(ec->cfp)[VM_ENV_DATA_INDEX_SPECVAL], new_block_handler);
2272 new_captured.self = refinement;
2273 return vm_yield_with_cref(ec, 0, argv,
RB_NO_KEYWORDS, cref, FALSE);
2281 rb_cref_t *cref = vm_cref_push(GET_EC(), self, NULL, FALSE, singleton);
2284 return eval_string_with_cref(self, src, cref, file, line);
2288specific_eval(
int argc,
const VALUE *argv,
VALUE self,
int singleton,
int kw_splat)
2292 return yield_under(self, singleton, 1, &self, kw_splat);
2310 file = get_eval_default_path();
2313 return eval_under(self, singleton, code, file, line);
2351rb_obj_instance_eval_internal(
int argc,
const VALUE *argv,
VALUE self)
2381rb_obj_instance_exec_internal(
int argc,
const VALUE *argv,
VALUE self)
2420rb_mod_module_eval_internal(
int argc,
const VALUE *argv,
VALUE mod)
2454rb_mod_module_exec_internal(
int argc,
const VALUE *argv,
VALUE mod)
2479uncaught_throw_init(
int argc,
const VALUE *argv,
VALUE exc)
2496uncaught_throw_tag(
VALUE exc)
2509uncaught_throw_value(
VALUE exc)
2522uncaught_throw_to_s(
VALUE exc)
2524 VALUE mesg = rb_attr_get(exc, id_mesg);
2525 VALUE tag = uncaught_throw_tag(exc);
2558 if (tt->tag == tag) {
2572 ec->errinfo = (
VALUE)THROW_DATA_NEW(tag, NULL, TAG_THROW);
2573 EC_JUMP_TAG(ec, TAG_THROW);
2585 return rb_yield_0(1, &tag);
2657 enum ruby_tag_type state;
2665 if ((state = EC_EXEC_TAG()) == TAG_NONE) {
2667 val = (*func)(tag, data, 1, (
const VALUE *)&tag,
Qnil);
2669 else if (state == TAG_THROW && THROW_DATA_VAL((
struct vm_throw_data *)ec->errinfo) == tag) {
2670 rb_vm_rewind_cfp(ec, saved_cfp);
2671 val = ec->tag->retval;
2685 return vm_catch_protect(t, func, data, stateptr, GET_EC());
2691 enum ruby_tag_type state;
2694 if (state) EC_JUMP_TAG(ec, state);
2701 vars->tbl = rb_ident_hash_new();
2702 RBASIC_CLEAR_CLASS(vars->tbl);
2709 VALUE ary = rb_hash_keys(vars->tbl);
2710 rb_hash_clear(vars->tbl);
2716local_var_list_update(st_data_t *key, st_data_t *value, st_data_t arg,
int existing)
2718 if (existing)
return ST_STOP;
2719 *value = (st_data_t)
Qtrue;
2723extern int rb_numparam_id_p(
ID id);
2733 if (rb_numparam_id_p(lid))
return;
2736 rb_hash_stlike_update(vars->tbl,
ID2SYM(lid), local_var_list_update, idx);
2747 if (rb_numparam_id_p(lid)) {
2749 rb_hash_stlike_update(vars->tbl,
ID2SYM(lid), local_var_list_update, idx);
2767rb_f_local_variables(
VALUE _)
2771 rb_control_frame_t *cfp = vm_get_ruby_level_caller_cfp(ec, RUBY_VM_PREVIOUS_CONTROL_FRAME(ec->cfp));
2774 local_var_list_init(&vars);
2776 if (CFP_ISEQ(cfp)) {
2777 for (i = 0; i < ISEQ_BODY(CFP_ISEQ(cfp))->local_table_size; i++) {
2778 local_var_list_add(&vars, ISEQ_BODY(CFP_ISEQ(cfp))->local_table[i]);
2781 if (!VM_ENV_LOCAL_P(cfp->ep)) {
2783 const VALUE *ep = VM_CF_PREV_EP(cfp);
2785 if (vm_collect_local_variables_in_heap(ep, &vars)) {
2789 while (cfp->ep != ep) {
2790 cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
2798 return local_var_list_finish(&vars);
2822rb_f_block_given_p(
VALUE _)
2826 cfp = vm_get_ruby_level_caller_cfp(ec, RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp));
2828 return RBOOL(cfp != NULL && VM_CF_BLOCK_HANDLER(cfp) != VM_BLOCK_HANDLER_NONE);
2839rb_f_iterator_p(
VALUE self)
2841 rb_warn_deprecated(
"iterator?",
"block_given?");
2842 return rb_f_block_given_p(self);
2846rb_current_realfilepath(
void)
2850 cfp = vm_get_ruby_level_caller_cfp(ec, RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp));
2853 VALUE path = rb_iseq_realpath(iseq);
2854 if (
RTEST(path))
return path;
2856 path = rb_iseq_path(iseq);
2857 if (path == eval_default_path) {
2862 const long len = RSTRING_LEN(path);
2863 if (
len > EVAL_LOCATION_MARK_LEN+1) {
2864 const char *
const ptr = RSTRING_PTR(path);
2865 if (ptr[
len - 1] ==
')' &&
2866 memcmp(ptr,
"("EVAL_LOCATION_MARK, EVAL_LOCATION_MARK_LEN+1) == 0) {
2879rb_current_ifunc(
void)
2882 VALUE ifunc = (
VALUE)CFP_ISEQ(GET_EC()->cfp);
2904 VM_METHOD_TYPE_OPTIMIZED, (
void *)OPTIMIZED_METHOD_TYPE_SEND, METHOD_VISI_PUBLIC);
2906 VM_METHOD_TYPE_OPTIMIZED, (
void *)OPTIMIZED_METHOD_TYPE_SEND, METHOD_VISI_PUBLIC);
2918 rb_eUncaughtThrow =
rb_define_class(
"UncaughtThrowError", rb_eArgError);
2919 rb_define_method(rb_eUncaughtThrow,
"initialize", uncaught_throw_init, -1);
#define RUBY_ASSERT_ALWAYS(expr,...)
A variant of RUBY_ASSERT that does not interface with RUBY_DEBUG.
#define RUBY_ASSERT(...)
Asserts that the given expression is truthy if and only if RUBY_DEBUG is truthy.
uint32_t pm_constant_id_t
A constant id is a unique identifier for a constant in the constant pool.
#define rb_define_method(klass, mid, func, arity)
Defines klass#mid.
#define rb_define_private_method(klass, mid, func, arity)
Defines klass#mid and makes it private.
#define rb_define_global_function(mid, func, arity)
Defines rb_mKernel #mid.
#define RUBY_EVENT_C_CALL
A method, written in C, is called.
#define RUBY_EVENT_C_RETURN
Return from a method, written in C.
VALUE rb_define_class(const char *name, VALUE super)
Defines a top-level class.
void rb_extend_object(VALUE obj, VALUE module)
Extend the object with the module.
VALUE rb_module_new(void)
Creates a new, anonymous module.
int rb_scan_args(int argc, const VALUE *argv, const char *fmt,...)
Retrieves argument from argc and argv to given VALUE references according to the format string.
int rb_keyword_given_p(void)
Determines if the current method is given a keyword argument.
int rb_block_given_p(void)
Determines if the current method is given a block.
#define rb_str_new2
Old name of rb_str_new_cstr.
#define T_COMPLEX
Old name of RUBY_T_COMPLEX.
#define T_FILE
Old name of RUBY_T_FILE.
#define T_STRING
Old name of RUBY_T_STRING.
#define T_MASK
Old name of RUBY_T_MASK.
#define Qundef
Old name of RUBY_Qundef.
#define T_NIL
Old name of RUBY_T_NIL.
#define T_FLOAT
Old name of RUBY_T_FLOAT.
#define T_IMEMO
Old name of RUBY_T_IMEMO.
#define ID2SYM
Old name of RB_ID2SYM.
#define T_BIGNUM
Old name of RUBY_T_BIGNUM.
#define SPECIAL_CONST_P
Old name of RB_SPECIAL_CONST_P.
#define T_STRUCT
Old name of RUBY_T_STRUCT.
#define OBJ_FREEZE
Old name of RB_OBJ_FREEZE.
#define T_FIXNUM
Old name of RUBY_T_FIXNUM.
#define UNREACHABLE_RETURN
Old name of RBIMPL_UNREACHABLE_RETURN.
#define SYM2ID
Old name of RB_SYM2ID.
#define T_DATA
Old name of RUBY_T_DATA.
#define CLASS_OF
Old name of rb_class_of.
#define T_NONE
Old name of RUBY_T_NONE.
#define T_NODE
Old name of RUBY_T_NODE.
#define rb_ary_new4
Old name of rb_ary_new_from_values.
#define xmalloc
Old name of ruby_xmalloc.
#define T_MODULE
Old name of RUBY_T_MODULE.
#define T_TRUE
Old name of RUBY_T_TRUE.
#define T_RATIONAL
Old name of RUBY_T_RATIONAL.
#define T_ICLASS
Old name of RUBY_T_ICLASS.
#define T_HASH
Old name of RUBY_T_HASH.
#define T_FALSE
Old name of RUBY_T_FALSE.
#define T_UNDEF
Old name of RUBY_T_UNDEF.
#define Qtrue
Old name of RUBY_Qtrue.
#define NUM2INT
Old name of RB_NUM2INT.
#define T_ZOMBIE
Old name of RUBY_T_ZOMBIE.
#define Qnil
Old name of RUBY_Qnil.
#define Qfalse
Old name of RUBY_Qfalse.
#define ENC_CODERANGE_BROKEN
Old name of RUBY_ENC_CODERANGE_BROKEN.
#define T_ARRAY
Old name of RUBY_T_ARRAY.
#define T_OBJECT
Old name of RUBY_T_OBJECT.
#define NIL_P
Old name of RB_NIL_P.
#define ALLOCV_N
Old name of RB_ALLOCV_N.
#define T_SYMBOL
Old name of RUBY_T_SYMBOL.
#define T_MATCH
Old name of RUBY_T_MATCH.
#define T_CLASS
Old name of RUBY_T_CLASS.
#define BUILTIN_TYPE
Old name of RB_BUILTIN_TYPE.
#define T_MOVED
Old name of RUBY_T_MOVED.
#define Check_TypedStruct(v, t)
Old name of rb_check_typeddata.
#define ALLOCV_END
Old name of RB_ALLOCV_END.
#define SYMBOL_P
Old name of RB_SYMBOL_P.
#define T_REGEXP
Old name of RUBY_T_REGEXP.
VALUE rb_eNotImpError
NotImplementedError exception.
void rb_exc_raise(VALUE mesg)
Raises an exception in the current thread.
VALUE rb_eNameError
NameError exception.
VALUE rb_eNoMethodError
NoMethodError exception.
VALUE rb_eRuntimeError
RuntimeError exception.
VALUE rb_mKernel
Kernel module.
VALUE rb_cObject
Object class.
VALUE rb_obj_alloc(VALUE klass)
Allocates an instance of the given class.
VALUE rb_class_new_instance(int argc, const VALUE *argv, VALUE klass)
Allocates, then initialises an instance of the given class.
VALUE rb_obj_class(VALUE obj)
Queries the class of an object.
VALUE rb_cBasicObject
BasicObject class.
VALUE rb_cModule
Module class.
VALUE rb_obj_clone(VALUE obj)
Produces a shallow copy of the given object.
VALUE rb_obj_is_kind_of(VALUE obj, VALUE klass)
Queries if the given object is an instance (of possibly descendants) of the given class.
VALUE rb_eval_string_wrap(const char *str, int *state)
Identical to rb_eval_string_protect(), except it evaluates the given string under a module binding in...
VALUE rb_funcall_passing_block(VALUE recv, ID mid, int argc, const VALUE *argv)
Identical to rb_funcallv_public(), except you can pass the passed block.
VALUE rb_funcall(VALUE recv, ID mid, int n,...)
Calls a method.
VALUE rb_funcallv_kw(VALUE recv, ID mid, int argc, const VALUE *argv, int kw_splat)
Identical to rb_funcallv(), except you can specify how to handle the last element of the given array.
VALUE rb_funcall_with_block(VALUE recv, ID mid, int argc, const VALUE *argv, VALUE procval)
Identical to rb_funcallv_public(), except you can pass a block.
VALUE rb_eval_string_protect(const char *str, int *state)
Identical to rb_eval_string(), except it avoids potential global escapes.
VALUE rb_call_super_kw(int argc, const VALUE *argv, int kw_splat)
Identical to rb_call_super(), except you can specify how to handle the last element of the given arra...
VALUE rb_funcallv_public(VALUE recv, ID mid, int argc, const VALUE *argv)
Identical to rb_funcallv(), except it only takes public methods into account.
VALUE rb_current_receiver(void)
This resembles ruby's self.
VALUE rb_funcall_passing_block_kw(VALUE recv, ID mid, int argc, const VALUE *argv, int kw_splat)
Identical to rb_funcallv_passing_block(), except you can specify how to handle the last element of th...
VALUE rb_funcall_with_block_kw(VALUE recv, ID mid, int argc, const VALUE *argv, VALUE procval, int kw_splat)
Identical to rb_funcallv_with_block(), except you can specify how to handle the last element of the g...
VALUE rb_eval_string(const char *str)
Evaluates the given string.
VALUE rb_call_super(int argc, const VALUE *argv)
This resembles ruby's super.
VALUE rb_funcallv_public_kw(VALUE recv, ID mid, int argc, const VALUE *argv, int kw_splat)
Identical to rb_funcallv_public(), except you can specify how to handle the last element of the given...
VALUE rb_ary_cat(VALUE ary, const VALUE *train, long len)
Destructively appends multiple elements at the end of the array.
VALUE rb_check_array_type(VALUE obj)
Try converting an object to its array representation using its to_ary method, if any.
VALUE rb_ary_pop(VALUE ary)
Destructively deletes an element from the end of the passed array and returns what was deleted.
VALUE rb_ary_hidden_new(long capa)
Allocates a hidden (no class) empty array.
VALUE rb_ary_subseq(VALUE ary, long beg, long len)
Obtains a part of the passed array.
#define UNLIMITED_ARGUMENTS
This macro is used in conjunction with rb_check_arity().
static int rb_check_arity(int argc, int min, int max)
Ensures that the passed integer is in the passed range.
int rb_is_local_id(ID id)
Classifies the given ID, then sees if it is a local variable.
VALUE rb_proc_lambda_p(VALUE recv)
Queries if the given object is a lambda.
#define rb_str_new_cstr(str)
Identical to rb_str_new, except it assumes the passed pointer is a pointer to a C string.
VALUE rb_str_intern(VALUE str)
Identical to rb_to_symbol(), except it assumes the receiver being an instance of RString.
VALUE rb_ivar_set(VALUE obj, ID name, VALUE val)
Identical to rb_iv_set(), except it accepts the name as an ID instead of a C string.
VALUE rb_ivar_get(VALUE obj, ID name)
Identical to rb_iv_get(), except it accepts the name as an ID instead of a C string.
VALUE rb_check_funcall(VALUE recv, ID mid, int argc, const VALUE *argv)
Identical to rb_funcallv(), except it returns RUBY_Qundef instead of raising rb_eNoMethodError.
VALUE rb_check_funcall_kw(VALUE recv, ID mid, int argc, const VALUE *argv, int kw_splat)
Identical to rb_check_funcall(), except you can specify how to handle the last element of the given a...
VALUE rb_mod_module_eval(int argc, const VALUE *argv, VALUE mod)
Identical to rb_obj_instance_eval(), except it evaluates within the context of module.
VALUE rb_mod_module_exec(int argc, const VALUE *argv, VALUE mod)
Identical to rb_obj_instance_exec(), except it evaluates within the context of module.
VALUE rb_obj_instance_exec(int argc, const VALUE *argv, VALUE recv)
Executes the given block within the context of the receiver.
VALUE rb_eval_cmd_kw(VALUE cmd, VALUE arg, int kw_splat)
This API is practically a variant of rb_proc_call_kw() now.
VALUE rb_apply(VALUE recv, ID mid, VALUE args)
Identical to rb_funcallv(), except it takes Ruby's array instead of C's.
VALUE rb_obj_instance_eval(int argc, const VALUE *argv, VALUE recv)
Evaluates a string containing Ruby source code, or the given block, within the context of the receive...
static ID rb_intern_const(const char *str)
This is a "tiny optimisation" over rb_intern().
ID rb_check_id(volatile VALUE *namep)
Detects if the given name is already interned or not.
int len
Length of the buffer.
VALUE rb_str_format(int argc, const VALUE *argv, VALUE fmt)
Formats a string.
#define RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg)
Shim for block function parameters.
VALUE rb_each(VALUE obj)
This is a shorthand of calling obj.each.
VALUE rb_yield_values(int n,...)
Identical to rb_yield(), except it takes variadic number of parameters and pass them to the block.
VALUE rb_yield_splat(VALUE ary)
Identical to rb_yield_values(), except it splats an array to generate the list of parameters.
void rb_throw(const char *tag, VALUE val)
Transfers control to the end of the active catch block waiting for tag.
VALUE rb_yield_values2(int n, const VALUE *argv)
Identical to rb_yield_values(), except it takes the parameters as a C array instead of variadic argum...
VALUE rb_yield(VALUE val)
Yields the block.
VALUE rb_yield_values_kw(int n, const VALUE *argv, int kw_splat)
Identical to rb_yield_values2(), except you can specify how to handle the last element of the given a...
rb_block_call_func * rb_block_call_func_t
Shorthand type that represents an iterator-written-in-C function pointer.
VALUE rb_yield_block(RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg))
Pass a passed block.
void rb_throw_obj(VALUE tag, VALUE val)
Identical to rb_throw(), except it allows arbitrary Ruby object to become a tag.
VALUE rb_block_call_func(RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg))
This is the type of a function that the interpreter expect for C-backended blocks.
VALUE rb_block_call_kw(VALUE obj, ID mid, int argc, const VALUE *argv, rb_block_call_func_t proc, VALUE data2, int kw_splat)
Identical to rb_funcallv_kw(), except it additionally passes a function as a block.
VALUE rb_yield_splat_kw(VALUE ary, int kw_splat)
Identical to rb_yield_splat(), except you can specify how to handle the last element of the given arr...
#define MEMCPY(p1, p2, type, n)
Handy macro to call memcpy.
#define ALLOCA_N(type, n)
#define MEMZERO(p, type, n)
Handy macro to erase a region of memory.
#define RB_GC_GUARD(v)
Prevents premature destruction of local objects.
VALUE rb_catch_obj(VALUE q, type *w, VALUE e)
An equivalent of Kernel#catch.
VALUE rb_catch(const char *q, type *w, VALUE e)
An equivalent of Kernel#catch.
VALUE rb_block_call(VALUE q, ID w, int e, const VALUE *r, type *t, VALUE y)
Call a method with a block.
VALUE type(ANYARGS)
ANYARGS-ed function type.
VALUE rb_rescue2(type *q, VALUE w, type *e, VALUE r,...)
An equivalent of rescue clause.
static const uint8_t PM_OPTIONS_SCOPE_FORWARDING_NONE
The default value for parameters.
static const uint8_t PM_OPTIONS_SCOPE_FORWARDING_ALL
When the scope is forwarding with the ... parameter.
static const uint8_t PM_OPTIONS_SCOPE_FORWARDING_POSITIONALS
When the scope is forwarding with the * parameter.
static const uint8_t PM_OPTIONS_SCOPE_FORWARDING_KEYWORDS
When the scope is forwarding with the ** parameter.
static const uint8_t PM_OPTIONS_SCOPE_FORWARDING_BLOCK
When the scope is forwarding with the & parameter.
static int RARRAY_LENINT(VALUE ary)
Identical to rb_array_len(), except it differs for the return type.
#define RARRAY_CONST_PTR
Just another name of rb_array_const_ptr.
static VALUE RBASIC_CLASS(VALUE obj)
Queries the class of an object.
#define RBASIC(obj)
Convenient casting macro.
#define RCLASS_SUPER
Just another name of rb_class_get_superclass.
#define RHASH_EMPTY_P(h)
Checks if the hash is empty.
#define StringValue(v)
Ensures that the parameter object is a String.
#define StringValuePtr(v)
Identical to StringValue, except it returns a char*.
#define RB_PASS_CALLED_KEYWORDS
Pass keywords if current method is called with keywords, useful for argument delegation.
#define RB_NO_KEYWORDS
Do not pass keywords.
#define RTEST
This is an old name of RB_TEST.
#define _(args)
This was a transition path from K&R to ANSI.
pm_parser_t * parser
The parser that will do the actual parsing.
pm_scope_node_t node
The resulting scope node that will hold the generated AST.
pm_options_t * options
The options that will be passed to the parser.
pm_arena_t * arena
The arena allocator for AST-lifetime memory.
pm_index_lookup_table_t index_lookup_table
A flat lookup table mapping constant IDs (or special IDs) to local variable indices.
A generic string type that can have various ownership semantics.
Internal header for Ruby Box.
IFUNC (Internal FUNCtion)
uintptr_t ID
Type that represents a Ruby identifier such as a variable name.
uintptr_t VALUE
Type that represents a Ruby object.
static void Check_Type(VALUE v, enum ruby_value_type t)
Identical to RB_TYPE_P(), except it raises exceptions on predication failure.
static bool RB_TYPE_P(VALUE obj, enum ruby_value_type t)
Queries if the given object is of given type.
ruby_value_type
C-level type of an object.