Skip to content

Commit 6927b51

Browse files
authored
Rollup merge of rust-lang#132151 - compiler-errors:coroutine-resume-outlives, r=spastorino
Ensure that resume arg outlives region bound for coroutines When proving that `{Coroutine}: 'region`, we must also prove that the coroutine's resume ty outlives that region as well. See the inline comment. Fixes rust-lang#132104
2 parents 84e6f4e + ad76564 commit 6927b51

File tree

5 files changed

+110
-0
lines changed

5 files changed

+110
-0
lines changed

compiler/rustc_type_ir/src/outlives.rs

+12
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,18 @@ impl<I: Interner> TypeVisitor<I> for OutlivesCollector<'_, I> {
110110
ty::Coroutine(_, args) => {
111111
args.as_coroutine().tupled_upvars_ty().visit_with(self);
112112

113+
// Coroutines may not outlive a region unless the resume
114+
// ty outlives a region. This is because the resume ty may
115+
// store data that lives shorter than this outlives region
116+
// across yield points, which may subsequently be accessed
117+
// after the coroutine is resumed again.
118+
//
119+
// Conceptually, you may think of the resume arg as an upvar
120+
// of `&mut Option<ResumeArgTy>`, since it is kinda like
121+
// storage shared between the callee of the coroutine and the
122+
// coroutine body.
123+
args.as_coroutine().resume_ty().visit_with(self);
124+
113125
// We ignore regions in the coroutine interior as we don't
114126
// want these to affect region inference
115127
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
// Regression test for 132104
2+
3+
#![feature(coroutine_trait, coroutines)]
4+
5+
use std::ops::Coroutine;
6+
use std::{thread, time};
7+
8+
fn demo<'not_static>(s: &'not_static str) -> thread::JoinHandle<()> {
9+
let mut generator = Box::pin({
10+
#[coroutine]
11+
move |_ctx| {
12+
let ctx: &'not_static str = yield;
13+
yield;
14+
dbg!(ctx);
15+
}
16+
});
17+
18+
// exploit:
19+
generator.as_mut().resume("");
20+
generator.as_mut().resume(s); // <- generator hoards it as `let ctx`.
21+
//~^ ERROR borrowed data escapes outside of function
22+
thread::spawn(move || {
23+
thread::sleep(time::Duration::from_millis(200));
24+
generator.as_mut().resume(""); // <- resumes from the last `yield`, running `dbg!(ctx)`.
25+
})
26+
}
27+
28+
fn main() {
29+
let local = String::from("...");
30+
let thread = demo(&local);
31+
drop(local);
32+
let _unrelated = String::from("UAF");
33+
thread.join().unwrap();
34+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
error[E0521]: borrowed data escapes outside of function
2+
--> $DIR/resume-arg-outlives-2.rs:20:5
3+
|
4+
LL | fn demo<'not_static>(s: &'not_static str) -> thread::JoinHandle<()> {
5+
| ----------- - `s` is a reference that is only valid in the function body
6+
| |
7+
| lifetime `'not_static` defined here
8+
...
9+
LL | generator.as_mut().resume(s); // <- generator hoards it as `let ctx`.
10+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
11+
| |
12+
| `s` escapes the function body here
13+
| argument requires that `'not_static` must outlive `'static`
14+
15+
error: aborting due to 1 previous error
16+
17+
For more information about this error, try `rustc --explain E0521`.
+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
// Regression test for 132104
2+
3+
#![feature(coroutine_trait, coroutines)]
4+
5+
use std::ops::Coroutine;
6+
use std::pin::Pin;
7+
8+
fn demo<'not_static>(s: &'not_static str) -> Pin<Box<impl Coroutine<&'not_static str> + 'static>> {
9+
let mut generator = Box::pin({
10+
#[coroutine]
11+
move |ctx: &'not_static str| {
12+
yield;
13+
dbg!(ctx);
14+
}
15+
});
16+
generator.as_mut().resume(s);
17+
generator
18+
//~^ ERROR lifetime may not live long enough
19+
}
20+
21+
fn main() {
22+
let local = String::from("...");
23+
let mut coro = demo(&local);
24+
drop(local);
25+
let _unrelated = String::from("UAF");
26+
coro.as_mut().resume("");
27+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
error: lifetime may not live long enough
2+
--> $DIR/resume-arg-outlives.rs:17:5
3+
|
4+
LL | fn demo<'not_static>(s: &'not_static str) -> Pin<Box<impl Coroutine<&'not_static str> + 'static>> {
5+
| ----------- lifetime `'not_static` defined here
6+
...
7+
LL | generator
8+
| ^^^^^^^^^ returning this value requires that `'not_static` must outlive `'static`
9+
|
10+
help: consider changing `impl Coroutine<&'not_static str> + 'static`'s explicit `'static` bound to the lifetime of argument `s`
11+
|
12+
LL | fn demo<'not_static>(s: &'not_static str) -> Pin<Box<impl Coroutine<&'not_static str> + 'not_static>> {
13+
| ~~~~~~~~~~~
14+
help: alternatively, add an explicit `'static` bound to this reference
15+
|
16+
LL | fn demo<'not_static>(s: &'static str) -> Pin<Box<impl Coroutine<&'not_static str> + 'static>> {
17+
| ~~~~~~~~~~~~
18+
19+
error: aborting due to 1 previous error
20+

0 commit comments

Comments
 (0)