mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 23:38:30 +08:00
[VM] Implemented stack_push(m) and stack_pop(m) opcodes.
This commit is contained in:
parent
345c9ab0dc
commit
dbe4f3ab78
@ -74,6 +74,10 @@ import {
|
|||||||
OP_UJMPI_LE,
|
OP_UJMPI_LE,
|
||||||
OP_JMP_LE,
|
OP_JMP_LE,
|
||||||
OP_JMPI_LE,
|
OP_JMPI_LE,
|
||||||
|
OP_STACK_POP,
|
||||||
|
OP_STACK_PUSH,
|
||||||
|
OP_STACK_PUSHM,
|
||||||
|
OP_STACK_POPM,
|
||||||
} from "../opcodes";
|
} from "../opcodes";
|
||||||
import Logger from "js-logger";
|
import Logger from "js-logger";
|
||||||
|
|
||||||
@ -81,6 +85,7 @@ const logger = Logger.get("quest_editor/scripting/vm");
|
|||||||
|
|
||||||
const REGISTER_COUNT = 256;
|
const REGISTER_COUNT = 256;
|
||||||
const REGISTER_SIZE = 4;
|
const REGISTER_SIZE = 4;
|
||||||
|
const VARIABLE_STACK_LENGTH = 16; // TODO: verify this value
|
||||||
|
|
||||||
export enum ExecutionResult {
|
export enum ExecutionResult {
|
||||||
Ok,
|
Ok,
|
||||||
@ -476,6 +481,19 @@ export class VirtualMachine {
|
|||||||
...conditional_jump_args(comparison_ops.lte),
|
...conditional_jump_args(comparison_ops.lte),
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
|
// variable stack operations
|
||||||
|
case OP_STACK_PUSH.code:
|
||||||
|
this.push_variable_stack(exec, arg0, 1);
|
||||||
|
break;
|
||||||
|
case OP_STACK_POP.code:
|
||||||
|
this.pop_variable_stack(exec, arg0, 1);
|
||||||
|
break;
|
||||||
|
case OP_STACK_PUSHM.code:
|
||||||
|
this.push_variable_stack(exec, arg0, arg1);
|
||||||
|
break;
|
||||||
|
case OP_STACK_POPM.code:
|
||||||
|
this.pop_variable_stack(exec, arg0, arg1);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`);
|
throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`);
|
||||||
}
|
}
|
||||||
@ -672,6 +690,38 @@ export class VirtualMachine {
|
|||||||
return arg;
|
return arg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private push_variable_stack(exec: Thread, base_reg: number, num_push: number): void {
|
||||||
|
const end = base_reg + num_push;
|
||||||
|
|
||||||
|
if (end > REGISTER_COUNT) {
|
||||||
|
throw new Error("Variable stack: Invalid register");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exec.variable_stack.length + num_push > VARIABLE_STACK_LENGTH) {
|
||||||
|
throw new Error("Variable stack: Stack overflow");
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let r = base_reg; r < end; r++) {
|
||||||
|
exec.variable_stack.push(this.get_uint(r));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private pop_variable_stack(exec: Thread, base_reg: number, num_pop: number): void {
|
||||||
|
const end = base_reg + num_pop;
|
||||||
|
|
||||||
|
if (end > REGISTER_COUNT) {
|
||||||
|
throw new Error("Variable stack: Invalid register");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (exec.variable_stack.length < num_pop) {
|
||||||
|
throw new Error("Variable stack: Stack underflow");
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let r = end - 1; r >= base_reg; r--) {
|
||||||
|
this.set_uint(r, exec.variable_stack.pop()!);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private get_next_instruction_from_thread(exec: Thread): Instruction {
|
private get_next_instruction_from_thread(exec: Thread): Instruction {
|
||||||
if (exec.call_stack.length) {
|
if (exec.call_stack.length) {
|
||||||
const top = exec.call_stack_top();
|
const top = exec.call_stack_top();
|
||||||
@ -710,6 +760,7 @@ class Thread {
|
|||||||
*/
|
*/
|
||||||
public call_stack: ExecutionLocation[] = [];
|
public call_stack: ExecutionLocation[] = [];
|
||||||
public arg_stack: Arg[] = [];
|
public arg_stack: Arg[] = [];
|
||||||
|
public variable_stack: number[] = [];
|
||||||
/**
|
/**
|
||||||
* Global or floor-local?
|
* Global or floor-local?
|
||||||
*/
|
*/
|
||||||
|
Loading…
Reference in New Issue
Block a user