Ruby 3.5.0dev (2025-06-16 revision c59f66b61ab5134110a8283a0ac44580e95d9120)
vm_sync.h (c59f66b61ab5134110a8283a0ac44580e95d9120)
1#ifndef RUBY_VM_SYNC_H
2#define RUBY_VM_SYNC_H
3
4#include "vm_debug.h"
5#include "debug_counter.h"
6
7#if USE_RUBY_DEBUG_LOG
8#define LOCATION_ARGS const char *file, int line
9#define LOCATION_PARAMS file, line
10#define APPEND_LOCATION_ARGS , const char *file, int line
11#define APPEND_LOCATION_PARAMS , file, line
12#else
13#define LOCATION_ARGS void
14#define LOCATION_PARAMS
15#define APPEND_LOCATION_ARGS
16#define APPEND_LOCATION_PARAMS
17#endif
18
19bool rb_vm_locked_p(void);
20void rb_vm_lock_body(LOCATION_ARGS);
21void rb_vm_unlock_body(LOCATION_ARGS);
22
23struct rb_ractor_struct;
24NOINLINE(void rb_vm_lock_enter_body_cr(struct rb_ractor_struct *cr, unsigned int *lev APPEND_LOCATION_ARGS));
25NOINLINE(void rb_vm_lock_enter_body_nb(unsigned int *lev APPEND_LOCATION_ARGS));
26NOINLINE(void rb_vm_lock_enter_body(unsigned int *lev APPEND_LOCATION_ARGS));
27void rb_vm_lock_leave_body_nb(unsigned int *lev APPEND_LOCATION_ARGS);
28void rb_vm_lock_leave_body(unsigned int *lev APPEND_LOCATION_ARGS);
29void rb_vm_barrier(void);
30
31#if RUBY_DEBUG
32// GET_VM()
33#include "vm_core.h"
34#endif
35
36RUBY_EXTERN struct rb_ractor_struct *ruby_single_main_ractor; // ractor.c
37
38static inline bool
39rb_multi_ractor_p(void)
40{
41 if (LIKELY(ruby_single_main_ractor)) {
42 // 0 on boot time.
43 RUBY_ASSERT(GET_VM()->ractor.cnt <= 1);
44 return false;
45 }
46 else {
47 // multi-ractor mode can run ractor.cnt == 1
48 return true;
49 }
50}
51
52static inline void
53rb_vm_lock(const char *file, int line)
54{
55 RB_DEBUG_COUNTER_INC(vm_sync_lock);
56
57 if (rb_multi_ractor_p()) {
58 rb_vm_lock_body(LOCATION_PARAMS);
59 }
60}
61
62static inline void
63rb_vm_unlock(const char *file, int line)
64{
65 if (rb_multi_ractor_p()) {
66 rb_vm_unlock_body(LOCATION_PARAMS);
67 }
68}
69
70static inline void
71rb_vm_lock_enter(unsigned int *lev, const char *file, int line)
72{
73 RB_DEBUG_COUNTER_INC(vm_sync_lock_enter);
74
75 if (rb_multi_ractor_p()) {
76 rb_vm_lock_enter_body(lev APPEND_LOCATION_PARAMS);
77 }
78}
79
80static inline void
81rb_vm_lock_enter_nb(unsigned int *lev, const char *file, int line)
82{
83 RB_DEBUG_COUNTER_INC(vm_sync_lock_enter_nb);
84
85 if (rb_multi_ractor_p()) {
86 rb_vm_lock_enter_body_nb(lev APPEND_LOCATION_PARAMS);
87 }
88}
89
90static inline void
91rb_vm_lock_leave_nb(unsigned int *lev, const char *file, int line)
92{
93 if (rb_multi_ractor_p()) {
94 rb_vm_lock_leave_body_nb(lev APPEND_LOCATION_PARAMS);
95 }
96}
97
98static inline void
99rb_vm_lock_leave(unsigned int *lev, const char *file, int line)
100{
101 if (rb_multi_ractor_p()) {
102 rb_vm_lock_leave_body(lev APPEND_LOCATION_PARAMS);
103 }
104}
105
106static inline void
107rb_vm_lock_enter_cr(struct rb_ractor_struct *cr, unsigned int *levp, const char *file, int line)
108{
109 RB_DEBUG_COUNTER_INC(vm_sync_lock_enter_cr);
110 rb_vm_lock_enter_body_cr(cr, levp APPEND_LOCATION_PARAMS);
111}
112
113static inline void
114rb_vm_lock_leave_cr(struct rb_ractor_struct *cr, unsigned int *levp, const char *file, int line)
115{
116 rb_vm_lock_leave_body(levp APPEND_LOCATION_PARAMS);
117}
118
119#define RB_VM_LOCKED_P() rb_vm_locked_p()
120
121#define RB_VM_LOCK() rb_vm_lock(__FILE__, __LINE__)
122#define RB_VM_UNLOCK() rb_vm_unlock(__FILE__, __LINE__)
123
124#define RB_VM_LOCK_ENTER_CR_LEV(cr, levp) rb_vm_lock_enter_cr(cr, levp, __FILE__, __LINE__)
125#define RB_VM_LOCK_LEAVE_CR_LEV(cr, levp) rb_vm_lock_leave_cr(cr, levp, __FILE__, __LINE__)
126#define RB_VM_LOCK_ENTER_LEV(levp) rb_vm_lock_enter(levp, __FILE__, __LINE__)
127#define RB_VM_LOCK_LEAVE_LEV(levp) rb_vm_lock_leave(levp, __FILE__, __LINE__)
128
129#define RB_VM_LOCK_ENTER() { unsigned int _lev; RB_VM_LOCK_ENTER_LEV(&_lev);
130#define RB_VM_LOCK_LEAVE() RB_VM_LOCK_LEAVE_LEV(&_lev); }
131#define RB_VM_LOCKING() \
132 for (unsigned int vm_locking_level, vm_locking_do = (RB_VM_LOCK_ENTER_LEV(&vm_locking_level), 1); \
133 vm_locking_do; RB_VM_LOCK_LEAVE_LEV(&vm_locking_level), vm_locking_do = 0)
134
135#define RB_VM_LOCK_ENTER_LEV_NB(levp) rb_vm_lock_enter_nb(levp, __FILE__, __LINE__)
136#define RB_VM_LOCK_LEAVE_LEV_NB(levp) rb_vm_lock_leave_nb(levp, __FILE__, __LINE__)
137#define RB_VM_LOCK_ENTER_NO_BARRIER() { unsigned int _lev; RB_VM_LOCK_ENTER_LEV_NB(&_lev);
138#define RB_VM_LOCK_LEAVE_NO_BARRIER() RB_VM_LOCK_LEAVE_LEV_NB(&_lev); }
139#define RB_VM_LOCKING_NO_BARRIER() \
140 for (unsigned int vm_locking_level, vm_locking_do = (RB_VM_LOCK_ENTER_LEV_NB(&vm_locking_level), 1); \
141 vm_locking_do; RB_VM_LOCK_LEAVE_LEV_NB(&vm_locking_level), vm_locking_do = 0)
142
143#if RUBY_DEBUG > 0
144void RUBY_ASSERT_vm_locking(void);
145void RUBY_ASSERT_vm_unlocking(void);
146#define ASSERT_vm_locking() RUBY_ASSERT_vm_locking()
147#define ASSERT_vm_unlocking() RUBY_ASSERT_vm_unlocking()
148#else
149#define ASSERT_vm_locking()
150#define ASSERT_vm_unlocking()
151#endif
152
153#endif // RUBY_VM_SYNC_H
#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