Fixed assembler bug related to arg_pushr.

This commit is contained in:
Daan Vanden Bosch 2019-08-05 22:33:07 +02:00
parent 2fc55cdc1a
commit 9284cf4a8a
4 changed files with 150 additions and 68 deletions

View File

@ -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);

View File

@ -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 {

View File

@ -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 }]);
});

View File

@ -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,67 +441,76 @@ 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;
} }
switch (param.type) { const [arg, token] = arg_and_token;
case TYPE_BYTE:
case TYPE_REG_REF:
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
break;
case TYPE_WORD:
case TYPE_LABEL:
case TYPE_I_LABEL:
case TYPE_D_LABEL:
case TYPE_S_LABEL:
this.add_instruction(Opcode.ARG_PUSHW, [arg]);
break;
case TYPE_DWORD:
case TYPE_FLOAT:
this.add_instruction(Opcode.ARG_PUSHL, [arg]);
break;
case TYPE_STRING:
this.add_instruction(Opcode.ARG_PUSHS, [arg]);
break;
default:
if (param.type instanceof RegTupRefType) {
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
} else {
logger.error(
`Line ${this.line_no}: Type ${param.type} not implemented.`
);
}
break; 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) {
case TYPE_BYTE:
case TYPE_REG_REF:
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
break;
case TYPE_WORD:
case TYPE_LABEL:
case TYPE_I_LABEL:
case TYPE_D_LABEL:
case TYPE_S_LABEL:
this.add_instruction(Opcode.ARG_PUSHW, [arg]);
break;
case TYPE_DWORD:
case TYPE_FLOAT:
this.add_instruction(Opcode.ARG_PUSHL, [arg]);
break;
case TYPE_STRING:
this.add_instruction(Opcode.ARG_PUSHS, [arg]);
break;
default:
if (param.type instanceof RegTupRefType) {
this.add_instruction(Opcode.ARG_PUSHB, [arg]);
} else {
logger.error(
`Line ${this.line_no}: Type ${param.type} not implemented.`
);
}
break;
}
} }
} }
} }
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, {
size: 4, value: token.value,
}); 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, {
size: 4, value: token.value,
}); 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, {
size: 2 * token.value.length + 2, value: token.value,
}); 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, {
size, value,
}); 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, {
size: 1, value,
}); size: 1,
},
token,
]);
} }
} }