Skip to content
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
686 changes: 437 additions & 249 deletions compiler/src/dotty/tools/dotc/transform/init/Objects.scala

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,4 @@ global-list.scala
t5366.scala
mutable-read7.scala
t9115.scala
Color.scala
unapplySeq-implicit-arg2.scala
unapplySeq-implicit-arg3.scala
Color.scala
20 changes: 20 additions & 0 deletions tests/init-global/pos/anon-class.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
abstract class Source[A] { self =>
def consume(a: A): Int
def contramap[B](f: B => A): Source[B] = {
new Source[B] { // OfClass($anon).outerValue = {OfClass(Source), OfClass($anon)} ???
override def consume(b: B) = self.consume(f(b))
}
}
}

object O {
val identity: Source[Int] = new Source[Int] {
override def consume(a: Int): Int = a
} // OfClass(Source[A])
val longToInt: Source[Long] = identity.contramap((l: Long) => l.toInt) // longToInt.outer == identity
val doubleToLongToInt: Source[Double] = longToInt.contramap((d: Double) => (d + 2.4).toLong) // doubleToLongToInt == longToInt
// OfClass(Source[Double]).outer = {LocalEnv(contramap)};
// LocalEnv(contramap).outer = {OfClass(Source[Long]), OfClass(Source[Double])}
println(doubleToLongToInt.consume(3.5))
}

