12#include "eval_intern.h"
14#include "internal/class.h"
15#include "internal/error.h"
16#include "internal/eval.h"
17#include "internal/gc.h"
18#include "internal/hash.h"
19#include "internal/object.h"
20#include "internal/proc.h"
21#include "internal/symbol.h"
25#include "ractor_core.h"
48static int method_arity(
VALUE);
49static int method_min_max_arity(
VALUE,
int *max);
54#define IS_METHOD_PROC_IFUNC(ifunc) ((ifunc)->func == bmcall)
57block_mark_and_move(
struct rb_block *block)
59 switch (block->type) {
61 case block_type_ifunc:
64 rb_gc_mark_and_move(&captured->self);
65 rb_gc_mark_and_move(&captured->code.val);
67 rb_gc_mark_and_move((
VALUE *)&captured->ep[VM_ENV_DATA_INDEX_ENV]);
71 case block_type_symbol:
72 rb_gc_mark_and_move(&block->as.symbol);
75 rb_gc_mark_and_move(&block->as.proc);
81proc_mark_and_move(
void *ptr)
84 block_mark_and_move((
struct rb_block *)&proc->block);
89 VALUE env[VM_ENV_DATA_SIZE + 1];
93proc_memsize(
const void *ptr)
96 if (proc->block.as.captured.ep == ((
const cfunc_proc_t *)ptr)->env+1)
112#define proc_data_type ruby_proc_data_type
115rb_proc_alloc(
VALUE klass)
124 return RBOOL(rb_typeddata_is_kind_of(proc, &proc_data_type));
129proc_clone(
VALUE self)
131 VALUE procval = rb_proc_dup(self);
132 return rb_obj_clone_setup(self, procval,
Qnil);
139 VALUE procval = rb_proc_dup(self);
140 return rb_obj_dup_setup(self, procval);
250 GetProcPtr(procval, proc);
252 return RBOOL(proc->is_lambda);
258binding_free(
void *ptr)
260 RUBY_FREE_ENTER(
"binding");
262 RUBY_FREE_LEAVE(
"binding");
266binding_mark_and_move(
void *ptr)
270 block_mark_and_move((
struct rb_block *)&bind->block);
271 rb_gc_mark_and_move((
VALUE *)&bind->pathobj);
275binding_memsize(
const void *ptr)
283 binding_mark_and_move,
286 binding_mark_and_move,
292rb_binding_alloc(
VALUE klass)
298 rb_yjit_collect_binding_alloc();
304binding_copy(
VALUE self)
308 GetBindingPtr(self, src);
309 GetBindingPtr(bindval, dst);
310 rb_vm_block_copy(bindval, &dst->block, &src->block);
312 dst->first_lineno = src->first_lineno;
318binding_dup(
VALUE self)
320 return rb_obj_dup_setup(self, binding_copy(self));
325binding_clone(
VALUE self)
327 return rb_obj_clone_setup(self, binding_copy(self),
Qnil);
334 return rb_vm_make_binding(ec, ec->cfp);
382rb_f_binding(
VALUE self)
404bind_eval(
int argc,
VALUE *argv,
VALUE bindval)
408 rb_scan_args(argc, argv,
"12", &args[0], &args[2], &args[3]);
410 return rb_f_eval(argc+1, args,
Qnil );
414get_local_variable_ptr(
const rb_env_t **envp,
ID lid,
bool search_outer)
418 if (!VM_ENV_FLAGS(env->ep, VM_FRAME_FLAG_CFRAME)) {
419 if (VM_ENV_FLAGS(env->ep, VM_ENV_FLAG_ISOLATED)) {
425 VM_ASSERT(rb_obj_is_iseq((
VALUE)iseq));
427 const unsigned int local_table_size = ISEQ_BODY(iseq)->local_table_size;
428 for (
unsigned int i=0; i<local_table_size; i++) {
429 if (ISEQ_BODY(iseq)->local_table[i] == lid) {
430 if (ISEQ_BODY(iseq)->local_iseq == iseq &&
431 ISEQ_BODY(iseq)->param.flags.has_block &&
432 (
unsigned int)ISEQ_BODY(iseq)->param.block_start == i) {
433 const VALUE *ep = env->ep;
434 if (!VM_ENV_FLAGS(ep, VM_FRAME_FLAG_MODIFIED_BLOCK_PARAM)) {
435 RB_OBJ_WRITE(env, &env->env[i], rb_vm_bh_to_procval(GET_EC(), VM_ENV_BLOCK_HANDLER(ep)));
436 VM_ENV_FLAGS_SET(ep, VM_FRAME_FLAG_MODIFIED_BLOCK_PARAM);
441 unsigned int last_lvar = env->env_size+VM_ENV_INDEX_LAST_LVAR
443 return &env->env[last_lvar - (local_table_size - i)];
451 }
while (search_outer && (env = rb_vm_env_prev_env(env)) != NULL);
463check_local_id(
VALUE bindval,
volatile VALUE *pname)
470 rb_name_err_raise(
"wrong local variable name '%1$s' for %2$s",
475 if (!rb_is_local_name(name)) {
476 rb_name_err_raise(
"wrong local variable name '%1$s' for %2$s",
503bind_local_variables(
VALUE bindval)
508 GetBindingPtr(bindval, bind);
509 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
510 return rb_vm_env_local_variables(env);
514rb_numparam_id_p(
ID id)
516 return (tNUMPARAM_1 << ID_SCOPE_SHIFT) <=
id &&
id < ((tNUMPARAM_1 + 9) << ID_SCOPE_SHIFT);
520rb_implicit_param_p(
ID id)
522 return id == idItImplicit || rb_numparam_id_p(
id);
543bind_local_variable_get(
VALUE bindval,
VALUE sym)
545 ID lid = check_local_id(bindval, &sym);
550 if (!lid)
goto undefined;
551 if (rb_numparam_id_p(lid)) {
552 rb_name_err_raise(
"numbered parameter '%1$s' is not a local variable",
556 GetBindingPtr(bindval, bind);
558 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
559 if ((ptr = get_local_variable_ptr(&env, lid, TRUE)) != NULL) {
565 rb_name_err_raise(
"local variable '%1$s' is not defined for %2$s",
598 ID lid = check_local_id(bindval, &sym);
603 if (!lid) lid = rb_intern_str(sym);
604 if (rb_numparam_id_p(lid)) {
605 rb_name_err_raise(
"numbered parameter '%1$s' is not a local variable",
609 GetBindingPtr(bindval, bind);
610 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
611 if ((ptr = get_local_variable_ptr(&env, lid, TRUE)) == NULL) {
613 ptr = rb_binding_add_dynavars(bindval, bind, 1, &lid);
614 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
618 rb_yjit_collect_binding_set();
644bind_local_variable_defined_p(
VALUE bindval,
VALUE sym)
646 ID lid = check_local_id(bindval, &sym);
651 if (rb_numparam_id_p(lid)) {
652 rb_name_err_raise(
"numbered parameter '%1$s' is not a local variable",
656 GetBindingPtr(bindval, bind);
657 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
658 return RBOOL(get_local_variable_ptr(&env, lid, TRUE));
682bind_implicit_parameters(
VALUE bindval)
687 GetBindingPtr(bindval, bind);
688 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
690 if (get_local_variable_ptr(&env, idItImplicit, FALSE)) {
691 return rb_ary_new_from_args(1,
ID2SYM(idIt));
694 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
695 return rb_vm_env_numbered_parameters(env);
719bind_implicit_parameter_get(
VALUE bindval,
VALUE sym)
721 ID lid = check_local_id(bindval, &sym);
726 if (lid == idIt) lid = idItImplicit;
728 if (!lid || !rb_implicit_param_p(lid)) {
729 rb_name_err_raise(
"'%1$s' is not an implicit parameter",
733 GetBindingPtr(bindval, bind);
735 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
736 if ((ptr = get_local_variable_ptr(&env, lid, FALSE)) != NULL) {
740 if (lid == idItImplicit) lid = idIt;
741 rb_name_err_raise(
"implicit parameter '%1$s' is not defined for %2$s", bindval,
ID2SYM(lid));
769bind_implicit_parameter_defined_p(
VALUE bindval,
VALUE sym)
771 ID lid = check_local_id(bindval, &sym);
775 if (lid == idIt) lid = idItImplicit;
777 if (!lid || !rb_implicit_param_p(lid)) {
778 rb_name_err_raise(
"'%1$s' is not an implicit parameter",
782 GetBindingPtr(bindval, bind);
783 env = VM_ENV_ENVVAL_PTR(vm_block_ep(&bind->block));
784 return RBOOL(get_local_variable_ptr(&env, lid, FALSE));
794bind_receiver(
VALUE bindval)
797 GetBindingPtr(bindval, bind);
798 return vm_block_self(&bind->block);
808bind_location(
VALUE bindval)
812 GetBindingPtr(bindval, bind);
813 loc[0] = pathobj_path(bind->pathobj);
814 loc[1] =
INT2FIX(bind->first_lineno);
827 proc = &sproc->basic;
828 vm_block_type_set(&proc->block, block_type_ifunc);
830 *(
VALUE **)&proc->block.as.captured.ep = ep = sproc->env + VM_ENV_DATA_SIZE-1;
831 ep[VM_ENV_DATA_INDEX_FLAGS] = VM_FRAME_MAGIC_IFUNC | VM_FRAME_FLAG_CFRAME | VM_ENV_FLAG_LOCAL | VM_ENV_FLAG_ESCAPED;
832 ep[VM_ENV_DATA_INDEX_ME_CREF] =
Qfalse;
833 ep[VM_ENV_DATA_INDEX_SPECVAL] = VM_BLOCK_HANDLER_NONE;
834 ep[VM_ENV_DATA_INDEX_ENV] =
Qundef;
837 RB_OBJ_WRITE(procval, &proc->block.as.captured.code.ifunc, ifunc);
838 proc->is_lambda = TRUE;
843rb_func_proc_dup(
VALUE src_obj)
845 RUBY_ASSERT(rb_typeddata_is_instance_of(src_obj, &proc_data_type));
848 GetProcPtr(src_obj, src_proc);
849 RUBY_ASSERT(vm_block_type(&src_proc->block) == block_type_ifunc);
854 memcpy(&proc->basic, src_proc,
sizeof(
rb_proc_t));
858 const VALUE *src_ep = src_proc->block.as.captured.ep;
859 VALUE *ep = *(
VALUE **)&proc->basic.block.as.captured.ep = proc->env + VM_ENV_DATA_SIZE - 1;
860 ep[VM_ENV_DATA_INDEX_FLAGS] = src_ep[VM_ENV_DATA_INDEX_FLAGS];
861 ep[VM_ENV_DATA_INDEX_ME_CREF] = src_ep[VM_ENV_DATA_INDEX_ME_CREF];
862 ep[VM_ENV_DATA_INDEX_SPECVAL] = src_ep[VM_ENV_DATA_INDEX_SPECVAL];
863 RB_OBJ_WRITE(proc_obj, &ep[VM_ENV_DATA_INDEX_ENV], src_ep[VM_ENV_DATA_INDEX_ENV]);
871 VALUE procval = rb_proc_alloc(klass);
873 GetProcPtr(procval, proc);
875 vm_block_type_set(&proc->block, block_type_symbol);
876 proc->is_lambda = TRUE;
886 min_argc >= (
int)(1U << (
SIZEOF_VALUE * CHAR_BIT) / 2) ||
889 rb_raise(
rb_eRangeError,
"minimum argument number out of range: %d",
894 max_argc >= (
int)(1U << (
SIZEOF_VALUE * CHAR_BIT) / 2) ||
897 rb_raise(
rb_eRangeError,
"maximum argument number out of range: %d",
902 struct vm_ifunc *ifunc = IMEMO_NEW(
struct vm_ifunc, imemo_ifunc, (
VALUE)rb_vm_svar_lep(ec, ec->cfp));
904 rb_gc_register_pinning_obj((
VALUE)ifunc);
908 ifunc->argc.min = min_argc;
909 ifunc->argc.max = max_argc;
917 struct vm_ifunc *ifunc = rb_vm_ifunc_new(func, (
void *)val, min_argc, max_argc);
921static const char proc_without_block[] =
"tried to create Proc object without a block";
924proc_new(
VALUE klass, int8_t is_lambda)
931 if ((block_handler = rb_vm_frame_block_handler(cfp)) == VM_BLOCK_HANDLER_NONE) {
932 rb_raise(rb_eArgError, proc_without_block);
936 switch (vm_block_handler_type(block_handler)) {
937 case block_handler_type_proc:
938 procval = VM_BH_TO_PROC(block_handler);
944 VALUE newprocval = rb_proc_dup(procval);
945 RBASIC_SET_CLASS(newprocval, klass);
950 case block_handler_type_symbol:
952 sym_proc_new(klass, VM_BH_TO_SYMBOL(block_handler)) :
953 rb_sym_to_proc(VM_BH_TO_SYMBOL(block_handler));
956 case block_handler_type_ifunc:
957 case block_handler_type_iseq:
958 return rb_vm_make_proc_lambda(ec, VM_BH_TO_CAPT_BLOCK(block_handler), klass, is_lambda);
960 VM_UNREACHABLE(proc_new);
979rb_proc_s_new(
int argc,
VALUE *argv,
VALUE klass)
981 VALUE block = proc_new(klass, FALSE);
1013f_lambda_filter_non_literal(
void)
1016 VALUE block_handler = rb_vm_frame_block_handler(cfp);
1018 if (block_handler == VM_BLOCK_HANDLER_NONE) {
1023 switch (vm_block_handler_type(block_handler)) {
1024 case block_handler_type_iseq:
1025 if (RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp)->ep == VM_BH_TO_ISEQ_BLOCK(block_handler)->ep) {
1029 case block_handler_type_symbol:
1031 case block_handler_type_proc:
1036 case block_handler_type_ifunc:
1040 rb_raise(rb_eArgError,
"the lambda method requires a literal block");
1054 f_lambda_filter_non_literal();
1109proc_call(
int argc,
VALUE *argv,
VALUE procval)
1115#if SIZEOF_LONG > SIZEOF_INT
1117check_argc(
long argc)
1119 if (argc > INT_MAX || argc < 0) {
1120 rb_raise(rb_eArgError,
"too many arguments (%lu)",
1121 (
unsigned long)argc);
1126#define check_argc(argc) (argc)
1136 GetProcPtr(self, proc);
1137 vret = rb_vm_invoke_proc(GET_EC(), proc, argc, argv,
1138 kw_splat, VM_BLOCK_HANDLER_NONE);
1151proc_to_block_handler(
VALUE procval)
1153 return NIL_P(procval) ? VM_BLOCK_HANDLER_NONE : procval;
1162 GetProcPtr(self, proc);
1163 vret = rb_vm_invoke_proc(ec, proc, argc, argv, kw_splat, proc_to_block_handler(passed_procval));
1217proc_arity(
VALUE self)
1224rb_iseq_min_max_arity(
const rb_iseq_t *iseq,
int *max)
1226 *max = ISEQ_BODY(iseq)->param.flags.has_rest == FALSE ?
1227 ISEQ_BODY(iseq)->param.lead_num + ISEQ_BODY(iseq)->param.opt_num + ISEQ_BODY(iseq)->param.post_num +
1228 (ISEQ_BODY(iseq)->param.flags.has_kw == TRUE || ISEQ_BODY(iseq)->param.flags.has_kwrest == TRUE || ISEQ_BODY(iseq)->param.flags.forwardable == TRUE)
1230 return ISEQ_BODY(iseq)->param.lead_num + ISEQ_BODY(iseq)->param.post_num + (ISEQ_BODY(iseq)->param.flags.has_kw && ISEQ_BODY(iseq)->param.keyword->required_num > 0);
1234rb_vm_block_min_max_arity(
const struct rb_block *block,
int *max)
1237 switch (vm_block_type(block)) {
1238 case block_type_iseq:
1239 return rb_iseq_min_max_arity(rb_iseq_check(block->as.captured.code.iseq), max);
1240 case block_type_proc:
1241 block = vm_proc_block(block->as.proc);
1243 case block_type_ifunc:
1245 const struct vm_ifunc *ifunc = block->as.captured.code.ifunc;
1246 if (IS_METHOD_PROC_IFUNC(ifunc)) {
1248 return method_min_max_arity((
VALUE)ifunc->data, max);
1250 *max = ifunc->argc.max;
1251 return ifunc->argc.min;
1253 case block_type_symbol:
1268rb_proc_min_max_arity(
VALUE self,
int *max)
1271 GetProcPtr(self, proc);
1272 return rb_vm_block_min_max_arity(&proc->block, max);
1280 GetProcPtr(self, proc);
1281 min = rb_vm_block_min_max_arity(&proc->block, &max);
1288 switch (vm_block_handler_type(block_handler)) {
1289 case block_handler_type_iseq:
1290 block->type = block_type_iseq;
1291 block->as.captured = *VM_BH_TO_ISEQ_BLOCK(block_handler);
1293 case block_handler_type_ifunc:
1294 block->type = block_type_ifunc;
1295 block->as.captured = *VM_BH_TO_IFUNC_BLOCK(block_handler);
1297 case block_handler_type_symbol:
1298 block->type = block_type_symbol;
1299 block->as.symbol = VM_BH_TO_SYMBOL(block_handler);
1301 case block_handler_type_proc:
1302 block->type = block_type_proc;
1303 block->as.proc = VM_BH_TO_PROC(block_handler);
1308rb_block_pair_yield_optimizable(
void)
1313 VALUE block_handler = rb_vm_frame_block_handler(cfp);
1316 if (block_handler == VM_BLOCK_HANDLER_NONE) {
1317 rb_raise(rb_eArgError,
"no block given");
1320 block_setup(&block, block_handler);
1321 min = rb_vm_block_min_max_arity(&block, &max);
1323 switch (vm_block_type(&block)) {
1324 case block_type_symbol:
1327 case block_type_proc:
1329 VALUE procval = block_handler;
1331 GetProcPtr(procval, proc);
1332 if (proc->is_lambda)
return 0;
1333 if (min != max)
return 0;
1337 case block_type_ifunc:
1339 const struct vm_ifunc *ifunc = block.as.captured.code.ifunc;
1340 if (ifunc->flags & IFUNC_YIELD_OPTIMIZABLE)
return 1;
1354 VALUE block_handler = rb_vm_frame_block_handler(cfp);
1357 if (block_handler == VM_BLOCK_HANDLER_NONE) {
1358 rb_raise(rb_eArgError,
"no block given");
1361 block_setup(&block, block_handler);
1363 switch (vm_block_type(&block)) {
1364 case block_type_symbol:
1367 case block_type_proc:
1371 min = rb_vm_block_min_max_arity(&block, &max);
1377rb_block_min_max_arity(
int *max)
1381 VALUE block_handler = rb_vm_frame_block_handler(cfp);
1384 if (block_handler == VM_BLOCK_HANDLER_NONE) {
1385 rb_raise(rb_eArgError,
"no block given");
1388 block_setup(&block, block_handler);
1389 return rb_vm_block_min_max_arity(&block, max);
1393rb_proc_get_iseq(
VALUE self,
int *is_proc)
1398 GetProcPtr(self, proc);
1399 block = &proc->block;
1400 if (is_proc) *is_proc = !proc->is_lambda;
1402 switch (vm_block_type(block)) {
1403 case block_type_iseq:
1404 return rb_iseq_check(block->as.captured.code.iseq);
1405 case block_type_proc:
1406 return rb_proc_get_iseq(block->as.proc, is_proc);
1407 case block_type_ifunc:
1409 const struct vm_ifunc *ifunc = block->as.captured.code.ifunc;
1410 if (IS_METHOD_PROC_IFUNC(ifunc)) {
1412 if (is_proc) *is_proc = 0;
1413 return rb_method_iseq((
VALUE)ifunc->data);
1419 case block_type_symbol:
1423 VM_UNREACHABLE(rb_proc_get_iseq);
1456 const rb_proc_t *self_proc, *other_proc;
1457 const struct rb_block *self_block, *other_block;
1463 GetProcPtr(self, self_proc);
1464 GetProcPtr(other, other_proc);
1466 if (self_proc->is_from_method != other_proc->is_from_method ||
1467 self_proc->is_lambda != other_proc->is_lambda) {
1471 self_block = &self_proc->block;
1472 other_block = &other_proc->block;
1474 if (vm_block_type(self_block) != vm_block_type(other_block)) {
1478 switch (vm_block_type(self_block)) {
1479 case block_type_iseq:
1480 if (self_block->as.captured.ep != \
1481 other_block->as.captured.ep ||
1482 self_block->as.captured.code.iseq != \
1483 other_block->as.captured.code.iseq) {
1487 case block_type_ifunc:
1488 if (self_block->as.captured.code.ifunc != \
1489 other_block->as.captured.code.ifunc) {
1500 case block_type_proc:
1501 if (self_block->as.proc != other_block->as.proc) {
1505 case block_type_symbol:
1506 if (self_block->as.symbol != other_block->as.symbol) {
1521 if (!iseq)
return Qnil;
1522 rb_iseq_check(iseq);
1523 loc[i++] = rb_iseq_path(iseq);
1537 return iseq_location(iseq);
1556rb_proc_location(
VALUE self)
1558 return iseq_location(rb_proc_get_iseq(self, 0));
1562rb_unnamed_parameters(
int arity)
1565 int n = (arity < 0) ? ~arity : arity;
1599rb_proc_parameters(
int argc,
VALUE *argv,
VALUE self)
1601 static ID keyword_ids[1];
1607 iseq = rb_proc_get_iseq(self, &is_proc);
1609 if (!keyword_ids[0]) {
1610 CONST_ID(keyword_ids[0],
"lambda");
1617 if (!
NIL_P(lambda)) {
1618 is_proc = !
RTEST(lambda);
1625 return rb_iseq_parameters(iseq, is_proc);
1629rb_hash_proc(st_index_t hash,
VALUE prc)
1632 GetProcPtr(prc, proc);
1634 switch (vm_block_type(&proc->block)) {
1635 case block_type_iseq:
1636 hash = rb_st_hash_uint(hash, (st_index_t)proc->block.as.captured.code.iseq->body);
1638 case block_type_ifunc:
1639 hash = rb_st_hash_uint(hash, (st_index_t)proc->block.as.captured.code.ifunc->func);
1640 hash = rb_st_hash_uint(hash, (st_index_t)proc->block.as.captured.code.ifunc->data);
1642 case block_type_symbol:
1643 hash = rb_st_hash_uint(hash, rb_any_hash(proc->block.as.symbol));
1645 case block_type_proc:
1646 hash = rb_st_hash_uint(hash, rb_any_hash(proc->block.as.proc));
1649 rb_bug(
"rb_hash_proc: unknown block type %d", vm_block_type(&proc->block));
1655 if (vm_block_type(&proc->block) != block_type_ifunc) {
1656 hash =
rb_hash_uint(hash, (st_index_t)proc->block.as.captured.ep);
1679rb_sym_to_proc(
VALUE sym)
1681 enum {SYM_PROC_CACHE_SIZE = 67};
1683 if (rb_ractor_main_p()) {
1684 if (!sym_proc_cache) {
1690 long index = (
id % SYM_PROC_CACHE_SIZE);
1692 if (
RTEST(procval)) {
1694 GetProcPtr(procval, proc);
1696 if (proc->block.as.symbol == sym) {
1701 procval = sym_proc_new(
rb_cProc, sym);
1707 return sym_proc_new(
rb_cProc, sym);
1721proc_hash(
VALUE self)
1725 hash = rb_hash_proc(hash, self);
1731rb_block_to_s(
VALUE self,
const struct rb_block *block,
const char *additional_info)
1734 VALUE str = rb_sprintf(
"#<%"PRIsVALUE
":", cname);
1737 switch (vm_block_type(block)) {
1738 case block_type_proc:
1739 block = vm_proc_block(block->as.proc);
1741 case block_type_iseq:
1743 const rb_iseq_t *iseq = rb_iseq_check(block->as.captured.code.iseq);
1744 rb_str_catf(str,
"%p %"PRIsVALUE
":%d", (
void *)self,
1746 ISEQ_BODY(iseq)->location.first_lineno);
1749 case block_type_symbol:
1750 rb_str_catf(str,
"%p(&%+"PRIsVALUE
")", (
void *)self, block->as.symbol);
1752 case block_type_ifunc:
1753 rb_str_catf(str,
"%p", (
void *)block->as.captured.code.ifunc);
1771proc_to_s(
VALUE self)
1774 GetProcPtr(self, proc);
1775 return rb_block_to_s(self, &proc->block, proc->is_lambda ?
" (lambda)" : NULL);
1787proc_to_proc(
VALUE self)
1793bm_mark_and_move(
void *ptr)
1795 struct METHOD *data = ptr;
1796 rb_gc_mark_and_move((
VALUE *)&data->recv);
1797 rb_gc_mark_and_move((
VALUE *)&data->klass);
1798 rb_gc_mark_and_move((
VALUE *)&data->iclass);
1799 rb_gc_mark_and_move((
VALUE *)&data->owner);
1817 return RBOOL(rb_typeddata_is_kind_of(m, &method_data_type));
1824 ID rmiss = idRespond_to_missing;
1826 if (UNDEF_P(obj))
return 0;
1827 if (rb_method_basic_definition_p(klass, rmiss))
return 0;
1845 def->type = VM_METHOD_TYPE_MISSING;
1846 def->original_id = id;
1848 me = rb_method_entry_create(
id, klass, METHOD_VISI_UNDEF, def);
1860 if (!respond_to_missing_p(klass, obj, vid, scope))
return Qfalse;
1861 return mnew_missing(klass, obj,
SYM2ID(vid), mclass);
1871 rb_method_visibility_t visi = METHOD_VISI_UNDEF;
1874 if (UNDEFINED_METHOD_ENTRY_P(me)) {
1875 if (respond_to_missing_p(klass, obj,
ID2SYM(
id), scope)) {
1876 return mnew_missing(klass, obj,
id, mclass);
1878 if (!error)
return Qnil;
1879 rb_print_undef(klass,
id, METHOD_VISI_UNDEF);
1881 if (visi == METHOD_VISI_UNDEF) {
1882 visi = METHOD_ENTRY_VISI(me);
1884 if (scope && (visi != METHOD_VISI_PUBLIC)) {
1885 if (!error)
return Qnil;
1886 rb_print_inaccessible(klass,
id, visi);
1889 if (me->def->type == VM_METHOD_TYPE_ZSUPER) {
1890 if (me->defined_class) {
1892 id = me->def->original_id;
1893 me = (
rb_method_entry_t *)rb_callable_method_entry_with_refinements(klass,
id, &iclass);
1897 id = me->def->original_id;
1898 me = rb_method_entry_without_refinements(klass,
id, &iclass);
1914 RB_OBJ_WRITE(method, &data->owner, original_me->owner);
1924 return mnew_internal(me, klass, iclass, obj,
id, mclass, scope, TRUE);
1934 me = (
rb_method_entry_t *)rb_callable_method_entry_with_refinements(klass,
id, &iclass);
1935 return mnew_from_me(me, klass, iclass, obj,
id, mclass, scope);
1939mnew_unbound(
VALUE klass,
ID id,
VALUE mclass,
int scope)
1944 me = rb_method_entry_with_refinements(klass,
id, &iclass);
1945 return mnew_from_me(me, klass, iclass,
Qundef,
id, mclass, scope);
1951 VALUE defined_class = me->defined_class;
1952 return defined_class ? defined_class : me->owner;
1996 VALUE klass1, klass2;
2004 m1 = (
struct METHOD *)RTYPEDDATA_GET_DATA(method);
2005 m2 = (
struct METHOD *)RTYPEDDATA_GET_DATA(other);
2007 klass1 = method_entry_defined_class(m1->me);
2008 klass2 = method_entry_defined_class(m2->me);
2012 if (!rb_method_entry_eq(m1->me, m2->me) ||
2014 m1->klass != m2->klass ||
2015 m1->recv != m2->recv) {
2036#define unbound_method_eq method_eq
2048method_hash(
VALUE method)
2055 hash = rb_hash_method_entry(hash, m->me);
2071method_unbind(
VALUE obj)
2074 struct METHOD *orig, *data;
2078 &method_data_type, data);
2083 RB_OBJ_WRITE(method, &data->me, rb_method_entry_clone(orig->me));
2098method_receiver(
VALUE obj)
2114method_name(
VALUE obj)
2119 return ID2SYM(data->me->called_id);
2136method_original_name(
VALUE obj)
2141 return ID2SYM(data->me->def->original_id);
2163method_owner(
VALUE obj)
2177method_box(
VALUE obj)
2183 box = data->me->def->box;
2184 if (!box)
return Qnil;
2185 if (box->box_object)
return box->box_object;
2186 rb_bug(
"Unexpected box on the method definition: %p", (
void*) box);
2193#define MSG(s) rb_fstring_lit("undefined method '%1$s' for"s" '%2$s'")
2197 if (RCLASS_SINGLETON_P(c)) {
2198 VALUE obj = RCLASS_ATTACHED_OBJECT(klass);
2215 rb_name_err_raise_str(s, c, str);
2227 VALUE m = mnew_missing_by_name(klass, obj, &vid, scope, mclass);
2229 rb_method_name_error(klass, vid);
2231 return mnew_callable(klass, obj,
id, mclass, scope);
2276 return obj_method(obj, vid, FALSE);
2289 return obj_method(obj, vid, TRUE);
2293rb_obj_singleton_method_lookup(
VALUE arg)
2300rb_obj_singleton_method_lookup_fail(
VALUE arg1,
VALUE arg2)
2337 NIL_P(klass = RCLASS_ORIGIN(sc)) ||
2338 !
NIL_P(rb_special_singleton_class(obj))) {
2347 VALUE args[2] = {obj, vid};
2348 VALUE ruby_method =
rb_rescue(rb_obj_singleton_method_lookup, (
VALUE)args, rb_obj_singleton_method_lookup_fail,
Qfalse);
2350 struct METHOD *method = (
struct METHOD *)RTYPEDDATA_GET_DATA(ruby_method);
2353 VALUE method_class = method->iclass;
2357 if (lookup_class == method_class) {
2361 }
while (lookup_class && lookup_class != stop_class);
2367 rb_name_err_raise(
"undefined singleton method '%1$s' for '%2$s'",
2408 rb_method_name_error(mod, vid);
2421rb_mod_public_instance_method(
VALUE mod,
VALUE vid)
2425 rb_method_name_error(mod, vid);
2436 int is_method = FALSE;
2455 "wrong argument type %s (expected Proc/Method/UnboundMethod)",
2462 struct METHOD *method = (
struct METHOD *)RTYPEDDATA_GET_DATA(body);
2465 if (RCLASS_SINGLETON_P(method->me->owner)) {
2467 "can't bind singleton method to a different class");
2471 "bind argument must be a subclass of % "PRIsVALUE,
2475 rb_method_entry_set(mod,
id, method->me, scope_visi->method_visi);
2476 if (scope_visi->module_func) {
2482 VALUE procval = rb_proc_dup(body);
2483 if (vm_proc_iseq(procval) != NULL) {
2485 GetProcPtr(procval, proc);
2486 proc->is_lambda = TRUE;
2487 proc->is_from_method = TRUE;
2489 rb_add_method(mod,
id, VM_METHOD_TYPE_BMETHOD, (
void *)procval, scope_visi->method_visi);
2490 if (scope_visi->module_func) {
2491 rb_add_method(
rb_singleton_class(mod),
id, VM_METHOD_TYPE_BMETHOD, (
void *)body, METHOD_VISI_PUBLIC);
2539rb_mod_define_method(
int argc,
VALUE *argv,
VALUE mod)
2541 const rb_cref_t *cref = rb_vm_cref_in_context(mod, mod);
2546 scope_visi = CREF_SCOPE_VISI(cref);
2549 return rb_mod_define_method_with_visibility(argc, argv, mod, scope_visi);
2584rb_obj_define_method(
int argc,
VALUE *argv,
VALUE obj)
2589 return rb_mod_define_method_with_visibility(argc, argv, klass, &scope_visi);
2600top_define_method(
int argc,
VALUE *argv,
VALUE obj)
2602 return rb_mod_define_method(argc, argv, rb_top_main_class(
"define_method"));
2623method_clone(
VALUE self)
2626 struct METHOD *orig, *data;
2630 rb_obj_clone_setup(self, clone,
Qnil);
2635 RB_OBJ_WRITE(clone, &data->me, rb_method_entry_clone(orig->me));
2641method_dup(
VALUE self)
2644 struct METHOD *orig, *data;
2648 rb_obj_dup_setup(self, clone);
2653 RB_OBJ_WRITE(clone, &data->me, rb_method_entry_clone(orig->me));
2682rb_method_call_pass_called_kw(
int argc,
const VALUE *argv,
VALUE method)
2702method_callable_method_entry(
const struct METHOD *data)
2704 if (data->me->defined_class == 0) rb_bug(
"method_callable_method_entry: not callable.");
2710 int argc,
const VALUE *argv,
VALUE passed_procval,
int kw_splat)
2712 vm_passed_block_handler_set(ec, proc_to_block_handler(passed_procval));
2713 return rb_vm_call_kw(ec, data->recv, data->me->called_id, argc, argv,
2714 method_callable_method_entry(data), kw_splat);
2720 const struct METHOD *data;
2724 if (UNDEF_P(data->recv)) {
2725 rb_raise(
rb_eTypeError,
"can't call unbound method; bind first");
2727 return call_method_data(ec, data, argc, argv, passed_procval, kw_splat);
2794 VALUE methclass = data->owner;
2795 VALUE iclass = data->me->defined_class;
2799 VALUE refined_class = rb_refinement_module_get_refined_class(methclass);
2800 if (!
NIL_P(refined_class)) methclass = refined_class;
2803 if (RCLASS_SINGLETON_P(methclass)) {
2805 "singleton method called for a different object");
2808 rb_raise(
rb_eTypeError,
"bind argument must be an instance of % "PRIsVALUE,
2815 me = rb_method_entry_clone(data->me);
2825 me = rb_method_entry_clone(me);
2827 VALUE ic = rb_class_search_ancestor(klass, me->owner);
2833 klass = rb_include_class_new(methclass, klass);
2835 me = (
const rb_method_entry_t *) rb_method_entry_complement_defined_class(me, me->called_id, klass);
2838 *methclass_out = methclass;
2840 *iclass_out = iclass;
2882 VALUE methclass, klass, iclass;
2884 const struct METHOD *data;
2886 convert_umethod_to_method_components(data, recv, &methclass, &klass, &iclass, &me,
true);
2908umethod_bind_call(
int argc,
VALUE *argv,
VALUE method)
2911 VALUE recv = argv[0];
2918 const struct METHOD *data;
2923 vm_passed_block_handler_set(ec, proc_to_block_handler(passed_procval));
2927 VALUE methclass, klass, iclass;
2929 convert_umethod_to_method_components(data, recv, &methclass, &klass, &iclass, &me,
false);
2930 struct METHOD bound = { recv, klass, 0, methclass, me };
2945 if (!def)
return *max = 0;
2946 switch (def->type) {
2947 case VM_METHOD_TYPE_CFUNC:
2948 if (def->body.cfunc.argc < 0) {
2952 return *max = check_argc(def->body.cfunc.argc);
2953 case VM_METHOD_TYPE_ZSUPER:
2956 case VM_METHOD_TYPE_ATTRSET:
2958 case VM_METHOD_TYPE_IVAR:
2960 case VM_METHOD_TYPE_ALIAS:
2961 def = def->body.alias.original_me->def;
2963 case VM_METHOD_TYPE_BMETHOD:
2964 return rb_proc_min_max_arity(def->body.bmethod.proc, max);
2965 case VM_METHOD_TYPE_ISEQ:
2966 return rb_iseq_min_max_arity(rb_iseq_check(def->body.iseq.
iseqptr), max);
2967 case VM_METHOD_TYPE_UNDEF:
2968 case VM_METHOD_TYPE_NOTIMPLEMENTED:
2970 case VM_METHOD_TYPE_MISSING:
2973 case VM_METHOD_TYPE_OPTIMIZED: {
2974 switch (def->body.optimized.type) {
2975 case OPTIMIZED_METHOD_TYPE_SEND:
2978 case OPTIMIZED_METHOD_TYPE_CALL:
2981 case OPTIMIZED_METHOD_TYPE_BLOCK_CALL:
2984 case OPTIMIZED_METHOD_TYPE_STRUCT_AREF:
2987 case OPTIMIZED_METHOD_TYPE_STRUCT_ASET:
2995 case VM_METHOD_TYPE_REFINED:
2999 rb_bug(
"method_def_min_max_arity: invalid method entry type (%d)", def->type);
3006 int max, min = method_def_min_max_arity(def, &max);
3007 return min == max ? min : -min-1;
3013 return method_def_arity(me->def);
3060method_arity_m(
VALUE method)
3062 int n = method_arity(method);
3067method_arity(
VALUE method)
3072 return rb_method_entry_arity(data->me);
3076original_method_entry(
VALUE mod,
ID id)
3080 while ((me = rb_method_entry(mod,
id)) != 0) {
3082 if (def->type != VM_METHOD_TYPE_ZSUPER)
break;
3084 id = def->original_id;
3090method_min_max_arity(
VALUE method,
int *max)
3092 const struct METHOD *data;
3095 return method_def_min_max_arity(data->me->def, max);
3103 return rb_method_entry_arity(me);
3113rb_callable_receiver(
VALUE callable)
3116 VALUE binding = proc_binding(callable);
3117 return rb_funcall(binding, rb_intern(
"receiver"), 0);
3120 return method_receiver(callable);
3128rb_method_def(
VALUE method)
3130 const struct METHOD *data;
3133 return data->me->def;
3139 switch (def->type) {
3140 case VM_METHOD_TYPE_ISEQ:
3141 return rb_iseq_check(def->body.iseq.
iseqptr);
3142 case VM_METHOD_TYPE_BMETHOD:
3143 return rb_proc_get_iseq(def->body.bmethod.proc, 0);
3144 case VM_METHOD_TYPE_ALIAS:
3145 return method_def_iseq(def->body.alias.original_me->def);
3146 case VM_METHOD_TYPE_CFUNC:
3147 case VM_METHOD_TYPE_ATTRSET:
3148 case VM_METHOD_TYPE_IVAR:
3149 case VM_METHOD_TYPE_ZSUPER:
3150 case VM_METHOD_TYPE_UNDEF:
3151 case VM_METHOD_TYPE_NOTIMPLEMENTED:
3152 case VM_METHOD_TYPE_OPTIMIZED:
3153 case VM_METHOD_TYPE_MISSING:
3154 case VM_METHOD_TYPE_REFINED:
3161rb_method_iseq(
VALUE method)
3163 return method_def_iseq(rb_method_def(method));
3167method_cref(
VALUE method)
3172 switch (def->type) {
3173 case VM_METHOD_TYPE_ISEQ:
3174 return def->body.iseq.
cref;
3175 case VM_METHOD_TYPE_ALIAS:
3176 def = def->body.alias.original_me->def;
3186 if (def->type == VM_METHOD_TYPE_ATTRSET || def->type == VM_METHOD_TYPE_IVAR) {
3187 if (!def->body.attr.location)
3191 return iseq_location(method_def_iseq(def));
3197 if (!me)
return Qnil;
3198 return method_def_location(me->def);
3217rb_method_location(
VALUE method)
3219 return method_def_location(rb_method_def(method));
3223vm_proc_method_def(
VALUE procval)
3229 GetProcPtr(procval, proc);
3230 block = &proc->block;
3232 if (vm_block_type(block) == block_type_ifunc &&
3233 IS_METHOD_PROC_IFUNC(ifunc = block->as.captured.code.ifunc)) {
3234 return rb_method_def((
VALUE)ifunc->data);
3247 switch (def->type) {
3248 case VM_METHOD_TYPE_ISEQ:
3249 iseq = method_def_iseq(def);
3250 return rb_iseq_parameters(iseq, 0);
3251 case VM_METHOD_TYPE_BMETHOD:
3252 if ((iseq = method_def_iseq(def)) != NULL) {
3253 return rb_iseq_parameters(iseq, 0);
3255 else if ((bmethod_def = vm_proc_method_def(def->body.bmethod.proc)) != NULL) {
3256 return method_def_parameters(bmethod_def);
3260 case VM_METHOD_TYPE_ALIAS:
3261 return method_def_parameters(def->body.alias.original_me->def);
3263 case VM_METHOD_TYPE_OPTIMIZED:
3264 if (def->body.optimized.type == OPTIMIZED_METHOD_TYPE_STRUCT_ASET) {
3265 VALUE param = rb_ary_new_from_args(2,
ID2SYM(rb_intern(
"req")),
ID2SYM(rb_intern(
"_")));
3266 return rb_ary_new_from_args(1, param);
3270 case VM_METHOD_TYPE_CFUNC:
3271 case VM_METHOD_TYPE_ATTRSET:
3272 case VM_METHOD_TYPE_IVAR:
3273 case VM_METHOD_TYPE_ZSUPER:
3274 case VM_METHOD_TYPE_UNDEF:
3275 case VM_METHOD_TYPE_NOTIMPLEMENTED:
3276 case VM_METHOD_TYPE_MISSING:
3277 case VM_METHOD_TYPE_REFINED:
3281 return rb_unnamed_parameters(method_def_arity(def));
3305rb_method_parameters(
VALUE method)
3307 return method_def_parameters(rb_method_def(method));
3344method_inspect(
VALUE method)
3348 const char *sharp =
"#";
3350 VALUE defined_class;
3353 str = rb_sprintf(
"#<% "PRIsVALUE
": ",
rb_obj_class(method));
3355 mklass = data->iclass;
3356 if (!mklass) mklass = data->klass;
3366 if (data->me->def->type == VM_METHOD_TYPE_ALIAS) {
3367 defined_class = data->me->def->body.alias.original_me->owner;
3370 defined_class = method_entry_defined_class(data->me);
3377 if (UNDEF_P(data->recv)) {
3381 else if (RCLASS_SINGLETON_P(mklass)) {
3382 VALUE v = RCLASS_ATTACHED_OBJECT(mklass);
3384 if (UNDEF_P(data->recv)) {
3387 else if (data->recv == v) {
3400 mklass = data->klass;
3401 if (RCLASS_SINGLETON_P(mklass)) {
3402 VALUE v = RCLASS_ATTACHED_OBJECT(mklass);
3410 if (defined_class != mklass) {
3411 rb_str_catf(str,
"(% "PRIsVALUE
")", defined_class);
3416 if (data->me->called_id != data->me->def->original_id) {
3417 rb_str_catf(str,
"(%"PRIsVALUE
")",
3418 rb_id2str(data->me->def->original_id));
3420 if (data->me->def->type == VM_METHOD_TYPE_NOTIMPLEMENTED) {
3426 VALUE params = rb_method_parameters(method);
3427 VALUE pair, name, kind;
3433 const VALUE keyrest =
ID2SYM(rb_intern(
"keyrest"));
3436 const VALUE noblock =
ID2SYM(rb_intern(
"noblock"));
3451 for (
int i = 0; i <
RARRAY_LEN(params); i++) {
3459 if (kind == req || kind == opt) {
3462 else if (kind == rest || kind == keyrest) {
3465 else if (kind == block) {
3468 else if (kind == nokey) {
3471 else if (kind == noblock) {
3480 rb_str_catf(str,
"%"PRIsVALUE, name);
3482 else if (kind == opt) {
3483 rb_str_catf(str,
"%"PRIsVALUE
"=...", name);
3485 else if (kind == keyreq) {
3486 rb_str_catf(str,
"%"PRIsVALUE
":", name);
3488 else if (kind == key) {
3489 rb_str_catf(str,
"%"PRIsVALUE
": ...", name);
3491 else if (kind == rest) {
3492 if (name ==
ID2SYM(
'*')) {
3496 rb_str_catf(str,
"*%"PRIsVALUE, name);
3499 else if (kind == keyrest) {
3500 if (name !=
ID2SYM(idPow)) {
3501 rb_str_catf(str,
"**%"PRIsVALUE, name);
3510 else if (kind == block) {
3511 if (name ==
ID2SYM(
'&')) {
3520 rb_str_catf(str,
"&%"PRIsVALUE, name);
3523 else if (kind == nokey) {
3526 else if (kind == noblock) {
3538 VALUE loc = rb_method_location(method);
3540 rb_str_catf(str,
" %"PRIsVALUE
":%"PRIsVALUE,
3573method_to_proc(
VALUE method)
3587 procval =
rb_block_call(rb_mRubyVMFrozenCore, idLambda, 0, 0, bmcall, method);
3588 GetProcPtr(procval, proc);
3589 proc->is_from_method = 1;
3593extern VALUE rb_find_defined_class_by_owner(
VALUE current_class,
VALUE target_owner);
3604method_super_method(
VALUE method)
3606 const struct METHOD *data;
3607 VALUE super_class, iclass;
3612 iclass = data->iclass;
3613 if (!iclass)
return Qnil;
3614 if (data->me->def->type == VM_METHOD_TYPE_ALIAS && data->me->defined_class) {
3615 super_class =
RCLASS_SUPER(rb_find_defined_class_by_owner(data->me->defined_class,
3616 data->me->def->body.alias.original_me->owner));
3617 mid = data->me->def->body.alias.original_me->def->original_id;
3621 mid = data->me->def->original_id;
3623 if (!super_class)
return Qnil;
3624 me = (
rb_method_entry_t *)rb_callable_method_entry_with_refinements(super_class, mid, &iclass);
3625 if (!me)
return Qnil;
3626 return mnew_internal(me, me->owner, iclass, data->recv, mid,
rb_obj_class(method), FALSE, FALSE);
3636localjump_xvalue(
VALUE exc)
3650localjump_reason(
VALUE exc)
3655rb_cref_t *rb_vm_cref_new_toplevel(
void);
3664 VM_ASSERT(env->ep > env->env);
3665 VM_ASSERT(VM_ENV_ESCAPED_P(env->ep));
3668 cref = rb_vm_cref_new_toplevel();
3672 new_ep = &new_body[env->ep - env->env];
3673 new_env = vm_env_new(new_ep, new_body, env->env_size, env->iseq);
3680 new_ep[VM_ENV_DATA_INDEX_ENV] = (
VALUE)new_env;
3682 VM_ASSERT(VM_ENV_ESCAPED_P(new_ep));
3700proc_binding(
VALUE self)
3709 GetProcPtr(self, proc);
3710 block = &proc->block;
3712 if (proc->is_isolated) rb_raise(rb_eArgError,
"Can't create Binding from isolated Proc");
3715 switch (vm_block_type(block)) {
3716 case block_type_iseq:
3717 iseq = block->as.captured.code.iseq;
3718 binding_self = block->as.captured.self;
3719 env = VM_ENV_ENVVAL_PTR(block->as.captured.ep);
3721 case block_type_proc:
3722 GetProcPtr(block->as.proc, proc);
3723 block = &proc->block;
3725 case block_type_ifunc:
3727 const struct vm_ifunc *ifunc = block->as.captured.code.ifunc;
3728 if (IS_METHOD_PROC_IFUNC(ifunc)) {
3730 VALUE name = rb_fstring_lit(
"<empty_iseq>");
3732 binding_self = method_receiver(method);
3733 iseq = rb_method_iseq(method);
3734 env = VM_ENV_ENVVAL_PTR(block->as.captured.ep);
3735 env = env_clone(env, method_cref(method));
3737 empty = rb_iseq_new(
Qnil, name, name,
Qnil, 0, ISEQ_TYPE_TOP);
3743 case block_type_symbol:
3744 rb_raise(rb_eArgError,
"Can't create Binding from C level Proc");
3749 GetBindingPtr(bindval, bind);
3750 RB_OBJ_WRITE(bindval, &bind->block.as.captured.self, binding_self);
3751 RB_OBJ_WRITE(bindval, &bind->block.as.captured.code.iseq, env->iseq);
3752 rb_vm_block_ep_update(bindval, &bind->block, env->ep);
3756 rb_iseq_check(iseq);
3757 RB_OBJ_WRITE(bindval, &bind->pathobj, ISEQ_BODY(iseq)->location.pathobj);
3758 bind->first_lineno = ISEQ_BODY(iseq)->location.first_lineno;
3762 rb_iseq_pathobj_new(rb_fstring_lit(
"(binding)"),
Qnil));
3763 bind->first_lineno = 1;
3778 GetProcPtr(proc, procp);
3779 is_lambda = procp->is_lambda;
3783 GetProcPtr(proc, procp);
3784 procp->is_lambda = is_lambda;
3791 VALUE proc, passed, arity;
3800 if (!
NIL_P(blockarg)) {
3801 rb_warn(
"given block not used");
3803 arity = make_curry_proc(proc, passed, arity);
3858proc_curry(
int argc,
const VALUE *argv,
VALUE self)
3860 int sarity, max_arity, min_arity = rb_proc_min_max_arity(self, &max_arity);
3873 return make_curry_proc(self,
rb_ary_new(), arity);
3909rb_method_curry(
int argc,
const VALUE *argv,
VALUE self)
3911 VALUE proc = method_to_proc(self);
3912 return proc_curry(argc, argv, proc);
3930 return rb_funcallv(f, idCall, 1, &fargs);
3941 mesg = rb_fstring_lit(
"callable object is expected");
3965 return rb_proc_compose_to_left(self, to_callable(g));
3971 VALUE proc, args, procs[2];
3977 args = rb_ary_tmp_new_from_values(0, 2, procs);
3980 GetProcPtr(g, procp);
3981 is_lambda = procp->is_lambda;
3989 GetProcPtr(proc, procp);
3990 procp->is_lambda = is_lambda;
4024 return rb_proc_compose_to_right(self, to_callable(g));
4030 VALUE proc, args, procs[2];
4036 args = rb_ary_tmp_new_from_values(0, 2, procs);
4038 GetProcPtr(self, procp);
4039 is_lambda = procp->is_lambda;
4042 GetProcPtr(proc, procp);
4043 procp->is_lambda = is_lambda;
4064rb_method_compose_to_left(
VALUE self,
VALUE g)
4067 self = method_to_proc(self);
4068 return proc_compose_to_left(self, g);
4087rb_method_compose_to_right(
VALUE self,
VALUE g)
4090 self = method_to_proc(self);
4091 return proc_compose_to_right(self, g);
4127proc_ruby2_keywords(
VALUE procval)
4130 GetProcPtr(procval, proc);
4132 rb_check_frozen(procval);
4134 if (proc->is_from_method) {
4135 rb_warn(
"Skipping set of ruby2_keywords flag for proc (proc created from method)");
4139 switch (proc->block.type) {
4140 case block_type_iseq:
4141 if (ISEQ_BODY(proc->block.as.captured.code.iseq)->param.flags.has_rest &&
4142 !ISEQ_BODY(proc->block.as.captured.code.iseq)->param.flags.has_post &&
4143 !ISEQ_BODY(proc->block.as.captured.code.iseq)->param.flags.has_kw &&
4144 !ISEQ_BODY(proc->block.as.captured.code.iseq)->param.flags.has_kwrest) {
4145 ISEQ_BODY(proc->block.as.captured.code.iseq)->param.flags.ruby2_keywords = 1;
4148 rb_warn(
"Skipping set of ruby2_keywords flag for proc (proc accepts keywords or post arguments or proc does not accept argument splat)");
4152 rb_warn(
"Skipping set of ruby2_keywords flag for proc (proc not defined in Ruby)");
4574 rb_add_method_optimized(
rb_cProc, idCall, OPTIMIZED_METHOD_TYPE_CALL, 0, METHOD_VISI_PUBLIC);
4575 rb_add_method_optimized(
rb_cProc, rb_intern(
"[]"), OPTIMIZED_METHOD_TYPE_CALL, 0, METHOD_VISI_PUBLIC);
4576 rb_add_method_optimized(
rb_cProc, rb_intern(
"==="), OPTIMIZED_METHOD_TYPE_CALL, 0, METHOD_VISI_PUBLIC);
4577 rb_add_method_optimized(
rb_cProc, rb_intern(
"yield"), OPTIMIZED_METHOD_TYPE_CALL, 0, METHOD_VISI_PUBLIC);
4611 rb_vm_register_special_exception(ruby_error_sysstack,
rb_eSysStackError,
"stack level too deep");
4680 "define_method", top_define_method, -1);
4721 rb_gc_register_address(&sym_proc_cache);
#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.
#define rb_define_method(klass, mid, func, arity)
Defines klass#mid.
#define rb_define_singleton_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.
VALUE rb_define_class(const char *name, VALUE super)
Defines a top-level class.
VALUE rb_singleton_class(VALUE obj)
Finds or creates the singleton class of the passed object.
VALUE rb_singleton_class_get(VALUE obj)
Returns the singleton class of obj, or nil if obj is not a singleton object.
void rb_define_alias(VALUE klass, const char *name1, const char *name2)
Defines an alias of a method.
void rb_undef_method(VALUE klass, const char *name)
Defines an undef of a method.
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_block_given_p(void)
Determines if the current method is given a block.
int rb_get_kwargs(VALUE keyword_hash, const ID *table, int required, int optional, VALUE *values)
Keyword argument deconstructor.
#define rb_str_new2
Old name of rb_str_new_cstr.
#define rb_str_buf_cat2
Old name of rb_usascii_str_new_cstr.
#define Qundef
Old name of RUBY_Qundef.
#define INT2FIX
Old name of RB_INT2FIX.
#define ID2SYM
Old name of RB_ID2SYM.
#define OBJ_FREEZE
Old name of RB_OBJ_FREEZE.
#define UNREACHABLE_RETURN
Old name of RBIMPL_UNREACHABLE_RETURN.
#define SYM2ID
Old name of RB_SYM2ID.
#define ZALLOC
Old name of RB_ZALLOC.
#define CLASS_OF
Old name of rb_class_of.
#define rb_ary_new4
Old name of rb_ary_new_from_values.
#define FIX2INT
Old name of RB_FIX2INT.
#define T_MODULE
Old name of RUBY_T_MODULE.
#define ASSUME
Old name of RBIMPL_ASSUME.
#define T_ICLASS
Old name of RUBY_T_ICLASS.
#define ALLOC_N
Old name of RB_ALLOC_N.
#define rb_ary_new3
Old name of rb_ary_new_from_args.
#define Qtrue
Old name of RUBY_Qtrue.
#define ST2FIX
Old name of RB_ST2FIX.
#define Qnil
Old name of RUBY_Qnil.
#define Qfalse
Old name of RUBY_Qfalse.
#define NIL_P
Old name of RB_NIL_P.
#define T_CLASS
Old name of RUBY_T_CLASS.
#define BUILTIN_TYPE
Old name of RB_BUILTIN_TYPE.
#define Check_TypedStruct(v, t)
Old name of rb_check_typeddata.
#define CONST_ID
Old name of RUBY_CONST_ID.
#define rb_ary_new2
Old name of rb_ary_new_capa.
VALUE rb_eLocalJumpError
LocalJumpError exception.
void rb_exc_raise(VALUE mesg)
Raises an exception in the current thread.
VALUE rb_eStandardError
StandardError exception.
VALUE rb_eRangeError
RangeError exception.
VALUE rb_eTypeError
TypeError exception.
void rb_warn(const char *fmt,...)
Identical to rb_warning(), except it reports unless $VERBOSE is nil.
VALUE rb_exc_new_str(VALUE etype, VALUE str)
Identical to rb_exc_new_cstr(), except it takes a Ruby's string instead of C's.
VALUE rb_eException
Mother of all exceptions.
VALUE rb_eSysStackError
SystemStackError exception.
VALUE rb_class_superclass(VALUE klass)
Queries the parent of the given class.
VALUE rb_cUnboundMethod
UnboundMethod class.
VALUE rb_mKernel
Kernel module.
VALUE rb_cObject
Object class.
VALUE rb_cBinding
Binding class.
VALUE rb_obj_class(VALUE obj)
Queries the class of an object.
VALUE rb_inspect(VALUE obj)
Generates a human-readable textual representation of the given object.
VALUE rb_cModule
Module class.
VALUE rb_class_inherited_p(VALUE scion, VALUE ascendant)
Determines if the given two modules are relatives.
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_cProc
Proc class.
VALUE rb_cMethod
Method class.
#define RB_OBJ_WRITTEN(old, oldv, young)
Identical to RB_OBJ_WRITE(), except it doesn't write any values, but only a WB declaration.
#define RB_OBJ_WRITE(old, slot, young)
Declaration of a "back" pointer.
VALUE rb_funcall(VALUE recv, ID mid, int n,...)
Calls a method.
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_ary_new_from_values(long n, const VALUE *elts)
Identical to rb_ary_new_from_args(), except how objects are passed.
VALUE rb_ary_dup(VALUE ary)
Duplicates an array.
VALUE rb_ary_plus(VALUE lhs, VALUE rhs)
Creates a new array, concatenating the former to the latter.
VALUE rb_ary_new(void)
Allocates a new, empty array.
VALUE rb_ary_hidden_new(long capa)
Allocates a hidden (no class) empty array.
VALUE rb_ary_push(VALUE ary, VALUE elem)
Special case of rb_ary_cat() that it adds only one element.
VALUE rb_ary_freeze(VALUE obj)
Freeze an array, preventing further modifications.
void rb_ary_store(VALUE ary, long key, VALUE val)
Destructively stores the passed value to the passed array's passed index.
#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_method_call_with_block(int argc, const VALUE *argv, VALUE recv, VALUE proc)
Identical to rb_proc_call(), except you can additionally pass a proc as a block.
int rb_obj_method_arity(VALUE obj, ID mid)
Identical to rb_mod_method_arity(), except it searches for singleton methods rather than instance met...
VALUE rb_proc_call(VALUE recv, VALUE args)
Evaluates the passed proc with the passed arguments.
VALUE rb_proc_call_with_block_kw(VALUE recv, int argc, const VALUE *argv, VALUE proc, int kw_splat)
Identical to rb_proc_call_with_block(), except you can specify how to handle the last element of the ...
VALUE rb_method_call_kw(int argc, const VALUE *argv, VALUE recv, int kw_splat)
Identical to rb_method_call(), except you can specify how to handle the last element of the given arr...
VALUE rb_obj_method(VALUE recv, VALUE mid)
Creates a method object.
VALUE rb_proc_lambda_p(VALUE recv)
Queries if the given object is a lambda.
VALUE rb_block_proc(void)
Constructs a Proc object from implicitly passed components.
VALUE rb_proc_call_with_block(VALUE recv, int argc, const VALUE *argv, VALUE proc)
Identical to rb_proc_call(), except you can additionally pass another proc object,...
int rb_mod_method_arity(VALUE mod, ID mid)
Queries the number of mandatory arguments of the method defined in the given module.
VALUE rb_method_call_with_block_kw(int argc, const VALUE *argv, VALUE recv, VALUE proc, int kw_splat)
Identical to rb_method_call_with_block(), except you can specify how to handle the last element of th...
VALUE rb_obj_is_method(VALUE recv)
Queries if the given object is a method.
VALUE rb_block_lambda(void)
Identical to rb_proc_new(), except it returns a lambda.
VALUE rb_proc_call_kw(VALUE recv, VALUE args, int kw_splat)
Identical to rb_proc_call(), except you can specify how to handle the last element of the given array...
VALUE rb_binding_new(void)
Snapshots the current execution context and turn it into an instance of rb_cBinding.
int rb_proc_arity(VALUE recv)
Queries the number of mandatory arguments of the given Proc.
VALUE rb_method_call(int argc, const VALUE *argv, VALUE recv)
Evaluates the passed method with the passed arguments.
VALUE rb_obj_is_proc(VALUE recv)
Queries if the given object is a proc.
#define rb_hash_uint(h, i)
Just another name of st_hash_uint.
#define rb_hash_end(h)
Just another name of st_hash_end.
VALUE rb_str_append(VALUE dst, VALUE src)
Identical to rb_str_buf_append(), except it converts the right hand side before concatenating.
VALUE rb_str_buf_append(VALUE dst, VALUE src)
Identical to rb_str_cat_cstr(), except it takes Ruby's string instead of C's.
void rb_str_set_len(VALUE str, long len)
Overwrites the length of the string.
st_index_t rb_hash_start(st_index_t i)
Starts a series of hashing.
#define rb_str_cat_cstr(buf, str)
Identical to rb_str_cat(), 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.
void rb_undef_alloc_func(VALUE klass)
Deletes the allocator function of a class.
int rb_obj_respond_to(VALUE obj, ID mid, int private_p)
Identical to rb_respond_to(), except it additionally takes the visibility parameter.
ID rb_check_id(volatile VALUE *namep)
Detects if the given name is already interned or not.
ID rb_to_id(VALUE str)
Identical to rb_intern_str(), except it tries to convert the parameter object to an instance of rb_cS...
VALUE rb_iv_get(VALUE obj, const char *name)
Obtains an instance variable.
#define RB_INT2NUM
Just another name of rb_int2num_inline.
#define RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg)
Shim for block function parameters.
rb_block_call_func * rb_block_call_func_t
Shorthand type that represents an iterator-written-in-C function pointer.
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.
#define MEMCPY(p1, p2, type, n)
Handy macro to call memcpy.
#define RB_GC_GUARD(v)
Prevents premature destruction of local objects.
VALUE rb_block_call(VALUE q, ID w, int e, const VALUE *r, type *t, VALUE y)
Call a method with a block.
VALUE rb_proc_new(type *q, VALUE w)
Creates a rb_cProc instance.
VALUE rb_rescue(type *q, VALUE w, type *e, VALUE r)
An equivalent of rescue clause.
#define RARRAY_LEN
Just another name of rb_array_len.
static void RARRAY_ASET(VALUE ary, long i, VALUE v)
Assigns an object in an array.
#define RARRAY_AREF(a, i)
#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 RCLASS_SUPER
Just another name of rb_class_get_superclass.
#define RUBY_TYPED_DEFAULT_FREE
This is a value you can set to rb_data_type_struct::dfree.
#define RUBY_TYPED_FREE_IMMEDIATELY
Macros to see if each corresponding flag is defined.
#define TypedData_Get_Struct(obj, type, data_type, sval)
Obtains a C struct from inside of a wrapper Ruby object.
#define TypedData_Make_Struct(klass, type, data_type, sval)
Identical to TypedData_Wrap_Struct, except it allocates a new data region internally instead of takin...
const char * rb_obj_classname(VALUE obj)
Queries the name of the class of the passed object.
#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.
Internal header for Ruby Box.
This is the struct that holds necessary info for a struct.
rb_cref_t * cref
class reference, should be marked
const rb_iseq_t * iseqptr
iseq pointer, should be separated from iseqval
IFUNC (Internal FUNCtion)
uintptr_t ID
Type that represents a Ruby identifier such as a variable name.
#define SIZEOF_VALUE
Identical to sizeof(VALUE), except it is a macro that can also be used inside of preprocessor directi...
uintptr_t VALUE
Type that represents a Ruby object.
static bool RB_TYPE_P(VALUE obj, enum ruby_value_type t)
Queries if the given object is of given type.