mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Fixed assembler bug related to arg_pushr.
This commit is contained in:
parent
2fc55cdc1a
commit
9284cf4a8a
@ -1,5 +1,7 @@
|
|||||||
import Logger from "js-logger";
|
import Logger from "js-logger";
|
||||||
import { Endianness } from "../..";
|
import { Endianness } from "../..";
|
||||||
|
import { ControlFlowGraph } from "../../../scripting/data_flow_analysis/ControlFlowGraph";
|
||||||
|
import { register_values } from "../../../scripting/data_flow_analysis/register_values";
|
||||||
import {
|
import {
|
||||||
Arg,
|
Arg,
|
||||||
DataSegment,
|
DataSegment,
|
||||||
@ -33,9 +35,6 @@ import { Cursor } from "../../cursor/Cursor";
|
|||||||
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { WritableCursor } from "../../cursor/WritableCursor";
|
import { WritableCursor } from "../../cursor/WritableCursor";
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
import { ControlFlowGraph } from "../../../scripting/data_flow_analysis/ControlFlowGraph";
|
|
||||||
import { register_values } from "../../../scripting/data_flow_analysis/register_values";
|
|
||||||
import { disassemble } from "../../../scripting/disassembly";
|
|
||||||
|
|
||||||
const logger = Logger.get("data_formats/parsing/quest/bin");
|
const logger = Logger.get("data_formats/parsing/quest/bin");
|
||||||
|
|
||||||
@ -533,6 +532,7 @@ function parse_instructions_segment(
|
|||||||
: instruction.args;
|
: instruction.args;
|
||||||
|
|
||||||
if (opcode.stack === StackInteraction.Push) {
|
if (opcode.stack === StackInteraction.Push) {
|
||||||
|
// TODO: correctly deal with arg_pushr.
|
||||||
stack.push(...args);
|
stack.push(...args);
|
||||||
} else {
|
} else {
|
||||||
const len = Math.min(params.length, args.length);
|
const len = Math.min(params.length, args.length);
|
||||||
|
@ -179,6 +179,10 @@ export class AssemblyLexer {
|
|||||||
return this.line.charAt(this.index);
|
return this.line.charAt(this.index);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private peek_prev(): string {
|
||||||
|
return this.line.charAt(this.index - 1);
|
||||||
|
}
|
||||||
|
|
||||||
private skip(): void {
|
private skip(): void {
|
||||||
this.index++;
|
this.index++;
|
||||||
}
|
}
|
||||||
@ -219,7 +223,7 @@ export class AssemblyLexer {
|
|||||||
is_float = true;
|
is_float = true;
|
||||||
this.skip();
|
this.skip();
|
||||||
}
|
}
|
||||||
} else if ("x" === char) {
|
} else if ("x" === char && this.marked_len() === 1 && this.peek_prev() === "0") {
|
||||||
if (is_float || is_hex) {
|
if (is_float || is_hex) {
|
||||||
break;
|
break;
|
||||||
} else {
|
} else {
|
||||||
|
@ -2,7 +2,7 @@ import { assemble } from "./assembly";
|
|||||||
import { InstructionSegment, SegmentType } from "./instructions";
|
import { InstructionSegment, SegmentType } from "./instructions";
|
||||||
import { Opcode } from "./opcodes";
|
import { Opcode } from "./opcodes";
|
||||||
|
|
||||||
test("", () => {
|
test("basic script", () => {
|
||||||
const { object_code, warnings, errors } = assemble(
|
const { object_code, warnings, errors } = assemble(
|
||||||
`
|
`
|
||||||
0: set_episode 0
|
0: set_episode 0
|
||||||
@ -75,3 +75,53 @@ test("", () => {
|
|||||||
expect(segment_2.instructions[0].opcode).toBe(Opcode.RET);
|
expect(segment_2.instructions[0].opcode).toBe(Opcode.RET);
|
||||||
expect(segment_2.instructions[0].args).toEqual([]);
|
expect(segment_2.instructions[0].args).toEqual([]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("pass the value of a register via the stack", () => {
|
||||||
|
const { object_code, warnings, errors } = assemble(
|
||||||
|
`
|
||||||
|
0:
|
||||||
|
leti r255, 7
|
||||||
|
exit r255
|
||||||
|
ret
|
||||||
|
`.split("\n")
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(warnings).toEqual([]);
|
||||||
|
expect(errors).toEqual([]);
|
||||||
|
|
||||||
|
expect(object_code.length).toBe(1);
|
||||||
|
|
||||||
|
const segment_0 = object_code[0] as InstructionSegment;
|
||||||
|
|
||||||
|
expect(segment_0.type).toBe(SegmentType.Instructions);
|
||||||
|
expect(segment_0.instructions.length).toBe(4);
|
||||||
|
|
||||||
|
expect(segment_0.instructions[1].opcode).toBe(Opcode.ARG_PUSHR);
|
||||||
|
expect(segment_0.instructions[1].args).toEqual([{ value: 255, size: 1 }]);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("pass a register reference via the stack", () => {
|
||||||
|
const { object_code, warnings, errors } = assemble(
|
||||||
|
`
|
||||||
|
0:
|
||||||
|
p_dead_v3 r200, 3
|
||||||
|
ret
|
||||||
|
`.split("\n")
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(warnings).toEqual([]);
|
||||||
|
expect(errors).toEqual([]);
|
||||||
|
|
||||||
|
expect(object_code.length).toBe(1);
|
||||||
|
|
||||||
|
const segment_0 = object_code[0] as InstructionSegment;
|
||||||
|
|
||||||
|
expect(segment_0.type).toBe(SegmentType.Instructions);
|
||||||
|
expect(segment_0.instructions.length).toBe(4);
|
||||||
|
|
||||||
|
expect(segment_0.instructions[0].opcode).toBe(Opcode.ARG_PUSHB);
|
||||||
|
expect(segment_0.instructions[0].args).toEqual([{ value: 200, size: 1 }]);
|
||||||
|
|
||||||
|
expect(segment_0.instructions[1].opcode).toBe(Opcode.ARG_PUSHL);
|
||||||
|
expect(segment_0.instructions[1].args).toEqual([{ value: 3, size: 4 }]);
|
||||||
|
});
|
||||||
|
@ -417,7 +417,7 @@ class Assembler {
|
|||||||
|
|
||||||
const last_token = this.tokens[this.tokens.length - 1];
|
const last_token = this.tokens[this.tokens.length - 1];
|
||||||
let error_length = last_token ? last_token.col + last_token.len - col : 0;
|
let error_length = last_token ? last_token.col + last_token.len - col : 0;
|
||||||
const ins_args: Arg[] = [];
|
const ins_args: [Arg, Token][] = [];
|
||||||
|
|
||||||
if (!varargs && arg_count !== param_count) {
|
if (!varargs && arg_count !== param_count) {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
@ -441,26 +441,34 @@ class Assembler {
|
|||||||
return;
|
return;
|
||||||
} else if (opcode.stack !== StackInteraction.Pop) {
|
} else if (opcode.stack !== StackInteraction.Pop) {
|
||||||
// Inline arguments.
|
// Inline arguments.
|
||||||
if (!this.parse_args(opcode.params, ins_args)) {
|
if (!this.parse_args(opcode.params, ins_args, false)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Stack arguments.
|
// Stack arguments.
|
||||||
// TODO: take into account that stack arguments can come from registers (arg_pushr).
|
const stack_args: [Arg, Token][] = [];
|
||||||
const stack_args: Arg[] = [];
|
|
||||||
|
|
||||||
if (!this.parse_args(opcode.params, stack_args)) {
|
if (!this.parse_args(opcode.params, stack_args, true)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let i = 0; i < opcode.params.length; i++) {
|
for (let i = 0; i < opcode.params.length; i++) {
|
||||||
const param = opcode.params[i];
|
const param = opcode.params[i];
|
||||||
const arg = stack_args[i];
|
const arg_and_token = stack_args[i];
|
||||||
|
|
||||||
if (arg == undefined) {
|
if (arg_and_token == undefined) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const [arg, token] = arg_and_token;
|
||||||
|
|
||||||
|
if (token.type === TokenType.Register) {
|
||||||
|
if (param.type instanceof RegTupRefType) {
|
||||||
|
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
|
||||||
|
} else {
|
||||||
|
this.add_instruction(Opcode.ARG_PUSHR, [arg]);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case TYPE_BYTE:
|
case TYPE_BYTE:
|
||||||
case TYPE_REG_REF:
|
case TYPE_REG_REF:
|
||||||
@ -493,15 +501,16 @@ class Assembler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
this.add_instruction(opcode, ins_args);
|
this.add_instruction(opcode, ins_args.map(([arg]) => arg));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @returns true if arguments can be translated to object code, possibly after truncation. False otherwise.
|
* @returns true if arguments can be translated to object code, possibly after truncation. False otherwise.
|
||||||
*/
|
*/
|
||||||
private parse_args(params: Param[], args: Arg[]): boolean {
|
private parse_args(params: Param[], arg_and_tokens: [Arg, Token][], stack: boolean): boolean {
|
||||||
let semi_valid = true;
|
let semi_valid = true;
|
||||||
let should_be_arg = true;
|
let should_be_arg = true;
|
||||||
let param_i = 0;
|
let param_i = 0;
|
||||||
@ -545,7 +554,7 @@ class Assembler {
|
|||||||
switch (param.type) {
|
switch (param.type) {
|
||||||
case TYPE_BYTE:
|
case TYPE_BYTE:
|
||||||
match = true;
|
match = true;
|
||||||
this.parse_int(1, token, args);
|
this.parse_int(1, token, arg_and_tokens);
|
||||||
break;
|
break;
|
||||||
case TYPE_WORD:
|
case TYPE_WORD:
|
||||||
case TYPE_LABEL:
|
case TYPE_LABEL:
|
||||||
@ -554,18 +563,21 @@ class Assembler {
|
|||||||
case TYPE_S_LABEL:
|
case TYPE_S_LABEL:
|
||||||
case TYPE_I_LABEL_VAR:
|
case TYPE_I_LABEL_VAR:
|
||||||
match = true;
|
match = true;
|
||||||
this.parse_int(2, token, args);
|
this.parse_int(2, token, arg_and_tokens);
|
||||||
break;
|
break;
|
||||||
case TYPE_DWORD:
|
case TYPE_DWORD:
|
||||||
match = true;
|
match = true;
|
||||||
this.parse_int(4, token, args);
|
this.parse_int(4, token, arg_and_tokens);
|
||||||
break;
|
break;
|
||||||
case TYPE_FLOAT:
|
case TYPE_FLOAT:
|
||||||
match = true;
|
match = true;
|
||||||
args.push({
|
arg_and_tokens.push([
|
||||||
|
{
|
||||||
value: token.value,
|
value: token.value,
|
||||||
size: 4,
|
size: 4,
|
||||||
});
|
},
|
||||||
|
token,
|
||||||
|
]);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
match = false;
|
match = false;
|
||||||
@ -576,29 +588,36 @@ class Assembler {
|
|||||||
match = param.type === TYPE_FLOAT;
|
match = param.type === TYPE_FLOAT;
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
args.push({
|
arg_and_tokens.push([
|
||||||
|
{
|
||||||
value: token.value,
|
value: token.value,
|
||||||
size: 4,
|
size: 4,
|
||||||
});
|
},
|
||||||
|
token,
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case TokenType.Register:
|
case TokenType.Register:
|
||||||
match =
|
match =
|
||||||
|
stack ||
|
||||||
param.type === TYPE_REG_REF ||
|
param.type === TYPE_REG_REF ||
|
||||||
param.type === TYPE_REG_REF_VAR ||
|
param.type === TYPE_REG_REF_VAR ||
|
||||||
param.type instanceof RegTupRefType;
|
param.type instanceof RegTupRefType;
|
||||||
|
|
||||||
this.parse_register(token, args);
|
this.parse_register(token, arg_and_tokens);
|
||||||
break;
|
break;
|
||||||
case TokenType.String:
|
case TokenType.String:
|
||||||
match = param.type === TYPE_STRING;
|
match = param.type === TYPE_STRING;
|
||||||
|
|
||||||
if (match) {
|
if (match) {
|
||||||
args.push({
|
arg_and_tokens.push([
|
||||||
|
{
|
||||||
value: token.value,
|
value: token.value,
|
||||||
size: 2 * token.value.length + 2,
|
size: 2 * token.value.length + 2,
|
||||||
});
|
},
|
||||||
|
token,
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -673,7 +692,8 @@ class Assembler {
|
|||||||
return semi_valid;
|
return semi_valid;
|
||||||
}
|
}
|
||||||
|
|
||||||
private parse_int(size: number, { col, len, value }: IntToken, args: Arg[]): void {
|
private parse_int(size: number, token: IntToken, arg_and_tokens: [Arg, Token][]): void {
|
||||||
|
const { value, col, len } = token;
|
||||||
const bit_size = 8 * size;
|
const bit_size = 8 * size;
|
||||||
const min_value = -Math.pow(2, bit_size - 1);
|
const min_value = -Math.pow(2, bit_size - 1);
|
||||||
const max_value = Math.pow(2, bit_size) - 1;
|
const max_value = Math.pow(2, bit_size) - 1;
|
||||||
@ -691,14 +711,19 @@ 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 {
|
||||||
args.push({
|
arg_and_tokens.push([
|
||||||
|
{
|
||||||
value,
|
value,
|
||||||
size,
|
size,
|
||||||
});
|
},
|
||||||
|
token,
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private parse_register({ col, len, value }: RegisterToken, args: Arg[]): void {
|
private parse_register(token: RegisterToken, arg_and_tokens: [Arg, Token][]): void {
|
||||||
|
const { col, len, value } = token;
|
||||||
|
|
||||||
if (value > 255) {
|
if (value > 255) {
|
||||||
this.add_error({
|
this.add_error({
|
||||||
col,
|
col,
|
||||||
@ -706,10 +731,13 @@ class Assembler {
|
|||||||
message: `Invalid register reference, expected r0-r255.`,
|
message: `Invalid register reference, expected r0-r255.`,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
args.push({
|
arg_and_tokens.push([
|
||||||
|
{
|
||||||
value,
|
value,
|
||||||
size: 1,
|
size: 1,
|
||||||
});
|
},
|
||||||
|
token,
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user