All bugs resulting from opcode typing changes have been fixed.

This commit is contained in:
Daan Vanden Bosch 2019-08-05 00:14:39 +02:00
parent dbe1b06fa6
commit 1ba7d3b6a7
8 changed files with 89 additions and 51 deletions

View File

@ -854,9 +854,11 @@ opcodes:
params:
- type: reg_tup_ref
reg_tup:
- type: byte
- type: dword
access: write
- type: dword
doc: Player slot.
stack: pop
- code: 0x6b
mnemonic: p_disablewarp
@ -2829,8 +2831,14 @@ opcodes:
- code: 0xf8dc
mnemonic: npc_action_string
params:
- type: dword
- type: dword
- type: reg_tup_ref
reg_tup:
- type: dword
access: read
- type: reg_tup_ref
reg_tup:
- type: dword
access: read
- type: string_label
- code: 0xf8dd
@ -3346,12 +3354,15 @@ opcodes:
- code: 0xf924
mnemonic: get_coord_player_detect
params:
- type: dword
doc: Player slot.
- type: reg_tup_ref
reg_tup: # TODO: determine type and access
- type: dword
access: write
- type: any
doc: Player slot.
access: read
- type: reg_tup_ref
reg_tup: # TODO: determine type and access
- type: any
access: read
- code: 0xf925
mnemonic: read_global_flag

View File

@ -303,7 +303,8 @@ function parse_object_code(
offset += segment.data.byteLength;
break;
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;
default:
throw new Error(`${SegmentType[segment!.type]} not implemented.`);
@ -501,6 +502,7 @@ function parse_instructions_segment(
break;
case TYPE_D_LABEL:
segment_type = SegmentType.Data;
break;
case TYPE_S_LABEL:
segment_type = SegmentType.String;
break;
@ -722,7 +724,9 @@ function write_object_code(
}
}
} 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 {
cursor.write_cursor(new ArrayBufferCursor(segment.data, cursor.endianness));
}

View File

@ -323,7 +323,7 @@ export class AssemblyLexer {
case ".data":
return { type: TokenType.DataSection, col, len: 5 };
case ".string":
return { type: TokenType.DataSection, col, len: 7 };
return { type: TokenType.StringSection, col, len: 7 };
default:
return { type: TokenType.InvalidSection, col, len: this.marked_len() };
}

View File

@ -1,6 +1,6 @@
import { editor } from "monaco-editor";
import { Segment } from "../data_formats/parsing/quest/bin";
import { AssemblyError } from "./assembly";
import { Segment } from "./instructions";
export type ScriptWorkerInput = NewAssemblyInput | AssemblyChangeInput;

View File

