From 99ba6de65806fdc4774fd7d75cc5e51919c06c11 Mon Sep 17 00:00:00 2001 From: odersky Date: Mon, 16 Sep 2024 19:29:36 +0200 Subject: [PATCH] Align SAM test and expansion --- compiler/src/dotty/tools/dotc/ast/tpd.scala | 22 ++++------------- .../src/dotty/tools/dotc/core/Types.scala | 24 ++++--------------- 2 files changed, 9 insertions(+), 37 deletions(-) diff --git a/compiler/src/dotty/tools/dotc/ast/tpd.scala b/compiler/src/dotty/tools/dotc/ast/tpd.scala index d4e585402feb..f97baa7f7889 100644 --- a/compiler/src/dotty/tools/dotc/ast/tpd.scala +++ b/compiler/src/dotty/tools/dotc/ast/tpd.scala @@ -328,21 +328,6 @@ object tpd extends Trees.Instance[Type] with TypedTreeInfo { superArgs: List[Tree] = Nil, adaptVarargs: Boolean = false)(using Context): TypeDef = val firstParent :: otherParents = cls.info.parents: @unchecked - def isApplicable(constr: Symbol): Boolean = - def recur(ctpe: Type): Boolean = ctpe match - case ctpe: PolyType => - recur(ctpe.instantiate(firstParent.argTypes)) - case ctpe: MethodType => - var paramInfos = ctpe.paramInfos - if adaptVarargs && paramInfos.length == superArgs.length + 1 - && atPhaseNoLater(Phases.elimRepeatedPhase)(constr.info.isVarArgsMethod) - then // accept missing argument for varargs parameter - paramInfos = paramInfos.init - superArgs.corresponds(paramInfos)(_.tpe <:< _) - case _ => - false - recur(constr.info) - def adaptedSuperArgs(ctpe: Type): List[Tree] = ctpe match case ctpe: PolyType => adaptedSuperArgs(ctpe.instantiate(firstParent.argTypes)) @@ -357,8 +342,11 @@ object tpd extends Trees.Instance[Type] with TypedTreeInfo { val superRef = if cls.is(Trait) then TypeTree(firstParent) else - val constr = firstParent.decl(nme.CONSTRUCTOR).suchThat(isApplicable) - New(firstParent, constr.symbol.asTerm, adaptedSuperArgs(constr.info)) + val parentConstr = firstParent.applicableConstructors(superArgs.tpes, adaptVarargs) match + case Nil => assert(false, i"no applicable parent constructor of $firstParent for supercall arguments $superArgs") + case constr :: Nil => constr + case _ => assert(false, i"multiple applicable parent constructors of $firstParent for supercall arguments $superArgs") + New(firstParent, parentConstr.asTerm, adaptedSuperArgs(parentConstr.info)) ClassDefWithParents(cls, constr, superRef :: otherParents.map(TypeTree(_)), body) end ClassDef diff --git a/compiler/src/dotty/tools/dotc/core/Types.scala b/compiler/src/dotty/tools/dotc/core/Types.scala index 5ab01aa08a1f..12de7f465f91 100644 --- a/compiler/src/dotty/tools/dotc/core/Types.scala +++ b/compiler/src/dotty/tools/dotc/core/Types.scala @@ -5945,30 +5945,14 @@ object Types extends TypeUtils { def samClass(tp: Type)(using Context): Symbol = tp match case tp: ClassInfo => val cls = tp.cls - def zeroParamsOLD(tp: Type): Boolean = tp.stripPoly match - case mt: MethodType => mt.paramInfos.isEmpty && !mt.resultType.isInstanceOf[MethodType] - case et: ExprType => true - case _ => false - val validCtorOLD = - val ctor = cls.primaryConstructor - // `ContextFunctionN` does not have constructors - !ctor.exists || zeroParamsOLD(ctor.info) - def takesNoArgs(tp: Type) = - !tp.classSymbol.primaryConstructor.exists // `ContextFunctionN` does not have constructors - || tp.applicableConstructors(Nil, adaptVarargs = true).nonEmpty - def firstParentCls = tp.parents.head.classSymbol + !tp.classSymbol.primaryConstructor.exists + // e.g. `ContextFunctionN` does not have constructors + || tp.applicableConstructors(Nil, adaptVarargs = true).lengthCompare(1) == 0 + // we require a unique constructor so that SAM expansion is deterministic val noArgsNeeded: Boolean = takesNoArgs(tp) && (!tp.cls.is(Trait) || takesNoArgs(tp.parents.head)) - - if noArgsNeeded != validCtorOLD then - println( - i"""SAM change for $tp with parent ${firstParentCls.fullName}, now $noArgsNeeded - |takesNoArgs: ${takesNoArgs(tp)} - |takesNoArgsParent: ${takesNoArgs(tp.cls.info.parents.head)} - |primary: ${firstParentCls.primaryConstructor.info}""") - def isInstantiable = !tp.cls.isOneOf(FinalOrSealed) && (tp.appliedRef <:< tp.selfType) if noArgsNeeded && isInstantiable then tp.cls