11 changes: 5 additions & 6 deletions tests/init-global/pos/inner-extends-outer.scala
Original file line number Diff line number Diff line change
@@ -1,14 +1,13 @@
class Outer {
val f = 5
class Inner extends Outer {
val g = Outer.this.f
class Outer(val f: Int) {
class Inner extends Outer(5) {
def g(): Int = this.f
}
}

object O {
def foo(i: Outer): Unit =
val i2 = new i.Inner // i2.outer should always be OfClass(Outer)
foo(i2)
println("i2.g = " + i2.g())

foo(new Outer)
foo(new Outer(6))
}
48 changes: 48 additions & 0 deletions tests/init-global/pos/multiple-outers.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
class A(val x: Int) {
class B {
println("A.this = " + A.this.hashCode()) // `a`
println("A.this.x = " + A.this.x) // B --> outer A (42 or 46)
def fooz = x
def fooz2 = x
class D {
println("B.this = " + B.this.hashCode()) // `c` in `foo`
def bar = fooz // expands to B.this.fooz, calls fooz in class B
def bar2 = fooz2 // expands to B.this.fooz, calls fooz2 in class C
}
}
}
class AA(y: Int) extends A(y+1) {
class E {}
def foo = {
val a = if true then new A(42) else new AA(46)
println("a = " + a.hashCode())
class C /*outer: AA(44) (`Main.aa`)*/ extends a.B /*outer: A(42) or AA(46) (`a`)*/ {
println("AA.this = " + AA.this.hashCode()) // Main.aa
println("AA.this.x = " + x) // C --> outer AA --> parent A (44)
override def fooz2 = x // 44
val z = fooz // (A.this.x)
println("z = " + z)

}
class B extends AA.this.E {}
val c: C = new C
println("c = " + c.hashCode())
val d = new c.D // outer: C (`c`)
println("d.bar = " + d.bar + ", d.bar2 = " + d.bar2)
d.bar + d.bar2
}
}

object O {
val aa = new AA(44)
val f = aa.foo
println("aa = " + aa.hashCode())
println("f = " + f)
}

object Main {
def main(args: Array[String]) = {
O
()
}
}
20 changes: 20 additions & 0 deletions tests/init-global/pos/resolve-outer-of-parent.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
class A {
val field_a = 5
def bar(): Int = A.this.field_a
}

class B extends A {
val field_b = field_a
class C {
def bar2() = B.this.field_b
val field_c = bar() // expands to B.this.bar()
val field_c2 = field_a // C --> outer B --> parent A
}
}

object O:
val b = new B
class D extends b.C { // D --> parent C --> outer B
val field_d = bar2()
}
val d = new D
19 changes: 19 additions & 0 deletions tests/init-global/pos/virtual-method.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
abstract class T {
def foo() = {
def bar() = 5
bar()
}
}

class A extends T {}
class B extends T {}
class C extends T {}

object O {
val a = new A
val b = new B
val c = new C
val d = a.foo()
val e = b.foo()
val f = c.foo()
}
20 changes: 6 additions & 14 deletions tests/init-global/warn/global-cycle6.check
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,12 @@
| │ ^
| ├── object B { [ global-cycle6.scala:8 ]
| │ ^
| └── val a = new A.Inner [ global-cycle6.scala:9 ]
| ^^^^^^^^^^^
| ├── val a = new A.Inner [ global-cycle6.scala:9 ]
| │ ^^^^^^^^^^^
| ├── class Inner { [ global-cycle6.scala:3 ]
| │ ^
| └── println(n) // warn [ global-cycle6.scala:4 ]
| ^
-- Warning: tests/init-global/warn/global-cycle6.scala:4:12 ------------------------------------------------------------
4 | println(n) // warn
| ^
Expand All @@ -22,15 +26,3 @@
| │ ^
| └── println(n) // warn [ global-cycle6.scala:4 ]
| ^
-- Warning: tests/init-global/warn/global-cycle6.scala:14:9 ------------------------------------------------------------
14 | object A { // warn
| ^
| Cyclic initialization: object A -> object B -> object A. Calling trace:
| ├── object A { // warn [ global-cycle6.scala:14 ]
| │ ^
| ├── val n: Int = B.m [ global-cycle6.scala:15 ]
| │ ^
| ├── object B { [ global-cycle6.scala:21 ]
| │ ^
| └── val a = new A.Inner [ global-cycle6.scala:22 ]
| ^^^^^^^^^^^
2 changes: 1 addition & 1 deletion tests/init-global/warn/global-cycle6.scala
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ object B {
}

object O {
object A { // warn
object A {
val n: Int = B.m
class Inner {
val x: Int = 4
Expand Down
16 changes: 16 additions & 0 deletions tests/init-global/warn/inner-extends-outer.check
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
-- Warning: tests/init-global/warn/inner-extends-outer.scala:22:19 -----------------------------------------------------
22 | def bar(): Int = f2 // warn
| ^^
| Access uninitialized field value f2. Calling trace:
| ├── object O extends T { [ inner-extends-outer.scala:15 ]
| │ ^
| ├── val f1 = foo(new Outer(this)) [ inner-extends-outer.scala:20 ]
| │ ^^^^^^^^^^^^^^^^^^^^
| ├── def foo(i: Outer): Int = [ inner-extends-outer.scala:16 ]
| │ ^
| ├── i2.g() [ inner-extends-outer.scala:18 ]
| │ ^^^^^^
| ├── def g(): Int = Outer.this.t.bar() [ inner-extends-outer.scala:11 ]
| │ ^^^^^^^^^^^^^^^^^^
| └── def bar(): Int = f2 // warn [ inner-extends-outer.scala:22 ]
| ^^
23 changes: 23 additions & 0 deletions tests/init-global/warn/inner-extends-outer.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
trait T {
def bar(): Int
}

class C extends T {
def bar(): Int = 5
}

class Outer(val t: T) {
class Inner extends Outer(new C) {
def g(): Int = Outer.this.t.bar()
}
}

object O extends T {
def foo(i: Outer): Int =
val i2 = new i.Inner // i2.outer should always be OfClass(Outer)
i2.g()

val f1 = foo(new Outer(this))
val f2 = 5
def bar(): Int = f2 // warn
}
22 changes: 22 additions & 0 deletions tests/init-global/warn/local-class.check
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
-- Warning: tests/init-global/warn/local-class.scala:2:14 --------------------------------------------------------------
2 | def m() = O.f2 // warn
| ^^^^
| Access uninitialized field value f2. Calling trace:
| ├── object O { [ local-class.scala:5 ]
| │ ^
| ├── val f1 = foo() [ local-class.scala:19 ]
| │ ^^^^^
| ├── def foo(): Int = { [ local-class.scala:6 ]
| │ ^
| ├── val d = new D [ local-class.scala:15 ]
| │ ^^^^^
| ├── class D { [ local-class.scala:8 ]
| │ ^
| ├── val f = bar() [ local-class.scala:13 ]
| │ ^^^^^
| ├── def bar() = { [ local-class.scala:9 ]
| │ ^
| ├── c.m() [ local-class.scala:10 ]
| │ ^^^^^
| └── def m() = O.f2 // warn [ local-class.scala:2 ]
| ^^^^
21 changes: 21 additions & 0 deletions tests/init-global/warn/local-class.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
class C {
def m() = O.f2 // warn
}

object O {
def foo(): Int = {
val c = new C
class D {
def bar() = {
c.m()
}

val f = bar()
}
val d = new D
d.f
}

val f1 = foo()
val f2: Int = 5
}
16 changes: 16 additions & 0 deletions tests/init-global/warn/resolve-outer-of-parent.check
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
-- Warning: tests/init-global/warn/resolve-outer-of-parent.scala:7:16 --------------------------------------------------
7 | def foo() = O.d // warn
| ^^^
| Access uninitialized field value d. Calling trace:
| ├── object O: [ resolve-outer-of-parent.scala:14 ]
| │ ^
| ├── val d = new D [ resolve-outer-of-parent.scala:19 ]
| │ ^^^^^
| ├── class D extends b.C { // D --> parent C --> outer B [ resolve-outer-of-parent.scala:16 ]
| │ ^
| ├── val field_d = bar2() [ resolve-outer-of-parent.scala:17 ]
| │ ^^^^^^
| ├── def bar2() = B.this.foo() [ resolve-outer-of-parent.scala:9 ]
| │ ^^^^^^^^^^^^
| └── def foo() = O.d // warn [ resolve-outer-of-parent.scala:7 ]
| ^^^
19 changes: 19 additions & 0 deletions tests/init-global/warn/resolve-outer-of-parent.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
class A {
val field_a = 5
def bar(): Int = A.this.field_a
}

class B extends A {
def foo() = O.d // warn
class C {
def bar2() = B.this.foo()
val field_c = bar() // expands to B.this.bar()
}
}

object O:
val b = new B
class D extends b.C { // D --> parent C --> outer B
val field_d = bar2()
}
val d = new D
8 changes: 8 additions & 0 deletions tests/init-global/warn/resolve-parent-this.check
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
-- Warning: tests/init-global/warn/resolve-parent-this.scala:7:21 ------------------------------------------------------
7 | val a: Int = foo().a // warn
| ^^^^^^^
| Access uninitialized field value a. Calling trace:
| ├── object O extends Delegate { [ resolve-parent-this.scala:6 ]
| │ ^
| └── val a: Int = foo().a // warn [ resolve-parent-this.scala:7 ]
| ^^^^^^^
8 changes: 8 additions & 0 deletions tests/init-global/warn/resolve-parent-this.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
class Delegate {
def foo() = f
val f: O.type = O
}

object O extends Delegate {
val a: Int = foo().a // warn
}
Loading