Improve Code
All checks were successful
CI / Build (push) Successful in 16s

This commit is contained in:
TheBrokenRail 2025-04-02 14:43:56 -04:00
parent 4018ff0fb9
commit 3365e5932a
5 changed files with 78 additions and 67 deletions

View File

@ -1,4 +1,4 @@
// Get Custom Data // Get Custom Wrapper From Object
template <typename Data> template <typename Data>
Data *custom_get(typename Data::__Self *self) { Data *custom_get(typename Data::__Self *self) {
return (Data *) (self + 1); return (Data *) (self + 1);
@ -86,6 +86,7 @@ protected:
void __set_vtable() { void __set_vtable() {
static __VTable *vtable = nullptr; static __VTable *vtable = nullptr;
if (!vtable) { if (!vtable) {
// Create VTable
vtable = __dup_vtable(__VTable::base); vtable = __dup_vtable(__VTable::base);
__patch_vtable(vtable); __patch_vtable(vtable);
} }

View File

@ -10,7 +10,6 @@ export const STRUCTURE_FILES: Record<string, string> = {};
export const COMMENT = '//'; export const COMMENT = '//';
export const INTERNAL = '__'; export const INTERNAL = '__';
export const BUILDING_SYMBOLS_GUARD = 'BUILDING_SYMBOLS_LIB'; export const BUILDING_SYMBOLS_GUARD = 'BUILDING_SYMBOLS_LIB';
export const CONSTRUCTOR_FUNCTION_NAME = 'constructor';
export const SELF_ARG = 'self'; export const SELF_ARG = 'self';
// Read Definition File // Read Definition File
export function readDefinition(name: string) { export function readDefinition(name: string) {

View File

@ -1,4 +1,5 @@
import { CONSTRUCTOR_FUNCTION_NAME, formatType, forwardArguments, INDENT, INTERNAL, SELF_ARG } from './common'; import { formatType, forwardArguments, INDENT, INTERNAL, SELF_ARG } from './common';
import { Method } from './method';
import { Struct } from './struct'; import { Struct } from './struct';
import { structuresWithVTableAddress } from './vtable'; import { structuresWithVTableAddress } from './vtable';
@ -15,9 +16,8 @@ export class CustomWrapperGenerator {
#getBase() { #getBase() {
return `${INTERNAL}CustomBase<${this.#struct.name}, ${this.#struct.getVTable().getName()}>`; return `${INTERNAL}CustomBase<${this.#struct.name}, ${this.#struct.getVTable().getName()}>`;
} }
#getMethods() { #getVirtualMethods() {
const out = []; const out = [];
out.push(this.#struct.getMainConstructor());
const methods = this.#struct.getVTable().getMethods(false); const methods = this.#struct.getVTable().getMethods(false);
for (const method of methods) { for (const method of methods) {
if (method) { if (method) {
@ -30,6 +30,55 @@ export class CustomWrapperGenerator {
return 'Custom' + this.#struct.name; return 'Custom' + this.#struct.name;
} }
// Generate Methods
#generateMethods(methods: Method[], isConstructor: boolean) {
let out = '';
for (const method of methods) {
// Generate Method Signature
out += INDENT;
if (isConstructor) {
out += 'explicit ' + this.#getName();
} else {
out += 'virtual ' + formatType(method.returnType) + method.shortName;
}
out += method.getArgs(false) + ';\n';
}
return out;
}
#generateMethodsCode(methods: Method[], isConstructor: boolean) {
let out = '';
for (const method of methods) {
const name = method.shortName;
// Generate Method Signature
if (!isConstructor) {
out += formatType(method.returnType);
}
out += this.#getName() + '::';
out += isConstructor ? this.#getName() : name;
const args = method.getArgs(false);
out += args + ' {\n';
// Generate Method Code
out += INDENT;
if (!isConstructor) {
// Virtual Method
if (method.doesReturnValue()) {
out += 'return ';
}
out += `${INTERNAL}VTable::base->${name}`;
} else {
// Constructor
out += method.getDirectCall();
}
out += forwardArguments(args, ['this->self']) + ';\n';
if (isConstructor) {
// Setup VTable
out += `${INDENT}${INTERNAL}set_vtable();\n`;
}
out += '}\n';
}
return out;
}
// Generate Header // Generate Header
generate() { generate() {
let out = ''; let out = '';
@ -40,26 +89,15 @@ export class CustomWrapperGenerator {
} }
// Structure // Structure
const base = this.#getBase(); out += `struct ${this.#getName()} : ${this.#getBase()} {\n`;
out += `struct ${this.#getName()} : ${base} {\n`;
// Methods // Methods
const methods = this.#getMethods(); out += this.#generateMethods(this.#struct.getConstructors(), true);
for (const method of methods) { out += this.#generateMethods(this.#getVirtualMethods(), false);
// Generate Method
const name = method.shortName;
out += INDENT;
if (name === CONSTRUCTOR_FUNCTION_NAME) {
out += 'explicit ' + this.#getName();
} else {
out += 'virtual ' + formatType(method.returnType) + name;
}
out += method.getArgs(false) + ';\n';
}
// VTable // VTable
out += 'private:\n'; out += 'private:\n';
out += INDENT + `virtual void ${INTERNAL}patch_vtable(${base}::${INTERNAL}VTable *) override;\n`; out += INDENT + `void ${INTERNAL}patch_vtable(${INTERNAL}VTable *) override;\n`;
out += '};\n'; out += '};\n';
// Return // Return
@ -71,36 +109,9 @@ export class CustomWrapperGenerator {
let out = ''; let out = '';
// Methods // Methods
const methods = this.#getMethods(); out += this.#generateMethodsCode(this.#struct.getConstructors(), true);
for (const method of methods) { const methods = this.#getVirtualMethods();
const name = method.shortName; out += this.#generateMethodsCode(methods, false);
const isConstructor = name === CONSTRUCTOR_FUNCTION_NAME;
// Generate Method
if (!isConstructor) {
out += formatType(method.returnType);
}
out += this.#getName() + '::';
out += isConstructor ? this.#getName() : name;
const args = method.getArgs(false);
out += args + ' {\n';
out += INDENT;
const forwardedArgs = forwardArguments(args, ['this->self']);
if (!isConstructor) {
// Virtual Method
if (method.doesReturnValue()) {
out += 'return ';
}
out += `${INTERNAL}VTable::base->${name}${forwardedArgs}`;
} else {
// Constructor
out += method.getDirectCall() + forwardedArgs;
}
out += ';\n';
if (isConstructor) {
out += `${INDENT}${INTERNAL}set_vtable();\n`;
}
out += '}\n';
}
// VTable // VTable
const base = this.#getBase(); const base = this.#getBase();
@ -108,9 +119,6 @@ export class CustomWrapperGenerator {
out += `${INDENT}${base}::${INTERNAL}patch_vtable(vtable);\n`; out += `${INDENT}${base}::${INTERNAL}patch_vtable(vtable);\n`;
for (const method of methods) { for (const method of methods) {
const name = method.shortName; const name = method.shortName;
if (name === CONSTRUCTOR_FUNCTION_NAME) {
continue;
}
// Patch Entry // Patch Entry
out += `${INDENT}vtable->${name} = []${method.getArgs()} {\n`; out += `${INDENT}vtable->${name} = []${method.getArgs()} {\n`;
out += INDENT + INDENT; out += INDENT + INDENT;

View File

@ -1,4 +1,4 @@
import { COMMENT, CONSTRUCTOR_FUNCTION_NAME, extendErrorMessage, EXTENSION, parseTypeAndName, readDefinition, safeParseInt, syntaxError } from './common'; import { COMMENT, extendErrorMessage, EXTENSION, parseTypeAndName, readDefinition, safeParseInt, syntaxError } from './common';
import { Method } from './method'; import { Method } from './method';
import { Property, StaticProperty } from './property'; import { Property, StaticProperty } from './property';
import { Struct } from './struct'; import { Struct } from './struct';
@ -146,27 +146,27 @@ export function load(target: Struct, name: string, isExtended: boolean) {
case 'method': { case 'method': {
// Add Method // Add Method
const method = parseMethod(args, name, true, isExtended); const method = parseMethod(args, name, true, isExtended);
target.addMethod(method, false); target.addMethod(method, false, false);
break; break;
} }
case 'virtual-method': { case 'virtual-method': {
// Add Virtual Method // Add Virtual Method
const method = parseMethod(args, target.name, true, isExtended); const method = parseMethod(args, target.name, true, isExtended);
target.addMethod(method, true); target.addMethod(method, true, false);
break; break;
} }
case 'static-method': { case 'static-method': {
// Add Static Method // Add Static Method
if (!isExtended) { if (!isExtended) {
const method = parseMethod(args, target.name, false, false); const method = parseMethod(args, target.name, false, false);
target.addMethod(method, false); target.addMethod(method, false, false);
} }
break; break;
} }
case 'constructor': { case 'constructor': {
// Constructor // Constructor
if (!isExtended) { if (!isExtended) {
let data = target.name + ' *' + CONSTRUCTOR_FUNCTION_NAME; let data = target.name + ' *constructor';
if (args.startsWith('(')) { if (args.startsWith('(')) {
// No Custom Name // No Custom Name
} else { } else {
@ -175,7 +175,7 @@ export function load(target: Struct, name: string, isExtended: boolean) {
} }
data += args; data += args;
const method = parseMethod(data, target.name, true, false); const method = parseMethod(data, target.name, true, false);
target.addMethod(method, false); target.addMethod(method, false, true);
} }
break; break;
} }

View File

@ -1,4 +1,4 @@
import { INDENT, STRUCTURE_FILES, toHex, assertSize, INTERNAL, preventConstruction, BUILDING_SYMBOLS_GUARD, formatType, forwardArguments, CONSTRUCTOR_FUNCTION_NAME } from './common'; import { INDENT, STRUCTURE_FILES, toHex, assertSize, INTERNAL, preventConstruction, BUILDING_SYMBOLS_GUARD, formatType, forwardArguments } from './common';
import { CustomWrapperGenerator } from './custom'; import { CustomWrapperGenerator } from './custom';
import { Method } from './method'; import { Method } from './method';
import { Property, StaticProperty } from './property'; import { Property, StaticProperty } from './property';
@ -9,6 +9,7 @@ export class Struct {
readonly name: string; readonly name: string;
#vtable: VTable | null; #vtable: VTable | null;
readonly #methods: Method[]; readonly #methods: Method[];
readonly #constructors: Method[];
readonly #properties: Property[]; readonly #properties: Property[];
#size: number | null; #size: number | null;
readonly #dependencies: string[]; readonly #dependencies: string[];
@ -21,6 +22,7 @@ export class Struct {
constructor(name: string) { constructor(name: string) {
this.name = name; this.name = name;
this.#methods = []; this.#methods = [];
this.#constructors = [];
this.#properties = []; this.#properties = [];
this.#vtable = null; this.#vtable = null;
this.#size = null; this.#size = null;
@ -54,7 +56,7 @@ export class Struct {
} }
// Add Method // Add Method
addMethod(method: Method, isVirtual: boolean) { addMethod(method: Method, isVirtual: boolean, isConstructor: boolean) {
if (method.returnType !== this.name && method.returnType in STRUCTURE_FILES) { if (method.returnType !== this.name && method.returnType in STRUCTURE_FILES) {
this.#addDependency(method.returnType); this.#addDependency(method.returnType);
} }
@ -66,6 +68,8 @@ export class Struct {
} else { } else {
if (method.isInherited) { if (method.isInherited) {
this.#addDependency(method.self); this.#addDependency(method.self);
} else if (isConstructor) {
this.#constructors.push(method);
} }
this.#methods.push(method); this.#methods.push(method);
} }
@ -106,13 +110,12 @@ export class Struct {
this.getVTable(); this.getVTable();
this.#custom = new CustomWrapperGenerator(this); this.#custom = new CustomWrapperGenerator(this);
} }
getMainConstructor() { getConstructors() {
for (const method of this.#methods) { const out = this.#constructors;
if (method.shortName === CONSTRUCTOR_FUNCTION_NAME) { if (out.length === 0) {
return method; throw new Error('Custom Wrappers Require Constructors');
}
} }
throw new Error('Unable To Find Main Constructor'); return out;
} }
// Generate Properties // Generate Properties