// SPDX-License-Identifier: GPL-2.0 // Copyright (C) 2024 Google LLC. use kernel::{ list::{ListArc, ListArcSafe}, prelude::*, seq_file::SeqFile, seq_print, sync::UniqueArc, uaccess::UserSliceWriter, }; use crate::{node::Node, thread::Thread, DArc, DLArc, DTRWrap, DeliverToRead}; use core::mem::MaybeUninit; pub(crate) struct CritIncrWrapper { inner: UniqueArc>>, } impl CritIncrWrapper { pub(crate) fn new() -> Result { Ok(CritIncrWrapper { inner: UniqueArc::try_new_uninit()?, }) } pub(super) fn init(self, node: DArc) -> DLArc { match self.inner.pin_init_with(DTRWrap::new(NodeWrapper { node })) { Ok(initialized) => ListArc::from_pin_unique(initialized) as _, Err(err) => match err {}, } } } struct NodeWrapper { node: DArc, } kernel::list::impl_list_arc_safe! { impl ListArcSafe<0> for NodeWrapper { untracked; } } impl DeliverToRead for NodeWrapper { fn do_work(self: DArc, _thread: &Thread, writer: &mut UserSliceWriter) -> Result { let node = &self.node; let mut owner_inner = node.owner.inner.lock(); let inner = node.inner.access_mut(&mut owner_inner); let ds = &mut inner.delivery_state; assert!(ds.has_pushed_wrapper); assert!(ds.has_strong_zero2one); ds.has_pushed_wrapper = false; ds.has_strong_zero2one = false; node.do_work_locked(writer, owner_inner) } fn cancel(self: DArc) {} fn on_thread_selected(&self, _thread: &Thread) {} fn should_sync_wakeup(&self) -> bool { false } #[inline(never)] fn debug_print(&self, m: &mut SeqFile, prefix: &str, _tprefix: &str) -> Result<()> { seq_print!( m, "{}node work {}: u{:016x} c{:016x}\n", prefix, self.node.debug_id, self.node.ptr, self.node.cookie, ); Ok(()) } }