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

closure handling for NLL #46509

Merged
merged 21 commits into from
Dec 7, 2017

Conversation

nikomatsakis
Copy link
Contributor

This branch overhauls how we handle closures and universally quantified regions in the NLL code. The goal is to lay the groundwork for "region erasure" by the HIR-based type checker, as well as to avoid "crazy hacks" when it comes to closures. This means that when we type-check a closure, we cannot make use of any of the precise values of free regions in its signature, since those are inferred by the HIR type-checker. Therefore, the new code handles closures by:

  • Creating fresh regions for every free region that appears in the closure's type, which now includes both its signature and the types of all upvars.
    • This means that the closure is type-checked without knowing about the connections.
  • When we encounter some region relationship that we can't locally verify, we propagate it to the closure's creator.
  • During MIR typeck, the closure creators then validates those relationships.

For a good example and explanation, see e.g. the test src/test/nll/closure-requirements/propagate-despite-same-free-region.rs.

Upcoming changes in the next NLL PR (not included in this PR in order to keep it manageable):

  • Improvements to the MIR type checker so that it handles a lot of stuff presently overlooked
  • Refactor how we store region values to make it more efficient, better encapsulated
  • Propagate "type-outlives" relationships like T: 'a in a similar fashion to the one covered in this PR (still in the works, but close)
  • Improvements to error reporting (still in the works)

r? @arielb1 or @pnkfelix

@kennytm kennytm added the S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. label Dec 5, 2017
}

struct ExtraComments<'cx, 'gcx: 'tcx, 'tcx: 'cx> {
_tcx: TyCtxt<'cx, 'gcx, 'tcx>, // don't need it now, but bet we will soon
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A potential alternative: Instead of having ExtraComments carry a tcx and get passed into a callback closure, you might consider having the closure itself be responsible for carrying the tcx?

At least, that's what I ended up deciding to do as part of a cleanup of the drop_flag_effects code, see here, where I found it more pleasant to put one or two type params in the signature than to pass around TyCtxt and its three required lifetimes...

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In this case, it's a mir visitor, not a closure.

Copy link
Contributor Author

@nikomatsakis nikomatsakis Dec 6, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

But also, I don't personally mind passing around the three lifetimes. Especially once the in-band lifetimes work lands makes its way into beta (it's in nightly already...), that will be relatively lightweight.

self.super_constant(constant, location);
let Constant { span, ty, literal } = constant;
self.push(&format!("mir::Constant"));
self.push(&format!("└ span: {:?}", span));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

box characters! 😒

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

but seriously, just in case someone eventually wants to write tests of such output by hand, why not stick to ascii?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

And this person doesn't have an editor that supports copy-and-paste? As it happens, you actually cannot test any of these comments our mir-opt infrastructure, since that infrastructure strips out comments right now. So in that event further changes would be needed anyhow.

I mean I don't care too much, I can change to some ASCII thing (maybe \?) if you prefer. But it seems like we should be able to use box characters in debug output gosh darn it. =)

/// contain any bound regions that would be bound by the
/// binder. This is commonly used to 'inject' a value T into a
/// different binding level.
pub fn new_not_binding<'tcx>(value: T) -> Binder<T>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(Personally i'd call this dummy_binding or new_dummy_binding (since in my head the binding exists, it just is useless), but I can cope with this function name.)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I could get behind Behinder::dummy or something.

@@ -11,7 +11,8 @@
use rustc::hir;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok nic

ref t => bug!("closure with non-closure type: {:?}", t),
}
NodeExpr(&hir::Expr { node: hir::ExprClosure(..), .. }) => {
// In order to property accommodate regions during NLL
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The reason a closure sig can't return fn signatures is because closure signatures are not fn signatures - e.g. they miss the self-argument, and in any case they are right in the closure's type.

// - takes an argument `y`
// - stores `y` into another, longer-lived spot
//
// *but* the signature of the closure doesn't indicate that `y` lives
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

but in the closure's signature, y and the slot are different unrelated higher-ranked regions. Because closure inference is not allowed to add where-clauses between higher-ranked regions, there is no set of free lifetimes that allows the closure to run, so the closure reports the error (and hence we see it before the closure's "external requirements" report).

