-
Notifications
You must be signed in to change notification settings - Fork 13k
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
Use 1-based indexing for columns in debuginfo #65676
Conversation
Thanks for working on this!
I don't really agree because, as the issue name and description says, it was about off-by-one column in DWARF and it shown both cases where it happens. This fixes one of the cases, but breaks the other one instead (ending column was correct before, but instead becomes off-by-one now). Introducing regression while fixing a bug is far from ideal, which is why I didn't make a PR yet and wrote on the issue that this needs further investigation.
Note that this is not fully correct:
I'd suggest holding off this PR for now and fixing the column for closing braces / return statement either as part of it or in a separate one before merging this one to avoid introducing new potential issues. |
This would be true if it was line 14, but the scope begins on line 15. Otherwise I agree with the facts above. |
Oh, you're right. Looks like original actually did the same - can you try your example without the patch? If it's still the same, then there is probably another place where |
Huh, that's a good point. I think there might be another place where we need to add +1 as well. The code this PR changes only seems to be involved in |
I think you'd need to change a column here for the scope to work.
|
@est31 Oh, race condition, haha :) |
Indeed that's my view. |
5c1876d
to
6ae5f70
Compare
|
||
// CHECK-LABEL: DISubprogram(name: "main" | ||
// CHECK: {{.*}}DILocalVariable(name: "x",{{.*}}line: 15{{.*}} | ||
// CHECK-NEXT: {{.*}}DILexicalBlock{{.*}} |
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.
Maybe turn this line into a test for the DILexicalBlock
change?
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'm not that fast okay 😄
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 was caught between commenting too early and risking that you already want to do it or commenting too late at which point you might already be doing something else. :).
My counter-argument is that it's not so much the count that matters, as severity. Previously columns were off, but at least both the [accidentally] correct ones (for return statements) and the incorrect ones (for everything else) stayed within bounds of the line, whereas now columns can point past the end of the line or even the file, which can create more serious issues for editors or other debugging tooling that doesn't expect this sort of overflow issues. |
I'd be a bit surprised if DWARF consumers borked on one-past-the-end column numbers, but I've seen (and written) dumber things. If we were to commit to never emitting such a location in DWARF, where should the span for the return statement inside the closure point to in the following? The fn func() -> fn() -> i32 {
|| 42
} |
Is there no way we can detect and change column at the LLVM debug location level, rather than MIR as you (@est31) described in the issue? In DWARF there is |
That's what it does right now (before the patch), which is admittedly a bit weird, but seems reasonable enough. |
The ret statement has nothing corresponding to it in the surface syntax of the language. So demanding that you find something corresponding is hard I guess :). Both the 2 as well as pointing towards the end of the line are bad choices here. I don't have an opinion onto which is better. |
After a sleep, I've decided to simply map zero-width spans to the preceding column (instead of the succeeding one). As a result, we should never emit columns pointing past the end of a line. My purpose in posting the example above was to illustrate a case where it would be desirable for a debuginfo column to point past the end of a line if that were supported by most DWARF consumers. I don't know if this is the case in practice, and no ruling is made in the DWARF spec AFAIK. @RReverser, perhaps you could share what tools you're using that make use of this information? It might be nice to test against them in the future, since LLVM's It would be ideal if spans pointed to the closing brace of a code block. This would give better errors in I think this should resolve everyone's concerns. |
I've edited the PR summary to reflect the new updates. |
f138930
to
af25bb2
Compare
Sounds reasonable to me!
I actually wonder if it could be as simple as what you're doing above, but in a proper place - that is, just modifying this span: rust/src/librustc_mir/build/mod.rs Line 634 in ac45719
that @est31 referenced earlier, to point to 1 character before? Then it should fix both DWARF and cases like error message you referenced, right? |
I tried that yesterday. The problem is that not all |
The BytePos(0) ones sound weird and should be excluded, but for others I think we still want to point to the last char, even if it's not a brace (like in example you provided above with |
@michaelwoerister Any chance of getting this reviewed? I'm confident that this is the best solution that doesn't require changing the attribution of spans while building the MIR (see #37310). It's pretty minimal, and it avoids creating columns that point past the end of a line. Although I've still not seen evidence that this causes a problem for DWARF tools in the wild, it seems prudent to avoid this for now. |
FWIW I agree - this PR with latest changes is resulting in correct locations for all mentioned cases, so as far as I'm concerned the issue is resolved. Thanks @ecstatic-morse! My suggestion on ways to clean this up further like above still applies, but refactoring MIR locations is generally a bigger task and can be certainly out of scope of this issue/PR. |
I put it on my todo list. |
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.
Thanks for the PR, @ecstatic-morse! It looks good to me in general (nice test!) but maybe we can clarify the comment in from_span
.
// x|yz = 1 | ||
// xy|z = 2 | ||
// | ||
// See discussion in /~https://github.com/rust-lang/rust/issues/65437 |
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 don't really understand what this comment means. Maybe we can try to make it clearer to the casual reader.
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.
How's this? Or is it the table specifically that's confusing?
// Rust likes to emit zero-width spans that point just after a closing brace to denote e.g.
// implicit return from a function. Instead of mapping zero-width spans to the column at
// `span.lo` like we do normally, map them to the column immediately before the span. This
// ensures that we point to a closing brace in the common case, and that debuginfo doesn't
// point past the end of a line. A zero-width span at the very start of the line gets
// mapped to `0`, which is used to represent "no column information" in DWARF. For example,
//
// Span len = 0 Span len = 1
//
// |xyz => 0 (None) |x|yz => 1
// x|yz => 1 x|y|z => 2
//
// See discussion in /~https://github.com/rust-lang/rust/issues/65437 for more info.
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.
Is there any documentation on when zero-width spans are emitted and for what reason? Or is this reverse-engineering the current (unspecified) behavior?
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 second one XD
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.
This is the code responsible (added in #37310):
rust/src/librustc_mir/build/mod.rs
Lines 633 to 634 in ac45719
// Attribute epilogue to function's closing brace | |
let fn_end = span.shrink_to_hi(); |
☔ The latest upstream changes (presumably #65718) made this pull request unmergeable. Please resolve the merge conflicts. |
I'll give this a closer look next week. |
534bbb3
to
f24bb99
Compare
Rebased to fix merge conflicts and squashed all commits that modified only the test. |
Alright, I've given this some more thought and my conclusions are:
The PR already adds a codegen test and I think it should extended to check for
Reading these tests is hard, so adding lots of explanatory comments is a good idea. |
I agree that more tests is better in the abstract, but I'm confused as to why this PR demands them? Currently, all column-info is off-by-one, with the exception of column-info for zero-width spans at the end of a function, which happen to be correct due to a second, offsetting off-by-one error (though it's not really an error I suppose, it's just a bit strange). All this PR does is fix the first off-by-one error, and mitigate the second. The worst thing that can happen as a result of this PR is we generate a bad column number (which we already do, everywhere). I can add a function with an explicit return statement and closures to the test, but that won't be testing anything in this PR, it will be testing the spans we generate. Those tests can be written without using LLVM's FileCheck, which is brittle and hard to read. |
The PR adds some non-intuitive logic via the from_span method. It is making assumptions about the spans that MIR construction generates and those assumptions should be tested, otherwise things can regress without anyone noticing before the regression lands.
My thinking is that we should test that the spans we generate early in the pipeline end up as the spans we expect in debuginfo. As far as I know we don't have tests for column numbers, so a PR that proposes to fix them should add regression tests.
I'm open to any method of testing! |
Very well. I don't expect to pick this up again anytime soon, so if anyone wants to take this across the finish line (or try a different approach entirely), I would be greatly appreciate it. |
@michaelwoerister what about a smple change that adds 1 unconditionally? That shouldn't be controversial and was my initial proposal for the first step. |
@ecstatic-morse I'm closing this based on the comment in #65676. Thanks for taking the time to contribute :) |
edited from original
Resolves #65437.
This fixes an off-by-one error when writing column numbers in DWARF, which uses 1-based indexing.
For the newly added test, the resulting debuginfo is:
You can see that the locations for
x
and32
are correct (columns 9 and 13 respectively), theDILexicalBlock
formain
begins at column 5 (the firstlet
statement) and the implicit return statement has column 1 (the closing}
).r? @michaelwoerister