Skip to content

Commit a8c7761

Browse files
Add tests
1 parent dccfeb9 commit a8c7761

File tree

5 files changed

+148
-19
lines changed

5 files changed

+148
-19
lines changed

compiler/rustc_ast_lowering/src/lib.rs

+16-9
Original file line numberDiff line numberDiff line change
@@ -132,6 +132,7 @@ struct LoweringContext<'a, 'hir> {
132132

133133
allow_try_trait: Lrc<[Symbol]>,
134134
allow_gen_future: Lrc<[Symbol]>,
135+
allow_async_iterator: Lrc<[Symbol]>,
135136

136137
/// Mapping from generics `def_id`s to TAIT generics `def_id`s.
137138
/// For each captured lifetime (e.g., 'a), we create a new lifetime parameter that is a generic
@@ -176,6 +177,8 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
176177
} else {
177178
[sym::gen_future].into()
178179
},
180+
// FIXME(gen_blocks): how does `closure_track_caller`
181+
allow_async_iterator: [sym::gen_future, sym::async_iterator].into(),
179182
generics_def_id_map: Default::default(),
180183
host_param_id: None,
181184
}
@@ -1900,14 +1903,18 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
19001903
fn_span: Span,
19011904
) -> hir::FnRetTy<'hir> {
19021905
let span = self.lower_span(fn_span);
1903-
let opaque_ty_span = self.mark_span_with_reason(DesugaringKind::Async, span, None);
19041906

