mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Only one bug remains after opcode typing changes.
This commit is contained in:
parent
8e5472044b
commit
dbe1b06fa6
@ -98,6 +98,7 @@
|
|||||||
"data_label",
|
"data_label",
|
||||||
"string_label",
|
"string_label",
|
||||||
"string",
|
"string",
|
||||||
|
"instruction_label_var",
|
||||||
"reg_ref",
|
"reg_ref",
|
||||||
"reg_tup_ref",
|
"reg_tup_ref",
|
||||||
"reg_ref_var",
|
"reg_ref_var",
|
||||||
|
@ -602,7 +602,7 @@ opcodes:
|
|||||||
reg_tup:
|
reg_tup:
|
||||||
- type: dword
|
- type: dword
|
||||||
access: read
|
access: read
|
||||||
- type: instruction_label
|
- type: instruction_label_var
|
||||||
|
|
||||||
- code: 0x41
|
- code: 0x41
|
||||||
mnemonic: switch_call
|
mnemonic: switch_call
|
||||||
@ -611,7 +611,7 @@ opcodes:
|
|||||||
reg_tup:
|
reg_tup:
|
||||||
- type: dword
|
- type: dword
|
||||||
access: read
|
access: read
|
||||||
- type: instruction_label
|
- type: instruction_label_var
|
||||||
|
|
||||||
- code: 0x42
|
- code: 0x42
|
||||||
mnemonic: stack_push
|
mnemonic: stack_push
|
||||||
@ -676,6 +676,23 @@ opcodes:
|
|||||||
- type: word
|
- type: word
|
||||||
stack: push
|
stack: push
|
||||||
|
|
||||||
|
- code: 0x4c
|
||||||
|
mnemonic: arg_pusha
|
||||||
|
doc: Pushes the memory address of the given register onto the stack. Not used by Sega.
|
||||||
|
params:
|
||||||
|
- type: reg_tup_ref
|
||||||
|
reg_tup:
|
||||||
|
- type: any
|
||||||
|
access: read
|
||||||
|
stack: push
|
||||||
|
|
||||||
|
- code: 0x4d
|
||||||
|
mnemonic: arg_pusho
|
||||||
|
doc: Pushes the memory address of the given label onto the stack. Not used by Sega.
|
||||||
|
params:
|
||||||
|
- type: label
|
||||||
|
stack: push
|
||||||
|
|
||||||
- code: 0x4e
|
- code: 0x4e
|
||||||
mnemonic: arg_pushs
|
mnemonic: arg_pushs
|
||||||
doc: Pushes the given value onto the stack.
|
doc: Pushes the given value onto the stack.
|
||||||
@ -2812,14 +2829,8 @@ opcodes:
|
|||||||
- code: 0xf8dc
|
- code: 0xf8dc
|
||||||
mnemonic: npc_action_string
|
mnemonic: npc_action_string
|
||||||
params:
|
params:
|
||||||
- type: reg_tup_ref
|
- type: dword
|
||||||
reg_tup: # TODO: determine type and access
|
- type: dword
|
||||||
- type: any
|
|
||||||
access: read
|
|
||||||
- type: reg_tup_ref
|
|
||||||
reg_tup: # TODO: determine type and access
|
|
||||||
- type: any
|
|
||||||
access: read
|
|
||||||
- type: string_label
|
- type: string_label
|
||||||
|
|
||||||
- code: 0xf8dd
|
- code: 0xf8dd
|
||||||
|
@ -144,7 +144,7 @@ function params_to_code(params: any[]) {
|
|||||||
|
|
||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case "any":
|
case "any":
|
||||||
type = "TYPE";
|
type = "TYPE_ANY";
|
||||||
break;
|
break;
|
||||||
case "byte":
|
case "byte":
|
||||||
type = "TYPE_BYTE";
|
type = "TYPE_BYTE";
|
||||||
@ -173,6 +173,9 @@ function params_to_code(params: any[]) {
|
|||||||
case "string":
|
case "string":
|
||||||
type = "TYPE_STRING";
|
type = "TYPE_STRING";
|
||||||
break;
|
break;
|
||||||
|
case "instruction_label_var":
|
||||||
|
type = "TYPE_I_LABEL_VAR";
|
||||||
|
break;
|
||||||
case "reg_ref":
|
case "reg_ref":
|
||||||
type = "TYPE_REG_REF";
|
type = "TYPE_REG_REF";
|
||||||
break;
|
break;
|
||||||
|
@ -26,6 +26,20 @@ export abstract class AbstractWritableCursor extends AbstractCursor implements W
|
|||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
write_i8(value: number): this {
|
||||||
|
this.ensure_size(1);
|
||||||
|
this.dv.setInt8(this.position, value);
|
||||||
|
this._position += 1;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
write_i16(value: number): this {
|
||||||
|
this.ensure_size(2);
|
||||||
|
this.dv.setInt16(this.position, value, this.little_endian);
|
||||||
|
this._position += 2;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
write_i32(value: number): this {
|
write_i32(value: number): this {
|
||||||
this.ensure_size(4);
|
this.ensure_size(4);
|
||||||
this.dv.setInt32(this.position, value, this.little_endian);
|
this.dv.setInt32(this.position, value, this.little_endian);
|
||||||
|
@ -113,6 +113,8 @@ function test_integer_write(method_name: string): void {
|
|||||||
test_integer_write("write_u8");
|
test_integer_write("write_u8");
|
||||||
test_integer_write("write_u16");
|
test_integer_write("write_u16");
|
||||||
test_integer_write("write_u32");
|
test_integer_write("write_u32");
|
||||||
|
test_integer_write("write_i8");
|
||||||
|
test_integer_write("write_i16");
|
||||||
test_integer_write("write_i32");
|
test_integer_write("write_i32");
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -22,6 +22,16 @@ export interface WritableCursor extends Cursor {
|
|||||||
*/
|
*/
|
||||||
write_u32(value: number): this;
|
write_u32(value: number): this;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes a signed 8-bit integer and increments position by 1.
|
||||||
|
*/
|
||||||
|
write_i8(value: number): this;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes a signed 16-bit integer and increments position by 2.
|
||||||
|
*/
|
||||||
|
write_i16(value: number): this;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Writes a signed 32-bit integer and increments position by 4.
|
* Writes a signed 32-bit integer and increments position by 4.
|
||||||
*/
|
*/
|
||||||
|
@ -7,6 +7,7 @@ import {
|
|||||||
InstructionSegment,
|
InstructionSegment,
|
||||||
Segment,
|
Segment,
|
||||||
SegmentType,
|
SegmentType,
|
||||||
|
StringSegment,
|
||||||
} from "../../../scripting/instructions";
|
} from "../../../scripting/instructions";
|
||||||
import {
|
import {
|
||||||
Opcode,
|
Opcode,
|
||||||
@ -48,7 +49,8 @@ export class BinFile {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const SEGMENT_PRIORITY: number[] = [];
|
const SEGMENT_PRIORITY: number[] = [];
|
||||||
SEGMENT_PRIORITY[SegmentType.Instructions] = 1;
|
SEGMENT_PRIORITY[SegmentType.Instructions] = 2;
|
||||||
|
SEGMENT_PRIORITY[SegmentType.String] = 1;
|
||||||
SEGMENT_PRIORITY[SegmentType.Data] = 0;
|
SEGMENT_PRIORITY[SegmentType.Data] = 0;
|
||||||
|
|
||||||
export function parse_bin(
|
export function parse_bin(
|
||||||
@ -300,6 +302,9 @@ function parse_object_code(
|
|||||||
case SegmentType.Data:
|
case SegmentType.Data:
|
||||||
offset += segment.data.byteLength;
|
offset += segment.data.byteLength;
|
||||||
break;
|
break;
|
||||||
|
case SegmentType.String:
|
||||||
|
offset += 2 * segment.value.length + 2;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`${SegmentType[segment!.type]} not implemented.`);
|
throw new Error(`${SegmentType[segment!.type]} not implemented.`);
|
||||||
}
|
}
|
||||||
@ -386,6 +391,9 @@ function parse_segment(
|
|||||||
case SegmentType.Data:
|
case SegmentType.Data:
|
||||||
parse_data_segment(offset_to_segment, cursor, end_offset, labels);
|
parse_data_segment(offset_to_segment, cursor, end_offset, labels);
|
||||||
break;
|
break;
|
||||||
|
case SegmentType.String:
|
||||||
|
parse_string_segment(offset_to_segment, cursor, end_offset, labels);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Segment type ${SegmentType[type]} not implemented.`);
|
throw new Error(`Segment type ${SegmentType[type]} not implemented.`);
|
||||||
}
|
}
|
||||||
@ -462,11 +470,11 @@ function parse_instructions_segment(
|
|||||||
: instruction.args;
|
: instruction.args;
|
||||||
|
|
||||||
if (opcode.stack === StackInteraction.Push) {
|
if (opcode.stack === StackInteraction.Push) {
|
||||||
for (const arg of args) {
|
stack.push(...args);
|
||||||
stack.push(arg);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
for (let i = 0; i < params.length && i < args.length; i++) {
|
const len = Math.min(params.length, args.length);
|
||||||
|
|
||||||
|
for (let i = 0; i < len; i++) {
|
||||||
const param_type = params[i].type;
|
const param_type = params[i].type;
|
||||||
const label = args[i].value;
|
const label = args[i].value;
|
||||||
let segment_type: SegmentType;
|
let segment_type: SegmentType;
|
||||||
@ -474,6 +482,22 @@ function parse_instructions_segment(
|
|||||||
switch (param_type) {
|
switch (param_type) {
|
||||||
case TYPE_I_LABEL:
|
case TYPE_I_LABEL:
|
||||||
segment_type = SegmentType.Instructions;
|
segment_type = SegmentType.Instructions;
|
||||||
|
break;
|
||||||
|
case TYPE_I_LABEL_VAR:
|
||||||
|
segment_type = SegmentType.Instructions;
|
||||||
|
|
||||||
|
// Eat all remaining arguments.
|
||||||
|
for (; i < args.length; i++) {
|
||||||
|
parse_segment(
|
||||||
|
offset_to_segment,
|
||||||
|
label_holder,
|
||||||
|
cursor,
|
||||||
|
args[i].value,
|
||||||
|
segment_type,
|
||||||
|
lenient
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case TYPE_D_LABEL:
|
case TYPE_D_LABEL:
|
||||||
segment_type = SegmentType.Data;
|
segment_type = SegmentType.Data;
|
||||||
@ -484,11 +508,6 @@ function parse_instructions_segment(
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!Number.isInteger(label)) {
|
|
||||||
logger.error(`Expected label reference but got ${label}.`);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
parse_segment(
|
parse_segment(
|
||||||
offset_to_segment,
|
offset_to_segment,
|
||||||
label_holder,
|
label_holder,
|
||||||
@ -502,11 +521,10 @@ function parse_instructions_segment(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Recurse on label drop-through.
|
// Recurse on label drop-through.
|
||||||
if (
|
if (next_label != undefined && instructions.length) {
|
||||||
next_label != undefined &&
|
const last_opcode = instructions[instructions.length - 1].opcode;
|
||||||
instructions.length &&
|
|
||||||
instructions[instructions.length - 1].opcode !== Opcode.RET
|
if (last_opcode !== Opcode.RET && last_opcode !== Opcode.JMP) {
|
||||||
) {
|
|
||||||
parse_segment(
|
parse_segment(
|
||||||
offset_to_segment,
|
offset_to_segment,
|
||||||
label_holder,
|
label_holder,
|
||||||
@ -517,6 +535,7 @@ function parse_instructions_segment(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function parse_data_segment(
|
function parse_data_segment(
|
||||||
offset_to_segment: Map<number, Segment>,
|
offset_to_segment: Map<number, Segment>,
|
||||||
@ -533,9 +552,25 @@ function parse_data_segment(
|
|||||||
offset_to_segment.set(start_offset, segment);
|
offset_to_segment.set(start_offset, segment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function parse_string_segment(
|
||||||
|
offset_to_segment: Map<number, Segment>,
|
||||||
|
cursor: Cursor,
|
||||||
|
end_offset: number,
|
||||||
|
labels: number[]
|
||||||
|
) {
|
||||||
|
const start_offset = cursor.position;
|
||||||
|
const segment: StringSegment = {
|
||||||
|
type: SegmentType.String,
|
||||||
|
labels,
|
||||||
|
value: cursor.string_utf16(end_offset - start_offset, true, true),
|
||||||
|
};
|
||||||
|
offset_to_segment.set(start_offset, segment);
|
||||||
|
}
|
||||||
|
|
||||||
function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
||||||
const args: Arg[] = [];
|
const args: Arg[] = [];
|
||||||
|
|
||||||
|
if (opcode.stack !== StackInteraction.Pop) {
|
||||||
for (const param of opcode.params) {
|
for (const param of opcode.params) {
|
||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case TYPE_BYTE:
|
case TYPE_BYTE:
|
||||||
@ -560,7 +595,11 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
|||||||
{
|
{
|
||||||
const start_pos = cursor.position;
|
const start_pos = cursor.position;
|
||||||
args.push({
|
args.push({
|
||||||
value: cursor.string_utf16(Math.min(4096, cursor.bytes_left), true, false),
|
value: cursor.string_utf16(
|
||||||
|
Math.min(4096, cursor.bytes_left),
|
||||||
|
true,
|
||||||
|
false
|
||||||
|
),
|
||||||
size: cursor.position - start_pos,
|
size: cursor.position - start_pos,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -589,6 +628,7 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode): Arg[] {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return args;
|
return args;
|
||||||
}
|
}
|
||||||
@ -616,6 +656,7 @@ function write_object_code(
|
|||||||
|
|
||||||
cursor.write_u8(opcode.code & 0xff);
|
cursor.write_u8(opcode.code & 0xff);
|
||||||
|
|
||||||
|
if (opcode.stack !== StackInteraction.Pop) {
|
||||||
for (let i = 0; i < opcode.params.length; i++) {
|
for (let i = 0; i < opcode.params.length; i++) {
|
||||||
const param = opcode.params[i];
|
const param = opcode.params[i];
|
||||||
const args = instruction.param_to_args[i];
|
const args = instruction.param_to_args[i];
|
||||||
@ -623,10 +664,18 @@ function write_object_code(
|
|||||||
|
|
||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case TYPE_BYTE:
|
case TYPE_BYTE:
|
||||||
|
if (arg.value >= 0) {
|
||||||
cursor.write_u8(arg.value);
|
cursor.write_u8(arg.value);
|
||||||
|
} else {
|
||||||
|
cursor.write_i8(arg.value);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case TYPE_WORD:
|
case TYPE_WORD:
|
||||||
|
if (arg.value >= 0) {
|
||||||
cursor.write_u16(arg.value);
|
cursor.write_u16(arg.value);
|
||||||
|
} else {
|
||||||
|
cursor.write_i16(arg.value);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case TYPE_DWORD:
|
case TYPE_DWORD:
|
||||||
if (arg.value >= 0) {
|
if (arg.value >= 0) {
|
||||||
@ -664,11 +713,16 @@ function write_object_code(
|
|||||||
cursor.write_u8(arg.value);
|
cursor.write_u8(arg.value);
|
||||||
} else {
|
} else {
|
||||||
// TYPE_ANY and TYPE_POINTER cannot be serialized.
|
// TYPE_ANY and TYPE_POINTER cannot be serialized.
|
||||||
throw new Error(`Parameter type ${param.type} not implemented.`);
|
throw new Error(
|
||||||
|
`Parameter type ${param.type} not implemented.`
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
} else if (segment.type === SegmentType.String) {
|
||||||
|
cursor.write_string_utf16(segment.value, 2 * segment.value.length + 2);
|
||||||
} else {
|
} else {
|
||||||
cursor.write_cursor(new ArrayBufferCursor(segment.data, cursor.endianness));
|
cursor.write_cursor(new ArrayBufferCursor(segment.data, cursor.endianness));
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,8 @@ import {
|
|||||||
QuestNpc,
|
QuestNpc,
|
||||||
QuestObject,
|
QuestObject,
|
||||||
} from "../../../domain";
|
} from "../../../domain";
|
||||||
|
import { Instruction, InstructionSegment, SegmentType } from "../../../scripting/instructions";
|
||||||
|
import { Opcode } from "../../../scripting/opcodes";
|
||||||
import { area_store } from "../../../stores/AreaStore";
|
import { area_store } from "../../../stores/AreaStore";
|
||||||
import { prs_compress } from "../../compression/prs/compress";
|
import { prs_compress } from "../../compression/prs/compress";
|
||||||
import { prs_decompress } from "../../compression/prs/decompress";
|
import { prs_decompress } from "../../compression/prs/decompress";
|
||||||
@ -16,9 +18,8 @@ import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
|||||||
import { Cursor } from "../../cursor/Cursor";
|
import { Cursor } from "../../cursor/Cursor";
|
||||||
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { Vec3 } from "../../vector";
|
import { Vec3 } from "../../vector";
|
||||||
import { BinFile, Instruction, InstructionSegment, parse_bin, SegmentType, write_bin } from "./bin";
|
import { BinFile, parse_bin, write_bin } from "./bin";
|
||||||
import { DatFile, DatNpc, DatObject, parse_dat, write_dat } from "./dat";
|
import { DatFile, DatNpc, DatObject, parse_dat, write_dat } from "./dat";
|
||||||
import { Opcode } from "../../../scripting/opcodes";
|
|
||||||
import { parse_qst, QstContainedFile, write_qst } from "./qst";
|
import { parse_qst, QstContainedFile, write_qst } from "./qst";
|
||||||
|
|
||||||
const logger = Logger.get("data_formats/parsing/quest");
|
const logger = Logger.get("data_formats/parsing/quest");
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import { action, computed, observable } from "mobx";
|
import { action, computed, observable } from "mobx";
|
||||||
import { Segment } from "../data_formats/parsing/quest/bin";
|
|
||||||
import { DatUnknown } from "../data_formats/parsing/quest/dat";
|
import { DatUnknown } from "../data_formats/parsing/quest/dat";
|
||||||
import { Vec3 } from "../data_formats/vector";
|
import { Vec3 } from "../data_formats/vector";
|
||||||
import { enum_values } from "../enums";
|
import { enum_values } from "../enums";
|
||||||
|
import { Segment } from "../scripting/instructions";
|
||||||
import { ItemType } from "./items";
|
import { ItemType } from "./items";
|
||||||
import { NpcType } from "./NpcType";
|
import { NpcType } from "./NpcType";
|
||||||
import { ObjectType } from "./ObjectType";
|
import { ObjectType } from "./ObjectType";
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import { observable } from "mobx";
|
import { observable } from "mobx";
|
||||||
import { editor } from "monaco-editor";
|
import { editor } from "monaco-editor";
|
||||||
import AssemblyWorker from "worker-loader!./assembly_worker";
|
import AssemblyWorker from "worker-loader!./assembly_worker";
|
||||||
import { Segment } from "../data_formats/parsing/quest/bin";
|
|
||||||
import { AssemblyChangeInput, NewAssemblyInput, ScriptWorkerOutput } from "./assembler_messages";
|
import { AssemblyChangeInput, NewAssemblyInput, ScriptWorkerOutput } from "./assembler_messages";
|
||||||
import { AssemblyError } from "./assembly";
|
import { AssemblyError } from "./assembly";
|
||||||
import { disassemble } from "./disassembly";
|
import { disassemble } from "./disassembly";
|
||||||
|
import { Segment } from "./instructions";
|
||||||
|
|
||||||
export class AssemblyAnalyser {
|
export class AssemblyAnalyser {
|
||||||
@observable errors: AssemblyError[] = [];
|
@observable errors: AssemblyError[] = [];
|
||||||
|
@ -6,6 +6,7 @@ export enum TokenType {
|
|||||||
Label,
|
Label,
|
||||||
CodeSection,
|
CodeSection,
|
||||||
DataSection,
|
DataSection,
|
||||||
|
StringSection,
|
||||||
InvalidSection,
|
InvalidSection,
|
||||||
String,
|
String,
|
||||||
UnterminatedString,
|
UnterminatedString,
|
||||||
@ -22,6 +23,7 @@ export type Token =
|
|||||||
| LabelToken
|
| LabelToken
|
||||||
| CodeSectionToken
|
| CodeSectionToken
|
||||||
| DataSectionToken
|
| DataSectionToken
|
||||||
|
| StringSectionToken
|
||||||
| InvalidSectionToken
|
| InvalidSectionToken
|
||||||
| StringToken
|
| StringToken
|
||||||
| UnterminatedStringToken
|
| UnterminatedStringToken
|
||||||
@ -75,6 +77,12 @@ export type DataSectionToken = {
|
|||||||
len: number;
|
len: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type StringSectionToken = {
|
||||||
|
type: TokenType.StringSection;
|
||||||
|
col: number;
|
||||||
|
len: number;
|
||||||
|
};
|
||||||
|
|
||||||
export type InvalidSectionToken = {
|
export type InvalidSectionToken = {
|
||||||
type: TokenType.InvalidSection;
|
type: TokenType.InvalidSection;
|
||||||
col: number;
|
col: number;
|
||||||
@ -293,7 +301,11 @@ export class AssemblyLexer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private tokenize_section(): CodeSectionToken | DataSectionToken | InvalidSectionToken {
|
private tokenize_section():
|
||||||
|
| CodeSectionToken
|
||||||
|
| DataSectionToken
|
||||||
|
| StringSectionToken
|
||||||
|
| InvalidSectionToken {
|
||||||
const col = this.col;
|
const col = this.col;
|
||||||
this.mark();
|
this.mark();
|
||||||
|
|
||||||
@ -310,6 +322,8 @@ export class AssemblyLexer {
|
|||||||
return { type: TokenType.CodeSection, col, len: 5 };
|
return { type: TokenType.CodeSection, col, len: 5 };
|
||||||
case ".data":
|
case ".data":
|
||||||
return { type: TokenType.DataSection, col, len: 5 };
|
return { type: TokenType.DataSection, col, len: 5 };
|
||||||
|
case ".string":
|
||||||
|
return { type: TokenType.DataSection, col, len: 7 };
|
||||||
default:
|
default:
|
||||||
return { type: TokenType.InvalidSection, col, len: this.marked_len() };
|
return { type: TokenType.InvalidSection, col, len: this.marked_len() };
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,8 @@ import {
|
|||||||
RegisterToken,
|
RegisterToken,
|
||||||
Token,
|
Token,
|
||||||
TokenType,
|
TokenType,
|
||||||
|
StringSectionToken,
|
||||||
|
StringToken,
|
||||||
} from "./AssemblyLexer";
|
} from "./AssemblyLexer";
|
||||||
import {
|
import {
|
||||||
Segment,
|
Segment,
|
||||||
@ -17,8 +19,27 @@ import {
|
|||||||
SegmentType,
|
SegmentType,
|
||||||
Instruction,
|
Instruction,
|
||||||
DataSegment,
|
DataSegment,
|
||||||
|
StringSegment,
|
||||||
} from "./instructions";
|
} from "./instructions";
|
||||||
import { Opcode, OPCODES_BY_MNEMONIC, Param } from "./opcodes";
|
import {
|
||||||
|
Opcode,
|
||||||
|
OPCODES_BY_MNEMONIC,
|
||||||
|
Param,
|
||||||
|
TYPE_I_LABEL_VAR,
|
||||||
|
TYPE_REG_REF_VAR,
|
||||||
|
StackInteraction,
|
||||||
|
TYPE_BYTE,
|
||||||
|
TYPE_DWORD,
|
||||||
|
TYPE_WORD,
|
||||||
|
TYPE_FLOAT,
|
||||||
|
TYPE_S_LABEL,
|
||||||
|
TYPE_D_LABEL,
|
||||||
|
TYPE_I_LABEL,
|
||||||
|
TYPE_LABEL,
|
||||||
|
TYPE_STRING,
|
||||||
|
TYPE_REG_REF,
|
||||||
|
RegTupRefType,
|
||||||
|
} from "./opcodes";
|
||||||
|
|
||||||
const logger = Logger.get("scripting/assembly");
|
const logger = Logger.get("scripting/assembly");
|
||||||
|
|
||||||
@ -53,8 +74,7 @@ class Assembler {
|
|||||||
private errors!: AssemblyError[];
|
private errors!: AssemblyError[];
|
||||||
// Encountered labels.
|
// Encountered labels.
|
||||||
private labels!: Set<number>;
|
private labels!: Set<number>;
|
||||||
// True iff we're in a code section, false iff we're in a data section.
|
private section_type: SegmentType = SegmentType.Instructions;
|
||||||
private code_section = true;
|
|
||||||
|
|
||||||
constructor(private assembly: string[], private manual_stack: boolean) {}
|
constructor(private assembly: string[], private manual_stack: boolean) {}
|
||||||
|
|
||||||
@ -68,7 +88,8 @@ class Assembler {
|
|||||||
this.warnings = [];
|
this.warnings = [];
|
||||||
this.errors = [];
|
this.errors = [];
|
||||||
this.labels = new Set();
|
this.labels = new Set();
|
||||||
this.code_section = true;
|
// Need to cast SegmentType.Instructions because of TypeScript bug.
|
||||||
|
this.section_type = SegmentType.Instructions as SegmentType;
|
||||||
|
|
||||||
for (const line of this.assembly) {
|
for (const line of this.assembly) {
|
||||||
this.tokens = this.lexer.tokenize_line(line);
|
this.tokens = this.lexer.tokenize_line(line);
|
||||||
@ -81,24 +102,34 @@ class Assembler {
|
|||||||
this.parse_label(token);
|
this.parse_label(token);
|
||||||
break;
|
break;
|
||||||
case TokenType.CodeSection:
|
case TokenType.CodeSection:
|
||||||
this.parse_code_section(token);
|
|
||||||
break;
|
|
||||||
case TokenType.DataSection:
|
case TokenType.DataSection:
|
||||||
this.parse_data_section(token);
|
case TokenType.StringSection:
|
||||||
|
this.parse_section(token);
|
||||||
break;
|
break;
|
||||||
case TokenType.Int:
|
case TokenType.Int:
|
||||||
if (this.code_section) {
|
if (this.section_type === SegmentType.Data) {
|
||||||
|
this.parse_bytes(token);
|
||||||
|
} else {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
col: token.col,
|
col: token.col,
|
||||||
length: token.len,
|
length: token.len,
|
||||||
message: "Unexpected token.",
|
message: "Unexpected token.",
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case TokenType.String:
|
||||||
|
if (this.section_type === SegmentType.String) {
|
||||||
|
this.parse_string(token);
|
||||||
} else {
|
} else {
|
||||||
this.parse_bytes(token);
|
this.add_error({
|
||||||
|
col: token.col,
|
||||||
|
length: token.len,
|
||||||
|
message: "Unexpected token.",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case TokenType.Ident:
|
case TokenType.Ident:
|
||||||
if (this.code_section) {
|
if (this.section_type === SegmentType.Instructions) {
|
||||||
this.parse_instruction(token);
|
this.parse_instruction(token);
|
||||||
} else {
|
} else {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
@ -181,6 +212,23 @@ class Assembler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private add_string(str: string): void {
|
||||||
|
if (!this.segment) {
|
||||||
|
// Unadressable data, technically valid.
|
||||||
|
const string_segment: StringSegment = {
|
||||||
|
labels: [],
|
||||||
|
type: SegmentType.String,
|
||||||
|
value: str,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.segment = string_segment;
|
||||||
|
this.object_code.push(string_segment);
|
||||||
|
} else {
|
||||||
|
const s_seg = this.segment as StringSegment;
|
||||||
|
s_seg.value += str;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private add_error({
|
private add_error({
|
||||||
col,
|
col,
|
||||||
length,
|
length,
|
||||||
@ -228,7 +276,8 @@ class Assembler {
|
|||||||
|
|
||||||
const next_token = this.tokens.shift();
|
const next_token = this.tokens.shift();
|
||||||
|
|
||||||
if (this.code_section) {
|
switch (this.section_type) {
|
||||||
|
case SegmentType.Instructions:
|
||||||
this.segment = {
|
this.segment = {
|
||||||
type: SegmentType.Instructions,
|
type: SegmentType.Instructions,
|
||||||
labels: [label],
|
labels: [label],
|
||||||
@ -247,7 +296,9 @@ class Assembler {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
|
break;
|
||||||
|
case SegmentType.Data:
|
||||||
this.segment = {
|
this.segment = {
|
||||||
type: SegmentType.Data,
|
type: SegmentType.Data,
|
||||||
labels: [label],
|
labels: [label],
|
||||||
@ -266,41 +317,60 @@ class Assembler {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private parse_code_section({ col, len }: CodeSectionToken): void {
|
break;
|
||||||
if (this.code_section) {
|
case SegmentType.String:
|
||||||
this.add_warning({
|
this.segment = {
|
||||||
col,
|
type: SegmentType.String,
|
||||||
length: len,
|
labels: [label],
|
||||||
message: "Unnecessary code section marker.",
|
value: "",
|
||||||
});
|
};
|
||||||
}
|
this.object_code.push(this.segment);
|
||||||
|
|
||||||
this.code_section = true;
|
|
||||||
|
|
||||||
const next_token = this.tokens.shift();
|
|
||||||
|
|
||||||
if (next_token) {
|
if (next_token) {
|
||||||
|
if (next_token.type === TokenType.String) {
|
||||||
|
this.parse_string(next_token);
|
||||||
|
} else {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
col: next_token.col,
|
col: next_token.col,
|
||||||
length: next_token.len,
|
length: next_token.len,
|
||||||
message: "Unexpected token.",
|
message: "Expected a string.",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private parse_data_section({ col, len }: DataSectionToken): void {
|
break;
|
||||||
if (!this.code_section) {
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private parse_section({
|
||||||
|
type,
|
||||||
|
col,
|
||||||
|
len,
|
||||||
|
}: CodeSectionToken | DataSectionToken | StringSectionToken): void {
|
||||||
|
let section_type!: SegmentType;
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case TokenType.CodeSection:
|
||||||
|
section_type = SegmentType.Instructions;
|
||||||
|
break;
|
||||||
|
case TokenType.DataSection:
|
||||||
|
section_type = SegmentType.Data;
|
||||||
|
break;
|
||||||
|
case TokenType.StringSection:
|
||||||
|
section_type = SegmentType.String;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.section_type === section_type) {
|
||||||
this.add_warning({
|
this.add_warning({
|
||||||
col,
|
col,
|
||||||
length: len,
|
length: len,
|
||||||
message: "Unnecessary data section marker.",
|
message: "Unnecessary section marker.",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
this.code_section = false;
|
this.section_type = section_type;
|
||||||
|
|
||||||
const next_token = this.tokens.shift();
|
const next_token = this.tokens.shift();
|
||||||
|
|
||||||
@ -324,11 +394,14 @@ class Assembler {
|
|||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
const varargs =
|
const varargs =
|
||||||
opcode.params.findIndex(p => p.type === Type.U8Var || p.type === Type.ILabelVar) !==
|
opcode.params.findIndex(
|
||||||
-1;
|
p => p.type === TYPE_I_LABEL_VAR || p.type === TYPE_REG_REF_VAR
|
||||||
|
) !== -1;
|
||||||
|
|
||||||
const param_count =
|
const param_count =
|
||||||
opcode.params.length + (this.manual_stack ? 0 : opcode.stack_params.length);
|
this.manual_stack && opcode.stack === StackInteraction.Pop
|
||||||
|
? 0
|
||||||
|
: opcode.params.length;
|
||||||
|
|
||||||
let arg_count = 0;
|
let arg_count = 0;
|
||||||
|
|
||||||
@ -362,7 +435,7 @@ class Assembler {
|
|||||||
});
|
});
|
||||||
|
|
||||||
return;
|
return;
|
||||||
} else if (varargs || arg_count === opcode.params.length) {
|
} else if (opcode.stack !== StackInteraction.Pop) {
|
||||||
// Inline arguments.
|
// Inline arguments.
|
||||||
if (!this.parse_args(opcode.params, ins_args)) {
|
if (!this.parse_args(opcode.params, ins_args)) {
|
||||||
return;
|
return;
|
||||||
@ -371,12 +444,12 @@ class Assembler {
|
|||||||
// Stack arguments.
|
// Stack arguments.
|
||||||
const stack_args: Arg[] = [];
|
const stack_args: Arg[] = [];
|
||||||
|
|
||||||
if (!this.parse_args(opcode.stack_params, stack_args)) {
|
if (!this.parse_args(opcode.params, stack_args)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let i = 0; i < opcode.stack_params.length; i++) {
|
for (let i = 0; i < opcode.params.length; i++) {
|
||||||
const param = opcode.stack_params[i];
|
const param = opcode.params[i];
|
||||||
const arg = stack_args[i];
|
const arg = stack_args[i];
|
||||||
|
|
||||||
if (arg == undefined) {
|
if (arg == undefined) {
|
||||||
@ -384,25 +457,32 @@ class Assembler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case Type.U8:
|
case TYPE_BYTE:
|
||||||
case Type.RegRef:
|
case TYPE_REG_REF:
|
||||||
this.add_instruction(Opcode.arg_pushb, [arg]);
|
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
|
||||||
break;
|
break;
|
||||||
case Type.U16:
|
case TYPE_WORD:
|
||||||
case Type.ILabel:
|
case TYPE_LABEL:
|
||||||
case Type.DLabel:
|
case TYPE_I_LABEL:
|
||||||
this.add_instruction(Opcode.arg_pushw, [arg]);
|
case TYPE_D_LABEL:
|
||||||
|
case TYPE_S_LABEL:
|
||||||
|
this.add_instruction(Opcode.ARG_PUSHW, [arg]);
|
||||||
break;
|
break;
|
||||||
case Type.U32:
|
case TYPE_DWORD:
|
||||||
case Type.I32:
|
case TYPE_FLOAT:
|
||||||
case Type.F32:
|
this.add_instruction(Opcode.ARG_PUSHL, [arg]);
|
||||||
this.add_instruction(Opcode.arg_pushl, [arg]);
|
|
||||||
break;
|
break;
|
||||||
case Type.String:
|
case TYPE_STRING:
|
||||||
this.add_instruction(Opcode.arg_pushs, [arg]);
|
this.add_instruction(Opcode.ARG_PUSHS, [arg]);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
logger.error(`Type ${Type[param.type]} not implemented.`);
|
if (param.type instanceof RegTupRefType) {
|
||||||
|
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
|
||||||
|
} else {
|
||||||
|
logger.error(`Type ${param.type} not implemented.`);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -431,7 +511,7 @@ class Assembler {
|
|||||||
message: "Expected an argument.",
|
message: "Expected an argument.",
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
if (param.type !== Type.U8Var && param.type !== Type.ILabelVar) {
|
if (param.type !== TYPE_I_LABEL_VAR && param.type !== TYPE_REG_REF_VAR) {
|
||||||
param_i++;
|
param_i++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -456,27 +536,24 @@ class Assembler {
|
|||||||
switch (token.type) {
|
switch (token.type) {
|
||||||
case TokenType.Int:
|
case TokenType.Int:
|
||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case Type.U8:
|
case TYPE_BYTE:
|
||||||
case Type.U8Var:
|
|
||||||
match = true;
|
match = true;
|
||||||
this.parse_uint(1, token, args);
|
this.parse_int(1, token, args);
|
||||||
break;
|
break;
|
||||||
case Type.U16:
|
case TYPE_WORD:
|
||||||
case Type.ILabel:
|
case TYPE_LABEL:
|
||||||
case Type.ILabelVar:
|
case TYPE_I_LABEL:
|
||||||
case Type.DLabel:
|
case TYPE_D_LABEL:
|
||||||
|
case TYPE_S_LABEL:
|
||||||
|
case TYPE_I_LABEL_VAR:
|
||||||
match = true;
|
match = true;
|
||||||
this.parse_uint(2, token, args);
|
this.parse_int(2, token, args);
|
||||||
break;
|
break;
|
||||||
case Type.U32:
|
case TYPE_DWORD:
|
||||||
match = true;
|
match = true;
|
||||||
this.parse_uint(4, token, args);
|
this.parse_int(4, token, args);
|
||||||
break;
|
break;
|
||||||
case Type.I32:
|
case TYPE_FLOAT:
|
||||||
match = true;
|
|
||||||
this.parse_sint(4, token, args);
|
|
||||||
break;
|
|
||||||
case Type.F32:
|
|
||||||
match = true;
|
match = true;
|
||||||
args.push({
|
args.push({
|
||||||
value: token.value,
|
value: token.value,
|
||||||
@ -489,7 +566,7 @@ class Assembler {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case TokenType.Float:
|
case TokenType.Float:
|
||||||
match = param.type === Type.F32;
|
match = param.type === TYPE_FLOAT;
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
args.push({
|
args.push({
|
||||||
@ -500,11 +577,15 @@ class Assembler {
|
|||||||
|
|
||||||
break;
|
break;
|
||||||
case TokenType.Register:
|
case TokenType.Register:
|
||||||
match = param.type === Type.RegRef;
|
match =
|
||||||
|
param.type === TYPE_REG_REF ||
|
||||||
|
param.type === TYPE_REG_REF_VAR ||
|
||||||
|
param.type instanceof RegTupRefType;
|
||||||
|
|
||||||
this.parse_register(token, args);
|
this.parse_register(token, args);
|
||||||
break;
|
break;
|
||||||
case TokenType.String:
|
case TokenType.String:
|
||||||
match = param.type === Type.String;
|
match = param.type === TYPE_STRING;
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
args.push({
|
args.push({
|
||||||
@ -522,49 +603,61 @@ class Assembler {
|
|||||||
if (!match) {
|
if (!match) {
|
||||||
semi_valid = false;
|
semi_valid = false;
|
||||||
|
|
||||||
let type_str = Type[param.type];
|
let type_str: string | undefined;
|
||||||
|
|
||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case Type.U8:
|
case TYPE_BYTE:
|
||||||
type_str = "an unsigned 8-bit integer";
|
type_str = "a 8-bit integer";
|
||||||
break;
|
break;
|
||||||
case Type.U16:
|
case TYPE_WORD:
|
||||||
type_str = "an unsigned 16-bit integer";
|
type_str = "a 16-bit integer";
|
||||||
break;
|
break;
|
||||||
case Type.U32:
|
case TYPE_DWORD:
|
||||||
type_str = "an unsigned 32-bit integer";
|
type_str = "a 32-bit integer";
|
||||||
break;
|
break;
|
||||||
case Type.I32:
|
case TYPE_FLOAT:
|
||||||
type_str = "a signed 32-bit integer";
|
|
||||||
break;
|
|
||||||
case Type.F32:
|
|
||||||
type_str = "a float";
|
type_str = "a float";
|
||||||
break;
|
break;
|
||||||
case Type.RegRef:
|
case TYPE_LABEL:
|
||||||
type_str = "a register reference";
|
type_str = "a label";
|
||||||
break;
|
break;
|
||||||
case Type.ILabel:
|
case TYPE_I_LABEL:
|
||||||
|
case TYPE_I_LABEL_VAR:
|
||||||
type_str = "an instruction label";
|
type_str = "an instruction label";
|
||||||
break;
|
break;
|
||||||
case Type.DLabel:
|
case TYPE_D_LABEL:
|
||||||
type_str = "a data label";
|
type_str = "a data label";
|
||||||
break;
|
break;
|
||||||
case Type.U8Var:
|
case TYPE_S_LABEL:
|
||||||
type_str = "an unsigned 8-bit integer";
|
type_str = "a string label";
|
||||||
break;
|
break;
|
||||||
case Type.ILabelVar:
|
case TYPE_STRING:
|
||||||
type_str = "an instruction label";
|
|
||||||
break;
|
|
||||||
case Type.String:
|
|
||||||
type_str = "a string";
|
type_str = "a string";
|
||||||
break;
|
break;
|
||||||
|
case TYPE_REG_REF:
|
||||||
|
case TYPE_REG_REF_VAR:
|
||||||
|
type_str = "a register reference";
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (param.type instanceof RegTupRefType) {
|
||||||
|
type_str = "a register reference";
|
||||||
|
}
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (type_str) {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
col: token.col,
|
col: token.col,
|
||||||
length: token.len,
|
length: token.len,
|
||||||
message: `Expected ${type_str}.`,
|
message: `Expected ${type_str}.`,
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
this.add_error({
|
||||||
|
col: token.col,
|
||||||
|
length: token.len,
|
||||||
|
message: `Unexpected token.`,
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -573,46 +666,22 @@ class Assembler {
|
|||||||
return semi_valid;
|
return semi_valid;
|
||||||
}
|
}
|
||||||
|
|
||||||
private parse_uint(size: number, { col, len, value }: IntToken, args: Arg[]): void {
|
private parse_int(size: number, { col, len, value }: IntToken, args: Arg[]): void {
|
||||||
const bit_size = 8 * size;
|
|
||||||
const max_value = Math.pow(2, bit_size) - 1;
|
|
||||||
|
|
||||||
if (value < 0) {
|
|
||||||
this.add_error({
|
|
||||||
col,
|
|
||||||
length: len,
|
|
||||||
message: `Unsigned ${bit_size}-bit integer can't be less than 0.`,
|
|
||||||
});
|
|
||||||
} else if (value > max_value) {
|
|
||||||
this.add_error({
|
|
||||||
col,
|
|
||||||
length: len,
|
|
||||||
message: `Unsigned ${bit_size}-bit integer can't be greater than ${max_value}.`,
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
args.push({
|
|
||||||
value,
|
|
||||||
size,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private parse_sint(size: number, { col, len, value }: IntToken, args: Arg[]): void {
|
|
||||||
const bit_size = 8 * size;
|
const bit_size = 8 * size;
|
||||||
const min_value = -Math.pow(2, bit_size - 1);
|
const min_value = -Math.pow(2, bit_size - 1);
|
||||||
const max_value = Math.pow(2, bit_size - 1) - 1;
|
const max_value = Math.pow(2, bit_size) - 1;
|
||||||
|
|
||||||
if (value < min_value) {
|
if (value < min_value) {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
col,
|
col,
|
||||||
length: len,
|
length: len,
|
||||||
message: `Signed ${bit_size}-bit integer can't be less than ${min_value}.`,
|
message: `${bit_size}-Bit integer can't be less than ${min_value}.`,
|
||||||
});
|
});
|
||||||
} else if (value > max_value) {
|
} else if (value > max_value) {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
col,
|
col,
|
||||||
length: len,
|
length: len,
|
||||||
message: `Signed ${bit_size}-bit integer can't be greater than ${max_value}.`,
|
message: `${bit_size}-Bit integer can't be greater than ${max_value}.`,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
args.push({
|
args.push({
|
||||||
@ -676,4 +745,18 @@ class Assembler {
|
|||||||
|
|
||||||
this.add_bytes(bytes);
|
this.add_bytes(bytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private parse_string(token: StringToken): void {
|
||||||
|
const next_token = this.tokens.shift();
|
||||||
|
|
||||||
|
if (next_token) {
|
||||||
|
this.add_error({
|
||||||
|
col: next_token.col,
|
||||||
|
length: next_token.len,
|
||||||
|
message: "Unexpected token.",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
this.add_string(token.value);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -15,35 +15,40 @@ import {
|
|||||||
export function disassemble(object_code: Segment[], manual_stack: boolean = false): string[] {
|
export function disassemble(object_code: Segment[], manual_stack: boolean = false): string[] {
|
||||||
const lines: string[] = [];
|
const lines: string[] = [];
|
||||||
const stack: Arg[] = [];
|
const stack: Arg[] = [];
|
||||||
let code_block: boolean | undefined;
|
let section_type: SegmentType | undefined;
|
||||||
|
|
||||||
for (const segment of object_code) {
|
for (const segment of object_code) {
|
||||||
if (segment.type === SegmentType.Data) {
|
// Section marker.
|
||||||
if (code_block !== false) {
|
let section_marker!: string;
|
||||||
code_block = false;
|
|
||||||
|
switch (segment.type) {
|
||||||
|
case SegmentType.Instructions:
|
||||||
|
section_marker = ".code";
|
||||||
|
break;
|
||||||
|
case SegmentType.Data:
|
||||||
|
section_marker = ".data";
|
||||||
|
break;
|
||||||
|
case SegmentType.String:
|
||||||
|
section_marker = ".string";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (section_type !== segment.type) {
|
||||||
|
section_type = segment.type;
|
||||||
|
|
||||||
if (lines.length) {
|
if (lines.length) {
|
||||||
lines.push("");
|
lines.push("");
|
||||||
}
|
}
|
||||||
|
|
||||||
lines.push(".data", "");
|
lines.push(section_marker, "");
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (code_block !== true) {
|
|
||||||
code_block = true;
|
|
||||||
|
|
||||||
if (lines.length) {
|
|
||||||
lines.push("");
|
|
||||||
}
|
|
||||||
|
|
||||||
lines.push(".code", "");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Labels.
|
||||||
for (const label of segment.labels) {
|
for (const label of segment.labels) {
|
||||||
lines.push(`${label}:`);
|
lines.push(`${label}:`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Code or data lines.
|
||||||
if (segment.type === SegmentType.Data) {
|
if (segment.type === SegmentType.Data) {
|
||||||
const bytes = new Uint8Array(segment.data);
|
const bytes = new Uint8Array(segment.data);
|
||||||
let line = " ";
|
let line = " ";
|
||||||
@ -63,7 +68,7 @@ export function disassemble(object_code: Segment[], manual_stack: boolean = fals
|
|||||||
lines.push(line);
|
lines.push(line);
|
||||||
}
|
}
|
||||||
} else if (segment.type === SegmentType.String) {
|
} else if (segment.type === SegmentType.String) {
|
||||||
lines.push(" " + segment.value);
|
lines.push(" " + JSON.stringify(segment.value));
|
||||||
} else {
|
} else {
|
||||||
for (const instruction of segment.instructions) {
|
for (const instruction of segment.instructions) {
|
||||||
if (!manual_stack && instruction.opcode.stack === StackInteraction.Push) {
|
if (!manual_stack && instruction.opcode.stack === StackInteraction.Push) {
|
||||||
|
@ -18,15 +18,13 @@ export class Instruction {
|
|||||||
readonly param_to_args: Arg[][] = [];
|
readonly param_to_args: Arg[][] = [];
|
||||||
|
|
||||||
constructor(readonly opcode: Opcode, readonly args: Arg[]) {
|
constructor(readonly opcode: Opcode, readonly args: Arg[]) {
|
||||||
for (let i = 0; i < opcode.params.length; i++) {
|
const len = Math.min(opcode.params.length, args.length);
|
||||||
|
|
||||||
|
for (let i = 0; i < len; i++) {
|
||||||
const type = opcode.params[i].type;
|
const type = opcode.params[i].type;
|
||||||
const arg = args[i];
|
const arg = args[i];
|
||||||
this.param_to_args[i] = [];
|
this.param_to_args[i] = [];
|
||||||
|
|
||||||
if (arg == undefined) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case TYPE_I_LABEL_VAR:
|
case TYPE_I_LABEL_VAR:
|
||||||
case TYPE_REG_REF_VAR:
|
case TYPE_REG_REF_VAR:
|
||||||
|
@ -1078,7 +1078,7 @@ export class Opcode {
|
|||||||
undefined,
|
undefined,
|
||||||
undefined
|
undefined
|
||||||
),
|
),
|
||||||
new Param(TYPE_I_LABEL, undefined, undefined),
|
new Param(TYPE_I_LABEL_VAR, undefined, undefined),
|
||||||
],
|
],
|
||||||
undefined
|
undefined
|
||||||
));
|
));
|
||||||
@ -1092,7 +1092,7 @@ export class Opcode {
|
|||||||
undefined,
|
undefined,
|
||||||
undefined
|
undefined
|
||||||
),
|
),
|
||||||
new Param(TYPE_I_LABEL, undefined, undefined),
|
new Param(TYPE_I_LABEL_VAR, undefined, undefined),
|
||||||
],
|
],
|
||||||
undefined
|
undefined
|
||||||
));
|
));
|
||||||
@ -1190,19 +1190,25 @@ export class Opcode {
|
|||||||
[new Param(TYPE_WORD, undefined, undefined)],
|
[new Param(TYPE_WORD, undefined, undefined)],
|
||||||
StackInteraction.Push
|
StackInteraction.Push
|
||||||
));
|
));
|
||||||
static readonly UNKNOWN_4C = (OPCODES[0x4c] = new Opcode(
|
static readonly ARG_PUSHA = (OPCODES[0x4c] = new Opcode(
|
||||||
0x4c,
|
0x4c,
|
||||||
"unknown_4c",
|
"arg_pusha",
|
||||||
|
"Pushes the memory address of the given register onto the stack. Not used by Sega.",
|
||||||
|
[
|
||||||
|
new Param(
|
||||||
|
new RegTupRefType(new Param(TYPE_ANY, undefined, ParamAccess.Read)),
|
||||||
undefined,
|
undefined,
|
||||||
[],
|
|
||||||
undefined
|
undefined
|
||||||
|
),
|
||||||
|
],
|
||||||
|
StackInteraction.Push
|
||||||
));
|
));
|
||||||
static readonly UNKNOWN_4D = (OPCODES[0x4d] = new Opcode(
|
static readonly ARG_PUSHO = (OPCODES[0x4d] = new Opcode(
|
||||||
0x4d,
|
0x4d,
|
||||||
"unknown_4d",
|
"arg_pusho",
|
||||||
undefined,
|
"Pushes the memory address of the given label onto the stack. Not used by Sega.",
|
||||||
[],
|
[new Param(TYPE_LABEL, undefined, undefined)],
|
||||||
undefined
|
StackInteraction.Push
|
||||||
));
|
));
|
||||||
static readonly ARG_PUSHS = (OPCODES[0x4e] = new Opcode(
|
static readonly ARG_PUSHS = (OPCODES[0x4e] = new Opcode(
|
||||||
0x4e,
|
0x4e,
|
||||||
@ -5043,16 +5049,8 @@ export class Opcode {
|
|||||||
"npc_action_string",
|
"npc_action_string",
|
||||||
undefined,
|
undefined,
|
||||||
[
|
[
|
||||||
new Param(
|
new Param(TYPE_DWORD, undefined, undefined),
|
||||||
new RegTupRefType(new Param(TYPE_ANY, undefined, ParamAccess.Read)),
|
new Param(TYPE_DWORD, undefined, undefined),
|
||||||
undefined,
|
|
||||||
undefined
|
|
||||||
),
|
|
||||||
new Param(
|
|
||||||
new RegTupRefType(new Param(TYPE_ANY, undefined, ParamAccess.Read)),
|
|
||||||
undefined,
|
|
||||||
undefined
|
|
||||||
),
|
|
||||||
new Param(TYPE_S_LABEL, undefined, undefined),
|
new Param(TYPE_S_LABEL, undefined, undefined),
|
||||||
],
|
],
|
||||||
undefined
|
undefined
|
||||||
|
@ -2,8 +2,8 @@ import { autorun } from "mobx";
|
|||||||
import { editor, languages, MarkerSeverity } from "monaco-editor";
|
import { editor, languages, MarkerSeverity } from "monaco-editor";
|
||||||
import React, { Component, createRef, ReactNode } from "react";
|
import React, { Component, createRef, ReactNode } from "react";
|
||||||
import { AutoSizer } from "react-virtualized";
|
import { AutoSizer } from "react-virtualized";
|
||||||
import { OPCODES } from "../../data_formats/parsing/quest/bin";
|
|
||||||
import { AssemblyAnalyser } from "../../scripting/AssemblyAnalyser";
|
import { AssemblyAnalyser } from "../../scripting/AssemblyAnalyser";
|
||||||
|
import { OPCODES } from "../../scripting/opcodes";
|
||||||
import { quest_editor_store } from "../../stores/QuestEditorStore";
|
import { quest_editor_store } from "../../stores/QuestEditorStore";
|
||||||
import { Action } from "../../undo";
|
import { Action } from "../../undo";
|
||||||
import styles from "./AssemblyEditorComponent.css";
|
import styles from "./AssemblyEditorComponent.css";
|
||||||
@ -77,6 +77,11 @@ const KEYWORD_SUGGESTIONS = [
|
|||||||
kind: languages.CompletionItemKind.Keyword,
|
kind: languages.CompletionItemKind.Keyword,
|
||||||
insertText: "data",
|
insertText: "data",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: ".string",
|
||||||
|
kind: languages.CompletionItemKind.Keyword,
|
||||||
|
insertText: "string",
|
||||||
|
},
|
||||||
] as languages.CompletionItem[];
|
] as languages.CompletionItem[];
|
||||||
|
|
||||||
languages.register({ id: "psoasm" });
|
languages.register({ id: "psoasm" });
|
||||||
|
Loading…
Reference in New Issue
Block a user