Ruby 3.5.0dev (2025-04-03 revision 1dddc6c78b5f6dc6ae18ee04ebe44abfce3b0433)
variable.c (1dddc6c78b5f6dc6ae18ee04ebe44abfce3b0433)
1/**********************************************************************
2
3 variable.c -
4
5 $Author$
6 created at: Tue Apr 19 23:55:15 JST 1994
7
8 Copyright (C) 1993-2007 Yukihiro Matsumoto
9 Copyright (C) 2000 Network Applied Communication Laboratory, Inc.
10 Copyright (C) 2000 Information-technology Promotion Agency, Japan
11
12**********************************************************************/
13
14#include "ruby/internal/config.h"
15#include <stddef.h>
17#include "ccan/list/list.h"
18#include "constant.h"
19#include "debug_counter.h"
20#include "id.h"
21#include "id_table.h"
22#include "internal.h"
23#include "internal/class.h"
24#include "internal/compilers.h"
25#include "internal/error.h"
26#include "internal/eval.h"
27#include "internal/hash.h"
28#include "internal/object.h"
29#include "internal/re.h"
30#include "internal/symbol.h"
31#include "internal/thread.h"
32#include "internal/variable.h"
33#include "ruby/encoding.h"
34#include "ruby/st.h"
35#include "ruby/util.h"
36#include "shape.h"
37#include "symbol.h"
38#include "variable.h"
39#include "vm_core.h"
40#include "ractor_core.h"
41#include "vm_sync.h"
42
43RUBY_EXTERN rb_serial_t ruby_vm_global_cvar_state;
44#define GET_GLOBAL_CVAR_STATE() (ruby_vm_global_cvar_state)
45
46typedef void rb_gvar_compact_t(void *var);
47
48static struct rb_id_table *rb_global_tbl;
49static ID autoload;
50
51// This hash table maps file paths to loadable features. We use this to track
52// autoload state until it's no longer needed.
53// feature (file path) => struct autoload_data
54static VALUE autoload_features;
55
56// This mutex is used to protect autoloading state. We use a global mutex which
57// is held until a per-feature mutex can be created. This ensures there are no
58// race conditions relating to autoload state.
59static VALUE autoload_mutex;
60
61static void check_before_mod_set(VALUE, ID, VALUE, const char *);
62static void setup_const_entry(rb_const_entry_t *, VALUE, VALUE, rb_const_flag_t);
63static VALUE rb_const_search(VALUE klass, ID id, int exclude, int recurse, int visibility);
64static st_table *generic_iv_tbl_;
65
66void
67Init_var_tables(void)
68{
69 rb_global_tbl = rb_id_table_create(0);
70 generic_iv_tbl_ = st_init_numtable();
71 autoload = rb_intern_const("__autoload__");
72
73 autoload_mutex = rb_mutex_new();
74 rb_obj_hide(autoload_mutex);
75 rb_vm_register_global_object(autoload_mutex);
76
77 autoload_features = rb_ident_hash_new();
78 rb_obj_hide(autoload_features);
79 rb_vm_register_global_object(autoload_features);
80}
81
82static inline bool
83rb_namespace_p(VALUE obj)
84{
85 if (RB_SPECIAL_CONST_P(obj)) return false;
86 switch (RB_BUILTIN_TYPE(obj)) {
87 case T_MODULE: case T_CLASS: return true;
88 default: break;
89 }
90 return false;
91}
92
103static VALUE
104classname(VALUE klass, bool *permanent)
105{
106 *permanent = false;
107
108 VALUE classpath = RCLASS_EXT(klass)->classpath;
109 if (classpath == 0) return Qnil;
110
111 *permanent = RCLASS_EXT(klass)->permanent_classpath;
112
113 return classpath;
114}
115
116VALUE
117rb_mod_name0(VALUE klass, bool *permanent)
118{
119 return classname(klass, permanent);
120}
121
122/*
123 * call-seq:
124 * mod.name -> string or nil
125 *
126 * Returns the name of the module <i>mod</i>. Returns +nil+ for anonymous modules.
127 */
128
129VALUE
131{
132 // YJIT needs this function to not allocate.
133 bool permanent;
134 return classname(mod, &permanent);
135}
136
137// Similar to logic in rb_mod_const_get().
138static bool
139is_constant_path(VALUE name)
140{
141 const char *path = RSTRING_PTR(name);
142 const char *pend = RSTRING_END(name);
143 rb_encoding *enc = rb_enc_get(name);
144
145 const char *p = path;
146
147 if (p >= pend || !*p) {
148 return false;
149 }
150
151 while (p < pend) {
152 if (p + 2 <= pend && p[0] == ':' && p[1] == ':') {
153 p += 2;
154 }
155
156 const char *pbeg = p;
157 while (p < pend && *p != ':') p++;
158
159 if (pbeg == p) return false;
160
161 if (rb_enc_symname_type(pbeg, p - pbeg, enc, 0) != ID_CONST) {
162 return false;
163 }
164 }
165
166 return true;
167}
168
170 VALUE names;
171 ID last;
172};
173
174static VALUE build_const_path(VALUE head, ID tail);
175static void set_sub_temporary_name_foreach(VALUE mod, struct sub_temporary_name_args *args, VALUE name);
176
177static VALUE
178set_sub_temporary_name_recursive(VALUE mod, VALUE data, int recursive)
179{
180 if (recursive) return Qfalse;
181
182 struct sub_temporary_name_args *args = (void *)data;
183 VALUE name = 0;
184 if (args->names) {
185 name = build_const_path(rb_ary_last(0, 0, args->names), args->last);
186 }
187 set_sub_temporary_name_foreach(mod, args, name);
188 return Qtrue;
189}
190
191static VALUE
192set_sub_temporary_name_topmost(VALUE mod, VALUE data, int recursive)
193{
194 if (recursive) return Qfalse;
195
196 struct sub_temporary_name_args *args = (void *)data;
197 VALUE name = args->names;
198 if (name) {
199 args->names = rb_ary_hidden_new(0);
200 }
201 set_sub_temporary_name_foreach(mod, args, name);
202 return Qtrue;
203}
204
205static enum rb_id_table_iterator_result
206set_sub_temporary_name_i(ID id, VALUE val, void *data)
207{
208 val = ((rb_const_entry_t *)val)->value;
209 if (rb_namespace_p(val) && !RCLASS_EXT(val)->permanent_classpath) {
210 VALUE arg = (VALUE)data;
211 struct sub_temporary_name_args *args = data;
212 args->last = id;
213 rb_exec_recursive_paired(set_sub_temporary_name_recursive, val, arg, arg);
214 }
215 return ID_TABLE_CONTINUE;
216}
217
218static void
219set_sub_temporary_name_foreach(VALUE mod, struct sub_temporary_name_args *args, VALUE name)
220{
221 RCLASS_SET_CLASSPATH(mod, name, FALSE);
222 struct rb_id_table *tbl = RCLASS_CONST_TBL(mod);
223 if (!tbl) return;
224 if (!name) {
225 rb_id_table_foreach(tbl, set_sub_temporary_name_i, args);
226 }
227 else {
228 long names_len = RARRAY_LEN(args->names); // paranoiac check?
229 rb_ary_push(args->names, name);
230 rb_id_table_foreach(tbl, set_sub_temporary_name_i, args);
231 rb_ary_set_len(args->names, names_len);
232 }
233}
234
235static void
236set_sub_temporary_name(VALUE mod, VALUE name)
237{
238 struct sub_temporary_name_args args = {name};
239 VALUE arg = (VALUE)&args;
240 rb_exec_recursive_paired(set_sub_temporary_name_topmost, mod, arg, arg);
241}
242
243/*
244 * call-seq:
245 * mod.set_temporary_name(string) -> self
246 * mod.set_temporary_name(nil) -> self
247 *
248 * Sets the temporary name of the module. This name is reflected in
249 * introspection of the module and the values that are related to it, such
250 * as instances, constants, and methods.
251 *
252 * The name should be +nil+ or a non-empty string that is not a valid constant
253 * path (to avoid confusing between permanent and temporary names).
254 *
255 * The method can be useful to distinguish dynamically generated classes and
256 * modules without assigning them to constants.
257 *
258 * If the module is given a permanent name by assigning it to a constant,
259 * the temporary name is discarded. A temporary name can't be assigned to
260 * modules that have a permanent name.
261 *
262 * If the given name is +nil+, the module becomes anonymous again.
263 *
264 * Example:
265 *
266 * m = Module.new # => #<Module:0x0000000102c68f38>
267 * m.name #=> nil
268 *
269 * m.set_temporary_name("fake_name") # => fake_name
270 * m.name #=> "fake_name"
271 *
272 * m.set_temporary_name(nil) # => #<Module:0x0000000102c68f38>
273 * m.name #=> nil
274 *
275 * c = Class.new
276 * c.set_temporary_name("MyClass(with description)")
277 *
278 * c.new # => #<MyClass(with description):0x0....>
279 *
280 * c::M = m
281 * c::M.name #=> "MyClass(with description)::M"
282 *
283 * # Assigning to a constant replaces the name with a permanent one
284 * C = c
285 *
286 * C.name #=> "C"
287 * C::M.name #=> "C::M"
288 * c.new # => #<C:0x0....>
289 */
290
291VALUE
292rb_mod_set_temporary_name(VALUE mod, VALUE name)
293{
294 // We don't allow setting the name if the classpath is already permanent:
295 if (RCLASS_EXT(mod)->permanent_classpath) {
296 rb_raise(rb_eRuntimeError, "can't change permanent name");
297 }
298
299 if (NIL_P(name)) {
300 // Set the temporary classpath to NULL (anonymous):
301 RB_VM_LOCK_ENTER();
302 set_sub_temporary_name(mod, 0);
303 RB_VM_LOCK_LEAVE();
304 }
305 else {
306 // Ensure the name is a string:
307 StringValue(name);
308
309 if (RSTRING_LEN(name) == 0) {
310 rb_raise(rb_eArgError, "empty class/module name");
311 }
312
313 if (is_constant_path(name)) {
314 rb_raise(rb_eArgError, "the temporary name must not be a constant path to avoid confusion");
315 }
316
317 name = rb_str_new_frozen(name);
318
319 // Set the temporary classpath to the given name:
320 RB_VM_LOCK_ENTER();
321 set_sub_temporary_name(mod, name);
322 RB_VM_LOCK_LEAVE();
323 }
324
325 return mod;
326}
327
328static VALUE
329make_temporary_path(VALUE obj, VALUE klass)
330{
331 VALUE path;
332 switch (klass) {
333 case Qnil:
334 path = rb_sprintf("#<Class:%p>", (void*)obj);
335 break;
336 case Qfalse:
337 path = rb_sprintf("#<Module:%p>", (void*)obj);
338 break;
339 default:
340 path = rb_sprintf("#<%"PRIsVALUE":%p>", klass, (void*)obj);
341 break;
342 }
343 OBJ_FREEZE(path);
344 return path;
345}
346
347typedef VALUE (*fallback_func)(VALUE obj, VALUE name);
348
349static VALUE
350rb_tmp_class_path(VALUE klass, bool *permanent, fallback_func fallback)
351{
352 VALUE path = classname(klass, permanent);
353
354 if (!NIL_P(path)) {
355 return path;
356 }
357
358 if (RB_TYPE_P(klass, T_MODULE)) {
359 if (rb_obj_class(klass) == rb_cModule) {
360 path = Qfalse;
361 }
362 else {
363 bool perm;
364 path = rb_tmp_class_path(RBASIC(klass)->klass, &perm, fallback);
365 }
366 }
367
368 *permanent = false;
369 return fallback(klass, path);
370}
371
372VALUE
374{
375 bool permanent;
376 VALUE path = rb_tmp_class_path(klass, &permanent, make_temporary_path);
377 if (!NIL_P(path)) path = rb_str_dup(path);
378 return path;
379}
380
381VALUE
383{
384 return rb_mod_name(klass);
385}
386
387static VALUE
388no_fallback(VALUE obj, VALUE name)
389{
390 return name;
391}
392
393VALUE
394rb_search_class_path(VALUE klass)
395{
396 bool permanent;
397 return rb_tmp_class_path(klass, &permanent, no_fallback);
398}
399
400static VALUE
401build_const_pathname(VALUE head, VALUE tail)
402{
403 VALUE path = rb_str_dup(head);
404 rb_str_cat2(path, "::");
405 rb_str_append(path, tail);
406 return rb_fstring(path);
407}
408
409static VALUE
410build_const_path(VALUE head, ID tail)
411{
412 return build_const_pathname(head, rb_id2str(tail));
413}
414
415void
417{
418 bool permanent = true;
419
420 VALUE str;
421 if (under == rb_cObject) {
422 str = rb_str_new_frozen(name);
423 }
424 else {
425 str = rb_tmp_class_path(under, &permanent, make_temporary_path);
426 str = build_const_pathname(str, name);
427 }
428
429 RCLASS_SET_CLASSPATH(klass, str, permanent);
430}
431
432void
433rb_set_class_path(VALUE klass, VALUE under, const char *name)
434{
435 VALUE str = rb_str_new2(name);
436 OBJ_FREEZE(str);
437 rb_set_class_path_string(klass, under, str);
438}
439
440VALUE
442{
443 rb_encoding *enc = rb_enc_get(pathname);
444 const char *pbeg, *pend, *p, *path = RSTRING_PTR(pathname);
445 ID id;
446 VALUE c = rb_cObject;
447
448 if (!rb_enc_asciicompat(enc)) {
449 rb_raise(rb_eArgError, "invalid class path encoding (non ASCII)");
450 }
451 pbeg = p = path;
452 pend = path + RSTRING_LEN(pathname);
453 if (path == pend || path[0] == '#') {
454 rb_raise(rb_eArgError, "can't retrieve anonymous class %"PRIsVALUE,
455 QUOTE(pathname));
456 }
457 while (p < pend) {
458 while (p < pend && *p != ':') p++;
459 id = rb_check_id_cstr(pbeg, p-pbeg, enc);
460 if (p < pend && p[0] == ':') {
461 if ((size_t)(pend - p) < 2 || p[1] != ':') goto undefined_class;
462 p += 2;
463 pbeg = p;
464 }
465 if (!id) {
466 goto undefined_class;
467 }
468 c = rb_const_search(c, id, TRUE, FALSE, FALSE);
469 if (UNDEF_P(c)) goto undefined_class;
470 if (!rb_namespace_p(c)) {
471 rb_raise(rb_eTypeError, "%"PRIsVALUE" does not refer to class/module",
472 pathname);
473 }
474 }
475 RB_GC_GUARD(pathname);
476
477 return c;
478
479 undefined_class:
480 rb_raise(rb_eArgError, "undefined class/module % "PRIsVALUE,
481 rb_str_subseq(pathname, 0, p-path));
483}
484
485VALUE
486rb_path2class(const char *path)
487{
488 return rb_path_to_class(rb_str_new_cstr(path));
489}
490
491VALUE
493{
494 return rb_class_path(rb_class_real(klass));
495}
496
497const char *
499{
500 bool permanent;
501 VALUE path = rb_tmp_class_path(rb_class_real(klass), &permanent, make_temporary_path);
502 if (NIL_P(path)) return NULL;
503 return RSTRING_PTR(path);
504}
505
506const char *
508{
509 return rb_class2name(CLASS_OF(obj));
510}
511
512struct trace_var {
513 int removed;
514 void (*func)(VALUE arg, VALUE val);
515 VALUE data;
516 struct trace_var *next;
517};
518
520 int counter;
521 int block_trace;
522 VALUE *data;
523 rb_gvar_getter_t *getter;
524 rb_gvar_setter_t *setter;
525 rb_gvar_marker_t *marker;
526 rb_gvar_compact_t *compactor;
527 struct trace_var *trace;
528};
529
531 struct rb_global_variable *var;
532 ID id;
533 bool ractor_local;
534};
535
536static void
537free_global_variable(struct rb_global_variable *var)
538{
539 RUBY_ASSERT(var->counter == 0);
540
541 struct trace_var *trace = var->trace;
542 while (trace) {
543 struct trace_var *next = trace->next;
544 xfree(trace);
545 trace = next;
546 }
547 xfree(var);
548}
549
550static enum rb_id_table_iterator_result
551free_global_entry_i(VALUE val, void *arg)
552{
553 struct rb_global_entry *entry = (struct rb_global_entry *)val;
554 entry->var->counter--;
555 if (entry->var->counter == 0) {
556 free_global_variable(entry->var);
557 }
558 ruby_xfree(entry);
559 return ID_TABLE_DELETE;
560}
561
562void
563rb_free_rb_global_tbl(void)
564{
565 rb_id_table_foreach_values(rb_global_tbl, free_global_entry_i, 0);
566 rb_id_table_free(rb_global_tbl);
567}
568
569void
570rb_free_generic_iv_tbl_(void)
571{
572 st_free_table(generic_iv_tbl_);
573}
574
575static struct rb_global_entry*
576rb_find_global_entry(ID id)
577{
578 struct rb_global_entry *entry;
579 VALUE data;
580
581 if (!rb_id_table_lookup(rb_global_tbl, id, &data)) {
582 entry = NULL;
583 }
584 else {
585 entry = (struct rb_global_entry *)data;
586 RUBY_ASSERT(entry != NULL);
587 }
588
589 if (UNLIKELY(!rb_ractor_main_p()) && (!entry || !entry->ractor_local)) {
590 rb_raise(rb_eRactorIsolationError, "can not access global variables %s from non-main Ractors", rb_id2name(id));
591 }
592
593 return entry;
594}
595
596void
597rb_gvar_ractor_local(const char *name)
598{
599 struct rb_global_entry *entry = rb_find_global_entry(rb_intern(name));
600 entry->ractor_local = true;
601}
602
603static void
604rb_gvar_undef_compactor(void *var)
605{
606}
607
608static struct rb_global_entry*
610{
611 struct rb_global_entry *entry = rb_find_global_entry(id);
612 if (!entry) {
613 struct rb_global_variable *var;
614 entry = ALLOC(struct rb_global_entry);
615 var = ALLOC(struct rb_global_variable);
616 entry->id = id;
617 entry->var = var;
618 entry->ractor_local = false;
619 var->counter = 1;
620 var->data = 0;
621 var->getter = rb_gvar_undef_getter;
622 var->setter = rb_gvar_undef_setter;
623 var->marker = rb_gvar_undef_marker;
624 var->compactor = rb_gvar_undef_compactor;
625
626 var->block_trace = 0;
627 var->trace = 0;
628 rb_id_table_insert(rb_global_tbl, id, (VALUE)entry);
629 }
630 return entry;
631}
632
633VALUE
635{
636 rb_warning("global variable '%"PRIsVALUE"' not initialized", QUOTE_ID(id));
637
638 return Qnil;
639}
640
641static void
642rb_gvar_val_compactor(void *_var)
643{
644 struct rb_global_variable *var = (struct rb_global_variable *)_var;
645
646 VALUE obj = (VALUE)var->data;
647
648 if (obj) {
649 VALUE new = rb_gc_location(obj);
650 if (new != obj) {
651 var->data = (void*)new;
652 }
653 }
654}
655
656void
658{
659 struct rb_global_variable *var = rb_global_entry(id)->var;
660 var->getter = rb_gvar_val_getter;
661 var->setter = rb_gvar_val_setter;
662 var->marker = rb_gvar_val_marker;
663 var->compactor = rb_gvar_val_compactor;
664
665 var->data = (void*)val;
666}
667
668void
670{
671}
672
673VALUE
674rb_gvar_val_getter(ID id, VALUE *data)
675{
676 return (VALUE)data;
677}
678
679void
681{
682 struct rb_global_variable *var = rb_global_entry(id)->var;
683 var->data = (void*)val;
684}
685
686void
688{
689 VALUE data = (VALUE)var;
690 if (data) rb_gc_mark_movable(data);
691}
692
693VALUE
695{
696 if (!var) return Qnil;
697 return *var;
698}
699
700void
701rb_gvar_var_setter(VALUE val, ID id, VALUE *data)
702{
703 *data = val;
704}
705
706void
708{
709 if (var) rb_gc_mark_maybe(*var);
710}
711
712void
714{
715 rb_name_error(id, "%"PRIsVALUE" is a read-only variable", QUOTE_ID(id));
716}
717
718static enum rb_id_table_iterator_result
719mark_global_entry(VALUE v, void *ignored)
720{
721 struct rb_global_entry *entry = (struct rb_global_entry *)v;
722 struct trace_var *trace;
723 struct rb_global_variable *var = entry->var;
724
725 (*var->marker)(var->data);
726 trace = var->trace;
727 while (trace) {
728 if (trace->data) rb_gc_mark_maybe(trace->data);
729 trace = trace->next;
730 }
731 return ID_TABLE_CONTINUE;
732}
733
734#define gc_mark_table(task) \
735 if (rb_global_tbl) { rb_id_table_foreach_values(rb_global_tbl, task##_global_entry, 0); }
736
737void
738rb_gc_mark_global_tbl(void)
739{
740 gc_mark_table(mark);
741}
742
743static enum rb_id_table_iterator_result
744update_global_entry(VALUE v, void *ignored)
745{
746 struct rb_global_entry *entry = (struct rb_global_entry *)v;
747 struct rb_global_variable *var = entry->var;
748
749 (*var->compactor)(var);
750 return ID_TABLE_CONTINUE;
751}
752
753void
754rb_gc_update_global_tbl(void)
755{
756 gc_mark_table(update);
757}
758
759static ID
760global_id(const char *name)
761{
762 ID id;
763
764 if (name[0] == '$') id = rb_intern(name);
765 else {
766 size_t len = strlen(name);
767 VALUE vbuf = 0;
768 char *buf = ALLOCV_N(char, vbuf, len+1);
769 buf[0] = '$';
770 memcpy(buf+1, name, len);
771 id = rb_intern2(buf, len+1);
772 ALLOCV_END(vbuf);
773 }
774 return id;
775}
776
777static ID
778find_global_id(const char *name)
779{
780 ID id;
781 size_t len = strlen(name);
782
783 if (name[0] == '$') {
784 id = rb_check_id_cstr(name, len, NULL);
785 }
786 else {
787 VALUE vbuf = 0;
788 char *buf = ALLOCV_N(char, vbuf, len+1);
789 buf[0] = '$';
790 memcpy(buf+1, name, len);
791 id = rb_check_id_cstr(buf, len+1, NULL);
792 ALLOCV_END(vbuf);
793 }
794
795 return id;
796}
797
798void
800 const char *name,
801 VALUE *var,
802 rb_gvar_getter_t *getter,
803 rb_gvar_setter_t *setter)
804{
805 volatile VALUE tmp = var ? *var : Qnil;
806 ID id = global_id(name);
807 struct rb_global_variable *gvar = rb_global_entry(id)->var;
808
809 gvar->data = (void*)var;
810 gvar->getter = getter ? (rb_gvar_getter_t *)getter : rb_gvar_var_getter;
811 gvar->setter = setter ? (rb_gvar_setter_t *)setter : rb_gvar_var_setter;
812 gvar->marker = rb_gvar_var_marker;
813
814 RB_GC_GUARD(tmp);
815}
816
817void
818rb_define_variable(const char *name, VALUE *var)
819{
820 rb_define_hooked_variable(name, var, 0, 0);
821}
822
823void
824rb_define_readonly_variable(const char *name, const VALUE *var)
825{
827}
828
829void
831 const char *name,
832 rb_gvar_getter_t *getter,
833 rb_gvar_setter_t *setter)
834{
835 if (!getter) getter = rb_gvar_val_getter;
836 if (!setter) setter = rb_gvar_readonly_setter;
837 rb_define_hooked_variable(name, 0, getter, setter);
838}
839
840static void
841rb_trace_eval(VALUE cmd, VALUE val)
842{
844}
845
846VALUE
847rb_f_trace_var(int argc, const VALUE *argv)
848{
849 VALUE var, cmd;
850 struct rb_global_entry *entry;
851 struct trace_var *trace;
852
853 if (rb_scan_args(argc, argv, "11", &var, &cmd) == 1) {
854 cmd = rb_block_proc();
855 }
856 if (NIL_P(cmd)) {
857 return rb_f_untrace_var(argc, argv);
858 }
859 entry = rb_global_entry(rb_to_id(var));
860 trace = ALLOC(struct trace_var);
861 trace->next = entry->var->trace;
862 trace->func = rb_trace_eval;
863 trace->data = cmd;
864 trace->removed = 0;
865 entry->var->trace = trace;
866
867 return Qnil;
868}
869
870static void
871remove_trace(struct rb_global_variable *var)
872{
873 struct trace_var *trace = var->trace;
874 struct trace_var t;
875 struct trace_var *next;
876
877 t.next = trace;
878 trace = &t;
879 while (trace->next) {
880 next = trace->next;
881 if (next->removed) {
882 trace->next = next->next;
883 xfree(next);
884 }
885 else {
886 trace = next;
887 }
888 }
889 var->trace = t.next;
890}
891
892VALUE
893rb_f_untrace_var(int argc, const VALUE *argv)
894{
895 VALUE var, cmd;
896 ID id;
897 struct rb_global_entry *entry;
898 struct trace_var *trace;
899
900 rb_scan_args(argc, argv, "11", &var, &cmd);
901 id = rb_check_id(&var);
902 if (!id) {
903 rb_name_error_str(var, "undefined global variable %"PRIsVALUE"", QUOTE(var));
904 }
905 if ((entry = rb_find_global_entry(id)) == NULL) {
906 rb_name_error(id, "undefined global variable %"PRIsVALUE"", QUOTE_ID(id));
907 }
908
909 trace = entry->var->trace;
910 if (NIL_P(cmd)) {
911 VALUE ary = rb_ary_new();
912
913 while (trace) {
914 struct trace_var *next = trace->next;
915 rb_ary_push(ary, (VALUE)trace->data);
916 trace->removed = 1;
917 trace = next;
918 }
919
920 if (!entry->var->block_trace) remove_trace(entry->var);
921 return ary;
922 }
923 else {
924 while (trace) {
925 if (trace->data == cmd) {
926 trace->removed = 1;
927 if (!entry->var->block_trace) remove_trace(entry->var);
928 return rb_ary_new3(1, cmd);
929 }
930 trace = trace->next;
931 }
932 }
933 return Qnil;
934}
935
937 struct trace_var *trace;
938 VALUE val;
939};
940
941static VALUE
942trace_ev(VALUE v)
943{
944 struct trace_data *data = (void *)v;
945 struct trace_var *trace = data->trace;
946
947 while (trace) {
948 (*trace->func)(trace->data, data->val);
949 trace = trace->next;
950 }
951
952 return Qnil;
953}
954
955static VALUE
956trace_en(VALUE v)
957{
958 struct rb_global_variable *var = (void *)v;
959 var->block_trace = 0;
960 remove_trace(var);
961 return Qnil; /* not reached */
962}
963
964static VALUE
965rb_gvar_set_entry(struct rb_global_entry *entry, VALUE val)
966{
967 struct trace_data trace;
968 struct rb_global_variable *var = entry->var;
969
970 (*var->setter)(val, entry->id, var->data);
971
972 if (var->trace && !var->block_trace) {
973 var->block_trace = 1;
974 trace.trace = var->trace;
975 trace.val = val;
976 rb_ensure(trace_ev, (VALUE)&trace, trace_en, (VALUE)var);
977 }
978 return val;
979}
980
981VALUE
982rb_gvar_set(ID id, VALUE val)
983{
984 struct rb_global_entry *entry;
985 entry = rb_global_entry(id);
986
987 return rb_gvar_set_entry(entry, val);
988}
989
990VALUE
991rb_gv_set(const char *name, VALUE val)
992{
993 return rb_gvar_set(global_id(name), val);
994}
995
996VALUE
997rb_gvar_get(ID id)
998{
999 struct rb_global_entry *entry = rb_global_entry(id);
1000 struct rb_global_variable *var = entry->var;
1001 return (*var->getter)(entry->id, var->data);
1002}
1003
1004VALUE
1005rb_gv_get(const char *name)
1006{
1007 ID id = find_global_id(name);
1008
1009 if (!id) {
1010 rb_warning("global variable '%s' not initialized", name);
1011 return Qnil;
1012 }
1013
1014 return rb_gvar_get(id);
1015}
1016
1017VALUE
1018rb_gvar_defined(ID id)
1019{
1020 struct rb_global_entry *entry = rb_global_entry(id);
1021 return RBOOL(entry->var->getter != rb_gvar_undef_getter);
1022}
1023
1025rb_gvar_getter_function_of(ID id)
1026{
1027 const struct rb_global_entry *entry = rb_global_entry(id);
1028 return entry->var->getter;
1029}
1030
1032rb_gvar_setter_function_of(ID id)
1033{
1034 const struct rb_global_entry *entry = rb_global_entry(id);
1035 return entry->var->setter;
1036}
1037
1038static enum rb_id_table_iterator_result
1039gvar_i(ID key, VALUE val, void *a)
1040{
1041 VALUE ary = (VALUE)a;
1042 rb_ary_push(ary, ID2SYM(key));
1043 return ID_TABLE_CONTINUE;
1044}
1045
1046VALUE
1048{
1049 VALUE ary = rb_ary_new();
1050 VALUE sym, backref = rb_backref_get();
1051
1052 if (!rb_ractor_main_p()) {
1053 rb_raise(rb_eRactorIsolationError, "can not access global variables from non-main Ractors");
1054 }
1055
1056 rb_id_table_foreach(rb_global_tbl, gvar_i, (void *)ary);
1057 if (!NIL_P(backref)) {
1058 char buf[2];
1059 int i, nmatch = rb_match_count(backref);
1060 buf[0] = '$';
1061 for (i = 1; i <= nmatch; ++i) {
1062 if (!RTEST(rb_reg_nth_defined(i, backref))) continue;
1063 if (i < 10) {
1064 /* probably reused, make static ID */
1065 buf[1] = (char)(i + '0');
1066 sym = ID2SYM(rb_intern2(buf, 2));
1067 }
1068 else {
1069 /* dynamic symbol */
1070 sym = rb_str_intern(rb_sprintf("$%d", i));
1071 }
1072 rb_ary_push(ary, sym);
1073 }
1074 }
1075 return ary;
1076}
1077
1078void
1080{
1081 struct rb_global_entry *entry1, *entry2;
1082 VALUE data1;
1083 struct rb_id_table *gtbl = rb_global_tbl;
1084
1085 if (!rb_ractor_main_p()) {
1086 rb_raise(rb_eRactorIsolationError, "can not access global variables from non-main Ractors");
1087 }
1088
1089 entry2 = rb_global_entry(name2);
1090 if (!rb_id_table_lookup(gtbl, name1, &data1)) {
1091 entry1 = ALLOC(struct rb_global_entry);
1092 entry1->id = name1;
1093 rb_id_table_insert(gtbl, name1, (VALUE)entry1);
1094 }
1095 else if ((entry1 = (struct rb_global_entry *)data1)->var != entry2->var) {
1096 struct rb_global_variable *var = entry1->var;
1097 if (var->block_trace) {
1098 rb_raise(rb_eRuntimeError, "can't alias in tracer");
1099 }
1100 var->counter--;
1101 if (var->counter == 0) {
1102 free_global_variable(var);
1103 }
1104 }
1105 else {
1106 return;
1107 }
1108 entry2->var->counter++;
1109 entry1->var = entry2->var;
1110}
1111
1112static void
1113IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(ID id)
1114{
1115 if (UNLIKELY(!rb_ractor_main_p())) {
1116 if (rb_is_instance_id(id)) { // check only normal ivars
1117 rb_raise(rb_eRactorIsolationError, "can not set instance variables of classes/modules by non-main Ractors");
1118 }
1119 }
1120}
1121
1122#define CVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR() \
1123 if (UNLIKELY(!rb_ractor_main_p())) { \
1124 rb_raise(rb_eRactorIsolationError, "can not access class variables from non-main Ractors"); \
1125 }
1126
1127static inline struct st_table *
1128generic_ivtbl(VALUE obj, ID id, bool force_check_ractor)
1129{
1130 ASSERT_vm_locking();
1131
1132 if ((force_check_ractor || LIKELY(rb_is_instance_id(id)) /* not internal ID */ ) &&
1133 !RB_OBJ_FROZEN_RAW(obj) &&
1134 UNLIKELY(!rb_ractor_main_p()) &&
1135 UNLIKELY(rb_ractor_shareable_p(obj))) {
1136
1137 rb_raise(rb_eRactorIsolationError, "can not access instance variables of shareable objects from non-main Ractors");
1138 }
1139 return generic_iv_tbl_;
1140}
1141
1142static inline struct st_table *
1143generic_ivtbl_no_ractor_check(VALUE obj)
1144{
1145 return generic_ivtbl(obj, 0, false);
1146}
1147
1148struct st_table *
1149rb_generic_ivtbl_get(void)
1150{
1151 return generic_iv_tbl_;
1152}
1153
1154int
1155rb_gen_ivtbl_get(VALUE obj, ID id, struct gen_ivtbl **ivtbl)
1156{
1158
1159 st_data_t data;
1160 int r = 0;
1161
1162 RB_VM_LOCK_ENTER();
1163 {
1164 if (st_lookup(generic_ivtbl(obj, id, false), (st_data_t)obj, &data)) {
1165 *ivtbl = (struct gen_ivtbl *)data;
1166 r = 1;
1167 }
1168 }
1169 RB_VM_LOCK_LEAVE();
1170
1171 return r;
1172}
1173
1174int
1175rb_ivar_generic_ivtbl_lookup(VALUE obj, struct gen_ivtbl **ivtbl)
1176{
1177 return rb_gen_ivtbl_get(obj, 0, ivtbl);
1178}
1179
1180static size_t
1181gen_ivtbl_bytes(size_t n)
1182{
1183 return offsetof(struct gen_ivtbl, as.shape.ivptr) + n * sizeof(VALUE);
1184}
1185
1186static struct gen_ivtbl *
1187gen_ivtbl_resize(struct gen_ivtbl *old, uint32_t n)
1188{
1189 RUBY_ASSERT(n > 0);
1190
1191 uint32_t len = old ? old->as.shape.numiv : 0;
1192 struct gen_ivtbl *ivtbl = xrealloc(old, gen_ivtbl_bytes(n));
1193
1194 ivtbl->as.shape.numiv = n;
1195 for (; len < n; len++) {
1196 ivtbl->as.shape.ivptr[len] = Qundef;
1197 }
1198
1199 return ivtbl;
1200}
1201
1202void
1203rb_mark_generic_ivar(VALUE obj)
1204{
1205 st_data_t data;
1206 if (st_lookup(generic_ivtbl_no_ractor_check(obj), (st_data_t)obj, &data)) {
1207 struct gen_ivtbl *ivtbl = (struct gen_ivtbl *)data;
1208 if (rb_shape_obj_too_complex(obj)) {
1209 rb_mark_tbl_no_pin(ivtbl->as.complex.table);
1210 }
1211 else {
1212 for (uint32_t i = 0; i < ivtbl->as.shape.numiv; i++) {
1213 rb_gc_mark_movable(ivtbl->as.shape.ivptr[i]);
1214 }
1215 }
1216 }
1217}
1218
1219void
1221{
1222 st_data_t key = (st_data_t)obj, value;
1223
1224 bool too_complex = rb_shape_obj_too_complex(obj);
1225
1226 if (st_delete(generic_ivtbl_no_ractor_check(obj), &key, &value)) {
1227 struct gen_ivtbl *ivtbl = (struct gen_ivtbl *)value;
1228
1229 if (UNLIKELY(too_complex)) {
1230 st_free_table(ivtbl->as.complex.table);
1231 }
1232
1233 xfree(ivtbl);
1234 }
1235}
1236
1237size_t
1238rb_generic_ivar_memsize(VALUE obj)
1239{
1240 struct gen_ivtbl *ivtbl;
1241
1242 if (rb_gen_ivtbl_get(obj, 0, &ivtbl)) {
1243 if (rb_shape_obj_too_complex(obj)) {
1244 return sizeof(struct gen_ivtbl) + st_memsize(ivtbl->as.complex.table);
1245 }
1246 else {
1247 return gen_ivtbl_bytes(ivtbl->as.shape.numiv);
1248 }
1249 }
1250 return 0;
1251}
1252
1253#if !SHAPE_IN_BASIC_FLAGS
1254shape_id_t
1255rb_generic_shape_id(VALUE obj)
1256{
1257 struct gen_ivtbl *ivtbl = 0;
1258 shape_id_t shape_id = 0;
1259
1260 RB_VM_LOCK_ENTER();
1261 {
1262 st_table* global_iv_table = generic_ivtbl(obj, 0, false);
1263
1264 if (global_iv_table && st_lookup(global_iv_table, obj, (st_data_t *)&ivtbl)) {
1265 shape_id = ivtbl->shape_id;
1266 }
1267 else if (OBJ_FROZEN(obj)) {
1268 shape_id = SPECIAL_CONST_SHAPE_ID;
1269 }
1270 }
1271 RB_VM_LOCK_LEAVE();
1272
1273 return shape_id;
1274}
1275#endif
1276
1277static size_t
1278gen_ivtbl_count(VALUE obj, const struct gen_ivtbl *ivtbl)
1279{
1280 uint32_t i;
1281 size_t n = 0;
1282
1283 if (rb_shape_obj_too_complex(obj)) {
1284 n = st_table_size(ivtbl->as.complex.table);
1285 }
1286 else {
1287 for (i = 0; i < ivtbl->as.shape.numiv; i++) {
1288 if (!UNDEF_P(ivtbl->as.shape.ivptr[i])) {
1289 n++;
1290 }
1291 }
1292 }
1293
1294 return n;
1295}
1296
1297VALUE
1298rb_ivar_lookup(VALUE obj, ID id, VALUE undef)
1299{
1300 if (SPECIAL_CONST_P(obj)) return undef;
1301
1302 shape_id_t shape_id;
1303 VALUE * ivar_list;
1304 rb_shape_t * shape;
1305
1306#if SHAPE_IN_BASIC_FLAGS
1307 shape_id = RBASIC_SHAPE_ID(obj);
1308#endif
1309
1310 switch (BUILTIN_TYPE(obj)) {
1311 case T_CLASS:
1312 case T_MODULE:
1313 {
1314 bool found = false;
1315 VALUE val;
1316
1317 RB_VM_LOCK_ENTER();
1318 {
1319#if !SHAPE_IN_BASIC_FLAGS
1320 shape_id = RCLASS_SHAPE_ID(obj);
1321#endif
1322
1323 if (rb_shape_obj_too_complex(obj)) {
1324 st_table * iv_table = RCLASS_IV_HASH(obj);
1325 if (rb_st_lookup(iv_table, (st_data_t)id, (st_data_t *)&val)) {
1326 found = true;
1327 }
1328 else {
1329 val = undef;
1330 }
1331 }
1332 else {
1333 attr_index_t index = 0;
1334 shape = rb_shape_get_shape_by_id(shape_id);
1335 found = rb_shape_get_iv_index(shape, id, &index);
1336
1337 if (found) {
1338 ivar_list = RCLASS_IVPTR(obj);
1339 RUBY_ASSERT(ivar_list);
1340
1341 val = ivar_list[index];
1342 }
1343 else {
1344 val = undef;
1345 }
1346 }
1347 }
1348 RB_VM_LOCK_LEAVE();
1349
1350 if (found &&
1351 rb_is_instance_id(id) &&
1352 UNLIKELY(!rb_ractor_main_p()) &&
1353 !rb_ractor_shareable_p(val)) {
1354 rb_raise(rb_eRactorIsolationError,
1355 "can not get unshareable values from instance variables of classes/modules from non-main Ractors");
1356 }
1357 return val;
1358 }
1359 case T_OBJECT:
1360 {
1361#if !SHAPE_IN_BASIC_FLAGS
1362 shape_id = ROBJECT_SHAPE_ID(obj);
1363#endif
1364 if (rb_shape_obj_too_complex(obj)) {
1365 st_table * iv_table = ROBJECT_IV_HASH(obj);
1366 VALUE val;
1367 if (rb_st_lookup(iv_table, (st_data_t)id, (st_data_t *)&val)) {
1368 return val;
1369 }
1370 else {
1371 return undef;
1372 }
1373 }
1374
1375 RUBY_ASSERT(!rb_shape_obj_too_complex(obj));
1376 ivar_list = ROBJECT_IVPTR(obj);
1377 break;
1378 }
1379 default:
1380 if (FL_TEST_RAW(obj, FL_EXIVAR)) {
1381 struct gen_ivtbl *ivtbl;
1382 rb_gen_ivtbl_get(obj, id, &ivtbl);
1383
1384 if (rb_shape_obj_too_complex(obj)) {
1385 VALUE val;
1386 if (rb_st_lookup(ivtbl->as.complex.table, (st_data_t)id, (st_data_t *)&val)) {
1387 return val;
1388 }
1389 else {
1390 return undef;
1391 }
1392 }
1393
1394#if !SHAPE_IN_BASIC_FLAGS
1395 shape_id = ivtbl->shape_id;
1396#endif
1397 ivar_list = ivtbl->as.shape.ivptr;
1398 }
1399 else {
1400 return undef;
1401 }
1402 break;
1403 }
1404
1405 attr_index_t index = 0;
1406 shape = rb_shape_get_shape_by_id(shape_id);
1407 if (rb_shape_get_iv_index(shape, id, &index)) {
1408 return ivar_list[index];
1409 }
1410
1411 return undef;
1412}
1413
1414VALUE
1416{
1417 VALUE iv = rb_ivar_lookup(obj, id, Qnil);
1418 RB_DEBUG_COUNTER_INC(ivar_get_base);
1419 return iv;
1420}
1421
1422VALUE
1423rb_attr_get(VALUE obj, ID id)
1424{
1425 return rb_ivar_lookup(obj, id, Qnil);
1426}
1427
1428static VALUE
1429rb_ivar_delete(VALUE obj, ID id, VALUE undef)
1430{
1431 rb_check_frozen(obj);
1432
1433 VALUE val = undef;
1434 rb_shape_t *shape = rb_shape_get_shape(obj);
1435
1436 if (BUILTIN_TYPE(obj) == T_CLASS || BUILTIN_TYPE(obj) == T_MODULE) {
1437 IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(id);
1438 }
1439
1440 if (!rb_shape_transition_shape_remove_ivar(obj, id, shape, &val)) {
1441 if (!rb_shape_obj_too_complex(obj)) {
1442 rb_evict_ivars_to_hash(obj);
1443 }
1444
1445 st_table *table = NULL;
1446 switch (BUILTIN_TYPE(obj)) {
1447 case T_CLASS:
1448 case T_MODULE:
1449 table = RCLASS_IV_HASH(obj);
1450 break;
1451
1452 case T_OBJECT:
1453 table = ROBJECT_IV_HASH(obj);
1454 break;
1455
1456 default: {
1457 struct gen_ivtbl *ivtbl;
1458 if (rb_gen_ivtbl_get(obj, 0, &ivtbl)) {
1459 table = ivtbl->as.complex.table;
1460 }
1461 break;
1462 }
1463 }
1464
1465 if (table) {
1466 if (!st_delete(table, (st_data_t *)&id, (st_data_t *)&val)) {
1467 val = undef;
1468 }
1469 }
1470 }
1471
1472 return val;
1473}
1474
1475VALUE
1476rb_attr_delete(VALUE obj, ID id)
1477{
1478 return rb_ivar_delete(obj, id, Qnil);
1479}
1480
1481void
1482rb_obj_convert_to_too_complex(VALUE obj, st_table *table)
1483{
1484 RUBY_ASSERT(!rb_shape_obj_too_complex(obj));
1485
1486 VALUE *old_ivptr = NULL;
1487
1488 switch (BUILTIN_TYPE(obj)) {
1489 case T_OBJECT:
1490 if (!(RBASIC(obj)->flags & ROBJECT_EMBED)) {
1491 old_ivptr = ROBJECT_IVPTR(obj);
1492 }
1493 rb_shape_set_shape_id(obj, OBJ_TOO_COMPLEX_SHAPE_ID);
1494 ROBJECT_SET_IV_HASH(obj, table);
1495 break;
1496 case T_CLASS:
1497 case T_MODULE:
1498 old_ivptr = RCLASS_IVPTR(obj);
1499 rb_shape_set_shape_id(obj, OBJ_TOO_COMPLEX_SHAPE_ID);
1500 RCLASS_SET_IV_HASH(obj, table);
1501 break;
1502 default:
1503 RB_VM_LOCK_ENTER();
1504 {
1505 struct st_table *gen_ivs = generic_ivtbl_no_ractor_check(obj);
1506
1507 struct gen_ivtbl *old_ivtbl = NULL;
1508 st_lookup(gen_ivs, (st_data_t)obj, (st_data_t *)&old_ivtbl);
1509
1510 if (old_ivtbl) {
1511 /* We need to modify old_ivtbl to have the too complex shape
1512 * and hold the table because the xmalloc could trigger a GC
1513 * compaction. We want the table to be updated rather than
1514 * the original ivptr. */
1515#if SHAPE_IN_BASIC_FLAGS
1516 rb_shape_set_shape_id(obj, OBJ_TOO_COMPLEX_SHAPE_ID);
1517#else
1518 old_ivtbl->shape_id = OBJ_TOO_COMPLEX_SHAPE_ID;
1519#endif
1520 old_ivtbl->as.complex.table = table;
1521 old_ivptr = (VALUE *)old_ivtbl;
1522 }
1523
1524 struct gen_ivtbl *ivtbl = xmalloc(sizeof(struct gen_ivtbl));
1525 ivtbl->as.complex.table = table;
1526 st_insert(gen_ivs, (st_data_t)obj, (st_data_t)ivtbl);
1527#if SHAPE_IN_BASIC_FLAGS
1528 rb_shape_set_shape_id(obj, OBJ_TOO_COMPLEX_SHAPE_ID);
1529#else
1530 ivtbl->shape_id = OBJ_TOO_COMPLEX_SHAPE_ID;
1531#endif
1532 }
1533 RB_VM_LOCK_LEAVE();
1534 }
1535
1536 xfree(old_ivptr);
1537}
1538
1539void
1540rb_evict_ivars_to_hash(VALUE obj)
1541{
1542 RUBY_ASSERT(!rb_shape_obj_too_complex(obj));
1543
1544 st_table *table = st_init_numtable_with_size(rb_ivar_count(obj));
1545
1546 // Evacuate all previous values from shape into id_table
1547 rb_obj_copy_ivs_to_hash_table(obj, table);
1548 rb_obj_convert_to_too_complex(obj, table);
1549
1550 RUBY_ASSERT(rb_shape_obj_too_complex(obj));
1551}
1552
1554 attr_index_t index;
1555 bool existing;
1556};
1557
1558static struct general_ivar_set_result
1559general_ivar_set(VALUE obj, ID id, VALUE val, void *data,
1560 VALUE *(*shape_ivptr_func)(VALUE, void *),
1561 void (*shape_resize_ivptr_func)(VALUE, attr_index_t, attr_index_t, void *),
1562 void (*set_shape_func)(VALUE, rb_shape_t *, void *),
1563 void (*transition_too_complex_func)(VALUE, void *),
1564 st_table *(*too_complex_table_func)(VALUE, void *))
1565{
1566 struct general_ivar_set_result result = {
1567 .index = 0,
1568 .existing = true
1569 };
1570
1571 rb_shape_t *current_shape = rb_shape_get_shape(obj);
1572
1573 if (UNLIKELY(current_shape->type == SHAPE_OBJ_TOO_COMPLEX)) {
1574 goto too_complex;
1575 }
1576
1577 attr_index_t index;
1578 if (!rb_shape_get_iv_index(current_shape, id, &index)) {
1579 result.existing = false;
1580
1581 index = current_shape->next_iv_index;
1582 if (index >= MAX_IVARS) {
1583 rb_raise(rb_eArgError, "too many instance variables");
1584 }
1585
1586 rb_shape_t *next_shape = rb_shape_get_next(current_shape, obj, id);
1587 if (UNLIKELY(next_shape->type == SHAPE_OBJ_TOO_COMPLEX)) {
1588 transition_too_complex_func(obj, data);
1589 goto too_complex;
1590 }
1591 else if (UNLIKELY(next_shape->capacity != current_shape->capacity)) {
1592 RUBY_ASSERT(next_shape->capacity > current_shape->capacity);
1593 shape_resize_ivptr_func(obj, current_shape->capacity, next_shape->capacity, data);
1594 }
1595
1596 RUBY_ASSERT(next_shape->type == SHAPE_IVAR);
1597 RUBY_ASSERT(index == (next_shape->next_iv_index - 1));
1598 set_shape_func(obj, next_shape, data);
1599 }
1600
1601 VALUE *table = shape_ivptr_func(obj, data);
1602 RB_OBJ_WRITE(obj, &table[index], val);
1603
1604 result.index = index;
1605 return result;
1606
1607too_complex:
1608 {
1609 RUBY_ASSERT(rb_shape_obj_too_complex(obj));
1610
1611 st_table *table = too_complex_table_func(obj, data);
1612 result.existing = st_insert(table, (st_data_t)id, (st_data_t)val);
1613 result.index = 0;
1614 RB_OBJ_WRITTEN(obj, Qundef, val);
1615 }
1616 return result;
1617}
1618
1620 VALUE obj;
1621 ID id;
1622 struct gen_ivtbl *ivtbl;
1623 rb_shape_t *shape;
1624 bool resize;
1625};
1626
1627static int
1628generic_ivar_lookup_ensure_size(st_data_t *k, st_data_t *v, st_data_t u, int existing)
1629{
1630 ASSERT_vm_locking();
1631
1632 struct gen_ivar_lookup_ensure_size *ivar_lookup = (struct gen_ivar_lookup_ensure_size *)u;
1633 struct gen_ivtbl *ivtbl = existing ? (struct gen_ivtbl *)*v : NULL;
1634
1635 if (!existing || ivar_lookup->resize) {
1636 if (existing) {
1637 RUBY_ASSERT(ivar_lookup->shape->type == SHAPE_IVAR);
1638 RUBY_ASSERT(rb_shape_get_shape_by_id(ivar_lookup->shape->parent_id)->capacity < ivar_lookup->shape->capacity);
1639 }
1640 else {
1642 }
1643
1644 ivtbl = gen_ivtbl_resize(ivtbl, ivar_lookup->shape->capacity);
1645 *v = (st_data_t)ivtbl;
1646 }
1647
1649
1650 ivar_lookup->ivtbl = ivtbl;
1651 if (ivar_lookup->shape) {
1652#if SHAPE_IN_BASIC_FLAGS
1653 rb_shape_set_shape(ivar_lookup->obj, ivar_lookup->shape);
1654#else
1655 ivtbl->shape_id = rb_shape_id(ivar_lookup->shape);
1656#endif
1657 }
1658
1659 return ST_CONTINUE;
1660}
1661
1662static VALUE *
1663generic_ivar_set_shape_ivptr(VALUE obj, void *data)
1664{
1665 RUBY_ASSERT(!rb_shape_obj_too_complex(obj));
1666
1667 struct gen_ivar_lookup_ensure_size *ivar_lookup = data;
1668
1669 RB_VM_LOCK_ENTER();
1670 {
1671 st_update(generic_ivtbl(obj, ivar_lookup->id, false), (st_data_t)obj, generic_ivar_lookup_ensure_size, (st_data_t)ivar_lookup);
1672 }
1673 RB_VM_LOCK_LEAVE();
1674
1675 FL_SET_RAW(obj, FL_EXIVAR);
1676
1677 return ivar_lookup->ivtbl->as.shape.ivptr;
1678}
1679
1680static void
1681generic_ivar_set_shape_resize_ivptr(VALUE obj, attr_index_t _old_capa, attr_index_t new_capa, void *data)
1682{
1683 struct gen_ivar_lookup_ensure_size *ivar_lookup = data;
1684
1685 ivar_lookup->resize = true;
1686}
1687
1688static void
1689generic_ivar_set_set_shape(VALUE obj, rb_shape_t *shape, void *data)
1690{
1691 struct gen_ivar_lookup_ensure_size *ivar_lookup = data;
1692
1693 ivar_lookup->shape = shape;
1694}
1695
1696static void
1697generic_ivar_set_transition_too_complex(VALUE obj, void *_data)
1698{
1699 rb_evict_ivars_to_hash(obj);
1700 FL_SET_RAW(obj, FL_EXIVAR);
1701}
1702
1703static st_table *
1704generic_ivar_set_too_complex_table(VALUE obj, void *data)
1705{
1706 struct gen_ivar_lookup_ensure_size *ivar_lookup = data;
1707
1708 struct gen_ivtbl *ivtbl;
1709 if (!rb_gen_ivtbl_get(obj, 0, &ivtbl)) {
1710 ivtbl = xmalloc(sizeof(struct gen_ivtbl));
1711#if !SHAPE_IN_BASIC_FLAGS
1712 ivtbl->shape_id = SHAPE_OBJ_TOO_COMPLEX;
1713#endif
1714 ivtbl->as.complex.table = st_init_numtable_with_size(1);
1715
1716 RB_VM_LOCK_ENTER();
1717 {
1718 st_insert(generic_ivtbl(obj, ivar_lookup->id, false), (st_data_t)obj, (st_data_t)ivtbl);
1719 }
1720 RB_VM_LOCK_LEAVE();
1721
1722 FL_SET_RAW(obj, FL_EXIVAR);
1723 }
1724
1725 RUBY_ASSERT(rb_shape_obj_too_complex(obj));
1726
1727 return ivtbl->as.complex.table;
1728}
1729
1730static void
1731generic_ivar_set(VALUE obj, ID id, VALUE val)
1732{
1733 struct gen_ivar_lookup_ensure_size ivar_lookup = {
1734 .obj = obj,
1735 .id = id,
1736 .resize = false,
1737 .shape = NULL,
1738 };
1739
1740 general_ivar_set(obj, id, val, &ivar_lookup,
1741 generic_ivar_set_shape_ivptr,
1742 generic_ivar_set_shape_resize_ivptr,
1743 generic_ivar_set_set_shape,
1744 generic_ivar_set_transition_too_complex,
1745 generic_ivar_set_too_complex_table);
1746}
1747
1748void
1749rb_ensure_iv_list_size(VALUE obj, uint32_t current_capacity, uint32_t new_capacity)
1750{
1751 RUBY_ASSERT(!rb_shape_obj_too_complex(obj));
1752
1753 if (RBASIC(obj)->flags & ROBJECT_EMBED) {
1754 VALUE *ptr = ROBJECT_IVPTR(obj);
1755 VALUE *newptr = ALLOC_N(VALUE, new_capacity);
1756 MEMCPY(newptr, ptr, VALUE, current_capacity);
1757 RB_FL_UNSET_RAW(obj, ROBJECT_EMBED);
1758 ROBJECT(obj)->as.heap.ivptr = newptr;
1759 }
1760 else {
1761 REALLOC_N(ROBJECT(obj)->as.heap.ivptr, VALUE, new_capacity);
1762 }
1763}
1764
1765static int
1766rb_obj_copy_ivs_to_hash_table_i(ID key, VALUE val, st_data_t arg)
1767{
1768 RUBY_ASSERT(!st_lookup((st_table *)arg, (st_data_t)key, NULL));
1769
1770 st_add_direct((st_table *)arg, (st_data_t)key, (st_data_t)val);
1771 return ST_CONTINUE;
1772}
1773
1774void
1775rb_obj_copy_ivs_to_hash_table(VALUE obj, st_table *table)
1776{
1777 rb_ivar_foreach(obj, rb_obj_copy_ivs_to_hash_table_i, (st_data_t)table);
1778}
1779
1780static VALUE *
1781obj_ivar_set_shape_ivptr(VALUE obj, void *_data)
1782{
1783 RUBY_ASSERT(!rb_shape_obj_too_complex(obj));
1784
1785 return ROBJECT_IVPTR(obj);
1786}
1787
1788static void
1789obj_ivar_set_shape_resize_ivptr(VALUE obj, attr_index_t old_capa, attr_index_t new_capa, void *_data)
1790{
1791 rb_ensure_iv_list_size(obj, old_capa, new_capa);
1792}
1793
1794static void
1795obj_ivar_set_set_shape(VALUE obj, rb_shape_t *shape, void *_data)
1796{
1797 rb_shape_set_shape(obj, shape);
1798}
1799
1800static void
1801obj_ivar_set_transition_too_complex(VALUE obj, void *_data)
1802{
1803 rb_evict_ivars_to_hash(obj);
1804}
1805
1806static st_table *
1807obj_ivar_set_too_complex_table(VALUE obj, void *_data)
1808{
1809 RUBY_ASSERT(rb_shape_obj_too_complex(obj));
1810
1811 return ROBJECT_IV_HASH(obj);
1812}
1813
1814attr_index_t
1815rb_obj_ivar_set(VALUE obj, ID id, VALUE val)
1816{
1817 return general_ivar_set(obj, id, val, NULL,
1818 obj_ivar_set_shape_ivptr,
1819 obj_ivar_set_shape_resize_ivptr,
1820 obj_ivar_set_set_shape,
1821 obj_ivar_set_transition_too_complex,
1822 obj_ivar_set_too_complex_table).index;
1823}
1824
1825/* Set the instance variable +val+ on object +obj+ at ivar name +id+.
1826 * This function only works with T_OBJECT objects, so make sure
1827 * +obj+ is of type T_OBJECT before using this function.
1828 */
1829VALUE
1830rb_vm_set_ivar_id(VALUE obj, ID id, VALUE val)
1831{
1832 rb_check_frozen(obj);
1833 rb_obj_ivar_set(obj, id, val);
1834 return val;
1835}
1836
1837bool
1838rb_shape_set_shape_id(VALUE obj, shape_id_t shape_id)
1839{
1840 if (rb_shape_get_shape_id(obj) == shape_id) {
1841 return false;
1842 }
1843
1844#if SHAPE_IN_BASIC_FLAGS
1845 RBASIC_SET_SHAPE_ID(obj, shape_id);
1846#else
1847 switch (BUILTIN_TYPE(obj)) {
1848 case T_OBJECT:
1849 ROBJECT_SET_SHAPE_ID(obj, shape_id);
1850 break;
1851 case T_CLASS:
1852 case T_MODULE:
1853 RCLASS_SET_SHAPE_ID(obj, shape_id);
1854 break;
1855 default:
1856 if (shape_id != SPECIAL_CONST_SHAPE_ID) {
1857 struct gen_ivtbl *ivtbl = 0;
1858 RB_VM_LOCK_ENTER();
1859 {
1860 st_table* global_iv_table = generic_ivtbl(obj, 0, false);
1861
1862 if (st_lookup(global_iv_table, obj, (st_data_t *)&ivtbl)) {
1863 ivtbl->shape_id = shape_id;
1864 }
1865 else {
1866 rb_bug("Expected shape_id entry in global iv table");
1867 }
1868 }
1869 RB_VM_LOCK_LEAVE();
1870 }
1871 }
1872#endif
1873
1874 return true;
1875}
1876
1878{
1879 if (RB_FL_ABLE(x)) {
1881 if (TYPE(x) == T_STRING) {
1882 RB_FL_UNSET_RAW(x, FL_USER2 | FL_USER3); // STR_CHILLED
1883 }
1884
1885 rb_shape_t * next_shape = rb_shape_transition_shape_frozen(x);
1886
1887 // If we're transitioning from "not complex" to "too complex"
1888 // then evict ivars. This can happen if we run out of shapes
1889 if (!rb_shape_obj_too_complex(x) && next_shape->type == SHAPE_OBJ_TOO_COMPLEX) {
1890 rb_evict_ivars_to_hash(x);
1891 }
1892 rb_shape_set_shape(x, next_shape);
1893
1894 if (RBASIC_CLASS(x)) {
1896 }
1897 }
1898}
1899
1900static void
1901ivar_set(VALUE obj, ID id, VALUE val)
1902{
1903 RB_DEBUG_COUNTER_INC(ivar_set_base);
1904
1905 switch (BUILTIN_TYPE(obj)) {
1906 case T_OBJECT:
1907 {
1908 rb_obj_ivar_set(obj, id, val);
1909 break;
1910 }
1911 case T_CLASS:
1912 case T_MODULE:
1913 IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(id);
1914 rb_class_ivar_set(obj, id, val);
1915
1916 break;
1917 default:
1918 generic_ivar_set(obj, id, val);
1919 break;
1920 }
1921}
1922
1923VALUE
1925{
1926 rb_check_frozen(obj);
1927 ivar_set(obj, id, val);
1928 return val;
1929}
1930
1931void
1932rb_ivar_set_internal(VALUE obj, ID id, VALUE val)
1933{
1934 // should be internal instance variable name (no @ prefix)
1935 VM_ASSERT(!rb_is_instance_id(id));
1936
1937 ivar_set(obj, id, val);
1938}
1939
1940VALUE
1942{
1943 attr_index_t index;
1944
1945 if (SPECIAL_CONST_P(obj)) return Qfalse;
1946 if (rb_shape_obj_too_complex(obj)) {
1947 VALUE idx;
1948 st_table *table = NULL;
1949 switch (BUILTIN_TYPE(obj)) {
1950 case T_CLASS:
1951 case T_MODULE:
1952 table = (st_table *)RCLASS_IVPTR(obj);
1953 break;
1954
1955 case T_OBJECT:
1956 table = ROBJECT_IV_HASH(obj);
1957 break;
1958
1959 default: {
1960 struct gen_ivtbl *ivtbl;
1961 if (rb_gen_ivtbl_get(obj, 0, &ivtbl)) {
1962 table = ivtbl->as.complex.table;
1963 }
1964 break;
1965 }
1966 }
1967
1968 if (!table || !rb_st_lookup(table, id, &idx)) {
1969 return Qfalse;
1970 }
1971
1972 return Qtrue;
1973 }
1974 else {
1975 return RBOOL(rb_shape_get_iv_index(rb_shape_get_shape(obj), id, &index));
1976 }
1977}
1978
1979typedef int rb_ivar_foreach_callback_func(ID key, VALUE val, st_data_t arg);
1980
1982 VALUE obj;
1983 struct gen_ivtbl * ivtbl;
1984 st_data_t arg;
1985 rb_ivar_foreach_callback_func *func;
1986};
1987
1988/*
1989 * Returns a flag to stop iterating depending on the result of +callback+.
1990 */
1991static bool
1992iterate_over_shapes_with_callback(rb_shape_t *shape, rb_ivar_foreach_callback_func *callback, struct iv_itr_data * itr_data)
1993{
1994 switch ((enum shape_type)shape->type) {
1995 case SHAPE_ROOT:
1996 case SHAPE_T_OBJECT:
1997 return false;
1998 case SHAPE_IVAR:
1999 ASSUME(callback);
2000 if (iterate_over_shapes_with_callback(rb_shape_get_parent(shape), callback, itr_data))
2001 return true;
2002 VALUE * iv_list;
2003 switch (BUILTIN_TYPE(itr_data->obj)) {
2004 case T_OBJECT:
2005 RUBY_ASSERT(!rb_shape_obj_too_complex(itr_data->obj));
2006 iv_list = ROBJECT_IVPTR(itr_data->obj);
2007 break;
2008 case T_CLASS:
2009 case T_MODULE:
2010 iv_list = RCLASS_IVPTR(itr_data->obj);
2011 break;
2012 default:
2013 iv_list = itr_data->ivtbl->as.shape.ivptr;
2014 break;
2015 }
2016 VALUE val = iv_list[shape->next_iv_index - 1];
2017 if (!UNDEF_P(val)) {
2018 switch (callback(shape->edge_name, val, itr_data->arg)) {
2019 case ST_CHECK:
2020 case ST_CONTINUE:
2021 break;
2022 case ST_STOP:
2023 return true;
2024 default:
2025 rb_bug("unreachable");
2026 }
2027 }
2028 return false;
2029 case SHAPE_FROZEN:
2030 return iterate_over_shapes_with_callback(rb_shape_get_parent(shape), callback, itr_data);
2031 case SHAPE_OBJ_TOO_COMPLEX:
2032 default:
2033 rb_bug("Unreachable");
2034 }
2035}
2036
2037static int
2038each_hash_iv(st_data_t id, st_data_t val, st_data_t data)
2039{
2040 struct iv_itr_data * itr_data = (struct iv_itr_data *)data;
2041 rb_ivar_foreach_callback_func *callback = itr_data->func;
2042 return callback((ID)id, (VALUE)val, itr_data->arg);
2043}
2044
2045static void
2046obj_ivar_each(VALUE obj, rb_ivar_foreach_callback_func *func, st_data_t arg)
2047{
2048 rb_shape_t* shape = rb_shape_get_shape(obj);
2049 struct iv_itr_data itr_data;
2050 itr_data.obj = obj;
2051 itr_data.arg = arg;
2052 itr_data.func = func;
2053 if (rb_shape_obj_too_complex(obj)) {
2054 rb_st_foreach(ROBJECT_IV_HASH(obj), each_hash_iv, (st_data_t)&itr_data);
2055 }
2056 else {
2057 iterate_over_shapes_with_callback(shape, func, &itr_data);
2058 }
2059}
2060
2061static void
2062gen_ivar_each(VALUE obj, rb_ivar_foreach_callback_func *func, st_data_t arg)
2063{
2064 rb_shape_t *shape = rb_shape_get_shape(obj);
2065 struct gen_ivtbl *ivtbl;
2066 if (!rb_gen_ivtbl_get(obj, 0, &ivtbl)) return;
2067
2068 struct iv_itr_data itr_data;
2069 itr_data.obj = obj;
2070 itr_data.ivtbl = ivtbl;
2071 itr_data.arg = arg;
2072 itr_data.func = func;
2073 if (rb_shape_obj_too_complex(obj)) {
2074 rb_st_foreach(ivtbl->as.complex.table, each_hash_iv, (st_data_t)&itr_data);
2075 }
2076 else {
2077 iterate_over_shapes_with_callback(shape, func, &itr_data);
2078 }
2079}
2080
2081static void
2082class_ivar_each(VALUE obj, rb_ivar_foreach_callback_func *func, st_data_t arg)
2083{
2085
2086 rb_shape_t* shape = rb_shape_get_shape(obj);
2087 struct iv_itr_data itr_data;
2088 itr_data.obj = obj;
2089 itr_data.arg = arg;
2090 itr_data.func = func;
2091 if (rb_shape_obj_too_complex(obj)) {
2092 rb_st_foreach(RCLASS_IV_HASH(obj), each_hash_iv, (st_data_t)&itr_data);
2093 }
2094 else {
2095 iterate_over_shapes_with_callback(shape, func, &itr_data);
2096 }
2097}
2098
2099void
2101{
2102 struct gen_ivtbl *obj_ivtbl;
2103 struct gen_ivtbl *new_ivtbl;
2104
2105 rb_check_frozen(clone);
2106
2107 if (!FL_TEST(obj, FL_EXIVAR)) {
2108 goto clear;
2109 }
2110
2111 if (rb_gen_ivtbl_get(obj, 0, &obj_ivtbl)) {
2112 if (gen_ivtbl_count(obj, obj_ivtbl) == 0)
2113 goto clear;
2114
2115 FL_SET(clone, FL_EXIVAR);
2116
2117 if (rb_shape_obj_too_complex(obj)) {
2118 new_ivtbl = xmalloc(sizeof(struct gen_ivtbl));
2119#if !SHAPE_IN_BASIC_FLAGS
2120 new_ivtbl->shape_id = SHAPE_OBJ_TOO_COMPLEX;
2121#endif
2122 new_ivtbl->as.complex.table = st_copy(obj_ivtbl->as.complex.table);
2123 }
2124 else {
2125 new_ivtbl = gen_ivtbl_resize(0, obj_ivtbl->as.shape.numiv);
2126
2127 for (uint32_t i=0; i<obj_ivtbl->as.shape.numiv; i++) {
2128 RB_OBJ_WRITE(clone, &new_ivtbl->as.shape.ivptr[i], obj_ivtbl->as.shape.ivptr[i]);
2129 }
2130 }
2131
2132 /*
2133 * c.ivtbl may change in gen_ivar_copy due to realloc,
2134 * no need to free
2135 */
2136 RB_VM_LOCK_ENTER();
2137 {
2138 generic_ivtbl_no_ractor_check(clone);
2139 st_insert(generic_ivtbl_no_ractor_check(obj), (st_data_t)clone, (st_data_t)new_ivtbl);
2140 }
2141 RB_VM_LOCK_LEAVE();
2142
2143 rb_shape_t * obj_shape = rb_shape_get_shape(obj);
2144 if (rb_shape_frozen_shape_p(obj_shape)) {
2145 rb_shape_set_shape_id(clone, obj_shape->parent_id);
2146 }
2147 else {
2148 rb_shape_set_shape(clone, obj_shape);
2149 }
2150 }
2151 return;
2152
2153 clear:
2154 if (FL_TEST(clone, FL_EXIVAR)) {
2155 rb_free_generic_ivar(clone);
2156 FL_UNSET(clone, FL_EXIVAR);
2157 }
2158}
2159
2160void
2161rb_ivar_foreach(VALUE obj, rb_ivar_foreach_callback_func *func, st_data_t arg)
2162{
2163 if (SPECIAL_CONST_P(obj)) return;
2164 switch (BUILTIN_TYPE(obj)) {
2165 case T_OBJECT:
2166 obj_ivar_each(obj, func, arg);
2167 break;
2168 case T_CLASS:
2169 case T_MODULE:
2170 IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(0);
2171 RB_VM_LOCK_ENTER();
2172 {
2173 class_ivar_each(obj, func, arg);
2174 }
2175 RB_VM_LOCK_LEAVE();
2176 break;
2177 default:
2178 if (FL_TEST(obj, FL_EXIVAR)) {
2179 gen_ivar_each(obj, func, arg);
2180 }
2181 break;
2182 }
2183}
2184
2185st_index_t
2187{
2188 if (SPECIAL_CONST_P(obj)) return 0;
2189
2190 switch (BUILTIN_TYPE(obj)) {
2191 case T_OBJECT:
2192 return ROBJECT_IV_COUNT(obj);
2193 case T_CLASS:
2194 case T_MODULE:
2195 return RCLASS_IV_COUNT(obj);
2196 default:
2197 if (FL_TEST(obj, FL_EXIVAR)) {
2198 struct gen_ivtbl *ivtbl;
2199
2200 if (rb_gen_ivtbl_get(obj, 0, &ivtbl)) {
2201 return gen_ivtbl_count(obj, ivtbl);
2202 }
2203 }
2204 break;
2205 }
2206 return 0;
2207}
2208
2209static int
2210ivar_i(ID key, VALUE v, st_data_t a)
2211{
2212 VALUE ary = (VALUE)a;
2213
2214 if (rb_is_instance_id(key)) {
2215 rb_ary_push(ary, ID2SYM(key));
2216 }
2217 return ST_CONTINUE;
2218}
2219
2220/*
2221 * call-seq:
2222 * obj.instance_variables -> array
2223 *
2224 * Returns an array of instance variable names for the receiver. Note
2225 * that simply defining an accessor does not create the corresponding
2226 * instance variable.
2227 *
2228 * class Fred
2229 * attr_accessor :a1
2230 * def initialize
2231 * @iv = 3
2232 * end
2233 * end
2234 * Fred.new.instance_variables #=> [:@iv]
2235 */
2236
2237VALUE
2239{
2240 VALUE ary;
2241
2242 ary = rb_ary_new();
2243 rb_ivar_foreach(obj, ivar_i, ary);
2244 return ary;
2245}
2246
2247#define rb_is_constant_id rb_is_const_id
2248#define rb_is_constant_name rb_is_const_name
2249#define id_for_var(obj, name, part, type) \
2250 id_for_var_message(obj, name, type, "'%1$s' is not allowed as "#part" "#type" variable name")
2251#define id_for_var_message(obj, name, type, message) \
2252 check_id_type(obj, &(name), rb_is_##type##_id, rb_is_##type##_name, message, strlen(message))
2253static ID
2254check_id_type(VALUE obj, VALUE *pname,
2255 int (*valid_id_p)(ID), int (*valid_name_p)(VALUE),
2256 const char *message, size_t message_len)
2257{
2258 ID id = rb_check_id(pname);
2259 VALUE name = *pname;
2260
2261 if (id ? !valid_id_p(id) : !valid_name_p(name)) {
2262 rb_name_err_raise_str(rb_fstring_new(message, message_len),
2263 obj, name);
2264 }
2265 return id;
2266}
2267
2268/*
2269 * call-seq:
2270 * obj.remove_instance_variable(symbol) -> obj
2271 * obj.remove_instance_variable(string) -> obj
2272 *
2273 * Removes the named instance variable from <i>obj</i>, returning that
2274 * variable's value. The name can be passed as a symbol or as a string.
2275 *
2276 * class Dummy
2277 * attr_reader :var
2278 * def initialize
2279 * @var = 99
2280 * end
2281 * def remove
2282 * remove_instance_variable(:@var)
2283 * end
2284 * end
2285 * d = Dummy.new
2286 * d.var #=> 99
2287 * d.remove #=> 99
2288 * d.var #=> nil
2289 */
2290
2291VALUE
2293{
2294 const ID id = id_for_var(obj, name, an, instance);
2295
2296 // Frozen check comes here because it's expected that we raise a
2297 // NameError (from the id_for_var check) before we raise a FrozenError
2298 rb_check_frozen(obj);
2299
2300 if (id) {
2301 VALUE val = rb_ivar_delete(obj, id, Qundef);
2302
2303 if (!UNDEF_P(val)) return val;
2304 }
2305
2306 rb_name_err_raise("instance variable %1$s not defined",
2307 obj, name);
2309}
2310
2311NORETURN(static void uninitialized_constant(VALUE, VALUE));
2312static void
2313uninitialized_constant(VALUE klass, VALUE name)
2314{
2315 if (klass && rb_class_real(klass) != rb_cObject)
2316 rb_name_err_raise("uninitialized constant %2$s::%1$s",
2317 klass, name);
2318 else
2319 rb_name_err_raise("uninitialized constant %1$s",
2320 klass, name);
2321}
2322
2323VALUE
2324rb_const_missing(VALUE klass, VALUE name)
2325{
2326 VALUE value = rb_funcallv(klass, idConst_missing, 1, &name);
2327 rb_vm_inc_const_missing_count();
2328 return value;
2329}
2330
2331
2332/*
2333 * call-seq:
2334 * mod.const_missing(sym) -> obj
2335 *
2336 * Invoked when a reference is made to an undefined constant in
2337 * <i>mod</i>. It is passed a symbol for the undefined constant, and
2338 * returns a value to be used for that constant. For example, consider:
2339 *
2340 * def Foo.const_missing(name)
2341 * name # return the constant name as Symbol
2342 * end
2343 *
2344 * Foo::UNDEFINED_CONST #=> :UNDEFINED_CONST: symbol returned
2345 *
2346 * As the example above shows, +const_missing+ is not required to create the
2347 * missing constant in <i>mod</i>, though that is often a side-effect. The
2348 * caller gets its return value when triggered. If the constant is also defined,
2349 * further lookups won't hit +const_missing+ and will return the value stored in
2350 * the constant as usual. Otherwise, +const_missing+ will be invoked again.
2351 *
2352 * In the next example, when a reference is made to an undefined constant,
2353 * +const_missing+ attempts to load a file whose path is the lowercase version
2354 * of the constant name (thus class <code>Fred</code> is assumed to be in file
2355 * <code>fred.rb</code>). If defined as a side-effect of loading the file, the
2356 * method returns the value stored in the constant. This implements an autoload
2357 * feature similar to Kernel#autoload and Module#autoload, though it differs in
2358 * important ways.
2359 *
2360 * def Object.const_missing(name)
2361 * @looked_for ||= {}
2362 * str_name = name.to_s
2363 * raise "Constant not found: #{name}" if @looked_for[str_name]
2364 * @looked_for[str_name] = 1
2365 * file = str_name.downcase
2366 * require file
2367 * const_get(name, false)
2368 * end
2369 *
2370 */
2371
2372VALUE
2373rb_mod_const_missing(VALUE klass, VALUE name)
2374{
2375 rb_execution_context_t *ec = GET_EC();
2376 VALUE ref = ec->private_const_reference;
2377 rb_vm_pop_cfunc_frame();
2378 if (ref) {
2379 ec->private_const_reference = 0;
2380 rb_name_err_raise("private constant %2$s::%1$s referenced", ref, name);
2381 }
2382 uninitialized_constant(klass, name);
2383
2385}
2386
2387static void
2388autoload_table_mark(void *ptr)
2389{
2390 rb_mark_tbl_no_pin((st_table *)ptr);
2391}
2392
2393static void
2394autoload_table_free(void *ptr)
2395{
2396 st_free_table((st_table *)ptr);
2397}
2398
2399static size_t
2400autoload_table_memsize(const void *ptr)
2401{
2402 const st_table *tbl = ptr;
2403 return st_memsize(tbl);
2404}
2405
2406static void
2407autoload_table_compact(void *ptr)
2408{
2409 rb_gc_ref_update_table_values_only((st_table *)ptr);
2410}
2411
2412static const rb_data_type_t autoload_table_type = {
2413 "autoload_table",
2414 {autoload_table_mark, autoload_table_free, autoload_table_memsize, autoload_table_compact,},
2415 0, 0, RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED
2416};
2417
2418#define check_autoload_table(av) \
2419 (struct st_table *)rb_check_typeddata((av), &autoload_table_type)
2420
2421static VALUE
2422autoload_data(VALUE mod, ID id)
2423{
2424 struct st_table *tbl;
2425 st_data_t val;
2426
2427 // If we are called with a non-origin ICLASS, fetch the autoload data from
2428 // the original module.
2429 if (RB_TYPE_P(mod, T_ICLASS)) {
2430 if (FL_TEST_RAW(mod, RICLASS_IS_ORIGIN)) {
2431 return 0;
2432 }
2433 else {
2434 mod = RBASIC(mod)->klass;
2435 }
2436 }
2437
2439
2440 // Look up the instance variable table for `autoload`, then index into that table with the given constant name `id`.
2441
2442 VALUE tbl_value = rb_ivar_lookup(mod, autoload, Qfalse);
2443 if (!RTEST(tbl_value) || !(tbl = check_autoload_table(tbl_value)) || !st_lookup(tbl, (st_data_t)id, &val)) {
2444 return 0;
2445 }
2446
2447 return (VALUE)val;
2448}
2449
2450// Every autoload constant has exactly one instance of autoload_const, stored in `autoload_features`. Since multiple autoload constants can refer to the same file, every `autoload_const` refers to a de-duplicated `autoload_data`.
2452 // The linked list node of all constants which are loaded by the related autoload feature.
2453 struct ccan_list_node cnode; /* <=> autoload_data.constants */
2454
2455 // The shared "autoload_data" if multiple constants are defined from the same feature.
2456 VALUE autoload_data_value;
2457
2458 // The module we are loading a constant into.
2459 VALUE module;
2460
2461 // The name of the constant we are loading.
2462 ID name;
2463
2464 // The value of the constant (after it's loaded).
2465 VALUE value;
2466
2467 // The constant entry flags which need to be re-applied after autoloading the feature.
2468 rb_const_flag_t flag;
2469
2470 // The source file and line number that defined this constant (different from feature path).
2471 VALUE file;
2472 int line;
2473};
2474
2475// Each `autoload_data` uniquely represents a specific feature which can be loaded, and a list of constants which it is able to define. We use a mutex to coordinate multiple threads trying to load the same feature.
2477 // The feature path to require to load this constant.
2478 VALUE feature;
2479
2480 // The mutex which is protecting autoloading this feature.
2481 VALUE mutex;
2482
2483 // The process fork serial number since the autoload mutex will become invalid on fork.
2484 rb_serial_t fork_gen;
2485
2486 // The linked list of all constants that are going to be loaded by this autoload.
2487 struct ccan_list_head constants; /* <=> autoload_const.cnode */
2488};
2489
2490static void
2491autoload_data_compact(void *ptr)
2492{
2493 struct autoload_data *p = ptr;
2494
2495 p->feature = rb_gc_location(p->feature);
2496 p->mutex = rb_gc_location(p->mutex);
2497}
2498
2499static void
2500autoload_data_mark(void *ptr)
2501{
2502 struct autoload_data *p = ptr;
2503
2504 rb_gc_mark_movable(p->feature);
2505 rb_gc_mark_movable(p->mutex);
2506}
2507
2508static void
2509autoload_data_free(void *ptr)
2510{
2511 struct autoload_data *p = ptr;
2512
2513 struct autoload_const *autoload_const, *next;
2514 ccan_list_for_each_safe(&p->constants, autoload_const, next, cnode) {
2515 ccan_list_del_init(&autoload_const->cnode);
2516 }
2517
2518 ruby_xfree(p);
2519}
2520
2521static size_t
2522autoload_data_memsize(const void *ptr)
2523{
2524 return sizeof(struct autoload_data);
2525}
2526
2527static const rb_data_type_t autoload_data_type = {
2528 "autoload_data",
2529 {autoload_data_mark, autoload_data_free, autoload_data_memsize, autoload_data_compact},
2530 0, 0, RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED
2531};
2532
2533static void
2534autoload_const_compact(void *ptr)
2535{
2536 struct autoload_const *ac = ptr;
2537
2538 ac->module = rb_gc_location(ac->module);
2539 ac->autoload_data_value = rb_gc_location(ac->autoload_data_value);
2540 ac->value = rb_gc_location(ac->value);
2541 ac->file = rb_gc_location(ac->file);
2542}
2543
2544static void
2545autoload_const_mark(void *ptr)
2546{
2547 struct autoload_const *ac = ptr;
2548
2549 rb_gc_mark_movable(ac->module);
2550 rb_gc_mark_movable(ac->autoload_data_value);
2551 rb_gc_mark_movable(ac->value);
2552 rb_gc_mark_movable(ac->file);
2553}
2554
2555static size_t
2556autoload_const_memsize(const void *ptr)
2557{
2558 return sizeof(struct autoload_const);
2559}
2560
2561static void
2562autoload_const_free(void *ptr)
2563{
2564 struct autoload_const *autoload_const = ptr;
2565
2566 ccan_list_del(&autoload_const->cnode);
2567 ruby_xfree(ptr);
2568}
2569
2570static const rb_data_type_t autoload_const_type = {
2571 "autoload_const",
2572 {autoload_const_mark, autoload_const_free, autoload_const_memsize, autoload_const_compact,},
2573 0, 0, RUBY_TYPED_FREE_IMMEDIATELY
2574};
2575
2576static struct autoload_data *
2577get_autoload_data(VALUE autoload_const_value, struct autoload_const **autoload_const_pointer)
2578{
2579 struct autoload_const *autoload_const = rb_check_typeddata(autoload_const_value, &autoload_const_type);
2580
2581 VALUE autoload_data_value = autoload_const->autoload_data_value;
2582 struct autoload_data *autoload_data = rb_check_typeddata(autoload_data_value, &autoload_data_type);
2583
2584 /* do not reach across stack for ->state after forking: */
2585 if (autoload_data && autoload_data->fork_gen != GET_VM()->fork_gen) {
2586 RB_OBJ_WRITE(autoload_data_value, &autoload_data->mutex, Qnil);
2587 autoload_data->fork_gen = 0;
2588 }
2589
2590 if (autoload_const_pointer) *autoload_const_pointer = autoload_const;
2591
2592 return autoload_data;
2593}
2594
2595void
2596rb_autoload(VALUE module, ID name, const char *feature)
2597{
2598 if (!feature || !*feature) {
2599 rb_raise(rb_eArgError, "empty feature name");
2600 }
2601
2602 rb_autoload_str(module, name, rb_fstring_cstr(feature));
2603}
2604
2605static void const_set(VALUE klass, ID id, VALUE val);
2606
2608 VALUE module;
2609 ID name;
2610 VALUE feature;
2611};
2612
2613static VALUE
2614autoload_feature_lookup_or_create(VALUE feature, struct autoload_data **autoload_data_pointer)
2615{
2616 RUBY_ASSERT_MUTEX_OWNED(autoload_mutex);
2617 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
2618
2619 VALUE autoload_data_value = rb_hash_aref(autoload_features, feature);
2621
2622 if (NIL_P(autoload_data_value)) {
2623 autoload_data_value = TypedData_Make_Struct(0, struct autoload_data, &autoload_data_type, autoload_data);
2624 RB_OBJ_WRITE(autoload_data_value, &autoload_data->feature, feature);
2625 RB_OBJ_WRITE(autoload_data_value, &autoload_data->mutex, Qnil);
2626 ccan_list_head_init(&autoload_data->constants);
2627
2628 if (autoload_data_pointer) *autoload_data_pointer = autoload_data;
2629
2630 rb_hash_aset(autoload_features, feature, autoload_data_value);
2631 }
2632 else if (autoload_data_pointer) {
2633 *autoload_data_pointer = rb_check_typeddata(autoload_data_value, &autoload_data_type);
2634 }
2635
2636 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
2637 return autoload_data_value;
2638}
2639
2640static VALUE
2641autoload_table_lookup_or_create(VALUE module)
2642{
2643 VALUE autoload_table_value = rb_ivar_lookup(module, autoload, Qfalse);
2644 if (RTEST(autoload_table_value)) {
2645 return autoload_table_value;
2646 }
2647 else {
2648 autoload_table_value = TypedData_Wrap_Struct(0, &autoload_table_type, NULL);
2649 rb_class_ivar_set(module, autoload, autoload_table_value);
2650 RTYPEDDATA_DATA(autoload_table_value) = st_init_numtable();
2651 return autoload_table_value;
2652 }
2653}
2654
2655static VALUE
2656autoload_synchronized(VALUE _arguments)
2657{
2658 struct autoload_arguments *arguments = (struct autoload_arguments *)_arguments;
2659
2660 rb_const_entry_t *constant_entry = rb_const_lookup(arguments->module, arguments->name);
2661 if (constant_entry && !UNDEF_P(constant_entry->value)) {
2662 return Qfalse;
2663 }
2664
2665 // Reset any state associated with any previous constant:
2666 const_set(arguments->module, arguments->name, Qundef);
2667
2668 VALUE autoload_table_value = autoload_table_lookup_or_create(arguments->module);
2669 struct st_table *autoload_table = check_autoload_table(autoload_table_value);
2670
2671 // Ensure the string is uniqued since we use an identity lookup:
2672 VALUE feature = rb_fstring(arguments->feature);
2673
2675 VALUE autoload_data_value = autoload_feature_lookup_or_create(feature, &autoload_data);
2676
2677 {
2679 VALUE autoload_const_value = TypedData_Make_Struct(0, struct autoload_const, &autoload_const_type, autoload_const);
2680 autoload_const->module = arguments->module;
2681 autoload_const->name = arguments->name;
2682 autoload_const->value = Qundef;
2683 autoload_const->flag = CONST_PUBLIC;
2684 autoload_const->autoload_data_value = autoload_data_value;
2685 ccan_list_add_tail(&autoload_data->constants, &autoload_const->cnode);
2686 st_insert(autoload_table, (st_data_t)arguments->name, (st_data_t)autoload_const_value);
2687 RB_OBJ_WRITTEN(autoload_table_value, Qundef, autoload_const_value);
2688 }
2689
2690 return Qtrue;
2691}
2692
2693void
2694rb_autoload_str(VALUE module, ID name, VALUE feature)
2695{
2696 if (!rb_is_const_id(name)) {
2697 rb_raise(rb_eNameError, "autoload must be constant name: %"PRIsVALUE"", QUOTE_ID(name));
2698 }
2699
2700 Check_Type(feature, T_STRING);
2701 if (!RSTRING_LEN(feature)) {
2702 rb_raise(rb_eArgError, "empty feature name");
2703 }
2704
2705 struct autoload_arguments arguments = {
2706 .module = module,
2707 .name = name,
2708 .feature = feature,
2709 };
2710
2711 VALUE result = rb_mutex_synchronize(autoload_mutex, autoload_synchronized, (VALUE)&arguments);
2712
2713 if (result == Qtrue) {
2714 rb_const_added(module, name);
2715 }
2716}
2717
2718static void
2719autoload_delete(VALUE module, ID name)
2720{
2721 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
2722
2723 st_data_t load = 0, key = name;
2724
2725 RUBY_ASSERT(RB_TYPE_P(module, T_CLASS) || RB_TYPE_P(module, T_MODULE));
2726
2727 VALUE table_value = rb_ivar_lookup(module, autoload, Qfalse);
2728 if (RTEST(table_value)) {
2729 struct st_table *table = check_autoload_table(table_value);
2730
2731 st_delete(table, &key, &load);
2732 RB_OBJ_WRITTEN(table_value, load, Qundef);
2733
2734 /* Qfalse can indicate already deleted */
2735 if (load != Qfalse) {
2737 struct autoload_data *autoload_data = get_autoload_data((VALUE)load, &autoload_const);
2738
2739 VM_ASSERT(autoload_data);
2740 VM_ASSERT(!ccan_list_empty(&autoload_data->constants));
2741
2742 /*
2743 * we must delete here to avoid "already initialized" warnings
2744 * with parallel autoload. Using list_del_init here so list_del
2745 * works in autoload_const_free
2746 */
2747 ccan_list_del_init(&autoload_const->cnode);
2748
2749 if (ccan_list_empty(&autoload_data->constants)) {
2750 rb_hash_delete(autoload_features, autoload_data->feature);
2751 }
2752
2753 // If the autoload table is empty, we can delete it.
2754 if (table->num_entries == 0) {
2755 rb_attr_delete(module, autoload);
2756 }
2757 }
2758 }
2759
2760 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
2761}
2762
2763static int
2764autoload_by_someone_else(struct autoload_data *ele)
2765{
2766 return ele->mutex != Qnil && !rb_mutex_owned_p(ele->mutex);
2767}
2768
2769static VALUE
2770check_autoload_required(VALUE mod, ID id, const char **loadingpath)
2771{
2772 VALUE autoload_const_value = autoload_data(mod, id);
2774 const char *loading;
2775
2776 if (!autoload_const_value || !(autoload_data = get_autoload_data(autoload_const_value, 0))) {
2777 return 0;
2778 }
2779
2780 VALUE feature = autoload_data->feature;
2781
2782 /*
2783 * if somebody else is autoloading, we MUST wait for them, since
2784 * rb_provide_feature can provide a feature before autoload_const_set
2785 * completes. We must wait until autoload_const_set finishes in
2786 * the other thread.
2787 */
2788 if (autoload_by_someone_else(autoload_data)) {
2789 return autoload_const_value;
2790 }
2791
2792 loading = RSTRING_PTR(feature);
2793
2794 if (!rb_feature_provided(loading, &loading)) {
2795 return autoload_const_value;
2796 }
2797
2798 if (loadingpath && loading) {
2799 *loadingpath = loading;
2800 return autoload_const_value;
2801 }
2802
2803 return 0;
2804}
2805
2806static struct autoload_const *autoloading_const_entry(VALUE mod, ID id);
2807
2808int
2809rb_autoloading_value(VALUE mod, ID id, VALUE* value, rb_const_flag_t *flag)
2810{
2811 struct autoload_const *ac = autoloading_const_entry(mod, id);
2812 if (!ac) return FALSE;
2813
2814 if (value) {
2815 *value = ac->value;
2816 }
2817
2818 if (flag) {
2819 *flag = ac->flag;
2820 }
2821
2822 return TRUE;
2823}
2824
2825static int
2826autoload_by_current(struct autoload_data *ele)
2827{
2828 return ele->mutex != Qnil && rb_mutex_owned_p(ele->mutex);
2829}
2830
2831// If there is an autoloading constant and it has been set by the current
2832// execution context, return it. This allows threads which are loading code to
2833// refer to their own autoloaded constants.
2834struct autoload_const *
2835autoloading_const_entry(VALUE mod, ID id)
2836{
2837 VALUE load = autoload_data(mod, id);
2838 struct autoload_data *ele;
2839 struct autoload_const *ac;
2840
2841 // Find the autoloading state:
2842 if (!load || !(ele = get_autoload_data(load, &ac))) {
2843 // Couldn't be found:
2844 return 0;
2845 }
2846
2847 // Check if it's being loaded by the current thread/fiber:
2848 if (autoload_by_current(ele)) {
2849 if (!UNDEF_P(ac->value)) {
2850 return ac;
2851 }
2852 }
2853
2854 return 0;
2855}
2856
2857static int
2858autoload_defined_p(VALUE mod, ID id)
2859{
2860 rb_const_entry_t *ce = rb_const_lookup(mod, id);
2861
2862 // If there is no constant or the constant is not undefined (special marker for autoloading):
2863 if (!ce || !UNDEF_P(ce->value)) {
2864 // We are not autoloading:
2865 return 0;
2866 }
2867
2868 // Otherwise check if there is an autoload in flight right now:
2869 return !rb_autoloading_value(mod, id, NULL, NULL);
2870}
2871
2872static void const_tbl_update(struct autoload_const *, int);
2873
2875 VALUE module;
2876 ID name;
2877 int flag;
2878
2879 VALUE mutex;
2880
2881 // The specific constant which triggered the autoload code to fire:
2883
2884 // The parent autoload data which is shared between multiple constants:
2886};
2887
2888static VALUE
2889autoload_const_set(struct autoload_const *ac)
2890{
2891 check_before_mod_set(ac->module, ac->name, ac->value, "constant");
2892
2893 RB_VM_LOCK_ENTER();
2894 {
2895 const_tbl_update(ac, true);
2896 }
2897 RB_VM_LOCK_LEAVE();
2898
2899 return 0; /* ignored */
2900}
2901
2902static VALUE
2903autoload_load_needed(VALUE _arguments)
2904{
2905 struct autoload_load_arguments *arguments = (struct autoload_load_arguments*)_arguments;
2906
2907 const char *loading = 0, *src;
2908
2909 if (!autoload_defined_p(arguments->module, arguments->name)) {
2910 return Qfalse;
2911 }
2912
2913 VALUE autoload_const_value = check_autoload_required(arguments->module, arguments->name, &loading);
2914 if (!autoload_const_value) {
2915 return Qfalse;
2916 }
2917
2918 src = rb_sourcefile();
2919 if (src && loading && strcmp(src, loading) == 0) {
2920 return Qfalse;
2921 }
2922
2925 if (!(autoload_data = get_autoload_data(autoload_const_value, &autoload_const))) {
2926 return Qfalse;
2927 }
2928
2929 if (NIL_P(autoload_data->mutex)) {
2930 RB_OBJ_WRITE(autoload_const->autoload_data_value, &autoload_data->mutex, rb_mutex_new());
2931 autoload_data->fork_gen = GET_VM()->fork_gen;
2932 }
2933 else if (rb_mutex_owned_p(autoload_data->mutex)) {
2934 return Qfalse;
2935 }
2936
2937 arguments->mutex = autoload_data->mutex;
2938 arguments->autoload_const = autoload_const;
2939
2940 return autoload_const_value;
2941}
2942
2943static VALUE
2944autoload_apply_constants(VALUE _arguments)
2945{
2946 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
2947
2948 struct autoload_load_arguments *arguments = (struct autoload_load_arguments*)_arguments;
2949
2950 struct autoload_const *autoload_const = 0; // for ccan_container_off_var()
2951 struct autoload_const *next;
2952
2953 // We use safe iteration here because `autoload_const_set` will eventually invoke
2954 // `autoload_delete` which will remove the constant from the linked list. In theory, once
2955 // the `autoload_data->constants` linked list is empty, we can remove it.
2956
2957 // Iterate over all constants and assign them:
2958 ccan_list_for_each_safe(&arguments->autoload_data->constants, autoload_const, next, cnode) {
2959 if (!UNDEF_P(autoload_const->value)) {
2960 autoload_const_set(autoload_const);
2961 }
2962 }
2963
2964 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
2965
2966 return Qtrue;
2967}
2968
2969static VALUE
2970autoload_feature_require(VALUE _arguments)
2971{
2972 struct autoload_load_arguments *arguments = (struct autoload_load_arguments*)_arguments;
2973
2974 struct autoload_const *autoload_const = arguments->autoload_const;
2975
2976 // We save this for later use in autoload_apply_constants:
2977 arguments->autoload_data = rb_check_typeddata(autoload_const->autoload_data_value, &autoload_data_type);
2978
2979 VALUE result = rb_funcall(rb_vm_top_self(), rb_intern("require"), 1, arguments->autoload_data->feature);
2980
2981 if (RTEST(result)) {
2982 return rb_mutex_synchronize(autoload_mutex, autoload_apply_constants, _arguments);
2983 }
2984
2985 return result;
2986}
2987
2988static VALUE
2989autoload_try_load(VALUE _arguments)
2990{
2991 struct autoload_load_arguments *arguments = (struct autoload_load_arguments*)_arguments;
2992
2993 VALUE result = autoload_feature_require(_arguments);
2994
2995 // After we loaded the feature, if the constant is not defined, we remove it completely:
2996 rb_const_entry_t *ce = rb_const_lookup(arguments->module, arguments->name);
2997
2998 if (!ce || UNDEF_P(ce->value)) {
2999 result = Qfalse;
3000
3001 rb_const_remove(arguments->module, arguments->name);
3002
3003 if (arguments->module == rb_cObject) {
3004 rb_warning(
3005 "Expected %"PRIsVALUE" to define %"PRIsVALUE" but it didn't",
3006 arguments->autoload_data->feature,
3007 ID2SYM(arguments->name)
3008 );
3009 }
3010 else {
3011 rb_warning(
3012 "Expected %"PRIsVALUE" to define %"PRIsVALUE"::%"PRIsVALUE" but it didn't",
3013 arguments->autoload_data->feature,
3014 arguments->module,
3015 ID2SYM(arguments->name)
3016 );
3017 }
3018 }
3019 else {
3020 // Otherwise, it was loaded, copy the flags from the autoload constant:
3021 ce->flag |= arguments->flag;
3022 }
3023
3024 return result;
3025}
3026
3027VALUE
3029{
3030 rb_const_entry_t *ce = rb_const_lookup(module, name);
3031
3032 // We bail out as early as possible without any synchronisation:
3033 if (!ce || !UNDEF_P(ce->value)) {
3034 return Qfalse;
3035 }
3036
3037 // At this point, we assume there might be autoloading, so fail if it's ractor:
3038 if (UNLIKELY(!rb_ractor_main_p())) {
3039 return rb_ractor_autoload_load(module, name);
3040 }
3041
3042 // This state is stored on the stack and is used during the autoload process.
3043 struct autoload_load_arguments arguments = {.module = module, .name = name, .mutex = Qnil};
3044
3045 // Figure out whether we can autoload the named constant:
3046 VALUE autoload_const_value = rb_mutex_synchronize(autoload_mutex, autoload_load_needed, (VALUE)&arguments);
3047
3048 // This confirms whether autoloading is required or not:
3049 if (autoload_const_value == Qfalse) return autoload_const_value;
3050
3051 arguments.flag = ce->flag & (CONST_DEPRECATED | CONST_VISIBILITY_MASK);
3052
3053 // Only one thread will enter here at a time:
3054 VALUE result = rb_mutex_synchronize(arguments.mutex, autoload_try_load, (VALUE)&arguments);
3055
3056 // If you don't guard this value, it's possible for the autoload constant to
3057 // be freed by another thread which loads multiple constants, one of which
3058 // resolves to the constant this thread is trying to load, so proteect this
3059 // so that it is not freed until we are done with it in `autoload_try_load`:
3060 RB_GC_GUARD(autoload_const_value);
3061
3062 return result;
3063}
3064
3065VALUE
3067{
3068 return rb_autoload_at_p(mod, id, TRUE);
3069}
3070
3071VALUE
3072rb_autoload_at_p(VALUE mod, ID id, int recur)
3073{
3074 VALUE load;
3075 struct autoload_data *ele;
3076
3077 while (!autoload_defined_p(mod, id)) {
3078 if (!recur) return Qnil;
3079 mod = RCLASS_SUPER(mod);
3080 if (!mod) return Qnil;
3081 }
3082 load = check_autoload_required(mod, id, 0);
3083 if (!load) return Qnil;
3084 return (ele = get_autoload_data(load, 0)) ? ele->feature : Qnil;
3085}
3086
3087void
3088rb_const_warn_if_deprecated(const rb_const_entry_t *ce, VALUE klass, ID id)
3089{
3090 if (RB_CONST_DEPRECATED_P(ce) &&
3091 rb_warning_category_enabled_p(RB_WARN_CATEGORY_DEPRECATED)) {
3092 if (klass == rb_cObject) {
3093 rb_category_warn(RB_WARN_CATEGORY_DEPRECATED, "constant ::%"PRIsVALUE" is deprecated", QUOTE_ID(id));
3094 }
3095 else {
3096 rb_category_warn(RB_WARN_CATEGORY_DEPRECATED, "constant %"PRIsVALUE"::%"PRIsVALUE" is deprecated",
3097 rb_class_name(klass), QUOTE_ID(id));
3098 }
3099 }
3100}
3101
3102static VALUE
3103rb_const_get_0(VALUE klass, ID id, int exclude, int recurse, int visibility)
3104{
3105 VALUE c = rb_const_search(klass, id, exclude, recurse, visibility);
3106 if (!UNDEF_P(c)) {
3107 if (UNLIKELY(!rb_ractor_main_p())) {
3108 if (!rb_ractor_shareable_p(c)) {
3109 rb_raise(rb_eRactorIsolationError, "can not access non-shareable objects in constant %"PRIsVALUE"::%s by non-main Ractor.", rb_class_path(klass), rb_id2name(id));
3110 }
3111 }
3112 return c;
3113 }
3114 return rb_const_missing(klass, ID2SYM(id));
3115}
3116
3117static VALUE
3118rb_const_search_from(VALUE klass, ID id, int exclude, int recurse, int visibility)
3119{
3120 VALUE value, current;
3121 bool first_iteration = true;
3122
3123 for (current = klass;
3124 RTEST(current);
3125 current = RCLASS_SUPER(current), first_iteration = false) {
3126 VALUE tmp;
3127 VALUE am = 0;
3128 rb_const_entry_t *ce;
3129
3130 if (!first_iteration && RCLASS_ORIGIN(current) != current) {
3131 // This item in the super chain has an origin iclass
3132 // that comes later in the chain. Skip this item so
3133 // prepended modules take precedence.
3134 continue;
3135 }
3136
3137 // Do lookup in original class or module in case we are at an origin
3138 // iclass in the chain.
3139 tmp = current;
3140 if (BUILTIN_TYPE(tmp) == T_ICLASS) tmp = RBASIC(tmp)->klass;
3141
3142 // Do the lookup. Loop in case of autoload.
3143 while ((ce = rb_const_lookup(tmp, id))) {
3144 if (visibility && RB_CONST_PRIVATE_P(ce)) {
3145 GET_EC()->private_const_reference = tmp;
3146 return Qundef;
3147 }
3148 rb_const_warn_if_deprecated(ce, tmp, id);
3149 value = ce->value;
3150 if (UNDEF_P(value)) {
3151 struct autoload_const *ac;
3152 if (am == tmp) break;
3153 am = tmp;
3154 ac = autoloading_const_entry(tmp, id);
3155 if (ac) return ac->value;
3156 rb_autoload_load(tmp, id);
3157 continue;
3158 }
3159 if (exclude && tmp == rb_cObject) {
3160 goto not_found;
3161 }
3162 return value;
3163 }
3164 if (!recurse) break;
3165 }
3166
3167 not_found:
3168 GET_EC()->private_const_reference = 0;
3169 return Qundef;
3170}
3171
3172static VALUE
3173rb_const_search(VALUE klass, ID id, int exclude, int recurse, int visibility)
3174{
3175 VALUE value;
3176
3177 if (klass == rb_cObject) exclude = FALSE;
3178 value = rb_const_search_from(klass, id, exclude, recurse, visibility);
3179 if (!UNDEF_P(value)) return value;
3180 if (exclude) return value;
3181 if (BUILTIN_TYPE(klass) != T_MODULE) return value;
3182 /* search global const too, if klass is a module */
3183 return rb_const_search_from(rb_cObject, id, FALSE, recurse, visibility);
3184}
3185
3186VALUE
3188{
3189 return rb_const_get_0(klass, id, TRUE, TRUE, FALSE);
3190}
3191
3192VALUE
3194{
3195 return rb_const_get_0(klass, id, FALSE, TRUE, FALSE);
3196}
3197
3198VALUE
3200{
3201 return rb_const_get_0(klass, id, TRUE, FALSE, FALSE);
3202}
3203
3204VALUE
3205rb_public_const_get_from(VALUE klass, ID id)
3206{
3207 return rb_const_get_0(klass, id, TRUE, TRUE, TRUE);
3208}
3209
3210VALUE
3211rb_public_const_get_at(VALUE klass, ID id)
3212{
3213 return rb_const_get_0(klass, id, TRUE, FALSE, TRUE);
3214}
3215
3216NORETURN(static void undefined_constant(VALUE mod, VALUE name));
3217static void
3218undefined_constant(VALUE mod, VALUE name)
3219{
3220 rb_name_err_raise("constant %2$s::%1$s not defined",
3221 mod, name);
3222}
3223
3224static VALUE
3225rb_const_location_from(VALUE klass, ID id, int exclude, int recurse, int visibility)
3226{
3227 while (RTEST(klass)) {
3228 rb_const_entry_t *ce;
3229
3230 while ((ce = rb_const_lookup(klass, id))) {
3231 if (visibility && RB_CONST_PRIVATE_P(ce)) {
3232 return Qnil;
3233 }
3234 if (exclude && klass == rb_cObject) {
3235 goto not_found;
3236 }
3237
3238 if (UNDEF_P(ce->value)) { // autoload
3239 VALUE autoload_const_value = autoload_data(klass, id);
3240 if (RTEST(autoload_const_value)) {
3242 struct autoload_data *autoload_data = get_autoload_data(autoload_const_value, &autoload_const);
3243
3244 if (!UNDEF_P(autoload_const->value) && RTEST(rb_mutex_owned_p(autoload_data->mutex))) {
3245 return rb_assoc_new(autoload_const->file, INT2NUM(autoload_const->line));
3246 }
3247 }
3248 }
3249
3250 if (NIL_P(ce->file)) return rb_ary_new();
3251 return rb_assoc_new(ce->file, INT2NUM(ce->line));
3252 }
3253 if (!recurse) break;
3254 klass = RCLASS_SUPER(klass);
3255 }
3256
3257 not_found:
3258 return Qnil;
3259}
3260
3261static VALUE
3262rb_const_location(VALUE klass, ID id, int exclude, int recurse, int visibility)
3263{
3264 VALUE loc;
3265
3266 if (klass == rb_cObject) exclude = FALSE;
3267 loc = rb_const_location_from(klass, id, exclude, recurse, visibility);
3268 if (!NIL_P(loc)) return loc;
3269 if (exclude) return loc;
3270 if (BUILTIN_TYPE(klass) != T_MODULE) return loc;
3271 /* search global const too, if klass is a module */
3272 return rb_const_location_from(rb_cObject, id, FALSE, recurse, visibility);
3273}
3274
3275VALUE
3276rb_const_source_location(VALUE klass, ID id)
3277{
3278 return rb_const_location(klass, id, FALSE, TRUE, FALSE);
3279}
3280
3281VALUE
3282rb_const_source_location_at(VALUE klass, ID id)
3283{
3284 return rb_const_location(klass, id, TRUE, FALSE, FALSE);
3285}
3286
3287/*
3288 * call-seq:
3289 * remove_const(sym) -> obj
3290 *
3291 * Removes the definition of the given constant, returning that
3292 * constant's previous value. If that constant referred to
3293 * a module, this will not change that module's name and can lead
3294 * to confusion.
3295 */
3296
3297VALUE
3299{
3300 const ID id = id_for_var(mod, name, a, constant);
3301
3302 if (!id) {
3303 undefined_constant(mod, name);
3304 }
3305 return rb_const_remove(mod, id);
3306}
3307
3308VALUE
3310{
3311 VALUE val;
3312 rb_const_entry_t *ce;
3313
3314 rb_check_frozen(mod);
3315
3316 ce = rb_const_lookup(mod, id);
3317 if (!ce || !rb_id_table_delete(RCLASS_CONST_TBL(mod), id)) {
3318 if (rb_const_defined_at(mod, id)) {
3319 rb_name_err_raise("cannot remove %2$s::%1$s", mod, ID2SYM(id));
3320 }
3321
3322 undefined_constant(mod, ID2SYM(id));
3323 }
3324
3325 rb_const_warn_if_deprecated(ce, mod, id);
3327
3328 val = ce->value;
3329
3330 if (UNDEF_P(val)) {
3331 autoload_delete(mod, id);
3332 val = Qnil;
3333 }
3334
3335 ruby_xfree(ce);
3336
3337 return val;
3338}
3339
3340static int
3341cv_i_update(st_data_t *k, st_data_t *v, st_data_t a, int existing)
3342{
3343 if (existing) return ST_STOP;
3344 *v = a;
3345 return ST_CONTINUE;
3346}
3347
3348static enum rb_id_table_iterator_result
3349sv_i(ID key, VALUE v, void *a)
3350{
3352 st_table *tbl = a;
3353
3354 if (rb_is_const_id(key)) {
3355 st_update(tbl, (st_data_t)key, cv_i_update, (st_data_t)ce);
3356 }
3357 return ID_TABLE_CONTINUE;
3358}
3359
3360static enum rb_id_table_iterator_result
3361rb_local_constants_i(ID const_name, VALUE const_value, void *ary)
3362{
3363 if (rb_is_const_id(const_name) && !RB_CONST_PRIVATE_P((rb_const_entry_t *)const_value)) {
3364 rb_ary_push((VALUE)ary, ID2SYM(const_name));
3365 }
3366 return ID_TABLE_CONTINUE;
3367}
3368
3369static VALUE
3370rb_local_constants(VALUE mod)
3371{
3372 struct rb_id_table *tbl = RCLASS_CONST_TBL(mod);
3373 VALUE ary;
3374
3375 if (!tbl) return rb_ary_new2(0);
3376
3377 RB_VM_LOCK_ENTER();
3378 {
3379 ary = rb_ary_new2(rb_id_table_size(tbl));
3380 rb_id_table_foreach(tbl, rb_local_constants_i, (void *)ary);
3381 }
3382 RB_VM_LOCK_LEAVE();
3383
3384 return ary;
3385}
3386
3387void*
3388rb_mod_const_at(VALUE mod, void *data)
3389{
3390 st_table *tbl = data;
3391 if (!tbl) {
3392 tbl = st_init_numtable();
3393 }
3394 if (RCLASS_CONST_TBL(mod)) {
3395 RB_VM_LOCK_ENTER();
3396 {
3397 rb_id_table_foreach(RCLASS_CONST_TBL(mod), sv_i, tbl);
3398 }
3399 RB_VM_LOCK_LEAVE();
3400 }
3401 return tbl;
3402}
3403
3404void*
3405rb_mod_const_of(VALUE mod, void *data)
3406{
3407 VALUE tmp = mod;
3408 for (;;) {
3409 data = rb_mod_const_at(tmp, data);
3410 tmp = RCLASS_SUPER(tmp);
3411 if (!tmp) break;
3412 if (tmp == rb_cObject && mod != rb_cObject) break;
3413 }
3414 return data;
3415}
3416
3417static int
3418list_i(st_data_t key, st_data_t value, VALUE ary)
3419{
3420 ID sym = (ID)key;
3421 rb_const_entry_t *ce = (rb_const_entry_t *)value;
3422 if (RB_CONST_PUBLIC_P(ce)) rb_ary_push(ary, ID2SYM(sym));
3423 return ST_CONTINUE;
3424}
3425
3426VALUE
3427rb_const_list(void *data)
3428{
3429 st_table *tbl = data;
3430 VALUE ary;
3431
3432 if (!tbl) return rb_ary_new2(0);
3433 ary = rb_ary_new2(tbl->num_entries);
3434 st_foreach_safe(tbl, list_i, ary);
3435 st_free_table(tbl);
3436
3437 return ary;
3438}
3439
3440/*
3441 * call-seq:
3442 * mod.constants(inherit=true) -> array
3443 *
3444 * Returns an array of the names of the constants accessible in
3445 * <i>mod</i>. This includes the names of constants in any included
3446 * modules (example at start of section), unless the <i>inherit</i>
3447 * parameter is set to <code>false</code>.
3448 *
3449 * The implementation makes no guarantees about the order in which the
3450 * constants are yielded.
3451 *
3452 * IO.constants.include?(:SYNC) #=> true
3453 * IO.constants(false).include?(:SYNC) #=> false
3454 *
3455 * Also see Module#const_defined?.
3456 */
3457
3458VALUE
3459rb_mod_constants(int argc, const VALUE *argv, VALUE mod)
3460{
3461 bool inherit = true;
3462
3463 if (rb_check_arity(argc, 0, 1)) inherit = RTEST(argv[0]);
3464
3465 if (inherit) {
3466 return rb_const_list(rb_mod_const_of(mod, 0));
3467 }
3468 else {
3469 return rb_local_constants(mod);
3470 }
3471}
3472
3473static int
3474rb_const_defined_0(VALUE klass, ID id, int exclude, int recurse, int visibility)
3475{
3476 VALUE tmp;
3477 int mod_retry = 0;
3478 rb_const_entry_t *ce;
3479
3480 tmp = klass;
3481 retry:
3482 while (tmp) {
3483 if ((ce = rb_const_lookup(tmp, id))) {
3484 if (visibility && RB_CONST_PRIVATE_P(ce)) {
3485 return (int)Qfalse;
3486 }
3487 if (UNDEF_P(ce->value) && !check_autoload_required(tmp, id, 0) &&
3488 !rb_autoloading_value(tmp, id, NULL, NULL))
3489 return (int)Qfalse;
3490
3491 if (exclude && tmp == rb_cObject && klass != rb_cObject) {
3492 return (int)Qfalse;
3493 }
3494
3495 return (int)Qtrue;
3496 }
3497 if (!recurse) break;
3498 tmp = RCLASS_SUPER(tmp);
3499 }
3500 if (!exclude && !mod_retry && BUILTIN_TYPE(klass) == T_MODULE) {
3501 mod_retry = 1;
3502 tmp = rb_cObject;
3503 goto retry;
3504 }
3505 return (int)Qfalse;
3506}
3507
3508int
3510{
3511 return rb_const_defined_0(klass, id, TRUE, TRUE, FALSE);
3512}
3513
3514int
3516{
3517 return rb_const_defined_0(klass, id, FALSE, TRUE, FALSE);
3518}
3519
3520int
3522{
3523 return rb_const_defined_0(klass, id, TRUE, FALSE, FALSE);
3524}
3525
3526int
3527rb_public_const_defined_from(VALUE klass, ID id)
3528{
3529 return rb_const_defined_0(klass, id, TRUE, TRUE, TRUE);
3530}
3531
3532static void
3533check_before_mod_set(VALUE klass, ID id, VALUE val, const char *dest)
3534{
3535 rb_check_frozen(klass);
3536}
3537
3538static void set_namespace_path(VALUE named_namespace, VALUE name);
3539
3540static enum rb_id_table_iterator_result
3541set_namespace_path_i(ID id, VALUE v, void *payload)
3542{
3544 VALUE value = ce->value;
3545 VALUE parental_path = *((VALUE *) payload);
3546 if (!rb_is_const_id(id) || !rb_namespace_p(value)) {
3547 return ID_TABLE_CONTINUE;
3548 }
3549
3550 bool has_permanent_classpath;
3551 classname(value, &has_permanent_classpath);
3552 if (has_permanent_classpath) {
3553 return ID_TABLE_CONTINUE;
3554 }
3555 set_namespace_path(value, build_const_path(parental_path, id));
3556
3557 if (!RCLASS_EXT(value)->permanent_classpath) {
3558 RCLASS_SET_CLASSPATH(value, 0, false);
3559 }
3560
3561 return ID_TABLE_CONTINUE;
3562}
3563
3564/*
3565 * Assign permanent classpaths to all namespaces that are directly or indirectly
3566 * nested under +named_namespace+. +named_namespace+ must have a permanent
3567 * classpath.
3568 */
3569static void
3570set_namespace_path(VALUE named_namespace, VALUE namespace_path)
3571{
3572 struct rb_id_table *const_table = RCLASS_CONST_TBL(named_namespace);
3573
3574 RB_VM_LOCK_ENTER();
3575 {
3576 RCLASS_SET_CLASSPATH(named_namespace, namespace_path, true);
3577
3578 if (const_table) {
3579 rb_id_table_foreach(const_table, set_namespace_path_i, &namespace_path);
3580 }
3581 }
3582 RB_VM_LOCK_LEAVE();
3583}
3584
3585void
3586rb_const_added(VALUE klass, ID const_name)
3587{
3588 if (GET_VM()->running) {
3589 VALUE name = ID2SYM(const_name);
3590 rb_funcallv(klass, idConst_added, 1, &name);
3591 }
3592}
3593
3594static void
3595const_set(VALUE klass, ID id, VALUE val)
3596{
3597 rb_const_entry_t *ce;
3598
3599 if (NIL_P(klass)) {
3600 rb_raise(rb_eTypeError, "no class/module to define constant %"PRIsVALUE"",
3601 QUOTE_ID(id));
3602 }
3603
3604 if (!rb_ractor_main_p() && !rb_ractor_shareable_p(val)) {
3605 rb_raise(rb_eRactorIsolationError, "can not set constants with non-shareable objects by non-main Ractors");
3606 }
3607
3608 check_before_mod_set(klass, id, val, "constant");
3609
3610 RB_VM_LOCK_ENTER();
3611 {
3612 struct rb_id_table *tbl = RCLASS_CONST_TBL(klass);
3613 if (!tbl) {
3614 RCLASS_CONST_TBL(klass) = tbl = rb_id_table_create(0);
3617 rb_id_table_insert(tbl, id, (VALUE)ce);
3618 setup_const_entry(ce, klass, val, CONST_PUBLIC);
3619 }
3620 else {
3621 struct autoload_const ac = {
3622 .module = klass, .name = id,
3623 .value = val, .flag = CONST_PUBLIC,
3624 /* fill the rest with 0 */
3625 };
3626 ac.file = rb_source_location(&ac.line);
3627 const_tbl_update(&ac, false);
3628 }
3629 }
3630 RB_VM_LOCK_LEAVE();
3631
3632 /*
3633 * Resolve and cache class name immediately to resolve ambiguity
3634 * and avoid order-dependency on const_tbl
3635 */
3636 if (rb_cObject && rb_namespace_p(val)) {
3637 bool val_path_permanent;
3638 VALUE val_path = classname(val, &val_path_permanent);
3639 if (NIL_P(val_path) || !val_path_permanent) {
3640 if (klass == rb_cObject) {
3641 set_namespace_path(val, rb_id2str(id));
3642 }
3643 else {
3644 bool parental_path_permanent;
3645 VALUE parental_path = classname(klass, &parental_path_permanent);
3646 if (NIL_P(parental_path)) {
3647 bool throwaway;
3648 parental_path = rb_tmp_class_path(klass, &throwaway, make_temporary_path);
3649 }
3650 if (parental_path_permanent && !val_path_permanent) {
3651 set_namespace_path(val, build_const_path(parental_path, id));
3652 }
3653 else if (!parental_path_permanent && NIL_P(val_path)) {
3654 RCLASS_SET_CLASSPATH(val, build_const_path(parental_path, id), false);
3655 }
3656 }
3657 }
3658 }
3659}
3660
3661void
3662rb_const_set_raw(VALUE klass, ID id, VALUE val)
3663{
3664 const_set(klass, id, val);
3665}
3666
3667void
3669{
3670 const_set(klass, id, val);
3671 rb_const_added(klass, id);
3672}
3673
3674static struct autoload_data *
3675autoload_data_for_named_constant(VALUE module, ID name, struct autoload_const **autoload_const_pointer)
3676{
3677 VALUE autoload_data_value = autoload_data(module, name);
3678 if (!autoload_data_value) return 0;
3679
3680 struct autoload_data *autoload_data = get_autoload_data(autoload_data_value, autoload_const_pointer);
3681 if (!autoload_data) return 0;
3682
3683 /* for autoloading thread, keep the defined value to autoloading storage */
3684 if (autoload_by_current(autoload_data)) {
3685 return autoload_data;
3686 }
3687
3688 return 0;
3689}
3690
3691static void
3692const_tbl_update(struct autoload_const *ac, int autoload_force)
3693{
3694 VALUE value;
3695 VALUE klass = ac->module;
3696 VALUE val = ac->value;
3697 ID id = ac->name;
3698 struct rb_id_table *tbl = RCLASS_CONST_TBL(klass);
3699 rb_const_flag_t visibility = ac->flag;
3700 rb_const_entry_t *ce;
3701
3702 if (rb_id_table_lookup(tbl, id, &value)) {
3703 ce = (rb_const_entry_t *)value;
3704 if (UNDEF_P(ce->value)) {
3705 RUBY_ASSERT_CRITICAL_SECTION_ENTER();
3706 VALUE file = ac->file;
3707 int line = ac->line;
3708 struct autoload_data *ele = autoload_data_for_named_constant(klass, id, &ac);
3709
3710 if (!autoload_force && ele) {
3712
3713 ac->value = val; /* autoload_data is non-WB-protected */
3714 ac->file = rb_source_location(&ac->line);
3715 }
3716 else {
3717 /* otherwise autoloaded constant, allow to override */
3718 autoload_delete(klass, id);
3719 ce->flag = visibility;
3720 RB_OBJ_WRITE(klass, &ce->value, val);
3721 RB_OBJ_WRITE(klass, &ce->file, file);
3722 ce->line = line;
3723 }
3724 RUBY_ASSERT_CRITICAL_SECTION_LEAVE();
3725 return;
3726 }
3727 else {
3728 VALUE name = QUOTE_ID(id);
3729 visibility = ce->flag;
3730 if (klass == rb_cObject)
3731 rb_warn("already initialized constant %"PRIsVALUE"", name);
3732 else
3733 rb_warn("already initialized constant %"PRIsVALUE"::%"PRIsVALUE"",
3734 rb_class_name(klass), name);
3735 if (!NIL_P(ce->file) && ce->line) {
3736 rb_compile_warn(RSTRING_PTR(ce->file), ce->line,
3737 "previous definition of %"PRIsVALUE" was here", name);
3738 }
3739 }
3741 setup_const_entry(ce, klass, val, visibility);
3742 }
3743 else {
3745
3747 rb_id_table_insert(tbl, id, (VALUE)ce);
3748 setup_const_entry(ce, klass, val, visibility);
3749 }
3750}
3751
3752static void
3753setup_const_entry(rb_const_entry_t *ce, VALUE klass, VALUE val,
3754 rb_const_flag_t visibility)
3755{
3756 ce->flag = visibility;
3757 RB_OBJ_WRITE(klass, &ce->value, val);
3758 RB_OBJ_WRITE(klass, &ce->file, rb_source_location(&ce->line));
3759}
3760
3761void
3762rb_define_const(VALUE klass, const char *name, VALUE val)
3763{
3764 ID id = rb_intern(name);
3765
3766 if (!rb_is_const_id(id)) {
3767 rb_warn("rb_define_const: invalid name '%s' for constant", name);
3768 }
3769 if (!RB_SPECIAL_CONST_P(val)) {
3770 rb_vm_register_global_object(val);
3771 }
3772 rb_const_set(klass, id, val);
3773}
3774
3775void
3776rb_define_global_const(const char *name, VALUE val)
3777{
3778 rb_define_const(rb_cObject, name, val);
3779}
3780
3781static void
3782set_const_visibility(VALUE mod, int argc, const VALUE *argv,
3783 rb_const_flag_t flag, rb_const_flag_t mask)
3784{
3785 int i;
3786 rb_const_entry_t *ce;
3787 ID id;
3788
3790 if (argc == 0) {
3791 rb_warning("%"PRIsVALUE" with no argument is just ignored",
3792 QUOTE_ID(rb_frame_callee()));
3793 return;
3794 }
3795
3796 for (i = 0; i < argc; i++) {
3797 struct autoload_const *ac;
3798 VALUE val = argv[i];
3799 id = rb_check_id(&val);
3800 if (!id) {
3801 undefined_constant(mod, val);
3802 }
3803 if ((ce = rb_const_lookup(mod, id))) {
3804 ce->flag &= ~mask;
3805 ce->flag |= flag;
3806 if (UNDEF_P(ce->value)) {
3807 struct autoload_data *ele;
3808
3809 ele = autoload_data_for_named_constant(mod, id, &ac);
3810 if (ele) {
3811 ac->flag &= ~mask;
3812 ac->flag |= flag;
3813 }
3814 }
3816 }
3817 else {
3818 undefined_constant(mod, ID2SYM(id));
3819 }
3820 }
3821}
3822
3823void
3824rb_deprecate_constant(VALUE mod, const char *name)
3825{
3826 rb_const_entry_t *ce;
3827 ID id;
3828 long len = strlen(name);
3829
3831 if (!(id = rb_check_id_cstr(name, len, NULL))) {
3832 undefined_constant(mod, rb_fstring_new(name, len));
3833 }
3834 if (!(ce = rb_const_lookup(mod, id))) {
3835 undefined_constant(mod, ID2SYM(id));
3836 }
3837 ce->flag |= CONST_DEPRECATED;
3838}
3839
3840/*
3841 * call-seq:
3842 * mod.private_constant(symbol, ...) => mod
3843 *
3844 * Makes a list of existing constants private.
3845 */
3846
3847VALUE
3848rb_mod_private_constant(int argc, const VALUE *argv, VALUE obj)
3849{
3850 set_const_visibility(obj, argc, argv, CONST_PRIVATE, CONST_VISIBILITY_MASK);
3851 return obj;
3852}
3853
3854/*
3855 * call-seq:
3856 * mod.public_constant(symbol, ...) => mod
3857 *
3858 * Makes a list of existing constants public.
3859 */
3860
3861VALUE
3862rb_mod_public_constant(int argc, const VALUE *argv, VALUE obj)
3863{
3864 set_const_visibility(obj, argc, argv, CONST_PUBLIC, CONST_VISIBILITY_MASK);
3865 return obj;
3866}
3867
3868/*
3869 * call-seq:
3870 * mod.deprecate_constant(symbol, ...) => mod
3871 *
3872 * Makes a list of existing constants deprecated. Attempt
3873 * to refer to them will produce a warning.
3874 *
3875 * module HTTP
3876 * NotFound = Exception.new
3877 * NOT_FOUND = NotFound # previous version of the library used this name
3878 *
3879 * deprecate_constant :NOT_FOUND
3880 * end
3881 *
3882 * HTTP::NOT_FOUND
3883 * # warning: constant HTTP::NOT_FOUND is deprecated
3884 *
3885 */
3886
3887VALUE
3888rb_mod_deprecate_constant(int argc, const VALUE *argv, VALUE obj)
3889{
3890 set_const_visibility(obj, argc, argv, CONST_DEPRECATED, CONST_DEPRECATED);
3891 return obj;
3892}
3893
3894static VALUE
3895original_module(VALUE c)
3896{
3897 if (RB_TYPE_P(c, T_ICLASS))
3898 return RBASIC(c)->klass;
3899 return c;
3900}
3901
3902static int
3903cvar_lookup_at(VALUE klass, ID id, st_data_t *v)
3904{
3905 if (RB_TYPE_P(klass, T_ICLASS)) {
3906 if (FL_TEST_RAW(klass, RICLASS_IS_ORIGIN)) {
3907 return 0;
3908 }
3909 else {
3910 // check the original module
3911 klass = RBASIC(klass)->klass;
3912 }
3913 }
3914
3915 VALUE n = rb_ivar_lookup(klass, id, Qundef);
3916 if (UNDEF_P(n)) return 0;
3917
3918 if (v) *v = n;
3919 return 1;
3920}
3921
3922static VALUE
3923cvar_front_klass(VALUE klass)
3924{
3925 if (RCLASS_SINGLETON_P(klass)) {
3926 VALUE obj = RCLASS_ATTACHED_OBJECT(klass);
3927 if (rb_namespace_p(obj)) {
3928 return obj;
3929 }
3930 }
3931 return RCLASS_SUPER(klass);
3932}
3933
3934static void
3935cvar_overtaken(VALUE front, VALUE target, ID id)
3936{
3937 if (front && target != front) {
3938 if (original_module(front) != original_module(target)) {
3939 rb_raise(rb_eRuntimeError,
3940 "class variable % "PRIsVALUE" of %"PRIsVALUE" is overtaken by %"PRIsVALUE"",
3941 ID2SYM(id), rb_class_name(original_module(front)),
3942 rb_class_name(original_module(target)));
3943 }
3944 if (BUILTIN_TYPE(front) == T_CLASS) {
3945 rb_ivar_delete(front, id, Qundef);
3946 }
3947 }
3948}
3949
3950#define CVAR_FOREACH_ANCESTORS(klass, v, r) \
3951 for (klass = cvar_front_klass(klass); klass; klass = RCLASS_SUPER(klass)) { \
3952 if (cvar_lookup_at(klass, id, (v))) { \
3953 r; \
3954 } \
3955 }
3956
3957#define CVAR_LOOKUP(v,r) do {\
3958 CVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR(); \
3959 if (cvar_lookup_at(klass, id, (v))) {r;}\
3960 CVAR_FOREACH_ANCESTORS(klass, v, r);\
3961} while(0)
3962
3963static VALUE
3964find_cvar(VALUE klass, VALUE * front, VALUE * target, ID id)
3965{
3966 VALUE v = Qundef;
3967 CVAR_LOOKUP(&v, {
3968 if (!*front) {
3969 *front = klass;
3970 }
3971 *target = klass;
3972 });
3973
3974 return v;
3975}
3976
3977static void
3978check_for_cvar_table(VALUE subclass, VALUE key)
3979{
3980 // Must not check ivar on ICLASS
3981 if (!RB_TYPE_P(subclass, T_ICLASS) && RTEST(rb_ivar_defined(subclass, key))) {
3982 RB_DEBUG_COUNTER_INC(cvar_class_invalidate);
3983 ruby_vm_global_cvar_state++;
3984 return;
3985 }
3986
3987 rb_class_foreach_subclass(subclass, check_for_cvar_table, key);
3988}
3989
3990void
3991rb_cvar_set(VALUE klass, ID id, VALUE val)
3992{
3993 VALUE tmp, front = 0, target = 0;
3994
3995 tmp = klass;
3996 CVAR_LOOKUP(0, {if (!front) front = klass; target = klass;});
3997 if (target) {
3998 cvar_overtaken(front, target, id);
3999 }
4000 else {
4001 target = tmp;
4002 }
4003
4004 if (RB_TYPE_P(target, T_ICLASS)) {
4005 target = RBASIC(target)->klass;
4006 }
4007 check_before_mod_set(target, id, val, "class variable");
4008
4009 int result = rb_class_ivar_set(target, id, val);
4010
4011 struct rb_id_table *rb_cvc_tbl = RCLASS_CVC_TBL(target);
4012
4013 if (!rb_cvc_tbl) {
4014 rb_cvc_tbl = RCLASS_CVC_TBL(target) = rb_id_table_create(2);
4015 }
4016
4017 struct rb_cvar_class_tbl_entry *ent;
4018 VALUE ent_data;
4019
4020 if (!rb_id_table_lookup(rb_cvc_tbl, id, &ent_data)) {
4021 ent = ALLOC(struct rb_cvar_class_tbl_entry);
4022 ent->class_value = target;
4023 ent->global_cvar_state = GET_GLOBAL_CVAR_STATE();
4024 ent->cref = 0;
4025 rb_id_table_insert(rb_cvc_tbl, id, (VALUE)ent);
4026 RB_DEBUG_COUNTER_INC(cvar_inline_miss);
4027 }
4028 else {
4029 ent = (void *)ent_data;
4030 ent->global_cvar_state = GET_GLOBAL_CVAR_STATE();
4031 }
4032
4033 // Break the cvar cache if this is a new class variable
4034 // and target is a module or a subclass with the same
4035 // cvar in this lookup.
4036 if (result == 0) {
4037 if (RB_TYPE_P(target, T_CLASS)) {
4038 if (RCLASS_SUBCLASSES(target)) {
4039 rb_class_foreach_subclass(target, check_for_cvar_table, id);
4040 }
4041 }
4042 }
4043}
4044
4045VALUE
4046rb_cvar_find(VALUE klass, ID id, VALUE *front)
4047{
4048 VALUE target = 0;
4049 VALUE value;
4050
4051 value = find_cvar(klass, front, &target, id);
4052 if (!target) {
4053 rb_name_err_raise("uninitialized class variable %1$s in %2$s",
4054 klass, ID2SYM(id));
4055 }
4056 cvar_overtaken(*front, target, id);
4057 return (VALUE)value;
4058}
4059
4060VALUE
4062{
4063 VALUE front = 0;
4064 return rb_cvar_find(klass, id, &front);
4065}
4066
4067VALUE
4069{
4070 if (!klass) return Qfalse;
4071 CVAR_LOOKUP(0,return Qtrue);
4072 return Qfalse;
4073}
4074
4075static ID
4076cv_intern(VALUE klass, const char *name)
4077{
4078 ID id = rb_intern(name);
4079 if (!rb_is_class_id(id)) {
4080 rb_name_err_raise("wrong class variable name %1$s",
4081 klass, rb_str_new_cstr(name));
4082 }
4083 return id;
4084}
4085
4086void
4087rb_cv_set(VALUE klass, const char *name, VALUE val)
4088{
4089 ID id = cv_intern(klass, name);
4090 rb_cvar_set(klass, id, val);
4091}
4092
4093VALUE
4094rb_cv_get(VALUE klass, const char *name)
4095{
4096 ID id = cv_intern(klass, name);
4097 return rb_cvar_get(klass, id);
4098}
4099
4100void
4101rb_define_class_variable(VALUE klass, const char *name, VALUE val)
4102{
4103 rb_cv_set(klass, name, val);
4104}
4105
4106static int
4107cv_i(ID key, VALUE v, st_data_t a)
4108{
4109 st_table *tbl = (st_table *)a;
4110
4111 if (rb_is_class_id(key)) {
4112 st_update(tbl, (st_data_t)key, cv_i_update, 0);
4113 }
4114 return ST_CONTINUE;
4115}
4116
4117static void*
4118mod_cvar_at(VALUE mod, void *data)
4119{
4120 st_table *tbl = data;
4121 if (!tbl) {
4122 tbl = st_init_numtable();
4123 }
4124 mod = original_module(mod);
4125
4126 rb_ivar_foreach(mod, cv_i, (st_data_t)tbl);
4127 return tbl;
4128}
4129
4130static void*
4131mod_cvar_of(VALUE mod, void *data)
4132{
4133 VALUE tmp = mod;
4134 if (RCLASS_SINGLETON_P(mod)) {
4135 if (rb_namespace_p(RCLASS_ATTACHED_OBJECT(mod))) {
4136 data = mod_cvar_at(tmp, data);
4137 tmp = cvar_front_klass(tmp);
4138 }
4139 }
4140 for (;;) {
4141 data = mod_cvar_at(tmp, data);
4142 tmp = RCLASS_SUPER(tmp);
4143 if (!tmp) break;
4144 }
4145 return data;
4146}
4147
4148static int
4149cv_list_i(st_data_t key, st_data_t value, VALUE ary)
4150{
4151 ID sym = (ID)key;
4152 rb_ary_push(ary, ID2SYM(sym));
4153 return ST_CONTINUE;
4154}
4155
4156static VALUE
4157cvar_list(void *data)
4158{
4159 st_table *tbl = data;
4160 VALUE ary;
4161
4162 if (!tbl) return rb_ary_new2(0);
4163 ary = rb_ary_new2(tbl->num_entries);
4164 st_foreach_safe(tbl, cv_list_i, ary);
4165 st_free_table(tbl);
4166
4167 return ary;
4168}
4169
4170/*
4171 * call-seq:
4172 * mod.class_variables(inherit=true) -> array
4173 *
4174 * Returns an array of the names of class variables in <i>mod</i>.
4175 * This includes the names of class variables in any included
4176 * modules, unless the <i>inherit</i> parameter is set to
4177 * <code>false</code>.
4178 *
4179 * class One
4180 * @@var1 = 1
4181 * end
4182 * class Two < One
4183 * @@var2 = 2
4184 * end
4185 * One.class_variables #=> [:@@var1]
4186 * Two.class_variables #=> [:@@var2, :@@var1]
4187 * Two.class_variables(false) #=> [:@@var2]
4188 */
4189
4190VALUE
4191rb_mod_class_variables(int argc, const VALUE *argv, VALUE mod)
4192{
4193 bool inherit = true;
4194 st_table *tbl;
4195
4196 if (rb_check_arity(argc, 0, 1)) inherit = RTEST(argv[0]);
4197 if (inherit) {
4198 tbl = mod_cvar_of(mod, 0);
4199 }
4200 else {
4201 tbl = mod_cvar_at(mod, 0);
4202 }
4203 return cvar_list(tbl);
4204}
4205
4206/*
4207 * call-seq:
4208 * remove_class_variable(sym) -> obj
4209 *
4210 * Removes the named class variable from the receiver, returning that
4211 * variable's value.
4212 *
4213 * class Example
4214 * @@var = 99
4215 * puts remove_class_variable(:@@var)
4216 * p(defined? @@var)
4217 * end
4218 *
4219 * <em>produces:</em>
4220 *
4221 * 99
4222 * nil
4223 */
4224
4225VALUE
4227{
4228 const ID id = id_for_var_message(mod, name, class, "wrong class variable name %1$s");
4229 st_data_t val;
4230
4231 if (!id) {
4232 goto not_defined;
4233 }
4234 rb_check_frozen(mod);
4235 val = rb_ivar_delete(mod, id, Qundef);
4236 if (!UNDEF_P(val)) {
4237 return (VALUE)val;
4238 }
4239 if (rb_cvar_defined(mod, id)) {
4240 rb_name_err_raise("cannot remove %1$s for %2$s", mod, ID2SYM(id));
4241 }
4242 not_defined:
4243 rb_name_err_raise("class variable %1$s not defined for %2$s",
4244 mod, name);
4246}
4247
4248VALUE
4249rb_iv_get(VALUE obj, const char *name)
4250{
4251 ID id = rb_check_id_cstr(name, strlen(name), rb_usascii_encoding());
4252
4253 if (!id) {
4254 return Qnil;
4255 }
4256 return rb_ivar_get(obj, id);
4257}
4258
4259VALUE
4260rb_iv_set(VALUE obj, const char *name, VALUE val)
4261{
4262 ID id = rb_intern(name);
4263
4264 return rb_ivar_set(obj, id, val);
4265}
4266
4267static VALUE *
4268class_ivar_set_shape_ivptr(VALUE obj, void *_data)
4269{
4270 RUBY_ASSERT(!rb_shape_obj_too_complex(obj));
4271
4272 return RCLASS_IVPTR(obj);
4273}
4274
4275static void
4276class_ivar_set_shape_resize_ivptr(VALUE obj, attr_index_t _old_capa, attr_index_t new_capa, void *_data)
4277{
4278 REALLOC_N(RCLASS_IVPTR(obj), VALUE, new_capa);
4279}
4280
4281static void
4282class_ivar_set_set_shape(VALUE obj, rb_shape_t *shape, void *_data)
4283{
4284 rb_shape_set_shape(obj, shape);
4285}
4286
4287static void
4288class_ivar_set_transition_too_complex(VALUE obj, void *_data)
4289{
4290 rb_evict_ivars_to_hash(obj);
4291}
4292
4293static st_table *
4294class_ivar_set_too_complex_table(VALUE obj, void *_data)
4295{
4296 RUBY_ASSERT(rb_shape_obj_too_complex(obj));
4297
4298 return RCLASS_IV_HASH(obj);
4299}
4300
4301int
4302rb_class_ivar_set(VALUE obj, ID id, VALUE val)
4303{
4305 bool existing = false;
4306 rb_check_frozen(obj);
4307
4308 RB_VM_LOCK_ENTER();
4309 {
4310 existing = general_ivar_set(obj, id, val, NULL,
4311 class_ivar_set_shape_ivptr,
4312 class_ivar_set_shape_resize_ivptr,
4313 class_ivar_set_set_shape,
4314 class_ivar_set_transition_too_complex,
4315 class_ivar_set_too_complex_table).existing;
4316 }
4317 RB_VM_LOCK_LEAVE();
4318
4319 return existing;
4320}
4321
4322static int
4323tbl_copy_i(ID key, VALUE val, st_data_t dest)
4324{
4325 rb_class_ivar_set((VALUE)dest, key, val);
4326
4327 return ST_CONTINUE;
4328}
4329
4330void
4331rb_iv_tbl_copy(VALUE dst, VALUE src)
4332{
4333 RUBY_ASSERT(rb_type(dst) == rb_type(src));
4335
4336 RUBY_ASSERT(rb_shape_get_shape(dst)->type == SHAPE_ROOT);
4337 RUBY_ASSERT(!RCLASS_IVPTR(dst));
4338
4339 rb_ivar_foreach(src, tbl_copy_i, dst);
4340}
4341
4343rb_const_lookup(VALUE klass, ID id)
4344{
4345 struct rb_id_table *tbl = RCLASS_CONST_TBL(klass);
4346
4347 if (tbl) {
4348 VALUE val;
4349 bool r;
4350 RB_VM_LOCK_ENTER();
4351 {
4352 r = rb_id_table_lookup(tbl, id, &val);
4353 }
4354 RB_VM_LOCK_LEAVE();
4355
4356 if (r) return (rb_const_entry_t *)val;
4357 }
4358 return NULL;
4359}
#define RUBY_ASSERT(...)
Asserts that the given expression is truthy if and only if RUBY_DEBUG is truthy.
Definition assert.h:219
#define RUBY_EXTERN
Declaration of externally visible global variables.
Definition dllexport.h:45
static VALUE RB_OBJ_FROZEN_RAW(VALUE obj)
This is an implementation detail of RB_OBJ_FROZEN().
Definition fl_type.h:883
static bool RB_FL_ABLE(VALUE obj)
Checks if the object is flaggable.
Definition fl_type.h:444
static void RB_FL_SET_RAW(VALUE obj, VALUE flags)
This is an implementation detail of RB_FL_SET().
Definition fl_type.h:606
void rb_obj_freeze_inline(VALUE obj)
Prevents further modifications to the given object.
Definition variable.c:1877
static void RB_FL_UNSET_RAW(VALUE obj, VALUE flags)
This is an implementation detail of RB_FL_UNSET().
Definition fl_type.h:666
@ RUBY_FL_FREEZE
This flag has something to do with data immutability.
Definition fl_type.h:324
void rb_class_modify_check(VALUE klass)
Asserts that klass is not a frozen class.
Definition eval.c:418
void rb_freeze_singleton_class(VALUE x)
This is an implementation detail of RB_OBJ_FREEZE().
Definition class.c:2266
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.
Definition class.c:2638
#define rb_str_new2
Old name of rb_str_new_cstr.
Definition string.h:1675
#define TYPE(_)
Old name of rb_type.
Definition value_type.h:108
#define FL_EXIVAR
Old name of RUBY_FL_EXIVAR.
Definition fl_type.h:66
#define FL_USER3
Old name of RUBY_FL_USER3.
Definition fl_type.h:74
#define REALLOC_N
Old name of RB_REALLOC_N.
Definition memory.h:403
#define ALLOC
Old name of RB_ALLOC.
Definition memory.h:400
#define T_STRING
Old name of RUBY_T_STRING.
Definition value_type.h:78
#define xfree
Old name of ruby_xfree.
Definition xmalloc.h:58
#define Qundef
Old name of RUBY_Qundef.
#define OBJ_FROZEN
Old name of RB_OBJ_FROZEN.
Definition fl_type.h:137
#define rb_str_cat2
Old name of rb_str_cat_cstr.
Definition string.h:1683
#define xrealloc
Old name of ruby_xrealloc.
Definition xmalloc.h:56
#define ID2SYM
Old name of RB_ID2SYM.
Definition symbol.h:44
#define SPECIAL_CONST_P
Old name of RB_SPECIAL_CONST_P.
#define OBJ_FREEZE
Old name of RB_OBJ_FREEZE.
Definition fl_type.h:135
#define UNREACHABLE_RETURN
Old name of RBIMPL_UNREACHABLE_RETURN.
Definition assume.h:29
#define ZALLOC
Old name of RB_ZALLOC.
Definition memory.h:402
#define CLASS_OF
Old name of rb_class_of.
Definition globals.h:203
#define xmalloc
Old name of ruby_xmalloc.
Definition xmalloc.h:53
#define T_MODULE
Old name of RUBY_T_MODULE.
Definition value_type.h:70
#define ASSUME
Old name of RBIMPL_ASSUME.
Definition assume.h:27
#define T_ICLASS
Old name of RUBY_T_ICLASS.
Definition value_type.h:66
#define ALLOC_N
Old name of RB_ALLOC_N.
Definition memory.h:399
#define FL_TEST_RAW
Old name of RB_FL_TEST_RAW.
Definition fl_type.h:132
#define FL_SET
Old name of RB_FL_SET.
Definition fl_type.h:129
#define rb_ary_new3
Old name of rb_ary_new_from_args.
Definition array.h:658
#define FL_USER2
Old name of RUBY_FL_USER2.
Definition fl_type.h:73
#define Qtrue
Old name of RUBY_Qtrue.
#define INT2NUM
Old name of RB_INT2NUM.
Definition int.h:43
#define Qnil
Old name of RUBY_Qnil.
#define Qfalse
Old name of RUBY_Qfalse.
#define T_OBJECT
Old name of RUBY_T_OBJECT.
Definition value_type.h:75
#define NIL_P
Old name of RB_NIL_P.
#define ALLOCV_N
Old name of RB_ALLOCV_N.
Definition memory.h:405
#define T_CLASS
Old name of RUBY_T_CLASS.
Definition value_type.h:58
#define BUILTIN_TYPE
Old name of RB_BUILTIN_TYPE.
Definition value_type.h:85
#define FL_TEST
Old name of RB_FL_TEST.
Definition fl_type.h:131
#define FL_UNSET
Old name of RB_FL_UNSET.
Definition fl_type.h:133
#define rb_ary_new2
Old name of rb_ary_new_capa.
Definition array.h:657
#define FL_SET_RAW
Old name of RB_FL_SET_RAW.
Definition fl_type.h:130
#define ALLOCV_END
Old name of RB_ALLOCV_END.
Definition memory.h:406
void rb_category_warn(rb_warning_category_t category, const char *fmt,...)
Identical to rb_category_warning(), except it reports unless $VERBOSE is nil.
Definition error.c:476
void rb_name_error(ID id, const char *fmt,...)
Raises an instance of rb_eNameError.
Definition error.c:2344
VALUE rb_eTypeError
TypeError exception.
Definition error.c:1430
void rb_name_error_str(VALUE str, const char *fmt,...)
Identical to rb_name_error(), except it takes a VALUE instead of ID.
Definition error.c:2359
VALUE rb_eNameError
NameError exception.
Definition error.c:1435
VALUE rb_eRuntimeError
RuntimeError exception.
Definition error.c:1428
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.
Definition error.c:1397
void rb_warn(const char *fmt,...)
Identical to rb_warning(), except it reports unless $VERBOSE is nil.
Definition error.c:466
void rb_warning(const char *fmt,...)
Issues a warning.
Definition error.c:497
@ RB_WARN_CATEGORY_DEPRECATED
Warning is for deprecated features.
Definition error.h:48
VALUE rb_obj_hide(VALUE obj)
Make the object invisible from Ruby code.
Definition object.c:104
VALUE rb_obj_class(VALUE obj)
Queries the class of an object.
Definition object.c:247
VALUE rb_cModule
Module class.
Definition object.c:67
VALUE rb_class_real(VALUE klass)
Finds a "real" class.
Definition object.c:237
#define RB_OBJ_WRITTEN(old, oldv, young)
Identical to RB_OBJ_WRITE(), except it doesn't write any values, but only a WB declaration.
Definition gc.h:615
#define RB_OBJ_WRITE(old, slot, young)
Declaration of a "back" pointer.
Definition gc.h:603
Encoding relates APIs.
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.
Definition symbol.c:1231
VALUE rb_funcall(VALUE recv, ID mid, int n,...)
Calls a method.
Definition vm_eval.c:1099
static int rb_check_arity(int argc, int min, int max)
Ensures that the passed integer is in the passed range.
Definition error.h:284
#define st_foreach_safe
Just another name of rb_st_foreach_safe.
Definition hash.h:51
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...
Definition load.c:687
VALUE rb_backref_get(void)
Queries the last match, or Regexp.last_match, or the $~.
Definition vm.c:1836
int rb_is_instance_id(ID id)
Classifies the given ID, then sees if it is an instance variable.
Definition symbol.c:1097
int rb_is_const_id(ID id)
Classifies the given ID, then sees if it is a constant.
Definition symbol.c:1079
int rb_is_class_id(ID id)
Classifies the given ID, then sees if it is a class variable.
Definition symbol.c:1085
VALUE rb_block_proc(void)
Constructs a Proc object from implicitly passed components.
Definition proc.c:839
VALUE rb_reg_nth_defined(int n, VALUE md)
Identical to rb_reg_nth_match(), except it just returns Boolean.
Definition re.c:1909
VALUE rb_str_append(VALUE dst, VALUE src)
Identical to rb_str_buf_append(), except it converts the right hand side before concatenating.
Definition string.c:3695
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...
Definition string.c:3070
VALUE rb_str_new_frozen(VALUE str)
Creates a frozen copy of the string, if necessary.
Definition string.c:1470
VALUE rb_str_dup(VALUE str)
Duplicates a string.
Definition string.c:1933
#define rb_str_new_cstr(str)
Identical to rb_str_new, except it assumes the passed pointer is a pointer to a C string.
Definition string.h:1514
VALUE rb_str_intern(VALUE str)
Identical to rb_to_symbol(), except it assumes the receiver being an instance of RString.
Definition symbol.c:894
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.
Definition variable.c:4226
VALUE rb_obj_instance_variables(VALUE obj)
Resembles Object#instance_variables.
Definition variable.c:2238
VALUE rb_f_untrace_var(int argc, const VALUE *argv)
Deletes the passed tracer from the passed global variable, or if omitted, deletes everything.
Definition variable.c:893
VALUE rb_const_get(VALUE space, ID name)
Identical to rb_const_defined(), except it returns the actual defined value.
Definition variable.c:3193
VALUE rb_const_list(void *)
This is another mysterious API that comes with no documents at all.
Definition variable.c:3427
VALUE rb_path2class(const char *path)
Resolves a Q::W::E::R-style path string to the actual class it points.
Definition variable.c:486
VALUE rb_autoload_p(VALUE space, ID name)
Queries if an autoload is defined at a point.
Definition variable.c:3066
void rb_set_class_path(VALUE klass, VALUE space, const char *name)
Names a class.
Definition variable.c:433
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.
Definition variable.c:1924
VALUE rb_mod_remove_const(VALUE space, VALUE name)
Resembles Module#remove_const.
Definition variable.c:3298
VALUE rb_class_path_cached(VALUE mod)
Just another name of rb_mod_name.
Definition variable.c:382
VALUE rb_f_trace_var(int argc, const VALUE *argv)
Traces a global variable.
Definition variable.c:847
void rb_cvar_set(VALUE klass, ID name, VALUE val)
Assigns a value to a class variable.
Definition variable.c:3991
VALUE rb_cvar_get(VALUE klass, ID name)
Obtains a value from a class variable.
Definition variable.c:4061
VALUE rb_mod_constants(int argc, const VALUE *argv, VALUE recv)
Resembles Module#constants.
Definition variable.c:3459
VALUE rb_cvar_find(VALUE klass, ID name, VALUE *front)
Identical to rb_cvar_get(), except it takes additional "front" pointer.
Definition variable.c:4046
VALUE rb_path_to_class(VALUE path)
Identical to rb_path2class(), except it accepts the path as Ruby's string instead of C's.
Definition variable.c:441
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.
Definition variable.c:1415
void rb_const_set(VALUE space, ID name, VALUE val)
Names a constant.
Definition variable.c:3668
VALUE rb_autoload_load(VALUE space, ID name)
Kicks the autoload procedure as if it was "touched".
Definition variable.c:3028
VALUE rb_mod_name(VALUE mod)
Queries the name of a module.
Definition variable.c:130
VALUE rb_class_name(VALUE obj)
Queries the name of the given object's class.
Definition variable.c:492
VALUE rb_const_get_at(VALUE space, ID name)
Identical to rb_const_defined_at(), except it returns the actual defined value.
Definition variable.c:3199
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.
Definition variable.c:416
void rb_alias_variable(ID dst, ID src)
Aliases a global variable.
Definition variable.c:1079
void rb_define_class_variable(VALUE, const char *, VALUE)
Just another name of rb_cv_set.
Definition variable.c:4101
VALUE rb_obj_remove_instance_variable(VALUE obj, VALUE name)
Resembles Object#remove_instance_variable.
Definition variable.c:2292
void * rb_mod_const_of(VALUE, void *)
This is a variant of rb_mod_const_at().
Definition variable.c:3405
st_index_t rb_ivar_count(VALUE obj)
Number of instance variables defined on an object.
Definition variable.c:2186
void * rb_mod_const_at(VALUE, void *)
This API is mysterious.
Definition variable.c:3388
VALUE rb_const_remove(VALUE space, ID name)
Identical to rb_mod_remove_const(), except it takes the name as ID instead of VALUE.
Definition variable.c:3309
VALUE rb_const_get_from(VALUE space, ID name)
Identical to rb_const_defined_at(), except it returns the actual defined value.
Definition variable.c:3187
VALUE rb_ivar_defined(VALUE obj, ID name)
Queries if the instance variable is defined at the object.
Definition variable.c:1941
VALUE rb_cv_get(VALUE klass, const char *name)
Identical to rb_cvar_get(), except it accepts C's string instead of ID.
Definition variable.c:4094
int rb_const_defined_at(VALUE space, ID name)
Identical to rb_const_defined(), except it doesn't look for parent classes.
Definition variable.c:3521
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.
Definition variable.c:4087
VALUE rb_mod_class_variables(int argc, const VALUE *argv, VALUE recv)
Resembles Module#class_variables.
Definition variable.c:4191
VALUE rb_f_global_variables(void)
Queries the list of global variables.
Definition variable.c:1047
VALUE rb_cvar_defined(VALUE klass, ID name)
Queries if the given class has the given class variable.
Definition variable.c:4068
VALUE rb_class_path(VALUE mod)
Identical to rb_mod_name(), except it returns #<Class: ...> style inspection for anonymous modules.
Definition variable.c:373
int rb_const_defined_from(VALUE space, ID name)
Identical to rb_const_defined(), except it returns false for private constants.
Definition variable.c:3509
int rb_const_defined(VALUE space, ID name)
Queries if the constant is defined at the namespace.
Definition variable.c:3515
void rb_free_generic_ivar(VALUE obj)
Frees the list of instance variables.
Definition variable.c:1220
const char * rb_sourcefile(void)
Resembles __FILE__.
Definition vm.c:1873
void rb_clear_constant_cache_for_id(ID id)
Clears the inline constant caches associated with a particular ID.
Definition vm_method.c:138
VALUE rb_eval_cmd_kw(VALUE cmd, VALUE arg, int kw_splat)
This API is practically a variant of rb_proc_call_kw() now.
Definition vm_eval.c:2123
static ID rb_intern_const(const char *str)
This is a "tiny optimisation" over rb_intern().
Definition symbol.h:284
ID rb_check_id(volatile VALUE *namep)
Detects if the given name is already interned or not.
Definition symbol.c:1133
ID rb_to_id(VALUE str)
Definition string.c:12512
rb_gvar_setter_t rb_gvar_var_setter
Definition variable.h:119
rb_gvar_marker_t rb_gvar_var_marker
Definition variable.h:128
void rb_define_global_const(const char *name, VALUE val)
Identical to rb_define_const(), except it defines that of "global", i.e.
Definition variable.c:3776
VALUE rb_gv_get(const char *name)
Obtains a global variable.
Definition variable.c:1005
void rb_define_variable(const char *name, VALUE *var)
"Shares" a global variable between Ruby and C.
Definition variable.c:818
void rb_gvar_marker_t(VALUE *var)
Type that represents a global variable marker function.
Definition variable.h:53
void rb_deprecate_constant(VALUE mod, const char *name)
Asserts that the given constant is deprecated.
Definition variable.c:3824
void rb_gvar_setter_t(VALUE val, ID id, VALUE *data)
Type that represents a global variable setter function.
Definition variable.h:46
rb_gvar_setter_t rb_gvar_val_setter
This is the setter function that backs global variables defined from a ruby script.
Definition variable.h:94
rb_gvar_marker_t rb_gvar_undef_marker
Definition variable.h:80
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...
Definition variable.c:824
rb_gvar_setter_t rb_gvar_readonly_setter
This function just raises rb_eNameError.
Definition variable.h:135
rb_gvar_getter_t rb_gvar_undef_getter
Definition variable.h:62
VALUE rb_gv_set(const char *name, VALUE val)
Assigns to a global variable.
Definition variable.c:991
rb_gvar_marker_t rb_gvar_val_marker
This is the setter function that backs global variables defined from a ruby script.
Definition variable.h:101
VALUE rb_gvar_getter_t(ID id, VALUE *data)
Type that represents a global variable getter function.
Definition variable.h:37
VALUE rb_iv_get(VALUE obj, const char *name)
Obtains an instance variable.
Definition variable.c:4249
rb_gvar_setter_t rb_gvar_undef_setter
Definition variable.h:71
rb_gvar_getter_t rb_gvar_val_getter
This is the getter function that backs global variables defined from a ruby script.
Definition variable.h:87
VALUE rb_iv_set(VALUE obj, const char *name, VALUE val)
Assigns to an instance variable.
Definition variable.c:4260
rb_gvar_getter_t rb_gvar_var_getter
Definition variable.h:110
int len
Length of the buffer.
Definition io.h:8
static bool rb_ractor_shareable_p(VALUE obj)
Queries if multiple Ractors can share the passed object or not.
Definition ractor.h:249
#define MEMCPY(p1, p2, type, n)
Handy macro to call memcpy.
Definition memory.h:372
#define RB_GC_GUARD(v)
Prevents premature destruction of local objects.
Definition memory.h:167
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.
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.
Definition variable.c:2100
#define RARRAY_LEN
Just another name of rb_array_len.
Definition rarray.h:51
static VALUE RBASIC_CLASS(VALUE obj)
Queries the class of an object.
Definition rbasic.h:150
#define RBASIC(obj)
Convenient casting macro.
Definition rbasic.h:40
#define RCLASS_SUPER
Just another name of rb_class_get_superclass.
Definition rclass.h:44
#define ROBJECT(obj)
Convenient casting macro.
Definition robject.h:43
static VALUE * ROBJECT_IVPTR(VALUE obj)
Queries the instance variables.
Definition robject.h:126
#define StringValue(v)
Ensures that the parameter object is a String.
Definition rstring.h:66
static char * RSTRING_END(VALUE str)
Queries the end of the contents pointer of the string.
Definition rstring.h:442
#define RTYPEDDATA_DATA(v)
Convenient getter macro.
Definition rtypeddata.h:102
#define TypedData_Wrap_Struct(klass, data_type, sval)
Converts sval, a pointer to your struct, into a Ruby object.
Definition rtypeddata.h:449
#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...
Definition rtypeddata.h:497
const char * rb_class2name(VALUE klass)
Queries the name of the passed class.
Definition variable.c:498
const char * rb_obj_classname(VALUE obj)
Queries the name of the class of the passed object.
Definition variable.c:507
#define RB_NO_KEYWORDS
Do not pass keywords.
Definition scan_args.h:69
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.
Definition stdarg.h:35
C99 shim for <stdbool.h>
Definition constant.h:33
Definition class.h:36
This is the struct that holds necessary info for a struct.
Definition rtypeddata.h:200
Definition variable.c:530
Definition st.h:79
uintptr_t ID
Type that represents a Ruby identifier such as a variable name.
Definition value.h:52
uintptr_t VALUE
Type that represents a Ruby object.
Definition value.h:40
static enum ruby_value_type rb_type(VALUE obj)
Identical to RB_BUILTIN_TYPE(), except it can also accept special constants.
Definition value_type.h:225
static enum ruby_value_type RB_BUILTIN_TYPE(VALUE obj)
Queries the type of the object.
Definition value_type.h:182
static void Check_Type(VALUE v, enum ruby_value_type t)
Identical to RB_TYPE_P(), except it raises exceptions on predication failure.
Definition value_type.h:433
static bool RB_TYPE_P(VALUE obj, enum ruby_value_type t)
Queries if the given object is of given type.
Definition value_type.h:376