-
Notifications
You must be signed in to change notification settings - Fork 59
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[skip ci] Add sample channel (require z_owned_sample_t)
- Loading branch information
Showing
5 changed files
with
267 additions
and
32 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,132 @@ | ||
use crate::{ | ||
z_closure_owned_sample_drop, z_closure_sample_drop, z_owned_closure_owned_sample_t, | ||
z_owned_closure_sample_t, z_owned_sample_t, z_sample_clone, z_sample_t, | ||
}; | ||
|
||
#[repr(C)] | ||
pub struct z_owned_sample_fifo_channel_t { | ||
pub send: z_owned_closure_sample_t, | ||
pub recv: z_owned_closure_owned_sample_t, | ||
pub try_recv: z_owned_closure_owned_sample_t, | ||
} | ||
#[no_mangle] | ||
pub extern "C" fn z_sample_fifo_channel_drop(channel: &mut z_owned_sample_fifo_channel_t) { | ||
z_closure_sample_drop(&mut channel.send); | ||
z_closure_owned_sample_drop(&mut channel.recv); | ||
z_closure_owned_sample_drop(&mut channel.try_recv); | ||
} | ||
/// Constructs a null safe-to-drop value of 'z_owned_sample_fifo_channel_t' type | ||
#[no_mangle] | ||
pub extern "C" fn z_sample_fifo_channel_null() -> z_owned_sample_fifo_channel_t { | ||
z_owned_sample_fifo_channel_t { | ||
send: z_owned_closure_sample_t::empty(), | ||
recv: z_owned_closure_owned_sample_t::empty(), | ||
try_recv: z_owned_closure_owned_sample_t::empty(), | ||
} | ||
} | ||
|
||
/// Creates a new blocking fifo channel, returned as a pair of closures. | ||
/// | ||
/// If `bound` is different from 0, that channel will be bound and apply back-pressure when full. | ||
/// | ||
/// The `send` end should be passed as callback to a `z_get` call. | ||
/// | ||
/// The `recv` end is a synchronous closure that will block until either a `z_owned_sample_t` is available, | ||
/// which it will then return; or until the `send` closure is dropped and all replies have been consumed, | ||
/// at which point it will return an invalidated `z_owned_sample_t`, and so will further calls. | ||
#[no_mangle] | ||
pub extern "C" fn z_sample_fifo_channel_new(bound: usize) -> z_owned_sample_fifo_channel_t { | ||
// TODO(sashacmc): switch to handlers::FifoChannel | ||
let (tx, rx) = if bound == 0 { | ||
crossbeam_channel::unbounded() | ||
} else { | ||
crossbeam_channel::bounded(bound) | ||
}; | ||
let rx_clone = rx.clone(); | ||
z_owned_sample_fifo_channel_t { | ||
send: From::from(move |sample: &z_sample_t| { | ||
let mut osample = z_sample_clone(Some(sample)); | ||
if let Some(osample) = osample.take() { | ||
if let Err(e) = tx.send(osample) { | ||
log::error!("Attempted to push onto a closed sample_ring: {}", e) | ||
} | ||
} | ||
}), | ||
recv: From::from(move |receptacle: &mut z_owned_sample_t| { | ||
*receptacle = match rx.recv() { | ||
Ok(val) => val.into(), | ||
Err(_) => None.into(), | ||
}; | ||
}), | ||
try_recv: From::from(move |receptacle: &mut z_owned_sample_t| { | ||
*receptacle = match rx_clone.try_recv() { | ||
Ok(val) => val.into(), | ||
Err(_) => None.into(), | ||
}; | ||
}), | ||
} | ||
} | ||
|
||
#[repr(C)] | ||
pub struct z_owned_sample_ring_channel_t { | ||
pub send: z_owned_closure_sample_t, | ||
pub recv: z_owned_closure_owned_sample_t, | ||
pub try_recv: z_owned_closure_owned_sample_t, | ||
} | ||
#[no_mangle] | ||
pub extern "C" fn z_sample_ring_channel_drop(channel: &mut z_owned_sample_ring_channel_t) { | ||
z_closure_sample_drop(&mut channel.send); | ||
z_closure_owned_sample_drop(&mut channel.recv); | ||
z_closure_owned_sample_drop(&mut channel.try_recv); | ||
} | ||
/// Constructs a null safe-to-drop value of 'z_owned_sample_ring_channel_t' type | ||
#[no_mangle] | ||
pub extern "C" fn z_sample_ring_channel_null() -> z_owned_sample_ring_channel_t { | ||
z_owned_sample_ring_channel_t { | ||
send: z_owned_closure_sample_t::empty(), | ||
recv: z_owned_closure_owned_sample_t::empty(), | ||
try_recv: z_owned_closure_owned_sample_t::empty(), | ||
} | ||
} | ||
|
||
/// Creates a new blocking ring channel, returned as a pair of closures. | ||
/// | ||
/// If `bound` is different from 0, that channel will be bound and apply back-pressure when full. | ||
/// | ||
/// The `send` end should be passed as callback to a `z_get` call. | ||
/// | ||
/// The `recv` end is a synchronous closure that will block until either a `z_owned_sample_t` is available, | ||
/// which it will then return; or until the `send` closure is dropped and all replies have been consumed, | ||
/// at which point it will return an invalidated `z_owned_sample_t`, and so will further calls. | ||
#[no_mangle] | ||
pub extern "C" fn z_sample_ring_channel_new(bound: usize) -> z_owned_sample_ring_channel_t { | ||
// TODO(sashacmc): switch to handlers::RingChannel | ||
let (tx, rx) = if bound == 0 { | ||
crossbeam_channel::unbounded() | ||
} else { | ||
crossbeam_channel::bounded(bound) | ||
}; | ||
let rx_clone = rx.clone(); | ||
z_owned_sample_ring_channel_t { | ||
send: From::from(move |sample: &z_sample_t| { | ||
let mut osample = z_sample_clone(Some(sample)); | ||
if let Some(osample) = osample.take() { | ||
if let Err(e) = tx.send(osample) { | ||
log::error!("Attempted to push onto a closed sample_ring: {}", e) | ||
} | ||
} | ||
}), | ||
recv: From::from(move |receptacle: &mut z_owned_sample_t| { | ||
*receptacle = match rx.recv() { | ||
Ok(val) => val.into(), | ||
Err(_) => None.into(), | ||
}; | ||
}), | ||
try_recv: From::from(move |receptacle: &mut z_owned_sample_t| { | ||
*receptacle = match rx_clone.try_recv() { | ||
Ok(val) => val.into(), | ||
Err(_) => None.into(), | ||
}; | ||
}), | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters