From 4b353882673e2cc7b1a91a35075914d169e6114c Mon Sep 17 00:00:00 2001 From: Eric Holk Date: Wed, 29 Sep 2021 10:40:07 -0700 Subject: [PATCH 1/4] Document auto trait inference for async blocks We are in the process of changing this (rust-lang/#69663), but it would be good to document the existing rules before changing them. This should also help explain the compilation errors people are getting in the meantime. --- src/expressions/block-expr.md | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/src/expressions/block-expr.md b/src/expressions/block-expr.md index a68b27e56..49af0f83d 100644 --- a/src/expressions/block-expr.md +++ b/src/expressions/block-expr.md @@ -117,6 +117,24 @@ loop { } ``` +### Auto traits and `async` blocks + +Auto trait inference for `async` blocks follow the same [rules as closures] except that [temporary values that are in scope][temporary-scopes] at an `await` expression are also considered. For example, consider the following block: + +```rust +#fn bar() -> i32 { 42 } +#async fn foo() {} +async { + match bar() { + _ => foo().await, + } +} +#; +``` + +Here the result of `bar()` is in scope during the await of `foo()`, so the result of `bar()` will impact the inferred auto traits. +If `bar()` is not `Send`, then the future for the whole match block will also not be `Send`. + ## `unsafe` blocks > **Syntax**\ @@ -189,3 +207,5 @@ fn is_unix_platform() -> bool { [tuple expressions]: tuple-expr.md [unsafe operations]: ../unsafety.md [value expressions]: ../expressions.md#place-expressions-and-value-expressions +[rules as closures]: ../special-types-and-traits.md#auto-traits +[temporary-scopes]: ../destructors.md#temporary-scopes From 00fce4a28636d9f4b68836d8574b65696347b21d Mon Sep 17 00:00:00 2001 From: Eric Holk Date: Wed, 29 Sep 2021 11:26:56 -0700 Subject: [PATCH 2/4] Fix tests --- src/expressions/block-expr.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/expressions/block-expr.md b/src/expressions/block-expr.md index 49af0f83d..8fbbce18a 100644 --- a/src/expressions/block-expr.md +++ b/src/expressions/block-expr.md @@ -122,14 +122,14 @@ loop { Auto trait inference for `async` blocks follow the same [rules as closures] except that [temporary values that are in scope][temporary-scopes] at an `await` expression are also considered. For example, consider the following block: ```rust -#fn bar() -> i32 { 42 } -#async fn foo() {} +# fn bar() -> i32 { 42 } +# async fn foo() {} async { match bar() { _ => foo().await, } } -#; +# ; ``` Here the result of `bar()` is in scope during the await of `foo()`, so the result of `bar()` will impact the inferred auto traits. From b33a14e1e76d70b37ca11ffa1bdfc1fa3a4e2a39 Mon Sep 17 00:00:00 2001 From: Eric Holk Date: Wed, 20 Oct 2021 11:03:05 -0700 Subject: [PATCH 3/4] Add positive and negative examples to try to make the behavior more clear --- src/expressions/block-expr.md | 44 ++++++++++++++++++++++++++++++++++- 1 file changed, 43 insertions(+), 1 deletion(-) diff --git a/src/expressions/block-expr.md b/src/expressions/block-expr.md index 8fbbce18a..9efbf6047 100644 --- a/src/expressions/block-expr.md +++ b/src/expressions/block-expr.md @@ -132,9 +132,51 @@ async { # ; ``` -Here the result of `bar()` is in scope during the await of `foo()`, so the result of `bar()` will impact the inferred auto traits. +Here the result of `bar()` is in scope for the await of `foo()`, so the result of `bar()` will impact the inferred auto traits. If `bar()` is not `Send`, then the future for the whole match block will also not be `Send`. +For a more complex example, consider the case below. +```rust +# struct Client; +# impl Client { +# fn new() -> Self { Client } +# fn status(&self) -> u32 { 200 } +# } +# async fn foo() {} +async { + let client = Client::new(); + match client.status() { + _ => foo().await, + } +} +# ; +``` +Here, `status()` is declared as `fn status(&self) -> u32`. +Because there is an implicit borrow of `client` in the `client.status()` call, a value of type `&Client` will be in scope for the `await` expression. +If `&Client` is `Send`, then the whole future will be `Send`. +If `&Client` is not `Send`, then the future will also not be `Send`. +The borrow of `client` remains in scope for the `match` body because the scrutineee of a `match` expression [does not introduce a temporary scope]. +However, we can work around this by explicitly introducing a scope, as shown below. +```rust +# struct Client; +# impl Client { +# fn new() -> Self { Client } +# fn status(&self) -> u32 { 200 } +# } +# async fn foo() {} +async { + let client = Client::new(); + match { let status = client.status(); status } { + _ => foo().await, + } +} +# ; +``` +Explicitly introducing a block means that the borrow of `client` will be released when the block exits, and therefore `&Client` will not be considered in determining auto traits for the overall future. +In this case, only the `u32` returned from `status` would be relevant. + +[does not introduce a temporary scope]: ../destructors.md#temporary-scopes + ## `unsafe` blocks > **Syntax**\ From 734fc47b2b6d3e2dbed10c0e3730806727a9d139 Mon Sep 17 00:00:00 2001 From: Eric Holk Date: Wed, 20 Oct 2021 11:25:12 -0700 Subject: [PATCH 4/4] Trim trailing whitespace --- src/expressions/block-expr.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/expressions/block-expr.md b/src/expressions/block-expr.md index 9efbf6047..6fe28e800 100644 --- a/src/expressions/block-expr.md +++ b/src/expressions/block-expr.md @@ -140,7 +140,7 @@ For a more complex example, consider the case below. # struct Client; # impl Client { # fn new() -> Self { Client } -# fn status(&self) -> u32 { 200 } +# fn status(&self) -> u32 { 200 } # } # async fn foo() {} async { @@ -161,7 +161,7 @@ However, we can work around this by explicitly introducing a scope, as shown bel # struct Client; # impl Client { # fn new() -> Self { Client } -# fn status(&self) -> u32 { 200 } +# fn status(&self) -> u32 { 200 } # } # async fn foo() {} async {