rb_sys/
utils.rs

1//! Internal utility functions.
2
3/// Check if the Ruby VM is globally available.
4///
5/// Unfortunately there is no public API for this check, but there's a hidden
6/// `ruby_current_vm_ptr` symbol in libruby 2.5 - 3.2 which we can use to
7/// determine if the VM has been initialized, or shut down.
8///
9/// # Notes
10///
11/// Ruby 2.4 and below don't have a global VM pointer, so we can't check if it's
12/// null. Ruby 2.4 is EOL, and support will be dropped soon anyway.
13//
14/// In Ruby 3.3, the global VM pointer is no longer exported, so there's no
15/// simple way to check the global VM pointer, so instead we check if known
16/// static value is non-zero.
17///
18/// On Ruby < 3.3, we also need to check if the global VM pointer is null to
19/// ensure the VM hasn't stopped, which makes the function name a bit of a
20/// misnomer... but in actuality this function can only guarantee that the
21/// VM is started, not that it's still running.
22#[allow(dead_code)]
23pub(crate) unsafe fn is_ruby_vm_started() -> bool {
24    #[cfg(ruby_engine = "mri")]
25    let ret = {
26        #[cfg(all(ruby_gt_2_4, ruby_lte_3_2))]
27        let ret = !crate::hidden::ruby_current_vm_ptr.is_null();
28
29        #[cfg(any(ruby_lte_2_4, ruby_gt_3_2))]
30        let ret = crate::rb_cBasicObject != 0;
31
32        ret
33    };
34
35    #[cfg(ruby_engine = "truffleruby")]
36    let ret = crate::rb_cBasicObject != 0;
37
38    ret
39}
40
41/// Macro for conditionally asserting type checks in Ruby, only active when RUBY_DEBUG is enabled.
42/// This matches Ruby's behavior of only checking types in debug mode.
43#[macro_export]
44macro_rules! debug_ruby_assert_type {
45    ($obj:expr, $type:expr, $message:expr) => {
46        #[cfg(ruby_ruby_debug = "true")]
47        {
48            #[allow(clippy::macro_metavars_in_unsafe)]
49            unsafe {
50                assert!(
51                    !$crate::SPECIAL_CONST_P($obj) && $crate::RB_BUILTIN_TYPE($obj) == $type,
52                    $message
53                );
54            }
55        }
56        #[cfg(not(ruby_ruby_debug = "true"))]
57        {
58            let _ = ($obj, $type, $message); // Prevent unused variable warnings
59        }
60    };
61}
62
63#[cfg(test)]
64mod tests {
65    use super::*;
66    use rusty_fork::rusty_fork_test;
67    use std::ptr::addr_of_mut;
68
69    rusty_fork_test! {
70        #[test]
71        fn test_is_ruby_vm_started() {
72            assert!(!unsafe { is_ruby_vm_started() });
73
74            // Call ruby_sysinit which handles platform-specific initialization
75            // (rb_w32_sysinit on Windows) and sets up standard file descriptors
76            let mut argc = 0;
77            let mut argv: [*mut std::os::raw::c_char; 0] = [];
78            let mut argv_ptr = argv.as_mut_ptr();
79            unsafe { crate::ruby_sysinit(&mut argc, &mut argv_ptr) };
80
81            // ruby_init_stack must be called before ruby_setup, especially on
82            // Windows where it's required for proper GC stack scanning
83            let mut stack_marker: crate::VALUE = 0;
84            unsafe { crate::ruby_init_stack(addr_of_mut!(stack_marker) as *mut _) };
85
86            match unsafe { crate::ruby_setup() } {
87                0 => {}
88                code => panic!("Failed to setup Ruby (error code: {})", code),
89            };
90
91            assert!(unsafe { is_ruby_vm_started() });
92        }
93    }
94}