Skip to content

Conversation

@som-snytt
Copy link
Contributor

@som-snytt som-snytt commented Jul 17, 2025

Fixes #23541

When supplying a default arg for an implicit application, warn if there is an implicit value available.

That is for f(using x = v) where a default is supplied in f(using x = v, y = default). The user might think the implicit value is supplied, which is not the case for explicit using; this is especially confusing in a recursive call, where the user intends to "replace" an implicit parameter.

Separately, behind a flag, warn if any recursive call uses a default arg instead of passing the current parameter value.

-Wrecurse-with-default, "Warn when a method calls itself with a default argument."

@som-snytt
Copy link
Contributor Author

I didn't think to bootstrap; there are many instances, including the classic jsig!

[warn] 250 |            jsig(defn.ObjectType)
[warn]     |            ^^^^^^^^^^^^^^^^^^^^^
[warn]     |Argument for parameter toplevel was supplied using a default argument.

@som-snytt
Copy link
Contributor Author

Could offer a quickfix that inlines the default.

@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from 4bb349c to 9e53b78 Compare July 18, 2025 04:06
@odersky
Copy link
Contributor

odersky commented Jul 18, 2025

I think this warning would be too noisy. Implicit defaults are there for a reason, we do not want to warn each time we use one.

Also the change set looks too big and risky for such a relatively minor improvment.

@bracevac
Copy link
Contributor

It would be interesting to see how frequently this warning pops up in the open community build to have an understanding of how frequently these situations occur.

@som-snytt
Copy link
Contributor Author

som-snytt commented Jul 18, 2025

The recursion case is the Scala 2 lint. I find it very confusing for the reason cited on the ticket: was this intentional or an error? The compiler can't help you decide, but you feel better about it if someone had to type in p = true. The expectation is that such "config" is passed down the stack.

For the using case, it tries to warn only if a default was used when an implicit would have been supplied from scope; not sure I did that the best way, but how often is that actually intended?

In fact, it might be cleaner to refuse to supply defaults in those cases. However, I haven't spent time thinking abstractly about it.

Noisy warnings are usually behind a flag; I wasn't sure if the compiler team wants to go that route, since flags (even for warnings) are also a maintenance burden and a challenge for cross-builds (especially under -Werror).

@som-snytt
Copy link
Contributor Author

This was the highlight of the effort.
image

Probably that is an ad for the mechanism described in the ticket to explicitly supply remaining implicits.

@som-snytt
Copy link
Contributor Author

Not sure about indirect recursion (where Scala 2 warns for the same reason)

class C {
  def f(x: Int, s: String = ""): List[Int] = {
    List(new C).flatMap(_.f(42))
  }
}

For fun, I'll try doing the check at tailrec or tailcalls whatever.

@som-snytt
Copy link
Contributor Author

Similar to the Scala 2 lint for "named boolean parameters" where you must name them if necessary to avoid ambiguity or accidentally switching args, one boolean parameter with a default might by OK (where there is one use site which requires the alternate behavior or state) but more than one is hard to follow, especially when recursing.

But that sounds like a third-party lint. (The named boolean lint was added as a project-internal lint on Scala 2.)

Implicit primitives add extra eye blinking.

@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from 9e53b78 to f038825 Compare July 18, 2025 16:20
@som-snytt
Copy link
Contributor Author

Put the recursion under a flag, but leave the titular warning.

I didn't retry the scaladoc test, which probably requires the "special syntax" for a satisfying fix.

@som-snytt
Copy link
Contributor Author

As suspected, the scaladoc case is noticed, appropriately.

[E219] Type Warning: /__w/scala3/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:272:66

I will nowarn it for now, rather than munging the code.

@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from f038825 to c7045ea Compare July 18, 2025 17:03
@odersky
Copy link
Contributor

odersky commented Jul 18, 2025

The recursion case is the Scala 2 lint. I find it very confusing for the reason cited on the ticket: was this intentional or an error? The compiler can't help you decide, but you feel better about it if someone had to type in p = true. The expectation is that such "config" is passed down the stack.

I overlooked that. Yes, this is indeed confusing and should be avoided.

@som-snytt som-snytt marked this pull request as ready for review July 18, 2025 22:51
@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from c7045ea to b144877 Compare July 20, 2025 02:07
@Gedochao Gedochao requested a review from odersky July 21, 2025 10:39
Copy link
Contributor

