phantasmal-world/assets_generation/update_generic_data.ts
2019-08-04 17:52:01 +02:00

214 lines
6.5 KiB
TypeScript

import { readFileSync, writeFileSync } from "fs";
import Logger from "js-logger";
import { ASSETS_DIR, RESOURCE_DIR, SRC_DIR } from ".";
import { Endianness } from "../src/data_formats";
import { BufferCursor } from "../src/data_formats/cursor/BufferCursor";
import { parse_rlc } from "../src/data_formats/parsing/rlc";
import YAML from "yaml";
const logger = Logger.get("assets_generation/update_generic_data");
Logger.useDefaults({ defaultLevel: Logger.TRACE });
update();
function update(): void {
logger.info("Updating generic static data.");
update_opcodes();
extract_player_animations();
logger.info("Done updating generic static data.");
}
function extract_player_animations(): void {
logger.info("Extracting player animations.");
const buf = readFileSync(`${RESOURCE_DIR}/plymotiondata.rlc`);
let i = 0;
for (const file of parse_rlc(new BufferCursor(buf, Endianness.Big))) {
writeFileSync(
`${ASSETS_DIR}/player/animation/animation_${(i++).toString().padStart(3, "0")}.njm`,
new Uint8Array(file.array_buffer())
);
}
logger.info("Done extracting player animations.");
}
function update_opcodes(): void {
logger.info("Generating opcodes.");
// Add manual code.
const opcodes_src = readFileSync(`${SRC_DIR}/scripting/opcodes.ts`, { encoding: "UTF-8" });
const file_lines: string[] = [];
let in_manual_code = true;
let generated_lines_insert_point = 0;
opcodes_src.split("\n").forEach((line, i) => {
if (in_manual_code) {
if (line.includes("!!! GENERATED_CODE_START !!!")) {
in_manual_code = false;
generated_lines_insert_point = i + 1;
}
file_lines.push(line);
} else {
if (line.includes("!!! GENERATED_CODE_END !!!")) {
in_manual_code = true;
file_lines.push(line);
}
}
});
// Add generated code.
const yml = readFileSync(`${RESOURCE_DIR}/scripting/opcodes.yml`, { encoding: "UTF-8" });
const input = YAML.parse(yml);
const generated_lines: string[] = [];
let i = 0;
for (let code = 0; code <= 0xff; code++) {
const opcode = input.opcodes[i];
if (opcode && opcode.code === code) {
opcode_to_code(generated_lines, code, opcode);
i++;
} else {
opcode_to_code(generated_lines, code);
}
}
for (let code = 0xf800; code <= 0xf9ff; code++) {
const opcode = input.opcodes[i];
if (opcode && opcode.code === code) {
opcode_to_code(generated_lines, code, opcode);
i++;
} else {
opcode_to_code(generated_lines, code);
}
}
// Write final file.
file_lines.splice(generated_lines_insert_point, 0, ...generated_lines);
writeFileSync(`${SRC_DIR}/scripting/opcodes.ts`, file_lines.join("\n"));
logger.info("Done generating opcodes.");
}
function opcode_to_code(output: string[], code: number, opcode?: any): void {
const code_str = code.toString(16).padStart(code < 256 ? 2 : 4, "0");
const mnemonic: string = (opcode && opcode.mnemonic) || `unknown_${code_str}`;
const var_name = mnemonic
.replace("!=", "ne")
.replace("<=", "le")
.replace(">=", "ge")
.replace("<", "l")
.replace(">", "g")
.replace("=", "e")
.toUpperCase();
if (opcode) {
const stack_interaction =
opcode.stack === "push"
? "StackInteraction.Push"
: opcode.stack === "pop"
? "StackInteraction.Pop"
: "undefined";
const params = params_to_code(opcode.params);
output.push(` static readonly ${var_name} = (OPCODES[0x${code_str}] = new Opcode(
0x${code_str},
"${mnemonic}",
${(opcode.doc && JSON.stringify(opcode.doc)) || "undefined"},
[${params}],
${stack_interaction}
));`);
} else {
output.push(` static readonly ${var_name} = (OPCODES[0x${code_str}] = new Opcode(
0x${code_str},
"${mnemonic}",
undefined,
[],
undefined
));`);
}
}
function params_to_code(params: any[]) {
return params
.map((param: any) => {
let type: string;
switch (param.type) {
case "any":
type = "TYPE";
break;
case "byte":
type = "TYPE_BYTE";
break;
case "word":
type = "TYPE_WORD";
break;
case "dword":
type = "TYPE_DWORD";
break;
case "float":
type = "TYPE_FLOAT";
break;
case "label":
type = "TYPE_LABEL";
break;
case "instruction_label":
type = "TYPE_I_LABEL";
break;
case "data_label":
type = "TYPE_D_LABEL";
break;
case "string_label":
type = "TYPE_S_LABEL";
break;
case "string":
type = "TYPE_STRING";
break;
case "reg_ref":
type = "TYPE_REG_REF";
break;
case "reg_tup_ref":
type = `new RegTupRefType(${params_to_code(param.reg_tup)})`;
break;
case "reg_ref_var":
type = "TYPE_REG_REF_VAR";
break;
case "pointer":
type = "TYPE_POINTER";
break;
default:
throw new Error(`Type ${param.type} not implemented.`);
}
const doc = (param.doc && JSON.stringify(param.doc)) || "undefined";
let access: string;
switch (param.access) {
case "read":
access = "ParamAccess.Read";
break;
case "write":
access = "ParamAccess.Write";
break;
case "read_write":
access = "ParamAccess.ReadWrite";
break;
default:
access = "undefined";
break;
}
return `new Param(${type}, ${doc}, ${access})`;
})
.join(", ");
}