mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
Fixed bug that would disassemble arg_pushr arguments incorrectly in automatic stack management mode.
This commit is contained in:
parent
bdee123ee3
commit
3c6b28f536
@ -2,27 +2,27 @@ import { editor, languages } from "monaco-editor";
|
|||||||
import AssemblyWorker from "worker-loader!./assembly_worker";
|
import AssemblyWorker from "worker-loader!./assembly_worker";
|
||||||
import {
|
import {
|
||||||
AssemblyChangeInput,
|
AssemblyChangeInput,
|
||||||
|
AssemblySettingsChangeInput,
|
||||||
AssemblyWorkerOutput,
|
AssemblyWorkerOutput,
|
||||||
InputMessageType,
|
InputMessageType,
|
||||||
NewAssemblyInput,
|
NewAssemblyInput,
|
||||||
OutputMessageType,
|
OutputMessageType,
|
||||||
SignatureHelpInput,
|
SignatureHelpInput,
|
||||||
AssemblySettingsChangeInput,
|
|
||||||
} from "./assembly_worker_messages";
|
} from "./assembly_worker_messages";
|
||||||
import { AssemblyError, AssemblyWarning, AssemblySettings } from "./assembly";
|
import { AssemblyError, AssemblySettings, AssemblyWarning } from "./assembly";
|
||||||
import { disassemble } from "./disassembly";
|
import { disassemble } from "./disassembly";
|
||||||
import { QuestModel } from "../model/QuestModel";
|
import { QuestModel } from "../model/QuestModel";
|
||||||
import { Kind, OPCODES } from "./opcodes";
|
import { Kind, OPCODES } from "./opcodes";
|
||||||
import { Property } from "../../core/observable/property/Property";
|
import { Property } from "../../core/observable/property/Property";
|
||||||
import { property } from "../../core/observable";
|
import { property } from "../../core/observable";
|
||||||
import { WritableProperty } from "../../core/observable/property/WritableProperty";
|
import { WritableProperty } from "../../core/observable/property/WritableProperty";
|
||||||
|
import { Disposable } from "../../core/observable/Disposable";
|
||||||
import CompletionList = languages.CompletionList;
|
import CompletionList = languages.CompletionList;
|
||||||
import CompletionItemKind = languages.CompletionItemKind;
|
import CompletionItemKind = languages.CompletionItemKind;
|
||||||
import CompletionItem = languages.CompletionItem;
|
import CompletionItem = languages.CompletionItem;
|
||||||
import IModelContentChange = editor.IModelContentChange;
|
import IModelContentChange = editor.IModelContentChange;
|
||||||
import SignatureHelp = languages.SignatureHelp;
|
import SignatureHelp = languages.SignatureHelp;
|
||||||
import ParameterInformation = languages.ParameterInformation;
|
import ParameterInformation = languages.ParameterInformation;
|
||||||
import { Disposable } from "../../core/observable/Disposable";
|
|
||||||
|
|
||||||
const INSTRUCTION_SUGGESTIONS = OPCODES.filter(opcode => opcode != null).map(opcode => {
|
const INSTRUCTION_SUGGESTIONS = OPCODES.filter(opcode => opcode != null).map(opcode => {
|
||||||
return ({
|
return ({
|
||||||
@ -75,7 +75,7 @@ export class AssemblyAnalyser implements Disposable {
|
|||||||
this.worker.onmessage = this.process_worker_message;
|
this.worker.onmessage = this.process_worker_message;
|
||||||
}
|
}
|
||||||
|
|
||||||
disassemble(quest: QuestModel, manual_stack?: boolean): string[] {
|
disassemble(quest: QuestModel, manual_stack: boolean): string[] {
|
||||||
this.quest = quest;
|
this.quest = quest;
|
||||||
const assembly = disassemble(quest.object_code, manual_stack);
|
const assembly = disassemble(quest.object_code, manual_stack);
|
||||||
const message: NewAssemblyInput = { type: InputMessageType.NewAssembly, assembly };
|
const message: NewAssemblyInput = { type: InputMessageType.NewAssembly, assembly };
|
||||||
|
@ -47,7 +47,15 @@ export function assemble(
|
|||||||
warnings: AssemblyWarning[];
|
warnings: AssemblyWarning[];
|
||||||
errors: AssemblyError[];
|
errors: AssemblyError[];
|
||||||
} {
|
} {
|
||||||
return new Assembler(assembly, manual_stack).assemble();
|
logger.trace("assemble start");
|
||||||
|
|
||||||
|
const result = new Assembler(assembly, manual_stack).assemble();
|
||||||
|
|
||||||
|
logger.trace(
|
||||||
|
`assemble end with ${result.warnings.length} warnings and ${result.errors.length} errors.`,
|
||||||
|
);
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
class Assembler {
|
class Assembler {
|
||||||
|
@ -1,14 +1,27 @@
|
|||||||
import { reinterpret_i32_as_f32 } from "../../core/primitive_conversion";
|
import { reinterpret_i32_as_f32 } from "../../core/primitive_conversion";
|
||||||
import { Arg, Segment, SegmentType } from "./instructions";
|
import { Arg, Segment, SegmentType } from "./instructions";
|
||||||
import { Kind, Param, StackInteraction } from "./opcodes";
|
import { AnyType, Kind, Param, StackInteraction } from "./opcodes";
|
||||||
|
import Logger from "js-logger";
|
||||||
|
|
||||||
|
const logger = Logger.get("quest_editor/scripting/disassembly");
|
||||||
|
|
||||||
|
type ArgWithType = Arg & {
|
||||||
|
/**
|
||||||
|
* Type inferred from the specific instruction used to push this argument onto the stack.
|
||||||
|
*/
|
||||||
|
type: AnyType;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param manual_stack If true, will output stack management instructions (argpush variants). Otherwise the arguments of stack management instructions will be output as arguments to the instruction that pops them from the stack.
|
* @param object_code - The object code to disassemble.
|
||||||
|
* @param manual_stack - If true, will output stack management instructions (argpush variants). Otherwise the arguments of stack management instructions will be output as arguments to the instruction that pops them from the stack.
|
||||||
*/
|
*/
|
||||||
export function disassemble(object_code: Segment[], manual_stack: boolean = false): string[] {
|
export function disassemble(object_code: Segment[], manual_stack: boolean = false): string[] {
|
||||||
|
logger.trace("disassemble start");
|
||||||
|
|
||||||
const lines: string[] = [];
|
const lines: string[] = [];
|
||||||
const stack: Arg[] = [];
|
const stack: ArgWithType[] = [];
|
||||||
let section_type: SegmentType | undefined;
|
let section_type: SegmentType | undefined = undefined;
|
||||||
|
|
||||||
for (const segment of object_code) {
|
for (const segment of object_code) {
|
||||||
// Section marker.
|
// Section marker.
|
||||||
@ -65,7 +78,7 @@ export function disassemble(object_code: Segment[], manual_stack: boolean = fals
|
|||||||
} 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) {
|
||||||
stack.push(...instruction.args);
|
stack.push(...add_type_to_args(instruction.opcode.params, instruction.args));
|
||||||
} else {
|
} else {
|
||||||
let args: string[] = [];
|
let args: string[] = [];
|
||||||
|
|
||||||
@ -81,7 +94,11 @@ export function disassemble(object_code: Segment[], manual_stack: boolean = fals
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
args = args_to_strings(instruction.opcode.params, instruction.args, false);
|
args = args_to_strings(
|
||||||
|
instruction.opcode.params,
|
||||||
|
add_type_to_args(instruction.opcode.params, instruction.args),
|
||||||
|
false,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
lines.push(
|
lines.push(
|
||||||
@ -99,10 +116,22 @@ export function disassemble(object_code: Segment[], manual_stack: boolean = fals
|
|||||||
lines.push("");
|
lines.push("");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
logger.trace(`disassemble end, line count: ${lines.length}`);
|
||||||
return lines;
|
return lines;
|
||||||
}
|
}
|
||||||
|
|
||||||
function args_to_strings(params: Param[], args: Arg[], stack: boolean): string[] {
|
function add_type_to_args(params: Param[], args: Arg[]): ArgWithType[] {
|
||||||
|
const args_with_type: ArgWithType[] = [];
|
||||||
|
const len = Math.min(params.length, args.length);
|
||||||
|
|
||||||
|
for (let i = 0; i < len; i++) {
|
||||||
|
args_with_type.push({ ...args[i], type: params[i].type });
|
||||||
|
}
|
||||||
|
|
||||||
|
return args_with_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
function args_to_strings(params: Param[], args: ArgWithType[], stack: boolean): string[] {
|
||||||
const arg_strings: string[] = [];
|
const arg_strings: string[] = [];
|
||||||
|
|
||||||
for (let i = 0; i < params.length; i++) {
|
for (let i = 0; i < params.length; i++) {
|
||||||
@ -114,35 +143,39 @@ function args_to_strings(params: Param[], args: Arg[], stack: boolean): string[]
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (type.kind) {
|
if (arg.type.kind === Kind.RegTupRef) {
|
||||||
case Kind.Float:
|
arg_strings.push("r" + arg.value);
|
||||||
// Floats are pushed onto the stack as integers with arg_pushl.
|
} else {
|
||||||
if (stack) {
|
switch (type.kind) {
|
||||||
arg_strings.push(reinterpret_i32_as_f32(arg.value).toString());
|
case Kind.Float:
|
||||||
} else {
|
// Floats are pushed onto the stack as integers with arg_pushl.
|
||||||
|
if (stack) {
|
||||||
|
arg_strings.push(reinterpret_i32_as_f32(arg.value).toString());
|
||||||
|
} else {
|
||||||
|
arg_strings.push(arg.value.toString());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case Kind.ILabelVar:
|
||||||
|
for (; i < args.length; i++) {
|
||||||
|
arg_strings.push(args[i].value.toString());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case Kind.RegRefVar:
|
||||||
|
for (; i < args.length; i++) {
|
||||||
|
arg_strings.push("r" + args[i].value);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case Kind.RegRef:
|
||||||
|
case Kind.RegTupRef:
|
||||||
|
arg_strings.push("r" + arg.value);
|
||||||
|
break;
|
||||||
|
case Kind.String:
|
||||||
|
arg_strings.push(JSON.stringify(arg.value));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
arg_strings.push(arg.value.toString());
|
arg_strings.push(arg.value.toString());
|
||||||
}
|
break;
|
||||||
break;
|
}
|
||||||
case Kind.ILabelVar:
|
|
||||||
for (; i < args.length; i++) {
|
|
||||||
arg_strings.push(args[i].value.toString());
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case Kind.RegRefVar:
|
|
||||||
for (; i < args.length; i++) {
|
|
||||||
arg_strings.push("r" + args[i].value);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case Kind.RegRef:
|
|
||||||
case Kind.RegTupRef:
|
|
||||||
arg_strings.push("r" + arg.value);
|
|
||||||
break;
|
|
||||||
case Kind.String:
|
|
||||||
arg_strings.push(JSON.stringify(arg.value));
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
arg_strings.push(arg.value.toString());
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,7 +22,7 @@ export enum Kind {
|
|||||||
/**
|
/**
|
||||||
* Abstract super type of all types.
|
* Abstract super type of all types.
|
||||||
*/
|
*/
|
||||||
type AnyType =
|
export type AnyType =
|
||||||
| ValueType
|
| ValueType
|
||||||
| RefType
|
| RefType
|
||||||
| PointerType
|
| PointerType
|
||||||
@ -33,40 +33,40 @@ type AnyType =
|
|||||||
/**
|
/**
|
||||||
* Purely abstract super type of all value types.
|
* Purely abstract super type of all value types.
|
||||||
*/
|
*/
|
||||||
type ValueType = ByteType | WordType | DWordType | FloatType | LabelType;
|
export type ValueType = ByteType | WordType | DWordType | FloatType | LabelType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 8-Bit integer.
|
* 8-Bit integer.
|
||||||
*/
|
*/
|
||||||
type ByteType = {
|
export type ByteType = {
|
||||||
readonly kind: Kind.Byte;
|
readonly kind: Kind.Byte;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 16-Bit integer.
|
* 16-Bit integer.
|
||||||
*/
|
*/
|
||||||
type WordType = {
|
export type WordType = {
|
||||||
readonly kind: Kind.Word;
|
readonly kind: Kind.Word;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 32-Bit integer.
|
* 32-Bit integer.
|
||||||
*/
|
*/
|
||||||
type DWordType = {
|
export type DWordType = {
|
||||||
readonly kind: Kind.DWord;
|
readonly kind: Kind.DWord;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 32-Bit floating point number.
|
* 32-Bit floating point number.
|
||||||
*/
|
*/
|
||||||
type FloatType = {
|
export type FloatType = {
|
||||||
readonly kind: Kind.Float;
|
readonly kind: Kind.Float;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract super type of all label types.
|
* Abstract super type of all label types.
|
||||||
*/
|
*/
|
||||||
type LabelType =
|
export type LabelType =
|
||||||
| ILabelType
|
| ILabelType
|
||||||
| DLabelType
|
| DLabelType
|
||||||
| SLabelType
|
| SLabelType
|
||||||
@ -79,47 +79,47 @@ type LabelType =
|
|||||||
/**
|
/**
|
||||||
* Named reference to an instruction.
|
* Named reference to an instruction.
|
||||||
*/
|
*/
|
||||||
type ILabelType = {
|
export type ILabelType = {
|
||||||
readonly kind: Kind.ILabel;
|
readonly kind: Kind.ILabel;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Named reference to a data segment.
|
* Named reference to a data segment.
|
||||||
*/
|
*/
|
||||||
type DLabelType = {
|
export type DLabelType = {
|
||||||
readonly kind: Kind.DLabel;
|
readonly kind: Kind.DLabel;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Named reference to a string segment.
|
* Named reference to a string segment.
|
||||||
*/
|
*/
|
||||||
type SLabelType = {
|
export type SLabelType = {
|
||||||
readonly kind: Kind.SLabel;
|
readonly kind: Kind.SLabel;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* String of arbitrary size.
|
* String of arbitrary size.
|
||||||
*/
|
*/
|
||||||
type StringType = {
|
export type StringType = {
|
||||||
readonly kind: Kind.String;
|
readonly kind: Kind.String;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Arbitrary amount of instruction labels.
|
* Arbitrary amount of instruction labels.
|
||||||
*/
|
*/
|
||||||
type ILabelVarType = {
|
export type ILabelVarType = {
|
||||||
readonly kind: Kind.ILabelVar;
|
readonly kind: Kind.ILabelVar;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Purely abstract super type of all reference types.
|
* Purely abstract super type of all reference types.
|
||||||
*/
|
*/
|
||||||
type RefType = RegRefType | RegTupRefType | RegRefVarType;
|
export type RefType = RegRefType | RegTupRefType | RegRefVarType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reference to one or more registers.
|
* Reference to one or more registers.
|
||||||
*/
|
*/
|
||||||
type RegRefType = {
|
export type RegRefType = {
|
||||||
readonly kind: Kind.RegRef;
|
readonly kind: Kind.RegRef;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -127,7 +127,7 @@ type RegRefType = {
|
|||||||
* Reference to a fixed amount of consecutive registers of specific types.
|
* Reference to a fixed amount of consecutive registers of specific types.
|
||||||
* The only parameterized type.
|
* The only parameterized type.
|
||||||
*/
|
*/
|
||||||
type RegTupRefType = {
|
export type RegTupRefType = {
|
||||||
readonly kind: Kind.RegTupRef;
|
readonly kind: Kind.RegTupRef;
|
||||||
readonly register_tuples: Param[];
|
readonly register_tuples: Param[];
|
||||||
};
|
};
|
||||||
@ -135,33 +135,33 @@ type RegTupRefType = {
|
|||||||
/**
|
/**
|
||||||
* Arbitrary amount of register references.
|
* Arbitrary amount of register references.
|
||||||
*/
|
*/
|
||||||
type RegRefVarType = {
|
export type RegRefVarType = {
|
||||||
readonly kind: Kind.RegRefVar;
|
readonly kind: Kind.RegRefVar;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Raw memory pointer.
|
* Raw memory pointer.
|
||||||
*/
|
*/
|
||||||
type PointerType = {
|
export type PointerType = {
|
||||||
readonly kind: Kind.Pointer;
|
readonly kind: Kind.Pointer;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Singleton type constants.
|
// Singleton type constants.
|
||||||
// No singleton constant for `RegTupRef` because it is parameterized.
|
// No singleton constant for `RegTupRef` because it is parameterized.
|
||||||
const TYPE_ANY: AnyType = { kind: Kind.Any };
|
export const TYPE_ANY: AnyType = { kind: Kind.Any };
|
||||||
const TYPE_BYTE: ByteType = { kind: Kind.Byte };
|
export const TYPE_BYTE: ByteType = { kind: Kind.Byte };
|
||||||
const TYPE_WORD: WordType = { kind: Kind.Word };
|
export const TYPE_WORD: WordType = { kind: Kind.Word };
|
||||||
const TYPE_DWORD: DWordType = { kind: Kind.DWord };
|
export const TYPE_DWORD: DWordType = { kind: Kind.DWord };
|
||||||
const TYPE_FLOAT: FloatType = { kind: Kind.Float };
|
export const TYPE_FLOAT: FloatType = { kind: Kind.Float };
|
||||||
const TYPE_LABEL: LabelType = { kind: Kind.Label };
|
export const TYPE_LABEL: LabelType = { kind: Kind.Label };
|
||||||
const TYPE_I_LABEL: ILabelType = { kind: Kind.ILabel };
|
export const TYPE_I_LABEL: ILabelType = { kind: Kind.ILabel };
|
||||||
const TYPE_D_LABEL: DLabelType = { kind: Kind.DLabel };
|
export const TYPE_D_LABEL: DLabelType = { kind: Kind.DLabel };
|
||||||
const TYPE_S_LABEL: SLabelType = { kind: Kind.SLabel };
|
export const TYPE_S_LABEL: SLabelType = { kind: Kind.SLabel };
|
||||||
const TYPE_STRING: StringType = { kind: Kind.String };
|
export const TYPE_STRING: StringType = { kind: Kind.String };
|
||||||
const TYPE_I_LABEL_VAR: ILabelVarType = { kind: Kind.ILabelVar };
|
export const TYPE_I_LABEL_VAR: ILabelVarType = { kind: Kind.ILabelVar };
|
||||||
const TYPE_REG_REF: RegRefType = { kind: Kind.RegRef };
|
export const TYPE_REG_REF: RegRefType = { kind: Kind.RegRef };
|
||||||
const TYPE_REG_REF_VAR: RegRefVarType = { kind: Kind.RegRefVar };
|
export const TYPE_REG_REF_VAR: RegRefVarType = { kind: Kind.RegRefVar };
|
||||||
const TYPE_POINTER: PointerType = { kind: Kind.Pointer };
|
export const TYPE_POINTER: PointerType = { kind: Kind.Pointer };
|
||||||
|
|
||||||
export const MIN_SIGNED_DWORD_VALUE = -Math.pow(2, 31);
|
export const MIN_SIGNED_DWORD_VALUE = -Math.pow(2, 31);
|
||||||
export const MAX_SIGNED_DWORD_VALUE = Math.pow(2, 31) - 1;
|
export const MAX_SIGNED_DWORD_VALUE = Math.pow(2, 31) - 1;
|
||||||
|
@ -99,7 +99,11 @@ export class AsmEditorStore implements Disposable {
|
|||||||
// don't allow changing inline args mode if there are issues
|
// don't allow changing inline args mode if there are issues
|
||||||
if (!this.has_issues.val) {
|
if (!this.has_issues.val) {
|
||||||
this._inline_args_mode.val = inline_args_mode;
|
this._inline_args_mode.val = inline_args_mode;
|
||||||
this.update_assembly_settings();
|
|
||||||
|
assembly_analyser.update_settings({
|
||||||
|
manual_stack: !this.inline_args_mode.val,
|
||||||
|
});
|
||||||
|
|
||||||
this.update_model();
|
this.update_model();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -213,12 +217,6 @@ export class AsmEditorStore implements Disposable {
|
|||||||
this._model.val = undefined;
|
this._model.val = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private update_assembly_settings(): void {
|
|
||||||
assembly_analyser.update_settings({
|
|
||||||
manual_stack: !this.inline_args_mode.val,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const asm_editor_store = new AsmEditorStore();
|
export const asm_editor_store = new AsmEditorStore();
|
||||||
|
Loading…
Reference in New Issue
Block a user