From 563b3a1286b15354126a8f5dbb432449d9ac7990 Mon Sep 17 00:00:00 2001 From: Hamza Remmal <56235032+hamzaremmal@users.noreply.github.com> Date: Wed, 15 Nov 2023 18:15:11 +0100 Subject: [PATCH] Add checks for the consistency of the parents in TreeChecker --- .../dotty/tools/dotc/transform/Erasure.scala | 14 ++++++++++- .../tools/dotc/transform/TreeChecker.scala | 20 +++++++++++++++- tests/neg-macros/i18677-a.check | 22 ++++++++++++++++++ tests/neg-macros/i18677-a/Macro_1.scala | 18 +++++++++++++++ tests/neg-macros/i18677-a/Test_2.scala | 4 ++++ tests/neg-macros/i18677-b.check | 22 ++++++++++++++++++ tests/neg-macros/i18677-b/Macro_1.scala | 18 +++++++++++++++ tests/neg-macros/i18677-b/Test_2.scala | 4 ++++ tests/neg-macros/wrong-owner.check | 23 +++++++++---------- 9 files changed, 131 insertions(+), 14 deletions(-) create mode 100644 tests/neg-macros/i18677-a.check create mode 100644 tests/neg-macros/i18677-a/Macro_1.scala create mode 100644 tests/neg-macros/i18677-a/Test_2.scala create mode 100644 tests/neg-macros/i18677-b.check create mode 100644 tests/neg-macros/i18677-b/Macro_1.scala create mode 100644 tests/neg-macros/i18677-b/Test_2.scala diff --git a/compiler/src/dotty/tools/dotc/transform/Erasure.scala b/compiler/src/dotty/tools/dotc/transform/Erasure.scala index 8582420d64ee..505907f991ff 100644 --- a/compiler/src/dotty/tools/dotc/transform/Erasure.scala +++ b/compiler/src/dotty/tools/dotc/transform/Erasure.scala @@ -34,6 +34,8 @@ import ExplicitOuter.* import core.Mode import util.Property import reporting.* +import dotty.tools.dotc.ast.Trees +import dotty.tools.dotc.core.unpickleScala2.Scala2Erasure.isTrait class Erasure extends Phase with DenotTransformer { @@ -1044,7 +1046,17 @@ object Erasure { override def typedClassDef(cdef: untpd.TypeDef, cls: ClassSymbol)(using Context): Tree = if cls.is(Flags.Erased) then erasedDef(cls) - else super.typedClassDef(cdef, cls) + else + val typedTree@TypeDef(name, impl @ Template(constr, _, self, _)) = super.typedClassDef(cdef, cls): @unchecked + // In the case where a trait extends a class, we need to strip any non trait class from the signature + // and accept the first one (see tests/run/mixins.scala) + val newParents = impl.parents.head :: impl.parents.tail.filterConserve: tree => + def isTraitConstructor = tree match + case Trees.Block(_, expr) => // Specific management for trait constructors (See #9216) + expr.symbol.isConstructor && expr.symbol.owner.isTrait + case _ => tree.symbol.isConstructor && tree.symbol.owner.isTrait + tree.symbol.isTrait || isTraitConstructor + cpy.TypeDef(typedTree)(rhs = cpy.Template(impl)(parents = newParents)) override def typedAnnotated(tree: untpd.Annotated, pt: Type)(using Context): Tree = typed(tree.arg, pt) diff --git a/compiler/src/dotty/tools/dotc/transform/TreeChecker.scala b/compiler/src/dotty/tools/dotc/transform/TreeChecker.scala index d847d14603a2..17298a45e01e 100644 --- a/compiler/src/dotty/tools/dotc/transform/TreeChecker.scala +++ b/compiler/src/dotty/tools/dotc/transform/TreeChecker.scala @@ -547,8 +547,24 @@ object TreeChecker { i"owner chain = ${tree.symbol.ownersIterator.toList}%, %, ctxOwners = ${ctx.outersIterator.map(_.owner).toList}%, %") } + private def checkParents(tree: untpd.TypeDef)(using Context): Unit = { + val TypeDef(_, impl: Template) = tree: @unchecked + assert(ctx.owner.isClass) + val sym = ctx.owner.asClass + if !sym.isPrimitiveValueClass then + val symbolParents = sym.classInfo.parents.map(_.dealias.typeSymbol) + val treeParents = impl.parents.map(_.tpe.dealias.typeSymbol) + assert(symbolParents == treeParents, + i"""Parents of class symbol differs from the parents in the tree for $sym + | + |Parents in symbol: $symbolParents + |Parents in tree: $treeParents + |""".stripMargin) + } + override def typedTypeDef(tdef: untpd.TypeDef, sym: Symbol)(using Context): Tree = { assert(sym.info.isInstanceOf[ClassInfo | TypeBounds], i"wrong type, expect a template or type bounds for ${sym.fullName}, but found: ${sym.info}") + if sym.isClass then checkParents(tdef) super.typedTypeDef(tdef, sym) } @@ -561,6 +577,8 @@ object TreeChecker { checkOwner(impl) checkOwner(impl.constr) + checkParents(cdef) + def isNonMagicalMember(x: Symbol) = !x.isValueClassConvertMethod && !x.name.is(DocArtifactName) && @@ -812,7 +830,7 @@ object TreeChecker { else err.getStackTrace.nn.mkString(" ", " \n", "") report.error( - s"""Malformed tree was found while expanding macro with -Xcheck-macros. + em"""Malformed tree was found while expanding macro with -Xcheck-macros. |The tree does not conform to the compiler's tree invariants. | |Macro was: diff --git a/tests/neg-macros/i18677-a.check b/tests/neg-macros/i18677-a.check new file mode 100644 index 000000000000..976ff7bddda2 --- /dev/null +++ b/tests/neg-macros/i18677-a.check @@ -0,0 +1,22 @@ + +-- Error: tests/neg-macros/i18677-a/Test_2.scala:4:6 ------------------------------------------------------------------- +3 |@extendFoo +4 |class AFoo // error + |^ + |Malformed tree was found while expanding macro with -Xcheck-macros. + |The tree does not conform to the compiler's tree invariants. + | + |Macro was: + |@scala.annotation.internal.SourceFile("tests/neg-macros/i18677-a/Test_2.scala") @extendFoo class AFoo() + | + |The macro returned: + |@scala.annotation.internal.SourceFile("tests/neg-macros/i18677-a/Test_2.scala") @extendFoo class AFoo() extends Foo + | + |Error: + |assertion failed: Parents of class symbol differs from the parents in the tree for class AFoo + | + |Parents in symbol: [class Object] + |Parents in tree: [trait Foo] + | + | + |stacktrace available when compiling with `-Ydebug` diff --git a/tests/neg-macros/i18677-a/Macro_1.scala b/tests/neg-macros/i18677-a/Macro_1.scala new file mode 100644 index 000000000000..f624d2d15782 --- /dev/null +++ b/tests/neg-macros/i18677-a/Macro_1.scala @@ -0,0 +1,18 @@ +//> using -expermiental + +import annotation.MacroAnnotation +import quoted.* + +trait Foo + +class extendFoo extends MacroAnnotation : + override def transform(using Quotes)(tree: quotes.reflect.Definition): List[quotes.reflect.Definition] = + import quotes.reflect.* + tree match + case ClassDef(name, ctr, p, self, body) => + val parents = List(TypeTree.of[Foo]) + val newTree = ClassDef.copy(tree)(name, ctr, parents, self, body) + newTree :: Nil + case _ => + report.error("@extendFoo can only annotate class definitions") + tree :: Nil \ No newline at end of file diff --git a/tests/neg-macros/i18677-a/Test_2.scala b/tests/neg-macros/i18677-a/Test_2.scala new file mode 100644 index 000000000000..486d7e09a4bc --- /dev/null +++ b/tests/neg-macros/i18677-a/Test_2.scala @@ -0,0 +1,4 @@ +//> using -expermiental + +@extendFoo +class AFoo // error \ No newline at end of file diff --git a/tests/neg-macros/i18677-b.check b/tests/neg-macros/i18677-b.check new file mode 100644 index 000000000000..b6d817a53706 --- /dev/null +++ b/tests/neg-macros/i18677-b.check @@ -0,0 +1,22 @@ + +-- Error: tests/neg-macros/i18677-b/Test_2.scala:4:6 ------------------------------------------------------------------- +3 |@extendFoo +4 |class AFoo // error + |^ + |Malformed tree was found while expanding macro with -Xcheck-macros. + |The tree does not conform to the compiler's tree invariants. + | + |Macro was: + |@scala.annotation.internal.SourceFile("tests/neg-macros/i18677-b/Test_2.scala") @extendFoo class AFoo() + | + |The macro returned: + |@scala.annotation.internal.SourceFile("tests/neg-macros/i18677-b/Test_2.scala") @extendFoo class AFoo() extends Foo + | + |Error: + |assertion failed: Parents of class symbol differs from the parents in the tree for class AFoo + | + |Parents in symbol: [class Object] + |Parents in tree: [class Foo] + | + | + |stacktrace available when compiling with `-Ydebug` diff --git a/tests/neg-macros/i18677-b/Macro_1.scala b/tests/neg-macros/i18677-b/Macro_1.scala new file mode 100644 index 000000000000..b2ff946ad302 --- /dev/null +++ b/tests/neg-macros/i18677-b/Macro_1.scala @@ -0,0 +1,18 @@ +//> using -expermiental + +import annotation.MacroAnnotation +import quoted.* + +class Foo + +class extendFoo extends MacroAnnotation : + override def transform(using Quotes)(tree: quotes.reflect.Definition): List[quotes.reflect.Definition] = + import quotes.reflect.* + tree match + case ClassDef(name, ctr, p, self, body) => + val parents = List(TypeTree.of[Foo]) + val newTree = ClassDef.copy(tree)(name, ctr, parents, self, body) + newTree :: Nil + case _ => + report.error("@extendFoo can only annotate class definitions") + tree :: Nil \ No newline at end of file diff --git a/tests/neg-macros/i18677-b/Test_2.scala b/tests/neg-macros/i18677-b/Test_2.scala new file mode 100644 index 000000000000..486d7e09a4bc --- /dev/null +++ b/tests/neg-macros/i18677-b/Test_2.scala @@ -0,0 +1,4 @@ +//> using -expermiental + +@extendFoo +class AFoo // error \ No newline at end of file diff --git a/tests/neg-macros/wrong-owner.check b/tests/neg-macros/wrong-owner.check index ccaca98e3948..ca8751d0fe1c 100644 --- a/tests/neg-macros/wrong-owner.check +++ b/tests/neg-macros/wrong-owner.check @@ -5,19 +5,18 @@ 5 |class Foo // error |^ |Malformed tree was found while expanding macro with -Xcheck-macros. - | |The tree does not conform to the compiler's tree invariants. - | | - | |Macro was: - | |@scala.annotation.internal.SourceFile("tests/neg-macros/wrong-owner/Test_2.scala") @wrongOwner @scala.annotation.experimental class Foo() - | | - | |The macro returned: - | |@scala.annotation.internal.SourceFile("tests/neg-macros/wrong-owner/Test_2.scala") @wrongOwner @scala.annotation.experimental class Foo() { + |The tree does not conform to the compiler's tree invariants. + | + |Macro was: + |@scala.annotation.internal.SourceFile("tests/neg-macros/wrong-owner/Test_2.scala") @wrongOwner @scala.annotation.experimental class Foo() + | + |The macro returned: + |@scala.annotation.internal.SourceFile("tests/neg-macros/wrong-owner/Test_2.scala") @wrongOwner @scala.annotation.experimental class Foo() { | override def toString(): java.lang.String = "Hello from macro" |} - | | - | |Error: - | |assertion failed: bad owner; method toString has owner class String, expected was class Foo + | + |Error: + |assertion failed: bad owner; method toString has owner class String, expected was class Foo |owner chain = method toString, class String, package java.lang, package java, package , ctxOwners = class Foo, class Foo, package , package , package , package , package , package , package , package , package , , , , , - | | + | |stacktrace available when compiling with `-Ydebug` - | |