Skip to content

Commit f7dc85b

Browse files
committed
Add an -Yimplicit-as-given flag to easily test changes in the ecosystem.
It replaces every implicit definition with `given` (including classes, which I was surprised to see working), and every implicit parameter clause with `using`, all in the parser phase. Symbols and trees unpickled from tasty are unaffected by this flag. Trees generated using the reflection api are unaffected as well.
1 parent a893f7f commit f7dc85b

File tree

6 files changed

+44
-1
lines changed

6 files changed

+44
-1
lines changed

compiler/src/dotty/tools/dotc/config/ScalaSettings.scala

+1
Original file line numberDiff line numberDiff line change
@@ -458,6 +458,7 @@ private sealed trait YSettings:
458458

459459
val Yinstrument: Setting[Boolean] = BooleanSetting(ForkSetting, "Yinstrument", "Add instrumentation code that counts allocations and closure creations.")
460460
val YinstrumentDefs: Setting[Boolean] = BooleanSetting(ForkSetting, "Yinstrument-defs", "Add instrumentation code that counts method calls; needs -Yinstrument to be set, too.")
461+
val YimplicitAsGiven: Setting[Boolean] = BooleanSetting(ForkSetting, "Yimplicit-as-given", "Interpret the compiled implicit keywords as their scala-3 given counterparts.")
461462

462463
// Deprecated: lifted from -Y to -X
463464
@deprecated(message = "Lifted to -X, Scheduled for removal.", since = "3.5.0")

compiler/src/dotty/tools/dotc/parsing/Parsers.scala

+2-1
Original file line numberDiff line numberDiff line change
@@ -3283,6 +3283,7 @@ object Parsers {
32833283
private def modOfToken(tok: Int, name: Name): Mod = tok match {
32843284
case ABSTRACT => Mod.Abstract()
32853285
case FINAL => Mod.Final()
3286+
case IMPLICIT if ctx.settings.YimplicitAsGiven.value => Mod.Given()
32863287
case IMPLICIT => Mod.Implicit()
32873288
case GIVEN => Mod.Given()
32883289
case LAZY => Mod.Lazy()
@@ -3552,7 +3553,7 @@ object Parsers {
35523553

35533554
def paramMods() =
35543555
if in.token == IMPLICIT then
3555-
addParamMod(() => Mod.Implicit())
3556+
addParamMod(() => if (ctx.settings.YimplicitAsGiven.value) Mod.Given() else Mod.Implicit())
35563557
else if isIdent(nme.using) then
35573558
if initialMods.is(Given) then
35583559
syntaxError(em"`using` is already implied here, should not be given explicitly", in.offset)

tests/pos-macros/i22482/Macro_1.scala

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
//> using options -Yimplicit-as-given
2+
import scala.quoted._
3+
object Macro:
4+
inline def test[I] = ${testImpl[I]}
5+
def testImpl[I: Type](using Quotes): Expr[Unit] =
6+
import quotes.reflect._
7+
8+
val implicitClause = '{
9+
def clause(implicit num: Int) = ???
10+
}
11+
val expectedImplicitClause =
12+
"""|{
13+
| def clause(using num: scala.Int): scala.Nothing = scala.Predef.???
14+
| ()
15+
|}""".stripMargin
16+
17+
assert(implicitClause.show == expectedImplicitClause)
18+
19+
// test ImplicitClauseInCaseClass
20+
assert(TypeRepr.of[I].typeSymbol.primaryConstructor.paramSymss(1)(0).flags.is(Flags.Given))
21+
22+
'{()}

tests/pos-macros/i22482/Test_2.scala

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
//> using options -Yimplicit-as-given
2+
case class ImplicitClauseInCaseClass(dummy: Int)(implicit num: Int)
3+
def Test() = Macro.test[ImplicitClauseInCaseClass]

tests/run/i22482.check

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
found if given

tests/run/i22482.scala

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
//> using options -Yimplicit-as-given
2+
3+
object Lib:
4+
class LibComponent:
5+
def msg = "found if given"
6+
implicit val libComponent: LibComponent = LibComponent()
7+
8+
class UserComponent extends LibComponent:
9+
override def msg = "found if implicit"
10+
implicit val userComponent: UserComponent = UserComponent()
11+
12+
def printComponent(implicit c: LibComponent) = println(c.msg)
13+
14+
15+
@main def Test = Lib.printComponent

0 commit comments

Comments
 (0)