@ -74,7 +74,7 @@ class Assembler {
private errors!: AssemblyError[];
// Encountered labels.
private labels!: Set<number>;
private section_type: SegmentType = SegmentType.Instructions;
private section: SegmentType = SegmentType.Instructions;
constructor(private assembly: string[], private manual_stack: boolean) {}
@ -89,7 +89,7 @@ class Assembler {
this.errors = [];
this.labels = new Set();
// 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) {
this.tokens = this.lexer.tokenize_line(line);
@ -107,7 +107,7 @@ class Assembler {
this.parse_section(token);
break;
case TokenType.Int:
if (this.section_type === SegmentType.Data) {
if (this.section === SegmentType.Data) {
this.parse_bytes(token);
} else {
this.add_error({
@ -118,7 +118,7 @@ class Assembler {
}
break;
case TokenType.String:
if (this.section_type === SegmentType.String) {
if (this.section === SegmentType.String) {
this.parse_string(token);
} else {
this.add_error({
@ -129,7 +129,7 @@ class Assembler {
}
break;
case TokenType.Ident:
if (this.section_type === SegmentType.Instructions) {
if (this.section === SegmentType.Instructions) {
this.parse_instruction(token);
} else {
this.add_error({
@ -184,9 +184,11 @@ class Assembler {
this.segment = 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 {
@ -200,15 +202,16 @@ class Assembler {
this.segment = data_segment;
this.object_code.push(data_segment);
} else {
const d_seg = this.segment as DataSegment;
const buf = new ArrayBuffer(d_seg.data.byteLength + bytes.length);
} else if (this.segment.type === SegmentType.Data) {
const buf = new ArrayBuffer(this.segment.data.byteLength + bytes.length);
const arr = new Uint8Array(buf);
arr.set(new Uint8Array(d_seg.data));
arr.set(new Uint8Array(this.segment.data));
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.object_code.push(string_segment);
} else if (this.segment.type === SegmentType.String) {
this.segment.value += str;
} else {
const s_seg = this.segment as StringSegment;
s_seg.value += str;
logger.error(`Line ${this.line_no}: Expected string segment.`);
}
}
@ -276,7 +280,7 @@ class Assembler {
const next_token = this.tokens.shift();
switch (this.section_type) {
switch (this.section) {
case SegmentType.Instructions:
this.segment = {
type: SegmentType.Instructions,
@ -348,21 +352,21 @@ class Assembler {
col,
len,
}: CodeSectionToken | DataSectionToken | StringSectionToken): void {
let section_type!: SegmentType;
let section!: SegmentType;
switch (type) {
case TokenType.CodeSection:
section_type = SegmentType.Instructions;
section = SegmentType.Instructions;
break;
case TokenType.DataSection:
section_type = SegmentType.Data;
section = SegmentType.Data;
break;
case TokenType.StringSection:
section_type = SegmentType.String;
section = SegmentType.String;
break;
}
if (this.section_type === section_type) {
if (this.section === section) {
this.add_warning({
col,
length: len,
@ -370,7 +374,7 @@ class Assembler {
});
}
this.section_type = section_type;
this.section = section;
const next_token = this.tokens.shift();
@ -479,7 +483,9 @@ class Assembler {
if (param.type instanceof RegTupRefType) {
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
} else {
logger.error(`Type ${param.type} not implemented.`);
logger.error(
`Line ${this.line_no}: Type ${param.type} not implemented.`
);
}
break;

View File

@ -1,5 +1,10 @@
import { NewObjectCodeOutput, ScriptWorkerInput } from "./assembler_messages";
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;

View File

@ -1437,13 +1437,13 @@ export class Opcode {
undefined,
[
new Param(
new RegTupRefType(new Param(TYPE_BYTE, undefined, ParamAccess.Write)),
new RegTupRefType(new Param(TYPE_DWORD, undefined, ParamAccess.Write)),
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(
0x6b,
@ -5049,8 +5049,16 @@ export class Opcode {
"npc_action_string",
undefined,
[
new Param(TYPE_DWORD, undefined, undefined),
new Param(TYPE_DWORD, undefined, undefined),
new Param(
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),
],
undefined
@ -5907,9 +5915,13 @@ export class Opcode {
"get_coord_player_detect",
undefined,
[
new Param(TYPE_DWORD, "Player slot.", undefined),
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
),

View File

@ -154,7 +154,7 @@ type MonacoProps = {
class MonacoComponent extends Component<MonacoProps> {
private div_ref = createRef<HTMLDivElement>();
private editor?: editor.IStandaloneCodeEditor;
private assembler?: AssemblyAnalyser;
private assembly_analyser?: AssemblyAnalyser;
private disposers: (() => void)[] = [];
render(): ReactNode {
@ -173,7 +173,7 @@ class MonacoComponent extends Component<MonacoProps> {
wrappingIndent: "indent",
});
this.assembler = new AssemblyAnalyser();
this.assembly_analyser = new AssemblyAnalyser();
this.disposers.push(
this.dispose,
@ -205,8 +205,8 @@ class MonacoComponent extends Component<MonacoProps> {
private update_model = () => {
const quest = quest_editor_store.current_quest;
if (quest && this.editor && this.assembler) {
const assembly = this.assembler.disassemble(quest.object_code);
if (quest && this.editor && this.assembly_analyser) {
const assembly = this.assembly_analyser.disassemble(quest.object_code);
const model = editor.createModel(assembly.join("\n"), "psoasm");
quest_editor_store.script_undo.action = new Action(
@ -256,8 +256,8 @@ class MonacoComponent extends Component<MonacoProps> {
current_version = version;
if (!this.assembler) return;
this.assembler.update_assembly(e.changes);
if (!this.assembly_analyser) return;
this.assembly_analyser.update_assembly(e.changes);
});
this.disposers.push(() => disposable.dispose());
@ -269,10 +269,10 @@ class MonacoComponent extends Component<MonacoProps> {
};
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.
this.assembler.errors.length;
this.assembly_analyser.errors.length;
const model = this.editor.getModel();
if (!model) return;
@ -280,7 +280,7 @@ class MonacoComponent extends Component<MonacoProps> {
editor.setModelMarkers(
model,
"psoasm",
this.assembler.errors.map(error => ({
this.assembly_analyser.errors.map(error => ({
severity: MarkerSeverity.Error,
message: error.message,
startLineNumber: error.line_no,
@ -299,8 +299,8 @@ class MonacoComponent extends Component<MonacoProps> {
this.editor = undefined;
}
if (this.assembler) {
this.assembler.dispose();
if (this.assembly_analyser) {
this.assembly_analyser.dispose();
}
};
}