mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
[VM] Implemented get_random opcode and added a unit test for it.
This commit is contained in:
parent
95da6e9e57
commit
260c070f34
@ -1,7 +1,12 @@
|
|||||||
|
/**
|
||||||
|
* @jest-environment jsdom
|
||||||
|
*/
|
||||||
|
|
||||||
import { VirtualMachine, ExecutionResult } from ".";
|
import { VirtualMachine, ExecutionResult } from ".";
|
||||||
import { VMIOStub } from "./VMIOStub";
|
import { VMIOStub } from "./VMIOStub";
|
||||||
import { to_instructions } from "../../../../test/src/utils";
|
import { to_instructions } from "../../../../test/src/utils";
|
||||||
import { Segment } from "../instructions";
|
import { Segment } from "../instructions";
|
||||||
|
import { srand } from "./windows";
|
||||||
|
|
||||||
test("integer arithmetic opcodes", () => {
|
test("integer arithmetic opcodes", () => {
|
||||||
class TestIO extends VMIOStub {
|
class TestIO extends VMIOStub {
|
||||||
@ -158,3 +163,45 @@ test("basic window_msg output", () => {
|
|||||||
expect(io.winend).toBeCalledTimes(1);
|
expect(io.winend).toBeCalledTimes(1);
|
||||||
expect(io.error).toBeCalledTimes(0);
|
expect(io.error).toBeCalledTimes(0);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("opcode get_random", () => {
|
||||||
|
const result_reg = 102;
|
||||||
|
const obj_code = to_instructions(`
|
||||||
|
.code
|
||||||
|
0:
|
||||||
|
leti r100, 0
|
||||||
|
leti r101, 65535
|
||||||
|
get_random r100, r${result_reg}
|
||||||
|
get_random r100, r${result_reg}
|
||||||
|
get_random r100, r${result_reg}
|
||||||
|
get_random r100, r${result_reg}
|
||||||
|
get_random r100, r${result_reg}
|
||||||
|
get_random r100, r${result_reg}
|
||||||
|
get_random r100, r${result_reg}
|
||||||
|
`);
|
||||||
|
|
||||||
|
const vm = new VirtualMachine();
|
||||||
|
srand(123);
|
||||||
|
vm.load_object_code(obj_code);
|
||||||
|
vm.start_thread(0);
|
||||||
|
|
||||||
|
// run `let`s
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
|
||||||
|
// test correct get_random sequence
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
expect(vm.get_register_unsigned(result_reg)).toBe(879);
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
expect(vm.get_register_unsigned(result_reg)).toBe(38105);
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
expect(vm.get_register_unsigned(result_reg)).toBe(46149);
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
expect(vm.get_register_unsigned(result_reg)).toBe(26207);
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
expect(vm.get_register_unsigned(result_reg)).toBe(64725);
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Ok);
|
||||||
|
expect(vm.get_register_unsigned(result_reg)).toBe(6529);
|
||||||
|
expect(vm.execute()).toBe(ExecutionResult.Halted);
|
||||||
|
expect(vm.get_register_unsigned(result_reg)).toBe(61497);
|
||||||
|
});
|
||||||
|
@ -79,6 +79,7 @@ import {
|
|||||||
OP_LETA,
|
OP_LETA,
|
||||||
OP_FLET,
|
OP_FLET,
|
||||||
OP_FLETI,
|
OP_FLETI,
|
||||||
|
OP_GET_RANDOM,
|
||||||
} from "../opcodes";
|
} from "../opcodes";
|
||||||
import { VirtualMachineMemoryBuffer, VirtualMachineMemory } from "./memory";
|
import { VirtualMachineMemoryBuffer, VirtualMachineMemory } from "./memory";
|
||||||
import {
|
import {
|
||||||
@ -92,6 +93,7 @@ import {
|
|||||||
} from "./utils";
|
} from "./utils";
|
||||||
import { VirtualMachineIO } from "./io";
|
import { VirtualMachineIO } from "./io";
|
||||||
import { VMIOStub } from "./VMIOStub";
|
import { VMIOStub } from "./VMIOStub";
|
||||||
|
import { rand, srand, GetTickCount } from "./windows";
|
||||||
|
|
||||||
const REGISTERS_BASE_ADDRESS = 0x00a954b0;
|
const REGISTERS_BASE_ADDRESS = 0x00a954b0;
|
||||||
const REGISTER_COUNT = 256;
|
const REGISTER_COUNT = 256;
|
||||||
@ -125,7 +127,9 @@ 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()) {}
|
constructor(private io: VirtualMachineIO = new VMIOStub()) {
|
||||||
|
srand(GetTickCount());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Halts and resets the VM, then loads new object code.
|
* Halts and resets the VM, then loads new object code.
|
||||||
@ -547,6 +551,22 @@ export class VirtualMachine {
|
|||||||
this.io.winend();
|
this.io.winend();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case OP_GET_RANDOM.code:
|
||||||
|
{
|
||||||
|
const low = this.get_register_signed(arg0);
|
||||||
|
const hi = this.get_register_signed(arg0 + 1);
|
||||||
|
|
||||||
|
const r = rand();
|
||||||
|
let result = Math.floor(Math.fround(r / 32768.0) * hi);
|
||||||
|
|
||||||
|
// intentional. this is how the game does it.
|
||||||
|
if (low >= result) {
|
||||||
|
result = low;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.set_register_signed(arg1, result);
|
||||||
|
}
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`);
|
throw new Error(`Unsupported instruction: ${inst.opcode.mnemonic}.`);
|
||||||
}
|
}
|
||||||
|
23
src/quest_editor/scripting/vm/windows.ts
Normal file
23
src/quest_editor/scripting/vm/windows.ts
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
/**
|
||||||
|
* @file Implementations of some parts of the Win32 API and the MSVCRT C standard library.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
var holdrand = 1;
|
||||||
|
|
||||||
|
export function srand(seed: number): void {
|
||||||
|
holdrand = seed;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function rand(): number {
|
||||||
|
const r = (holdrand * 0x343fd + 0x269ec3) >>> 0;
|
||||||
|
holdrand = r;
|
||||||
|
return r >>> 0x10 & 0x7fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function GetTickCount(): number {
|
||||||
|
// GetTickCount returns time elapsed since system start
|
||||||
|
// performance.now returns time elapsed since document load
|
||||||
|
// but both are monotonic so it's probably close enough?
|
||||||
|
return Math.floor(performance.now());
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user