Ruby 4.1.0dev (2026-05-17 revision 71749b882622c03028dfc55f40e5c3cceb24eaaf)
gc.h
1#ifndef GC_GC_H
2#define GC_GC_H
12#include "ruby/ruby.h"
13#include "ruby/assert.h"
14
15#include "ruby/thread_native.h"
16
17#ifndef VM_CHECK_MODE
18# define VM_CHECK_MODE RUBY_DEBUG
19#endif
20
21// From ractor_core.h
22#ifndef RACTOR_CHECK_MODE
23# define RACTOR_CHECK_MODE (VM_CHECK_MODE || RUBY_DEBUG) && (SIZEOF_UINT64_T == SIZEOF_VALUE)
24#endif
25
26#if RACTOR_CHECK_MODE
27void rb_ractor_setup_belonging(VALUE obj);
28
29struct rb_gc_obj_suffix {
30 uint32_t _ractor_belonging_id;
31};
32
33# define RB_GC_OBJ_HAS_SUFFIX 1
34# define RB_GC_OBJ_SUFFIX_SIZE (sizeof(struct rb_gc_obj_suffix))
35#else
36# define RB_GC_OBJ_HAS_SUFFIX 0
37# define RB_GC_OBJ_SUFFIX_SIZE 0
38#endif
39
41 rb_nativethread_lock_t lock;
42
44};
45
46typedef int (*vm_table_foreach_callback_func)(VALUE value, void *data);
47typedef int (*vm_table_update_callback_func)(VALUE *value, void *data);
48
49enum rb_gc_vm_weak_tables {
50 RB_GC_VM_CI_TABLE,
51 RB_GC_VM_OVERLOADED_CME_TABLE,
52 RB_GC_VM_GLOBAL_SYMBOLS_TABLE,
53 RB_GC_VM_ID2REF_TABLE,
54 RB_GC_VM_GENERIC_FIELDS_TABLE,
55 RB_GC_VM_FROZEN_STRINGS_TABLE,
56 RB_GC_VM_WEAK_TABLE_COUNT
57};
58
59#define RB_GC_VM_LOCK() rb_gc_vm_lock(__FILE__, __LINE__)
60#define RB_GC_VM_UNLOCK(lev) rb_gc_vm_unlock(lev, __FILE__, __LINE__)
61#define RB_GC_CR_LOCK() rb_gc_cr_lock(__FILE__, __LINE__)
62#define RB_GC_CR_UNLOCK(lev) rb_gc_cr_unlock(lev, __FILE__, __LINE__)
63#define RB_GC_VM_LOCK_NO_BARRIER() rb_gc_vm_lock_no_barrier(__FILE__, __LINE__)
64#define RB_GC_VM_UNLOCK_NO_BARRIER(lev) rb_gc_vm_unlock_no_barrier(lev, __FILE__, __LINE__)
65
66#if USE_MODULAR_GC
67# define MODULAR_GC_FN
68#else
69// This takes advantage of internal linkage winning when appearing first.
70// See C99 6.2.2p4.
71# define MODULAR_GC_FN static
72#endif
73
74#if USE_MODULAR_GC
75RUBY_SYMBOL_EXPORT_BEGIN
76#endif
77
78// These functions cannot be defined as static because they are used by other
79// files in Ruby.
80size_t rb_size_mul_or_raise(size_t x, size_t y, VALUE exc);
81void rb_objspace_reachable_objects_from(VALUE obj, void (func)(VALUE, void *), void *data);
82const char *rb_raw_obj_info(char *const buff, const size_t buff_size, VALUE obj);
83const char *rb_obj_info(VALUE obj);
84size_t rb_obj_memsize_of(VALUE obj);
85bool ruby_free_at_exit_p(void);
86void rb_objspace_reachable_objects_from_root(void (func)(const char *category, VALUE, void *), void *passing_data);
87void rb_gc_verify_shareable(VALUE);
88
89MODULAR_GC_FN unsigned int rb_gc_vm_lock(const char *file, int line);
90MODULAR_GC_FN void rb_gc_vm_unlock(unsigned int lev, const char *file, int line);
91MODULAR_GC_FN unsigned int rb_gc_cr_lock(const char *file, int line);
92MODULAR_GC_FN void rb_gc_cr_unlock(unsigned int lev, const char *file, int line);
93MODULAR_GC_FN unsigned int rb_gc_vm_lock_no_barrier(const char *file, int line);
94MODULAR_GC_FN void rb_gc_vm_unlock_no_barrier(unsigned int lev, const char *file, int line);
95MODULAR_GC_FN void rb_gc_vm_barrier(void);
96MODULAR_GC_FN size_t rb_gc_obj_optimal_size(VALUE obj);
97MODULAR_GC_FN void rb_gc_mark_children(void *objspace, VALUE obj);
98MODULAR_GC_FN void rb_gc_vm_weak_table_foreach(vm_table_foreach_callback_func callback, vm_table_update_callback_func update_callback, void *data, bool weak_only, enum rb_gc_vm_weak_tables table);
99MODULAR_GC_FN void rb_gc_update_object_references(void *objspace, VALUE obj);
100MODULAR_GC_FN void rb_gc_update_vm_references(void *objspace);
101MODULAR_GC_FN void rb_gc_event_hook(VALUE obj, rb_event_flag_t event);
102MODULAR_GC_FN void *rb_gc_get_objspace(void);
103MODULAR_GC_FN void rb_gc_run_obj_finalizer(VALUE objid, long count, VALUE (*callback)(long i, void *data), void *data);
104MODULAR_GC_FN void rb_gc_set_pending_interrupt(void);
105MODULAR_GC_FN void rb_gc_unset_pending_interrupt(void);
106MODULAR_GC_FN void rb_gc_obj_free_vm_weak_references(VALUE obj);
107MODULAR_GC_FN bool rb_gc_obj_free(void *objspace, VALUE obj);
108MODULAR_GC_FN void rb_gc_save_machine_context(void);
109MODULAR_GC_FN void rb_gc_mark_roots(void *objspace, const char **categoryp);
110MODULAR_GC_FN void rb_gc_ractor_newobj_cache_foreach(void (*func)(void *cache, void *data), void *data);
111MODULAR_GC_FN bool rb_gc_multi_ractor_p(void);
112MODULAR_GC_FN bool rb_gc_shutdown_call_finalizer_p(VALUE obj);
113MODULAR_GC_FN void rb_gc_obj_changed_pool(VALUE obj, size_t heap_id);
114MODULAR_GC_FN void rb_gc_prepare_heap_process_object(VALUE obj);
115MODULAR_GC_FN bool rb_memerror_reentered(void);
116MODULAR_GC_FN bool rb_obj_id_p(VALUE);
117MODULAR_GC_FN void rb_gc_before_updating_jit_code(void);
118MODULAR_GC_FN void rb_gc_after_updating_jit_code(void);
119MODULAR_GC_FN bool rb_gc_obj_shareable_p(VALUE);
120MODULAR_GC_FN void rb_gc_rp(VALUE);
121MODULAR_GC_FN void rb_gc_handle_weak_references(VALUE obj);
122MODULAR_GC_FN bool rb_gc_obj_needs_cleanup_p(VALUE obj);
123
124#if USE_MODULAR_GC
125MODULAR_GC_FN bool rb_gc_event_hook_required_p(rb_event_flag_t event);
126MODULAR_GC_FN void *rb_gc_get_ractor_newobj_cache(void);
127MODULAR_GC_FN void rb_gc_initialize_vm_context(struct rb_gc_vm_context *context);
128MODULAR_GC_FN void rb_gc_move_obj_during_marking(VALUE from, VALUE to);
129MODULAR_GC_FN void rb_gc_print_backtrace();
130#endif
131
132#if USE_MODULAR_GC
133RUBY_SYMBOL_EXPORT_END
134#endif
135
136void rb_ractor_finish_marking(void);
137
138// -------------------Private section begin------------------------
139// Functions in this section are private to the default GC and gc.c
140
141#ifdef BUILDING_MODULAR_GC
143RBIMPL_WARNING_IGNORED(-Wunused-function)
144#endif
145
146/* RGENGC_CHECK_MODE
147 * 0: disable all assertions
148 * 1: enable assertions (to debug RGenGC)
149 * 2: enable internal consistency check at each GC (for debugging)
150 * 3: enable internal consistency check at each GC steps (for debugging)
151 * 4: enable liveness check
152 * 5: show all references
153 */
154#ifndef RGENGC_CHECK_MODE
155# define RGENGC_CHECK_MODE 0
156#endif
157
158#ifndef GC_ASSERT
159# define GC_ASSERT(expr, ...) RUBY_ASSERT_MESG_WHEN(RGENGC_CHECK_MODE > 0, expr, #expr RBIMPL_VA_OPT_ARGS(__VA_ARGS__))
160#endif
161
162static int
163hash_foreach_replace_value(st_data_t key, st_data_t value, st_data_t argp, int error)
164{
165 if (rb_gc_location((VALUE)value) != (VALUE)value) {
166 return ST_REPLACE;
167 }
168 return ST_CONTINUE;
169}
170
171static int
172hash_replace_ref_value(st_data_t *key, st_data_t *value, st_data_t argp, int existing)
173{
174 *value = rb_gc_location((VALUE)*value);
175
176 return ST_CONTINUE;
177}
178
179static void
180gc_ref_update_table_values_only(st_table *tbl)
181{
182 if (!tbl || tbl->num_entries == 0) return;
183
184 if (st_foreach_with_replace(tbl, hash_foreach_replace_value, hash_replace_ref_value, 0)) {
185 rb_raise(rb_eRuntimeError, "hash modified during iteration");
186 }
187}
188
189static int
190gc_mark_tbl_no_pin_i(st_data_t key, st_data_t value, st_data_t data)
191{
192 rb_gc_mark_movable((VALUE)value);
193
194 return ST_CONTINUE;
195}
196
197static int
198hash_foreach_replace(st_data_t key, st_data_t value, st_data_t argp, int error)
199{
200 if (rb_gc_location((VALUE)key) != (VALUE)key) {
201 return ST_REPLACE;
202 }
203
204 if (rb_gc_location((VALUE)value) != (VALUE)value) {
205 return ST_REPLACE;
206 }
207
208 return ST_CONTINUE;
209}
210
211static int
212hash_replace_ref(st_data_t *key, st_data_t *value, st_data_t argp, int existing)
213{
214 if (rb_gc_location((VALUE)*key) != (VALUE)*key) {
215 *key = rb_gc_location((VALUE)*key);
216 }
217
218 if (rb_gc_location((VALUE)*value) != (VALUE)*value) {
219 *value = rb_gc_location((VALUE)*value);
220 }
221
222 return ST_CONTINUE;
223}
224
225static void
226gc_update_table_refs(st_table *tbl)
227{
228 if (!tbl || tbl->num_entries == 0) return;
229
230 if (st_foreach_with_replace(tbl, hash_foreach_replace, hash_replace_ref, 0)) {
231 rb_raise(rb_eRuntimeError, "hash modified during iteration");
232 }
233}
234
235static inline size_t
236xmalloc2_size(const size_t count, const size_t elsize)
237{
238 return rb_size_mul_or_raise(count, elsize, rb_eArgError);
239}
240
241static VALUE
242type_sym(size_t type)
243{
244 switch (type) {
245#define COUNT_TYPE(t) case (t): return ID2SYM(rb_intern(#t)); break;
246 COUNT_TYPE(T_NONE);
247 COUNT_TYPE(T_OBJECT);
248 COUNT_TYPE(T_CLASS);
249 COUNT_TYPE(T_MODULE);
250 COUNT_TYPE(T_FLOAT);
251 COUNT_TYPE(T_STRING);
252 COUNT_TYPE(T_REGEXP);
253 COUNT_TYPE(T_ARRAY);
254 COUNT_TYPE(T_HASH);
255 COUNT_TYPE(T_STRUCT);
256 COUNT_TYPE(T_BIGNUM);
257 COUNT_TYPE(T_FILE);
258 COUNT_TYPE(T_DATA);
259 COUNT_TYPE(T_MATCH);
260 COUNT_TYPE(T_COMPLEX);
261 COUNT_TYPE(T_RATIONAL);
262 COUNT_TYPE(T_NIL);
263 COUNT_TYPE(T_TRUE);
264 COUNT_TYPE(T_FALSE);
265 COUNT_TYPE(T_SYMBOL);
266 COUNT_TYPE(T_FIXNUM);
267 COUNT_TYPE(T_IMEMO);
268 COUNT_TYPE(T_UNDEF);
269 COUNT_TYPE(T_NODE);
270 COUNT_TYPE(T_ICLASS);
271 COUNT_TYPE(T_ZOMBIE);
272 COUNT_TYPE(T_MOVED);
273#undef COUNT_TYPE
274 default: return SIZET2NUM(type); break;
275 }
276}
277
278#ifdef BUILDING_MODULAR_GC
280#endif
281// -------------------Private section end------------------------
282
283#endif
uint32_t rb_event_flag_t
Represents event(s).
Definition event.h:108
#define T_COMPLEX
Old name of RUBY_T_COMPLEX.
Definition value_type.h:59
#define T_FILE
Old name of RUBY_T_FILE.
Definition value_type.h:62
#define T_STRING
Old name of RUBY_T_STRING.
Definition value_type.h:78
#define T_NIL
Old name of RUBY_T_NIL.
Definition value_type.h:72
#define T_FLOAT
Old name of RUBY_T_FLOAT.
Definition value_type.h:64
#define T_IMEMO
Old name of RUBY_T_IMEMO.
Definition value_type.h:67
#define T_BIGNUM
Old name of RUBY_T_BIGNUM.
Definition value_type.h:57
#define T_STRUCT
Old name of RUBY_T_STRUCT.
Definition value_type.h:79
#define T_FIXNUM
Old name of RUBY_T_FIXNUM.
Definition value_type.h:63
#define T_DATA
Old name of RUBY_T_DATA.
Definition value_type.h:60
#define T_NONE
Old name of RUBY_T_NONE.
Definition value_type.h:74
#define T_NODE
Old name of RUBY_T_NODE.
Definition value_type.h:73
#define SIZET2NUM
Old name of RB_SIZE2NUM.
Definition size_t.h:62
#define T_MODULE
Old name of RUBY_T_MODULE.
Definition value_type.h:70
#define T_TRUE
Old name of RUBY_T_TRUE.
Definition value_type.h:81
#define T_RATIONAL
Old name of RUBY_T_RATIONAL.
Definition value_type.h:76
#define T_ICLASS
Old name of RUBY_T_ICLASS.
Definition value_type.h:66
#define T_HASH
Old name of RUBY_T_HASH.
Definition value_type.h:65
#define T_FALSE
Old name of RUBY_T_FALSE.
Definition value_type.h:61
#define T_UNDEF
Old name of RUBY_T_UNDEF.
Definition value_type.h:82
#define T_ZOMBIE
Old name of RUBY_T_ZOMBIE.
Definition value_type.h:83
#define T_ARRAY
Old name of RUBY_T_ARRAY.
Definition value_type.h:56
#define T_OBJECT
Old name of RUBY_T_OBJECT.
Definition value_type.h:75
#define T_SYMBOL
Old name of RUBY_T_SYMBOL.
Definition value_type.h:80
#define T_MATCH
Old name of RUBY_T_MATCH.
Definition value_type.h:69
#define T_CLASS
Old name of RUBY_T_CLASS.
Definition value_type.h:58
#define T_MOVED
Old name of RUBY_T_MOVED.
Definition value_type.h:71
#define T_REGEXP
Old name of RUBY_T_REGEXP.
Definition value_type.h:77
VALUE rb_eRuntimeError
RuntimeError exception.
Definition error.c:1425
VALUE type(ANYARGS)
ANYARGS-ed function type.
Definition st.h:79
uintptr_t VALUE
Type that represents a Ruby object.
Definition value.h:40
#define RBIMPL_WARNING_IGNORED(flag)
Suppresses a warning.
#define RBIMPL_WARNING_PUSH()
Pushes compiler warning state.
#define RBIMPL_WARNING_POP()
Pops compiler warning state.