Skip to content

Commit f56aa22

Browse files
committed
Restrict the new rules to givens that don't define a new class
`given ... with` or `given ... = new { ... }` kinds of definitions now follow the old rules. This allows recursive `given...with` definitions as they are found in protoQuill. We still have the old check in a later phase against directly recursive methods. Of the three loops in the original i15474 we now detect #2 and #3 with new new restrictions. #1 slips through since it is a loop involving a `given...with` instance of `Conversion`, but is caught later with the recursive method check. Previously tests #1 and #3 were detected with the recursive methods check and #2 slipped through altogether. The new rules are enough for defining simple givens with `=` without fear of looping.
1 parent ed9f427 commit f56aa22

File tree

8 files changed

+25
-38
lines changed

8 files changed

+25
-38
lines changed

compiler/src/dotty/tools/dotc/semanticdb/Scala3.scala

-3
Original file line numberDiff line numberDiff line change
@@ -77,9 +77,6 @@ object Scala3:
7777
type SemanticSymbol = Symbol | FakeSymbol
7878

7979
given SemanticSymbolOps : AnyRef with
80-
import SymbolOps.*
81-
import StringOps.*
82-
8380
extension (sym: SemanticSymbol)
8481
def name(using Context): Name = sym match
8582
case s: Symbol => s.name

compiler/src/dotty/tools/dotc/typer/Implicits.scala

+5-3
Original file line numberDiff line numberDiff line change
@@ -1569,16 +1569,18 @@ trait Implicits:
15691569

15701570
/** Does candidate `cand` come too late for it to be considered as an
15711571
* eligible candidate? This is the case if `cand` appears in the same
1572-
* scope as a given definition enclosing the search point and comes
1573-
* later in the source or coincides with that given definition.
1572+
* scope as a given definition enclosing the search point (with no
1573+
* class methods between the given definition and the search point)
1574+
* and `cand` comes later in the source or coincides with that given
1575+
* definition.
15741576
*/
15751577
def comesTooLate(cand: Candidate): Boolean =
15761578
val candSym = cand.ref.symbol
15771579
def candSucceedsGiven(sym: Symbol): Boolean =
15781580
if sym.owner == candSym.owner then
15791581
if sym.is(ModuleClass) then candSucceedsGiven(sym.sourceModule)
15801582
else sym.is(Given) && sym.span.exists && sym.span.start <= candSym.span.start
1581-
else if sym.is(Package) then false
1583+
else if sym.owner.isClass then false
15821584
else candSucceedsGiven(sym.owner)
15831585

15841586
ctx.isTyper

compiler/src/scala/quoted/runtime/impl/QuotesImpl.scala

-4
Original file line numberDiff line numberDiff line change
@@ -1781,8 +1781,6 @@ class QuotesImpl private (using val ctx: Context) extends Quotes, QuoteUnpickler
17811781
end TypeRepr
17821782

17831783
given TypeReprMethods: TypeReprMethods with
1784-
import SymbolMethods.*
1785-
17861784
extension (self: TypeRepr)
17871785

17881786
def show(using printer: Printer[TypeRepr]): String = printer.show(self)
@@ -2610,8 +2608,6 @@ class QuotesImpl private (using val ctx: Context) extends Quotes, QuoteUnpickler
26102608
end Symbol
26112609

26122610
given SymbolMethods: SymbolMethods with
2613-
import FlagsMethods.*
2614-
26152611
extension (self: Symbol)
26162612
def owner: Symbol = self.denot.owner
26172613
def maybeOwner: Symbol = self.denot.maybeOwner

tests/neg/i15474.check

+7-20
Original file line numberDiff line numberDiff line change
@@ -1,31 +1,18 @@
1-
-- Error: tests/neg/i15474.scala:7:35 ----------------------------------------------------------------------------------
2-
7 | def apply(from: String): Int = from.toInt // error
3-
| ^^^^
1+
-- Error: tests/neg/i15474.scala:6:39 ----------------------------------------------------------------------------------
2+
6 | given c: Conversion[ String, Int ] = _.toInt // error
3+
| ^
44
| Warning: result of implicit search for ?{ toInt: ? } will change.
5-
| Current result Test1.c will be no longer eligible
5+
| Current result Test2.c will be no longer eligible
66
| because it is not defined before the search position.
77
| Result with new rules: augmentString.
88
| To opt into the new rules, use the `experimental.avoidLoopingGivens` language import.
99
|
1010
| To fix the problem without the language import, you could try one of the following:
11-
| - rearrange definitions so that Test1.c comes earlier,
11+
| - rearrange definitions so that Test2.c comes earlier,
1212
| - use an explicit conversion,
1313
| - use an import to get extension method into scope.
14-
-- Error: tests/neg/i15474.scala:10:39 ---------------------------------------------------------------------------------
15-
10 | given c: Conversion[ String, Int ] = _.toInt // error
16-
| ^
17-
| Warning: result of implicit search for ?{ toInt: ? } will change.
18-
| Current result Test2.c will be no longer eligible
19-
| because it is not defined before the search position.
20-
| Result with new rules: augmentString.
21-
| To opt into the new rules, use the `experimental.avoidLoopingGivens` language import.
22-
|
23-
| To fix the problem without the language import, you could try one of the following:
24-
| - rearrange definitions so that Test2.c comes earlier,
25-
| - use an explicit conversion,
26-
| - use an import to get extension method into scope.
27-
-- Error: tests/neg/i15474.scala:16:56 ---------------------------------------------------------------------------------
28-
16 | given Ordering[Price] = summon[Ordering[BigDecimal]] // error
14+
-- Error: tests/neg/i15474.scala:12:56 ---------------------------------------------------------------------------------
15+
12 | given Ordering[Price] = summon[Ordering[BigDecimal]] // error
2916
| ^
3017
| Warning: result of implicit search for Ordering[BigDecimal] will change.
3118
| Current result Prices.Price.given_Ordering_Price will be no longer eligible

tests/neg/i15474.scala

-4
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,6 @@
22

33
import scala.language.implicitConversions
44

5-
object Test1:
6-
given c: Conversion[ String, Int ] with
7-
def apply(from: String): Int = from.toInt // error
8-
95
object Test2:
106
given c: Conversion[ String, Int ] = _.toInt // error
117

tests/neg/i15474b.check

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
-- Error: tests/neg/i15474b.scala:7:40 ---------------------------------------------------------------------------------
2+
7 | def apply(from: String): Int = from.toInt // error: infinite loop in function body
3+
| ^^^^^^^^^^
4+
| Infinite loop in function body
5+
| Test1.c.apply(from).toInt

tests/neg/i15474b.scala

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
//> using options -Xfatal-warnings
2+
3+
import scala.language.implicitConversions
4+
5+
object Test1:
6+
given c: Conversion[ String, Int ] with
7+
def apply(from: String): Int = from.toInt // error: infinite loop in function body
8+

tests/pos/i15474.scala

-4
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,6 @@
22
import scala.language.implicitConversions
33
import scala.language.experimental.avoidLoopingGivens
44

5-
object Test1:
6-
given c: Conversion[ String, Int ] with
7-
def apply(from: String): Int = from.toInt // was error, now avoided
8-
95
object Test2:
106
given c: Conversion[ String, Int ] = _.toInt // now avoided, was loop not detected, could be used as a fallback to avoid the warning.
117

0 commit comments

Comments
 (0)