1905-
let opaque_ty_node_id = match coro {
1906-
CoroutineKind::Async { return_impl_trait_id, .. }
1907-
| CoroutineKind::Gen { return_impl_trait_id, .. }
1908-
| CoroutineKind::AsyncGen { return_impl_trait_id, .. } => return_impl_trait_id,
1907+
let (opaque_ty_node_id, allowed_features) = match coro {
1908+
CoroutineKind::Async { return_impl_trait_id, .. } => (return_impl_trait_id, None),
1909+
CoroutineKind::Gen { return_impl_trait_id, .. } => (return_impl_trait_id, None),
1910+
CoroutineKind::AsyncGen { return_impl_trait_id, .. } => {
1911+
(return_impl_trait_id, Some(self.allow_async_iterator.clone()))
1912+
}
19091913
};
19101914

1915+
let opaque_ty_span =
1916+
self.mark_span_with_reason(DesugaringKind::Async, span, allowed_features);
1917+
19111918
let captured_lifetimes: Vec<_> = self
19121919
.resolver
19131920
.take_extra_lifetime_params(opaque_ty_node_id)
@@ -1926,7 +1933,7 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
19261933
let bound = this.lower_coroutine_fn_output_type_to_bound(
19271934
output,
19281935
coro,
1929-
span,
1936+
opaque_ty_span,
19301937
ImplTraitContext::ReturnPositionOpaqueTy {
19311938
origin: hir::OpaqueTyOrigin::FnReturn(fn_def_id),
19321939
fn_kind,
@@ -1945,7 +1952,7 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
19451952
&mut self,
19461953
output: &FnRetTy,
19471954
coro: CoroutineKind,
1948-
span: Span,
1955+
opaque_ty_span: Span,
19491956
nested_impl_trait_context: ImplTraitContext,
19501957
) -> hir::GenericBound<'hir> {
19511958
// Compute the `T` in `Future<Output = T>` from the return type.
@@ -1968,14 +1975,14 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
19681975

19691976
let future_args = self.arena.alloc(hir::GenericArgs {
19701977
args: &[],
1971-
bindings: arena_vec![self; self.assoc_ty_binding(assoc_ty_name, span, output_ty)],
1978+
bindings: arena_vec![self; self.assoc_ty_binding(assoc_ty_name, opaque_ty_span, output_ty)],
19721979
parenthesized: hir::GenericArgsParentheses::No,
19731980
span_ext: DUMMY_SP,
19741981
});
19751982

19761983
hir::GenericBound::LangItemTrait(
19771984
trait_lang_item,
1978-
self.lower_span(span),
1985+
opaque_ty_span,
19791986
self.next_id(),
19801987
future_args,
19811988
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
error[E0658]: gen blocks are experimental
2+
--> $DIR/async_gen_fn.rs:4:1
3+
|
4+
LL | async gen fn foo() {}
5+
| ^^^^^^^^^
6+
|
7+
= note: see issue #117078 <https://github.com/rust-lang/rust/issues/117078> for more information
8+
= help: add `#![feature(gen_blocks)]` to the crate attributes to enable
9+
10+
error: aborting due to 1 previous error
11+
12+
For more information about this error, try `rustc --explain E0658`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
error[E0670]: `async fn` is not permitted in Rust 2015
2+
--> $DIR/async_gen_fn.rs:4:1
3+
|
4+
LL | async gen fn foo() {}
5+
| ^^^^^ to use `async fn`, switch to Rust 2018 or later
6+
|
7+
= help: pass `--edition 2021` to `rustc`
8+
= note: for more on editions, read https://doc.rust-lang.org/edition-guide
9+
10+
error: expected one of `extern`, `fn`, or `unsafe`, found `gen`
11+
--> $DIR/async_gen_fn.rs:4:7
12+
|
13+
LL | async gen fn foo() {}
14+
| ^^^ expected one of `extern`, `fn`, or `unsafe`
15+
16+
error: aborting due to 2 previous errors
17+
18+
For more information about this error, try `rustc --explain E0670`.

tests/ui/coroutine/async_gen_fn.rs

+6-10
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,9 @@
1-
// edition: 2024
2-
// compile-flags: -Zunstable-options
3-
// check-pass
1+
// revisions: e2024 none
2+
//[e2024] compile-flags: --edition 2024 -Zunstable-options
43

5-
#![feature(gen_blocks, async_iterator)]
6-
7-
async fn bar() {}
8-
9-
async gen fn foo() {
10-
yield bar().await;
11-
}
4+
async gen fn foo() {}
5+
//[none]~^ ERROR: `async fn` is not permitted in Rust 2015
6+
//[none]~| ERROR: expected one of `extern`, `fn`, or `unsafe`, found `gen`
7+
//[e2024]~^^^ ERROR: gen blocks are experimental
128

139
fn main() {}
+96
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
// edition: 2024
2+
// compile-flags: -Zunstable-options
3+
// run-pass
4+
5+
#![feature(gen_blocks, async_iterator)]
6+
7+
// make sure that a ridiculously simple async gen fn works as an iterator.
8+
9+
async fn pause() {
10+
// this doesn't actually do anything, lol
11+
}
12+
13+
async fn one() -> i32 {
14+
1
15+
}
16+
17+
async fn two() -> i32 {
18+
2
19+
}
20+
21+
async gen fn foo() -> i32 {
22+
yield one().await;
23+
pause().await;
24+
yield two().await;
25+
pause().await;
26+
yield 3;
27+
pause().await;
28+
}
29+
30+
async fn async_main() {
31+
let mut iter = std::pin::pin!(foo());
32+
assert_eq!(iter.next().await, Some(1));
33+
assert_eq!(iter.as_mut().next().await, Some(2));
34+
assert_eq!(iter.as_mut().next().await, Some(3));
35+
assert_eq!(iter.as_mut().next().await, None);
36+
}
37+
38+
// ------------------------------------------------------------------------- //
39+
// Implementation Details Below...
40+
41+
use std::pin::Pin;
42+
use std::task::*;
43+
use std::async_iter::AsyncIterator;
44+
use std::future::Future;
45+
46+
trait AsyncIterExt {
47+
fn next(&mut self) -> Next<'_, Self>;
48+
}
49+
50+
impl<T> AsyncIterExt for T {
51+
fn next(&mut self) -> Next<'_, Self> {
52+
Next { s: self }
53+
}
54+
}
55+
56+
struct Next<'s, S: ?Sized> {
57+
s: &'s mut S,
58+
}
59+
60+
impl<'s, S: AsyncIterator> Future for Next<'s, S> where S: Unpin {
61+
type Output = Option<S::Item>;
62+
63+
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<S::Item>> {
64+
Pin::new(&mut *self.s).poll_next(cx)
65+
}
66+
}
67+
68+
pub fn noop_waker() -> Waker {
69+
let raw = RawWaker::new(std::ptr::null(), &NOOP_WAKER_VTABLE);
70+
71+
// SAFETY: the contracts for RawWaker and RawWakerVTable are upheld
72+
unsafe { Waker::from_raw(raw) }
73+
}
74+
75+
const NOOP_WAKER_VTABLE: RawWakerVTable = RawWakerVTable::new(noop_clone, noop, noop, noop);
76+
77+
unsafe fn noop_clone(_p: *const ()) -> RawWaker {
78+
RawWaker::new(std::ptr::null(), &NOOP_WAKER_VTABLE)
79+
}
80+
81+
unsafe fn noop(_p: *const ()) {}
82+
83+
fn main() {
84+
let mut fut = async_main();
85+
86+
// Poll loop, just to test the future...
87+
let waker = noop_waker();
88+
let ctx = &mut Context::from_waker(&waker);
89+
90+
loop {
91+
match unsafe { Pin::new_unchecked(&mut fut).poll(ctx) } {
92+
Poll::Pending => {}
93+
Poll::Ready(()) => break,
94+
}
95+
}
96+
}

0 commit comments

Comments
 (0)