mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
Added semantic highlighting for registers to the ASM editor.
This commit is contained in:
parent
2c6f9ba680
commit
b5a6ca6dc3
@ -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
|
||||||
|
@ -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)
|
||||||
|
Loading…
Reference in New Issue
Block a user