// - stores `y` into another, longer-lived spot
//
// but is invoked with a spot that doesn't live long
// enough to store `y`.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here the regions are higher-ranked in the closure signature but explicitly linked, so (I hope) this only checks that the closure signature is obeyed.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes

// option. This file may not be copied, modified, or distributed
// except according to those terms.

// As in via-upvar, test closure that:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As in escape-upvar-ref?

@@ -0,0 +1,87 @@
note: External requirements
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where is via-upvar-nested.rs?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

oops, deleted

// Test closure that:
// - captures a variable `y`
// - stores reference to `y` into another, longer-lived spot

Copy link
Contributor

@arielb1 arielb1 Dec 6, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here both the spot and y are upvars and therefore closure-external ("free") regions

// Note: the use of `Cell` here is to introduce invariance. One less
// variable.
//
// FIXME: The `supply` function *ought* to generate an error, but it
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you try to make this an issue on the milestone?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It was #45827 -- this is currently marked as closed however since the work has landed on nll-master (and the test is fixed there).

// option. This file may not be copied, modified, or distributed
// except according to those terms.

// Test a case where we fail to approximate one of the regions and
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Similarly to escape-argument-callee, a test case where the closure requires a relationship between 2 unrelated higher-ranked regions, with no helpful relations between the HBRs and free regions.

Maybe also add to this test cases where:

'c: 'x, 'b: 'y
\- CAN'T HELP, because we can't prevent 'x and 'y from being assigned "different" short lifetimes

'y: 'c
\- can't help, because we can't prevent 'x from being short.


// Callee knows that:
//
// 'x: 'a
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ok

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actually, digging into this revealed something weird. Why is this not getting an error reported? It's propagating the right relationships, but they're not yielding any errors...

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh, it's another case of "is fixed later on in nll-master". In particular, it's #45827 I think but maybe one of the other issues. I can add a fixme, but I just tested and I do get the expected errors on nll-master.

@@ -0,0 +1,26 @@
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can't you find some compile-fail tests and run them with NLL?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Conceivably. =) Yeah, I'll search.

// a variety of errors from the older, AST-based machinery (notably
// borrowck), and then we get the NLL error at the end.

// compile-flags:-Znll -Zborrowck=mir
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do you have different tests with -Z nll and with -Z nll -Z borrowck=mir?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Mistake. It used to be that -Znll implies -Zborrowck-mir so maybe it dates from then. They should really all have -Znll -Zborrowck=mir -Zverbose.

}
}

