-
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 always initializes variables with them in (debuginfo) scope. #32949
Comments
This might be interesting here:
How would I best be able to take a look at what the scoping structure of a function looks like (i.e. what expression is assigned to what scope)? |
@michaelwoerister MIR scopes? |
@eddyb Thanks! I'll take a look. |
Oops, wrong tab... |
@eddyb This looks rather suspicious to me: rust/src/librustc_mir/build/block.rs Line 55 in 8b09372
At least from a visibility point of view, the init-scope should not be nested within the remainder-scope. |
@michaelwoerister that nesting is necessary for MIR borrowck AFAICT - if a variable is not in scope, you can't actually initialize it, since it doesn't exist. |
Hm, then I'm inclined to suggest that we do not re-use these scopes for debuginfo and instead build an additional visibility scope tree, as not to conflate two things that are not really the same. |
How terrible would it be to have two scopes, one starting before the |
So, the this code... let x = 3 + 4;
<subsequent statements> ... would map to the following? { // "lifetime scope"
let x;
{ // "initializer scope"
tmp1 = 3 + 4;
x = tmp1;
}
{ // scope within which x is visible, i.e. "remainder scope" of the let binding
<subsequent statements>
}
} That would be worth a try, I think. If it turns out not to cause to many problems with "accidental" interactions with borrow-checking, it would save us the work of maintaining two scope trees. |
@michaelwoerister Yeah, that's what I meant. I'm pretty sure it can work, although I'd like someone who actually is going to work on MIR borrowck look at it. @nikomatsakis? |
Get all (but one) of debuginfo tests to pass with MIR codegen. I didn't get much feedback in #31005 so I went ahead and implemented something simple. Closes #31005, as MIR debuginfo should work now for most usecases. The `no-debug-attribute` test no longer assumes variables are in scope of `return`. We might also want to revisit that in #32949, but the test is more reliable now either way. In order to get one last function in the `associated-type` test pass, this PR also fixes #32790.
@eddyb having an extra scope (the innermost enclosing block) seems... fine. I'm still somewhat unconvinced that we want those variables to be visible before you have stepped to the |
@nikomatsakis Why would they be visible before? I'm only suggesting changing the scope we use in debuginfo from starting "just before the |
@eddyb oh, hmm, I was not understanding what you meant. The issue is that today the "suffix" scopes include the initializer, and you would rather that they do not? In any case, having an extra scope seems fine. |
@nikomatsakis Correct, that would result in the expected behavior in a debugger. |
@eddyb I think borrowck does want the lifetime to include the initializer, in order to allow things like |
So it appears that the whole "definition scope" isn't used right now, but rather "remainder scope", which is the visibility scope, and the problem, as @michaelwoerister pointed out, is that the initializer scope is nested in it at the MIR level (its parent is the remainder scope), even though they're not lexically nested. |
[MIR] Make scopes debuginfo-specific (visibility scopes). Fixes #32949 by having MIR (visibility) scopes mimic the lexical structure. Unlike #33235, this PR also removes all scopes without variable bindings. Printing of scopes also changed, e.g. for: ```rust fn foo(x: i32, y: i32) { let a = 0; let b = 0; let c = 0; } ``` Before my changes: ```rust fn foo(arg0: i32, arg1: i32) -> () { let var0: i32; // "x" in scope 1 at <anon>:1:8: 1:9 let var1: i32; // "y" in scope 1 at <anon>:1:16: 1:17 let var2: i32; // "a" in scope 3 at <anon>:1:30: 1:31 let var3: i32; // "b" in scope 6 at <anon>:1:41: 1:42 let var4: i32; // "c" in scope 9 at <anon>:1:52: 1:53 ... scope tree: 0 1 2 3 { 4 5 6 { 7 8 9 10 11 } } } ``` After my changes: ```rust fn foo(arg0: i32, arg1: i32) -> () { scope 1 { let var0: i32; // "x" in scope 1 at <anon>:1:8: 1:9 let var1: i32; // "y" in scope 1 at <anon>:1:16: 1:17 scope 2 { let var2: i32; // "a" in scope 2 at <anon>:1:30: 1:31 scope 3 { let var3: i32; // "b" in scope 3 at <anon>:1:41: 1:42 scope 4 { let var4: i32; // "c" in scope 4 at <anon>:1:52: 1:53 } } } } ... }
That is, given
let x = expr;
, MIR will produce the same code aslet x; x = expr;
(modulo borrows).When breaking inside
expr
in a debugger,x
will be in scope, uninitialized.It can be really annoying when dealing with shadowing or arguments.
For example, if you break on the call in
let x = ...; let x = f(x);
and try to printx
, you will get the secondx
, which is uninitialized at that point.cc @michaelwoerister
The text was updated successfully, but these errors were encountered: