mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
Fixed some bugs in object code parsing.
This commit is contained in:
parent
3416ef5676
commit
0d05714730
@ -92,7 +92,7 @@ val generateOpcodes = tasks.register("generateOpcodes") {
|
|||||||
|
|
||||||
fun opcodeToCode(writer: PrintWriter, opcode: Map<String, Any>) {
|
fun opcodeToCode(writer: PrintWriter, opcode: Map<String, Any>) {
|
||||||
val code = (opcode["code"] as String).drop(2).toInt(16)
|
val code = (opcode["code"] as String).drop(2).toInt(16)
|
||||||
val codeStr = code.toString(16).padStart(2, '0')
|
val codeStr = code.toString(16).toUpperCase().padStart(2, '0')
|
||||||
val mnemonic = opcode["mnemonic"] as String? ?: "unknown_$codeStr"
|
val mnemonic = opcode["mnemonic"] as String? ?: "unknown_$codeStr"
|
||||||
val description = opcode["description"] as String?
|
val description = opcode["description"] as String?
|
||||||
val stack = opcode["stack"] as String?
|
val stack = opcode["stack"] as String?
|
||||||
@ -119,6 +119,7 @@ fun opcodeToCode(writer: PrintWriter, opcode: Map<String, Any>) {
|
|||||||
in 0xF900..0xF9FF -> "OPCODES_F9"
|
in 0xF900..0xF9FF -> "OPCODES_F9"
|
||||||
else -> error("Invalid opcode $codeStr ($mnemonic).")
|
else -> error("Invalid opcode $codeStr ($mnemonic).")
|
||||||
}
|
}
|
||||||
|
val indexStr = (code and 0xFF).toString(16).toUpperCase().padStart(2, '0')
|
||||||
|
|
||||||
writer.println(
|
writer.println(
|
||||||
"""
|
"""
|
||||||
@ -128,7 +129,7 @@ fun opcodeToCode(writer: PrintWriter, opcode: Map<String, Any>) {
|
|||||||
| ${description?.let { "\"$it\"" }},
|
| ${description?.let { "\"$it\"" }},
|
||||||
| $params,
|
| $params,
|
||||||
| $stackInteraction,
|
| $stackInteraction,
|
||||||
|).also { ${array}[0x$codeStr] = it }""".trimMargin()
|
|).also { ${array}[0x$indexStr] = it }""".trimMargin()
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6,7 +6,7 @@ import world.phantasmal.lib.assembly.*
|
|||||||
private val logger = KotlinLogging.logger {}
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Computes the possible values of a stack element at the nth position from the top right before a
|
* Computes the possible values of a stack element at the nth position from the top, right before a
|
||||||
* specific instruction.
|
* specific instruction.
|
||||||
*/
|
*/
|
||||||
fun getStackValue(cfg: ControlFlowGraph, instruction: Instruction, position: Int): ValueSet {
|
fun getStackValue(cfg: ControlFlowGraph, instruction: Instruction, position: Int): ValueSet {
|
||||||
|
@ -106,8 +106,14 @@ expect class Buffer {
|
|||||||
fun fill(value: Byte): Buffer
|
fun fill(value: Byte): Buffer
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
/**
|
||||||
|
* Returns a new buffer the given initial capacity and size 0.
|
||||||
|
*/
|
||||||
fun withCapacity(initialCapacity: Int, endianness: Endianness = Endianness.Little): Buffer
|
fun withCapacity(initialCapacity: Int, endianness: Endianness = Endianness.Little): Buffer
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a new buffer with an initial size and capacity of [initialSize].
|
||||||
|
*/
|
||||||
fun withSize(initialSize: Int, endianness: Endianness = Endianness.Little): Buffer
|
fun withSize(initialSize: Int, endianness: Endianness = Endianness.Little): Buffer
|
||||||
|
|
||||||
fun fromByteArray(array: ByteArray, endianness: Endianness = Endianness.Little): Buffer
|
fun fromByteArray(array: ByteArray, endianness: Endianness = Endianness.Little): Buffer
|
||||||
|
@ -6,6 +6,7 @@ import world.phantasmal.core.PwResultBuilder
|
|||||||
import world.phantasmal.core.Severity
|
import world.phantasmal.core.Severity
|
||||||
import world.phantasmal.lib.assembly.*
|
import world.phantasmal.lib.assembly.*
|
||||||
import world.phantasmal.lib.assembly.dataFlowAnalysis.ControlFlowGraph
|
import world.phantasmal.lib.assembly.dataFlowAnalysis.ControlFlowGraph
|
||||||
|
import world.phantasmal.lib.assembly.dataFlowAnalysis.ValueSet
|
||||||
import world.phantasmal.lib.assembly.dataFlowAnalysis.getRegisterValue
|
import world.phantasmal.lib.assembly.dataFlowAnalysis.getRegisterValue
|
||||||
import world.phantasmal.lib.assembly.dataFlowAnalysis.getStackValue
|
import world.phantasmal.lib.assembly.dataFlowAnalysis.getStackValue
|
||||||
import world.phantasmal.lib.buffer.Buffer
|
import world.phantasmal.lib.buffer.Buffer
|
||||||
@ -22,6 +23,9 @@ val SEGMENT_PRIORITY = mapOf(
|
|||||||
SegmentType.Data to 0,
|
SegmentType.Data to 0,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* These functions are built into the client and can optionally be overridden on BB.
|
||||||
|
*/
|
||||||
val BUILTIN_FUNCTIONS = setOf(
|
val BUILTIN_FUNCTIONS = setOf(
|
||||||
60,
|
60,
|
||||||
70,
|
70,
|
||||||
@ -45,8 +49,8 @@ fun parseObjectCode(
|
|||||||
objectCode: Buffer,
|
objectCode: Buffer,
|
||||||
labelOffsets: IntArray,
|
labelOffsets: IntArray,
|
||||||
entryLabels: Set<Int>,
|
entryLabels: Set<Int>,
|
||||||
lenient: Boolean,
|
|
||||||
dcGcFormat: Boolean,
|
dcGcFormat: Boolean,
|
||||||
|
lenient: Boolean,
|
||||||
): PwResult<List<Segment>> {
|
): PwResult<List<Segment>> {
|
||||||
val cursor = BufferCursor(objectCode)
|
val cursor = BufferCursor(objectCode)
|
||||||
val labelHolder = LabelHolder(labelOffsets)
|
val labelHolder = LabelHolder(labelOffsets)
|
||||||
@ -215,27 +219,35 @@ private fun findAndParseSegments(
|
|||||||
|
|
||||||
is RegTupRefType -> {
|
is RegTupRefType -> {
|
||||||
// Never on the stack.
|
// Never on the stack.
|
||||||
val arg = instruction.args[i]
|
var firstRegister: ValueSet? = null
|
||||||
|
|
||||||
for (j in param.type.registerTuples.indices) {
|
for (j in param.type.registerTuples.indices) {
|
||||||
val regTup = param.type.registerTuples[j]
|
val regTup = param.type.registerTuples[j]
|
||||||
|
|
||||||
if (regTup.type is ILabelType) {
|
if (regTup.type is ILabelType) {
|
||||||
val labelValues = getRegisterValue(
|
if (firstRegister == null) {
|
||||||
cfg,
|
firstRegister = getStackValue(cfg, instruction, i)
|
||||||
instruction,
|
}
|
||||||
arg.value as Int + j,
|
|
||||||
)
|
|
||||||
|
|
||||||
if (labelValues.size <= 10) {
|
for (reg in firstRegister) {
|
||||||
for (label in labelValues) {
|
val labelValues = getRegisterValue(
|
||||||
newLabels[label] = SegmentType.Instructions
|
cfg,
|
||||||
|
instruction,
|
||||||
|
reg + j,
|
||||||
|
)
|
||||||
|
|
||||||
|
if (labelValues.size <= 10) {
|
||||||
|
for (label in labelValues) {
|
||||||
|
newLabels[label] = SegmentType.Instructions
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
i++
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -378,7 +390,7 @@ private fun parseInstructionsSegment(
|
|||||||
val mainOpcode = cursor.u8()
|
val mainOpcode = cursor.u8()
|
||||||
|
|
||||||
val fullOpcode = when (mainOpcode.toInt()) {
|
val fullOpcode = when (mainOpcode.toInt()) {
|
||||||
0xF8, 0xF9 -> ((mainOpcode.toUInt() shl 8) or cursor.u8().toUInt()).toInt()
|
0xF8, 0xF9 -> ((mainOpcode.toInt() shl 8) or cursor.u8().toInt())
|
||||||
else -> mainOpcode.toInt()
|
else -> mainOpcode.toInt()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -479,6 +491,8 @@ private fun parseInstructionArguments(
|
|||||||
val args = mutableListOf<Arg>()
|
val args = mutableListOf<Arg>()
|
||||||
|
|
||||||
if (opcode.stack != StackInteraction.Pop) {
|
if (opcode.stack != StackInteraction.Pop) {
|
||||||
|
var varargCount = 0
|
||||||
|
|
||||||
for (param in opcode.params) {
|
for (param in opcode.params) {
|
||||||
when (param.type) {
|
when (param.type) {
|
||||||
is ByteType ->
|
is ByteType ->
|
||||||
@ -521,6 +535,7 @@ private fun parseInstructionArguments(
|
|||||||
}
|
}
|
||||||
|
|
||||||
is ILabelVarType -> {
|
is ILabelVarType -> {
|
||||||
|
varargCount++
|
||||||
val argSize = cursor.u8()
|
val argSize = cursor.u8()
|
||||||
args.addAll(cursor.u16Array(argSize.toInt()).map { Arg(it.toInt()) })
|
args.addAll(cursor.u16Array(argSize.toInt()).map { Arg(it.toInt()) })
|
||||||
}
|
}
|
||||||
@ -532,6 +547,7 @@ private fun parseInstructionArguments(
|
|||||||
}
|
}
|
||||||
|
|
||||||
is RegRefVarType -> {
|
is RegRefVarType -> {
|
||||||
|
varargCount++
|
||||||
val argSize = cursor.u8()
|
val argSize = cursor.u8()
|
||||||
args.addAll(cursor.u8Array(argSize.toInt()).map { Arg(it.toInt()) })
|
args.addAll(cursor.u8Array(argSize.toInt()).map { Arg(it.toInt()) })
|
||||||
}
|
}
|
||||||
@ -539,6 +555,12 @@ private fun parseInstructionArguments(
|
|||||||
else -> error("Parameter type ${param.type} not implemented.")
|
else -> error("Parameter type ${param.type} not implemented.")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val minExpectedArgs = opcode.params.size - varargCount
|
||||||
|
|
||||||
|
check(args.size >= minExpectedArgs) {
|
||||||
|
"Expected to parse at least $minExpectedArgs, only parsed ${args.size}."
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return args
|
return args
|
||||||
|
@ -67,8 +67,8 @@ fun parseBinDatToQuest(
|
|||||||
bin.objectCode,
|
bin.objectCode,
|
||||||
bin.labelOffsets,
|
bin.labelOffsets,
|
||||||
extractScriptEntryPoints(objects, npcs),
|
extractScriptEntryPoints(objects, npcs),
|
||||||
lenient,
|
|
||||||
bin.format == BinFormat.DC_GC,
|
bin.format == BinFormat.DC_GC,
|
||||||
|
lenient,
|
||||||
)
|
)
|
||||||
|
|
||||||
rb.addResult(objectCodeResult)
|
rb.addResult(objectCodeResult)
|
||||||
|
@ -0,0 +1,43 @@
|
|||||||
|
package world.phantasmal.lib.fileFormats.quest
|
||||||
|
|
||||||
|
import world.phantasmal.core.Success
|
||||||
|
import world.phantasmal.lib.assembly.InstructionSegment
|
||||||
|
import world.phantasmal.lib.assembly.OP_BB_MAP_DESIGNATE
|
||||||
|
import world.phantasmal.lib.assembly.OP_SET_EPISODE
|
||||||
|
import world.phantasmal.lib.buffer.Buffer
|
||||||
|
import kotlin.test.Test
|
||||||
|
import kotlin.test.assertEquals
|
||||||
|
import kotlin.test.assertTrue
|
||||||
|
|
||||||
|
class ObjectCode {
|
||||||
|
@Test
|
||||||
|
fun minimal() {
|
||||||
|
val buffer = Buffer.fromByteArray(ubyteArrayOf(
|
||||||
|
0xF8u, 0xBCu, 0x01u, 0x00u, 0x00u, 0x00u, // set_episode 1
|
||||||
|
0xF9u, 0x51u, 0x03u, 0x15u, 0x00u, 0x02u, 0x00u, // bb_map_designate 3, 21, 2, 0
|
||||||
|
0x01u // ret
|
||||||
|
).toByteArray())
|
||||||
|
|
||||||
|
val result = parseObjectCode(
|
||||||
|
buffer,
|
||||||
|
labelOffsets = intArrayOf(0),
|
||||||
|
entryLabels = setOf(0),
|
||||||
|
dcGcFormat = false,
|
||||||
|
lenient = false
|
||||||
|
)
|
||||||
|
|
||||||
|
assertTrue(result is Success)
|
||||||
|
assertTrue(result.problems.isEmpty())
|
||||||
|
|
||||||
|
val segments = result.value
|
||||||
|
val segment = segments[0]
|
||||||
|
|
||||||
|
assertTrue(segment is InstructionSegment)
|
||||||
|
assertEquals(OP_SET_EPISODE, segment.instructions[0].opcode)
|
||||||
|
assertEquals(1, segment.instructions[0].args[0].value)
|
||||||
|
assertEquals(OP_BB_MAP_DESIGNATE, segment.instructions[1].opcode)
|
||||||
|
assertEquals(3, segment.instructions[1].args[0].value)
|
||||||
|
assertEquals(21, segment.instructions[1].args[1].value)
|
||||||
|
assertEquals(2, segment.instructions[1].args[2].value)
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user