interprocess 2.4.0

Interprocess communication toolkit
Documentation
#[cfg(feature = "tokio")]
use crate::os::windows::{tokio_flusher::TokioFlusher, OptArcIRC};
use {
    super::super::*,
    crate::os::windows::{MaybeArc, OptArc},
    std::sync::Arc,
};

/// Tags for [`PipeStream`]'s generic arguments that specify the directionality of the stream and
/// how it receives and/or sends data (as bytes or as messages).
///
/// This is a sort of const-generic incarnation of the [`PipeMode`] enumeration that allows
/// `PipeStream` to be consolidated into a single struct with generic parameters that decide which
/// traits are implemented on it.
///
/// Some examples of how different `PipeStream` signatures would look:
/// - **`PipeStream<Bytes, Bytes>`** (or, thanks to default generic arguments, simply `PipeStream`)
///   is a duplex stream that receives and sends bytes.
/// - **`PipeStream<Messages, ()>`** is a receive-only message stream.
/// - **`PipeStream<Bytes, Messages>`** is a duplex stream that receives bytes but sends messages.
pub mod pipe_mode {
    use super::*;

    mod seal {
        use super::*;
        pub trait PipeModeTag: Copy + std::fmt::Debug + Eq + Send + Sync + Unpin {
            const MODE: Option<PipeMode>;
            /// Used on the send mode to determine if the raw instance should always be in an
            /// `Arc` to enable `&self` refcloning or not.
            type OptArc<T>: OptArc<Value = T>
            where
                T: Send + Sync;
            #[cfg(feature = "tokio")]
            type TokioFlusher: std::fmt::Debug + Default;
        }
        // Hack for working around the lack of associated type bounds on 1.75.
        #[cfg(feature = "tokio")]
        pub(crate) trait PmtOptArcIRC: PipeModeTag {
            fn cast_oai(
                oa: &Self::OptArc<tokio::RawPipeStream>,
            ) -> &(impl OptArcIRC<Value = tokio::RawPipeStream> + 'static);
        }
        #[cfg(feature = "tokio")]
        impl<T: PipeModeTag> PmtOptArcIRC for T
        where
            T::OptArc<tokio::RawPipeStream>: OptArcIRC + 'static,
        {
            #[cfg(feature = "tokio")]
            fn cast_oai(
                oa: &Self::OptArc<tokio::RawPipeStream>,
            ) -> &(impl OptArcIRC<Value = tokio::RawPipeStream> + 'static) {
                oa
            }
        }
        #[cfg(feature = "tokio")]
        #[allow(private_bounds)]
        pub trait NotNone: PipeModeTag<TokioFlusher = TokioFlusher> + PmtOptArcIRC {}
        #[cfg(not(feature = "tokio"))]
        pub trait NotNone: PipeModeTag {}
    }
    pub(crate) use seal::*;

    macro_rules! present_tag {
        ($(#[$attr:meta])* $tag:ident is $mode:expr ; no_tokio_flusher) => {
                tag_enum!($( #[$attr] )* $tag);
                impl PipeModeTag for $tag {
                    const MODE: Option<PipeMode> = $mode;
                    type OptArc<T> = MaybeArc<T> where T: Send + Sync;
                    #[cfg(feature = "tokio")]
                    type TokioFlusher = ();
                }
        };
        ($(#[$attr:meta])* $tag:ident is $mode:expr) => {
                tag_enum!($( #[$attr] )* $tag);
                impl PipeModeTag for $tag {
                    const MODE: Option<PipeMode> = $mode;
                    type OptArc<T> = Arc<T> where T: Send + Sync;
                    #[cfg(feature = "tokio")]
                    type TokioFlusher = TokioFlusher;
                }
        };
        ($($(#[$attr:meta])* $tag:ident is $mode:expr $(; $yayornay:ident)?),+ $(,)?) => {
            $(present_tag!($( #[$attr] )* $tag is $mode $(; $yayornay)?);)+
        };
    }

    present_tag! {
        /// Tags a direction of a [`PipeStream`] to be absent.
        None is None ; no_tokio_flusher,
        /// Tags a direction of a [`PipeStream`] to be present with byte-wise semantics.
        Bytes is Some(PipeMode::Bytes),
        /// Tags a direction of a [`PipeStream`] to be present with message-wise semantics.
        Messages is Some(PipeMode::Messages),
    }
    impl NotNone for Bytes {}
    impl NotNone for Messages {}
}
pub(crate) use pipe_mode::{NotNone as PmtNotNone, PipeModeTag};