-
Notifications
You must be signed in to change notification settings - Fork 14
New issue
Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? # to your account
Generating better JVM bytecode for debuggers #811
Comments
Match expressions on types don't generate enough debugging informationLet's examine the following Scala code. import java.util.concurrent.atomic.AtomicInteger
object main {
val counter: AtomicInteger = new AtomicInteger(0)
def str(): Any = {
counter.incrementAndGet()
"abc"
}
def main(args: Array[String]): Unit = {
str() match {
case s: String =>
println(s) // Breakpoint here.
case n: Int =>
println(n)
case _ =>
}
}
} The important things to note are the side-effectful Let's suppose that we have a breakpoint on line 14 ( Invoking the debugger on this code snippet will stop on the provided breakpoint, as expected. Both the IntelliJ and the Metals debugger show only A much better user experience would be to also show the The difference comes from the emitted bytecode (which we will examine a bit later in much more detail) between Scala 2 and Scala 3. How it affects expression evaluationUnfortunately, showing bound variables in this context is only one problem. Expression evaluation is also negatively affected by this. Here's where the side-effectful nature of In theory, all of the local variables of a method can be (and are) queried according to their index, and not their name. This could be exploited to examine what local variables are written to right after the Unfortunately, I have found examples where the debugger is unable to read unnamed local variables and throwing cryptic exceptions, with varying behavior in situations like reordered Coming back to the expression evaluation example. Because the debugger is unable to find the value of Examining Scala 2 and Scala 3 bytecodeScala 2 bytecode
The important part is the Scala 3 bytecode
Examining the Discussion about the reportI hope that I've been able to present a clear improvement with this report. I expect the other instances that I will file to follow a similar style. Please do not hesitate to ask for more clarifications and suggest further improvements. Thanks for reading. |
I'll repeat one thing I said on Discord, namely:
|
I'm wondering if this overlaps at all with scala/scala#9121 ? (which seems abandoned now for Scala 2, though similar work is going forward for Scala 3) |
As far as I can tell, that PR (and the one in Scala 3) mainly deal with enabling better inspections in inlined code, which is a whole can of worms on its own. 😆 |
@vasilmkd-jetbrains having bytecode for matches that the debugger understands better would be excellent. I'm happy to assist where I can. |
@lrytz Please assume that I have no knowledge of the bytecode generation backend. Where would I start with something like the above? Thanks. |
Without looking in detail - is this possibly "only" about the local variables table? There's a method |
Thanks, that's still a helpful pointer. |
To my knowledge, we don't have learning resources oriented towards The absolute best way to learn is to ask very specific questions on issues and PRs, especially your own PRs. Half-finished PRs and not-even-half-finished PRs in the scala/scala repo are totally fine to open and then ask questions on, so we can help you get unstuck and take the next steps. I haven't done much work on the back end itself (Lukas is by far the most expert on that), but I have built an alternate back end, so I know a lot about the tree shapes that come in to the back end. There is other knowledge spread around among Scala 2 team & contributors and the Scala 3 ones too. One thing that perhaps you already know, but I'll state explicitly just in case, is that the Scala 2 and Scala 3 back ends are basically the same code. They've drifted a bit out of sync with each other, as improvements haven't always been forward-ported or back-ported. But they're not drastically different. Moving improvements (and knowledge) back and forth between the two is often pretty straightforward. (Though for pattern matching in particular, the Scala 2 tree shapes and the Scala 3 tree shapes are rather different...) |
So, I took a look at the backend. It seems that if (!isSynth) { // there are case <synthetic> ValDef's emitted by patmat
varsInScope ::= (sym -> localVarStart)
} I compared this code to the Dotty backend, and it seems to be the same. This hints to me that Scala 2 is marking patmat vals as |
|
Should I ask the Scala 3 team as well, for their thoughts on the matter? |
Um, not sure. It's hard to be sure who you would ask exactly, as you're asking about the absence of a flag. You might look through the git history in their repo and see if you turn up any leads (though again, it's hard to gather evidence about an absence). Perhaps you could ask them a meta-question about whether someone in particular is knowledgeable in this area. (It's often tempting to say "ask Guillaume" since he often seems to know everything, but of course always asking Guillaume doesn't scale...) |
My idea was more or less to have a high level discussion about whether patmat variables should be synthetic or not. My claim is that there is merit to them being not synthetic and visible. It would also shine a light on whether Scala 3's behavior is a bug or a feature. 😄 |
It seems reasonable that patmat variables are not synthetic, since they are just like other variables. @vasilmkd-jetbrains I encourage you to make them not synthetic in a PR. Then, I am not sure exactly how we could assess the impact of such a change. Do you guys think there could be some tooling relying on patmat variables being flagged as synthetic…? |
Exactly, I'm also a bit vary of such a change having a larger impact than wanted or anticipated. It's also why I would still like to invite someone from the Scala 3 team to discuss it. |
I'm kind of responsible for the pattern matcher in Scala 3, although I did not write its first iteration. I can't tell how the history of synthetic-ness for patmat variables happened. That said, I'm pretty sure that any definition visible in the source code should not be synthetic. So I'd be in favor of making patmat variables non-synthetic. |
Thank you @sjrd for chiming in! |
I just took a closer look at scala/scala@f6ee85b and I no longer think it's relevant — it looks to apply only to synthetic temporaries. Sorry for the misdirection.
Agree. |
I asked the following question on the Scala Discord and I was invited by @SethTisue to open a discussion here, in hopes of getting more eyes on the issue.
I'm including the text from the Discord message verbatim:
I would like to use this tracking issue to present current pain points with the Scala 2 debugging experience that I believe are feasible to solve on the compiler side. I will post those examples as follow up comments to this issue.
I haven't worked on the compiler backend to date, but I'm willing to learn and help in the process, as well as provide feedback and testing of the implementations. Thanks in advance.
The text was updated successfully, but these errors were encountered: