shadow_rs/host/
status_listener.rs
1use crate::core::work::task::TaskRef;
4use crate::core::worker::Worker;
5use crate::cshadow;
6use crate::host::descriptor::FileState;
7use crate::utility::HostTreePointer;
8
9use super::host::Host;
10
11#[derive(Debug)]
16pub struct StatusListener {
17 ptr: HostTreePointer<cshadow::StatusListener>,
18}
19
20impl Ord for StatusListener {
21 fn cmp(&self, other: &Self) -> std::cmp::Ordering {
22 Worker::with_active_host(|host| {
23 let lhs = unsafe { self.ptr.ptr_with_host(host) };
25 let rhs = unsafe { other.ptr.ptr_with_host(host) };
26
27 match unsafe { cshadow::status_listener_compare(lhs.cast(), rhs.cast()) } {
28 -1 => std::cmp::Ordering::Less,
29 0 => std::cmp::Ordering::Equal,
30 1 => std::cmp::Ordering::Greater,
31 other => panic!("Unexpected comparison result: {other}"),
32 }
33 })
34 .unwrap()
35 }
36}
37
38impl PartialOrd for StatusListener {
39 fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
40 Some(self.cmp(other))
41 }
42}
43
44impl Eq for StatusListener {}
45
46impl PartialEq for StatusListener {
47 fn eq(&self, other: &Self) -> bool {
48 matches!(self.cmp(other), std::cmp::Ordering::Equal)
49 }
50}
51
52impl StatusListener {
53 pub unsafe fn clone_from_c(host: &Host, ptr: *mut cshadow::StatusListener) -> Self {
60 unsafe { cshadow::statuslistener_ref(ptr) };
61 let ptr = HostTreePointer::new_for_host(host.id(), ptr);
62 Self { ptr }
63 }
64
65 pub fn new(host: &Host, notify: impl Fn(&Host) + Send + Sync + 'static) -> Self {
67 extern "C-unwind" fn notify_fn(
68 callback_object: *mut std::ffi::c_void,
69 _callback_arg: *mut std::ffi::c_void,
70 ) {
71 let task: *const TaskRef = callback_object.cast();
72 let task = unsafe { &*task };
73 Worker::with_active_host(|host| task.execute(host)).unwrap()
74 }
75 extern "C-unwind" fn object_free_fn(callback_object: *mut std::ffi::c_void) {
76 let task: *mut TaskRef = callback_object.cast();
77 drop(unsafe { Box::from_raw(task) })
78 }
79
80 let callback_obj = Box::into_raw(Box::new(TaskRef::new(notify)));
81 let status_listener = unsafe {
82 cshadow::statuslistener_new(
83 Some(notify_fn),
84 callback_obj.cast::<std::ffi::c_void>(),
85 Some(object_free_fn),
86 std::ptr::null_mut(),
87 None,
88 host,
89 )
90 };
91 Self {
92 ptr: HostTreePointer::new_for_host(host.id(), status_listener),
93 }
94 }
95
96 pub fn handle_status_change(&self, host: &Host, current: FileState, transitions: FileState) {
103 unsafe {
104 cshadow::statuslistener_onStatusChanged(
105 self.ptr.ptr_with_host(host),
106 current,
107 transitions,
108 )
109 }
110 }
111
112 pub fn set_monitor_status(
116 &self,
117 host: &Host,
118 status: FileState,
119 filter: cshadow::StatusListenerFilter,
120 ) {
121 unsafe {
122 cshadow::statuslistener_setMonitorStatus(self.ptr.ptr_with_host(host), status, filter)
123 }
124 }
125}
126
127impl Drop for StatusListener {
128 fn drop(&mut self) {
129 unsafe { cshadow::statuslistener_unref(self.ptr.ptr()) };
130 }
131}