From 64088894704b1645222a80baac22943c48f12536 Mon Sep 17 00:00:00 2001 From: Matt Bovel Date: Tue, 27 Feb 2024 19:15:07 +0100 Subject: [PATCH] Use isStatic intsead of isStaticOwner in hasLocalInstantiation Co-Authored-By: Dale Wijnand Co-Authored-By: noti0na1 Co-Authored-By: odersky <795990+odersky@users.noreply.github.com> --- .../tools/dotc/transform/ExplicitOuter.scala | 2 +- .../outer-ref-elimination.check | 9 +++++ .../outer-ref-elimination.scala | 36 +++++++++++++++++++ 3 files changed, 46 insertions(+), 1 deletion(-) create mode 100644 tests/generic-java-signatures/outer-ref-elimination.check create mode 100644 tests/generic-java-signatures/outer-ref-elimination.scala diff --git a/compiler/src/dotty/tools/dotc/transform/ExplicitOuter.scala b/compiler/src/dotty/tools/dotc/transform/ExplicitOuter.scala index f57595293ae1..e34dbfe3e1d8 100644 --- a/compiler/src/dotty/tools/dotc/transform/ExplicitOuter.scala +++ b/compiler/src/dotty/tools/dotc/transform/ExplicitOuter.scala @@ -227,7 +227,7 @@ object ExplicitOuter { private def hasLocalInstantiation(cls: ClassSymbol)(using Context): Boolean = // Modules are normally locally instantiated, except if they are declared in a trait, // in which case they will be instantiated in the classes that mix in the trait. - cls.owner.ownersIterator.takeWhile(!_.isStatic).exists(_.isTerm) + cls.owner.ownersIterator.takeWhile(!_.isStaticOwner).exists(_.isTerm) || cls.is(Private, butNot = Module) || cls.is(Module) && !cls.owner.is(Trait) diff --git a/tests/generic-java-signatures/outer-ref-elimination.check b/tests/generic-java-signatures/outer-ref-elimination.check new file mode 100644 index 000000000000..628f7abc476c --- /dev/null +++ b/tests/generic-java-signatures/outer-ref-elimination.check @@ -0,0 +1,9 @@ +List(public T1$C1()) +List(public T2$$anon$1$C2()) +List(public T3$C3$1()) +List(public T4$$anon$2$C4()) +List(public T5$C5(T5)) +List(public T6$$anon$3$C6()) +List(public T7$C7$1()) +List(public T8$$anon$4$C8()) +List(public T9$C9(T9)) diff --git a/tests/generic-java-signatures/outer-ref-elimination.scala b/tests/generic-java-signatures/outer-ref-elimination.scala new file mode 100644 index 000000000000..19bdb0995613 --- /dev/null +++ b/tests/generic-java-signatures/outer-ref-elimination.scala @@ -0,0 +1,36 @@ +// This test checks that references to outer classes in inner classes are +// eliminated in some cases when they are not used. This is done is the +// ExplicitOuter phase. See issue #19569 for discussions. + +object helper { + def test(cls: Class[?]) = println(cls.getDeclaredConstructors.toList) +} + +import helper.test + +object T1 { class C1; test(classOf[C1]) } +object T2 { new AnyRef { class C2; test(classOf[C2]) } } +object T3 { def t3(): Unit = { class C3; test(classOf[C3]) } } +object T4 { def t4(): Unit = new AnyRef { class C4; test(classOf[C4]) } } + +class T5 { class C5; test(classOf[C5]) } // outer ref currently not elided +class T6 { new AnyRef { class C6; test(classOf[C6]) } } +class T7 { def t7(): Unit = { class C7; test(classOf[C7]) } } +class T8 { def t8(): Unit = new AnyRef { class C8; test(classOf[C8]) } } + +// The outer ref is not elided when it is used: +class T9 { var x = 451; class C9 {def getX = x}; test(classOf[C9]) } + +object Test { + def main(args: Array[String]): Unit = { + T1 + T2 + T3.t3() + T4.t4() + new T5() + new T6() + new T7().t7() + new T8().t8() + new T9() + } +}