mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 23:38:30 +08:00
[VM] Added an interface for handling the virtual machine's IO.
This commit is contained in:
parent
4cd0b58f66
commit
f3c26f9eda
21
src/quest_editor/scripting/vm/VMIOStub.ts
Normal file
21
src/quest_editor/scripting/vm/VMIOStub.ts
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import { stub } from "../../../core/decorators";
|
||||||
|
import { AsmToken } from "../instructions";
|
||||||
|
import { VirtualMachineIO } from "./io";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* All methods of VirtualMachineIO implemented as stubs.
|
||||||
|
*/
|
||||||
|
export class VMIOStub implements VirtualMachineIO {
|
||||||
|
@stub
|
||||||
|
advance_msg(): Promise<any> {
|
||||||
|
return Promise.resolve();
|
||||||
|
}
|
||||||
|
|
||||||
|
@stub window_msg(msg: string): void {}
|
||||||
|
@stub message(msg: string): void {}
|
||||||
|
@stub add_msg(msg: string): void {}
|
||||||
|
@stub winend(): void {}
|
||||||
|
@stub mesend(): void {}
|
||||||
|
@stub warning(msg: string, srcloc?: AsmToken): void {}
|
||||||
|
@stub error(msg: string, srcloc?: AsmToken): void {}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
import { Instruction, InstructionSegment, Segment, SegmentType } from "../instructions";
|
import { Instruction, InstructionSegment, Segment, SegmentType, AsmToken } from "../instructions";
|
||||||
import {
|
import {
|
||||||
OP_ADD,
|
OP_ADD,
|
||||||
OP_ADDI,
|
OP_ADDI,
|
||||||
@ -89,6 +89,9 @@ import {
|
|||||||
andsecond,
|
andsecond,
|
||||||
andreduce,
|
andreduce,
|
||||||
} from "./utils";
|
} from "./utils";
|
||||||
|
import { VirtualMachineIO } from "./io";
|
||||||
|
import { VMIOStub } from "./VMIOStub";
|
||||||
|
import { Source } from "webpack-sources";
|
||||||
|
|
||||||
const logger = Logger.get("quest_editor/scripting/vm");
|
const logger = Logger.get("quest_editor/scripting/vm");
|
||||||
|
|
||||||
@ -123,6 +126,8 @@ export class VirtualMachine {
|
|||||||
private thread_idx = 0;
|
private thread_idx = 0;
|
||||||
private window_msg_open = false;
|
private window_msg_open = false;
|
||||||
|
|
||||||
|
constructor(private io: VirtualMachineIO = new VMIOStub()) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Halts and resets the VM, then loads new object code.
|
* Halts and resets the VM, then loads new object code.
|
||||||
*/
|
*/
|
||||||
@ -163,6 +168,7 @@ export class VirtualMachine {
|
|||||||
|
|
||||||
this.thread.push(
|
this.thread.push(
|
||||||
new Thread(
|
new Thread(
|
||||||
|
this.io,
|
||||||
new ExecutionLocation(seg_idx!, 0),
|
new ExecutionLocation(seg_idx!, 0),
|
||||||
this.memory.allocate(ARG_STACK_SLOT_SIZE * ARG_STACK_LENGTH),
|
this.memory.allocate(ARG_STACK_SLOT_SIZE * ARG_STACK_LENGTH),
|
||||||
true,
|
true,
|
||||||
@ -177,16 +183,30 @@ export class VirtualMachine {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Executes the next instruction if one is scheduled.
|
* Executes the next instruction if one is scheduled.
|
||||||
*
|
|
||||||
* @returns true if an instruction was executed, false otherwise.
|
|
||||||
*/
|
*/
|
||||||
execute(): ExecutionResult {
|
execute(): ExecutionResult {
|
||||||
|
let srcloc: AsmToken | undefined;
|
||||||
|
try {
|
||||||
|
const exec = this.thread[this.thread_idx];
|
||||||
|
const inst = this.get_next_instruction_from_thread(exec);
|
||||||
|
|
||||||
|
if (inst.asm && inst.asm.mnemonic) {
|
||||||
|
srcloc = inst.asm.mnemonic;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.execute_instruction(exec, inst);
|
||||||
|
} catch (err) {
|
||||||
|
this.halt();
|
||||||
|
this.io.error(err, srcloc);
|
||||||
|
|
||||||
|
return ExecutionResult.Halted;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private execute_instruction(exec: Thread, inst: Instruction): ExecutionResult {
|
||||||
if (this.thread.length === 0) return ExecutionResult.Halted;
|
if (this.thread.length === 0) return ExecutionResult.Halted;
|
||||||
if (this.thread_idx >= this.thread.length) return ExecutionResult.WaitingVsync;
|
if (this.thread_idx >= this.thread.length) return ExecutionResult.WaitingVsync;
|
||||||
|
|
||||||
const exec = this.thread[this.thread_idx];
|
|
||||||
const inst = this.get_next_instruction_from_thread(exec);
|
|
||||||
|
|
||||||
const arg_vals = inst.args.map(arg => arg.value);
|
const arg_vals = inst.args.map(arg => arg.value);
|
||||||
// eslint-disable-next-line
|
// eslint-disable-next-line
|
||||||
const [arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7] = arg_vals;
|
const [arg0, arg1, arg2, arg3, arg4, arg5, arg6, arg7] = arg_vals;
|
||||||
@ -480,8 +500,7 @@ export class VirtualMachine {
|
|||||||
const str = this.deref_string(args[0]);
|
const str = this.deref_string(args[0]);
|
||||||
|
|
||||||
this.window_msg_open = true;
|
this.window_msg_open = true;
|
||||||
console.group("window_msg");
|
this.io.window_msg(str);
|
||||||
console.log(str);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case OP_ADD_MSG.code:
|
case OP_ADD_MSG.code:
|
||||||
@ -489,13 +508,13 @@ export class VirtualMachine {
|
|||||||
const args = exec.fetch_args(inst.opcode.params);
|
const args = exec.fetch_args(inst.opcode.params);
|
||||||
const str = this.deref_string(args[0]);
|
const str = this.deref_string(args[0]);
|
||||||
|
|
||||||
console.log(str);
|
this.io.add_msg(str);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case OP_WINEND.code:
|
case OP_WINEND.code:
|
||||||
if (this.window_msg_open) {
|
if (this.window_msg_open) {
|
||||||
this.window_msg_open = false;
|
this.window_msg_open = false;
|
||||||
console.groupEnd();
|
this.io.winend();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@ -581,12 +600,12 @@ export class VirtualMachine {
|
|||||||
const seg_idx = this.label_to_seg_idx.get(label);
|
const seg_idx = this.label_to_seg_idx.get(label);
|
||||||
|
|
||||||
if (seg_idx == undefined) {
|
if (seg_idx == undefined) {
|
||||||
logger.warn(`Invalid label called: ${label}.`);
|
this.io.warning(`Invalid label called: ${label}.`);
|
||||||
} else {
|
} else {
|
||||||
const segment = this.object_code[seg_idx];
|
const segment = this.object_code[seg_idx];
|
||||||
|
|
||||||
if (segment.type !== SegmentType.Instructions) {
|
if (segment.type !== SegmentType.Instructions) {
|
||||||
logger.warn(
|
this.io.warning(
|
||||||
`Label ${label} points to a ${SegmentType[segment.type]} segment, expecting ${
|
`Label ${label} points to a ${SegmentType[segment.type]} segment, expecting ${
|
||||||
SegmentType[SegmentType.Instructions]
|
SegmentType[SegmentType.Instructions]
|
||||||
}.`,
|
}.`,
|
||||||
@ -620,7 +639,7 @@ export class VirtualMachine {
|
|||||||
const seg_idx = this.label_to_seg_idx.get(label);
|
const seg_idx = this.label_to_seg_idx.get(label);
|
||||||
|
|
||||||
if (seg_idx == undefined) {
|
if (seg_idx == undefined) {
|
||||||
logger.warn(`Invalid jump label: ${label}.`);
|
this.io.warning(`Invalid jump label: ${label}.`);
|
||||||
} else {
|
} else {
|
||||||
top.seg_idx = seg_idx;
|
top.seg_idx = seg_idx;
|
||||||
top.inst_idx = -1;
|
top.inst_idx = -1;
|
||||||
@ -784,7 +803,12 @@ class Thread {
|
|||||||
*/
|
*/
|
||||||
public global: boolean;
|
public global: boolean;
|
||||||
|
|
||||||
constructor(next: ExecutionLocation, arg_stack: VirtualMachineMemoryBuffer, global: boolean) {
|
constructor(
|
||||||
|
public io: VirtualMachineIO,
|
||||||
|
next: ExecutionLocation,
|
||||||
|
arg_stack: VirtualMachineMemoryBuffer,
|
||||||
|
global: boolean,
|
||||||
|
) {
|
||||||
this.call_stack = [next];
|
this.call_stack = [next];
|
||||||
this.global = global;
|
this.global = global;
|
||||||
|
|
||||||
@ -810,14 +834,14 @@ class Thread {
|
|||||||
const args: number[] = [];
|
const args: number[] = [];
|
||||||
|
|
||||||
if (params.length !== this.arg_stack_counter) {
|
if (params.length !== this.arg_stack_counter) {
|
||||||
logger.warn("Argument stack: Argument count mismatch");
|
this.io.warning("Argument stack: Argument count mismatch");
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let i = 0; i < params.length; i++) {
|
for (let i = 0; i < params.length; i++) {
|
||||||
const param = params[i];
|
const param = params[i];
|
||||||
|
|
||||||
if (param.type.kind !== this.arg_stack_types[i]) {
|
if (param.type.kind !== this.arg_stack_types[i]) {
|
||||||
logger.warn("Argument stack: Argument type mismatch");
|
this.io.warning("Argument stack: Argument type mismatch");
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (param.type.kind) {
|
switch (param.type.kind) {
|
||||||
@ -832,7 +856,7 @@ class Thread {
|
|||||||
args.push(this.arg_stack.u32_at(i * ARG_STACK_SLOT_SIZE));
|
args.push(this.arg_stack.u32_at(i * ARG_STACK_SLOT_SIZE));
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Unhandled param kind: Kind.${Kind[param.type.kind]}`);
|
throw new Error(`Argument stack: Unhandled param kind: Kind.${Kind[param.type.kind]}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
42
src/quest_editor/scripting/vm/io.ts
Normal file
42
src/quest_editor/scripting/vm/io.ts
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
import { AsmToken } from "../instructions";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The virtual machine calls these methods when it requires input.
|
||||||
|
*/
|
||||||
|
export interface VirtualMachineInput {
|
||||||
|
/**
|
||||||
|
* End the current message and move to the next message.
|
||||||
|
*/
|
||||||
|
advance_msg(): Promise<any>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The virtual machine calls these methods when it outputs something.
|
||||||
|
*/
|
||||||
|
export interface VirtualMachineOutput {
|
||||||
|
window_msg(msg: string): void;
|
||||||
|
message(msg: string): void;
|
||||||
|
add_msg(msg: string): void;
|
||||||
|
winend(): void;
|
||||||
|
mesend(): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Methods that are outside of the context of the game.
|
||||||
|
*/
|
||||||
|
export interface VirtualMachineMetaIO {
|
||||||
|
/**
|
||||||
|
* The virtual machine emits warning messages about suspicious execution
|
||||||
|
* patterns that could possibly cause problems or have unintended effects.
|
||||||
|
*/
|
||||||
|
warning(msg: string, srcloc?: AsmToken): void;
|
||||||
|
error(msg: string, srcloc?: AsmToken): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handles input/output to/from the virtual machine.
|
||||||
|
*/
|
||||||
|
export interface VirtualMachineIO
|
||||||
|
extends VirtualMachineInput,
|
||||||
|
VirtualMachineOutput,
|
||||||
|
VirtualMachineMetaIO {}
|
Loading…
Reference in New Issue
Block a user