Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Backport "Report only non-overridden unimplemented members" to LTS #22092

Merged
merged 1 commit into from
Dec 3, 2024
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
12 changes: 11 additions & 1 deletion compiler/src/dotty/tools/dotc/typer/RefChecks.scala
Original file line number Diff line number Diff line change
Expand Up @@ -600,6 +600,15 @@ object RefChecks {
&& withMode(Mode.IgnoreCaptures)(mbrDenot.matchesLoosely(impl, alwaysCompareTypes = true)))
.exists

/** Filter out symbols from `syms` that are overridden by a symbol appearing later in the list.
* Symbols that are not overridden are kept. */
def lastOverrides(syms: List[Symbol]): List[Symbol] =
val deduplicated =
syms.foldLeft(List.empty[Symbol]):
case (acc, sym) if acc.exists(s => isOverridingPair(s, sym, clazz.thisType)) => acc
case (acc, sym) => sym :: acc
deduplicated.reverse

/** The term symbols in this class and its baseclasses that are
* abstract in this class. We can't use memberNames for that since
* a concrete member might have the same signature as an abstract
Expand All @@ -622,7 +631,8 @@ object RefChecks {

val missingMethods = grouped.toList flatMap {
case (name, syms) =>
syms.filterConserve(!_.isSetter)
lastOverrides(syms)
.filterConserve(!_.isSetter)
.distinctBy(_.signature) // Avoid duplication for similar definitions (#19731)
}

Expand Down
8 changes: 8 additions & 0 deletions tests/neg/i21335.check
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
-- Error: tests/neg/i21335.scala:7:6 -----------------------------------------------------------------------------------
7 |class Z1 extends Bar1 // error
| ^
| class Z1 needs to be abstract, since override def bar(): Bar1 in trait Bar1 is not defined
-- Error: tests/neg/i21335.scala:12:6 ----------------------------------------------------------------------------------
12 |class Z2 extends Bar2 // error
| ^
| class Z2 needs to be abstract, since def bar(): Bar2 in trait Bar2 is not defined
12 changes: 12 additions & 0 deletions tests/neg/i21335.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
trait Foo:
def bar(): Foo

trait Bar1 extends Foo:
override def bar(): Bar1

class Z1 extends Bar1 // error

trait Bar2 extends Foo:
def bar(): Bar2

class Z2 extends Bar2 // error
Loading