rust/tests/run-pass/generator.rs

97 lines
1.8 KiB
Rust
Raw Normal View History

2019-11-22 09:50:22 +01:00
#![feature(generators, generator_trait)]
2017-08-30 11:13:01 +02:00
use std::ops::{GeneratorState, Generator};
2019-01-30 11:51:06 +01:00
use std::pin::Pin;
2017-08-30 11:13:01 +02:00
fn finish<T>(mut amt: usize, mut t: T) -> T::Return
2018-11-20 12:51:55 +01:00
where T: Generator<Yield = usize>
2017-08-30 11:13:01 +02:00
{
2019-01-30 11:51:06 +01:00
// We are not moving the `t` around until it gets dropped, so this is okay.
let mut t = unsafe { Pin::new_unchecked(&mut t) };
2017-08-30 11:13:01 +02:00
loop {
2019-01-30 11:51:06 +01:00
match t.as_mut().resume() {
2018-11-20 12:51:55 +01:00
GeneratorState::Yielded(y) => amt -= y,
2017-08-30 11:13:01 +02:00
GeneratorState::Complete(ret) => {
assert_eq!(amt, 0);
return ret
}
}
}
}
2017-08-30 11:13:01 +02:00
enum Never {}
fn never() -> Never {
panic!()
2017-08-30 11:13:01 +02:00
}
fn main() {
2018-11-20 12:51:55 +01:00
finish(1, || yield 1);
2017-08-30 11:13:01 +02:00
finish(3, || {
let mut x = 0;
2018-11-20 12:51:55 +01:00
yield 1;
2017-08-30 11:13:01 +02:00
x += 1;
2018-11-20 12:51:55 +01:00
yield 1;
2017-08-30 11:13:01 +02:00
x += 1;
2018-11-20 12:51:55 +01:00
yield 1;
2017-08-30 11:13:01 +02:00
assert_eq!(x, 2);
});
2018-11-20 12:51:55 +01:00
finish(7*8/2, || {
for i in 0..8 {
yield i;
2017-08-30 11:13:01 +02:00
}
});
2017-08-30 11:13:01 +02:00
finish(1, || {
if true {
2018-11-20 12:51:55 +01:00
yield 1;
2017-08-30 11:13:01 +02:00
} else {
}
});
2017-08-30 11:13:01 +02:00
finish(1, || {
if false {
} else {
2018-11-20 12:51:55 +01:00
yield 1;
2017-08-30 11:13:01 +02:00
}
});
2017-08-30 11:13:01 +02:00
finish(2, || {
2018-11-20 12:51:55 +01:00
if { yield 1; false } {
yield 1;
2017-08-30 11:13:01 +02:00
panic!()
}
2018-11-20 12:51:55 +01:00
yield 1;
2017-08-30 11:13:01 +02:00
});
2018-11-20 12:51:55 +01:00
// also test a self-referential generator
assert_eq!(
finish(5, || {
let mut x = Box::new(5);
let y = &mut *x;
*y = 5;
yield *y;
*y = 10;
*x
}),
10
);
let b = true;
finish(1, || {
yield 1;
if b { return; }
#[allow(unused)]
let x = never();
yield 2;
drop(x);
});
finish(3, || {
yield 1;
#[allow(unreachable_code)]
let _x: (String, !) = (String::new(), { yield 2; return });
});
2017-08-30 11:13:01 +02:00
}