Skip to content

Commit d77f6d5

Browse files
committed
Auto merge of #21657 - pnkfelix:block-remainder-extents, r=nikomatsakis
Add `CodeExtent::Remainder` variant; pre-req for new scoping/drop rules. This new enum variant introduces finer-grain code extents, i.e. we now track that a binding lives only for a suffix of a block, and (importantly) will be dropped when it goes out of scope *before* the bindings that occurred earlier in the block. Both of these notions are neatly captured by marking the block (and each suffix) as an enclosing scope of the next suffix beneath it. This is work that is part of the foundation for issue #8861. (It actually has been seen in earlier posted pull requests, in particular #21022; I have just factored it out into its own PR to ease my own near-future rebasing, and also get people used to the new rules.) ---- These finer grained scopes do mean that some code is newly rejected by `rustc`; for example: ```rust let mut map : HashMap<u8, &u8> = HashMap::new(); let tmp = Box::new(2); map.insert(43, &*tmp); ``` This will now fail to compile with a message that `*tmp` does not live long enough, because the scope of `tmp` is now strictly smaller than that of `map`, and the use of `&u8` in map's type requires that the borrowed references are all to data that live at least as long as the map. The usual fix for a case like this is to move the binding for `tmp` up above that of `map`; note that you can still leave the initialization in the original spot, like so: ```rust let tmp; let mut map : HashMap<u8, &u8> = HashMap::new(); tmp = box 2; map.insert(43, &*tmp); ``` Similarly, one can encounter an analogous situation with `Vec`: one would need to rewrite: ```rust let mut vec = Vec::new(); let tmp = 'c'; vec.push(&tmp); ``` as: ```rust let tmp; let mut vec = Vec::new(); tmp = 'c'; vec.push(&tmp); ``` ---- In some corner cases, it does not suffice to reorder the bindings; in particular, when the types for both bindings need to reflect exactly the *same* code extent, and a parent/child relationship between them does not work. In pnkfelix's experience this has arisen most often when mixing uses of cyclic data structures while also allowing a lifetime parameter `'a` to flow into a type parameter context where the type is *invariant* with respect to the type parameter. An important instance of this is `arena::TypedArena<T>`, which is invariant with respect to `T`. (The reason that variance is relevant is this: *if* `TypedArena` were covariant with respect to its type parameter, then we could assign it the longer lifetime when it is initialized, and then convert it to a subtype (via covariance) with a shorter lifetime when necessary. But `TypedArena` is invariant with respect to its type parameter, and thus if `S` is a subtype of `T` (in particular, if `S` has a lifetime parameter that is shorter than that of `T`), then a `TypedArena<S>` is unrelated to `TypedArena<T>`.) Concretely, consider code like this: ```rust struct Node<'a> { sibling: Option<&'a Node<'a>> } struct Context<'a> { // because of this field, `Context<'a>` is invariant with respect to `'a`. arena: &'a TypedArena<Node<'a>>, ... } fn new_ctxt<'a>(arena: &'a TypedArena<Node<'a>>) -> Context<'a> { ... } fn use_ctxt<'a>(fcx: &'a Context<'a>) { ... } let arena = TypedArena::new(); let ctxt = new_ctxt(&arena); use_ctxt(&ctxt); ``` In these situations, if you try to introduce two bindings via two distinct `let` statements, each is (with this commit) assigned a distinct extent, and the region inference system cannot find a single region to assign to the lifetime `'a` that works for both of the bindings. So you get an error that `ctxt` does not live long enough; but moving its binding up above that of `arena` just shifts the error so now the compiler complains that `arena` does not live long enough. * SO: What to do? The easiest fix in this case is to ensure that the two bindings *do* get assigned the same static extent, by stuffing both bindings into the same let statement, like so: ```rust let (arena, ctxt): (TypedArena, Context); arena = TypedArena::new(); ctxt = new_ctxt(&arena); use_ctxt(&ctxt); ``` ---- Due to the new code restrictions outlined above, this is a ... [breaking-change]
2 parents e365e4c + d6bf04a commit d77f6d5

File tree

20 files changed

+404
-128
lines changed

20 files changed

+404
-128
lines changed

src/librustc/metadata/tydecode.rs

+8
Original file line numberDiff line numberDiff line change
@@ -377,6 +377,14 @@ fn parse_scope(st: &mut PState) -> region::CodeExtent {
377377
let node_id = parse_uint(st) as ast::NodeId;
378378
region::CodeExtent::Misc(node_id)
379379
}
380+
'B' => {
381+
let node_id = parse_uint(st) as ast::NodeId;
382+
let first_stmt_index = parse_uint(st);
383+
let block_remainder = region::BlockRemainder {
384+
block: node_id, first_statement_index: first_stmt_index,
385+
};
386+
region::CodeExtent::Remainder(block_remainder)
387+
}
380388
_ => panic!("parse_scope: bad input")
381389
}
382390
}

src/librustc/metadata/tyencode.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -276,7 +276,9 @@ pub fn enc_region(w: &mut SeekableMemWriter, cx: &ctxt, r: ty::Region) {
276276

277277
fn enc_scope(w: &mut SeekableMemWriter, _cx: &ctxt, scope: region::CodeExtent) {
278278
match scope {
279-
region::CodeExtent::Misc(node_id) => mywrite!(w, "M{}", node_id)
279+
region::CodeExtent::Misc(node_id) => mywrite!(w, "M{}", node_id),
280+
region::CodeExtent::Remainder(region::BlockRemainder {
281+
block: b, first_statement_index: i }) => mywrite!(w, "B{}{}", b, i),
280282
}
281283
}
282284

src/librustc/middle/infer/error_reporting.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -809,6 +809,8 @@ impl<'a, 'tcx> ErrorReporting<'tcx> for InferCtxt<'a, 'tcx> {
809809
let scope_id = same_regions[0].scope_id;
810810
let parent = self.tcx.map.get_parent(scope_id);
811811
let parent_node = self.tcx.map.find(parent);
812+
let taken = lifetimes_in_scope(self.tcx, scope_id);
813+
let life_giver = LifeGiver::with_taken(&taken[]);
812814
let node_inner = match parent_node {
813815
Some(ref node) => match *node {
814816
ast_map::NodeItem(ref item) => {
@@ -851,8 +853,6 @@ impl<'a, 'tcx> ErrorReporting<'tcx> for InferCtxt<'a, 'tcx> {
851853
};
852854
let (fn_decl, generics, unsafety, ident, expl_self, span)
853855
= node_inner.expect("expect item fn");
854-
let taken = lifetimes_in_scope(self.tcx, scope_id);
855-
let life_giver = LifeGiver::with_taken(&taken[]);
856856
let rebuilder = Rebuilder::new(self.tcx, fn_decl, expl_self,
857857
generics, same_regions, &life_giver);
858858
let (fn_decl, expl_self, generics) = rebuilder.rebuild();

0 commit comments

Comments
 (0)