-
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
MIR-borrowck: locals (and thread locals) can remain borrowed after the function ends #45704
Comments
The same is true for function arguments (see fn cplusplus_mode(x: isize) -> &'static isize { &x }
fn cplusplus_mode_exceptionally_unsafe(x: &mut Option<&'static mut isize>) {
let z = 0;
*x = Some(&mut z);
panic!("catch me for a dangling pointer!")
} I'm having all 3 in a single issue because the fix (preventing outstanding borrows to locals at function exit) should catch all 3. |
@arielb1 hmm, what exact fix do you have in mind? With respect to local variables, I think the problem there is that we don't have a proper storage-dead on the unwinding path, no? That is, would it error if you removed the With respect to thread-locals, we don't have a I could imagine trying not to add storage-dead onto the unwind path, but it seems like that might go wrong, since you might have nested scopes that could cause trouble? (i.e., inner variables getting freed before outer ones?) |
We know that all locals have their storage killed at the end of the function, so we don't need to represent that explicitly in MIR. With lexical lifetimes, I suppose you could observe the ordering of Unless we are planning to deploy MIR borrowck without NLL, that shouldn't be a problem. |
@arielb1 so, to be clear, is your expectation that we will not add I'm trying to put a finger on what I think that might permit that we might not want to. Given that the checks on StorageDead are a subset of those that occur on DROP, it seems like very little, unless we are talking about structures without destructors -- but those are permitted to have cycles and data that outlives their destruction anyway. So it may be nothing. |
I think it will allow you to have "NLL-style" cycles if your function doesn't have a destructor. aka: use std::cell::Cell;
// no destructor
struct S<'a>(Cell<Option<&'a S<'a>>>);
fn main() {
#[cfg(break)] let _d = Box::new(0);
let x = S(Cell::new(None));
let y = S(Cell::new(None));
x.0.set(Some(&y));
y.0.set(Some(&x));
panic!()
} This currently creates a storagedead -> endregion -> storagedead sequence, which causes a "borrow does not live long enough" error, while either with NLL, or if we have a "mass storagedead at function exit", it should compile. However, with the "None terminator", if we add a destructor in scope, the code will start issuing a borrowck error and therefore be inconsistent. I think one good way to deal with it consistently, if we want a stable MIR borrowck without NLL, would be to introduce a bit of CFG-based analysis to MIR borrowck: if we find a sequence of storagedead and endregion instructions (possibly connected by gotos across basic blocks) end all regions first and then kill all the storage. |
I am more and more thinking we will not wind up wanting this -- the timing just doesn't seem to be working out that way. |
Mentoring instructionsWell, whatever we do, I think we all agree that it makes sense as a minimal starting point to have some code in borrow-check such that when you return (or unwind) from a function, we report an error if there are any outstanding loans. In other words, any of the "returning terminators" (listed below) are treated as if they kill the storage for all locals. The "returning terminators" are those that have no successor, with the exception of From the POV of borrow-check, we want to treat those basically the same as if they had done a rust/src/librustc_mir/borrow_check.rs Lines 291 to 296 in 8efbf7a
As you can see, it invokes When the borrow check encounters one of our returning terminators, it currently does nothing: rust/src/librustc_mir/borrow_check.rs Lines 373 to 380 in 8efbf7a
I think we want to separate out the returning terminators from that match, and then have them iterate over through all the locals in the MIR. The local variables in the MIR are stored in the for local in self.mir.local_decls.indices() {
// insert call to access_lvalue() we saw before in StorageDead here
} Once that's done, you'll have to edit the tests. Also, we'll want to handle thread-locals, but we'll come to that later. (Note to future self: thread-locals will need 'deep' style access checks.) |
You also want to end the borrows for all the local (aka let x = 2;
let y = &x;
panic!() |
Hmm, I left out something important from those instructions. We also want to modify the bit of code that figures out which borrows are in effect. We need to consider the "return" to cancel all the borrows whose duration are local to the function (an But I am now wondering if maybe it's better to iterate over the live borrows and examine them, rather than iterating over all the variables and trying to check if accessing them would be legal. |
Well, be forewarned, this may be one of those things where we try a few different stabs at it before we are happy. =) |
I'm taking a look at this. |
Kill the storage for all locals on returning terminators Fixes #45704.
e.g. this does not give a MIR borrowck error:
This is a problem because, as #17954 shows, thread locals end when the current thread ends. AST borrowck currently makes it an error when thread locals are borrowed for more than the current function.
The text was updated successfully, but these errors were encountered: