Renamed object code to byte code.

This commit is contained in:
Daan Vanden Bosch 2020-10-29 00:20:58 +01:00
parent 924b084db4
commit 2e6b3c9cdb
8 changed files with 39 additions and 39 deletions

View File

@ -91,7 +91,7 @@ Features that are in ***bold italics*** are planned but not yet implemented.
- ***Spawn NPCs*** - ***Spawn NPCs***
- ***Reorder actions*** - ***Reorder actions***
## Script Object Code ## Script Byte Code
- Disassembler - Disassembler
- Assembler - Assembler

View File

@ -40,7 +40,7 @@ fun assemble(
private class Assembler(private val assembly: List<String>, private val manualStack: Boolean) { private class Assembler(private val assembly: List<String>, private val manualStack: Boolean) {
private var lineNo = 1 private var lineNo = 1
private lateinit var tokens: MutableList<Token> private lateinit var tokens: MutableList<Token>
private var objectCode: MutableList<Segment> = mutableListOf() private var ir: MutableList<Segment> = mutableListOf()
/** /**
* The current segment. * The current segment.
@ -113,7 +113,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
lineNo++ lineNo++
} }
return result.success(objectCode) return result.success(ir)
} }
private fun addInstruction( private fun addInstruction(
@ -133,7 +133,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
srcLoc = SegmentSrcLoc() srcLoc = SegmentSrcLoc()
) )
objectCode.add(segment!!) ir.add(segment!!)
} }
is InstructionSegment -> { is InstructionSegment -> {
@ -172,7 +172,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
srcLoc = SegmentSrcLoc() srcLoc = SegmentSrcLoc()
) )
objectCode.add(segment!!) ir.add(segment!!)
} }
is DataSegment -> { is DataSegment -> {
@ -200,7 +200,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
srcLoc = SegmentSrcLoc() srcLoc = SegmentSrcLoc()
) )
objectCode.add(segment!!) ir.add(segment!!)
} }
is StringSegment -> { is StringSegment -> {
@ -260,7 +260,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
val srcLoc = SrcLoc(lineNo, token.col, token.len) val srcLoc = SrcLoc(lineNo, token.col, token.len)
if (prevLineHadLabel) { if (prevLineHadLabel) {
val segment = objectCode.last() val segment = ir.last()
segment.labels.add(label) segment.labels.add(label)
segment.srcLoc.labels.add(srcLoc) segment.srcLoc.labels.add(srcLoc)
} }
@ -274,7 +274,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
srcLoc = SegmentSrcLoc(labels = mutableListOf(srcLoc)), srcLoc = SegmentSrcLoc(labels = mutableListOf(srcLoc)),
) )
objectCode.add(segment!!) ir.add(segment!!)
} }
if (nextToken != null) { if (nextToken != null) {
@ -293,7 +293,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
data = Buffer.withCapacity(0), data = Buffer.withCapacity(0),
srcLoc = SegmentSrcLoc(labels = mutableListOf(srcLoc)), srcLoc = SegmentSrcLoc(labels = mutableListOf(srcLoc)),
) )
objectCode.add(segment!!) ir.add(segment!!)
} }
if (nextToken != null) { if (nextToken != null) {
@ -312,7 +312,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
value = "", value = "",
srcLoc = SegmentSrcLoc(labels = mutableListOf(srcLoc)), srcLoc = SegmentSrcLoc(labels = mutableListOf(srcLoc)),
) )
objectCode.add(segment!!) ir.add(segment!!)
} }
if (nextToken != null) { if (nextToken != null) {
@ -509,7 +509,7 @@ private class Assembler(private val assembly: List<String>, private val manualSt
} }
/** /**
* @returns true if arguments can be translated to object code, possibly after truncation. False otherwise. * Returns true iff arguments can be translated to byte code, possibly after truncation.
*/ */
private fun parseArgs( private fun parseArgs(
params: List<Param>, params: List<Param>,

View File

@ -100,7 +100,7 @@ enum class SegmentType {
} }
/** /**
* Segment of object code. A segment starts with an instruction, byte or string character that is * Segment of byte code. A segment starts with an instruction, byte or string character that is
* referenced by one or more labels. The segment ends right before the next instruction, byte or * referenced by one or more labels. The segment ends right before the next instruction, byte or
* string character that is referenced by a label. * string character that is referenced by a label.
*/ */

View File

@ -129,11 +129,11 @@ enum class StackInteraction {
} }
/** /**
* Opcode for script object code. Invoked by instructions. * Opcode for script byte code. Invoked by instructions.
*/ */
class Opcode( class Opcode(
/** /**
* 1- Or 2-byte big-endian representation of this opcode as used in object code. * 1- Or 2-byte big-endian representation of this opcode as used in byte code.
*/ */
val code: Int, val code: Int,
/** /**

View File

@ -17,7 +17,7 @@ class BinFile(
val questName: String, val questName: String,
val shortDescription: String, val shortDescription: String,
val longDescription: String, val longDescription: String,
val objectCode: Buffer, val byteCode: Buffer,
val labelOffsets: IntArray, val labelOffsets: IntArray,
val shopItems: UIntArray, val shopItems: UIntArray,
) )
@ -40,18 +40,18 @@ enum class BinFormat {
} }
fun parseBin(cursor: Cursor): BinFile { fun parseBin(cursor: Cursor): BinFile {
val objectCodeOffset = cursor.int() val byteCodeOffset = cursor.int()
val labelOffsetTableOffset = cursor.int() // Relative offsets val labelOffsetTableOffset = cursor.int() // Relative offsets
val size = cursor.int() val size = cursor.int()
cursor.seek(4) // Always seems to be 0xFFFFFFFF. cursor.seek(4) // Always seems to be 0xFFFFFFFF.
val format = when (objectCodeOffset) { val format = when (byteCodeOffset) {
DC_GC_OBJECT_CODE_OFFSET -> BinFormat.DC_GC DC_GC_OBJECT_CODE_OFFSET -> BinFormat.DC_GC
PC_OBJECT_CODE_OFFSET -> BinFormat.PC PC_OBJECT_CODE_OFFSET -> BinFormat.PC
BB_OBJECT_CODE_OFFSET -> BinFormat.BB BB_OBJECT_CODE_OFFSET -> BinFormat.BB
else -> { else -> {
logger.warn { logger.warn {
"Object code at unexpected offset $objectCodeOffset, assuming file is a PC file." "Byte code at unexpected offset $byteCodeOffset, assuming file is a PC file."
} }
BinFormat.PC BinFormat.PC
} }
@ -100,9 +100,9 @@ fun parseBin(cursor: Cursor): BinFile {
.seekStart(labelOffsetTableOffset) .seekStart(labelOffsetTableOffset)
.intArray(labelOffsetCount) .intArray(labelOffsetCount)
val objectCode = cursor val byteCode = cursor
.seekStart(objectCodeOffset) .seekStart(byteCodeOffset)
.buffer(labelOffsetTableOffset - objectCodeOffset) .buffer(labelOffsetTableOffset - byteCodeOffset)
return BinFile( return BinFile(
format, format,
@ -111,7 +111,7 @@ fun parseBin(cursor: Cursor): BinFile {
questName, questName,
shortDescription, shortDescription,
longDescription, longDescription,
objectCode, byteCode,
labelOffsets, labelOffsets,
shopItems, shopItems,
) )

View File

@ -44,14 +44,14 @@ val BUILTIN_FUNCTIONS = setOf(
860, 860,
) )
fun parseObjectCode( fun parseByteCode(
objectCode: Buffer, byteCode: Buffer,
labelOffsets: IntArray, labelOffsets: IntArray,
entryLabels: Set<Int>, entryLabels: Set<Int>,
dcGcFormat: Boolean, dcGcFormat: Boolean,
lenient: Boolean, lenient: Boolean,
): PwResult<List<Segment>> { ): PwResult<List<Segment>> {
val cursor = BufferCursor(objectCode) val cursor = BufferCursor(byteCode)
val labelHolder = LabelHolder(labelOffsets) val labelHolder = LabelHolder(labelOffsets)
val result = PwResultBuilder<List<Segment>>(logger) val result = PwResultBuilder<List<Segment>>(logger)
val offsetToSegment = mutableMapOf<Int, Segment>() val offsetToSegment = mutableMapOf<Int, Segment>()
@ -138,7 +138,7 @@ fun parseObjectCode(
} }
} }
// Sanity check parsed object code. // Sanity check parsed byte code.
if (cursor.size != offset) { if (cursor.size != offset) {
result.addProblem( result.addProblem(
Severity.Error, Severity.Error,
@ -351,7 +351,7 @@ private fun parseSegment(
} }
} catch (e: Throwable) { } catch (e: Throwable) {
if (lenient) { if (lenient) {
logger.error(e) { "Couldn't fully parse object code segment." } logger.error(e) { "Couldn't fully parse byte code segment." }
} else { } else {
throw e throw e
} }

View File

@ -25,7 +25,7 @@ class Quest(
val npcs: List<QuestNpc>, val npcs: List<QuestNpc>,
val events: List<DatEvent>, val events: List<DatEvent>,
val datUnknowns: List<DatUnknown>, val datUnknowns: List<DatUnknown>,
val objectCode: List<Segment>, val byteCodeIr: List<Segment>,
val shopItems: UIntArray, val shopItems: UIntArray,
val mapDesignations: Map<Int, Int>, val mapDesignations: Map<Int, Int>,
) )
@ -59,30 +59,30 @@ fun parseBinDatToQuest(
// Initialize NPCs with random episode and correct it later. // Initialize NPCs with random episode and correct it later.
val npcs = dat.npcs.map { QuestNpc(Episode.I, it.areaId, it.data) } val npcs = dat.npcs.map { QuestNpc(Episode.I, it.areaId, it.data) }
// Extract episode and map designations from object code. // Extract episode and map designations from byte code.
var episode = Episode.I var episode = Episode.I
var mapDesignations = emptyMap<Int, Int>() var mapDesignations = emptyMap<Int, Int>()
val objectCodeResult = parseObjectCode( val parseByteCodeResult = parseByteCode(
bin.objectCode, bin.byteCode,
bin.labelOffsets, bin.labelOffsets,
extractScriptEntryPoints(objects, npcs), extractScriptEntryPoints(objects, npcs),
bin.format == BinFormat.DC_GC, bin.format == BinFormat.DC_GC,
lenient, lenient,
) )
rb.addResult(objectCodeResult) rb.addResult(parseByteCodeResult)
if (objectCodeResult !is Success) { if (parseByteCodeResult !is Success) {
return rb.failure() return rb.failure()
} }
val objectCode = objectCodeResult.value val byteCodeIr = parseByteCodeResult.value
if (objectCode.isEmpty()) { if (byteCodeIr.isEmpty()) {
rb.addProblem(Severity.Warning, "File contains no instruction labels.") rb.addProblem(Severity.Warning, "File contains no instruction labels.")
} else { } else {
val instructionSegments = objectCode.filterIsInstance<InstructionSegment>() val instructionSegments = byteCodeIr.filterIsInstance<InstructionSegment>()
var label0Segment: InstructionSegment? = null var label0Segment: InstructionSegment? = null
@ -117,7 +117,7 @@ fun parseBinDatToQuest(
npcs, npcs,
events = dat.events, events = dat.events,
datUnknowns = dat.unknowns, datUnknowns = dat.unknowns,
objectCode, byteCodeIr,
shopItems = bin.shopItems, shopItems = bin.shopItems,
mapDesignations, mapDesignations,
)) ))

View File

@ -9,7 +9,7 @@ import kotlin.test.Test
import kotlin.test.assertEquals import kotlin.test.assertEquals
import kotlin.test.assertTrue import kotlin.test.assertTrue
class ObjectCode { class ByteCode {
@Test @Test
fun minimal() { fun minimal() {
val buffer = Buffer.fromByteArray(ubyteArrayOf( val buffer = Buffer.fromByteArray(ubyteArrayOf(
@ -18,7 +18,7 @@ class ObjectCode {
0x01u // ret 0x01u // ret
).toByteArray()) ).toByteArray())
val result = parseObjectCode( val result = parseByteCode(
buffer, buffer,
labelOffsets = intArrayOf(0), labelOffsets = intArrayOf(0),
entryLabels = setOf(0), entryLabels = setOf(0),