mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
All bugs resulting from opcode typing changes have been fixed.
This commit is contained in:
parent
dbe1b06fa6
commit
1ba7d3b6a7
@ -854,9 +854,11 @@ opcodes:
|
|||||||
params:
|
params:
|
||||||
- type: reg_tup_ref
|
- type: reg_tup_ref
|
||||||
reg_tup:
|
reg_tup:
|
||||||
- type: byte
|
- type: dword
|
||||||
access: write
|
access: write
|
||||||
- type: dword
|
- type: dword
|
||||||
|
doc: Player slot.
|
||||||
|
stack: pop
|
||||||
|
|
||||||
- code: 0x6b
|
- code: 0x6b
|
||||||
mnemonic: p_disablewarp
|
mnemonic: p_disablewarp
|
||||||
@ -2829,8 +2831,14 @@ opcodes:
|
|||||||
- code: 0xf8dc
|
- code: 0xf8dc
|
||||||
mnemonic: npc_action_string
|
mnemonic: npc_action_string
|
||||||
params:
|
params:
|
||||||
- type: dword
|
- type: reg_tup_ref
|
||||||
- type: dword
|
reg_tup:
|
||||||
|
- type: dword
|
||||||
|
access: read
|
||||||
|
- type: reg_tup_ref
|
||||||
|
reg_tup:
|
||||||
|
- type: dword
|
||||||
|
access: read
|
||||||
- type: string_label
|
- type: string_label
|
||||||
|
|
||||||
- code: 0xf8dd
|
- code: 0xf8dd
|
||||||
@ -3346,12 +3354,15 @@ opcodes:
|
|||||||
- code: 0xf924
|
- code: 0xf924
|
||||||
mnemonic: get_coord_player_detect
|
mnemonic: get_coord_player_detect
|
||||||
params:
|
params:
|
||||||
- type: dword
|
|
||||||
doc: Player slot.
|
|
||||||
- type: reg_tup_ref
|
- type: reg_tup_ref
|
||||||
reg_tup: # TODO: determine type and access
|
reg_tup: # TODO: determine type and access
|
||||||
- type: dword
|
- type: any
|
||||||
access: write
|
doc: Player slot.
|
||||||
|
access: read
|
||||||
|
- type: reg_tup_ref
|
||||||
|
reg_tup: # TODO: determine type and access
|
||||||
|
- type: any
|
||||||
|
access: read
|
||||||
|
|
||||||
- code: 0xf925
|
- code: 0xf925
|
||||||
mnemonic: read_global_flag
|
mnemonic: read_global_flag
|
||||||
|
@ -303,7 +303,8 @@ function parse_object_code(
|
|||||||
offset += segment.data.byteLength;
|
offset += segment.data.byteLength;
|
||||||
break;
|
break;
|
||||||
case SegmentType.String:
|
case SegmentType.String:
|
||||||
offset += 2 * segment.value.length + 2;
|
// String segments should be multiples of 4 bytes.
|
||||||
|
offset += 4 * Math.ceil((segment.value.length + 1) / 2);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`${SegmentType[segment!.type]} not implemented.`);
|
throw new Error(`${SegmentType[segment!.type]} not implemented.`);
|
||||||
@ -501,6 +502,7 @@ function parse_instructions_segment(
|
|||||||
break;
|
break;
|
||||||
case TYPE_D_LABEL:
|
case TYPE_D_LABEL:
|
||||||
segment_type = SegmentType.Data;
|
segment_type = SegmentType.Data;
|
||||||
|
break;
|
||||||
case TYPE_S_LABEL:
|
case TYPE_S_LABEL:
|
||||||
segment_type = SegmentType.String;
|
segment_type = SegmentType.String;
|
||||||
break;
|
break;
|
||||||
@ -722,7 +724,9 @@ function write_object_code(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (segment.type === SegmentType.String) {
|
} else if (segment.type === SegmentType.String) {
|
||||||
cursor.write_string_utf16(segment.value, 2 * segment.value.length + 2);
|
// String segments should be multiples of 4 bytes.
|
||||||
|
const byte_length = 4 * Math.ceil((segment.value.length + 1) / 2);
|
||||||
|
cursor.write_string_utf16(segment.value, byte_length);
|
||||||
} else {
|
} else {
|
||||||
cursor.write_cursor(new ArrayBufferCursor(segment.data, cursor.endianness));
|
cursor.write_cursor(new ArrayBufferCursor(segment.data, cursor.endianness));
|
||||||
}
|
}
|
||||||
|
@ -323,7 +323,7 @@ export class AssemblyLexer {
|
|||||||
case ".data":
|
case ".data":
|
||||||
return { type: TokenType.DataSection, col, len: 5 };
|
return { type: TokenType.DataSection, col, len: 5 };
|
||||||
case ".string":
|
case ".string":
|
||||||
return { type: TokenType.DataSection, col, len: 7 };
|
return { type: TokenType.StringSection, col, len: 7 };
|
||||||
default:
|
default:
|
||||||
return { type: TokenType.InvalidSection, col, len: this.marked_len() };
|
return { type: TokenType.InvalidSection, col, len: this.marked_len() };
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { editor } from "monaco-editor";
|
import { editor } from "monaco-editor";
|
||||||
import { Segment } from "../data_formats/parsing/quest/bin";
|
|
||||||
import { AssemblyError } from "./assembly";
|
import { AssemblyError } from "./assembly";
|
||||||
|
import { Segment } from "./instructions";
|
||||||
|
|
||||||
export type ScriptWorkerInput = NewAssemblyInput | AssemblyChangeInput;
|
export type ScriptWorkerInput = NewAssemblyInput | AssemblyChangeInput;
|
||||||
|
|
||||||
|
@ -74,7 +74,7 @@ class Assembler {
|
|||||||
private errors!: AssemblyError[];
|
private errors!: AssemblyError[];
|
||||||
// Encountered labels.
|
// Encountered labels.
|
||||||
private labels!: Set<number>;
|
private labels!: Set<number>;
|
||||||
private section_type: SegmentType = SegmentType.Instructions;
|
private section: SegmentType = SegmentType.Instructions;
|
||||||
|
|
||||||
constructor(private assembly: string[], private manual_stack: boolean) {}
|
constructor(private assembly: string[], private manual_stack: boolean) {}
|
||||||
|
|
||||||
@ -89,7 +89,7 @@ class Assembler {
|
|||||||
this.errors = [];
|
this.errors = [];
|
||||||
this.labels = new Set();
|
this.labels = new Set();
|
||||||
// Need to cast SegmentType.Instructions because of TypeScript bug.
|
// Need to cast SegmentType.Instructions because of TypeScript bug.
|
||||||
this.section_type = SegmentType.Instructions as SegmentType;
|
this.section = 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);
|
||||||
@ -107,7 +107,7 @@ class Assembler {
|
|||||||
this.parse_section(token);
|
this.parse_section(token);
|
||||||
break;
|
break;
|
||||||
case TokenType.Int:
|
case TokenType.Int:
|
||||||
if (this.section_type === SegmentType.Data) {
|
if (this.section === SegmentType.Data) {
|
||||||
this.parse_bytes(token);
|
this.parse_bytes(token);
|
||||||
} else {
|
} else {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
@ -118,7 +118,7 @@ class Assembler {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case TokenType.String:
|
case TokenType.String:
|
||||||
if (this.section_type === SegmentType.String) {
|
if (this.section === SegmentType.String) {
|
||||||
this.parse_string(token);
|
this.parse_string(token);
|
||||||
} else {
|
} else {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
@ -129,7 +129,7 @@ class Assembler {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case TokenType.Ident:
|
case TokenType.Ident:
|
||||||
if (this.section_type === SegmentType.Instructions) {
|
if (this.section === SegmentType.Instructions) {
|
||||||
this.parse_instruction(token);
|
this.parse_instruction(token);
|
||||||
} else {
|
} else {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
@ -184,9 +184,11 @@ class Assembler {
|
|||||||
|
|
||||||
this.segment = instruction_segment;
|
this.segment = instruction_segment;
|
||||||
this.object_code.push(instruction_segment);
|
this.object_code.push(instruction_segment);
|
||||||
|
} else if (this.segment.type === SegmentType.Instructions) {
|
||||||
|
this.segment.instructions.push(new Instruction(opcode, args));
|
||||||
|
} else {
|
||||||
|
logger.error(`Line ${this.line_no}: Expected instructions segment.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
(this.segment as InstructionSegment).instructions.push(new Instruction(opcode, args));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private add_bytes(bytes: number[]): void {
|
private add_bytes(bytes: number[]): void {
|
||||||
@ -200,15 +202,16 @@ class Assembler {
|
|||||||
|
|
||||||
this.segment = data_segment;
|
this.segment = data_segment;
|
||||||
this.object_code.push(data_segment);
|
this.object_code.push(data_segment);
|
||||||
} else {
|
} else if (this.segment.type === SegmentType.Data) {
|
||||||
const d_seg = this.segment as DataSegment;
|
const buf = new ArrayBuffer(this.segment.data.byteLength + bytes.length);
|
||||||
const buf = new ArrayBuffer(d_seg.data.byteLength + bytes.length);
|
|
||||||
const arr = new Uint8Array(buf);
|
const arr = new Uint8Array(buf);
|
||||||
|
|
||||||
arr.set(new Uint8Array(d_seg.data));
|
arr.set(new Uint8Array(this.segment.data));
|
||||||
arr.set(new Uint8Array(bytes));
|
arr.set(new Uint8Array(bytes));
|
||||||
|
|
||||||
d_seg.data = buf;
|
this.segment.data = buf;
|
||||||
|
} else {
|
||||||
|
logger.error(`Line ${this.line_no}: Expected data segment.`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -223,9 +226,10 @@ class Assembler {
|
|||||||
|
|
||||||
this.segment = string_segment;
|
this.segment = string_segment;
|
||||||
this.object_code.push(string_segment);
|
this.object_code.push(string_segment);
|
||||||
|
} else if (this.segment.type === SegmentType.String) {
|
||||||
|
this.segment.value += str;
|
||||||
} else {
|
} else {
|
||||||
const s_seg = this.segment as StringSegment;
|
logger.error(`Line ${this.line_no}: Expected string segment.`);
|
||||||
s_seg.value += str;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -276,7 +280,7 @@ class Assembler {
|
|||||||
|
|
||||||
const next_token = this.tokens.shift();
|
const next_token = this.tokens.shift();
|
||||||
|
|
||||||
switch (this.section_type) {
|
switch (this.section) {
|
||||||
case SegmentType.Instructions:
|
case SegmentType.Instructions:
|
||||||
this.segment = {
|
this.segment = {
|
||||||
type: SegmentType.Instructions,
|
type: SegmentType.Instructions,
|
||||||
@ -348,21 +352,21 @@ class Assembler {
|
|||||||
col,
|
col,
|
||||||
len,
|
len,
|
||||||
}: CodeSectionToken | DataSectionToken | StringSectionToken): void {
|
}: CodeSectionToken | DataSectionToken | StringSectionToken): void {
|
||||||
let section_type!: SegmentType;
|
let section!: SegmentType;
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case TokenType.CodeSection:
|
case TokenType.CodeSection:
|
||||||
section_type = SegmentType.Instructions;
|
section = SegmentType.Instructions;
|
||||||
break;
|
break;
|
||||||
case TokenType.DataSection:
|
case TokenType.DataSection:
|
||||||
section_type = SegmentType.Data;
|
section = SegmentType.Data;
|
||||||
break;
|
break;
|
||||||
case TokenType.StringSection:
|
case TokenType.StringSection:
|
||||||
section_type = SegmentType.String;
|
section = SegmentType.String;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.section_type === section_type) {
|
if (this.section === section) {
|
||||||
this.add_warning({
|
this.add_warning({
|
||||||
col,
|
col,
|
||||||
length: len,
|
length: len,
|
||||||
@ -370,7 +374,7 @@ class Assembler {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
this.section_type = section_type;
|
this.section = section;
|
||||||
|
|
||||||
const next_token = this.tokens.shift();
|
const next_token = this.tokens.shift();
|
||||||
|
|
||||||
@ -479,7 +483,9 @@ class Assembler {
|
|||||||
if (param.type instanceof RegTupRefType) {
|
if (param.type instanceof RegTupRefType) {
|
||||||
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
|
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
|
||||||
} else {
|
} else {
|
||||||
logger.error(`Type ${param.type} not implemented.`);
|
logger.error(
|
||||||
|
`Line ${this.line_no}: Type ${param.type} not implemented.`
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
@ -1,5 +1,10 @@
|
|||||||
import { NewObjectCodeOutput, ScriptWorkerInput } from "./assembler_messages";
|
import { NewObjectCodeOutput, ScriptWorkerInput } from "./assembler_messages";
|
||||||
import { assemble } from "./assembly";
|
import { assemble } from "./assembly";
|
||||||
|
import Logger from "js-logger";
|
||||||
|
|
||||||
|
Logger.useDefaults({
|
||||||
|
defaultLevel: (Logger as any)[process.env["LOG_LEVEL"] || "OFF"],
|
||||||
|
});
|
||||||
|
|
||||||
const ctx: Worker = self as any;
|
const ctx: Worker = self as any;
|
||||||
|
|
||||||
|
@ -1437,13 +1437,13 @@ export class Opcode {
|
|||||||
undefined,
|
undefined,
|
||||||
[
|
[
|
||||||
new Param(
|
new Param(
|
||||||
new RegTupRefType(new Param(TYPE_BYTE, undefined, ParamAccess.Write)),
|
new RegTupRefType(new Param(TYPE_DWORD, undefined, ParamAccess.Write)),
|
||||||
undefined,
|
undefined,
|
||||||
undefined
|
undefined
|
||||||
),
|
),
|
||||||
new Param(TYPE_DWORD, undefined, undefined),
|
new Param(TYPE_DWORD, "Player slot.", undefined),
|
||||||
],
|
],
|
||||||
undefined
|
StackInteraction.Pop
|
||||||
));
|
));
|
||||||
static readonly P_DISABLEWARP = (OPCODES[0x6b] = new Opcode(
|
static readonly P_DISABLEWARP = (OPCODES[0x6b] = new Opcode(
|
||||||
0x6b,
|
0x6b,
|
||||||
@ -5049,8 +5049,16 @@ export class Opcode {
|
|||||||
"npc_action_string",
|
"npc_action_string",
|
||||||
undefined,
|
undefined,
|
||||||
[
|
[
|
||||||
new Param(TYPE_DWORD, undefined, undefined),
|
new Param(
|
||||||
new Param(TYPE_DWORD, undefined, undefined),
|
new RegTupRefType(new Param(TYPE_DWORD, undefined, ParamAccess.Read)),
|
||||||
|
undefined,
|
||||||
|
undefined
|
||||||
|
),
|
||||||
|
new Param(
|
||||||
|
new RegTupRefType(new Param(TYPE_DWORD, undefined, ParamAccess.Read)),
|
||||||
|
undefined,
|
||||||
|
undefined
|
||||||
|
),
|
||||||
new Param(TYPE_S_LABEL, undefined, undefined),
|
new Param(TYPE_S_LABEL, undefined, undefined),
|
||||||
],
|
],
|
||||||
undefined
|
undefined
|
||||||
@ -5907,9 +5915,13 @@ export class Opcode {
|
|||||||
"get_coord_player_detect",
|
"get_coord_player_detect",
|
||||||
undefined,
|
undefined,
|
||||||
[
|
[
|
||||||
new Param(TYPE_DWORD, "Player slot.", undefined),
|
|
||||||
new Param(
|
new Param(
|
||||||
new RegTupRefType(new Param(TYPE_DWORD, undefined, ParamAccess.Write)),
|
new RegTupRefType(new Param(TYPE_ANY, "Player slot.", ParamAccess.Read)),
|
||||||
|
undefined,
|
||||||
|
undefined
|
||||||
|
),
|
||||||
|
new Param(
|
||||||
|
new RegTupRefType(new Param(TYPE_ANY, undefined, ParamAccess.Read)),
|
||||||
undefined,
|
undefined,
|
||||||
undefined
|
undefined
|
||||||
),
|
),
|
||||||
|
@ -154,7 +154,7 @@ type MonacoProps = {
|
|||||||
class MonacoComponent extends Component<MonacoProps> {
|
class MonacoComponent extends Component<MonacoProps> {
|
||||||
private div_ref = createRef<HTMLDivElement>();
|
private div_ref = createRef<HTMLDivElement>();
|
||||||
private editor?: editor.IStandaloneCodeEditor;
|
private editor?: editor.IStandaloneCodeEditor;
|
||||||
private assembler?: AssemblyAnalyser;
|
private assembly_analyser?: AssemblyAnalyser;
|
||||||
private disposers: (() => void)[] = [];
|
private disposers: (() => void)[] = [];
|
||||||
|
|
||||||
render(): ReactNode {
|
render(): ReactNode {
|
||||||
@ -173,7 +173,7 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
wrappingIndent: "indent",
|
wrappingIndent: "indent",
|
||||||
});
|
});
|
||||||
|
|
||||||
this.assembler = new AssemblyAnalyser();
|
this.assembly_analyser = new AssemblyAnalyser();
|
||||||
|
|
||||||
this.disposers.push(
|
this.disposers.push(
|
||||||
this.dispose,
|
this.dispose,
|
||||||
@ -205,8 +205,8 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
private update_model = () => {
|
private update_model = () => {
|
||||||
const quest = quest_editor_store.current_quest;
|
const quest = quest_editor_store.current_quest;
|
||||||
|
|
||||||
if (quest && this.editor && this.assembler) {
|
if (quest && this.editor && this.assembly_analyser) {
|
||||||
const assembly = this.assembler.disassemble(quest.object_code);
|
const assembly = this.assembly_analyser.disassemble(quest.object_code);
|
||||||
const model = editor.createModel(assembly.join("\n"), "psoasm");
|
const model = editor.createModel(assembly.join("\n"), "psoasm");
|
||||||
|
|
||||||
quest_editor_store.script_undo.action = new Action(
|
quest_editor_store.script_undo.action = new Action(
|
||||||
@ -256,8 +256,8 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
|
|
||||||
current_version = version;
|
current_version = version;
|
||||||
|
|
||||||
if (!this.assembler) return;
|
if (!this.assembly_analyser) return;
|
||||||
this.assembler.update_assembly(e.changes);
|
this.assembly_analyser.update_assembly(e.changes);
|
||||||
});
|
});
|
||||||
|
|
||||||
this.disposers.push(() => disposable.dispose());
|
this.disposers.push(() => disposable.dispose());
|
||||||
@ -269,10 +269,10 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
private update_model_markers = () => {
|
private update_model_markers = () => {
|
||||||
if (!this.editor || !this.assembler) return;
|
if (!this.editor || !this.assembly_analyser) return;
|
||||||
|
|
||||||
// Reference errors here to make sure we get mobx updates.
|
// Reference errors here to make sure we get mobx updates.
|
||||||
this.assembler.errors.length;
|
this.assembly_analyser.errors.length;
|
||||||
|
|
||||||
const model = this.editor.getModel();
|
const model = this.editor.getModel();
|
||||||
if (!model) return;
|
if (!model) return;
|
||||||
@ -280,7 +280,7 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
editor.setModelMarkers(
|
editor.setModelMarkers(
|
||||||
model,
|
model,
|
||||||
"psoasm",
|
"psoasm",
|
||||||
this.assembler.errors.map(error => ({
|
this.assembly_analyser.errors.map(error => ({
|
||||||
severity: MarkerSeverity.Error,
|
severity: MarkerSeverity.Error,
|
||||||
message: error.message,
|
message: error.message,
|
||||||
startLineNumber: error.line_no,
|
startLineNumber: error.line_no,
|
||||||
@ -299,8 +299,8 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
this.editor = undefined;
|
this.editor = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.assembler) {
|
if (this.assembly_analyser) {
|
||||||
this.assembler.dispose();
|
this.assembly_analyser.dispose();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user