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

Lifetime Pointer to Lifetime Closure Bug #12717

Closed
bvssvni opened this issue Mar 5, 2014 · 5 comments
Closed

Lifetime Pointer to Lifetime Closure Bug #12717

bvssvni opened this issue Mar 5, 2014 · 5 comments

Comments

@bvssvni
Copy link

bvssvni commented Mar 5, 2014

I was told in the IRC channel the bug is Test::new missing a "'a" and should give the same warning as Test2::new.

rustc 0.10-pre (0a5138c 2014-03-04 13:16:41 -0800)

/// A struct with borrowed pointer to a closure.
/// This is to be able to reuse the closure then struct runs out of scope.

pub struct Test<'a> {
    f: &'a 'a ||
}

impl<'a> Test<'a> {
        // This should give the same warning as Test2.
    pub fn new(f: &'a ||) -> Test<'a> {
        Test { f: f }
    }
}

pub struct Test2<'a> {
    f: Option<&'a 'a ||>
}

impl<'a> Test2<'a> {
    pub fn new(f: &'a ||) -> Test2<'a> {
        Test2 { f: Some(f) }
    }
}

fn main() {
    let x = || println!("hello!");

    // Works
    (*Test::new(&x).f)();

    // Fails
    (*Test2::new(&x).f.unwrap())();
}


Compiler error:
optionclosure.rs:21:3: 21:23 error: cannot infer an appropriate lifetime for region in type/impl due to conflicting requirements
optionclosure.rs:21         Test2 { f: Some(f) }
                            ^~~~~~~~~~~~~~~~~~~~
optionclosure.rs:21:14: 21:21 note: first, the lifetime cannot outlive the call at 21:13...
optionclosure.rs:21         Test2 { f: Some(f) }
                                       ^~~~~~~
optionclosure.rs:21:14: 21:21 note: ...so that return value is valid for the call
optionclosure.rs:21         Test2 { f: Some(f) }
                                       ^~~~~~~
optionclosure.rs:20:37: 22:3 note: but, the lifetime must be valid for the lifetime &'a  as defined on the block at 20:36...
optionclosure.rs:20     pub fn new(f: &'a ||) -> Test2<'a> {
optionclosure.rs:21         Test2 { f: Some(f) }
optionclosure.rs:22     }
optionclosure.rs:21:3: 21:23 note: ...so that types are compatible (expected `Test2<'a>` but found `Test2<>`)
optionclosure.rs:21         Test2 { f: Some(f) }
@Aatch
Copy link
Contributor

Aatch commented Mar 5, 2014

From what I can tell, the anonymous lifetime assigned to the closure in the new call should cause the first example to fail with a similar error. However, I am not familiar enough with these specific rules to be sure.

@bvssvni
Copy link
Author

bvssvni commented Mar 5, 2014

If "&'a ||" always equals "&'a 'a ||" the second example should work, but neither do I know the specific rules. It is better to have both working or failing, because it is possible to define traits this way and get surprising errors in the impl.

@bvssvni
Copy link
Author

bvssvni commented Mar 5, 2014

This is the code where I discovered the bug:

When I change to "&'a 'a ||" in the trait it gives me this lifetime error. If I change to "&'a ||" in the trait it gives me same error as the first example. Maybe these two bugs are related?

rustc 0.10-pre (0a5138c 2014-03-04 13:16:41 -0800)


pub trait Expr<'a, T> {
    fn then(&'a self, f: &'a 'a ||) -> T;
}

impl<'a> Expr<'a, ()> for () {
    fn then(&'a self, f: &'a 'a ||) -> () {
        (*f)();
    }
}

pub struct Action<'a> {
    f: Option<&'a 'a ||>,
    next: Option<&'a Action<'a>>,
}

impl<'a> Action<'a> {
    pub fn new(f: &'a 'a ||) -> Action<'a> {
        Action { f: Some(f), next: None }
    }

    pub fn run(&self) {
        match self.next {
            Some(action) => action.run(),
            None => {},
        }

        match self.f {
            None => {},
            Some(f) => (*f)(),
        };
    }
}

impl<'a> Expr<'a, Action<'a>> for Action<'a> {
    fn then(&'a self, f: &'a 'a ||) -> Action<'a> {
        Action { f: Some(f), next: Some(self) }
    }
}

pub fn main() {
    let x = || println!("Hello!");
    let y = || println!("Again!");
    let z = || println!("So...");

    ().then(&x).then(&y).then(&z);

    Action::new(&x).then(&y).then(&z).run();
}


Compiler error:
impl2.rs:46:2: 46:4 error: cannot infer an appropriate lifetime for region in type/impl due to conflicting requirements
impl2.rs:46     ().then(&x).then(&y).then(&z);
                ^~
impl2.rs:48:2: 48:26 note: first, the lifetime must be contained by the method call at 48:1...
impl2.rs:48     Action::new(&x).then(&y).then(&z).run();
                ^~~~~~~~~~~~~~~~~~~~~~~~
impl2.rs:48:2: 48:17 note: ...so that method receiver is valid for the method call
impl2.rs:48     Action::new(&x).then(&y).then(&z).run();
                ^~~~~~~~~~~~~~~
impl2.rs:46:10: 46:12 note: but, the lifetime must also be contained by the expression at 46:9...
impl2.rs:46     ().then(&x).then(&y).then(&z);

@edwardw
Copy link
Contributor

edwardw commented Mar 17, 2014

Interestingly, with #12828 both work. Maybe we can close this bug?

@reem
Copy link
Contributor

reem commented Dec 15, 2014

Triage: both examples fail, but because of calling a closure through &, not due to any lifetimes. Should be closed.

@bvssvni bvssvni closed this as completed Dec 15, 2014
flip1995 pushed a commit to flip1995/rust that referenced this issue Feb 27, 2025
fixes rust-lang#12717.

r? @xFrednet

----

changelog: add [`io_other_error`] lint
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

4 participants