14#include "ruby/internal/config.h"
17#include "ccan/list/list.h"
19#include "debug_counter.h"
23#include "internal/box.h"
24#include "internal/class.h"
25#include "internal/compilers.h"
26#include "internal/error.h"
27#include "internal/eval.h"
28#include "internal/hash.h"
29#include "internal/object.h"
30#include "internal/gc.h"
31#include "internal/re.h"
32#include "internal/struct.h"
33#include "internal/symbol.h"
34#include "internal/thread.h"
35#include "internal/variable.h"
43#include "ractor_core.h"
47#define GET_GLOBAL_CVAR_STATE() (ruby_vm_global_cvar_state)
49typedef void rb_gvar_compact_t(
void *var);
57static VALUE autoload_features;
62static VALUE autoload_mutex;
64static void check_before_mod_set(
VALUE,
ID,
VALUE,
const char *);
65static void setup_const_entry(rb_const_entry_t *,
VALUE,
VALUE, rb_const_flag_t);
66static VALUE rb_const_search(
VALUE klass,
ID id,
int exclude,
int recurse,
int visibility,
VALUE *found_in);
67static st_table *generic_fields_tbl_;
69typedef int rb_ivar_foreach_callback_func(
ID key,
VALUE val, st_data_t arg);
70static void rb_field_foreach(
VALUE obj, rb_ivar_foreach_callback_func *func, st_data_t arg,
bool ivar_only);
75 rb_global_tbl = rb_id_table_create(0);
76 generic_fields_tbl_ = st_init_numtable();
81 rb_vm_register_global_object(autoload_mutex);
83 autoload_features = rb_ident_hash_new();
85 rb_vm_register_global_object(autoload_features);
89rb_namespace_p(
VALUE obj)
110classname(
VALUE klass,
bool *permanent)
114 VALUE classpath = RCLASS_CLASSPATH(klass);
115 if (classpath == 0)
return Qnil;
117 *permanent = RCLASS_PERMANENT_CLASSPATH_P(klass);
123rb_mod_name0(
VALUE klass,
bool *permanent)
125 return classname(klass, permanent);
140 return classname(mod, &permanent);
145is_constant_path(
VALUE name)
147 const char *path = RSTRING_PTR(name);
149 rb_encoding *enc = rb_enc_get(name);
151 const char *p = path;
153 if (p >= pend || !*p) {
158 if (p + 2 <= pend && p[0] ==
':' && p[1] ==
':') {
162 const char *pbeg = p;
163 while (p < pend && *p !=
':') p++;
165 if (pbeg == p)
return false;
167 if (rb_enc_symname_type(pbeg, p - pbeg, enc, 0) != ID_CONST) {
184set_sub_temporary_name_recursive(
VALUE mod,
VALUE data,
int recursive)
186 if (recursive)
return Qfalse;
191 name = build_const_path(rb_ary_last(0, 0, args->names), args->last);
193 set_sub_temporary_name_foreach(mod, args, name);
198set_sub_temporary_name_topmost(
VALUE mod,
VALUE data,
int recursive)
200 if (recursive)
return Qfalse;
203 VALUE name = args->names;
207 set_sub_temporary_name_foreach(mod, args, name);
211static enum rb_id_table_iterator_result
212set_sub_temporary_name_i(
ID id,
VALUE val,
void *data)
214 val = ((rb_const_entry_t *)val)->value;
215 if (rb_namespace_p(val) && !RCLASS_PERMANENT_CLASSPATH_P(val)) {
221 return ID_TABLE_CONTINUE;
227 RCLASS_WRITE_CLASSPATH(mod, name, FALSE);
231 rb_id_table_foreach(tbl, set_sub_temporary_name_i, args);
236 rb_id_table_foreach(tbl, set_sub_temporary_name_i, args);
237 rb_ary_set_len(args->names, names_len);
298rb_mod_set_temporary_name(
VALUE mod,
VALUE name)
301 if (RCLASS_PERMANENT_CLASSPATH_P(mod)) {
308 set_sub_temporary_name(mod, 0);
315 if (RSTRING_LEN(name) == 0) {
316 rb_raise(rb_eArgError,
"empty class/module name");
319 if (is_constant_path(name)) {
320 rb_raise(rb_eArgError,
"the temporary name must not be a constant path to avoid confusion");
324 RB_OBJ_SET_SHAREABLE(name);
328 set_sub_temporary_name(mod, name);
341 path = rb_sprintf(
"#<Class:%p>", (
void*)obj);
344 path = rb_sprintf(
"#<Module:%p>", (
void*)obj);
347 path = rb_sprintf(
"#<%"PRIsVALUE
":%p>", klass, (
void*)obj);
357rb_tmp_class_path(
VALUE klass,
bool *permanent, fallback_func fallback)
359 VALUE path = classname(klass, permanent);
371 path = rb_tmp_class_path(
RBASIC(klass)->klass, &perm, fallback);
376 return fallback(klass, path);
383 VALUE path = rb_tmp_class_path(klass, &permanent, make_temporary_path);
401rb_search_class_path(
VALUE klass)
404 return rb_tmp_class_path(klass, &permanent, no_fallback);
413 return rb_fstring(path);
417build_const_path(
VALUE head,
ID tail)
419 return build_const_pathname(head, rb_id2str(tail));
425 bool permanent =
true;
428 if (under == rb_cObject) {
432 str = rb_tmp_class_path(under, &permanent, make_temporary_path);
433 str = build_const_pathname(str, name);
436 RB_OBJ_SET_SHAREABLE(str);
437 RCLASS_SET_CLASSPATH(klass, str, permanent);
451 rb_encoding *enc = rb_enc_get(pathname);
452 const char *pbeg, *pend, *p, *path = RSTRING_PTR(pathname);
454 VALUE c = rb_cObject;
456 if (!rb_enc_asciicompat(enc)) {
457 rb_raise(rb_eArgError,
"invalid class path encoding (non ASCII)");
460 pend = path + RSTRING_LEN(pathname);
461 if (path == pend || path[0] ==
'#') {
462 rb_raise(rb_eArgError,
"can't retrieve anonymous class %"PRIsVALUE,
466 while (p < pend && *p !=
':') p++;
468 if (p < pend && p[0] ==
':') {
469 if ((
size_t)(pend - p) < 2 || p[1] !=
':')
goto undefined_class;
474 goto undefined_class;
476 c = rb_const_search(c,
id, TRUE, FALSE, FALSE, NULL);
477 if (UNDEF_P(c))
goto undefined_class;
478 if (!rb_namespace_p(c)) {
479 rb_raise(
rb_eTypeError,
"%"PRIsVALUE
" does not refer to class/module",
488 rb_raise(rb_eArgError,
"undefined class/module % "PRIsVALUE,
509 VALUE path = rb_tmp_class_path(
rb_class_real(klass), &permanent, make_temporary_path);
510 if (
NIL_P(path))
return NULL;
511 return RSTRING_PTR(path);
534 rb_gvar_compact_t *compactor;
560static enum rb_id_table_iterator_result
561free_global_entry_i(
VALUE val,
void *arg)
564 entry->var->counter--;
565 if (entry->var->counter == 0) {
566 free_global_variable(entry->var);
569 return ID_TABLE_DELETE;
573rb_free_rb_global_tbl(
void)
575 rb_id_table_foreach_values(rb_global_tbl, free_global_entry_i, 0);
576 rb_id_table_free(rb_global_tbl);
580rb_free_generic_fields_tbl_(
void)
582 st_free_table(generic_fields_tbl_);
586rb_find_global_entry(
ID id)
592 if (!rb_id_table_lookup(rb_global_tbl,
id, &data)) {
601 if (UNLIKELY(!rb_ractor_main_p()) && (!entry || !entry->ractor_local)) {
602 rb_raise(rb_eRactorIsolationError,
"can not access global variable %s from non-main Ractor", rb_id2name(
id));
609rb_gvar_ractor_local(
const char *name)
612 entry->ractor_local =
true;
616rb_gvar_box_ready(
const char *name)
619 entry->var->box_ready =
true;
623rb_gvar_box_dynamic(
const char *name)
626 entry->var->box_dynamic =
true;
630rb_gvar_undef_compactor(
void *var)
639 entry = rb_find_global_entry(
id);
646 entry->ractor_local =
false;
652 var->compactor = rb_gvar_undef_compactor;
654 var->block_trace = 0;
656 var->box_ready =
false;
657 var->box_dynamic =
false;
658 rb_id_table_insert(rb_global_tbl,
id, (
VALUE)entry);
667 rb_warning(
"global variable '%"PRIsVALUE
"' not initialized", QUOTE_ID(
id));
673rb_gvar_val_compactor(
void *_var)
680 VALUE new = rb_gc_location(obj);
682 var->data = (
void*)
new;
694 var->compactor = rb_gvar_val_compactor;
696 var->data = (
void*)val;
714 var->data = (
void*)val;
721 if (data) rb_gc_mark_movable(data);
727 if (!var)
return Qnil;
740 if (var) rb_gc_mark_maybe(*var);
746 rb_name_error(
id,
"%"PRIsVALUE
" is a read-only variable", QUOTE_ID(
id));
749static enum rb_id_table_iterator_result
750mark_global_entry(
VALUE v,
void *ignored)
756 (*var->marker)(var->data);
759 if (trace->data) rb_gc_mark_maybe(trace->data);
762 return ID_TABLE_CONTINUE;
765#define gc_mark_table(task) \
766 if (rb_global_tbl) { rb_id_table_foreach_values(rb_global_tbl, task##_global_entry, 0); }
769rb_gc_mark_global_tbl(
void)
774static enum rb_id_table_iterator_result
775update_global_entry(
VALUE v,
void *ignored)
780 (*var->compactor)(var);
781 return ID_TABLE_CONTINUE;
785rb_gc_update_global_tbl(
void)
787 gc_mark_table(update);
791global_id(
const char *name)
795 if (name[0] ==
'$')
id = rb_intern(name);
797 size_t len = strlen(name);
801 memcpy(buf+1, name,
len);
802 id = rb_intern2(buf,
len+1);
809find_global_id(
const char *name)
812 size_t len = strlen(name);
814 if (name[0] ==
'$') {
821 memcpy(buf+1, name,
len);
837 ID id = global_id(name);
840 gvar->data = (
void*)var;
892 trace->next = entry->var->trace;
893 trace->func = rb_trace_eval;
896 entry->var->trace = trace;
910 while (trace->next) {
913 trace->next = next->next;
936 if ((entry = rb_find_global_entry(
id)) == NULL) {
937 rb_name_error(
id,
"undefined global variable %"PRIsVALUE
"", QUOTE_ID(
id));
940 trace = entry->var->trace;
951 if (!entry->var->block_trace) remove_trace(entry->var);
956 if (trace->data == cmd) {
958 if (!entry->var->block_trace) remove_trace(entry->var);
979 (*trace->func)(trace->data, data->val);
990 var->block_trace = 0;
1001 (*var->setter)(val, entry->id, var->data);
1003 if (var->trace && !var->block_trace) {
1004 var->block_trace = 1;
1005 trace.trace = var->trace;
1013gvar_use_box_tbl(
const rb_box_t *box,
const struct rb_global_entry *entry)
1015 return BOX_USER_P(box) &&
1016 !entry->var->box_dynamic &&
1025 const rb_box_t *box = rb_current_box();
1026 bool use_box_tbl =
false;
1031 if (gvar_use_box_tbl(box, entry)) {
1033 rb_hash_aset(box->gvar_tbl,
rb_id2sym(entry->id), val);
1040 retval = rb_gvar_set_entry(entry, val);
1048 return rb_gvar_set(global_id(name), val);
1054 VALUE retval, gvars, key;
1055 const rb_box_t *box = rb_current_box();
1056 bool use_box_tbl =
false;
1064 if (gvar_use_box_tbl(box, entry)) {
1066 gvars = box->gvar_tbl;
1068 if (
RTEST(rb_hash_has_key(gvars, key))) {
1069 retval = rb_hash_aref(gvars, key);
1074 retval = (*var->getter)(entry->id, var->data);
1076 retval =
rb_funcall(retval, rb_intern(
"clone"), 0);
1080 rb_hash_aset(gvars, key, retval);
1085 retval = (*var->getter)(entry->id, var->data);
1093 ID id = find_global_id(name);
1096 rb_warning(
"global variable '%s' not initialized", name);
1100 return rb_gvar_get(
id);
1104rb_gvar_defined(
ID id)
1111rb_gvar_getter_function_of(
ID id)
1114 return entry->var->getter;
1118rb_gvar_setter_function_of(
ID id)
1121 return entry->var->setter;
1124static enum rb_id_table_iterator_result
1125gvar_i(
ID key,
VALUE val,
void *a)
1129 return ID_TABLE_CONTINUE;
1138 if (!rb_ractor_main_p()) {
1139 rb_raise(rb_eRactorIsolationError,
"can not access global variables from non-main Ractors");
1143 rb_id_table_foreach(rb_global_tbl, gvar_i, (
void *)ary);
1144 if (!
NIL_P(backref)) {
1146 int i, nmatch = rb_match_count(backref);
1148 for (i = 1; i <= nmatch; ++i) {
1152 buf[1] = (char)(i +
'0');
1153 sym =
ID2SYM(rb_intern2(buf, 2));
1172 if (!rb_ractor_main_p()) {
1173 rb_raise(rb_eRactorIsolationError,
"can not access global variables from non-main Ractors");
1178 if (!rb_id_table_lookup(gtbl, name1, &data1)) {
1181 rb_id_table_insert(gtbl, name1, (
VALUE)entry1);
1183 else if ((entry1 = (
struct rb_global_entry *)data1)->var != entry2->var) {
1185 if (var->block_trace) {
1190 if (var->counter == 0) {
1191 free_global_variable(var);
1194 if (entry1->var != entry2->var) {
1195 entry2->var->counter++;
1196 entry1->var = entry2->var;
1202IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(
ID id)
1204 if (UNLIKELY(!rb_ractor_main_p())) {
1206 rb_raise(rb_eRactorIsolationError,
"can not set instance variables of classes/modules by non-main Ractors");
1212CVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(
VALUE klass,
ID id)
1214 if (UNLIKELY(!rb_ractor_main_p())) {
1215 rb_raise(rb_eRactorIsolationError,
"can not access class variables from non-main Ractors (%"PRIsVALUE
" from %"PRIsVALUE
")", rb_id2str(
id), klass);
1220ivar_ractor_check(
VALUE obj,
ID id)
1224 UNLIKELY(!rb_ractor_main_p()) &&
1227 rb_raise(rb_eRactorIsolationError,
"can not access instance variables of shareable objects from non-main Ractors");
1231static inline struct st_table *
1232generic_fields_tbl_no_ractor_check(
void)
1234 ASSERT_vm_locking();
1236 return generic_fields_tbl_;
1240rb_generic_fields_tbl_get(
void)
1242 return generic_fields_tbl_;
1246rb_mark_generic_ivar(
VALUE obj)
1250 if (st_lookup(generic_fields_tbl_, (st_data_t)obj, (st_data_t *)&data)) {
1251 rb_gc_mark_movable(data);
1256rb_obj_fields_generic_uncached(
VALUE obj)
1258 VALUE fields_obj = 0;
1260 if (!st_lookup(generic_fields_tbl_, (st_data_t)obj, (st_data_t *)&fields_obj)) {
1261 rb_bug(
"Object is missing entry in generic_fields_tbl");
1268rb_obj_fields(
VALUE obj,
ID field_name)
1271 ivar_ractor_check(obj, field_name);
1273 VALUE fields_obj = 0;
1274 if (rb_shape_obj_has_fields(obj)) {
1281 goto generic_fields;
1283 if (LIKELY(!
FL_TEST_RAW(obj, RSTRUCT_GEN_FIELDS))) {
1284 fields_obj = RSTRUCT_FIELDS_OBJ(obj);
1287 goto generic_fields;
1291 rb_execution_context_t *ec = GET_EC();
1292 if (ec->gen_fields_cache.obj == obj && !UNDEF_P(ec->gen_fields_cache.fields_obj) && rb_imemo_fields_owner(ec->gen_fields_cache.fields_obj) == obj) {
1293 fields_obj = ec->gen_fields_cache.fields_obj;
1294 RUBY_ASSERT(fields_obj == rb_obj_fields_generic_uncached(obj));
1297 fields_obj = rb_obj_fields_generic_uncached(obj);
1298 ec->gen_fields_cache.fields_obj = fields_obj;
1299 ec->gen_fields_cache.obj = obj;
1310 if (rb_obj_gen_fields_p(obj)) {
1311 st_data_t key = (st_data_t)obj, value;
1318 goto generic_fields;
1320 if (LIKELY(!
FL_TEST_RAW(obj, RSTRUCT_GEN_FIELDS))) {
1321 RSTRUCT_SET_FIELDS_OBJ(obj, 0);
1324 goto generic_fields;
1330 rb_execution_context_t *ec = GET_EC();
1331 if (ec->gen_fields_cache.obj == obj) {
1332 ec->gen_fields_cache.obj =
Qundef;
1333 ec->gen_fields_cache.fields_obj =
Qundef;
1336 if (!st_delete(generic_fields_tbl_no_ractor_check(), &key, &value)) {
1337 rb_bug(
"Object is missing entry in generic_fields_tbl");
1342 RBASIC_SET_SHAPE_ID(obj, ROOT_SHAPE_ID);
1347rb_obj_set_fields(
VALUE obj,
VALUE fields_obj,
ID field_name,
VALUE original_fields_obj)
1349 ivar_ractor_check(obj, field_name);
1354 rb_imemo_fields_clear(original_fields_obj);
1358 RUBY_ASSERT(IMEMO_TYPE_P(fields_obj, imemo_fields));
1359 RUBY_ASSERT(!original_fields_obj || IMEMO_TYPE_P(original_fields_obj, imemo_fields));
1361 if (fields_obj != original_fields_obj) {
1368 goto generic_fields;
1370 if (LIKELY(!
FL_TEST_RAW(obj, RSTRUCT_GEN_FIELDS))) {
1371 RSTRUCT_SET_FIELDS_OBJ(obj, fields_obj);
1374 goto generic_fields;
1379 st_insert(generic_fields_tbl_, (st_data_t)obj, (st_data_t)fields_obj);
1383 rb_execution_context_t *ec = GET_EC();
1384 if (ec->gen_fields_cache.fields_obj != fields_obj) {
1385 ec->gen_fields_cache.obj = obj;
1386 ec->gen_fields_cache.fields_obj = fields_obj;
1391 if (original_fields_obj) {
1393 rb_imemo_fields_clear(original_fields_obj);
1397 RBASIC_SET_SHAPE_ID(obj, RBASIC_SHAPE_ID(fields_obj));
1401rb_obj_replace_fields(
VALUE obj,
VALUE fields_obj)
1404 VALUE original_fields_obj = rb_obj_fields_no_ractor_check(obj);
1405 rb_obj_set_fields(obj, fields_obj, 0, original_fields_obj);
1410rb_obj_field_get(
VALUE obj, shape_id_t target_shape_id)
1413 RUBY_ASSERT(RSHAPE_TYPE_P(target_shape_id, SHAPE_IVAR) || RSHAPE_TYPE_P(target_shape_id, SHAPE_OBJ_ID));
1420 fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
1430 fields_obj = rb_obj_fields(obj, RSHAPE_EDGE_NAME(target_shape_id));
1434 if (UNLIKELY(rb_shape_too_complex_p(target_shape_id))) {
1435 st_table *fields_hash = rb_imemo_fields_complex_tbl(fields_obj);
1437 st_lookup(fields_hash, RSHAPE_EDGE_NAME(target_shape_id), &value);
1442 attr_index_t index = RSHAPE_INDEX(target_shape_id);
1443 return rb_imemo_fields_ptr(fields_obj)[index];
1457 VALUE val = rb_ivar_lookup(RCLASS_WRITABLE_FIELDS_OBJ(obj),
id, undef);
1460 UNLIKELY(!rb_ractor_main_p()) &&
1462 rb_raise(rb_eRactorIsolationError,
1463 "can not get unshareable values from instance variables of classes/modules from non-main Ractors (%"PRIsVALUE
" from %"PRIsVALUE
")",
1464 rb_id2str(
id), obj);
1477 fields_obj = rb_obj_fields(obj,
id);
1485 shape_id_t shape_id = RBASIC_SHAPE_ID(fields_obj);
1487 if (UNLIKELY(rb_shape_too_complex_p(shape_id))) {
1488 st_table *iv_table = rb_imemo_fields_complex_tbl(fields_obj);
1490 if (rb_st_lookup(iv_table, (st_data_t)
id, (st_data_t *)&val)) {
1496 attr_index_t index = 0;
1497 if (rb_shape_get_iv_index(shape_id,
id, &index)) {
1498 return rb_imemo_fields_ptr(fields_obj)[index];
1507 VALUE iv = rb_ivar_lookup(obj,
id,
Qnil);
1508 RB_DEBUG_COUNTER_INC(ivar_get_base);
1513rb_ivar_get_at(
VALUE obj, attr_index_t index,
ID id)
1524 VALUE fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
1525 VALUE val = rb_imemo_fields_ptr(fields_obj)[index];
1528 rb_raise(rb_eRactorIsolationError,
1529 "can not get unshareable values from instance variables of classes/modules from non-main Ractors");
1536 VALUE fields_obj = rb_obj_fields(obj,
id);
1537 return rb_imemo_fields_ptr(fields_obj)[index];
1543rb_ivar_get_at_no_ractor_check(
VALUE obj, attr_index_t index)
1553 fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
1556 fields_obj = rb_obj_fields_no_ractor_check(obj);
1559 return rb_imemo_fields_ptr(fields_obj)[index];
1565 return rb_ivar_lookup(obj,
id,
Qnil);
1568void rb_obj_copy_fields_to_hash_table(
VALUE obj, st_table *table);
1569static VALUE imemo_fields_complex_from_obj(
VALUE owner,
VALUE source_fields_obj, shape_id_t shape_id);
1572obj_transition_too_complex(
VALUE obj, st_table *table)
1575 shape_id_t shape_id = rb_shape_transition_complex(obj);
1580 VALUE *old_fields = NULL;
1587 RBASIC_SET_SHAPE_ID(obj, shape_id);
1588 ROBJECT_SET_FIELDS_HASH(obj, table);
1602 RBASIC_SET_SHAPE_ID(fields_obj, shape_id);
1603 rb_obj_replace_fields(obj, fields_obj);
1612rb_evict_fields_to_hash(
VALUE obj)
1616 st_table *table = st_init_numtable_with_size(RSHAPE_LEN(RBASIC_SHAPE_ID(obj)));
1617 rb_obj_copy_fields_to_hash_table(obj, table);
1618 shape_id_t new_shape_id = obj_transition_too_complex(obj, table);
1621 return new_shape_id;
1625rb_evict_ivars_to_hash(
VALUE obj)
1629 st_table *table = st_init_numtable_with_size(
rb_ivar_count(obj));
1632 rb_obj_copy_ivs_to_hash_table(obj, table);
1633 obj_transition_too_complex(obj, table);
1641 rb_check_frozen(obj);
1645 bool concurrent =
false;
1651 IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(
id);
1653 fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
1654 if (rb_multi_ractor_p()) {
1662 fields_obj = rb_obj_fields(obj,
id);
1671 const VALUE original_fields_obj = fields_obj;
1673 fields_obj = rb_imemo_fields_clone(fields_obj);
1676 shape_id_t old_shape_id = RBASIC_SHAPE_ID(fields_obj);
1677 shape_id_t removed_shape_id;
1678 shape_id_t next_shape_id = rb_shape_transition_remove_ivar(fields_obj,
id, &removed_shape_id);
1680 if (UNLIKELY(rb_shape_too_complex_p(next_shape_id))) {
1681 if (UNLIKELY(!rb_shape_too_complex_p(old_shape_id))) {
1683 rb_evict_fields_to_hash(obj);
1686 fields_obj = imemo_fields_complex_from_obj(obj, fields_obj, next_shape_id);
1690 if (!st_delete(rb_imemo_fields_complex_tbl(fields_obj), &key, (st_data_t *)&val)) {
1695 if (next_shape_id == old_shape_id) {
1699 RUBY_ASSERT(removed_shape_id != INVALID_SHAPE_ID);
1700 RUBY_ASSERT(RSHAPE_LEN(next_shape_id) == RSHAPE_LEN(old_shape_id) - 1);
1702 VALUE *fields = rb_imemo_fields_ptr(fields_obj);
1703 attr_index_t removed_index = RSHAPE_INDEX(removed_shape_id);
1704 val = fields[removed_index];
1706 attr_index_t new_fields_count = RSHAPE_LEN(next_shape_id);
1707 if (new_fields_count) {
1708 size_t trailing_fields = new_fields_count - removed_index;
1710 MEMMOVE(&fields[removed_index], &fields[removed_index + 1],
VALUE, trailing_fields);
1711 RBASIC_SET_SHAPE_ID(fields_obj, next_shape_id);
1718 MEMCPY(rb_imemo_fields_ptr(fields_obj), fields,
VALUE, new_fields_count);
1728 RBASIC_SET_SHAPE_ID(obj, next_shape_id);
1729 if (fields_obj != original_fields_obj) {
1735 RCLASS_WRITABLE_SET_FIELDS_OBJ(obj, fields_obj);
1738 rb_obj_set_fields(obj, fields_obj,
id, original_fields_obj);
1747rb_attr_delete(
VALUE obj,
ID id)
1749 return rb_ivar_delete(obj,
id,
Qnil);
1753rb_obj_init_too_complex(
VALUE obj, st_table *table)
1757 RUBY_ASSERT(rb_shape_canonical_p(RBASIC_SHAPE_ID(obj)));
1758 RUBY_ASSERT(RSHAPE_LEN(RBASIC_SHAPE_ID(obj)) == 0);
1760 obj_transition_too_complex(obj, table);
1764imemo_fields_complex_from_obj_i(
ID key,
VALUE val, st_data_t arg)
1767 st_table *table = rb_imemo_fields_complex_tbl(fields);
1769 RUBY_ASSERT(!st_lookup(table, (st_data_t)key, NULL));
1770 st_add_direct(table, (st_data_t)key, (st_data_t)val);
1777imemo_fields_complex_from_obj(
VALUE owner,
VALUE source_fields_obj, shape_id_t shape_id)
1779 attr_index_t
len = source_fields_obj ? RSHAPE_LEN(RBASIC_SHAPE_ID(source_fields_obj)) : 0;
1782 rb_field_foreach(source_fields_obj, imemo_fields_complex_from_obj_i, (st_data_t)fields_obj,
false);
1783 RBASIC_SET_SHAPE_ID(fields_obj, shape_id);
1789imemo_fields_copy_capa(
VALUE owner,
VALUE source_fields_obj, attr_index_t new_size)
1792 if (source_fields_obj) {
1793 attr_index_t fields_count = RSHAPE_LEN(RBASIC_SHAPE_ID(source_fields_obj));
1794 VALUE *fields = rb_imemo_fields_ptr(fields_obj);
1795 MEMCPY(fields, rb_imemo_fields_ptr(source_fields_obj),
VALUE, fields_count);
1796 RBASIC_SET_SHAPE_ID(fields_obj, RBASIC_SHAPE_ID(source_fields_obj));
1797 for (attr_index_t i = 0; i < fields_count; i++) {
1805imemo_fields_set(
VALUE owner,
VALUE fields_obj, shape_id_t target_shape_id,
ID field_name,
VALUE val,
bool concurrent)
1807 const VALUE original_fields_obj = fields_obj;
1808 shape_id_t current_shape_id = fields_obj ? RBASIC_SHAPE_ID(fields_obj) : ROOT_SHAPE_ID;
1810 if (UNLIKELY(rb_shape_too_complex_p(target_shape_id))) {
1811 if (rb_shape_too_complex_p(current_shape_id)) {
1816 fields_obj = rb_imemo_fields_clone(fields_obj);
1820 fields_obj = imemo_fields_complex_from_obj(owner, original_fields_obj, target_shape_id);
1821 current_shape_id = target_shape_id;
1824 st_table *table = rb_imemo_fields_complex_tbl(fields_obj);
1827 st_insert(table, (st_data_t)field_name, (st_data_t)val);
1829 RBASIC_SET_SHAPE_ID(fields_obj, target_shape_id);
1832 attr_index_t index = RSHAPE_INDEX(target_shape_id);
1833 if (concurrent || index >= RSHAPE_CAPACITY(current_shape_id)) {
1834 fields_obj = imemo_fields_copy_capa(owner, original_fields_obj, RSHAPE_CAPACITY(target_shape_id));
1837 VALUE *table = rb_imemo_fields_ptr(fields_obj);
1840 if (RSHAPE_LEN(target_shape_id) > RSHAPE_LEN(current_shape_id)) {
1841 RBASIC_SET_SHAPE_ID(fields_obj, target_shape_id);
1849generic_field_set(
VALUE obj, shape_id_t target_shape_id,
ID field_name,
VALUE val)
1852 field_name = RSHAPE_EDGE_NAME(target_shape_id);
1856 const VALUE original_fields_obj = rb_obj_fields(obj, field_name);
1857 VALUE fields_obj = imemo_fields_set(obj, original_fields_obj, target_shape_id, field_name, val,
false);
1859 rb_obj_set_fields(obj, fields_obj, field_name, original_fields_obj);
1860 return rb_shape_too_complex_p(target_shape_id) ? ATTR_INDEX_NOT_SET : RSHAPE_INDEX(target_shape_id);
1864generic_shape_ivar(
VALUE obj,
ID id,
bool *new_ivar_out)
1866 bool new_ivar =
false;
1867 shape_id_t current_shape_id = RBASIC_SHAPE_ID(obj);
1868 shape_id_t target_shape_id = current_shape_id;
1870 if (!rb_shape_too_complex_p(current_shape_id)) {
1871 if (!rb_shape_find_ivar(current_shape_id,
id, &target_shape_id)) {
1872 if (RSHAPE_LEN(current_shape_id) >= SHAPE_MAX_FIELDS) {
1873 rb_raise(rb_eArgError,
"too many instance variables");
1877 target_shape_id = rb_shape_transition_add_ivar(obj,
id);
1881 *new_ivar_out = new_ivar;
1882 return target_shape_id;
1889 shape_id_t target_shape_id = generic_shape_ivar(obj,
id, &dontcare);
1890 return generic_field_set(obj, target_shape_id,
id, val);
1894rb_ensure_iv_list_size(
VALUE obj, uint32_t current_len, uint32_t new_capacity)
1906 ROBJECT(obj)->as.heap.fields = newptr;
1911rb_obj_copy_ivs_to_hash_table_i(
ID key,
VALUE val, st_data_t arg)
1913 RUBY_ASSERT(!st_lookup((st_table *)arg, (st_data_t)key, NULL));
1915 st_add_direct((st_table *)arg, (st_data_t)key, (st_data_t)val);
1920rb_obj_copy_ivs_to_hash_table(
VALUE obj, st_table *table)
1922 rb_ivar_foreach(obj, rb_obj_copy_ivs_to_hash_table_i, (st_data_t)table);
1926rb_obj_copy_fields_to_hash_table(
VALUE obj, st_table *table)
1928 rb_field_foreach(obj, rb_obj_copy_ivs_to_hash_table_i, (st_data_t)table,
false);
1932obj_field_set(
VALUE obj, shape_id_t target_shape_id,
ID field_name,
VALUE val)
1934 shape_id_t current_shape_id = RBASIC_SHAPE_ID(obj);
1936 if (UNLIKELY(rb_shape_too_complex_p(target_shape_id))) {
1937 if (UNLIKELY(!rb_shape_too_complex_p(current_shape_id))) {
1938 current_shape_id = rb_evict_fields_to_hash(obj);
1941 if (RSHAPE_LEN(target_shape_id) > RSHAPE_LEN(current_shape_id)) {
1942 RBASIC_SET_SHAPE_ID(obj, target_shape_id);
1946 field_name = RSHAPE_EDGE_NAME(target_shape_id);
1950 st_insert(ROBJECT_FIELDS_HASH(obj), (st_data_t)field_name, (st_data_t)val);
1953 return ATTR_INDEX_NOT_SET;
1956 attr_index_t index = RSHAPE_INDEX(target_shape_id);
1958 if (index >= RSHAPE_LEN(current_shape_id)) {
1959 if (UNLIKELY(index >= RSHAPE_CAPACITY(current_shape_id))) {
1960 rb_ensure_iv_list_size(obj, RSHAPE_CAPACITY(current_shape_id), RSHAPE_CAPACITY(target_shape_id));
1962 RBASIC_SET_SHAPE_ID(obj, target_shape_id);
1975 shape_id_t target_shape_id = generic_shape_ivar(obj,
id, &dontcare);
1976 return obj_field_set(obj, target_shape_id,
id, val);
1986 rb_check_frozen(obj);
1987 obj_ivar_set(obj,
id, val);
1999 RB_SET_SHAPE_ID(x, rb_shape_transition_frozen(x));
2007static attr_index_t class_ivar_set(
VALUE obj,
ID id,
VALUE val,
bool *new_ivar);
2012 RB_DEBUG_COUNTER_INC(ivar_set_base);
2016 return obj_ivar_set(obj,
id, val);
2020 IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(
id);
2022 return class_ivar_set(obj,
id, val, &dontcare);
2025 return generic_ivar_set(obj,
id, val);
2032 rb_check_frozen(obj);
2033 ivar_set(obj,
id, val);
2040 return ivar_set(obj,
id, val);
2049 ivar_set(obj,
id, val);
2053rb_obj_field_set(
VALUE obj, shape_id_t target_shape_id,
ID field_name,
VALUE val)
2057 return obj_field_set(obj, target_shape_id, field_name, val);
2061 rb_bug(
"Unreachable");
2064 return generic_field_set(obj, target_shape_id, field_name, val);
2069ivar_defined0(
VALUE obj,
ID id)
2073 if (rb_shape_obj_too_complex_p(obj)) {
2075 st_table *table = NULL;
2079 rb_bug(
"Unreachable");
2084 table = rb_imemo_fields_complex_tbl(obj);
2088 table = ROBJECT_FIELDS_HASH(obj);
2092 VALUE fields_obj = rb_obj_fields_no_ractor_check(obj);
2093 table = rb_imemo_fields_complex_tbl(fields_obj);
2097 if (!table || !rb_st_lookup(table,
id, &idx)) {
2104 return RBOOL(rb_shape_get_iv_index(RBASIC_SHAPE_ID(obj),
id, &index));
2118 VALUE fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
2120 defined = ivar_defined0(fields_obj,
id);
2125 defined = ivar_defined0(obj,
id);
2133 struct gen_fields_tbl *fields_tbl;
2135 rb_ivar_foreach_callback_func *func;
2141iterate_over_shapes_callback(shape_id_t shape_id,
void *data)
2145 if (itr_data->ivar_only && !RSHAPE_TYPE_P(shape_id, SHAPE_IVAR)) {
2152 RUBY_ASSERT(!rb_shape_obj_too_complex_p(itr_data->obj));
2156 RUBY_ASSERT(IMEMO_TYPE_P(itr_data->obj, imemo_fields));
2157 RUBY_ASSERT(!rb_shape_obj_too_complex_p(itr_data->obj));
2159 fields = rb_imemo_fields_ptr(itr_data->obj);
2162 rb_bug(
"Unreachable");
2165 VALUE val = fields[RSHAPE_INDEX(shape_id)];
2166 return itr_data->func(RSHAPE_EDGE_NAME(shape_id), val, itr_data->arg);
2173iterate_over_shapes(shape_id_t shape_id, rb_ivar_foreach_callback_func *callback,
struct iv_itr_data *itr_data)
2175 rb_shape_foreach_field(shape_id, iterate_over_shapes_callback, itr_data);
2179each_hash_iv(st_data_t
id, st_data_t val, st_data_t data)
2182 rb_ivar_foreach_callback_func *callback = itr_data->func;
2183 if (is_internal_id((
ID)
id)) {
2186 return callback((
ID)
id, (
VALUE)val, itr_data->arg);
2190obj_fields_each(
VALUE obj, rb_ivar_foreach_callback_func *func, st_data_t arg,
bool ivar_only)
2196 .ivar_only = ivar_only,
2199 shape_id_t shape_id = RBASIC_SHAPE_ID(obj);
2200 if (rb_shape_too_complex_p(shape_id)) {
2201 rb_st_foreach(ROBJECT_FIELDS_HASH(obj), each_hash_iv, (st_data_t)&itr_data);
2205 iterate_over_shapes(shape_id, func, &itr_data);
2210imemo_fields_each(
VALUE fields_obj, rb_ivar_foreach_callback_func *func, st_data_t arg,
bool ivar_only)
2212 IMEMO_TYPE_P(fields_obj, imemo_fields);
2218 .ivar_only = ivar_only,
2221 shape_id_t shape_id = RBASIC_SHAPE_ID(fields_obj);
2222 if (rb_shape_too_complex_p(shape_id)) {
2223 rb_st_foreach(rb_imemo_fields_complex_tbl(fields_obj), each_hash_iv, (st_data_t)&itr_data);
2226 itr_data.fields = rb_imemo_fields_ptr(fields_obj);
2227 iterate_over_shapes(shape_id, func, &itr_data);
2234 VALUE new_fields_obj;
2236 rb_check_frozen(dest);
2238 if (!rb_obj_gen_fields_p(obj)) {
2242 shape_id_t src_shape_id = rb_obj_shape_id(obj);
2244 VALUE fields_obj = rb_obj_fields_no_ractor_check(obj);
2251 if (rb_shape_too_complex_p(src_shape_id)) {
2252 rb_shape_copy_complex_ivars(dest, obj, src_shape_id, rb_imemo_fields_complex_tbl(fields_obj));
2256 shape_id_t dest_shape_id = src_shape_id;
2257 shape_id_t initial_shape_id = rb_obj_shape_id(dest);
2259 if (!rb_shape_canonical_p(src_shape_id)) {
2260 RUBY_ASSERT(RSHAPE_TYPE_P(initial_shape_id, SHAPE_ROOT));
2262 dest_shape_id = rb_shape_rebuild(initial_shape_id, src_shape_id);
2263 if (UNLIKELY(rb_shape_too_complex_p(dest_shape_id))) {
2264 st_table *table = rb_st_init_numtable_with_size(src_num_ivs);
2265 rb_obj_copy_ivs_to_hash_table(obj, table);
2266 rb_obj_init_too_complex(dest, table);
2271 if (!RSHAPE_LEN(dest_shape_id)) {
2272 RBASIC_SET_SHAPE_ID(dest, dest_shape_id);
2276 new_fields_obj = rb_imemo_fields_new(dest, RSHAPE_CAPACITY(dest_shape_id),
RB_OBJ_SHAREABLE_P(dest));
2277 VALUE *src_buf = rb_imemo_fields_ptr(fields_obj);
2278 VALUE *dest_buf = rb_imemo_fields_ptr(new_fields_obj);
2279 rb_shape_copy_fields(new_fields_obj, dest_buf, dest_shape_id, src_buf, src_shape_id);
2280 RBASIC_SET_SHAPE_ID(new_fields_obj, dest_shape_id);
2282 rb_obj_replace_fields(dest, new_fields_obj);
2291rb_replace_generic_ivar(
VALUE clone,
VALUE obj)
2294 st_data_t fields_tbl, obj_data = (st_data_t)obj;
2295 if (st_delete(generic_fields_tbl_, &obj_data, &fields_tbl)) {
2296 st_insert(generic_fields_tbl_, (st_data_t)clone, fields_tbl);
2300 rb_bug(
"unreachable");
2306rb_field_foreach(
VALUE obj, rb_ivar_foreach_callback_func *func, st_data_t arg,
bool ivar_only)
2311 if (IMEMO_TYPE_P(obj, imemo_fields)) {
2312 imemo_fields_each(obj, func, arg, ivar_only);
2316 obj_fields_each(obj, func, arg, ivar_only);
2321 IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(0);
2322 VALUE fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
2324 imemo_fields_each(fields_obj, func, arg, ivar_only);
2330 VALUE fields_obj = rb_obj_fields_no_ractor_check(obj);
2332 imemo_fields_each(fields_obj, func, arg, ivar_only);
2342 rb_field_foreach(obj, func, arg,
true);
2350 st_index_t iv_count = 0;
2353 iv_count = ROBJECT_FIELDS_COUNT(obj);
2359 VALUE fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
2363 if (rb_shape_obj_too_complex_p(fields_obj)) {
2364 iv_count = rb_st_table_size(rb_imemo_fields_complex_tbl(fields_obj));
2367 iv_count = RBASIC_FIELDS_COUNT(fields_obj);
2375 if (rb_shape_obj_too_complex_p(obj)) {
2376 iv_count = rb_st_table_size(rb_imemo_fields_complex_tbl(obj));
2379 iv_count = RBASIC_FIELDS_COUNT(obj);
2385 VALUE fields_obj = rb_obj_fields_no_ractor_check(obj);
2387 if (rb_shape_obj_too_complex_p(fields_obj)) {
2388 rb_st_table_size(rb_imemo_fields_complex_tbl(fields_obj));
2391 iv_count = RBASIC_FIELDS_COUNT(obj);
2398 if (rb_shape_obj_has_id(obj)) {
2406ivar_i(
ID key,
VALUE v, st_data_t a)
2443#define rb_is_constant_id rb_is_const_id
2444#define rb_is_constant_name rb_is_const_name
2445#define id_for_var(obj, name, part, type) \
2446 id_for_var_message(obj, name, type, "'%1$s' is not allowed as "#part" "#type" variable name")
2447#define id_for_var_message(obj, name, type, message) \
2448 check_id_type(obj, &(name), rb_is_##type##_id, rb_is_##type##_name, message, strlen(message))
2451 int (*valid_id_p)(
ID),
int (*valid_name_p)(
VALUE),
2452 const char *message,
size_t message_len)
2455 VALUE name = *pname;
2457 if (
id ? !valid_id_p(
id) : !valid_name_p(name)) {
2458 rb_name_err_raise_str(rb_fstring_new(message, message_len),
2490 const ID id = id_for_var(obj, name, an, instance);
2494 rb_check_frozen(obj);
2499 if (!UNDEF_P(val))
return val;
2502 rb_name_err_raise(
"instance variable %1$s not defined",
2507NORETURN(
static void uninitialized_constant(
VALUE,
VALUE));
2509uninitialized_constant(
VALUE klass,
VALUE name)
2512 rb_name_err_raise(
"uninitialized constant %2$s::%1$s",
2515 rb_name_err_raise(
"uninitialized constant %1$s",
2522 VALUE value = rb_funcallv(klass, idConst_missing, 1, &name);
2523 rb_vm_inc_const_missing_count();
2571 rb_execution_context_t *ec = GET_EC();
2572 VALUE ref = ec->private_const_reference;
2573 rb_vm_pop_cfunc_frame();
2575 ec->private_const_reference = 0;
2576 rb_name_err_raise(
"private constant %2$s::%1$s referenced", ref, name);
2578 uninitialized_constant(klass, name);
2584autoload_table_mark(
void *ptr)
2586 rb_mark_tbl_no_pin((st_table *)ptr);
2590autoload_table_free(
void *ptr)
2592 st_free_table((st_table *)ptr);
2596autoload_table_memsize(
const void *ptr)
2598 const st_table *tbl = ptr;
2599 return st_memsize(tbl);
2603autoload_table_compact(
void *ptr)
2605 rb_gc_ref_update_table_values_only((st_table *)ptr);
2610 {autoload_table_mark, autoload_table_free, autoload_table_memsize, autoload_table_compact,},
2611 0, 0, RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED
2614#define check_autoload_table(av) \
2615 (struct st_table *)rb_check_typeddata((av), &autoload_table_type)
2620 struct st_table *tbl;
2626 if (RICLASS_IS_ORIGIN_P(mod)) {
2630 mod =
RBASIC(mod)->klass;
2638 VALUE tbl_value = rb_ivar_lookup(mod, autoload,
Qfalse);
2639 if (!
RTEST(tbl_value) || !(tbl = check_autoload_table(tbl_value)) || !st_lookup(tbl, (st_data_t)
id, &val)) {
2649 struct ccan_list_node cnode;
2652 VALUE autoload_data_value;
2668 rb_const_flag_t flag;
2684 rb_serial_t fork_gen;
2687 struct ccan_list_head constants;
2691autoload_data_mark_and_move(
void *ptr)
2695 rb_gc_mark_and_move(&p->feature);
2696 rb_gc_mark_and_move(&p->mutex);
2700autoload_data_free(
void *ptr)
2705 ccan_list_for_each_safe(&p->constants,
autoload_const, next, cnode) {
2713autoload_data_memsize(
const void *ptr)
2720 {autoload_data_mark_and_move, autoload_data_free, autoload_data_memsize, autoload_data_mark_and_move},
2721 0, 0, RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED
2725autoload_const_mark_and_move(
void *ptr)
2729 rb_gc_mark_and_move(&ac->module);
2730 rb_gc_mark_and_move(&ac->autoload_data_value);
2731 rb_gc_mark_and_move(&ac->value);
2732 rb_gc_mark_and_move(&ac->file);
2733 rb_gc_mark_and_move(&ac->box_value);
2737autoload_const_memsize(
const void *ptr)
2743autoload_const_free(
void *ptr)
2753 {autoload_const_mark_and_move, autoload_const_free, autoload_const_memsize, autoload_const_mark_and_move,},
2754 0, 0, RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED
2758get_autoload_data(
VALUE autoload_const_value,
struct autoload_const **autoload_const_pointer)
2771 if (autoload_const_pointer) *autoload_const_pointer =
autoload_const;
2777 VALUE dst_tbl_value;
2779 const rb_box_t *box;
2783autoload_copy_table_for_box_i(st_data_t key, st_data_t value, st_data_t arg)
2787 struct st_table *tbl = data->dst_tbl;
2788 VALUE tbl_value = data->dst_tbl_value;
2789 const rb_box_t *box = data->box;
2794 VALUE autoload_data_value = src_const->autoload_data_value;
2806 st_insert(tbl, (st_data_t)
autoload_const->name, (st_data_t)new_value);
2813rb_autoload_copy_table_for_box(st_table *iv_ptr,
const rb_box_t *box)
2815 struct st_table *src_tbl, *dst_tbl;
2816 VALUE src_tbl_value, dst_tbl_value;
2817 if (!rb_st_lookup(iv_ptr, (st_data_t)autoload, (st_data_t *)&src_tbl_value)) {
2821 if (!
RTEST(src_tbl_value) || !(src_tbl = check_autoload_table(src_tbl_value))) {
2825 src_tbl = check_autoload_table(src_tbl_value);
2831 .dst_tbl_value = dst_tbl_value,
2836 st_foreach(src_tbl, autoload_copy_table_for_box_i, (st_data_t)&data);
2837 st_insert(iv_ptr, (st_data_t)autoload, (st_data_t)dst_tbl_value);
2841rb_autoload(
VALUE module,
ID name,
const char *feature)
2843 if (!feature || !*feature) {
2844 rb_raise(rb_eArgError,
"empty feature name");
2847 rb_autoload_str(module, name, rb_fstring_cstr(feature));
2851static void const_added(
VALUE klass,
ID const_name);
2861autoload_feature_lookup_or_create(
VALUE feature,
struct autoload_data **autoload_data_pointer)
2863 RUBY_ASSERT_MUTEX_OWNED(autoload_mutex);
2864 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
2866 VALUE autoload_data_value = rb_hash_aref(autoload_features, feature);
2869 if (
NIL_P(autoload_data_value)) {
2875 if (autoload_data_pointer) *autoload_data_pointer =
autoload_data;
2877 rb_hash_aset(autoload_features, feature, autoload_data_value);
2879 else if (autoload_data_pointer) {
2880 *autoload_data_pointer =
rb_check_typeddata(autoload_data_value, &autoload_data_type);
2883 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
2884 return autoload_data_value;
2888autoload_table_lookup_or_create(
VALUE module)
2890 VALUE autoload_table_value = rb_ivar_lookup(module, autoload,
Qfalse);
2891 if (
RTEST(autoload_table_value)) {
2892 return autoload_table_value;
2896 rb_class_ivar_set(module, autoload, autoload_table_value);
2898 return autoload_table_value;
2903autoload_synchronized(
VALUE _arguments)
2907 rb_const_entry_t *constant_entry = rb_const_lookup(arguments->module, arguments->name);
2908 if (constant_entry && !UNDEF_P(constant_entry->value)) {
2913 const_set(arguments->module, arguments->name,
Qundef);
2915 VALUE autoload_table_value = autoload_table_lookup_or_create(arguments->module);
2916 struct st_table *autoload_table = check_autoload_table(autoload_table_value);
2919 VALUE feature = rb_fstring(arguments->feature);
2922 VALUE autoload_data_value = autoload_feature_lookup_or_create(feature, &
autoload_data);
2934 st_insert(autoload_table, (st_data_t)arguments->name, (st_data_t)autoload_const_value);
2944 const rb_box_t *box = rb_current_box();
2945 VALUE current_box_value = rb_get_box_object((rb_box_t *)box);
2948 rb_raise(
rb_eNameError,
"autoload must be constant name: %"PRIsVALUE
"", QUOTE_ID(name));
2952 if (!RSTRING_LEN(feature)) {
2953 rb_raise(rb_eArgError,
"empty feature name");
2960 .box_value = current_box_value,
2965 if (result ==
Qtrue) {
2966 const_added(module, name);
2971autoload_delete(
VALUE module,
ID name)
2973 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
2975 st_data_t load = 0, key = name;
2979 VALUE table_value = rb_ivar_lookup(module, autoload,
Qfalse);
2980 if (
RTEST(table_value)) {
2981 struct st_table *table = check_autoload_table(table_value);
2983 st_delete(table, &key, &load);
3006 if (table->num_entries == 0) {
3007 rb_attr_delete(module, autoload);
3012 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
3018 return ele->mutex !=
Qnil && !rb_mutex_owned_p(ele->mutex);
3022check_autoload_required(
VALUE mod,
ID id,
const char **loadingpath)
3026 const char *loading;
3028 if (!autoload_const_value || !(
autoload_data = get_autoload_data(autoload_const_value, 0))) {
3041 return autoload_const_value;
3044 loading = RSTRING_PTR(feature);
3047 return autoload_const_value;
3050 if (loadingpath && loading) {
3051 *loadingpath = loading;
3052 return autoload_const_value;
3061rb_autoloading_value(
VALUE mod,
ID id,
VALUE* value, rb_const_flag_t *flag)
3064 if (!ac)
return FALSE;
3080 return ele->mutex !=
Qnil && rb_mutex_owned_p(ele->mutex);
3087autoloading_const_entry(
VALUE mod,
ID id)
3094 if (!load || !(ele = get_autoload_data(load, &ac))) {
3100 if (autoload_by_current(ele)) {
3101 if (!UNDEF_P(ac->value)) {
3110autoload_defined_p(
VALUE mod,
ID id)
3112 rb_const_entry_t *ce = rb_const_lookup(mod,
id);
3115 if (!ce || !UNDEF_P(ce->value)) {
3121 return !rb_autoloading_value(mod,
id, NULL, NULL);
3134 struct autoload_const *autoload_const;
3137 struct autoload_data *autoload_data;
3143 check_before_mod_set(ac->module, ac->name, ac->value,
"constant");
3146 const_tbl_update(ac,
true);
3153autoload_load_needed(
VALUE _arguments)
3157 const char *loading = 0, *src;
3159 if (!autoload_defined_p(arguments->module, arguments->name)) {
3163 VALUE autoload_const_value = check_autoload_required(arguments->module, arguments->name, &loading);
3164 if (!autoload_const_value) {
3169 if (src && loading && strcmp(src, loading) == 0) {
3190 return autoload_const_value;
3194autoload_apply_constants(
VALUE _arguments)
3196 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
3208 ccan_list_for_each_safe(&arguments->autoload_data->constants,
autoload_const, next, cnode) {
3214 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
3220autoload_feature_require(
VALUE _arguments)
3222 VALUE receiver = rb_vm_top_self();
3232 if (rb_box_available() && BOX_OBJ_P(autoload_box_value))
3233 receiver = autoload_box_value;
3241 rb_gccct_clear_table(
Qnil);
3243 VALUE result =
rb_funcall(receiver, rb_intern(
"require"), 1, arguments->autoload_data->feature);
3245 if (
RTEST(result)) {
3252autoload_try_load(
VALUE _arguments)
3256 VALUE result = autoload_feature_require(_arguments);
3259 rb_const_entry_t *ce = rb_const_lookup(arguments->module, arguments->name);
3261 if (!ce || UNDEF_P(ce->value)) {
3266 if (arguments->module == rb_cObject) {
3268 "Expected %"PRIsVALUE
" to define %"PRIsVALUE
" but it didn't",
3269 arguments->autoload_data->feature,
3275 "Expected %"PRIsVALUE
" to define %"PRIsVALUE
"::%"PRIsVALUE
" but it didn't",
3276 arguments->autoload_data->feature,
3284 ce->flag |= arguments->flag;
3293 rb_const_entry_t *ce = rb_const_lookup(module, name);
3296 if (!ce || !UNDEF_P(ce->value)) {
3301 if (UNLIKELY(!rb_ractor_main_p())) {
3302 return rb_ractor_autoload_load(module, name);
3312 if (autoload_const_value ==
Qfalse)
return autoload_const_value;
3314 arguments.flag = ce->flag & (CONST_DEPRECATED | CONST_VISIBILITY_MASK);
3331 return rb_autoload_at_p(mod,
id, TRUE);
3335rb_autoload_at_p(
VALUE mod,
ID id,
int recur)
3340 while (!autoload_defined_p(mod,
id)) {
3341 if (!recur)
return Qnil;
3343 if (!mod)
return Qnil;
3345 load = check_autoload_required(mod,
id, 0);
3346 if (!load)
return Qnil;
3347 return (ele = get_autoload_data(load, 0)) ? ele->feature :
Qnil;
3351rb_const_warn_if_deprecated(
const rb_const_entry_t *ce,
VALUE klass,
ID id)
3353 if (RB_CONST_DEPRECATED_P(ce) &&
3355 if (klass == rb_cObject) {
3366rb_const_get_0(
VALUE klass,
ID id,
int exclude,
int recurse,
int visibility)
3369 VALUE c = rb_const_search(klass,
id, exclude, recurse, visibility, &found_in);
3371 if (UNLIKELY(!rb_ractor_main_p())) {
3373 rb_raise(rb_eRactorIsolationError,
"can not access non-shareable objects in constant %"PRIsVALUE
"::%"PRIsVALUE
" by non-main Ractor.",
rb_class_path(found_in), rb_id2str(
id));
3378 return rb_const_missing(klass,
ID2SYM(
id));
3382rb_const_search_from(
VALUE klass,
ID id,
int exclude,
int recurse,
int visibility,
VALUE *found_in)
3384 VALUE value, current;
3385 bool first_iteration =
true;
3387 for (current = klass;
3389 current =
RCLASS_SUPER(current), first_iteration =
false) {
3392 rb_const_entry_t *ce;
3394 if (!first_iteration && RCLASS_ORIGIN(current) != current) {
3407 while ((ce = rb_const_lookup(tmp,
id))) {
3408 if (visibility && RB_CONST_PRIVATE_P(ce)) {
3409 GET_EC()->private_const_reference = tmp;
3412 rb_const_warn_if_deprecated(ce, tmp,
id);
3414 if (UNDEF_P(value)) {
3416 if (am == tmp)
break;
3418 ac = autoloading_const_entry(tmp,
id);
3420 if (found_in) { *found_in = tmp; }
3426 if (exclude && tmp == rb_cObject) {
3429 if (found_in) { *found_in = tmp; }
3432 if (!recurse)
break;
3436 GET_EC()->private_const_reference = 0;
3441rb_const_search(
VALUE klass,
ID id,
int exclude,
int recurse,
int visibility,
VALUE *found_in)
3445 if (klass == rb_cObject) exclude = FALSE;
3446 value = rb_const_search_from(klass,
id, exclude, recurse, visibility, found_in);
3447 if (!UNDEF_P(value))
return value;
3448 if (exclude)
return value;
3451 return rb_const_search_from(rb_cObject,
id, FALSE, recurse, visibility, found_in);
3457 return rb_const_get_0(klass,
id, TRUE, TRUE, FALSE);
3463 return rb_const_get_0(klass,
id, FALSE, TRUE, FALSE);
3469 return rb_const_get_0(klass,
id, TRUE, FALSE, FALSE);
3473rb_public_const_get_from(
VALUE klass,
ID id)
3475 return rb_const_get_0(klass,
id, TRUE, TRUE, TRUE);
3479rb_public_const_get_at(
VALUE klass,
ID id)
3481 return rb_const_get_0(klass,
id, TRUE, FALSE, TRUE);
3484NORETURN(
static void undefined_constant(
VALUE mod,
VALUE name));
3488 rb_name_err_raise(
"constant %2$s::%1$s not defined",
3493rb_const_location_from(
VALUE klass,
ID id,
int exclude,
int recurse,
int visibility)
3495 while (
RTEST(klass)) {
3496 rb_const_entry_t *ce;
3498 while ((ce = rb_const_lookup(klass,
id))) {
3499 if (visibility && RB_CONST_PRIVATE_P(ce)) {
3502 if (exclude && klass == rb_cObject) {
3506 if (UNDEF_P(ce->value)) {
3508 if (
RTEST(autoload_const_value)) {
3521 if (!recurse)
break;
3530rb_const_location(
VALUE klass,
ID id,
int exclude,
int recurse,
int visibility)
3534 if (klass == rb_cObject) exclude = FALSE;
3535 loc = rb_const_location_from(klass,
id, exclude, recurse, visibility);
3536 if (!
NIL_P(loc))
return loc;
3537 if (exclude)
return loc;
3540 return rb_const_location_from(rb_cObject,
id, FALSE, recurse, visibility);
3544rb_const_source_location(
VALUE klass,
ID id)
3546 return rb_const_location(klass,
id, FALSE, TRUE, FALSE);
3550rb_const_source_location_at(
VALUE klass,
ID id)
3552 return rb_const_location(klass,
id, TRUE, FALSE, FALSE);
3568 const ID id = id_for_var(mod, name, a, constant);
3571 undefined_constant(mod, name);
3576static rb_const_entry_t * const_lookup(
struct rb_id_table *tbl,
ID id);
3582 rb_const_entry_t *ce;
3584 rb_check_frozen(mod);
3586 ce = rb_const_lookup(mod,
id);
3590 rb_name_err_raise(
"cannot remove %2$s::%1$s", mod,
ID2SYM(
id));
3593 undefined_constant(mod,
ID2SYM(
id));
3596 VALUE writable_ce = 0;
3597 if (rb_id_table_lookup(RCLASS_WRITABLE_CONST_TBL(mod),
id, &writable_ce)) {
3598 rb_id_table_delete(RCLASS_WRITABLE_CONST_TBL(mod),
id);
3599 if ((rb_const_entry_t *)writable_ce != ce) {
3600 xfree((rb_const_entry_t *)writable_ce);
3604 rb_const_warn_if_deprecated(ce, mod,
id);
3610 autoload_delete(mod,
id);
3614 if (ce != const_lookup(RCLASS_PRIME_CONST_TBL(mod),
id)) {
3623cv_i_update(st_data_t *k, st_data_t *v, st_data_t a,
int existing)
3625 if (existing)
return ST_STOP;
3630static enum rb_id_table_iterator_result
3633 rb_const_entry_t *ce = (rb_const_entry_t *)v;
3637 st_update(tbl, (st_data_t)key, cv_i_update, (st_data_t)ce);
3639 return ID_TABLE_CONTINUE;
3642static enum rb_id_table_iterator_result
3643rb_local_constants_i(
ID const_name,
VALUE const_value,
void *ary)
3645 if (
rb_is_const_id(const_name) && !RB_CONST_PRIVATE_P((rb_const_entry_t *)const_value)) {
3648 return ID_TABLE_CONTINUE;
3652rb_local_constants(
VALUE mod)
3661 rb_id_table_foreach(tbl, rb_local_constants_i, (
void *)ary);
3670 st_table *tbl = data;
3672 tbl = st_init_numtable();
3674 if (RCLASS_CONST_TBL(mod)) {
3676 rb_id_table_foreach(RCLASS_CONST_TBL(mod), sv_i, tbl);
3690 if (tmp == rb_cObject && mod != rb_cObject)
break;
3696list_i(st_data_t key, st_data_t value,
VALUE ary)
3699 rb_const_entry_t *ce = (rb_const_entry_t *)value;
3707 st_table *tbl = data;
3739 bool inherit =
true;
3747 return rb_local_constants(mod);
3752rb_const_defined_0(
VALUE klass,
ID id,
int exclude,
int recurse,
int visibility)
3756 rb_const_entry_t *ce;
3761 if ((ce = rb_const_lookup(tmp,
id))) {
3762 if (visibility && RB_CONST_PRIVATE_P(ce)) {
3765 if (UNDEF_P(ce->value) && !check_autoload_required(tmp,
id, 0) &&
3766 !rb_autoloading_value(tmp,
id, NULL, NULL))
3769 if (exclude && tmp == rb_cObject && klass != rb_cObject) {
3775 if (!recurse)
break;
3789 return rb_const_defined_0(klass,
id, TRUE, TRUE, FALSE);
3795 return rb_const_defined_0(klass,
id, FALSE, TRUE, FALSE);
3801 return rb_const_defined_0(klass,
id, TRUE, FALSE, FALSE);
3805rb_public_const_defined_from(
VALUE klass,
ID id)
3807 return rb_const_defined_0(klass,
id, TRUE, TRUE, TRUE);
3811check_before_mod_set(
VALUE klass,
ID id,
VALUE val,
const char *dest)
3813 rb_check_frozen(klass);
3816static void set_namespace_path(
VALUE named_namespace,
VALUE name);
3818static enum rb_id_table_iterator_result
3819set_namespace_path_i(
ID id,
VALUE v,
void *payload)
3821 rb_const_entry_t *ce = (rb_const_entry_t *)v;
3822 VALUE value = ce->value;
3825 return ID_TABLE_CONTINUE;
3828 bool has_permanent_classpath;
3829 classname(value, &has_permanent_classpath);
3830 if (has_permanent_classpath) {
3831 return ID_TABLE_CONTINUE;
3833 set_namespace_path(value, build_const_path(parental_path,
id));
3835 if (!RCLASS_PERMANENT_CLASSPATH_P(value)) {
3836 RCLASS_WRITE_CLASSPATH(value, 0,
false);
3839 return ID_TABLE_CONTINUE;
3848set_namespace_path(
VALUE named_namespace,
VALUE namespace_path)
3850 struct rb_id_table *const_table = RCLASS_CONST_TBL(named_namespace);
3851 RB_OBJ_SET_SHAREABLE(namespace_path);
3854 RCLASS_WRITE_CLASSPATH(named_namespace, namespace_path,
true);
3857 rb_id_table_foreach(const_table, set_namespace_path_i, &namespace_path);
3863const_added(
VALUE klass,
ID const_name)
3865 if (GET_VM()->running) {
3867 rb_funcallv(klass, idConst_added, 1, &name);
3874 rb_const_entry_t *ce;
3877 rb_raise(
rb_eTypeError,
"no class/module to define constant %"PRIsVALUE
"",
3882 rb_raise(rb_eRactorIsolationError,
"can not set constants with non-shareable objects by non-main Ractors");
3885 check_before_mod_set(klass,
id, val,
"constant");
3888 struct rb_id_table *tbl = RCLASS_WRITABLE_CONST_TBL(klass);
3890 tbl = rb_id_table_create(0);
3891 RCLASS_WRITE_CONST_TBL(klass, tbl,
false);
3893 ce =
ZALLOC(rb_const_entry_t);
3894 rb_id_table_insert(tbl,
id, (
VALUE)ce);
3895 setup_const_entry(ce, klass, val, CONST_PUBLIC);
3899 .module = klass, .name = id,
3900 .value = val, .flag = CONST_PUBLIC,
3903 ac.file = rb_source_location(&ac.line);
3904 const_tbl_update(&ac,
false);
3912 if (rb_cObject && rb_namespace_p(val)) {
3913 bool val_path_permanent;
3914 VALUE val_path = classname(val, &val_path_permanent);
3915 if (
NIL_P(val_path) || !val_path_permanent) {
3916 if (klass == rb_cObject) {
3917 set_namespace_path(val, rb_id2str(
id));
3920 bool parental_path_permanent;
3921 VALUE parental_path = classname(klass, &parental_path_permanent);
3922 if (
NIL_P(parental_path)) {
3924 parental_path = rb_tmp_class_path(klass, &throwaway, make_temporary_path);
3926 if (parental_path_permanent && !val_path_permanent) {
3927 set_namespace_path(val, build_const_path(parental_path,
id));
3929 else if (!parental_path_permanent &&
NIL_P(val_path)) {
3930 VALUE path = build_const_path(parental_path,
id);
3931 RCLASS_SET_CLASSPATH(val, path,
false);
3941 const_set(klass,
id, val);
3942 const_added(klass,
id);
3946autoload_const_value_for_named_constant(
VALUE module,
ID name,
struct autoload_const **autoload_const_pointer)
3949 if (!autoload_const_value)
return Qfalse;
3956 return autoload_const_value;
3966 VALUE klass = ac->module;
3967 VALUE val = ac->value;
3969 struct rb_id_table *tbl = RCLASS_CONST_TBL(klass);
3970 rb_const_flag_t visibility = ac->flag;
3971 rb_const_entry_t *ce;
3973 if (rb_id_table_lookup(tbl,
id, &value)) {
3974 ce = (rb_const_entry_t *)value;
3975 if (UNDEF_P(ce->value)) {
3976 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
3977 VALUE file = ac->file;
3978 int line = ac->line;
3979 VALUE autoload_const_value = autoload_const_value_for_named_constant(klass,
id, &ac);
3981 if (!autoload_force && autoload_const_value) {
3985 RB_OBJ_WRITE(autoload_const_value, &ac->file, rb_source_location(&ac->line));
3989 autoload_delete(klass,
id);
3990 ce->flag = visibility;
3995 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
3999 VALUE name = QUOTE_ID(
id);
4000 visibility = ce->flag;
4001 if (klass == rb_cObject)
4002 rb_warn(
"already initialized constant %"PRIsVALUE
"", name);
4004 rb_warn(
"already initialized constant %"PRIsVALUE
"::%"PRIsVALUE
"",
4006 if (!
NIL_P(ce->file) && ce->line) {
4007 rb_compile_warn(RSTRING_PTR(ce->file), ce->line,
4008 "previous definition of %"PRIsVALUE
" was here", name);
4012 setup_const_entry(ce, klass, val, visibility);
4015 tbl = RCLASS_WRITABLE_CONST_TBL(klass);
4018 ce =
ZALLOC(rb_const_entry_t);
4019 rb_id_table_insert(tbl,
id, (
VALUE)ce);
4020 setup_const_entry(ce, klass, val, visibility);
4025setup_const_entry(rb_const_entry_t *ce,
VALUE klass,
VALUE val,
4026 rb_const_flag_t visibility)
4028 ce->flag = visibility;
4030 RB_OBJ_WRITE(klass, &ce->file, rb_source_location(&ce->line));
4036 ID id = rb_intern(name);
4039 rb_warn(
"rb_define_const: invalid name '%s' for constant", name);
4042 rb_vm_register_global_object(val);
4050 rb_define_const(rb_cObject, name, val);
4054set_const_visibility(
VALUE mod,
int argc,
const VALUE *argv,
4055 rb_const_flag_t flag, rb_const_flag_t mask)
4058 rb_const_entry_t *ce;
4063 rb_warning(
"%"PRIsVALUE
" with no argument is just ignored",
4068 for (i = 0; i < argc; i++) {
4070 VALUE val = argv[i];
4073 undefined_constant(mod, val);
4075 if ((ce = rb_const_lookup(mod,
id))) {
4078 if (UNDEF_P(ce->value)) {
4079 if (autoload_const_value_for_named_constant(mod,
id, &ac)) {
4087 undefined_constant(mod,
ID2SYM(
id));
4095 rb_const_entry_t *ce;
4097 long len = strlen(name);
4101 undefined_constant(mod, rb_fstring_new(name,
len));
4103 if (!(ce = rb_const_lookup(mod,
id))) {
4104 undefined_constant(mod,
ID2SYM(
id));
4106 ce->flag |= CONST_DEPRECATED;
4117rb_mod_private_constant(
int argc,
const VALUE *argv,
VALUE obj)
4119 set_const_visibility(obj, argc, argv, CONST_PRIVATE, CONST_VISIBILITY_MASK);
4131rb_mod_public_constant(
int argc,
const VALUE *argv,
VALUE obj)
4133 set_const_visibility(obj, argc, argv, CONST_PUBLIC, CONST_VISIBILITY_MASK);
4157rb_mod_deprecate_constant(
int argc,
const VALUE *argv,
VALUE obj)
4159 set_const_visibility(obj, argc, argv, CONST_DEPRECATED, CONST_DEPRECATED);
4164original_module(
VALUE c)
4172cvar_lookup_at(
VALUE klass,
ID id, st_data_t *v)
4175 if (RICLASS_IS_ORIGIN_P(klass)) {
4180 klass =
RBASIC(klass)->klass;
4185 if (UNDEF_P(n))
return 0;
4192cvar_front_klass(
VALUE klass)
4194 if (RCLASS_SINGLETON_P(klass)) {
4195 VALUE obj = RCLASS_ATTACHED_OBJECT(klass);
4196 if (rb_namespace_p(obj)) {
4206 if (front && target != front) {
4207 if (original_module(front) != original_module(target)) {
4209 "class variable % "PRIsVALUE
" of %"PRIsVALUE
" is overtaken by %"PRIsVALUE
"",
4214 rb_ivar_delete(front,
id,
Qundef);
4219#define CVAR_FOREACH_ANCESTORS(klass, v, r) \
4220 for (klass = cvar_front_klass(klass); klass; klass = RCLASS_SUPER(klass)) { \
4221 if (cvar_lookup_at(klass, id, (v))) { \
4226#define CVAR_LOOKUP(v,r) do {\
4227 CVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(klass, id); \
4228 if (cvar_lookup_at(klass, id, (v))) {r;}\
4229 CVAR_FOREACH_ANCESTORS(klass, v, r);\
4247check_for_cvar_table(
VALUE subclass,
VALUE key)
4252 RB_DEBUG_COUNTER_INC(cvar_class_invalidate);
4253 ruby_vm_global_cvar_state++;
4257 rb_class_foreach_subclass(subclass, check_for_cvar_table, key);
4263 VALUE tmp, front = 0, target = 0;
4266 CVAR_LOOKUP(0, {
if (!front) front = klass; target = klass;});
4268 cvar_overtaken(front, target,
id);
4275 target =
RBASIC(target)->klass;
4277 check_before_mod_set(target,
id, val,
"class variable");
4279 bool new_cvar = rb_class_ivar_set(target,
id, val);
4281 VALUE cvc_tbl = RCLASS_WRITABLE_CVC_TBL(target);
4286 if (!cvc_tbl || !rb_marked_id_table_lookup(cvc_tbl,
id, &ent_data)) {
4290 ent->global_cvar_state = GET_GLOBAL_CVAR_STATE();
4292 VALUE new_cvc_tbl = cvc_tbl;
4294 new_cvc_tbl = rb_marked_id_table_new(2);
4296 else if (rb_multi_ractor_p()) {
4297 new_cvc_tbl = rb_marked_id_table_new(rb_marked_id_table_size(cvc_tbl) + 1);
4300 rb_marked_id_table_insert(new_cvc_tbl,
id, (
VALUE)ent);
4301 if (new_cvc_tbl != cvc_tbl) {
4302 RCLASS_WRITE_CVC_TBL(target, new_cvc_tbl);
4304 RB_DEBUG_COUNTER_INC(cvar_inline_miss);
4307 ent = (
void *)ent_data;
4308 ent->global_cvar_state = GET_GLOBAL_CVAR_STATE();
4316 if (RCLASS_SUBCLASSES_FIRST(target)) {
4317 rb_class_foreach_subclass(target, check_for_cvar_table,
id);
4329 value = find_cvar(klass, front, &target,
id);
4331 rb_name_err_raise(
"uninitialized class variable %1$s in %2$s",
4334 cvar_overtaken(*front, target,
id);
4335 return (
VALUE)value;
4348 if (!klass)
return Qfalse;
4349 CVAR_LOOKUP(0,
return Qtrue);
4354cv_intern(
VALUE klass,
const char *name)
4356 ID id = rb_intern(name);
4358 rb_name_err_raise(
"wrong class variable name %1$s",
4367 ID id = cv_intern(klass, name);
4374 ID id = cv_intern(klass, name);
4385cv_i(
ID key,
VALUE v, st_data_t a)
4387 st_table *tbl = (st_table *)a;
4390 st_update(tbl, (st_data_t)key, cv_i_update, 0);
4396mod_cvar_at(
VALUE mod,
void *data)
4398 st_table *tbl = data;
4400 tbl = st_init_numtable();
4402 mod = original_module(mod);
4409mod_cvar_of(
VALUE mod,
void *data)
4412 if (RCLASS_SINGLETON_P(mod)) {
4413 if (rb_namespace_p(RCLASS_ATTACHED_OBJECT(mod))) {
4414 data = mod_cvar_at(tmp, data);
4415 tmp = cvar_front_klass(tmp);
4419 data = mod_cvar_at(tmp, data);
4427cv_list_i(st_data_t key, st_data_t value,
VALUE ary)
4435cvar_list(
void *data)
4437 st_table *tbl = data;
4471 bool inherit =
true;
4476 tbl = mod_cvar_of(mod, 0);
4479 tbl = mod_cvar_at(mod, 0);
4481 return cvar_list(tbl);
4506 const ID id = id_for_var_message(mod, name,
class,
"wrong class variable name %1$s");
4512 rb_check_frozen(mod);
4513 val = rb_ivar_delete(mod,
id,
Qundef);
4514 if (!UNDEF_P(val)) {
4518 rb_name_err_raise(
"cannot remove %1$s for %2$s", mod,
ID2SYM(
id));
4521 rb_name_err_raise(
"class variable %1$s not defined for %2$s",
4540 ID id = rb_intern(name);
4546class_fields_ivar_set(
VALUE klass,
VALUE fields_obj,
ID id,
VALUE val,
bool concurrent,
VALUE *new_fields_obj,
bool *new_ivar_out)
4548 const VALUE original_fields_obj = fields_obj;
4549 fields_obj = original_fields_obj ? original_fields_obj : rb_imemo_fields_new(klass, 1,
true);
4551 shape_id_t current_shape_id = RBASIC_SHAPE_ID(fields_obj);
4552 shape_id_t next_shape_id = current_shape_id;
4553 if (UNLIKELY(rb_shape_too_complex_p(current_shape_id))) {
4558 next_shape_id = generic_shape_ivar(fields_obj,
id, &new_ivar);
4560 if (UNLIKELY(rb_shape_too_complex_p(next_shape_id))) {
4561 fields_obj = imemo_fields_complex_from_obj(klass, fields_obj, next_shape_id);
4565 attr_index_t index = RSHAPE_INDEX(next_shape_id);
4567 if (index >= RSHAPE_CAPACITY(current_shape_id)) {
4570 fields_obj = imemo_fields_copy_capa(klass, fields_obj, RSHAPE_CAPACITY(next_shape_id));
4574 VALUE *fields = rb_imemo_fields_ptr(fields_obj);
4576 if (concurrent && original_fields_obj == fields_obj) {
4582 RB_OBJ_ATOMIC_WRITE(fields_obj, &fields[index], val);
4589 RBASIC_SET_SHAPE_ID(fields_obj, next_shape_id);
4592 *new_fields_obj = fields_obj;
4593 *new_ivar_out = new_ivar;
4598 if (concurrent && fields_obj == original_fields_obj) {
4602 fields_obj = rb_imemo_fields_clone(fields_obj);
4605 st_table *table = rb_imemo_fields_complex_tbl(fields_obj);
4606 new_ivar = !st_insert(table, (st_data_t)
id, (st_data_t)val);
4609 if (fields_obj != original_fields_obj) {
4610 RBASIC_SET_SHAPE_ID(fields_obj, next_shape_id);
4614 *new_fields_obj = fields_obj;
4615 *new_ivar_out = new_ivar;
4616 return ATTR_INDEX_NOT_SET;
4620class_ivar_set(
VALUE obj,
ID id,
VALUE val,
bool *new_ivar)
4622 rb_class_ensure_writable(obj);
4624 const VALUE original_fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(obj);
4625 VALUE new_fields_obj = 0;
4627 attr_index_t index = class_fields_ivar_set(obj, original_fields_obj,
id, val, rb_multi_ractor_p(), &new_fields_obj, new_ivar);
4629 if (new_fields_obj != original_fields_obj) {
4630 RCLASS_WRITABLE_SET_FIELDS_OBJ(obj, new_fields_obj);
4636 RBASIC_SET_SHAPE_ID(obj, RBASIC_SHAPE_ID(new_fields_obj));
4644 rb_check_frozen(obj);
4647 class_ivar_set(obj,
id, val, &new_ivar);
4656 RUBY_ASSERT(RSHAPE_TYPE_P(RBASIC_SHAPE_ID(dst), SHAPE_ROOT));
4658 VALUE fields_obj = RCLASS_WRITABLE_FIELDS_OBJ(src);
4660 RCLASS_WRITABLE_SET_FIELDS_OBJ(dst, rb_imemo_fields_clone(fields_obj));
4661 RBASIC_SET_SHAPE_ID(dst, RBASIC_SHAPE_ID(src));
4665static rb_const_entry_t *
4672 r = rb_id_table_lookup(tbl,
id, &val);
4675 if (r)
return (rb_const_entry_t *)val;
4681rb_const_lookup(
VALUE klass,
ID id)
4683 return const_lookup(RCLASS_CONST_TBL(klass),
id);
#define RUBY_ASSERT(...)
Asserts that the given expression is truthy if and only if RUBY_DEBUG is truthy.
#define RUBY_EXTERN
Declaration of externally visible global variables.
static VALUE RB_OBJ_FROZEN_RAW(VALUE obj)
This is an implementation detail of RB_OBJ_FROZEN().
static bool RB_FL_ABLE(VALUE obj)
Checks if the object is flaggable.
static void RB_FL_SET_RAW(VALUE obj, VALUE flags)
This is an implementation detail of RB_FL_SET().
void rb_obj_freeze_inline(VALUE obj)
Prevents further modifications to the given object.
static void RB_FL_UNSET_RAW(VALUE obj, VALUE flags)
This is an implementation detail of RB_FL_UNSET().
@ RUBY_FL_FREEZE
This flag has something to do with data immutability.
void rb_class_modify_check(VALUE klass)
Asserts that klass is not a frozen class.
void rb_freeze_singleton_class(VALUE x)
This is an implementation detail of RB_OBJ_FREEZE().
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.
#define rb_str_new2
Old name of rb_str_new_cstr.
#define TYPE(_)
Old name of rb_type.
#define FL_UNSET_RAW
Old name of RB_FL_UNSET_RAW.
#define FL_USER3
Old name of RUBY_FL_USER3.
#define REALLOC_N
Old name of RB_REALLOC_N.
#define ALLOC
Old name of RB_ALLOC.
#define T_STRING
Old name of RUBY_T_STRING.
#define xfree
Old name of ruby_xfree.
#define Qundef
Old name of RUBY_Qundef.
#define rb_str_cat2
Old name of rb_str_cat_cstr.
#define UNREACHABLE
Old name of RBIMPL_UNREACHABLE.
#define T_IMEMO
Old name of RUBY_T_IMEMO.
#define ID2SYM
Old name of RB_ID2SYM.
#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 UNREACHABLE_RETURN
Old name of RBIMPL_UNREACHABLE_RETURN.
#define T_DATA
Old name of RUBY_T_DATA.
#define ZALLOC
Old name of RB_ZALLOC.
#define CLASS_OF
Old name of rb_class_of.
#define T_MODULE
Old name of RUBY_T_MODULE.
#define T_ICLASS
Old name of RUBY_T_ICLASS.
#define ALLOC_N
Old name of RB_ALLOC_N.
#define FL_TEST_RAW
Old name of RB_FL_TEST_RAW.
#define rb_ary_new3
Old name of rb_ary_new_from_args.
#define FL_USER2
Old name of RUBY_FL_USER2.
#define Qtrue
Old name of RUBY_Qtrue.
#define INT2NUM
Old name of RB_INT2NUM.
#define Qnil
Old name of RUBY_Qnil.
#define Qfalse
Old name of RUBY_Qfalse.
#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_CLASS
Old name of RUBY_T_CLASS.
#define BUILTIN_TYPE
Old name of RB_BUILTIN_TYPE.
#define rb_ary_new2
Old name of rb_ary_new_capa.
#define FL_SET_RAW
Old name of RB_FL_SET_RAW.
#define ALLOCV_END
Old name of RB_ALLOCV_END.
void rb_category_warn(rb_warning_category_t category, const char *fmt,...)
Identical to rb_category_warning(), except it reports unless $VERBOSE is nil.
void rb_name_error(ID id, const char *fmt,...)
Raises an instance of rb_eNameError.
VALUE rb_eTypeError
TypeError exception.
void rb_name_error_str(VALUE str, const char *fmt,...)
Identical to rb_name_error(), except it takes a VALUE instead of ID.
VALUE rb_eNameError
NameError exception.
VALUE rb_eRuntimeError
RuntimeError exception.
void * rb_check_typeddata(VALUE obj, const rb_data_type_t *data_type)
Identical to rb_typeddata_is_kind_of(), except it raises exceptions instead of returning false.
void rb_warn(const char *fmt,...)
Identical to rb_warning(), except it reports unless $VERBOSE is nil.
void rb_warning(const char *fmt,...)
Issues a warning.
@ RB_WARN_CATEGORY_DEPRECATED
Warning is for deprecated features.
VALUE rb_obj_hide(VALUE obj)
Make the object invisible from Ruby code.
VALUE rb_obj_class(VALUE obj)
Queries the class of an object.
VALUE rb_cModule
Module class.
VALUE rb_class_real(VALUE klass)
Finds a "real" class.
size_t rb_obj_embedded_size(uint32_t fields_count)
Internal header for Object.
#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.
rb_encoding * rb_usascii_encoding(void)
Queries the encoding that represents US-ASCII.
ID rb_check_id_cstr(const char *ptr, long len, rb_encoding *enc)
Identical to rb_check_id(), except it takes a pointer to a memory region instead of Ruby's string.
VALUE rb_funcall(VALUE recv, ID mid, int n,...)
Calls a method.
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_assoc_new(VALUE car, VALUE cdr)
Identical to rb_ary_new_from_values(), except it expects exactly two parameters.
static int rb_check_arity(int argc, int min, int max)
Ensures that the passed integer is in the passed range.
ID rb_frame_callee(void)
Identical to rb_frame_this_func(), except it returns the named used to call the method.
#define st_foreach_safe
Just another name of rb_st_foreach_safe.
int rb_feature_provided(const char *feature, const char **loading)
Identical to rb_provided(), except it additionally returns the "canonical" name of the loaded feature...
VALUE rb_backref_get(void)
Queries the last match, or Regexp.last_match, or the $~.
int rb_is_instance_id(ID id)
Classifies the given ID, then sees if it is an instance variable.
int rb_is_const_id(ID id)
Classifies the given ID, then sees if it is a constant.
int rb_is_class_id(ID id)
Classifies the given ID, then sees if it is a class variable.
VALUE rb_block_proc(void)
Constructs a Proc object from implicitly passed components.
VALUE rb_reg_nth_defined(int n, VALUE md)
Identical to rb_reg_nth_match(), except it just returns Boolean.
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_subseq(VALUE str, long beg, long len)
Identical to rb_str_substr(), except the numbers are interpreted as byte offsets instead of character...
VALUE rb_str_new_frozen(VALUE str)
Creates a frozen copy of the string, if necessary.
VALUE rb_str_dup(VALUE str)
Duplicates a string.
#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_mutex_new(void)
Creates a mutex.
VALUE rb_mutex_synchronize(VALUE mutex, VALUE(*func)(VALUE arg), VALUE arg)
Obtains the lock, runs the passed function, and releases the lock when it completes.
VALUE rb_exec_recursive_paired(VALUE(*f)(VALUE g, VALUE h, int r), VALUE g, VALUE p, VALUE h)
Identical to rb_exec_recursive(), except it checks for the recursion on the ordered pair of { g,...
VALUE rb_mod_remove_cvar(VALUE mod, VALUE name)
Resembles Module#remove_class_variable.
VALUE rb_obj_instance_variables(VALUE obj)
Resembles Object#instance_variables.
VALUE rb_f_untrace_var(int argc, const VALUE *argv)
Deletes the passed tracer from the passed global variable, or if omitted, deletes everything.
VALUE rb_const_get(VALUE space, ID name)
Identical to rb_const_defined(), except it returns the actual defined value.
VALUE rb_const_list(void *)
This is another mysterious API that comes with no documents at all.
VALUE rb_path2class(const char *path)
Resolves a Q::W::E::R-style path string to the actual class it points.
VALUE rb_autoload_p(VALUE space, ID name)
Queries if an autoload is defined at a point.
void rb_set_class_path(VALUE klass, VALUE space, const char *name)
Names a class.
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_mod_remove_const(VALUE space, VALUE name)
Resembles Module#remove_const.
VALUE rb_class_path_cached(VALUE mod)
Just another name of rb_mod_name.
VALUE rb_f_trace_var(int argc, const VALUE *argv)
Traces a global variable.
void rb_cvar_set(VALUE klass, ID name, VALUE val)
Assigns a value to a class variable.
VALUE rb_cvar_get(VALUE klass, ID name)
Obtains a value from a class variable.
VALUE rb_mod_constants(int argc, const VALUE *argv, VALUE recv)
Resembles Module#constants.
VALUE rb_cvar_find(VALUE klass, ID name, VALUE *front)
Identical to rb_cvar_get(), except it takes additional "front" pointer.
VALUE rb_path_to_class(VALUE path)
Identical to rb_path2class(), except it accepts the path as Ruby's string instead of C's.
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.
void rb_const_set(VALUE space, ID name, VALUE val)
Names a constant.
VALUE rb_autoload_load(VALUE space, ID name)
Kicks the autoload procedure as if it was "touched".
VALUE rb_mod_name(VALUE mod)
Queries the name of a module.
VALUE rb_class_name(VALUE obj)
Queries the name of the given object's class.
VALUE rb_const_get_at(VALUE space, ID name)
Identical to rb_const_defined_at(), except it returns the actual defined value.
void rb_set_class_path_string(VALUE klass, VALUE space, VALUE name)
Identical to rb_set_class_path(), except it accepts the name as Ruby's string instead of C's.
void rb_alias_variable(ID dst, ID src)
Aliases a global variable.
void rb_define_class_variable(VALUE, const char *, VALUE)
Just another name of rb_cv_set.
VALUE rb_obj_remove_instance_variable(VALUE obj, VALUE name)
Resembles Object#remove_instance_variable.
void * rb_mod_const_of(VALUE, void *)
This is a variant of rb_mod_const_at().
st_index_t rb_ivar_count(VALUE obj)
Number of instance variables defined on an object.
void * rb_mod_const_at(VALUE, void *)
This API is mysterious.
VALUE rb_const_remove(VALUE space, ID name)
Identical to rb_mod_remove_const(), except it takes the name as ID instead of VALUE.
VALUE rb_const_get_from(VALUE space, ID name)
Identical to rb_const_defined_at(), except it returns the actual defined value.
VALUE rb_ivar_defined(VALUE obj, ID name)
Queries if the instance variable is defined at the object.
VALUE rb_cv_get(VALUE klass, const char *name)
Identical to rb_cvar_get(), except it accepts C's string instead of ID.
int rb_const_defined_at(VALUE space, ID name)
Identical to rb_const_defined(), except it doesn't look for parent classes.
void rb_cv_set(VALUE klass, const char *name, VALUE val)
Identical to rb_cvar_set(), except it accepts C's string instead of ID.
VALUE rb_mod_class_variables(int argc, const VALUE *argv, VALUE recv)
Resembles Module#class_variables.
VALUE rb_f_global_variables(void)
Queries the list of global variables.
VALUE rb_cvar_defined(VALUE klass, ID name)
Queries if the given class has the given class variable.
VALUE rb_class_path(VALUE mod)
Identical to rb_mod_name(), except it returns #<Class: ...> style inspection for anonymous modules.
int rb_const_defined_from(VALUE space, ID name)
Identical to rb_const_defined(), except it returns false for private constants.
int rb_const_defined(VALUE space, ID name)
Queries if the constant is defined at the namespace.
void rb_free_generic_ivar(VALUE obj)
Frees the list of instance variables.
const char * rb_sourcefile(void)
Resembles __FILE__.
void rb_clear_constant_cache_for_id(ID id)
Clears the inline constant caches associated with a particular ID.
int rb_obj_respond_to(VALUE obj, ID mid, int private_p)
Identical to rb_respond_to(), except it additionally takes the visibility parameter.
static ID rb_intern_const(const char *str)
This is a "tiny optimisation" over rb_intern().
VALUE rb_id2sym(ID id)
Allocates an instance of rb_cSymbol that has the given id.
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...
rb_gvar_setter_t rb_gvar_var_setter
rb_gvar_marker_t rb_gvar_var_marker
void rb_define_global_const(const char *name, VALUE val)
Identical to rb_define_const(), except it defines that of "global", i.e.
VALUE rb_gv_get(const char *name)
Obtains a global variable.
void rb_define_variable(const char *name, VALUE *var)
"Shares" a global variable between Ruby and C.
void rb_gvar_marker_t(VALUE *var)
Type that represents a global variable marker function.
void rb_deprecate_constant(VALUE mod, const char *name)
Asserts that the given constant is deprecated.
void rb_gvar_setter_t(VALUE val, ID id, VALUE *data)
Type that represents a global variable setter function.
rb_gvar_setter_t rb_gvar_val_setter
This is the setter function that backs global variables defined from a ruby script.
rb_gvar_marker_t rb_gvar_undef_marker
void rb_define_readonly_variable(const char *name, const VALUE *var)
Identical to rb_define_variable(), except it does not allow Ruby programs to assign values to such gl...
rb_gvar_setter_t rb_gvar_readonly_setter
This function just raises rb_eNameError.
rb_gvar_getter_t rb_gvar_undef_getter
VALUE rb_gv_set(const char *name, VALUE val)
Assigns to a global variable.
rb_gvar_marker_t rb_gvar_val_marker
This is the setter function that backs global variables defined from a ruby script.
VALUE rb_gvar_getter_t(ID id, VALUE *data)
Type that represents a global variable getter function.
VALUE rb_iv_get(VALUE obj, const char *name)
Obtains an instance variable.
rb_gvar_setter_t rb_gvar_undef_setter
rb_gvar_getter_t rb_gvar_val_getter
This is the getter function that backs global variables defined from a ruby script.
VALUE rb_iv_set(VALUE obj, const char *name, VALUE val)
Assigns to an instance variable.
rb_gvar_getter_t rb_gvar_var_getter
int len
Length of the buffer.
static bool rb_ractor_shareable_p(VALUE obj)
Queries if multiple Ractors can share the passed object or not.
#define RB_OBJ_SHAREABLE_P(obj)
Queries if the passed object has previously classified as shareable or not.
#define MEMCPY(p1, p2, type, n)
Handy macro to call memcpy.
#define RB_GC_GUARD(v)
Prevents premature destruction of local objects.
#define MEMMOVE(p1, p2, type, n)
Handy macro to call memmove.
void rb_define_hooked_variable(const char *q, VALUE *w, type *e, void_type *r)
Define a function-backended global variable.
VALUE type(ANYARGS)
ANYARGS-ed function type.
int st_foreach(st_table *q, int_type *w, st_data_t e)
Iteration over the given table.
void rb_define_virtual_variable(const char *q, type *w, void_type *e)
Define a function-backended global variable.
void rb_ivar_foreach(VALUE q, int_type *w, VALUE e)
Iteration over each instance variable of the object.
VALUE rb_ensure(type *q, VALUE w, type *e, VALUE r)
An equivalent of ensure clause.
void rb_copy_generic_ivar(VALUE clone, VALUE obj)
Copies the list of instance variables.
#define RARRAY_LEN
Just another name of rb_array_len.
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 ROBJECT(obj)
Convenient casting macro.
static VALUE * ROBJECT_FIELDS(VALUE obj)
Queries the instance variables.
#define StringValue(v)
Ensures that the parameter object is a String.
static char * RSTRING_END(VALUE str)
Queries the end of the contents pointer of the string.
static bool RTYPEDDATA_P(VALUE obj)
Checks whether the passed object is RTypedData or RData.
#define RTYPEDDATA_DATA(v)
Convenient getter macro.
#define TypedData_Wrap_Struct(klass, data_type, sval)
Converts sval, a pointer to your struct, into a Ruby object.
struct rb_data_type_struct rb_data_type_t
This is the struct that holds necessary info for a struct.
#define RTYPEDDATA(obj)
Convenient casting macro.
#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_class2name(VALUE klass)
Queries the name of the passed class.
const char * rb_obj_classname(VALUE obj)
Queries the name of the class of the passed object.
#define RB_NO_KEYWORDS
Do not pass keywords.
static bool RB_SPECIAL_CONST_P(VALUE obj)
Checks if the given object is of enum ruby_special_consts.
#define RTEST
This is an old name of RB_TEST.
#define _(args)
This was a transition path from K&R to ANSI.
C99 shim for <stdbool.h>.
uintptr_t ID
Type that represents a Ruby identifier such as a variable name.
uintptr_t VALUE
Type that represents a Ruby object.
static enum ruby_value_type rb_type(VALUE obj)
Identical to RB_BUILTIN_TYPE(), except it can also accept special constants.
static enum ruby_value_type RB_BUILTIN_TYPE(VALUE obj)
Queries the type of the 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.