Fixed various bugs.

This commit is contained in:
Daan Vanden Bosch 2020-10-29 19:09:23 +01:00
parent 2e6b3c9cdb
commit fdb3d5bbb6
21 changed files with 652 additions and 549 deletions

View File

@ -96,8 +96,8 @@
"enum": [
"any",
"byte",
"word",
"dword",
"short",
"int",
"float",
"label",
"instruction_label",

File diff suppressed because it is too large Load Diff

View File

@ -151,8 +151,8 @@ fun paramsToCode(params: List<Map<String, Any>>, indent: Int): String {
val type = when (param["type"]) {
"any" -> "AnyType()"
"byte" -> "ByteType"
"word" -> "WordType"
"dword" -> "DWordType"
"short" -> "ShortType"
"int" -> "IntType"
"float" -> "FloatType"
"label" -> "LabelType()"
"instruction_label" -> "ILabelType"

View File

@ -435,7 +435,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
)
}
is WordType,
is ShortType,
is LabelType,
is ILabelType,
is DLabelType,
@ -451,7 +451,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
)
}
is DWordType -> {
is IntType -> {
addInstruction(
OP_ARG_PUSHL,
listOf(arg),
@ -554,7 +554,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
match = true
parseInt(1, token, argAndTokens)
}
is WordType,
is ShortType,
is LabelType,
is ILabelType,
is DLabelType,
@ -564,7 +564,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
match = true
parseInt(2, token, argAndTokens)
}
is DWordType -> {
is IntType -> {
match = true
parseInt(4, token, argAndTokens)
}
@ -613,8 +613,8 @@ private class Assembler(private val assembly: List<String>, private val manualSt
val typeStr: String? = when (param.type) {
is ByteType -> "an 8-bit integer"
is WordType -> "a 16-bit integer"
is DWordType -> "a 32-bit integer"
is ShortType -> "a 16-bit integer"
is IntType -> "a 32-bit integer"
is FloatType -> "a float"
is LabelType -> "a label"

View File

@ -58,14 +58,14 @@ fun instructionSize(instruction: Instruction, dcGcFormat: Boolean): Int {
is RegTupRefType,
-> 1
is WordType,
is ShortType,
is LabelType,
is ILabelType,
is DLabelType,
is SLabelType,
-> 2
is DWordType,
is IntType,
is FloatType,
-> 4

View File

@ -29,12 +29,12 @@ object ByteType : ValueType()
/**
* 16-Bit integer.
*/
object WordType : ValueType()
object ShortType : ValueType()
/**
* 32-Bit integer.
*/
object DWordType : ValueType()
object IntType : ValueType()
/**
* 32-Bit floating point number.
@ -61,16 +61,16 @@ object DLabelType : LabelType()
*/
object SLabelType : LabelType()
/**
* String of arbitrary size.
*/
object StringType : LabelType()
/**
* Arbitrary amount of instruction labels.
*/
object ILabelVarType : LabelType()
/**
* String of arbitrary size.
*/
object StringType : ValueType()
/**
* Purely abstract super type of all reference types.
*/
@ -97,13 +97,6 @@ object RegRefVarType : RefType()
*/
object PointerType : AnyType()
const val MIN_SIGNED_DWORD_VALUE = Int.MIN_VALUE
const val MAX_SIGNED_DWORD_VALUE = Int.MAX_VALUE
const val MIN_UNSIGNED_DWORD_VALUE = UInt.MIN_VALUE
const val MAX_UNSIGNED_DWORD_VALUE = UInt.MAX_VALUE
const val MIN_DWORD_VALUE = MIN_SIGNED_DWORD_VALUE
const val MAX_DWORD_VALUE = MAX_UNSIGNED_DWORD_VALUE
enum class ParamAccess {
Read,
Write,

View File

@ -26,7 +26,7 @@ fun getRegisterValue(cfg: ControlFlowGraph, instruction: Instruction, register:
}
private class RegisterValueFinder {
var iterations = 0
private var iterations = 0
fun find(
path: MutableSet<BasicBlock>,

View File

@ -22,7 +22,7 @@ fun getStackValue(cfg: ControlFlowGraph, instruction: Instruction, position: Int
}
private class StackValueFinder {
var iterations = 0
private var iterations = 0
fun find(
path: MutableSet<BasicBlock>,
@ -54,7 +54,6 @@ private class StackValueFinder {
return getRegisterValue(cfg, instruction, args[0].value as Int)
} else {
pos--
break
}
}
@ -66,7 +65,6 @@ private class StackValueFinder {
return ValueSet.of(args[0].value as Int)
} else {
pos--
break
}
}
@ -78,11 +76,8 @@ private class StackValueFinder {
return ValueSet.all()
} else {
pos--
break
}
}
else -> break
}
}

View File

@ -112,6 +112,17 @@ protected constructor(protected val offset: Int) : WritableCursor {
return this
}
override fun writeByteArray(array: ByteArray): WritableCursor {
val len = array.size
requireSize(len)
for (i in 0 until len) {
writeByte(array[i])
}
return this
}
override fun writeIntArray(array: IntArray): WritableCursor {
val len = array.size
requireSize(4 * len)

View File

@ -125,6 +125,19 @@ class BufferCursor(
return array
}
override fun byteArray(n: Int): ByteArray {
requireSize(n)
val array = ByteArray(n)
for (i in 0 until n) {
array[i] = buffer.getByte(absolutePosition)
position++
}
return array
}
override fun intArray(n: Int): IntArray {
requireSize(4 * n)
@ -214,6 +227,11 @@ class BufferCursor(
return super.writeUIntArray(array)
}
override fun writeByteArray(array: ByteArray): WritableCursor {
ensureSpace(array.size)
return super.writeByteArray(array)
}
override fun writeIntArray(array: IntArray): WritableCursor {
ensureSpace(4 * array.size)
return super.writeIntArray(array)

View File

@ -95,6 +95,11 @@ interface Cursor {
*/
fun uIntArray(n: Int): UIntArray
/**
* Reads [n] signed 8-bit integers and increments position by [n].
*/
fun byteArray(n: Int): ByteArray
/**
* Reads [n] signed 32-bit integers and increments position by 4[n].
*/

View File

@ -64,6 +64,11 @@ interface WritableCursor : Cursor {
*/
fun writeUIntArray(array: UIntArray): WritableCursor
/**
* Writes an array of signed 8-bit integers and increments position by the array's length.
*/
fun writeByteArray(array: ByteArray): WritableCursor
/**
* Writes an array of signed 32-bit integers and increments position by four times the array's
* length.

View File

@ -490,15 +490,23 @@ private fun parseInstructionArguments(
is ByteType ->
args.add(Arg(cursor.uByte().toInt()))
is WordType ->
is ShortType ->
args.add(Arg(cursor.uShort().toInt()))
is DWordType ->
is IntType ->
args.add(Arg(cursor.int()))
is FloatType ->
args.add(Arg(cursor.float()))
// Ensure this case is before the LabelType case because ILabelVarType extends
// LabelType.
is ILabelVarType -> {
varargCount++
val argSize = cursor.uByte()
args.addAll(cursor.uShortArray(argSize.toInt()).map { Arg(it.toInt()) })
}
is LabelType,
is ILabelType,
is DLabelType,
@ -526,12 +534,6 @@ private fun parseInstructionArguments(
))
}
is ILabelVarType -> {
varargCount++
val argSize = cursor.uByte()
args.addAll(cursor.uShortArray(argSize.toInt()).map { Arg(it.toInt()) })
}
is RegRefType,
is RegTupRefType,
-> {

View File

@ -27,31 +27,31 @@ class DatEntity(
)
class DatEvent(
var id: UInt,
var sectionId: UShort,
var wave: UShort,
var delay: UShort,
var id: Int,
var sectionId: Short,
var wave: Short,
var delay: Short,
val actions: MutableList<DatEventAction>,
val areaId: Int,
val unknown: UShort,
val unknown: Short,
)
sealed class DatEventAction {
class SpawnNpcs(
val sectionId: UShort,
val appearFlag: UShort,
val sectionId: Short,
val appearFlag: Short,
) : DatEventAction()
class Unlock(
val doorId: UShort,
val doorId: Short,
) : DatEventAction()
class Lock(
val doorId: UShort,
val doorId: Short,
) : DatEventAction()
class TriggerEvent(
val eventId: UInt,
val eventId: Int,
) : DatEventAction()
}
@ -60,7 +60,7 @@ class DatUnknown(
val totalSize: Int,
val areaId: Int,
val entitiesSize: Int,
val data: UByteArray,
val data: ByteArray,
)
fun parseDat(cursor: Cursor): DatFile {
@ -95,7 +95,7 @@ fun parseDat(cursor: Cursor): DatFile {
totalSize,
areaId,
entitiesSize,
data = cursor.uByteArray(entitiesSize),
data = cursor.byteArray(entitiesSize),
))
}
}
@ -139,7 +139,7 @@ private fun parseEvents(cursor: Cursor, areaId: Int, events: MutableList<DatEven
cursor.seek(3) // Always 0
val eventType = cursor.byte()
require(eventType != (0x32).toByte()) {
require(eventType.toInt() != 0x32) {
"Can't parse challenge mode quests yet."
}
@ -148,12 +148,12 @@ private fun parseEvents(cursor: Cursor, areaId: Int, events: MutableList<DatEven
cursor.seekStart(16)
repeat(eventCount) {
val id = cursor.uInt()
val id = cursor.int()
cursor.seek(4) // Always 0x100
val sectionId = cursor.uShort()
val wave = cursor.uShort()
val delay = cursor.uShort()
val unknown = cursor.uShort() // "wavesetting"?
val sectionId = cursor.short()
val wave = cursor.short()
val delay = cursor.short()
val unknown = cursor.short() // "wavesetting"?
val eventActionsOffset = cursor.int()
val actions: MutableList<DatEventAction> =
@ -182,17 +182,17 @@ private fun parseEvents(cursor: Cursor, areaId: Int, events: MutableList<DatEven
}
}
var lastUByte: UByte = 0xffu
var lastByte: Byte = -1
while (actionsCursor.hasBytesLeft()) {
lastUByte = actionsCursor.uByte()
lastByte = actionsCursor.byte()
if (lastUByte != (0xffu).toUByte()) {
if (lastByte.toInt() != -1) {
break
}
}
if (lastUByte != (0xffu).toUByte()) {
if (lastByte.toInt() != -1) {
actionsCursor.seek(-1)
}
@ -209,23 +209,23 @@ private fun parseEventActions(cursor: Cursor): MutableList<DatEventAction> {
EVENT_ACTION_SPAWN_NPCS ->
actions.add(DatEventAction.SpawnNpcs(
sectionId = cursor.uShort(),
appearFlag = cursor.uShort(),
sectionId = cursor.short(),
appearFlag = cursor.short(),
))
EVENT_ACTION_UNLOCK ->
actions.add(DatEventAction.Unlock(
doorId = cursor.uShort(),
doorId = cursor.short(),
))
EVENT_ACTION_LOCK ->
actions.add(DatEventAction.Lock(
doorId = cursor.uShort(),
doorId = cursor.short(),
))
EVENT_ACTION_TRIGGER_EVENT ->
actions.add(DatEventAction.TriggerEvent(
eventId = cursor.uInt(),
eventId = cursor.int(),
))
else -> {

View File

@ -178,6 +178,17 @@ abstract class CursorTests {
testIntegerArrayRead(4, read, Endianness.Big)
}
@Test
fun byteArray() {
val read: Cursor.(Int) -> IntArray = { n ->
val arr = byteArray(n)
IntArray(n) { arr[it].toInt() }
}
testIntegerArrayRead(1, read, Endianness.Little)
testIntegerArrayRead(1, read, Endianness.Big)
}
@Test
fun intArray() {
val read: Cursor.(Int) -> IntArray = { n ->
@ -194,25 +205,16 @@ abstract class CursorTests {
read: Cursor.(Int) -> IntArray,
endianness: Endianness,
) {
// Generate array of the form 1, 2, 0xFF, 4, 5, 6, 7, 8.
// Generate array of the form 1, 2, 3, 4, 5, 6, 7, 8.
val bytes = ByteArray(8 * byteCount)
for (i in 0 until 8) {
if (i == 2) {
for (j in 0 until byteCount) {
bytes[i * byteCount + j] = (0xff).toByte()
}
} else {
if (endianness == Endianness.Little) {
bytes[i * byteCount] = (i + 1).toByte()
} else {
bytes[i * byteCount + byteCount - 1] = (i + 1).toByte()
}
}
}
var allOnes = 0
repeat(byteCount) { allOnes = ((allOnes shl 8) or 0xff) }
// Test cursor.
val cursor = createCursor(bytes, endianness)
@ -220,12 +222,12 @@ abstract class CursorTests {
val array1 = cursor.read(3)
assertEquals(1, array1[0])
assertEquals(2, array1[1])
assertEquals(allOnes, array1[2])
assertEquals(3, array1[2])
assertEquals(3 * byteCount, cursor.position)
cursor.seekStart(2 * byteCount)
val array2 = cursor.read(4)
assertEquals(allOnes, array2[0])
assertEquals(3, array2[0])
assertEquals(4, array2[1])
assertEquals(5, array2[2])
assertEquals(6, array2[3])

View File

@ -161,6 +161,20 @@ abstract class WritableCursorTests : CursorTests() {
testIntegerArrayWrite(4, read, write, Endianness.Big)
}
@Test
fun writeByteArray() {
val read: Cursor.(Int) -> IntArray = { n ->
val arr = byteArray(n)
IntArray(n) { arr[it].toInt() }
}
val write: WritableCursor.(IntArray) -> Unit = { a ->
writeByteArray(ByteArray(a.size) { a[it].toByte() })
}
testIntegerArrayWrite(1, read, write, Endianness.Little)
testIntegerArrayWrite(1, read, write, Endianness.Big)
}
@Test
fun writeIntArray() {
val read: Cursor.(Int) -> IntArray = { n ->

View File

@ -9,7 +9,7 @@ import kotlin.test.Test
import kotlin.test.assertEquals
import kotlin.test.assertTrue
class ByteCode {
class ByteCodeTests {
@Test
fun minimal() {
val buffer = Buffer.fromByteArray(ubyteArrayOf(

View File

@ -0,0 +1,16 @@
package world.phantasmal.lib.fileFormats.quest
import world.phantasmal.lib.test.asyncTest
import world.phantasmal.lib.test.readFile
import kotlin.test.Test
import kotlin.test.assertEquals
class DatTests {
@Test
fun parse_quest_towards_the_future() = asyncTest {
val dat = parseDat(readFile("/quest118_e_decompressed.dat"))
assertEquals(277, dat.objs.size)
assertEquals(216, dat.npcs.size)
}
}

View File

@ -1,6 +1,7 @@
package world.phantasmal.lib.fileFormats.quest
import world.phantasmal.core.Success
import world.phantasmal.lib.assembly.*
import world.phantasmal.lib.test.asyncTest
import world.phantasmal.lib.test.readFile
import kotlin.test.Test
@ -40,5 +41,33 @@ class QuestTests {
assertEquals(4, quest.mapDesignations[8])
assertEquals(4, quest.mapDesignations[10])
assertEquals(0, quest.mapDesignations[14])
val seg1 = quest.byteCodeIr[0]
assertTrue(seg1 is InstructionSegment)
assertTrue(0 in seg1.labels)
assertEquals(OP_SET_EPISODE, seg1.instructions[0].opcode)
assertEquals(0, seg1.instructions[0].args[0].value)
assertEquals(OP_ARG_PUSHL, seg1.instructions[1].opcode)
assertEquals(0, seg1.instructions[1].args[0].value)
assertEquals(OP_ARG_PUSHW, seg1.instructions[2].opcode)
assertEquals(150, seg1.instructions[2].args[0].value)
assertEquals(OP_SET_FLOOR_HANDLER, seg1.instructions[3].opcode)
val seg2 = quest.byteCodeIr[1]
assertTrue(seg2 is InstructionSegment)
assertTrue(1 in seg2.labels)
val seg3 = quest.byteCodeIr[2]
assertTrue(seg3 is InstructionSegment)
assertTrue(10 in seg3.labels)
val seg4 = quest.byteCodeIr[3]
assertTrue(seg4 is InstructionSegment)
assertTrue(150 in seg4.labels)
assertEquals(1, seg4.instructions.size)
assertEquals(OP_SWITCH_JMP, seg4.instructions[0].opcode)
assertEquals(0, seg4.instructions[0].args[0].value)
assertEquals(200, seg4.instructions[0].args[1].value)
assertEquals(201, seg4.instructions[0].args[2].value)
}
}

View File

@ -123,6 +123,19 @@ class ArrayBufferCursor(
return array
}
override fun byteArray(n: Int): ByteArray {
requireSize(n)
val array = ByteArray(n)
for (i in 0 until n) {
array[i] = dv.getInt8(absolutePosition)
position++
}
return array
}
override fun intArray(n: Int): IntArray {
requireSize(4 * n)