-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
closure handling for NLL #46509
Conversation
} | ||
|
||
struct ExtraComments<'cx, 'gcx: 'tcx, 'tcx: 'cx> { | ||
_tcx: TyCtxt<'cx, 'gcx, 'tcx>, // don't need it now, but bet we will soon |
There was a problem hiding this comment.
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...
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
box characters! 😒
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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. =)
src/librustc/ty/sty.rs
Outdated
/// 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> |
There was a problem hiding this comment.
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.)
There was a problem hiding this comment.
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; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ok nic
src/librustc_typeck/collect.rs
Outdated
ref t => bug!("closure with non-closure type: {:?}", t), | ||
} | ||
NodeExpr(&hir::Expr { node: hir::ExprClosure(..), .. }) => { | ||
// In order to property accommodate regions during NLL |
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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`. |
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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: |
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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
?
There was a problem hiding this comment.
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 | ||
|
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ok
There was a problem hiding this comment.
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...
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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
?
There was a problem hiding this comment.
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( |
There was a problem hiding this comment.
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
There was a problem hiding this comment.
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 implementsDebug
. - In one context, we want to prepend a
|
and usewriteln!
for each line, and in the other, we want to invokeerr.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 | ||
|
There was a problem hiding this comment.
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
})
}
There was a problem hiding this comment.
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 :-).
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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".
☔ The latest upstream changes (presumably #46533) made this pull request unmergeable. Please resolve the merge conflicts. |
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.
This includes the additional case suggested by arielb1.
as suggested by arielb1
suggested by arielb1
per pnkfelix's suggestion
e1b8261
to
1db58d7
Compare
|
||
// 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) { |
There was a problem hiding this comment.
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.
Nice PR. If you want to change the tests somewhat, that's OK by me. @bors r+ p=1 |
📌 Commit 1db58d7 has been approved by |
…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
☀️ Test successful - status-appveyor, status-travis |
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:
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):
T: 'a
in a similar fashion to the one covered in this PR (still in the works, but close)r? @arielb1 or @pnkfelix