mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 23:38:30 +08:00
[VM] Implemented opcode set_episode and made it possible to load the entire quest in the VM.
This commit is contained in:
parent
89a119e90d
commit
4441435302
@ -81,6 +81,7 @@ import {
|
|||||||
OP_FLETI,
|
OP_FLETI,
|
||||||
OP_GET_RANDOM,
|
OP_GET_RANDOM,
|
||||||
OP_GETTIME,
|
OP_GETTIME,
|
||||||
|
OP_SET_EPISODE,
|
||||||
} from "../opcodes";
|
} from "../opcodes";
|
||||||
import { VirtualMachineMemoryBuffer, VirtualMachineMemory } from "./memory";
|
import { VirtualMachineMemoryBuffer, VirtualMachineMemory } from "./memory";
|
||||||
import {
|
import {
|
||||||
@ -95,6 +96,9 @@ import {
|
|||||||
import { VirtualMachineIO } from "./io";
|
import { VirtualMachineIO } from "./io";
|
||||||
import { VMIOStub } from "./VMIOStub";
|
import { VMIOStub } from "./VMIOStub";
|
||||||
import { rand, srand, GetTickCount } from "./windows";
|
import { rand, srand, GetTickCount } from "./windows";
|
||||||
|
import { QuestModel } from "../../model/QuestModel";
|
||||||
|
import { convert_quest_from_model } from "../../stores/model_conversion";
|
||||||
|
import { Episode } from "../../../core/data_formats/parsing/quest/Episode";
|
||||||
|
|
||||||
const REGISTERS_BASE_ADDRESS = 0x00a954b0;
|
const REGISTERS_BASE_ADDRESS = 0x00a954b0;
|
||||||
const REGISTER_COUNT = 256;
|
const REGISTER_COUNT = 256;
|
||||||
@ -105,6 +109,7 @@ const ARG_STACK_LENGTH = 8;
|
|||||||
const STRING_ARG_STORE_ADDRESS = 0x00a92700;
|
const STRING_ARG_STORE_ADDRESS = 0x00a92700;
|
||||||
const STRING_ARG_STORE_SIZE = 1024; // TODO: verify this value
|
const STRING_ARG_STORE_SIZE = 1024; // TODO: verify this value
|
||||||
const FLOAT_EPSILON = 1.19e-7;
|
const FLOAT_EPSILON = 1.19e-7;
|
||||||
|
const ENTRY_SEGMENT = 0;
|
||||||
|
|
||||||
export enum ExecutionResult {
|
export enum ExecutionResult {
|
||||||
Ok,
|
Ok,
|
||||||
@ -112,6 +117,17 @@ export enum ExecutionResult {
|
|||||||
Halted,
|
Halted,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function encode_episode_number(ep: Episode): number {
|
||||||
|
switch (ep) {
|
||||||
|
case Episode.I:
|
||||||
|
return 0;
|
||||||
|
case Episode.II:
|
||||||
|
return 1;
|
||||||
|
case Episode.IV:
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export class VirtualMachine {
|
export class VirtualMachine {
|
||||||
private memory = new VirtualMachineMemory();
|
private memory = new VirtualMachineMemory();
|
||||||
private registers = this.memory.allocate(
|
private registers = this.memory.allocate(
|
||||||
@ -122,25 +138,41 @@ export class VirtualMachine {
|
|||||||
STRING_ARG_STORE_SIZE,
|
STRING_ARG_STORE_SIZE,
|
||||||
STRING_ARG_STORE_ADDRESS,
|
STRING_ARG_STORE_ADDRESS,
|
||||||
);
|
);
|
||||||
private object_code: Segment[] = [];
|
private quest?: QuestModel;
|
||||||
|
private object_code: readonly Segment[] = [];
|
||||||
private label_to_seg_idx: Map<number, number> = new Map();
|
private label_to_seg_idx: Map<number, number> = new Map();
|
||||||
private thread: Thread[] = [];
|
private thread: Thread[] = [];
|
||||||
private thread_idx = 0;
|
private thread_idx = 0;
|
||||||
private window_msg_open = false;
|
private window_msg_open = false;
|
||||||
|
private set_episode_called = false;
|
||||||
|
|
||||||
constructor(private io: VirtualMachineIO = new VMIOStub()) {
|
constructor(private io: VirtualMachineIO = new VMIOStub()) {
|
||||||
srand(GetTickCount());
|
srand(GetTickCount());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Halts and resets the VM, then loads new object code.
|
* Halts and resets the VM, then loads new quest.
|
||||||
*/
|
*/
|
||||||
load_object_code(object_code: Segment[]): void {
|
load_quest(quest_model: QuestModel): void {
|
||||||
|
const quest = convert_quest_from_model(quest_model);
|
||||||
|
this.load_object_code(quest.object_code);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Halts and resets the VM, then loads new object code.
|
||||||
|
* Opcodes which use quest data outside the object code
|
||||||
|
* will not work if calling this method directly.
|
||||||
|
* Use {@link VirtualMachine.load_quest} if full functionality is needed.
|
||||||
|
*/
|
||||||
|
load_object_code(object_code: readonly Segment[]): void {
|
||||||
this.halt();
|
this.halt();
|
||||||
|
this.quest = undefined;
|
||||||
this.registers.zero();
|
this.registers.zero();
|
||||||
this.string_arg_store.zero();
|
this.string_arg_store.zero();
|
||||||
this.object_code = object_code;
|
this.object_code = object_code;
|
||||||
this.label_to_seg_idx.clear();
|
this.label_to_seg_idx.clear();
|
||||||
|
this.set_episode_called = false;
|
||||||
|
|
||||||
let i = 0;
|
let i = 0;
|
||||||
|
|
||||||
for (const segment of this.object_code) {
|
for (const segment of this.object_code) {
|
||||||
@ -199,7 +231,7 @@ export class VirtualMachine {
|
|||||||
srcloc = inst.asm.mnemonic;
|
srcloc = inst.asm.mnemonic;
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.execute_instruction(exec, inst);
|
return this.execute_instruction(exec, inst, srcloc);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (!(err instanceof Error)) {
|
if (!(err instanceof Error)) {
|
||||||
err = new Error(String(err));
|
err = new Error(String(err));
|
||||||
@ -212,7 +244,11 @@ export class VirtualMachine {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private execute_instruction(exec: Thread, inst: Instruction): ExecutionResult {
|
private execute_instruction(
|
||||||
|
exec: Thread,
|
||||||
|
inst: Instruction,
|
||||||
|
srcloc?: AsmToken,
|
||||||
|
): 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;
|
||||||
|
|
||||||
@ -570,6 +606,37 @@ export class VirtualMachine {
|
|||||||
|
|
||||||
this.set_register_signed(arg1, result);
|
this.set_register_signed(arg1, result);
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
|
case OP_SET_EPISODE.code:
|
||||||
|
if (this.set_episode_called) {
|
||||||
|
this.io.warning("Calling set_episode more than once is not supported.", srcloc);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.set_episode_called = true;
|
||||||
|
|
||||||
|
if (this.get_current_segment_idx() !== ENTRY_SEGMENT) {
|
||||||
|
this.io.warning(
|
||||||
|
`Calling set_episode outside of segment ${ENTRY_SEGMENT} is not supported.`,
|
||||||
|
srcloc,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.quest) {
|
||||||
|
this.missing_quest_data_warning(OP_SET_EPISODE.mnemonic, srcloc);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (encode_episode_number(this.quest.episode) !== arg0) {
|
||||||
|
this.io.warning(
|
||||||
|
"Calling set_episode with an argument that does not" +
|
||||||
|
"match the quest's designated episode is not supported.",
|
||||||
|
srcloc,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`);
|
throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`);
|
||||||
@ -887,6 +954,17 @@ export class VirtualMachine {
|
|||||||
|
|
||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private get_current_segment_idx(): number {
|
||||||
|
return this.thread[this.thread_idx].call_stack_top().seg_idx;
|
||||||
|
}
|
||||||
|
|
||||||
|
private missing_quest_data_warning(info: string, srcloc?: AsmToken): void {
|
||||||
|
this.io.warning(
|
||||||
|
`Opcode execution failed because the VM was missing quest data: ${info}`,
|
||||||
|
srcloc,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class ExecutionLocation {
|
class ExecutionLocation {
|
||||||
|
Loading…
Reference in New Issue
Block a user