1#include "internal/gc.h"
2#include "internal/thread.h"
5#include "ractor_core.h"
15 return vm->ractor.sync.lock_owner == GET_RACTOR();
20RUBY_ASSERT_vm_locking(
void)
22 if (rb_multi_ractor_p()) {
24 VM_ASSERT(vm_locked(vm));
29RUBY_ASSERT_vm_unlocking(
void)
31 if (rb_multi_ractor_p()) {
33 VM_ASSERT(!vm_locked(vm));
41 return vm_locked(GET_VM());
45vm_need_barrier_waiting(
const rb_vm_t *vm)
47#ifdef RUBY_THREAD_PTHREAD_H
48 return vm->ractor.sched.barrier_waiting;
50 return vm->ractor.sync.barrier_waiting;
57#ifdef RUBY_THREAD_PTHREAD_H
58 return !no_barrier && cr->threads.sched.running != NULL && vm_need_barrier_waiting(vm);
60 return !no_barrier && vm_need_barrier_waiting(vm);
65vm_lock_enter(
rb_ractor_t *cr,
rb_vm_t *vm,
bool locked,
bool no_barrier,
unsigned int *lev APPEND_LOCATION_ARGS)
67 RUBY_DEBUG_LOG2(file, line,
"start locked:%d", locked);
75 VM_ASSERT(cr->sync.locked_by != rb_ractor_self(cr));
79 VM_ASSERT(vm->ractor.sync.lock_owner == NULL);
80 VM_ASSERT(vm->ractor.sync.lock_rec == 0);
83 if (vm_need_barrier(no_barrier, cr, vm)) {
85 RB_VM_SAVE_MACHINE_CONTEXT(rb_ec_thread_ptr(ec));
88 VM_ASSERT(vm_need_barrier_waiting(vm));
89 RUBY_DEBUG_LOG(
"barrier serial:%u", vm->ractor.sched.barrier_serial);
90 rb_ractor_sched_barrier_join(vm, cr);
91 }
while (vm_need_barrier_waiting(vm));
94 VM_ASSERT(vm->ractor.sync.lock_rec == 0);
95 VM_ASSERT(vm->ractor.sync.lock_owner == NULL);
96 vm->ractor.sync.lock_owner = cr;
99 vm->ractor.sync.lock_rec++;
100 *lev = vm->ractor.sync.lock_rec;
102 RUBY_DEBUG_LOG2(file, line,
"rec:%u owner:%u", vm->ractor.sync.lock_rec,
103 (
unsigned int)rb_ractor_id(vm->ractor.sync.lock_owner));
107vm_lock_leave(
rb_vm_t *vm,
bool no_barrier,
unsigned int *lev APPEND_LOCATION_ARGS)
109 MAYBE_UNUSED(
rb_ractor_t *cr = vm->ractor.sync.lock_owner);
111 RUBY_DEBUG_LOG2(file, line,
"rec:%u owner:%u%s", vm->ractor.sync.lock_rec,
112 (
unsigned int)rb_ractor_id(cr),
113 vm->ractor.sync.lock_rec == 1 ?
" (leave)" :
"");
116 VM_ASSERT(vm->ractor.sync.lock_rec > 0);
117 VM_ASSERT(vm->ractor.sync.lock_rec == *lev);
118 VM_ASSERT(cr == GET_RACTOR());
120#ifdef RUBY_THREAD_PTHREAD_H
121 if (vm->ractor.sched.barrier_ractor == cr &&
122 vm->ractor.sched.barrier_lock_rec == vm->ractor.sync.lock_rec) {
123 VM_ASSERT(!no_barrier);
124 rb_ractor_sched_barrier_end(vm, cr);
128 vm->ractor.sync.lock_rec--;
129 *lev = vm->ractor.sync.lock_rec;
131 if (vm->ractor.sync.lock_rec == 0) {
132 vm->ractor.sync.lock_owner = NULL;
138rb_vm_lock_enter_body(
unsigned int *lev APPEND_LOCATION_ARGS)
142 vm_lock_enter(NULL, vm,
true,
false, lev APPEND_LOCATION_PARAMS);
145 vm_lock_enter(GET_RACTOR(), vm,
false,
false, lev APPEND_LOCATION_PARAMS);
150rb_vm_lock_enter_body_nb(
unsigned int *lev APPEND_LOCATION_ARGS)
154 vm_lock_enter(NULL, vm,
true,
true, lev APPEND_LOCATION_PARAMS);
157 vm_lock_enter(GET_RACTOR(), vm,
false,
true, lev APPEND_LOCATION_PARAMS);
162rb_vm_lock_enter_body_cr(
rb_ractor_t *cr,
unsigned int *lev APPEND_LOCATION_ARGS)
165 vm_lock_enter(cr, vm, vm_locked(vm),
false, lev APPEND_LOCATION_PARAMS);
169rb_vm_lock_leave_body_nb(
unsigned int *lev APPEND_LOCATION_ARGS)
171 vm_lock_leave(GET_VM(),
true, lev APPEND_LOCATION_PARAMS);
175rb_vm_lock_leave_body(
unsigned int *lev APPEND_LOCATION_ARGS)
177 vm_lock_leave(GET_VM(),
false, lev APPEND_LOCATION_PARAMS);
181rb_vm_lock_body(LOCATION_ARGS)
184 ASSERT_vm_unlocking();
186 vm_lock_enter(GET_RACTOR(), vm,
false,
false, &vm->ractor.sync.lock_rec APPEND_LOCATION_PARAMS);
190rb_vm_unlock_body(LOCATION_ARGS)
194 VM_ASSERT(vm->ractor.sync.lock_rec == 1);
195 vm_lock_leave(vm,
false, &vm->ractor.sync.lock_rec APPEND_LOCATION_PARAMS);
199vm_cond_wait(
rb_vm_t *vm, rb_nativethread_cond_t *cond,
unsigned long msec)
202 unsigned int lock_rec = vm->ractor.sync.lock_rec;
205 vm->ractor.sync.lock_rec = 0;
206 vm->ractor.sync.lock_owner = NULL;
213 vm->ractor.sync.lock_rec = lock_rec;
214 vm->ractor.sync.lock_owner = cr;
218rb_vm_cond_wait(
rb_vm_t *vm, rb_nativethread_cond_t *cond)
220 vm_cond_wait(vm, cond, 0);
224rb_vm_cond_timedwait(
rb_vm_t *vm, rb_nativethread_cond_t *cond,
unsigned long msec)
226 vm_cond_wait(vm, cond, msec);
232#ifdef RUBY_THREAD_PTHREAD_H
233 return vm->ractor.sched.barrier_ractor == cr;
242 RB_DEBUG_COUNTER_INC(vm_sync_barrier);
244 if (!rb_multi_ractor_p()) {
253 VM_ASSERT(cr == GET_RACTOR());
254 VM_ASSERT(rb_ractor_status_p(cr, ractor_running));
256 if (vm_barrier_acquired_p(vm, cr)) {
261 VM_ASSERT(!vm->ractor.sched.barrier_waiting);
262 rb_ractor_sched_barrier_start(vm, cr);
269 unsigned int recorded_lock_rec,
270 unsigned int current_lock_rec)
272 VM_ASSERT(recorded_lock_rec != current_lock_rec);
274 if (UNLIKELY(recorded_lock_rec > current_lock_rec)) {
275 rb_bug(
"unexpected situation - recordd:%u current:%u",
276 recorded_lock_rec, current_lock_rec);
279 while (recorded_lock_rec < current_lock_rec) {
280 RB_VM_LOCK_LEAVE_LEV(¤t_lock_rec);
284 VM_ASSERT(recorded_lock_rec == rb_ec_vm_lock_rec(ec));
void rb_native_mutex_lock(rb_nativethread_lock_t *lock)
Just another name of rb_nativethread_lock_lock.
void rb_native_mutex_unlock(rb_nativethread_lock_t *lock)
Just another name of rb_nativethread_lock_unlock.
void rb_native_cond_wait(rb_nativethread_cond_t *cond, rb_nativethread_lock_t *mutex)
Waits for the passed condition variable to be signalled.
void rb_native_cond_timedwait(rb_nativethread_cond_t *cond, rb_nativethread_lock_t *mutex, unsigned long msec)
Identical to rb_native_cond_wait(), except it additionally takes timeout in msec resolution.