Skip to content

Commit e6cd09f

Browse files
committed
Let inlining report compiletime.error text
1 parent 4518ce5 commit e6cd09f

File tree

4 files changed

+91
-1
lines changed

4 files changed

+91
-1
lines changed

compiler/src/dotty/tools/dotc/inlines/Inlines.scala

+6-1
Original file line numberDiff line numberDiff line change
@@ -508,8 +508,13 @@ object Inlines:
508508
inContext(evCtx) {
509509
val evidence = evTyper.inferImplicitArg(tpe, callTypeArgs.head.span)
510510
evidence.tpe match
511+
case fail: Implicits.LateMismatchedImplicit =>
512+
val addendum = fail.errors match
513+
case Nil => ""
514+
case errs => errs.map(_.msg).mkString("a search with errors:\n ", "\n ", "")
515+
errorTree(call, evTyper.missingArgMsg(evidence, tpe, addendum))
511516
case fail: Implicits.SearchFailureType =>
512-
errorTree(call, evTyper.missingArgMsg(evidence, tpe, ""))
517+
errorTree(call, evTyper.missingArgMsg(evidence, tpe, where = ""))
513518
case _ =>
514519
evidence
515520
}

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

+7
Original file line numberDiff line numberDiff line change
@@ -616,6 +616,11 @@ object Implicits:
616616
def msg(using Context): Message =
617617
em"${errors.map(_.msg).mkString("\n")}"
618618
}
619+
620+
class LateMismatchedImplicit(ref: TermRef,
621+
expectedType: Type,
622+
argument: Tree,
623+
val errors: List[Diagnostic.Error]) extends MismatchedImplicit(ref, expectedType, argument)
619624
end Implicits
620625

621626
import Implicits.*
@@ -1233,6 +1238,8 @@ trait Implicits:
12331238
NoMatchingImplicitsFailure
12341239
else if Splicer.inMacroExpansion && tpe <:< pt then
12351240
SearchFailure(adapted.withType(new MacroErrorsFailure(ctx.reporter.allErrors.reverse, pt, argument)))
1241+
else if ctx.isAfterTyper then
1242+
SearchFailure(adapted.withType(LateMismatchedImplicit(ref, pt, argument, ctx.reporter.allErrors.reverse)))
12361243
else
12371244
SearchFailure(adapted.withType(new MismatchedImplicit(ref, pt, argument)))
12381245
}

tests/neg/i15788.check

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
-- Error: tests/neg/i15788.scala:39:13 ---------------------------------------------------------------------------------
2+
39 | fff1[Test2] // error
3+
| ^
4+
| Field 'a' not found
5+
-- Error: tests/neg/i15788.scala:42:15 ---------------------------------------------------------------------------------
6+
42 | summonInline[Test[Test2]].test // error
7+
| ^
8+
| Field 'a' not found
9+
-- [E172] Type Error: tests/neg/i15788.scala:45:6 ----------------------------------------------------------------------
10+
45 | fff2[Test2] // error
11+
| ^^^^^^^^^^^
12+
| No given instance of type Test[Test2] was found for a search with errors:
13+
| Field 'a' not found.
14+
| I found:
15+
|
16+
| Test.given_Test_A[Test2](
17+
| Test2.$asInstanceOf[
18+
| scala.deriving.Mirror.Product{
19+
| type MirroredMonoType = Test2; type MirroredType = Test2; type MirroredLabel = ("Test2" : String);
20+
| type MirroredElemTypes = (String, Int); type MirroredElemLabels = (("q" : String), ("w" : String))
21+
| }
22+
| ]
23+
| )
24+
|
25+
| But given instance given_Test_A in object Test does not match type Test[Test2].
26+
|--------------------------------------------------------------------------------------------------------------------
27+
|Inline stack trace
28+
|- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
29+
|This location contains code that was inlined from i15788.scala:35
30+
35 | summonInline[Test[P]].test
31+
| ^^^^^^^^^^^^^^^^^^^^^
32+
--------------------------------------------------------------------------------------------------------------------

tests/neg/i15788.scala

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
2+
import scala.deriving.Mirror
3+
import scala.compiletime.*
4+
5+
trait Test[A] {
6+
def test: String
7+
}
8+
9+
object Test {
10+
11+
inline def findA[T <: Tuple]: Unit =
12+
inline erasedValue[T] match {
13+
case _: EmptyTuple => error("Field 'a' not found")
14+
case _: ("a" *: tl) => ()
15+
case _: (_ *: tl) => findA[tl]
16+
}
17+
18+
inline given [A <: Product] => (mm: Mirror.ProductOf[A]) => Test[A] = new {
19+
override def test: String = {
20+
findA[mm.MirroredElemLabels]
21+
"test"
22+
}
23+
}
24+
}
25+
26+
final case class Test1(a: String, b: Int)
27+
final case class Test2(q: String, w: Int)
28+
29+
object Main {
30+
inline def fff1[P <: Product](using ggg: Test[P]): String = {
31+
ggg.test
32+
}
33+
34+
inline def fff2[P <: Product]: String = {
35+
summonInline[Test[P]].test
36+
}
37+
38+
fff1[Test1]
39+
fff1[Test2] // error
40+
41+
summonInline[Test[Test1]].test
42+
summonInline[Test[Test2]].test // error
43+
44+
fff2[Test1]
45+
fff2[Test2] // error
46+
}

0 commit comments

Comments
 (0)