@odersky odersky left a comment

Choose a reason for hiding this comment

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

It's on the right track, but i would try to do without the UsedDefault attachment.

override protected def explain(using Context): String =
"Usually it's intended to prefer the given in scope, but you must specify it explicitly."

final class TailrecUsedDefault(using Context) extends TypeMsg(TailrecUsedDefaultID):
Copy link
Contributor

Choose a reason for hiding this comment

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

Since it's not restricted to tailrec calls, the class should be renamed, maybe to RecursiveUsesDefault?

import tpd.*

/** Attachment indicating that an argument in an application was a default arg. */
val UsedDefaults = Property.StickyKey[Unit]
Copy link
Contributor

Choose a reason for hiding this comment

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

That will not work in from tasty compilation, since attachments are not pickled.


if (isRecursiveCall)
if isRecursiveCall then
if ctx.settings.Whas.recurseWithDefault && tree.args.exists(_.hasAttachment(UsedDefaults)) then
Copy link
Contributor

Choose a reason for hiding this comment

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

Consider finding a more local way to figure this out, one that does not span most of the compiler with a time-travelling attachment. Maybe look at spans? Or else check whether the argument's symbol is a default getter.


if !defaultArg.isEmpty then
if methodType.isImplicitMethod && ctx.mode.is(Mode.ImplicitsEnabled)
&& inferImplicitArg(formal, appPos.span).tpe.isError == false
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
&& inferImplicitArg(formal, appPos.span).tpe.isError == false
&& !inferImplicitArg(formal, appPos.span).tpe.isError

Copy link
Contributor Author

Choose a reason for hiding this comment

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

That was intentional, since the operator is so far away from the word it negates. I could introduce a var for such a case.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

It also reads as "did not infer an implicit arg"...

@odersky odersky assigned som-snytt and unassigned odersky Jul 23, 2025
@som-snytt
Copy link
Contributor Author

Reminder that this is especially brittle, even if it warns about unused nowarn. By the time my PR is merged, I don't know if the error IDs will have migrated. It needs a special comment to SYNC ERROR ID.

   // TODO #23 add support for all types signatures that make sense
+  @nowarn("id=E219")
   private def inner(

@som-snytt
Copy link
Contributor Author

som-snytt commented Jul 23, 2025

Only 13 warnings for compiler under -Wrecurse-with-default, which is narrower than warning when calling any enclosing method, but also feels like under-reporting to me. Choosing here to under-report ("false negative" or "less annoying").

The warnings for scaladoc:

[warn] -- [E219] Type Warning: /home/amarki/projects/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:273:66
[warn] 273 |                val sig = inParens(inner(qual, skipThisTypePrefix)(using skipTypeSuffix = true), shouldWrapInParens(qual, tp, true))
[warn]     |                                   ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
[warn]     |Argument for implicit parameter indent was supplied using a default argument.
[warn]     |
[warn]     | longer explanation available when compiling with `-explain`
[warn] -- [E219] Type Warning: /home/amarki/projects/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:280:43
[warn] 280 |            inner(qual, skipThisTypePrefix)(using skipTypeSuffix = true) ++ plain(".").l ++ suffix
[warn]     |            ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
[warn]     |Argument for implicit parameter indent was supplied using a default argument.
[warn]     |
[warn]     | longer explanation available when compiling with `-explain`
[warn] -- [E219] Type Warning: /home/amarki/projects/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:289:50
[warn] 289 |          case tp => inner(tp, skipThisTypePrefix)(using skipTypeSuffix = true) ++ plain(".").l
[warn]     |                     ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
[warn]     |Argument for implicit parameter indent was supplied using a default argument.
[warn]     |
[warn]     | longer explanation available when compiling with `-explain`
[warn] -- [E219] Type Warning: /home/amarki/projects/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:308:132
[warn] 308 |            keyword(caseSpaces + "case ").l ++ inner(from, skipThisTypePrefix) ++ keyword(" => ").l ++ inner(to, skipThisTypePrefix)(using indent = indent + 2) ++ plain("\n").l
[warn]     |                                                                                                       ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
[warn]     |Argument for implicit parameter skipTypeSuffix was supplied using a default argument.
[warn]     |
[warn]     | longer explanation available when compiling with `-explain`
[warn] -- [E219] Type Warning: /home/amarki/projects/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:310:132
[warn] 310 |            keyword(caseSpaces + "case ").l ++ inner(from, skipThisTypePrefix) ++ keyword(" => ").l ++ inner(to, skipThisTypePrefix)(using indent = indent + 2) ++ plain("\n").l
[warn]     |                                                                                                       ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
[warn]     |Argument for implicit parameter skipTypeSuffix was supplied using a default argument.
[warn]     |
[warn]     | longer explanation available when compiling with `-explain`
[warn] -- [E220] Type Warning: /home/amarki/projects/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:166:47
[warn] 166 |            val parsedMethod = parseRefinedElem(name, t.resType)
[warn]     |                               ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
[warn]     |                               Recursive call used a default argument.
[warn]     |
[warn]     | longer explanation available when compiling with `-explain`
[warn] -- [E220] Type Warning: /home/amarki/projects/scala3/scaladoc/src/dotty/tools/scaladoc/tasty/TypesSupport.scala:169:35
[warn] 169 |            } else parseRefinedElem(name, t.resType)
[warn]     |                   ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
[warn]     |                   Recursive call used a default argument.
[warn]     |
[warn]     | longer explanation available when compiling with `-explain`
[error] No warnings can be incurred under -Werror (or -Xfatal-warnings)
[warn] 7 warnings found

