From 338e3bd8a6453684b4de09b8c81add21bc24857a Mon Sep 17 00:00:00 2001 From: Tomasz Godzik Date: Tue, 22 Jun 2021 17:55:46 +0200 Subject: [PATCH] Do not widen selector type in inline matches Fixes https://github.com/lampepfl/dotty/issues/12715 --- .../src/dotty/tools/dotc/typer/Typer.scala | 4 +-- tests/pos/i12715/full.scala | 28 +++++++++++++++++++ tests/pos/i12715/minimized.scala | 8 ++++++ 3 files changed, 38 insertions(+), 2 deletions(-) create mode 100644 tests/pos/i12715/full.scala create mode 100644 tests/pos/i12715/minimized.scala diff --git a/compiler/src/dotty/tools/dotc/typer/Typer.scala b/compiler/src/dotty/tools/dotc/typer/Typer.scala index 536a80626380..d66c4e30b9d3 100644 --- a/compiler/src/dotty/tools/dotc/typer/Typer.scala +++ b/compiler/src/dotty/tools/dotc/typer/Typer.scala @@ -1465,8 +1465,8 @@ class Typer extends Namer case _ => if tree.isInline then checkInInlineContext("inline match", tree.srcPos) val sel1 = typedExpr(tree.selector) - val selType = fullyDefinedType(sel1.tpe, "pattern selector", tree.span).widen - + val rawSelectorTpe = fullyDefinedType(sel1.tpe, "pattern selector", tree.span) + val selType = if (tree.isInline) rawSelectorTpe else rawSelectorTpe.widen /** Extractor for match types hidden behind an AppliedType/MatchAlias */ object MatchTypeInDisguise { def unapply(tp: AppliedType): Option[MatchType] = tp match { diff --git a/tests/pos/i12715/full.scala b/tests/pos/i12715/full.scala new file mode 100644 index 000000000000..2f10dd5aaaf5 --- /dev/null +++ b/tests/pos/i12715/full.scala @@ -0,0 +1,28 @@ +package repro + +import compiletime.{constValue, erasedValue} + +sealed trait ValidateExprInt + +class And[A <: ValidateExprInt, B <: ValidateExprInt] extends ValidateExprInt +class GreaterThan[T <: Int] extends ValidateExprInt + +object Repro: + inline def validate[E <: ValidateExprInt](v: Int): String = + val failMsg = validateV[E](v) + if failMsg == "neverPass" then "neverPass" + else "something else" + + transparent inline def validateV[E <: ValidateExprInt](v: Int): String = + inline erasedValue[E] match + case _: GreaterThan[t] => + "GreaterThan" + case _: And[a, b] => + inline validateV[a](v) match + case "" => + validateV[b](v) + case other => + other + + @main def test(): Unit = + println(validate[And[GreaterThan[10], GreaterThan[12]]](5)) diff --git a/tests/pos/i12715/minimized.scala b/tests/pos/i12715/minimized.scala new file mode 100644 index 000000000000..4bd8a2b1a404 --- /dev/null +++ b/tests/pos/i12715/minimized.scala @@ -0,0 +1,8 @@ +transparent inline def f: String = + inline 10 match + case _ => + inline "foo" match + case x : String => x + +def test = + inline val failMsg = f