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

MIR always initializes variables with them in (debuginfo) scope. #32949

Closed
eddyb opened this issue Apr 14, 2016 · 16 comments
Closed

MIR always initializes variables with them in (debuginfo) scope. #32949

eddyb opened this issue Apr 14, 2016 · 16 comments
Assignees
Labels
A-debuginfo Area: Debugging information in compiled programs (DWARF, PDB, etc.) A-MIR Area: Mid-level IR (MIR) - https://blog.rust-lang.org/2016/04/19/MIR.html

Comments

@eddyb
Copy link
Member

eddyb commented Apr 14, 2016

That is, given let x = expr;, MIR will produce the same code as let 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 print x, you will get the second x, which is uninitialized at that point.

cc @michaelwoerister

@eddyb eddyb added A-debuginfo Area: Debugging information in compiled programs (DWARF, PDB, etc.) A-MIR Area: Mid-level IR (MIR) - https://blog.rust-lang.org/2016/04/19/MIR.html labels Apr 14, 2016
@michaelwoerister
Copy link
Member

This might be interesting here:

// LLVM does not properly generate 'DW_AT_start_scope' fields

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)?

@eddyb
Copy link
Member Author

eddyb commented Apr 14, 2016

@michaelwoerister MIR scopes? rustc --unpretty=mir -Z unstable-options on master should print scope IDs for all variables, statements and terminators.

@michaelwoerister
Copy link
Member

@eddyb Thanks! I'll take a look.

@michaelwoerister
Copy link
Member

r? @michaelwoerister
I'll review this if nobody minds.

Oops, wrong tab...

@michaelwoerister michaelwoerister self-assigned this Apr 14, 2016
@michaelwoerister
Copy link
Member

@eddyb This looks rather suspicious to me:

let remainder_scope_id = this.push_scope(remainder_scope, block);

At least from a visibility point of view, the init-scope should not be nested within the remainder-scope.

@eddyb
Copy link
Member Author

eddyb commented Apr 14, 2016

@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.

@michaelwoerister
Copy link
Member

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.

@eddyb
Copy link
Member Author

eddyb commented Apr 16, 2016

How terrible would it be to have two scopes, one starting before the let and one after it?
The visibility problems would be fixed by having each VarDecl track two scopes, the one in which it's declared and the one in which it's initialized.

@michaelwoerister
Copy link
Member

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.

@eddyb
Copy link
Member Author

eddyb commented Apr 16, 2016

@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?

bors added a commit that referenced this issue Apr 17, 2016
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.
@nikomatsakis
Copy link
Contributor

@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 let in question though, but I guess that's a (somewhat) separate question.

@eddyb
Copy link
Member Author

eddyb commented Apr 19, 2016

@nikomatsakis Why would they be visible before? I'm only suggesting changing the scope we use in debuginfo from starting "just before the let" to " just after the let", i.e. not including the let RHS.

@nikomatsakis
Copy link
Contributor

@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.

@eddyb
Copy link
Member Author

eddyb commented Apr 20, 2016

@nikomatsakis Correct, that would result in the expected behavior in a debugger.
But borrowck has to know that the initializer is in the scope of the definition, doesn't it?
Can we actually get away with only recording the post-initializer "suffix"?
Because that's the trivial fix @michaelwoerister mentioned, for let, while match already works like that.

@nikomatsakis
Copy link
Contributor

@eddyb I think borrowck does want the lifetime to include the initializer, in order to allow things like let foo = &3

@eddyb
Copy link
Member Author

eddyb commented Apr 27, 2016

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.

bors added a commit that referenced this issue Jun 8, 2016
[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
                }
            }
        }
    }

    ...
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-debuginfo Area: Debugging information in compiled programs (DWARF, PDB, etc.) A-MIR Area: Mid-level IR (MIR) - https://blog.rust-lang.org/2016/04/19/MIR.html
Projects
None yet
Development

No branches or pull requests

3 participants