Skip to content

Commit 8897dc4

Browse files
authored
Report only non-overridden unimplemented members (#21337)
Previously, when a concrete class A had unimplemented members that are overridden, all overrides would be reported as unimplemented in the error message. This would produce error messages that are not accurate, and that suggest stubs that are not correct. This patch fixes the issue by reporting in the error message only the unimplemented members that are not overridden by other unimplemented members. Fixes #21335
2 parents 590b496 + 0d50a30 commit 8897dc4

File tree

3 files changed

+31
-1
lines changed

3 files changed

+31
-1
lines changed

compiler/src/dotty/tools/dotc/typer/RefChecks.scala

+11-1
Original file line numberDiff line numberDiff line change
@@ -698,6 +698,15 @@ object RefChecks {
698698
&& withMode(Mode.IgnoreCaptures)(mbrDenot.matchesLoosely(impl, alwaysCompareTypes = true)))
699699
.exists
700700

701+
/** Filter out symbols from `syms` that are overridden by a symbol appearing later in the list.
702+
* Symbols that are not overridden are kept. */
703+
def lastOverrides(syms: List[Symbol]): List[Symbol] =
704+
val deduplicated =
705+
syms.foldLeft(List.empty[Symbol]):
706+
case (acc, sym) if acc.exists(s => isOverridingPair(s, sym, clazz.thisType)) => acc
707+
case (acc, sym) => sym :: acc
708+
deduplicated.reverse
709+
701710
/** The term symbols in this class and its baseclasses that are
702711
* abstract in this class. We can't use memberNames for that since
703712
* a concrete member might have the same signature as an abstract
@@ -720,7 +729,8 @@ object RefChecks {
720729

721730
val missingMethods = grouped.toList flatMap {
722731
case (name, syms) =>
723-
syms.filterConserve(!_.isSetter)
732+
lastOverrides(syms)
733+
.filterConserve(!_.isSetter)
724734
.distinctBy(_.signature) // Avoid duplication for similar definitions (#19731)
725735
}
726736

tests/neg/i21335.check

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
-- Error: tests/neg/i21335.scala:7:6 -----------------------------------------------------------------------------------
2+
7 |class Z1 extends Bar1 // error
3+
| ^
4+
| class Z1 needs to be abstract, since override def bar(): Bar1 in trait Bar1 is not defined
5+
-- Error: tests/neg/i21335.scala:12:6 ----------------------------------------------------------------------------------
6+
12 |class Z2 extends Bar2 // error
7+
| ^
8+
| class Z2 needs to be abstract, since def bar(): Bar2 in trait Bar2 is not defined

tests/neg/i21335.scala

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
trait Foo:
2+
def bar(): Foo
3+
4+
trait Bar1 extends Foo:
5+
override def bar(): Bar1
6+
7+
class Z1 extends Bar1 // error
8+
9+
trait Bar2 extends Foo:
10+
def bar(): Bar2
11+
12+
class Z2 extends Bar2 // error

0 commit comments

Comments
 (0)