fn for_each_region_constraint(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

that's one weird factoring. I would have a Display implementation for OutlivesRequirement

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Two reasons I did it this way:

  • We are not displaying to the "end user" -- this is still debug output, so Display feels inappropriate to me. But I could make some sort of newtype that implements Debug.
  • In one context, we want to prepend a | and use writeln! for each line, and in the other, we want to invoke err.note() with each line.

The latter is real killer, though I think err.note() can handle internal newlines.

| ^^^

error: aborting due to 2 previous errors

Copy link
Contributor

@arielb1 arielb1 Dec 6, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thinking about the tests, I didn't see a test where a free region was directly compared against a bound region. aka

fn foo<'a, F: for<'x> FnOnce(
    Cell<&'a u32>,
    Cell<&'x u32>)>(_cell: Cell<&'a u32>, _f: F) { }

fn case1() {
    let a = 0;
    let cell = Cell::new(&a);
    foo(cell, |cell_1, cell_2| {
        cell_1.set(cell_2.get()); // forces 'x: 'a, error in closure
    })
}

fn case1() {
    let a = 0;
    let cell = Cell::new(&a);
    foo(cell, |cell_1, cell_2| {
        cell_2.set(cell_1.get()); // forces 'a: 'x, implies 'a = 'static -> borrow error
    })
}

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You could also have a test where we can use a given to convert a free/bound relationship to a free/free relationship, but we already have tests that show we can use givens to convert bound/bound to free/free (propagate-despite-same-free-region) so I think it's less critical to add another test - you can still add one if you want :-).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yeah, I think existing tests cover that ok (free/bound -> free/free).

// need to propagate; but in fact we do because identity of free
// regions is erased.

// compile-flags:-Znll -Zborrowck=mir -Zverbose
Copy link
Contributor

@arielb1 arielb1 Dec 6, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

doesn't this need a compilation successful marker?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Those are not enforced by the UI framework right now. That reminds me that I was going to open an issue on that. It's kind of the "default".

@bors
Copy link
Collaborator

bors commented Dec 7, 2017

☔ The latest upstream changes (presumably #46533) made this pull request unmergeable. Please resolve the merge conflicts.

@nikomatsakis
Copy link
Contributor Author

I pushed changes addressing all existing comments, except that:

The overall format is now easier to read. Also, There is now graphviz
output, as well as a `#[rustc_regions]` annotation that dumps internal
state.
It's inefficient, and the substitution there doesn't account for the
extra regions used by NLL inference, so it's a bad thing to encourage.

As it happens all callers already know if they have a closure or not,
from what I can tell.
It's just not useful. It also makes it hard to have tests that probe
internal state, since the interning number is very sensitive.

Dumping the number in the case of gensym is not ideal but will do for
now.
@nikomatsakis nikomatsakis force-pushed the nll-master-to-rust-master-3 branch from e1b8261 to 1db58d7 Compare December 7, 2017 10:39

// Shrink `fr` until we find a non-local region (if we do).
// We'll call that `fr-` -- it's ever so slightly smaller than `fr`.
if let Some(fr_minus) = self.universal_regions.non_local_lower_bound(longer_fr) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The more I read this scheme, the less certain I become it has any chance of working in "real life". But I don't really know of a better way, and I suppose there's only 1 way to see.

@arielb1
Copy link
Contributor

arielb1 commented Dec 7, 2017

Nice PR. If you want to change the tests somewhat, that's OK by me.

@bors r+ p=1

@bors
Copy link
Collaborator

bors commented Dec 7, 2017

📌 Commit 1db58d7 has been approved by arielb1

@bors
Copy link
Collaborator

bors commented Dec 7, 2017

⌛ Testing commit 1db58d7 with merge a8437a0...

bors added a commit that referenced this pull request Dec 7, 2017
…ielb1

closure handling for NLL

This branch overhauls how we handle closures and universally quantified regions in the NLL code. The goal is to lay the groundwork for "region erasure" by the HIR-based type checker, as well as to avoid "crazy hacks" when it comes to closures. This means that when we type-check a closure, we cannot make use of *any* of the precise values of free regions in its signature, since those are inferred by the HIR type-checker. Therefore, the new code handles closures by:

- Creating fresh regions for every free region that appears in the closure's type, which now includes both its signature and the types of all upvars.
    - This means that the closure is type-checked without knowing about the connections.
- When we encounter some region relationship that we can't locally verify, we propagate it to the closure's creator.
- During MIR typeck, the closure creators then validates those relationships.

For a good example and explanation, see e.g. the test `src/test/nll/closure-requirements/propagate-despite-same-free-region.rs`.

Upcoming changes in the next NLL PR (not included in this PR in order to keep it manageable):

- Improvements to the MIR type checker so that it handles a lot of stuff presently overlooked
- Refactor how we store region values to make it more efficient, better encapsulated
- Propagate "type-outlives" relationships like `T: 'a` in a similar fashion to the one covered in this PR (still in the works, but close)
- Improvements to error reporting (still in the works)

r? @arielb1 or @pnkfelix
@bors
Copy link
Collaborator

bors commented Dec 7, 2017

☀️ Test successful - status-appveyor, status-travis
Approved by: arielb1
Pushing a8437a0 to master...

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
S-waiting-on-review Status: Awaiting review from the assignee but also interested parties.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants