2012-08-07 16:38:32 -05:00
|
|
|
// Examples from Eric's internship final presentation.
|
|
|
|
//
|
|
|
|
// Code is easier to write in emacs, and it's good to be sure all the
|
|
|
|
// code samples compile (or not) as they should.
|
|
|
|
|
2012-08-08 14:00:00 -05:00
|
|
|
// xfail-pretty
|
|
|
|
|
2012-08-07 16:38:32 -05:00
|
|
|
import double_buffer::client::*;
|
|
|
|
import double_buffer::give_buffer;
|
|
|
|
|
2012-08-22 19:47:11 -05:00
|
|
|
macro_rules! select_if (
|
2012-08-07 16:38:32 -05:00
|
|
|
{
|
|
|
|
$index:expr,
|
|
|
|
$count:expr,
|
|
|
|
$port:path => [
|
|
|
|
$($message:path$(($($x: ident),+))dont_type_this*
|
|
|
|
-> $next:ident $e:expr),+
|
|
|
|
],
|
|
|
|
$( $ports:path => [
|
|
|
|
$($messages:path$(($($xs: ident),+))dont_type_this*
|
|
|
|
-> $nexts:ident $es:expr),+
|
|
|
|
], )*
|
|
|
|
} => {
|
|
|
|
if $index == $count {
|
|
|
|
match move pipes::try_recv($port) {
|
|
|
|
$(some($message($($(copy $x,)+)* next)) => {
|
|
|
|
// FIXME (#2329) we really want move out of enum here.
|
|
|
|
let $next = unsafe { let x <- *ptr::addr_of(next); x };
|
|
|
|
$e
|
|
|
|
})+
|
|
|
|
_ => fail
|
|
|
|
}
|
|
|
|
} else {
|
2012-08-22 19:24:52 -05:00
|
|
|
select_if!(
|
2012-08-07 16:38:32 -05:00
|
|
|
$index,
|
|
|
|
$count + 1,
|
|
|
|
$( $ports => [
|
|
|
|
$($messages$(($($xs),+))dont_type_this*
|
|
|
|
-> $nexts $es),+
|
|
|
|
], )*
|
2012-08-22 19:24:52 -05:00
|
|
|
)
|
2012-08-07 16:38:32 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
{
|
|
|
|
$index:expr,
|
|
|
|
$count:expr,
|
|
|
|
} => {
|
|
|
|
fail
|
|
|
|
}
|
2012-08-22 19:47:11 -05:00
|
|
|
)
|
2012-08-07 16:38:32 -05:00
|
|
|
|
2012-08-22 19:47:11 -05:00
|
|
|
macro_rules! select (
|
2012-08-07 16:38:32 -05:00
|
|
|
{
|
|
|
|
$( $port:path => {
|
|
|
|
$($message:path$(($($x: ident),+))dont_type_this*
|
|
|
|
-> $next:ident $e:expr),+
|
|
|
|
} )+
|
|
|
|
} => {
|
|
|
|
let index = pipes::selecti([$(($port).header()),+]/_);
|
2012-08-22 19:24:52 -05:00
|
|
|
select_if!(index, 0, $( $port => [
|
2012-08-07 16:38:32 -05:00
|
|
|
$($message$(($($x),+))dont_type_this* -> $next $e),+
|
2012-08-22 19:24:52 -05:00
|
|
|
], )+)
|
2012-08-07 16:38:32 -05:00
|
|
|
}
|
2012-08-22 19:47:11 -05:00
|
|
|
)
|
2012-08-07 16:38:32 -05:00
|
|
|
|
|
|
|
// Types and protocols
|
|
|
|
struct Buffer {
|
|
|
|
foo: ();
|
|
|
|
|
|
|
|
drop { }
|
|
|
|
}
|
|
|
|
|
2012-08-22 20:10:48 -05:00
|
|
|
proto! double_buffer (
|
2012-08-07 16:38:32 -05:00
|
|
|
acquire:send {
|
|
|
|
request -> wait_buffer
|
|
|
|
}
|
|
|
|
|
|
|
|
wait_buffer:recv {
|
|
|
|
give_buffer(Buffer) -> release
|
|
|
|
}
|
|
|
|
|
|
|
|
release:send {
|
|
|
|
release(Buffer) -> acquire
|
|
|
|
}
|
2012-08-22 20:10:48 -05:00
|
|
|
)
|
2012-08-07 16:38:32 -05:00
|
|
|
|
|
|
|
// Code examples
|
|
|
|
fn render(_buffer: &Buffer) {
|
|
|
|
// A dummy function.
|
|
|
|
}
|
|
|
|
|
|
|
|
fn draw_frame(+channel: double_buffer::client::acquire) {
|
|
|
|
let channel = request(channel);
|
2012-08-22 19:24:52 -05:00
|
|
|
select! (
|
2012-08-07 16:38:32 -05:00
|
|
|
channel => {
|
|
|
|
give_buffer(buffer) -> channel {
|
|
|
|
render(&buffer);
|
|
|
|
release(channel, move buffer)
|
|
|
|
}
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
);
|
2012-08-07 16:38:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn draw_two_frames(+channel: double_buffer::client::acquire) {
|
|
|
|
let channel = request(channel);
|
2012-08-22 19:24:52 -05:00
|
|
|
let channel = select! (
|
2012-08-07 16:38:32 -05:00
|
|
|
channel => {
|
|
|
|
give_buffer(buffer) -> channel {
|
|
|
|
render(&buffer);
|
|
|
|
release(channel, move buffer)
|
|
|
|
}
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
);
|
2012-08-07 16:38:32 -05:00
|
|
|
let channel = request(channel);
|
2012-08-22 19:24:52 -05:00
|
|
|
select! (
|
2012-08-07 16:38:32 -05:00
|
|
|
channel => {
|
|
|
|
give_buffer(buffer) -> channel {
|
|
|
|
render(&buffer);
|
|
|
|
release(channel, move buffer)
|
|
|
|
}
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
);
|
2012-08-07 16:38:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(bad1)]
|
|
|
|
fn draw_two_frames_bad1(+channel: double_buffer::client::acquire) {
|
|
|
|
let channel = request(channel);
|
2012-08-22 19:24:52 -05:00
|
|
|
select! (
|
2012-08-07 16:38:32 -05:00
|
|
|
channel => {
|
|
|
|
give_buffer(buffer) -> channel {
|
|
|
|
render(&buffer);
|
|
|
|
}
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
);
|
2012-08-07 16:38:32 -05:00
|
|
|
let channel = request(channel);
|
2012-08-22 19:24:52 -05:00
|
|
|
select! (
|
2012-08-07 16:38:32 -05:00
|
|
|
channel => {
|
|
|
|
give_buffer(buffer) -> channel {
|
|
|
|
render(&buffer);
|
|
|
|
release(channel, move buffer)
|
|
|
|
}
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
);
|
2012-08-07 16:38:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(bad2)]
|
|
|
|
fn draw_two_frames_bad2(+channel: double_buffer::client::acquire) {
|
|
|
|
let channel = request(channel);
|
2012-08-22 19:24:52 -05:00
|
|
|
select! (
|
2012-08-07 16:38:32 -05:00
|
|
|
channel => {
|
|
|
|
give_buffer(buffer) -> channel {
|
|
|
|
render(&buffer);
|
|
|
|
release(channel, move buffer);
|
|
|
|
render(&buffer);
|
|
|
|
release(channel, move buffer);
|
|
|
|
}
|
|
|
|
}
|
2012-08-22 19:24:52 -05:00
|
|
|
);
|
2012-08-07 16:38:32 -05:00
|
|
|
}
|
|
|
|
|
2012-08-08 19:24:12 -05:00
|
|
|
fn main() { }
|