Added semantic highlighting for registers to the ASM editor.

This commit is contained in:
Daan Vanden Bosch 2021-04-27 19:58:44 +02:00
parent 2c6f9ba680
commit b5a6ca6dc3
2 changed files with 205 additions and 96 deletions

View File

@ -326,25 +326,56 @@ class AsmAnalyser {
} }
} }
} else { } else {
visitLabelArguments( visitArgs(
ir.inst, ir.inst,
accept = { argSrcLoc -> positionInside(lineNo, col, argSrcLoc.coarse) }, processParam = { VisitAction.Go },
processImmediateArg = { label, _ -> processImmediateArg = { param, arg, argSrcLoc ->
results.addAll(getLabelDefinitionsAndReferences(label)) if (positionInside(lineNo, col, argSrcLoc.coarse)) {
VisitAction.Return (arg as? IntArg)?.let {
when (param.type) {
is LabelType -> {
results.addAll(
getLabelDefinitionsAndReferences(arg.value)
)
}
is RegRefType -> {
results.addAll(getRegisterReferences(arg.value))
}
else -> Unit
}
}
VisitAction.Return
} else {
VisitAction.Continue
}
}, },
processStackArg = { labels, pushInst, _ -> processStackArgSrcLoc = { _, argSrcLoc ->
// Filter out arg_pushr labels, because register values could be if (positionInside(lineNo, col, argSrcLoc.coarse)) {
// used for anything. VisitAction.Go
if (pushInst != null && } else {
pushInst.opcode.code != OP_ARG_PUSHR.code && VisitAction.Continue
labels.size == 1L }
) { },
results.addAll(getLabelDefinitionsAndReferences(labels[0]!!)) processStackArg = { param, _, pushInst, _ ->
if (pushInst != null) {
val pushArg = pushInst.args.firstOrNull()
if (pushArg is IntArg) {
if (pushInst.opcode.code == OP_ARG_PUSHR.code ||
param.type is RegRefType
) {
results.addAll(getRegisterReferences(pushArg.value))
} else if (param.type is LabelType) {
results.addAll(
getLabelDefinitionsAndReferences(pushArg.value)
)
}
}
} }
VisitAction.Return VisitAction.Return
}, }
) )
} }
} }
@ -411,95 +442,48 @@ class AsmAnalyser {
return null return null
} }
/** private fun getRegisterReferences(register: Int): List<AsmRange> {
* Visits all label arguments of [instruction] with their value. val results = mutableListOf<AsmRange>()
*/
private fun visitLabelArguments(
instruction: Instruction,
accept: (ArgSrcLoc) -> Boolean,
processImmediateArg: (label: Int, ArgSrcLoc) -> VisitAction,
processStackArg: (label: ValueSet, Instruction?, ArgSrcLoc) -> VisitAction,
) {
visitArgs(
instruction,
processParam = { if (it.type is LabelType) VisitAction.Go else VisitAction.Continue },
processImmediateArg = { arg, srcLoc ->
if (accept(srcLoc)) {
processImmediateArg((arg as IntArg).value, srcLoc)
} else VisitAction.Continue
},
processStackArgSrcLoc = { srcLoc ->
if (accept(srcLoc)) VisitAction.Go
else VisitAction.Continue
},
processStackArg = { value, pushInst, srcLoc ->
processStackArg(value, pushInst, srcLoc)
}
)
}
private enum class VisitAction { for (segment in bytecodeIr.segments) {
Go, Break, Continue, Return if (segment is InstructionSegment) {
} for (inst in segment.instructions) {
visitArgs(
inst,
processParam = { VisitAction.Go },
processImmediateArg = { param, arg, argSrcLoc ->
if (param.type is RegRefType &&
arg is IntArg &&
arg.value == register
) {
results.add(argSrcLoc.precise.toAsmRange())
}
/** VisitAction.Go
* Visits all arguments of [instruction], including stack arguments. },
*/ processStackArgSrcLoc = { param, _ ->
private fun visitArgs( if (param.type is RegRefType) VisitAction.Go
instruction: Instruction, else VisitAction.Continue
processParam: (Param) -> VisitAction, },
processImmediateArg: (Arg, ArgSrcLoc) -> VisitAction, processStackArg = { _, _, pushInst, argSrcLoc ->
processStackArgSrcLoc: (ArgSrcLoc) -> VisitAction, if (pushInst != null &&
processStackArg: (ValueSet, Instruction?, ArgSrcLoc) -> VisitAction, pushInst.opcode.code != OP_ARG_PUSHR.code
) { ) {
for ((paramIdx, param) in instruction.opcode.params.withIndex()) { val pushArg = pushInst.args.firstOrNull()
when (processParam(param)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break // Same as Stop.
VisitAction.Continue -> continue
VisitAction.Return -> return
}
if (instruction.opcode.stack !== StackInteraction.Pop) { if (pushArg is IntArg && pushArg.value == register) {
// Immediate arguments. results.add(argSrcLoc.precise.toAsmRange())
val args = instruction.getArgs(paramIdx) }
val argSrcLocs = instruction.getArgSrcLocs(paramIdx) }
for (i in 0 until min(args.size, argSrcLocs.size)) { VisitAction.Go
val arg = args[i] }
val srcLoc = argSrcLocs[i] )
when (processImmediateArg(arg, srcLoc)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break
VisitAction.Continue -> continue // Same as Down.
VisitAction.Return -> return
}
}
} else {
// Stack arguments.
val argSrcLocs = instruction.getArgSrcLocs(paramIdx)
for ((i, srcLoc) in argSrcLocs.withIndex()) {
when (processStackArgSrcLoc(srcLoc)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break
VisitAction.Continue -> continue
VisitAction.Return -> return
}
val (labelValues, pushInstruction) =
getStackValue(cfg, instruction, argSrcLocs.lastIndex - i)
when (processStackArg(labelValues, pushInstruction, srcLoc)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break
VisitAction.Continue -> continue // Same as Down.
VisitAction.Return -> return
}
} }
} }
} }
return results
} }
/** /**
@ -568,6 +552,101 @@ class AsmAnalyser {
return results return results
} }
/**
* Visits all label arguments of [instruction] with their value.
*/
private fun visitLabelArguments(
instruction: Instruction,
accept: (ArgSrcLoc) -> Boolean,
processImmediateArg: (label: Int, ArgSrcLoc) -> VisitAction,
processStackArg: (label: ValueSet, Instruction?, ArgSrcLoc) -> VisitAction,
) {
visitArgs(
instruction,
processParam = { if (it.type is LabelType) VisitAction.Go else VisitAction.Continue },
processImmediateArg = { _, arg, srcLoc ->
if (accept(srcLoc) && arg is IntArg) {
processImmediateArg(arg.value, srcLoc)
} else VisitAction.Continue
},
processStackArgSrcLoc = { _, srcLoc ->
if (accept(srcLoc)) VisitAction.Go
else VisitAction.Continue
},
processStackArg = { _, value, pushInst, srcLoc ->
processStackArg(value, pushInst, srcLoc)
}
)
}
private enum class VisitAction {
Go, Break, Continue, Return
}
/**
* Visits all arguments of [instruction], including stack arguments.
*/
private fun visitArgs(
instruction: Instruction,
processParam: (Param) -> VisitAction,
processImmediateArg: (Param, Arg, ArgSrcLoc) -> VisitAction,
processStackArgSrcLoc: (Param, ArgSrcLoc) -> VisitAction,
processStackArg: (Param, ValueSet, Instruction?, ArgSrcLoc) -> VisitAction,
) {
for ((paramIdx, param) in instruction.opcode.params.withIndex()) {
when (processParam(param)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break // Same as Stop.
VisitAction.Continue -> continue
VisitAction.Return -> return
}
if (instruction.opcode.stack !== StackInteraction.Pop) {
// Immediate arguments.
val args = instruction.getArgs(paramIdx)
val argSrcLocs = instruction.getArgSrcLocs(paramIdx)
for (i in 0 until min(args.size, argSrcLocs.size)) {
val arg = args[i]
val srcLoc = argSrcLocs[i]
when (processImmediateArg(param, arg, srcLoc)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break
VisitAction.Continue -> continue // Same as Down.
VisitAction.Return -> return
}
}
} else {
// Stack arguments.
val argSrcLocs = instruction.getArgSrcLocs(paramIdx)
// Never varargs.
for (srcLoc in argSrcLocs) {
when (processStackArgSrcLoc(param, srcLoc)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break
VisitAction.Continue -> continue
VisitAction.Return -> return
}
val (labelValues, pushInstruction) = getStackValue(
cfg,
instruction,
instruction.opcode.params.lastIndex - paramIdx,
)
when (processStackArg(param, labelValues, pushInstruction, srcLoc)) {
VisitAction.Go -> Unit // Keep going.
VisitAction.Break -> break
VisitAction.Continue -> continue // Same as Down.
VisitAction.Return -> return
}
}
}
}
}
private fun positionInside(lineNo: Int, col: Int, srcLoc: SrcLoc?): Boolean = private fun positionInside(lineNo: Int, col: Int, srcLoc: SrcLoc?): Boolean =
if (srcLoc == null) { if (srcLoc == null) {
false false

View File

@ -232,6 +232,36 @@ class AsmAnalyserTests : AssemblyWorkerTestSuite() {
} }
} }
@Test
fun getHighlights_for_register() = test {
val analyser = createAsmAnalyser(
"""
.code
100:
leti r13, 4031
set_floor_handler 0, r13
leti r17, 5379
npc_param_v3 r13, 4
""".trimIndent()
)
val requestId = 2999
for ((lineNo, col) in listOf(
Pair(3, 11),
Pair(4, 27),
Pair(6, 19),
)) {
val response = analyser.getHighlights(requestId, lineNo, col)
assertEquals(requestId, response.id)
assertEquals(3, response.result.size)
assertEquals(AsmRange(3, 10, 3, 13), response.result[0])
assertEquals(AsmRange(4, 26, 4, 29), response.result[1])
assertEquals(AsmRange(6, 18, 6, 21), response.result[2])
}
}
private fun createAsmAnalyser(asm: String): AsmAnalyser { private fun createAsmAnalyser(asm: String): AsmAnalyser {
val analyser = AsmAnalyser() val analyser = AsmAnalyser()
analyser.setAsm(asm.split("\n"), inlineStackArgs = true) analyser.setAsm(asm.split("\n"), inlineStackArgs = true)