mirror of
https://github.com/KarolS/millfork.git
synced 2025-04-04 22:29:32 +00:00
Improved error reporting for constants used before definition
This commit is contained in:
parent
0b7183daf2
commit
d7f2b0688f
@ -56,6 +56,8 @@ There are no built-in encodings now, the include path needs to contain the neces
|
||||
|
||||
* Allow placing platform definitions in a dedicated subdirectory.
|
||||
|
||||
* Improved error reporting for constants used before their definitions.
|
||||
|
||||
* Z80: Intel syntax for all Z80 instructions, based on Digital Research's Z80.LIB.
|
||||
|
||||
* Commander X16: Updated to support VERA 0.9 and the new joystick API. Added mouse support.
|
||||
|
30
src/main/scala/millfork/MutableMultimap.scala
Normal file
30
src/main/scala/millfork/MutableMultimap.scala
Normal file
@ -0,0 +1,30 @@
|
||||
package millfork
|
||||
|
||||
import scala.collection.mutable
|
||||
|
||||
/**
|
||||
* @author Karol Stasiak
|
||||
*/
|
||||
class MutableMultimap[K, V]() {
|
||||
val inner: mutable.HashMap[K, mutable.HashSet[V]] = mutable.HashMap()
|
||||
|
||||
def addBinding(k: K, v: V): Unit = {
|
||||
inner.get(k) match {
|
||||
case None =>
|
||||
val s = mutable.HashSet[V]()
|
||||
s += v
|
||||
inner(k) = s
|
||||
case Some(s) =>
|
||||
s += v
|
||||
}
|
||||
}
|
||||
|
||||
def get(k: K): Set[V] = {
|
||||
inner.get(k) match {
|
||||
case None =>
|
||||
Set()
|
||||
case Some(s) =>
|
||||
s.toSet
|
||||
}
|
||||
}
|
||||
}
|
@ -36,7 +36,7 @@ object M6809MacroExpander extends MacroExpander[MLine] {
|
||||
case x => x
|
||||
}
|
||||
case (AssemblyOrMacroParam(typ, Placeholder(ph, phType), AssemblyParameterPassingBehaviour.ByConstant), actualParam) =>
|
||||
ctx.env.eval(actualParam).getOrElse(ctx.env.errorConstant("Non-constant expression was passed to an inlineable function as a `const` parameter", actualParam.position))
|
||||
ctx.env.eval(actualParam).getOrElse(ctx.env.errorConstant("Non-constant expression was passed to an inlineable function as a `const` parameter", Some(actualParam), actualParam.position))
|
||||
actualCode = actualCode.map {
|
||||
case a@Z80AssemblyStatement(_, _, _, expr, _) =>
|
||||
a.copy(expression = expr.replaceVariable(ph, actualParam))
|
||||
|
@ -34,7 +34,7 @@ object MosMacroExpander extends MacroExpander[AssemblyLine] {
|
||||
case x => x
|
||||
}
|
||||
case (AssemblyOrMacroParam(typ, Placeholder(ph, phType), AssemblyParameterPassingBehaviour.ByConstant), actualParam) =>
|
||||
ctx.env.eval(actualParam).getOrElse(ctx.env.errorConstant("Non-constant expression was passed to an inlineable function as a `const` parameter", actualParam.position))
|
||||
ctx.env.eval(actualParam).getOrElse(ctx.env.errorConstant("Non-constant expression was passed to an inlineable function as a `const` parameter", Some(actualParam), actualParam.position))
|
||||
actualCode = actualCode.map {
|
||||
case a@MosAssemblyStatement(_, _, expr, _) =>
|
||||
a.copy(expression = expr.replaceVariable(ph, actualParam))
|
||||
|
@ -360,7 +360,7 @@ object MosStatementCompiler extends AbstractStatementCompiler[AssemblyLine] {
|
||||
case FunctionCallExpression("hudson_transfer$", params) =>
|
||||
StructureConstant(env.get[StructType]("hudson_transfer$"), params.map(y => compileParameterForAssemblyStatement(env, o, y)))
|
||||
case _ =>
|
||||
env.evalForAsm(x).getOrElse(env.errorConstant(s"`$x` is not a constant", x.position))
|
||||
env.evalForAsm(x).getOrElse(env.errorConstant(s"`$x` is not a constant", Some(x), x.position))
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -36,7 +36,7 @@ object Z80MacroExpander extends MacroExpander[ZLine] {
|
||||
case x => x
|
||||
}
|
||||
case (AssemblyOrMacroParam(typ, Placeholder(ph, phType), AssemblyParameterPassingBehaviour.ByConstant), actualParam) =>
|
||||
ctx.env.eval(actualParam).getOrElse(ctx.env.errorConstant("Non-constant expression was passed to an inlineable function as a `const` parameter", actualParam.position))
|
||||
ctx.env.eval(actualParam).getOrElse(ctx.env.errorConstant("Non-constant expression was passed to an inlineable function as a `const` parameter", Some(actualParam), actualParam.position))
|
||||
actualCode = actualCode.map {
|
||||
case a@Z80AssemblyStatement(_, _, _, expr, _) =>
|
||||
a.copy(expression = expr.replaceVariable(ph, actualParam))
|
||||
|
@ -234,7 +234,7 @@ object Z80StatementCompiler extends AbstractStatementCompiler[ZLine] {
|
||||
env.evalForAsm(expression).orElse(env.maybeGet[ThingInMemory](name).map(_.toAddress)).getOrElse(MemoryAddressConstant(Label(name)))
|
||||
}
|
||||
case _ =>
|
||||
env.evalForAsm(expression).getOrElse(env.errorConstant(s"`$expression` is not a constant", expression.position))
|
||||
env.evalForAsm(expression).getOrElse(env.errorConstant(s"`$expression` is not a constant", Some(expression), expression.position))
|
||||
}
|
||||
val registers = (reg, offset) match {
|
||||
case (OneRegister(r), Some(o)) => env.evalForAsm(expression) match {
|
||||
|
96
src/main/scala/millfork/env/Environment.scala
vendored
96
src/main/scala/millfork/env/Environment.scala
vendored
@ -28,6 +28,8 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
@inline
|
||||
def nextLabel: LabelGenerator = jobContext.nextLabel
|
||||
|
||||
private val constantsThatShouldHaveBeenImportedEarlier = new MutableMultimap[String, Option[Position]]()
|
||||
|
||||
private var baseStackOffset: Int = cpuFamily match {
|
||||
case CpuFamily.M6502 => 0x101
|
||||
case CpuFamily.I80 => 0
|
||||
@ -35,9 +37,10 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
case CpuFamily.M6809 => 0
|
||||
}
|
||||
|
||||
def errorConstant(msg: String, position: Option[Position] = None): Constant = {
|
||||
log.error(msg, position)
|
||||
def errorConstant(msg: String, invalidExpression: Option[Expression], position: Option[Position] = None): Constant = {
|
||||
log.error(msg, position.orElse(invalidExpression.flatMap(_.position)))
|
||||
log.info("Did you forget to import an appropriate module?")
|
||||
invalidExpression.foreach(markAsConstantsThatShouldHaveBeenImportedEarlier)
|
||||
Constant.Zero
|
||||
}
|
||||
|
||||
@ -919,7 +922,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
}
|
||||
}
|
||||
|
||||
def evalForAsm(e: Expression, op: MOpcode.Value = MOpcode.NOP): Option[Constant] = {
|
||||
def evalForAsm(e: Expression, op: MOpcode.Value = MOpcode.NOP, silent: Boolean = false): Option[Constant] = {
|
||||
e match {
|
||||
// TODO: hmmm
|
||||
case VariableExpression(name) =>
|
||||
@ -934,7 +937,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
case ConstantArrayElementExpression(c) => Some(c)
|
||||
case VariableExpression(name) =>
|
||||
val result = maybeGet[ConstantThing](name).map(_.value).orElse(maybeGet[ThingInMemory](name).map(_.toAddress))
|
||||
if (result.isEmpty) log.warn(s"$name is not known")
|
||||
if (!silent && result.isEmpty) log.warn(s"$name is not known", e.position)
|
||||
result
|
||||
case IndexedExpression(name, index) => (evalForAsm(VariableExpression(name)), evalForAsm(index)) match {
|
||||
case (Some(a), Some(b)) => Some(CompoundConstant(MathOperator.Plus, a, b).quickSimplify)
|
||||
@ -984,13 +987,13 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
case "lo" =>
|
||||
oneArgFunctionForAsm(_.loByte, params)
|
||||
case ">>>>" | ">>'" | "<<'" | ">" | "<" =>
|
||||
log.error(s"Operator `$name` not supported in inline assembly", e.position)
|
||||
if (!silent) log.error(s"Operator `$name` not supported in inline assembly", e.position)
|
||||
None
|
||||
case _ if name.endsWith("=") =>
|
||||
log.error(s"Operator `$name` not supported in inline assembly", e.position)
|
||||
if (!silent) log.error(s"Operator `$name` not supported in inline assembly", e.position)
|
||||
None
|
||||
case "nonet" | "sin" | "cos" | "tan" =>
|
||||
log.error("Function not supported in inline assembly", e.position)
|
||||
if (!silent) log.error("Function not supported in inline assembly", e.position)
|
||||
None
|
||||
case "sizeof" =>
|
||||
Some(evalSizeof(params.head))
|
||||
@ -1048,7 +1051,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
for((name, optValue) <- stmt.variants) {
|
||||
optValue match {
|
||||
case Some(v) =>
|
||||
value = eval(v).getOrElse(errorConstant(s"Enum constant `${stmt.name}.$name` is not a constant", stmt.position))
|
||||
value = eval(v).getOrElse(errorConstant(s"Enum constant `${stmt.name}.$name` is not a constant", Some(v), stmt.position))
|
||||
case _ =>
|
||||
}
|
||||
addThing(ConstantThing(name, value.fitInto(t), t), stmt.position)
|
||||
@ -1214,7 +1217,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
case None =>
|
||||
log.error(s"Extern function `${stmt.name}`needs an address", stmt.position)
|
||||
case Some(a) =>
|
||||
val addr = eval(a).getOrElse(errorConstant(s"Address of `${stmt.name}` is not a constant", stmt.position))
|
||||
val addr = eval(a).getOrElse(errorConstant(s"Address of `${stmt.name}` is not a constant", Some(a), stmt.position))
|
||||
val mangled = ExternFunction(
|
||||
name,
|
||||
resultType,
|
||||
@ -1339,7 +1342,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
params,
|
||||
env,
|
||||
stackVariablesSize,
|
||||
stmt.address.map(a => this.eval(a).getOrElse(errorConstant(s"Address of `${stmt.name}` is not a constant"))),
|
||||
stmt.address.map(a => this.eval(a).getOrElse(errorConstant(s"Address of `${stmt.name}` is not a constant", Some(a), a.position))),
|
||||
executableStatements ++ paramForAutomaticReturn.map(param => ReturnStatement(param).pos(executableStatements.lastOption.fold(stmt.position)(_.position))),
|
||||
hasElidedReturnVariable = hasElidedReturnVariable,
|
||||
interrupt = stmt.interrupt,
|
||||
@ -1596,6 +1599,34 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
}
|
||||
}
|
||||
|
||||
//noinspection ScalaUnusedExpression
|
||||
private def markAsConstantsThatShouldHaveBeenImportedEarlier(node: Expression): Unit = {
|
||||
node match {
|
||||
case VariableExpression(v) =>
|
||||
if (eval(node).isEmpty && evalForAsm(node, silent = true).isEmpty) {
|
||||
log.error(s"The constant $v is undefined", node.position)
|
||||
log.info("Did you forget to import an appropriate module?")
|
||||
constantsThatShouldHaveBeenImportedEarlier.addBinding(v.takeWhile(_ != '.').mkString(""), node.position)
|
||||
}
|
||||
case FunctionCallExpression(_, params) =>
|
||||
params.foreach(markAsConstantsThatShouldHaveBeenImportedEarlier)
|
||||
case SumExpression(params, _) =>
|
||||
params.foreach(i => markAsConstantsThatShouldHaveBeenImportedEarlier(i._2))
|
||||
case SeparateBytesExpression(h, l) =>
|
||||
markAsConstantsThatShouldHaveBeenImportedEarlier(h)
|
||||
markAsConstantsThatShouldHaveBeenImportedEarlier(l)
|
||||
case IndexedExpression(a, i) =>
|
||||
constantsThatShouldHaveBeenImportedEarlier.addBinding(a, node.position)
|
||||
markAsConstantsThatShouldHaveBeenImportedEarlier(i)
|
||||
case DerefExpression(p, _, _) =>
|
||||
markAsConstantsThatShouldHaveBeenImportedEarlier(p)
|
||||
case DerefDebuggingExpression(p, _) =>
|
||||
markAsConstantsThatShouldHaveBeenImportedEarlier(p)
|
||||
case _ =>
|
||||
// not a variable
|
||||
}
|
||||
}
|
||||
|
||||
def extractArrayContents(contents1: ArrayContents): List[Expression] = contents1 match {
|
||||
case LiteralContents(xs) => xs
|
||||
case CombinedContents(xs) => xs.flatMap(extractArrayContents)
|
||||
@ -1606,7 +1637,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
log.error(s"Invalid struct array contents", pc.position)
|
||||
Nil
|
||||
case pc@ProcessedContents(f, xs) => pc.getAllExpressions(options.isBigEndian)
|
||||
case ForLoopContents(v, start, end, direction, body) =>
|
||||
case flc@ForLoopContents(v, start, end, direction, body) =>
|
||||
(eval(start), eval(end)) match {
|
||||
case (Some(NumericConstant(s, sz1)), Some(NumericConstant(e, sz2))) =>
|
||||
val size = sz1 max sz2
|
||||
@ -1617,10 +1648,21 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
}).toList
|
||||
range.flatMap(i => extractArrayContents(body).map(_.replaceVariable(v, LiteralExpression(i, size))))
|
||||
case (Some(_), Some(_)) =>
|
||||
log.error("Array range bounds cannot be evaluated")
|
||||
log.error("Array range bounds cannot be evaluated", flc.position.orElse(flc.start.position))
|
||||
Nil
|
||||
case _ =>
|
||||
log.error("Non-constant array range bounds")
|
||||
case (a, b) =>
|
||||
if (a.isEmpty) {
|
||||
log.error("Non-constant array range bounds", flc.start.position)
|
||||
}
|
||||
if (b.isEmpty) {
|
||||
log.error("Non-constant array range bounds", flc.`end`.position)
|
||||
}
|
||||
if (a.isEmpty){
|
||||
markAsConstantsThatShouldHaveBeenImportedEarlier(flc.start)
|
||||
}
|
||||
if (b.isEmpty){
|
||||
markAsConstantsThatShouldHaveBeenImportedEarlier(flc.`end`)
|
||||
}
|
||||
Nil
|
||||
|
||||
}
|
||||
@ -1677,11 +1719,11 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
log.error(s"Type $name cannot be used as an array index", l.position)
|
||||
w -> Constant.Zero
|
||||
case _ =>
|
||||
val constant = eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", stmt.position))
|
||||
val constant = eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", Some(l), stmt.position))
|
||||
w -> constant
|
||||
}
|
||||
case _ =>
|
||||
val constant = eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", stmt.position))
|
||||
val constant = eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", Some(l), stmt.position))
|
||||
w -> constant
|
||||
}
|
||||
lengthConst match {
|
||||
@ -1754,10 +1796,10 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
log.error(s"Type $name cannot be used as an array index", l.position)
|
||||
w -> Constant.Zero
|
||||
case _ =>
|
||||
w -> eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", stmt.position))
|
||||
w -> eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", Some(l), stmt.position))
|
||||
}
|
||||
case _ =>
|
||||
w -> eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", stmt.position))
|
||||
w -> eval(l).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant length", Some(l), stmt.position))
|
||||
}
|
||||
lengthConst match {
|
||||
case NumericConstant(ll, _) =>
|
||||
@ -1769,7 +1811,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
val length = contents.length
|
||||
if (length > 0xffff || length < 0) log.error(s"Array `${stmt.name}` has invalid length", stmt.position)
|
||||
val alignment = stmt.alignment.getOrElse(defaultArrayAlignment(options, length)) & e.alignment
|
||||
val address = stmt.address.map(a => eval(a).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant address", stmt.position)))
|
||||
val address = stmt.address.map(a => eval(a).getOrElse(errorConstant(s"Array `${stmt.name}` has non-constant address", Some(a), stmt.position)))
|
||||
for (element <- contents) {
|
||||
AbstractExpressionCompiler.checkAssignmentTypeLoosely(this, element, e)
|
||||
}
|
||||
@ -1830,11 +1872,23 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
val typ = get[VariableType](stmt.typ)
|
||||
val alignment = stmt.alignment.getOrElse(defaultVariableAlignment(options, typ.size)) & typ.alignment
|
||||
if (stmt.constant) {
|
||||
val invalidUsesBefore = constantsThatShouldHaveBeenImportedEarlier.get(stmt.name)
|
||||
if (invalidUsesBefore.nonEmpty) {
|
||||
log.info(s"The constant ${stmt.name} has been used before it was defined in a way that requires a definition beforehand", stmt.position)
|
||||
invalidUsesBefore.foreach{use =>
|
||||
if (use.isDefined) {
|
||||
log.info(s"here:", use)
|
||||
}
|
||||
}
|
||||
if (invalidUsesBefore(None)) {
|
||||
log.info("and in some other place or places.")
|
||||
}
|
||||
}
|
||||
if (stmt.stack) log.error(s"`$name` is a constant and cannot be on stack", position)
|
||||
if (stmt.register) log.error(s"`$name` is a constant and cannot be in a register", position)
|
||||
if (stmt.address.isDefined) log.error(s"`$name` is a constant and cannot have an address", position)
|
||||
if (stmt.initialValue.isEmpty) log.error(s"`$name` is a constant and requires a value", position)
|
||||
val rawConstantValue = stmt.initialValue.flatMap(eval).getOrElse(errorConstant(s"`$name` has a non-constant value", position)).quickSimplify
|
||||
val rawConstantValue = stmt.initialValue.flatMap(eval).getOrElse(errorConstant(s"`$name` has a non-constant value", stmt.initialValue, position)).quickSimplify
|
||||
rawConstantValue match {
|
||||
case NumericConstant(nv, _) if nv >= 2 && typ.size < 8 =>
|
||||
if (nv >= 1L.<<(8*typ.size)) {
|
||||
@ -1902,7 +1956,7 @@ class Environment(val parent: Option[Environment], val prefix: String, val cpuFa
|
||||
registerAddressConstant(v, stmt.position, options, Some(typ))
|
||||
(v, v.toAddress)
|
||||
})(a => {
|
||||
val addr = eval(a).getOrElse(errorConstant(s"Address of `$name` has a non-constant value", position))
|
||||
val addr = eval(a).getOrElse(errorConstant(s"Address of `$name` has a non-constant value", Some(a), position))
|
||||
val zp = addr match {
|
||||
case NumericConstant(n, _) => n < 0x100
|
||||
case _ => false
|
||||
|
@ -43,6 +43,7 @@ class ConsoleLogger extends Logger {
|
||||
override def info(msg: String, position: Option[Position] = None): Unit = {
|
||||
if (verbosity < 0) return
|
||||
println("INFO: " + f(position) + msg)
|
||||
printErrorContext(position)
|
||||
flushOutput()
|
||||
}
|
||||
|
||||
|
@ -84,7 +84,7 @@ object UnusedFunctions extends NodeOptimization {
|
||||
}
|
||||
}
|
||||
if (unusedFunctions.nonEmpty) {
|
||||
options.log.debug("Removing unused functions: " + unusedFunctions.mkString(", "))
|
||||
options.log.debug("Removing unused functions: " + unusedFunctions.toSeq.sorted.mkString(", "))
|
||||
optimize(removeFunctionsFromProgram(nodes, unusedFunctions), options)
|
||||
} else {
|
||||
nodes
|
||||
|
@ -23,7 +23,7 @@ object UnusedGlobalVariables extends NodeOptimization {
|
||||
val allReadVariables = resolveAliases(aliases, getAllReadVariables(nodes).toSet)
|
||||
val unusedVariables = allNonvolatileGlobalVariables -- allReadVariables
|
||||
if (unusedVariables.nonEmpty) {
|
||||
options.log.debug("Removing unused global variables: " + unusedVariables.mkString(", "))
|
||||
options.log.debug("Removing unused global variables: " + unusedVariables.toSeq.sorted.mkString(", "))
|
||||
}
|
||||
removeVariablesFromProgram(nodes, unusedVariables.flatMap(v => Set(v, v + ".hi", v + ".lo")))
|
||||
}
|
||||
|
@ -63,7 +63,7 @@ object UnusedLocalVariables extends NodeOptimization {
|
||||
}.flatMap(getAllReadVariables(_)).toSet
|
||||
val localsToRemove = allLocals.filterNot(allRead).toSet
|
||||
if (localsToRemove.nonEmpty) {
|
||||
log.debug("Removing unused local variables: " + localsToRemove.mkString(", "))
|
||||
log.debug("Removing unused local variables: " + localsToRemove.toSeq.sorted.mkString(", "))
|
||||
}
|
||||
removeVariables(statements, localsToRemove)
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user