Skip to content

Commit 5246e16

Browse files
committed
Fix false positive in WUnused for renamed path-dependent imports (attempt 2)
1 parent 982c91b commit 5246e16

File tree

2 files changed

+34
-22
lines changed

2 files changed

+34
-22
lines changed

compiler/src/dotty/tools/dotc/transform/CheckUnused.scala

+24-22
Original file line numberDiff line numberDiff line change
@@ -337,7 +337,7 @@ object CheckUnused:
337337
* - usage
338338
*/
339339
private class UnusedData:
340-
import collection.mutable.{Set => MutSet, Map => MutMap, Stack => MutStack}
340+
import collection.mutable.{Set => MutSet, Map => MutMap, Stack => MutStack, ListBuffer => MutList}
341341
import UnusedData.*
342342

343343
/** The current scope during the tree traversal */
@@ -346,7 +346,7 @@ object CheckUnused:
346346
var unusedAggregate: Option[UnusedResult] = None
347347

348348
/* IMPORTS */
349-
private val impInScope = MutStack(MutSet[tpd.Import]())
349+
private val impInScope = MutStack(MutList[tpd.Import]())
350350
/**
351351
* We store the symbol along with their accessibility without import.
352352
* Accessibility to their definition in outer context/scope
@@ -449,7 +449,7 @@ object CheckUnused:
449449
def pushScope(newScopeType: ScopeType): Unit =
450450
// unused imports :
451451
currScopeType.push(newScopeType)
452-
impInScope.push(MutSet())
452+
impInScope.push(MutList())
453453
usedInScope.push(MutSet())
454454

455455
def registerSetVar(sym: Symbol): Unit =
@@ -664,25 +664,27 @@ object CheckUnused:
664664

665665
/** Given an import and accessibility, return selector that matches import<->symbol */
666666
private def isInImport(imp: tpd.Import, isAccessible: Boolean, symName: Option[Name], isDerived: Boolean)(using Context): Option[ImportSelector] =
667-
val tpd.Import(qual, sels) = imp
668-
val dealiasedSym = dealias(sym)
669-
val simpleSelections = qual.tpe.member(sym.name).alternatives
670-
val typeSelections = sels.flatMap(n => qual.tpe.member(n.name.toTypeName).alternatives)
671-
val termSelections = sels.flatMap(n => qual.tpe.member(n.name.toTermName).alternatives)
672-
val selectionsToDealias = typeSelections ::: termSelections
673-
val qualHasSymbol = simpleSelections.map(_.symbol).contains(sym) || (simpleSelections ::: selectionsToDealias).map(_.symbol).map(dealias).contains(dealiasedSym)
674-
def selector = sels.find(sel => (sel.name.toTermName == sym.name || sel.name.toTypeName == sym.name) && symName.map(n => n.toTermName == sel.rename).getOrElse(true))
675-
def dealiasedSelector = if(isDerived) sels.flatMap(sel => selectionsToDealias.map(m => (sel, m.symbol))).collect {
676-
case (sel, sym) if dealias(sym) == dealiasedSym => sel
677-
}.headOption else None
678-
def givenSelector = if sym.is(Given) || sym.is(Implicit)
679-
then sels.filter(sel => sel.isGiven && !sel.bound.isEmpty).find(sel => sel.boundTpe =:= sym.info)
680-
else None
681-
def wildcard = sels.find(sel => sel.isWildcard && ((sym.is(Given) == sel.isGiven && sel.bound.isEmpty) || sym.is(Implicit)))
682-
if qualHasSymbol && (!isAccessible || sym.isRenamedSymbol(symName)) && sym.exists then
683-
selector.orElse(dealiasedSelector).orElse(givenSelector).orElse(wildcard) // selector with name or wildcard (or given)
684-
else
685-
None
667+
val tpd.Import(qual, sels) = imp
668+
val dealiasedSym = dealias(sym)
669+
val simpleSelections = qual.tpe.member(sym.name).alternatives
670+
val typeSelections = sels.flatMap(n => qual.tpe.member(n.name.toTypeName).alternatives)
671+
val termSelections = sels.flatMap(n => qual.tpe.member(n.name.toTermName).alternatives)
672+
val sameTermPath = qual.isTerm && sym.exists && sym.owner.isType && qual.tpe.typeSymbol == sym.owner.asType
673+
val selectionsToDealias = typeSelections ::: termSelections
674+
val renamedSelection = if sameTermPath then sels.find(sel => sel.imported.name == sym.name) else None
675+
val qualHasSymbol = simpleSelections.map(_.symbol).contains(sym) || (simpleSelections ::: selectionsToDealias).map(_.symbol).map(dealias).contains(dealiasedSym) || renamedSelection.isDefined
676+
def selector = sels.find(sel => (sel.name.toTermName == sym.name || sel.name.toTypeName == sym.name) && symName.map(n => n.toTermName == sel.rename).getOrElse(true))
677+
def dealiasedSelector = if(isDerived) sels.flatMap(sel => selectionsToDealias.map(m => (sel, m.symbol))).collect {
678+
case (sel, sym) if dealias(sym) == dealiasedSym => sel
679+
}.headOption else None
680+
def givenSelector = if sym.is(Given) || sym.is(Implicit)
681+
then sels.filter(sel => sel.isGiven && !sel.bound.isEmpty).find(sel => sel.boundTpe =:= sym.info)
682+
else None
683+
def wildcard = sels.find(sel => sel.isWildcard && ((sym.is(Given) == sel.isGiven && sel.bound.isEmpty) || sym.is(Implicit)))
684+
if qualHasSymbol && (!isAccessible || sym.isRenamedSymbol(symName)) && sym.exists then
685+
selector.orElse(dealiasedSelector).orElse(givenSelector).orElse(wildcard).orElse(renamedSelection) // selector with name or wildcard (or given)
686+
else
687+
None
686688

687689
private def isRenamedSymbol(symNameInScope: Option[Name])(using Context) =
688690
sym.name != nme.NO_NAME && symNameInScope.exists(_.toSimpleName != sym.name.toSimpleName)

tests/pos/i18366.scala

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
//> using options -Xfatal-warnings -Wunused:all
2+
3+
trait Builder {
4+
def foo(): Unit
5+
}
6+
7+
def repro =
8+
val builder: Builder = ???
9+
import builder.{foo => bar}
10+
bar()

0 commit comments

Comments
 (0)