Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

False positives of local variables crossing await scopes. #137721

Closed
taoseng opened this issue Feb 27, 2025 · 6 comments
Closed

False positives of local variables crossing await scopes. #137721

taoseng opened this issue Feb 27, 2025 · 6 comments
Labels
A-async-await Area: Async & Await

Comments

@taoseng
Copy link

taoseng commented Feb 27, 2025

Hello, I have a piece of code that fails to compile, but I believe it is a false positive.

use rand::Rng;
use std::future::Future;
use std::mem;

#[tokio::main]
async fn main() {
    accept_send_future(foo()).await;
}

fn accept_send_future<F>(f: F) -> F
where
    F: Future + Send,
{
    f
}

async fn foo() {
    let mut rng = rand::rng();                 // rng is not Send.
    let idx = rng.random_range(0..100);
    
    mem::drop(rng);  // drop the rng

    bar(idx).await;
}

async fn bar(id: usize) {
    println!("bar: {}", id);
}

i got an error:

error: future cannot be sent between threads safely
  --> src/main.rs:7:24
   |
7  |     accept_send_future(foo()).await;
   |                        ^^^^^ future returned by `foo` is not `Send`
   |
   = help: within `impl Future<Output = ()>`, the trait `Send` is not implemented for `Rc<UnsafeCell<ReseedingRng<rand_chacha::chacha::ChaCha12Core, OsRng>>>`, which is required by `impl Future<Output = ()>: Send`
note: future is not `Send` as this value is used across an await
  --> src/main.rs:29:14
   |
26 |     let mut rng = rand::rng();
   |         ------- has type `ThreadRng` which is not `Send`
...
29 |     bar(idx).await;
   |              ^^^^^ await occurs here, with `mut rng` maybe used later
note: required by a bound in `accept_send_future`
  --> src/main.rs:12:17
   |
10 | fn accept_send_future<F>(f: F) -> F
   |    ------------------ required by a bound in this function
11 | where
12 |     F: Future + Send,
   |                 ^^^^ required by this bound in `accept_send_future`

the 'rng' is expliciyly dropped, and it never cross the await.

@rustbot rustbot added the needs-triage This issue may need triage. Remove it if it has been sufficiently triaged. label Feb 27, 2025
@taoseng
Copy link
Author

taoseng commented Feb 27, 2025

I had to modify the code of foo() as follows for it to compile successfully:

async fn foo() {
    let idx = {
        let mut rng = rand::rng();
        rng.random_range(0..100)
    };
    bar(idx).await;
}

@taoseng
Copy link
Author

taoseng commented Feb 27, 2025

oh, Let me add something here, the 'rng' is Copy, so drop is useless.

@theemathas
Copy link
Contributor

Duplicate of #63768

@lqd
Copy link
Member

lqd commented Feb 27, 2025

the 'rng' is Copy

Is that accurate, https://docs.rs/rand/latest/rand/fn.rng.html returns https://docs.rs/rand/latest/rand/rngs/struct.ThreadRng.html which is not Copy?

@jieyouxu jieyouxu added the A-async-await Area: Async & Await label Feb 27, 2025
@taoseng
Copy link
Author

taoseng commented Feb 27, 2025

the 'rng' is Copy

Is that accurate, https://docs.rs/rand/latest/rand/fn.rng.html returns https://docs.rs/rand/latest/rand/rngs/struct.ThreadRng.html which is not Copy?

sorry, I got the wrong version.
In rand_v0.7.3 ThreadRng is Copy, but in 0.9.0 ThreadRng is not Copy.

My example is based on version 0.9.0, and this ThreadRng is indeed not Copy.

@Noratrieb
Copy link
Member

going to close this as a duplicate as suggested by @theemathas which seems correct to me, but correct me if I'm wrong.

@jieyouxu jieyouxu removed the needs-triage This issue may need triage. Remove it if it has been sufficiently triaged. label Mar 3, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-async-await Area: Async & Await
Projects
None yet
Development

No branches or pull requests

7 participants