From 77a8902c52207a88d0c3cc452941b5781378d0e9 Mon Sep 17 00:00:00 2001 From: Nicolas Stucki Date: Thu, 23 Jun 2022 17:43:47 +0200 Subject: [PATCH] Fix stability check for inline parameters Inline parameters are not stable. Two references to the same parameter may not be idempotent. --- community-build/community-projects/protoquill | 2 +- compiler/src/dotty/tools/dotc/core/SymDenotations.scala | 2 +- library/src/scala/quoted/Quotes.scala | 2 +- library/src/scala/runtime/stdLibPatches/Predef.scala | 2 +- .../source-dependencies/inline-rec-change-inline/B.scala | 2 +- .../inline-rec-change-inline/changes/B1.scala | 2 +- scaladoc/src/dotty/tools/scaladoc/tasty/reflect.scala | 2 +- tests/neg/inline-param-unstable-path.scala | 6 ++++++ tests/neg/inline-val-in-inline-method.scala | 8 ++++++++ tests/pos/i11163.scala | 4 ++-- tests/pos/i12379a.scala | 4 ++-- tests/run/deriving-constructor-order.scala | 2 +- 12 files changed, 26 insertions(+), 12 deletions(-) create mode 100644 tests/neg/inline-param-unstable-path.scala create mode 100644 tests/neg/inline-val-in-inline-method.scala diff --git a/community-build/community-projects/protoquill b/community-build/community-projects/protoquill index 16d26fcb3072..494c2ddc06e7 160000 --- a/community-build/community-projects/protoquill +++ b/community-build/community-projects/protoquill @@ -1 +1 @@ -Subproject commit 16d26fcb30720b9aa81d29f08b9da10916e269a2 +Subproject commit 494c2ddc06e71f1c7f13b382675525130feee9a0 diff --git a/compiler/src/dotty/tools/dotc/core/SymDenotations.scala b/compiler/src/dotty/tools/dotc/core/SymDenotations.scala index 33fa77fc7c6e..7f86fb0ad38e 100644 --- a/compiler/src/dotty/tools/dotc/core/SymDenotations.scala +++ b/compiler/src/dotty/tools/dotc/core/SymDenotations.scala @@ -773,7 +773,7 @@ object SymDenotations { * So the first call to a stable member might fail and/or produce side effects. */ final def isStableMember(using Context): Boolean = { - def isUnstableValue = isOneOf(UnstableValueFlags) || info.isInstanceOf[ExprType] + def isUnstableValue = isOneOf(UnstableValueFlags) || info.isInstanceOf[ExprType] || isAllOf(InlineParam) isType || is(StableRealizable) || exists && !isUnstableValue } diff --git a/library/src/scala/quoted/Quotes.scala b/library/src/scala/quoted/Quotes.scala index 038a97d52069..875e60ac3d67 100644 --- a/library/src/scala/quoted/Quotes.scala +++ b/library/src/scala/quoted/Quotes.scala @@ -14,7 +14,7 @@ import scala.reflect.TypeTest * } * ``` */ -transparent inline def quotes(using inline q: Quotes): q.type = q +transparent inline def quotes(using q: Quotes): q.type = q /** Quotation context provided by a macro expansion or in the scope of `scala.quoted.staging.run`. * Used to perform all operations on quoted `Expr` or `Type`. diff --git a/library/src/scala/runtime/stdLibPatches/Predef.scala b/library/src/scala/runtime/stdLibPatches/Predef.scala index 3b7d009ff6f3..09feaf11c31d 100644 --- a/library/src/scala/runtime/stdLibPatches/Predef.scala +++ b/library/src/scala/runtime/stdLibPatches/Predef.scala @@ -31,7 +31,7 @@ object Predef: * @tparam T the type of the value to be summoned * @return the given value typed: the provided type parameter */ - transparent inline def summon[T](using inline x: T): x.type = x + transparent inline def summon[T](using x: T): x.type = x // Extension methods for working with explicit nulls diff --git a/sbt-test/source-dependencies/inline-rec-change-inline/B.scala b/sbt-test/source-dependencies/inline-rec-change-inline/B.scala index 61e61a620957..eaeef8d57ece 100644 --- a/sbt-test/source-dependencies/inline-rec-change-inline/B.scala +++ b/sbt-test/source-dependencies/inline-rec-change-inline/B.scala @@ -1,5 +1,5 @@ object B { - inline def inlinedAny(x: String): x.type = x + inline def inlinedAny(x: String): String = x } diff --git a/sbt-test/source-dependencies/inline-rec-change-inline/changes/B1.scala b/sbt-test/source-dependencies/inline-rec-change-inline/changes/B1.scala index 4a1c47d38572..63104570fed4 100644 --- a/sbt-test/source-dependencies/inline-rec-change-inline/changes/B1.scala +++ b/sbt-test/source-dependencies/inline-rec-change-inline/changes/B1.scala @@ -1,5 +1,5 @@ object B { - inline def inlinedAny(inline x: String): x.type = x + inline def inlinedAny(inline x: String): String = x } diff --git a/scaladoc/src/dotty/tools/scaladoc/tasty/reflect.scala b/scaladoc/src/dotty/tools/scaladoc/tasty/reflect.scala index b48519e29d28..419beac50134 100644 --- a/scaladoc/src/dotty/tools/scaladoc/tasty/reflect.scala +++ b/scaladoc/src/dotty/tools/scaladoc/tasty/reflect.scala @@ -4,4 +4,4 @@ package tasty import scala.quoted._ /** Shorthand for `quotes.reflect` */ -transparent inline def reflect(using inline q: Quotes): q.reflect.type = q.reflect +transparent inline def reflect(using q: Quotes): q.reflect.type = q.reflect diff --git a/tests/neg/inline-param-unstable-path.scala b/tests/neg/inline-param-unstable-path.scala new file mode 100644 index 000000000000..be2d7142bc2f --- /dev/null +++ b/tests/neg/inline-param-unstable-path.scala @@ -0,0 +1,6 @@ +inline val a = 3 +inline def f(inline x: Int, y: Int, z: => Int): Unit = + val x2: x.type = x // error: (x : Int) is not a valid singleton type, since it is not an immutable path + val y2: y.type = y + val z2: z.type = z // error: (z : Int) is not a valid singleton type, since it is not an immutable path + val a2: a.type = a diff --git a/tests/neg/inline-val-in-inline-method.scala b/tests/neg/inline-val-in-inline-method.scala new file mode 100644 index 000000000000..fbd0f69ff2d5 --- /dev/null +++ b/tests/neg/inline-val-in-inline-method.scala @@ -0,0 +1,8 @@ +inline def f(inline x: Int): Unit = + inline val b = x + val c: b.type = b + +def test = + f(1) + def a = 1 + f(a) // error: inline value must have a literal constant type diff --git a/tests/pos/i11163.scala b/tests/pos/i11163.scala index acf5629d1ae9..7f220d15e392 100644 --- a/tests/pos/i11163.scala +++ b/tests/pos/i11163.scala @@ -1,5 +1,5 @@ inline def summonA[T](using x: T): x.type = x -inline def summonB[T](using inline x: T): x.type = x +// inline def summonB[T](using inline x: T): x.type = x // inline parameters are unstable inline def summonC[T](using inline x: T): T = x trait Foo: @@ -7,6 +7,6 @@ trait Foo: def test(using Foo) = summonA[Foo].f - summonB[Foo].f + // summonB[Foo].f summonC[Foo].f () diff --git a/tests/pos/i12379a.scala b/tests/pos/i12379a.scala index 0a1bc8461226..9ca2404e0871 100644 --- a/tests/pos/i12379a.scala +++ b/tests/pos/i12379a.scala @@ -1,7 +1,7 @@ -inline def convFail[Of, From](inline from : From) : Unit = +inline def convFail[Of, From](from : From) : Unit = // removed inline from parameter to avoid unsound path selection val c = compiletime.summonInline[Conversion[from.type, Of]] -inline def convOK[Of, From](inline from : From)(using c : Conversion[from.type, Of]) : Unit = {} +inline def convOK[Of, From](from : From)(using c : Conversion[from.type, Of]) : Unit = {} // removed inline from parameter to avoid unsound path selection class Bar[T](value : T) given [T <: Int] : Conversion[T, Bar[T]] = Bar(_) diff --git a/tests/run/deriving-constructor-order.scala b/tests/run/deriving-constructor-order.scala index 0a3fcde7f84f..d71e73c95d23 100644 --- a/tests/run/deriving-constructor-order.scala +++ b/tests/run/deriving-constructor-order.scala @@ -2,7 +2,7 @@ import scala.compiletime.erasedValue import scala.deriving.Mirror object Test extends App { - inline def checkElems[A, T](using inline A: Mirror.SumOf[A]): Unit = + inline def checkElems[A, T](using A: Mirror.SumOf[A]): Unit = // removed inline from parameter to avoid unsound path selection inline erasedValue[A.MirroredElemTypes] match { case _: T => () }