mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Fixed bug in object code writer. Strings in the object code are now written correctly when the format is DC/GC.
This commit is contained in:
parent
b276ba988e
commit
50d1ff7f93
@ -1,5 +1,6 @@
|
|||||||
import { Kind, Opcode } from "./opcodes";
|
import { Kind, Opcode } from "./opcodes";
|
||||||
import { array_buffers_equal, arrays_equal } from "../../util";
|
import { array_buffers_equal, arrays_equal } from "../../util";
|
||||||
|
import { BinFormat } from "../parsing/quest/BinFormat";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Instruction invocation.
|
* Instruction invocation.
|
||||||
@ -7,14 +8,6 @@ import { array_buffers_equal, arrays_equal } from "../../util";
|
|||||||
export type Instruction = {
|
export type Instruction = {
|
||||||
readonly opcode: Opcode;
|
readonly opcode: Opcode;
|
||||||
readonly args: readonly Arg[];
|
readonly args: readonly Arg[];
|
||||||
/**
|
|
||||||
* Byte size of the argument list.
|
|
||||||
*/
|
|
||||||
readonly arg_size: number;
|
|
||||||
/**
|
|
||||||
* Byte size of the entire instruction, i.e. the sum of the opcode size and all argument sizes.
|
|
||||||
*/
|
|
||||||
readonly size: number;
|
|
||||||
/**
|
/**
|
||||||
* Maps each parameter by index to its arguments.
|
* Maps each parameter by index to its arguments.
|
||||||
*/
|
*/
|
||||||
@ -25,7 +18,6 @@ export type Instruction = {
|
|||||||
export function new_instruction(opcode: Opcode, args: Arg[], asm?: InstructionAsm): Instruction {
|
export function new_instruction(opcode: Opcode, args: Arg[], asm?: InstructionAsm): Instruction {
|
||||||
const len = Math.min(opcode.params.length, args.length);
|
const len = Math.min(opcode.params.length, args.length);
|
||||||
const param_to_args: Arg[][] = [];
|
const param_to_args: Arg[][] = [];
|
||||||
let arg_size = 0;
|
|
||||||
|
|
||||||
for (let i = 0; i < len; i++) {
|
for (let i = 0; i < len; i++) {
|
||||||
const type = opcode.params[i].type;
|
const type = opcode.params[i].type;
|
||||||
@ -35,16 +27,12 @@ export function new_instruction(opcode: Opcode, args: Arg[], asm?: InstructionAs
|
|||||||
switch (type.kind) {
|
switch (type.kind) {
|
||||||
case Kind.ILabelVar:
|
case Kind.ILabelVar:
|
||||||
case Kind.RegRefVar:
|
case Kind.RegRefVar:
|
||||||
arg_size++;
|
|
||||||
|
|
||||||
for (let j = i; j < args.length; j++) {
|
for (let j = i; j < args.length; j++) {
|
||||||
param_to_args[i].push(args[j]);
|
param_to_args[i].push(args[j]);
|
||||||
arg_size += args[j].size;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
arg_size += arg.size;
|
|
||||||
param_to_args[i].push(arg);
|
param_to_args[i].push(arg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -53,13 +41,62 @@ export function new_instruction(opcode: Opcode, args: Arg[], asm?: InstructionAs
|
|||||||
return {
|
return {
|
||||||
opcode,
|
opcode,
|
||||||
args,
|
args,
|
||||||
arg_size,
|
|
||||||
size: opcode.size + arg_size,
|
|
||||||
param_to_args,
|
param_to_args,
|
||||||
asm,
|
asm,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns The byte size of the entire instruction, i.e. the sum of the opcode size and all
|
||||||
|
* argument sizes.
|
||||||
|
*/
|
||||||
|
export function instruction_size(instruction: Instruction, format: BinFormat): number {
|
||||||
|
const opcode = instruction.opcode;
|
||||||
|
const p_len = Math.min(opcode.params.length, instruction.param_to_args.length);
|
||||||
|
let arg_size = 0;
|
||||||
|
|
||||||
|
for (let i = 0; i < p_len; i++) {
|
||||||
|
const type = opcode.params[i].type;
|
||||||
|
const args = instruction.param_to_args[i];
|
||||||
|
|
||||||
|
switch (type.kind) {
|
||||||
|
case Kind.Byte:
|
||||||
|
case Kind.RegRef:
|
||||||
|
case Kind.RegTupRef:
|
||||||
|
arg_size++;
|
||||||
|
break;
|
||||||
|
case Kind.Word:
|
||||||
|
case Kind.Label:
|
||||||
|
case Kind.ILabel:
|
||||||
|
case Kind.DLabel:
|
||||||
|
case Kind.SLabel:
|
||||||
|
arg_size += 2;
|
||||||
|
break;
|
||||||
|
case Kind.DWord:
|
||||||
|
case Kind.Float:
|
||||||
|
arg_size += 4;
|
||||||
|
break;
|
||||||
|
case Kind.String:
|
||||||
|
if (format == BinFormat.DC_GC) {
|
||||||
|
arg_size += (args[0].value as string).length + 1;
|
||||||
|
} else {
|
||||||
|
arg_size += 2 * (args[0].value as string).length + 2;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case Kind.ILabelVar:
|
||||||
|
arg_size += 1 + 2 * args.length;
|
||||||
|
break;
|
||||||
|
case Kind.RegRefVar:
|
||||||
|
arg_size += 1 + args.length;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Error(`Parameter type ${Kind[type.kind]} not implemented.`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return opcode.size + arg_size;
|
||||||
|
}
|
||||||
|
|
||||||
function instructions_equal(a: Instruction, b: Instruction): boolean {
|
function instructions_equal(a: Instruction, b: Instruction): boolean {
|
||||||
return a.opcode.code === b.opcode.code && arrays_equal(a.args, b.args, args_equal);
|
return a.opcode.code === b.opcode.code && arrays_equal(a.args, b.args, args_equal);
|
||||||
}
|
}
|
||||||
@ -68,8 +105,6 @@ export function clone_instruction(instr: Instruction): Instruction {
|
|||||||
return {
|
return {
|
||||||
opcode: instr.opcode,
|
opcode: instr.opcode,
|
||||||
args: instr.args.map(arg => ({ ...arg })),
|
args: instr.args.map(arg => ({ ...arg })),
|
||||||
arg_size: instr.arg_size,
|
|
||||||
size: instr.size,
|
|
||||||
param_to_args: instr.param_to_args.map(args => args.map(arg => ({ ...arg }))),
|
param_to_args: instr.param_to_args.map(args => args.map(arg => ({ ...arg }))),
|
||||||
asm: instr.asm,
|
asm: instr.asm,
|
||||||
};
|
};
|
||||||
@ -80,18 +115,14 @@ export function clone_instruction(instr: Instruction): Instruction {
|
|||||||
*/
|
*/
|
||||||
export type Arg = {
|
export type Arg = {
|
||||||
readonly value: any;
|
readonly value: any;
|
||||||
readonly size: number;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export function new_arg(value: any, size: number): Arg {
|
export function new_arg(value: any): Arg {
|
||||||
return {
|
return { value };
|
||||||
value,
|
|
||||||
size,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function args_equal(a: Arg, b: Arg): boolean {
|
function args_equal(a: Arg, b: Arg): boolean {
|
||||||
return a.value === b.value && a.size === b.size;
|
return a.value === b.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -5,6 +5,12 @@ import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
|||||||
import { BufferCursor } from "../../cursor/BufferCursor";
|
import { BufferCursor } from "../../cursor/BufferCursor";
|
||||||
import { parse_qst_to_quest, write_quest_qst } from "./index";
|
import { parse_qst_to_quest, write_quest_qst } from "./index";
|
||||||
import { ObjectType } from "./object_types";
|
import { ObjectType } from "./object_types";
|
||||||
|
import {
|
||||||
|
DataSegment,
|
||||||
|
InstructionSegment,
|
||||||
|
SegmentType,
|
||||||
|
StringSegment,
|
||||||
|
} from "../../asm/instructions";
|
||||||
|
|
||||||
test("parse Towards the Future", () => {
|
test("parse Towards the Future", () => {
|
||||||
const buffer = readFileSync("test/resources/quest118_e.qst");
|
const buffer = readFileSync("test/resources/quest118_e.qst");
|
||||||
@ -104,6 +110,35 @@ function round_trip_test(path: string, file_name: string, contents: Buffer): voi
|
|||||||
expect(test_quest.object_code.length).toBe(orig_quest.object_code.length);
|
expect(test_quest.object_code.length).toBe(orig_quest.object_code.length);
|
||||||
|
|
||||||
for (let i = 0; i < orig_quest.object_code.length; i++) {
|
for (let i = 0; i < orig_quest.object_code.length; i++) {
|
||||||
|
const orig_segment = orig_quest.object_code[i];
|
||||||
|
const test_segment = test_quest.object_code[i];
|
||||||
|
|
||||||
|
expect(test_segment.type).toBe(orig_segment.type);
|
||||||
|
expect(test_segment.labels).toEqual(orig_segment.labels);
|
||||||
|
|
||||||
|
switch (orig_segment.type) {
|
||||||
|
case SegmentType.Instructions:
|
||||||
|
expect((test_segment as InstructionSegment).instructions.length).toBe(
|
||||||
|
orig_segment.instructions.length,
|
||||||
|
);
|
||||||
|
|
||||||
|
for (let j = 0; j < orig_segment.instructions.length; j++) {
|
||||||
|
const orig_inst = orig_segment.instructions[j];
|
||||||
|
const test_inst = (test_segment as InstructionSegment).instructions[j];
|
||||||
|
|
||||||
|
expect(test_inst.opcode.code).toBe(orig_inst.opcode.code);
|
||||||
|
expect(test_inst.args).toEqual(orig_inst.args);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
case SegmentType.Data:
|
||||||
|
expect((test_segment as DataSegment).data).toEqual(orig_segment.data);
|
||||||
|
break;
|
||||||
|
case SegmentType.String:
|
||||||
|
expect((test_segment as StringSegment).value).toBe(orig_segment.value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
expect(test_quest.object_code[i]).toEqual(orig_quest.object_code[i]);
|
expect(test_quest.object_code[i]).toEqual(orig_quest.object_code[i]);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -2,6 +2,7 @@ import {
|
|||||||
Arg,
|
Arg,
|
||||||
DataSegment,
|
DataSegment,
|
||||||
Instruction,
|
Instruction,
|
||||||
|
instruction_size,
|
||||||
InstructionSegment,
|
InstructionSegment,
|
||||||
new_arg,
|
new_arg,
|
||||||
new_instruction,
|
new_instruction,
|
||||||
@ -110,9 +111,15 @@ export function write_object_code(
|
|||||||
break;
|
break;
|
||||||
case Kind.String:
|
case Kind.String:
|
||||||
if (format === BinFormat.DC_GC) {
|
if (format === BinFormat.DC_GC) {
|
||||||
cursor.write_string_ascii(arg.value, arg.size);
|
cursor.write_string_ascii(
|
||||||
|
arg.value,
|
||||||
|
(arg.value as string).length + 1,
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
cursor.write_string_utf16(arg.value, arg.size);
|
cursor.write_string_utf16(
|
||||||
|
arg.value,
|
||||||
|
2 * (arg.value as string).length + 2,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case Kind.ILabelVar:
|
case Kind.ILabelVar:
|
||||||
@ -229,7 +236,7 @@ function internal_parse_object_code(
|
|||||||
switch (segment.type) {
|
switch (segment.type) {
|
||||||
case SegmentType.Instructions:
|
case SegmentType.Instructions:
|
||||||
for (const instruction of segment.instructions) {
|
for (const instruction of segment.instructions) {
|
||||||
offset += instruction.size;
|
offset += instruction_size(instruction, format);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -595,33 +602,31 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode, format: Bin
|
|||||||
for (const param of opcode.params) {
|
for (const param of opcode.params) {
|
||||||
switch (param.type.kind) {
|
switch (param.type.kind) {
|
||||||
case Kind.Byte:
|
case Kind.Byte:
|
||||||
args.push(new_arg(cursor.u8(), 1));
|
args.push(new_arg(cursor.u8()));
|
||||||
break;
|
break;
|
||||||
case Kind.Word:
|
case Kind.Word:
|
||||||
args.push(new_arg(cursor.u16(), 2));
|
args.push(new_arg(cursor.u16()));
|
||||||
break;
|
break;
|
||||||
case Kind.DWord:
|
case Kind.DWord:
|
||||||
args.push(new_arg(cursor.i32(), 4));
|
args.push(new_arg(cursor.i32()));
|
||||||
break;
|
break;
|
||||||
case Kind.Float:
|
case Kind.Float:
|
||||||
args.push(new_arg(cursor.f32(), 4));
|
args.push(new_arg(cursor.f32()));
|
||||||
break;
|
break;
|
||||||
case Kind.Label:
|
case Kind.Label:
|
||||||
case Kind.ILabel:
|
case Kind.ILabel:
|
||||||
case Kind.DLabel:
|
case Kind.DLabel:
|
||||||
case Kind.SLabel:
|
case Kind.SLabel:
|
||||||
args.push(new_arg(cursor.u16(), 2));
|
args.push(new_arg(cursor.u16()));
|
||||||
break;
|
break;
|
||||||
case Kind.String:
|
case Kind.String:
|
||||||
{
|
{
|
||||||
const start_pos = cursor.position;
|
|
||||||
const max_bytes = Math.min(4096, cursor.bytes_left);
|
const max_bytes = Math.min(4096, cursor.bytes_left);
|
||||||
args.push(
|
args.push(
|
||||||
new_arg(
|
new_arg(
|
||||||
format === BinFormat.DC_GC
|
format === BinFormat.DC_GC
|
||||||
? cursor.string_ascii(max_bytes, true, false)
|
? cursor.string_ascii(max_bytes, true, false)
|
||||||
: cursor.string_utf16(max_bytes, true, false),
|
: cursor.string_utf16(max_bytes, true, false),
|
||||||
cursor.position - start_pos,
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -629,17 +634,17 @@ function parse_instruction_arguments(cursor: Cursor, opcode: Opcode, format: Bin
|
|||||||
case Kind.ILabelVar:
|
case Kind.ILabelVar:
|
||||||
{
|
{
|
||||||
const arg_size = cursor.u8();
|
const arg_size = cursor.u8();
|
||||||
args.push(...cursor.u16_array(arg_size).map(value => new_arg(value, 2)));
|
args.push(...cursor.u16_array(arg_size).map(value => new_arg(value)));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case Kind.RegRef:
|
case Kind.RegRef:
|
||||||
case Kind.RegTupRef:
|
case Kind.RegTupRef:
|
||||||
args.push(new_arg(cursor.u8(), 1));
|
args.push(new_arg(cursor.u8()));
|
||||||
break;
|
break;
|
||||||
case Kind.RegRefVar:
|
case Kind.RegRefVar:
|
||||||
{
|
{
|
||||||
const arg_size = cursor.u8();
|
const arg_size = cursor.u8();
|
||||||
args.push(...cursor.u8_array(arg_size).map(value => new_arg(value, 1)));
|
args.push(...cursor.u8_array(arg_size).map(value => new_arg(value)));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -37,7 +37,7 @@ test("basic script", () => {
|
|||||||
expect(segment_0.instructions.length).toBe(9);
|
expect(segment_0.instructions.length).toBe(9);
|
||||||
|
|
||||||
expect(segment_0.instructions[0].opcode).toBe(OP_SET_EPISODE);
|
expect(segment_0.instructions[0].opcode).toBe(OP_SET_EPISODE);
|
||||||
expect(segment_0.instructions[0].args).toEqual([{ value: 0, size: 4 }]);
|
expect(segment_0.instructions[0].args).toEqual([{ value: 0 }]);
|
||||||
expect(segment_0.instructions[0].asm).toEqual({
|
expect(segment_0.instructions[0].asm).toEqual({
|
||||||
mnemonic: { line_no: 2, col: 10, len: 11 },
|
mnemonic: { line_no: 2, col: 10, len: 11 },
|
||||||
args: [{ line_no: 2, col: 22, len: 1 }],
|
args: [{ line_no: 2, col: 22, len: 1 }],
|
||||||
@ -46,10 +46,10 @@ test("basic script", () => {
|
|||||||
|
|
||||||
expect(segment_0.instructions[1].opcode).toBe(OP_BB_MAP_DESIGNATE);
|
expect(segment_0.instructions[1].opcode).toBe(OP_BB_MAP_DESIGNATE);
|
||||||
expect(segment_0.instructions[1].args).toEqual([
|
expect(segment_0.instructions[1].args).toEqual([
|
||||||
{ value: 1, size: 1 },
|
{ value: 1 },
|
||||||
{ value: 2, size: 2 },
|
{ value: 2 },
|
||||||
{ value: 3, size: 1 },
|
{ value: 3 },
|
||||||
{ value: 4, size: 1 },
|
{ value: 4 },
|
||||||
]);
|
]);
|
||||||
expect(segment_0.instructions[1].asm).toEqual({
|
expect(segment_0.instructions[1].asm).toEqual({
|
||||||
mnemonic: { line_no: 3, col: 10, len: 16 },
|
mnemonic: { line_no: 3, col: 10, len: 16 },
|
||||||
@ -63,13 +63,13 @@ test("basic script", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
expect(segment_0.instructions[2].opcode).toBe(OP_ARG_PUSHL);
|
expect(segment_0.instructions[2].opcode).toBe(OP_ARG_PUSHL);
|
||||||
expect(segment_0.instructions[2].args).toEqual([{ value: 0, size: 4 }]);
|
expect(segment_0.instructions[2].args).toEqual([{ value: 0 }]);
|
||||||
expect(segment_0.instructions[2].asm).toEqual({
|
expect(segment_0.instructions[2].asm).toEqual({
|
||||||
args: [{ line_no: 4, col: 28, len: 1 }],
|
args: [{ line_no: 4, col: 28, len: 1 }],
|
||||||
stack_args: [],
|
stack_args: [],
|
||||||
});
|
});
|
||||||
expect(segment_0.instructions[3].opcode).toBe(OP_ARG_PUSHW);
|
expect(segment_0.instructions[3].opcode).toBe(OP_ARG_PUSHW);
|
||||||
expect(segment_0.instructions[3].args).toEqual([{ value: 150, size: 2 }]);
|
expect(segment_0.instructions[3].args).toEqual([{ value: 150 }]);
|
||||||
expect(segment_0.instructions[3].asm).toEqual({
|
expect(segment_0.instructions[3].asm).toEqual({
|
||||||
args: [{ line_no: 4, col: 31, len: 3 }],
|
args: [{ line_no: 4, col: 31, len: 3 }],
|
||||||
stack_args: [],
|
stack_args: [],
|
||||||
@ -86,13 +86,13 @@ test("basic script", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
expect(segment_0.instructions[5].opcode).toBe(OP_ARG_PUSHL);
|
expect(segment_0.instructions[5].opcode).toBe(OP_ARG_PUSHL);
|
||||||
expect(segment_0.instructions[5].args).toEqual([{ value: 1, size: 4 }]);
|
expect(segment_0.instructions[5].args).toEqual([{ value: 1 }]);
|
||||||
expect(segment_0.instructions[5].asm).toEqual({
|
expect(segment_0.instructions[5].asm).toEqual({
|
||||||
args: [{ line_no: 5, col: 28, len: 1 }],
|
args: [{ line_no: 5, col: 28, len: 1 }],
|
||||||
stack_args: [],
|
stack_args: [],
|
||||||
});
|
});
|
||||||
expect(segment_0.instructions[6].opcode).toBe(OP_ARG_PUSHW);
|
expect(segment_0.instructions[6].opcode).toBe(OP_ARG_PUSHW);
|
||||||
expect(segment_0.instructions[6].args).toEqual([{ value: 151, size: 2 }]);
|
expect(segment_0.instructions[6].args).toEqual([{ value: 151 }]);
|
||||||
expect(segment_0.instructions[6].asm).toEqual({
|
expect(segment_0.instructions[6].asm).toEqual({
|
||||||
args: [{ line_no: 5, col: 31, len: 3 }],
|
args: [{ line_no: 5, col: 31, len: 3 }],
|
||||||
stack_args: [],
|
stack_args: [],
|
||||||
@ -117,7 +117,7 @@ test("basic script", () => {
|
|||||||
expect(segment_1.instructions.length).toBe(3);
|
expect(segment_1.instructions.length).toBe(3);
|
||||||
|
|
||||||
expect(segment_1.instructions[0].opcode).toBe(OP_ARG_PUSHL);
|
expect(segment_1.instructions[0].opcode).toBe(OP_ARG_PUSHL);
|
||||||
expect(segment_1.instructions[0].args).toEqual([{ value: 1, size: 4 }]);
|
expect(segment_1.instructions[0].args).toEqual([{ value: 1 }]);
|
||||||
expect(segment_1.instructions[0].asm).toEqual({
|
expect(segment_1.instructions[0].asm).toEqual({
|
||||||
args: [{ line_no: 7, col: 23, len: 1 }],
|
args: [{ line_no: 7, col: 23, len: 1 }],
|
||||||
stack_args: [],
|
stack_args: [],
|
||||||
@ -163,7 +163,7 @@ test("pass the value of a register via the stack", () => {
|
|||||||
expect(segment_0.instructions.length).toBe(4);
|
expect(segment_0.instructions.length).toBe(4);
|
||||||
|
|
||||||
expect(segment_0.instructions[1].opcode).toBe(OP_ARG_PUSHR);
|
expect(segment_0.instructions[1].opcode).toBe(OP_ARG_PUSHR);
|
||||||
expect(segment_0.instructions[1].args).toEqual([{ value: 255, size: 1 }]);
|
expect(segment_0.instructions[1].args).toEqual([{ value: 255 }]);
|
||||||
});
|
});
|
||||||
|
|
||||||
test("pass a register reference via the stack", () => {
|
test("pass a register reference via the stack", () => {
|
||||||
@ -186,8 +186,8 @@ test("pass a register reference via the stack", () => {
|
|||||||
expect(segment_0.instructions.length).toBe(4);
|
expect(segment_0.instructions.length).toBe(4);
|
||||||
|
|
||||||
expect(segment_0.instructions[0].opcode).toBe(OP_ARG_PUSHB);
|
expect(segment_0.instructions[0].opcode).toBe(OP_ARG_PUSHB);
|
||||||
expect(segment_0.instructions[0].args).toEqual([{ value: 200, size: 1 }]);
|
expect(segment_0.instructions[0].args).toEqual([{ value: 200 }]);
|
||||||
|
|
||||||
expect(segment_0.instructions[1].opcode).toBe(OP_ARG_PUSHL);
|
expect(segment_0.instructions[1].opcode).toBe(OP_ARG_PUSHL);
|
||||||
expect(segment_0.instructions[1].args).toEqual([{ value: 3, size: 4 }]);
|
expect(segment_0.instructions[1].args).toEqual([{ value: 3 }]);
|
||||||
});
|
});
|
||||||
|
@ -603,7 +603,7 @@ class Assembler {
|
|||||||
case Kind.Float:
|
case Kind.Float:
|
||||||
this.add_instruction(
|
this.add_instruction(
|
||||||
OP_ARG_PUSHL,
|
OP_ARG_PUSHL,
|
||||||
[{ value: reinterpret_f32_as_i32(arg.value), size: 4 }],
|
[{ value: reinterpret_f32_as_i32(arg.value) }],
|
||||||
[],
|
[],
|
||||||
undefined,
|
undefined,
|
||||||
[arg_token],
|
[arg_token],
|
||||||
@ -728,13 +728,7 @@ class Assembler {
|
|||||||
break;
|
break;
|
||||||
case Kind.Float:
|
case Kind.Float:
|
||||||
match = true;
|
match = true;
|
||||||
arg_and_tokens.push([
|
arg_and_tokens.push([{ value: token.value }, token]);
|
||||||
{
|
|
||||||
value: token.value,
|
|
||||||
size: 4,
|
|
||||||
},
|
|
||||||
token,
|
|
||||||
]);
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
match = false;
|
match = false;
|
||||||
@ -745,13 +739,7 @@ class Assembler {
|
|||||||
match = param.type.kind === Kind.Float;
|
match = param.type.kind === Kind.Float;
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
arg_and_tokens.push([
|
arg_and_tokens.push([{ value: token.value }, token]);
|
||||||
{
|
|
||||||
value: token.value,
|
|
||||||
size: 4,
|
|
||||||
},
|
|
||||||
token,
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -768,13 +756,7 @@ class Assembler {
|
|||||||
match = param.type.kind === Kind.String;
|
match = param.type.kind === Kind.String;
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
arg_and_tokens.push([
|
arg_and_tokens.push([{ value: token.value }, token]);
|
||||||
{
|
|
||||||
value: token.value,
|
|
||||||
size: 2 * token.value.length + 2,
|
|
||||||
},
|
|
||||||
token,
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -864,13 +846,7 @@ class Assembler {
|
|||||||
message: `${bit_size}-Bit integer can't be greater than ${max_value}.`,
|
message: `${bit_size}-Bit integer can't be greater than ${max_value}.`,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
arg_and_tokens.push([
|
arg_and_tokens.push([{ value }, token]);
|
||||||
{
|
|
||||||
value,
|
|
||||||
size,
|
|
||||||
},
|
|
||||||
token,
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -884,13 +860,7 @@ class Assembler {
|
|||||||
message: `Invalid register reference, expected r0-r255.`,
|
message: `Invalid register reference, expected r0-r255.`,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
arg_and_tokens.push([
|
arg_and_tokens.push([{ value }, token]);
|
||||||
{
|
|
||||||
value,
|
|
||||||
size: 1,
|
|
||||||
},
|
|
||||||
token,
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -34,10 +34,10 @@ test("vararg instructions should be disassembled correctly", () => {
|
|||||||
labels: [0],
|
labels: [0],
|
||||||
instructions: [
|
instructions: [
|
||||||
new_instruction(OP_SWITCH_JMP, [
|
new_instruction(OP_SWITCH_JMP, [
|
||||||
new_arg(90, 1),
|
new_arg(90),
|
||||||
new_arg(100, 2),
|
new_arg(100),
|
||||||
new_arg(101, 2),
|
new_arg(101),
|
||||||
new_arg(102, 2),
|
new_arg(102),
|
||||||
]),
|
]),
|
||||||
new_instruction(OP_RET, []),
|
new_instruction(OP_RET, []),
|
||||||
],
|
],
|
||||||
@ -64,8 +64,8 @@ test("va list instructions should be disassembled correctly", () => {
|
|||||||
labels: [0],
|
labels: [0],
|
||||||
instructions: [
|
instructions: [
|
||||||
new_instruction(OP_VA_START, []),
|
new_instruction(OP_VA_START, []),
|
||||||
new_instruction(OP_ARG_PUSHW, [new_arg(1337, 2)]),
|
new_instruction(OP_ARG_PUSHW, [new_arg(1337)]),
|
||||||
new_instruction(OP_VA_CALL, [new_arg(100, 2)]),
|
new_instruction(OP_VA_CALL, [new_arg(100)]),
|
||||||
new_instruction(OP_VA_END, []),
|
new_instruction(OP_VA_END, []),
|
||||||
new_instruction(OP_RET, []),
|
new_instruction(OP_RET, []),
|
||||||
],
|
],
|
||||||
|
@ -1 +1 @@
|
|||||||
47
|
48
|
||||||
|
Loading…
Reference in New Issue
Block a user