I will fix the E219s at least, where I think "use current implicit" was intended.

@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from b144877 to 27588b0 Compare July 23, 2025 21:11
@som-snytt
Copy link
Contributor Author

som-snytt commented Jul 23, 2025

Much improved by following the review advice.

Rebased and squashed. The scaladoc fix is a separate commit in case my guesses were wrong.

I see it doesn't report the parameter name: Recursive call used a default argument.

@som-snytt som-snytt requested a review from odersky July 24, 2025 15:09
@Gedochao Gedochao assigned odersky and unassigned som-snytt Jul 31, 2025
@Gedochao
Copy link
Contributor

@som-snytt seems a rebase would be necessary.

@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from f9832df to 631a471 Compare July 31, 2025 12:04
@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from 631a471 to abdd6c9 Compare July 31, 2025 16:25
@som-snytt
Copy link
Contributor Author

Don't forget to update the check file.

@som-snytt som-snytt force-pushed the issue/23541-warn-tricky-defaults branch from abdd6c9 to 8ba7cd3 Compare August 1, 2025 15:50
@som-snytt
Copy link
Contributor Author

Where to go for a code review.

image

Copy link
Contributor

@odersky odersky left a comment

Choose a reason for hiding this comment

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

Looks all good to me now. Sorry for the lagging review. I was on vacation last week.

@odersky odersky merged commit 4d98df9 into scala:main Aug 6, 2025
45 checks passed
@som-snytt som-snytt deleted the issue/23541-warn-tricky-defaults branch August 6, 2025 19:33
@som-snytt
Copy link
Contributor Author

@odersky since I'm Swiss on my father's side, I'm glad you took the holiday. Thanks again for the review. It's always a challenge as an outside contributor to make a PR that pays its way or pulls its weight in LOC and review time. "Martin's Rubbish Removal" is just a location I noticed in upstate New York; but if the Scala store had a t-shirt with that logo, I would wear it.

@Gedochao Gedochao added the release-notes Should be mentioned in the release notes label Aug 7, 2025
@WojciechMazur WojciechMazur added the backport:nominated If we agree to backport this PR, replace this tag with "backport:accepted", otherwise delete it. label Aug 7, 2025
WojciechMazur added a commit that referenced this pull request Aug 12, 2025
Backports #23559 to the 3.7.3-RC2.

PR submitted by the release tooling.
@WojciechMazur WojciechMazur added backport:done This PR was successfully backported. and removed backport:nominated If we agree to backport this PR, replace this tag with "backport:accepted", otherwise delete it. labels Aug 12, 2025
@WojciechMazur WojciechMazur added this to the 3.7.3 milestone Aug 12, 2025
@som-snytt
Copy link
Contributor Author

The linked issue
#23842
is an example in the wild. The use site was test code, which may be why the behavior was not noticed.

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. release-notes Should be mentioned in the release notes

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Add a lint/warning for the Interaction between using and default parameters

5 participants