Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Dealias before checking for member in lint #22708

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
55 changes: 27 additions & 28 deletions compiler/src/dotty/tools/dotc/typer/RefChecks.scala
Original file line number Diff line number Diff line change
Expand Up @@ -1137,14 +1137,24 @@ object RefChecks {
end checkUnaryMethods

/** Check that an extension method is not hidden, i.e., that it is callable as an extension method.
*
* For example, it is not possible to define a type-safe extension `contains` for `Set`,
* since for any parameter type, the existing `contains` method will compile and would be used.
*
* An extension method is hidden if it does not offer a parameter that is not subsumed
* by the corresponding parameter of the member with the same name (or of all alternatives of an overload).
*
* This check is suppressed if this method is an override.
* This check is suppressed if the method is an override. (Because the type of the receiver
* may be narrower in the override.)
*
* For example, it is not possible to define a type-safe extension `contains` for `Set`,
* since for any parameter type, the existing `contains` method will compile and would be used.
* If the extension method is nullary, it is always hidden by a member of the same name.
* (Either the member is nullary, or the reference is taken as the eta-expansion of the member.)
*
* This check is in lieu of a more expensive use-site check that an application failed to use an extension.
* That check would account for accessibility and opacity. As a limitation, this check considers
* only public members for which corresponding method parameters are either both opaque types or both not.
* It is intended to warn if the receiver type from a third-party library has been augmented with a member
* that nullifies an existing extension.
*
* If the member has a leading implicit parameter list, then the extension method must also have
* a leading implicit parameter list. The reason is that if the implicit arguments are inferred,
Expand All @@ -1155,42 +1165,31 @@ object RefChecks {
* If the member does not have a leading implicit parameter list, then the argument cannot be explicitly
* supplied with `using`, as typechecking would fail. But the extension method may have leading implicit
* parameters, which are necessarily supplied implicitly in the application. The first non-implicit
* parameters of the extension method must be distinguishable from the member parameters, as described.
*
* If the extension method is nullary, it is always hidden by a member of the same name.
* (Either the member is nullary, or the reference is taken as the eta-expansion of the member.)
*
* This check is in lieu of a more expensive use-site check that an application failed to use an extension.
* That check would account for accessibility and opacity. As a limitation, this check considers
* only public members, a target receiver that is not an alias, and corresponding method parameters
* that are either both opaque types or both not.
* parameters of the extension method must be distinguishable from the member parameters, as described above.
*/
def checkExtensionMethods(sym: Symbol)(using Context): Unit =
if sym.is(Extension) then
extension (tp: Type)
def explicit = Applications.stripImplicit(tp.stripPoly, wildcardOnly = true)
def hasImplicitParams = tp.stripPoly match { case mt: MethodType => mt.isImplicitMethod case _ => false }
val explicitInfo = sym.info.explicit // consider explicit value params
val target = explicitInfo.firstParamTypes.head.typeSymbol.info // required for extension method, the putative receiver
val target0 = explicitInfo.firstParamTypes.head // required for extension method, the putative receiver
val target = target0.dealiasKeepOpaques.typeSymbol.info
val methTp = explicitInfo.resultType // skip leading implicits and the "receiver" parameter
def memberMatchesMethod(member: Denotation) =
val memberIsImplicit = member.info.hasImplicitParams
val paramTps =
if memberIsImplicit then methTp.stripPoly.firstParamTypes
else methTp.explicit.firstParamTypes
inline def paramsCorrespond =
val memberParamTps = member.info.stripPoly.firstParamTypes
memberParamTps.corresponds(paramTps): (m, x) =>
m.typeSymbol.denot.isOpaqueAlias == x.typeSymbol.denot.isOpaqueAlias && (x frozen_<:< m)
paramTps.isEmpty || memberIsImplicit && !methTp.hasImplicitParams || paramsCorrespond
Comment on lines +1179 to +1188
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The refactor looks correct to me. The only question is: were the isEmpty and length checks there for happy-path-optimization? I suppose that RefChecks is not a performance bottleneck, so it should be fine.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I almost added a code comment at the time: paramTps.isEmpty was checked, and corresponds ensures same length, so the extra length compare was only because of lazyZip. My question was why didn't I make this edit last time I touched the code? I guess it's OK to focus on correctness first and then performance & style eventually.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Simple coding decisions become harder when the domain model is harder (do I need dealias here? stripPoly there?) and still some mental friction in Scala 3 syntax (can I put a colon here? is this easy to read?).

def hidden =
target.nonPrivateMember(sym.name)
.filterWithPredicate: member =>
member.symbol.isPublic && {
val memberIsImplicit = member.info.hasImplicitParams
val paramTps =
if memberIsImplicit then methTp.stripPoly.firstParamTypes
else methTp.explicit.firstParamTypes

paramTps.isEmpty || memberIsImplicit && !methTp.hasImplicitParams || {
val memberParamTps = member.info.stripPoly.firstParamTypes
!memberParamTps.isEmpty
&& memberParamTps.lengthCompare(paramTps) == 0
&& memberParamTps.lazyZip(paramTps).forall: (m, x) =>
m.typeSymbol.denot.isOpaqueAlias == x.typeSymbol.denot.isOpaqueAlias
&& (x frozen_<:< m)
}
}
member.symbol.isPublic && memberMatchesMethod(member)
.exists
if sym.is(HasDefaultParams) then
val getterDenot =
Expand Down
2 changes: 1 addition & 1 deletion tests/warn/ext-override.scala
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
//> using options -Xfatal-warnings
//> using options -Werror

trait Foo[T]:
extension (x: T)
Expand Down
2 changes: 1 addition & 1 deletion tests/warn/i16743.scala
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,7 @@ trait DungeonDweller:
trait SadDungeonDweller:
def f[A](x: Dungeon.IArray[A]) = 27 // x.length // just to confirm, length is not a member

trait Quote:
trait Quote: // see tests/warn/ext-override.scala
type Tree <: AnyRef
given TreeMethods: TreeMethods
trait TreeMethods:
Expand Down
5 changes: 5 additions & 0 deletions tests/warn/i22232.scala
Original file line number Diff line number Diff line change
Expand Up @@ -23,10 +23,15 @@ object Upperbound3:
object NonUpperbound1:
opaque type MyString[+T] = String
extension (arr: MyString[Byte]) def length: Int = 0 // nowarn

object NonUpperbound2:
opaque type MyString[+T] = String
extension [T <: MyString[Byte]](arr: T) def length2: Int = 0 // nowarn

object NonUpperbound3:
opaque type MyString[+T] = String
extension [T](arr: T) def length: Int = 0 // nowarn

object NonUpperbound4:
opaque type MyString = String
extension (arr: MyString) def length: Int = 0 // nowarn
28 changes: 28 additions & 0 deletions tests/warn/i22705.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
//> using options -Werror

object Native {
class Obj:
def f: String = "F"
}

object Types {

opaque type Node = Native.Obj

type S = Node

object S:
def apply(): S = new Node

extension (s: S)
def f: String = "S"
}

import Types.*

object Main {
def main(args: Array[String]): Unit = {
val v: S = S()
println(v.f)
}
}
30 changes: 30 additions & 0 deletions tests/warn/i22706.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
//> using options -Werror

object Native {
class O {
def f: String = "F"
}
class M extends O
}

object Types {
opaque type N = Native.O
opaque type GS = Native.M

type S = N | GS

object S:
def apply(): S = new N

extension (s: S)
def f: String = "S"
}

import Types.*

object Main {
def main(args: Array[String]): Unit = {
val v: S = S()
println(v.f)
}
}
14 changes: 14 additions & 0 deletions tests/warn/i22727.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
//> using options -Werror

object Main {
type IXY = (Int, Int)

extension (xy: IXY) {
def map(f: Int => Int): (Int, Int) = (f(xy._1), f(xy._2))
}

def main(args: Array[String]): Unit = {
val a = (0, 1)
println(a)
}
}
Loading