dotty.tools.dotc.transform.ForwardDepChecks.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of scala3-compiler_3 Show documentation
Show all versions of scala3-compiler_3 Show documentation
scala3-compiler-bootstrapped
package dotty.tools
package dotc
package transform
import core.*
import Symbols.*, Types.*, Contexts.*, Flags.*, Decorators.*, reporting.*
import util.Spans.Span
import util.Store
import collection.immutable
import ast.tpd
import MegaPhase.MiniPhase
object ForwardDepChecks:
import tpd.*
val name: String = "forwardDepChecks"
val description: String = "ensure no forward references to local vals"
type LevelAndIndex = immutable.Map[Symbol, (LevelInfo, Int)]
class OptLevelInfo {
def levelAndIndex: LevelAndIndex = Map()
def enterReference(sym: Symbol, span: Span): Unit = ()
}
/** A class to help in forward reference checking */
class LevelInfo(val outer: OptLevelInfo, val owner: Symbol, stats: List[Tree])(using Context)
extends OptLevelInfo {
override val levelAndIndex: LevelAndIndex =
stats.foldLeft(outer.levelAndIndex, 0) {(mi, stat) =>
val (m, idx) = mi
val m1 = stat match {
case stat: MemberDef => m.updated(stat.symbol, (this: @unchecked, idx))
case _ => m
}
(m1, idx + 1)
}._1
var maxIndex: Int = Int.MinValue
var refSpan: Span = _
var refSym: Symbol = _
override def enterReference(sym: Symbol, span: Span): Unit =
if (sym.exists && sym.owner.isTerm)
levelAndIndex.get(sym) match {
case Some((level, idx)) if (level.maxIndex < idx) =>
level.maxIndex = idx
level.refSpan = span
level.refSym = sym
case _ =>
}
}
val NoLevelInfo: OptLevelInfo = new OptLevelInfo()
class ForwardDepChecks extends MiniPhase:
import ForwardDepChecks.*
import tpd.*
override def phaseName: String = ForwardDepChecks.name
override def description: String = ForwardDepChecks.description
override def runsAfter: Set[String] = Set(ElimByName.name)
private var LevelInfo: Store.Location[OptLevelInfo] = _
private def currentLevel(using Context): OptLevelInfo = ctx.store(LevelInfo)
override def initContext(ctx: FreshContext): Unit =
LevelInfo = ctx.addLocation(NoLevelInfo)
override def prepareForStats(trees: List[Tree])(using Context): Context =
if (ctx.owner.isTerm)
ctx.fresh.updateStore(LevelInfo, new LevelInfo(currentLevel, ctx.owner, trees))
else ctx
override def transformValDef(tree: ValDef)(using Context): ValDef =
val sym = tree.symbol
if sym.exists && sym.owner.isTerm && !sym.is(Lazy) then
currentLevel.levelAndIndex.get(sym) match
case Some((level, symIdx)) if symIdx <= level.maxIndex =>
report.error(ForwardReferenceExtendsOverDefinition(sym, level.refSym),
ctx.source.atSpan(level.refSpan))
case _ =>
tree
override def transformIdent(tree: Ident)(using Context): Ident = {
currentLevel.enterReference(tree.symbol, tree.span)
tree
}
/** Check that self constructor call does not contain references to vals or defs
* defined later in the secondary constructor's right hand side. This is tricky
* since the complete self constructor might itself be a block that originated from
* expanding named and default parameters. In that case we have to go outwards
* and find the enclosing expression that consists of that block. Test cases in
* {pos,neg}/complex-self-call.scala.
*/
private def checkSelfConstructorCall()(using Context): Unit =
// Find level info corresponding to constructor's RHS. This is the info of the
// outermost LevelInfo that has the constructor as owner.
def rhsLevelInfo(l: OptLevelInfo): OptLevelInfo = l match
case l: LevelInfo if l.owner == ctx.owner =>
rhsLevelInfo(l.outer) match
case l1: LevelInfo => l1
case _ => l
case _ =>
NoLevelInfo
rhsLevelInfo(currentLevel) match
case level: LevelInfo =>
if level.maxIndex > 0 then
report.debuglog("refsym = " + level.refSym.showLocated)
report.error("forward reference not allowed from self constructor invocation",
ctx.source.atSpan(level.refSpan))
case _ =>
assert(false, s"${ctx.owner.showLocated}")
end checkSelfConstructorCall
override def transformApply(tree: Apply)(using Context): Apply =
if (isSelfConstrCall(tree))
assert(ctx.owner.isConstructor)
checkSelfConstructorCall()
tree
override def transformNew(tree: New)(using Context): New = {
currentLevel.enterReference(tree.tpe.typeSymbol, tree.span)
tree.tpe.dealias.foreachPart {
case TermRef(_, s: Symbol) => currentLevel.enterReference(s, tree.span)
case _ =>
}
tree
}
end ForwardDepChecks
© 2015 - 2025 Weber Informatics LLC | Privacy Policy