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

Fix coverage instrumentation of Java-defined and parameterless methods #15648

Merged
merged 4 commits into from
Jul 25, 2022

Conversation

TheElectronWill
Copy link
Contributor

@TheElectronWill TheElectronWill commented Jul 11, 2022

Note: dear reviewer, don't worry! Most of the changes are "expect" files for the tests. The actual changes in the compiler codebase are small :)

#13880 introduced code instrumentation for coverage analysis with the compiler option -coverage-out. #15530 fixed a few corner cases in the instrumentation, but 3 bugs remained:

  1. Select and TypeApply trees weren't properly handled for Java-defined symbols, leading to a crash when selecting a method with parameter types. See java-methods/test.scala. I found this bug while trying to run coverage on ScalaPB's generated code.
  2. Select and Ident trees weren't properly handled, and many occurences of parameterless methods were missed. For example:
def f = ()
f // was not instrumented, and thus the coverage analysis was wrong

def g() = ()
g() // was instrumented properly, and still is

The right way to detect parameterless methods seems to be

sym.is(Method) && sym.info.isParameterless

Most of the changes in the expect files are caused by this fix.

  1. Some methods like asInstanceOf and isInstanceOf must not be instrumented, otherwise it crashes in Erasure. That is, we should never generate this:
{
  Invoker.invoked(id, dir)
  x.asInstanceOf
}[T]

I've also ignored the calls to methods on primitive types, like Int.+, because it doesn't exist at runtime and doesn't seem interesting to instrument. Note that in a + f(), the call to f will be instrumented properly. Also, if a is a parameterless method, the call to a will be instrumented.

For example:

1 + f()

previously resulted in:

{
  invoked(...) // covered method: "+" of Int
  {
    val f$1 = {
      invoked(...) // covered method: "f"
      f()
    }
    1 + f$1
  }
}

and now gives:

{
  val f$1 = {
    invoked(...) // covered method: "f"
    f()
  }
1 + f$1
}

@TheElectronWill TheElectronWill requested a review from ckipp01 July 11, 2022 16:20
@TheElectronWill
Copy link
Contributor Author

Like #15530, I'll backport this to release-3.0 once it's good.

@ckipp01
Copy link
Member

ckipp01 commented Jul 11, 2022

Hey @TheElectronWill, just to be transparent, I know you tagged me on this but I'm no longer really working on any coverage related stuff apart from just cutting releases of scoverage, so probably best to tag someone else for a more timely review.

@ckipp01 ckipp01 removed their request for review July 11, 2022 19:15
@TheElectronWill
Copy link
Contributor Author

TheElectronWill commented Jul 11, 2022

Hey @TheElectronWill, just to be transparent, I know you tagged me on this but I'm no longer really working on any coverage related stuff apart from just cutting releases of scoverage, so probably best to tag someone else for a more timely review.

No problem! Thanks for your transparence :)

Tentatively assigning my homonym @smarter, who is suggested by GitHub (feel free to assign someone else if you don't have the time to review this, I know you've got things to write ^^).

@TheElectronWill TheElectronWill requested a review from smarter July 11, 2022 20:12
@smarter
Copy link
Member

smarter commented Jul 11, 2022

Tentatively assigning my homonym @smarter, who is suggested by GitHub (feel free to assign someone else if you don't have the time to review this, I know you've got things to write ^^).

Yeah I wish I could but as I'm handing in my thesis in two weeks I definitely don't have the time to look at github, so passing the buck to @dwijnand 😅

@smarter smarter requested review from dwijnand and removed request for smarter July 11, 2022 20:19
@TheElectronWill
Copy link
Contributor Author

Best of luck for your thesis!

@TheElectronWill TheElectronWill added the backport:nominated If we agree to backport this PR, replace this tag with "backport:accepted", otherwise delete it. label Jul 12, 2022
1. Select and TypeApply trees weren't properly handled for JavaDefined symbols, leading to a crash when selecting a static method with parameter types.
2. Select and Ident trees weren't properly handled, and many occurences of parameterless methods were missed.
3. Some methods like asInstanceOf and isInstanceOf must not be instrumented, otherwise it crashes in Erasure.
@griggt
Copy link
Contributor

griggt commented Jul 19, 2022

While investigating a compiler crash using coverage (separate issue, ticket may be forthcoming), I observed what seems to be a regression with this PR.

The following snippet now emits an error:

import scala.compiletime.uninitialized
class Foo:
  var x: AnyRef = uninitialized
> scalac -coverage-out:. /tmp/coverage/uninitialized.scala

-- Error: /tmp/coverage/uninitialized.scala:3:18 -------------------------------
3 |  var x: AnyRef = uninitialized
  |                  ^^^^^^^^^^^^^
  |`uninitialized` can only be used as the right hand side of a mutable field definition

It compiles without error using -coverage-out on 3.2.0-RC2 and 3.2.1-RC1-bin-20220717-5c43324-NIGHTLY.

@TheElectronWill
Copy link
Contributor Author

TheElectronWill commented Jul 19, 2022

Ah yes, that's because it gets instrumented like this:

var x = {
  invoked(...)
  uninitialized
}

I'll add it to the blacklist.
Thank you for reporting this!

EDIT: done ✔️

@smarter
Copy link
Member

smarter commented Jul 23, 2022

@sjrd do you by any chance have some time to take a look at this PR?

@sjrd sjrd self-assigned this Jul 23, 2022
@sjrd
Copy link
Member

sjrd commented Jul 23, 2022

I'll find some time in the coming week.

@dwijnand dwijnand requested review from sjrd and removed request for dwijnand July 25, 2022 08:35
Copy link
Member

@sjrd sjrd left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The changes to the compiler code LGTM. I have not checked the test changes, as I don't know what I'm looking at. ;)

@TheElectronWill
Copy link
Contributor Author

TheElectronWill commented Jul 25, 2022

Auto-generated boring coverage data, essentially 😄
When looking at such data, I check that the line numbers are globally correct, and that all the methods that are called in the source code appear in the generated file.

Thanks for the review!

@TheElectronWill TheElectronWill merged commit 19eff87 into scala:main Jul 25, 2022
@TheElectronWill TheElectronWill added backport:accepted This PR needs to be backported, once it's been backported replace this tag by "backport:done" and removed backport:nominated If we agree to backport this PR, replace this tag with "backport:accepted", otherwise delete it. labels Jul 25, 2022
@Kordyjan Kordyjan added backport:done This PR was successfully backported. and removed backport:accepted This PR needs to be backported, once it's been backported replace this tag by "backport:done" labels Jul 27, 2022
@Kordyjan Kordyjan added this to the 3.2.1 milestone Aug 1, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
backport:done This PR was successfully backported.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

7 participants