mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-06 06:05:40 +12:00
487 lines
16 KiB
Perl
487 lines
16 KiB
Perl
#!/usr/bin/perl
|
|
use strict;
|
|
use warnings;
|
|
use diagnostics;
|
|
|
|
# This works for IMGUI 1.60 and does not get all functions
|
|
#
|
|
# to use ./generate_imgui_bindings.pl <../imgui/imgui.h >imgui_iterator.inl
|
|
# and define macros properly as in example imgui_lua_bindings.cpp
|
|
#
|
|
# check imgui_iterator for explanations of why some functions are not supported yet
|
|
|
|
require "./parse_blocks.pl";
|
|
|
|
sub generateNamespaceImgui {
|
|
my ($imguiCodeBlock) = @_;
|
|
|
|
my $lineCaptureRegex = qr" *(IMGUI_API) *((const char\*)|([^ ]+)) *([^\(]+)\(([^\;]*)\);";
|
|
my $doEndStackOptions = 1;
|
|
my $terminator = "} \/\/ namespace ImGui";
|
|
my $callPrefix = "";
|
|
my $functionSuffix = "";
|
|
|
|
#define bannedNames with keys of functions to exclude them
|
|
# EXAMPLE:
|
|
my %bannedNames = (
|
|
"NewFrame" => "banned",
|
|
"Render" => "banned",
|
|
"Shutdown" => "banned" );
|
|
#
|
|
# This is only useful for ENABLE_IM_LUA_END_STACK
|
|
# We hold a list of differnet 'things' that can be pushed to the stack
|
|
# i.e. Group for BeginGroup
|
|
# It usually works like this BeginBlah EndBlah
|
|
|
|
# We have to redefine stuff when it doesn't work so cleanly
|
|
my %beginN = (
|
|
"TreeNode" => "Tree",
|
|
"TreePush" => "Tree",
|
|
"PushStyleVar" => "StyleVar"
|
|
);
|
|
my %changeN = (
|
|
"Tree" => "TreePop",
|
|
"StyleVar"=> "PopStyleVar"
|
|
);
|
|
my %endN = (
|
|
"TreePop" => "Tree",
|
|
"PopStyleVar" => "StyleVar"
|
|
);
|
|
my %endOverride = (
|
|
"PopupModal" => "Popup",
|
|
"PopupContextItem" => "Popup",
|
|
"PopupContextWindow" => "Popup",
|
|
"PopupContextVoid" => "Popup" );
|
|
|
|
generateImguiGeneric(
|
|
$lineCaptureRegex,
|
|
$doEndStackOptions,
|
|
$terminator,
|
|
$callPrefix,
|
|
$functionSuffix,
|
|
\%bannedNames,
|
|
\%beginN,
|
|
\%changeN,
|
|
\%endN,
|
|
\%endOverride,
|
|
$imguiCodeBlock)
|
|
}
|
|
|
|
sub generateDrawListFunctions {
|
|
my ($imguiCodeBlock) = @_;
|
|
|
|
my $lineCaptureRegex = qr" *(IMGUI_API|inline) *((const char\*)|([^ ]+)) *([^\(]+)\(([^\;]*)\);";
|
|
my $doEndStackOptions = 0;
|
|
my $terminator = 0;
|
|
my $callPrefix = "DRAW_LIST_";
|
|
my $functionSuffix = "_DRAW_LIST";
|
|
|
|
#define bannedNames with keys of functions to exclude them
|
|
# EXAMPLE:
|
|
my %bannedNames = (
|
|
);
|
|
#
|
|
# This is only useful for ENABLE_IM_LUA_END_STACK
|
|
# We hold a list of differnet 'things' that can be pushed to the stack
|
|
# i.e. Group for BeginGroup
|
|
# It usually works like this BeginBlah EndBlah
|
|
|
|
# We have to redefine stuff when it doesn't work so cleanly
|
|
my %beginN = (
|
|
);
|
|
my %changeN = (
|
|
);
|
|
my %endN = (
|
|
);
|
|
my %endOverride = (
|
|
);
|
|
|
|
|
|
generateImguiGeneric(
|
|
$lineCaptureRegex,
|
|
$doEndStackOptions,
|
|
$terminator,
|
|
$callPrefix,
|
|
$functionSuffix,
|
|
\%bannedNames,
|
|
\%beginN,
|
|
\%changeN,
|
|
\%endN,
|
|
\%endOverride,
|
|
$imguiCodeBlock)
|
|
}
|
|
|
|
sub generateImguiGeneric {
|
|
my $lineCaptureRegex = shift;
|
|
my $doEndStackOptions = shift;
|
|
my $terminator = shift;
|
|
my $callPrefix = shift;
|
|
my $functionSuffix = shift;
|
|
|
|
#define bannedNames with keys of functions to exclude them
|
|
# EXAMPLE:
|
|
my $bannedNamesRef = shift;
|
|
my %bannedNames = %{$bannedNamesRef};
|
|
#
|
|
# This is only useful for ENABLE_IM_LUA_END_STACK
|
|
# We hold a list of differnet 'things' that can be pushed to the stack
|
|
# i.e. Group for BeginGroup
|
|
# It usually works like this BeginBlah EndBlah
|
|
|
|
# We have to redefine stuff when it doesn't work so cleanly
|
|
my $beginNRef = shift;
|
|
my %beginN = %{$beginNRef};
|
|
my $changeNRef = shift;
|
|
my %changeN = %{$changeNRef};
|
|
my $endNRef = shift;
|
|
my %endN = %{$endNRef};
|
|
my $endOverrideRef = shift;
|
|
my %endOverride = %{$endOverrideRef};
|
|
|
|
my ($imguiCodeBlock) = @_;
|
|
|
|
|
|
my $numSupported = 0;
|
|
my $numUnsupported = 0;
|
|
my $line;
|
|
my %funcNames;
|
|
my %endTypeToInt;
|
|
my @endTypes;
|
|
my @functionsAlreadyAdded;
|
|
foreach $line (split /\n/, $imguiCodeBlock) {
|
|
#replace ImVec2(x, y) with ImVec2 x, y so it's easier for regex (and ImVec4)
|
|
$line =~ s/ImVec2\(([^,]*),([^\)]*)\)/ImVec2 $1 $2/g;
|
|
$line =~ s/ImVec4\(([^,]*),([^\)]*),([^\)]*),([^\)]*)\)/ImVec4 $1 $2 $3 $4/g;
|
|
|
|
#delete this so it's eaiser for regexes
|
|
$line =~ s/ IM_PRINTFARGS\(.\);/;/g;
|
|
if ($line =~ m/$lineCaptureRegex/) {
|
|
print "//" . $line . "\n";
|
|
# this will be set to 0 if something is not supported yet
|
|
my $shouldPrint = 1;
|
|
my @args = split(',', $6);
|
|
# things to do before calling real c++ function
|
|
my @before;
|
|
# arguments to real c++ function
|
|
my @funcArgs;
|
|
# things to do after callign real c++ function
|
|
my @after;
|
|
# real c++ function name
|
|
my $funcName = $5;
|
|
|
|
#say STDERR "Parsing function: " . $funcName;
|
|
if (grep(/^$funcName$/, @functionsAlreadyAdded)) {
|
|
#say STDERR $funcName;
|
|
}
|
|
push @functionsAlreadyAdded, $funcName;
|
|
|
|
if (defined($bannedNames{$funcName})) {
|
|
print "//Not allowed to use this function\n";
|
|
$shouldPrint = 0;
|
|
}
|
|
# c++ type of return value
|
|
my $retType = $2;
|
|
# macro used for calling function
|
|
my $callMacro;
|
|
# if it has a return value (yes I know this is not the cleanest code)
|
|
my $hasRet = 1;
|
|
if ($retType =~ /^void$/) {
|
|
$callMacro = "${callPrefix}CALL_FUNCTION_NO_RET";
|
|
$hasRet = 0;
|
|
} elsif ($retType =~ /^const char\*$/) {
|
|
$callMacro = "${callPrefix}CALL_FUNCTION";
|
|
push(@funcArgs, "const char*");
|
|
push(@after, "PUSH_STRING(ret)");
|
|
} elsif ($retType =~ /^bool$/) {
|
|
$callMacro = "${callPrefix}CALL_FUNCTION";
|
|
push(@funcArgs, "bool");
|
|
push(@after, "PUSH_BOOL(ret)");
|
|
} elsif ($retType =~ /^float$/) {
|
|
$callMacro = "${callPrefix}CALL_FUNCTION";
|
|
push(@funcArgs, "float");
|
|
push(@after, "PUSH_NUMBER(ret)");
|
|
} elsif ($retType =~ /^ImVec2$/) {
|
|
$callMacro = "${callPrefix}CALL_FUNCTION";
|
|
push(@funcArgs, "ImVec2");
|
|
push(@after, "PUSH_NUMBER(ret.x)");
|
|
push(@after, "PUSH_NUMBER(ret.y)");
|
|
} elsif ($retType =~ /^(unsigned int|ImGuiID|ImU32)$/) {
|
|
$callMacro = "${callPrefix}CALL_FUNCTION";
|
|
push(@funcArgs, "unsigned int");
|
|
push(@after, "PUSH_NUMBER(ret)");
|
|
} else {
|
|
print "// Unsupported return type $retType\n";
|
|
$shouldPrint = 0;
|
|
}
|
|
for (my $i = 0; $i < @args; $i++) {
|
|
# bool * x = NULL or bool * x
|
|
if ($args[$i] =~ m/^ *bool *\* *([^ =\[]*)( = NULL|) *$/) {
|
|
my $name = $1;
|
|
if ($2 =~ m/^ = NULL$/) {
|
|
push(@before, "OPTIONAL_BOOL_POINTER_ARG($name)");
|
|
} else {
|
|
push(@before, "BOOL_POINTER_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
push(@after, "END_BOOL_POINTER($name)");
|
|
# float * x
|
|
} elsif ($args[$i] =~ m/^ *float *\* *([^ =\[]*)$/) {
|
|
my $name = $1;
|
|
push(@before, "FLOAT_POINTER_ARG($name)");
|
|
push(@funcArgs, $name);
|
|
push(@after, "END_FLOAT_POINTER($name)");
|
|
# const float *, int count
|
|
} elsif ($args[$i] =~ m/^ *const float *\* *([^ =\[]*)$/ && $args[$i + 1] =~ m/^ *int *([^ =\[]*)_count$/) {
|
|
my $name = $1;
|
|
push(@before, "FLOAT_ARRAY_ARG($name)");
|
|
push(@funcArgs, $name);
|
|
push(@funcArgs, "${name}_count");
|
|
push(@after, "END_FLOAT_ARRAY($name)");
|
|
$i++;
|
|
# float a or float a = number
|
|
} elsif ($args[$i] =~ m/^ *float *([^ =\[]*)( *= *[^ ]*|)$/) {
|
|
my $name = $1;
|
|
if ($2 =~ m/^ *= *([^ ]*)$/) {
|
|
push(@before, "OPTIONAL_NUMBER_ARG($name, $1)");
|
|
} else {
|
|
push(@before, "NUMBER_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
# const char* a or const char* a = NULL or "blah"
|
|
} elsif ($args[$i] =~ m/^ *const char\* *([^ =\[]*)( *= *(NULL|".*")|) *$/) {
|
|
my $name = $1;
|
|
if ($2 =~ m/^ *= *([^ ]*)$/) {
|
|
push(@before, "OPTIONAL_LABEL_ARG($name, $1)");
|
|
} else {
|
|
push(@before, "LABEL_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
#const ImVec2& with default or not
|
|
} elsif ($args[$i] =~ m/^ *(const)? ImVec2&? +([^ ]*) *(= * ImVec2 [^ ]* +[^ ]*|) *$/) {
|
|
my $name = $2;
|
|
if ($3 =~ m/^= * ImVec2 ([^ ]*) +([^ ]*)$/) {
|
|
push(@before, "OPTIONAL_IM_VEC_2_ARG($name, $1, $2)");
|
|
} else {
|
|
push(@before, "IM_VEC_2_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
# ImVec2
|
|
} elsif ($args[$i] =~ m/^ *ImVec2 ([^ ]*) *$/) {
|
|
my $name = $1;
|
|
push(@before, "IM_VEC_2_ARG($name)");
|
|
push(@funcArgs, $name);
|
|
#const ImVec4& with default or not
|
|
} elsif ($args[$i] =~ m/^ *const ImVec4& +([^ ]*) *(= * ImVec4 [^ ]* +[^ ]* +[^ ]* +[^ ]*|) *$/) {
|
|
my $name = $1;
|
|
if ($2 =~ m/^= * ImVec4 +([^ ]*) +([^ ]*) +([^ ]*) +([^ ]*)$/) {
|
|
push(@before, "OPTIONAL_IM_VEC_4_ARG($name, $1, $2, $3, $4)");
|
|
} else {
|
|
push(@before, "IM_VEC_4_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
# one of the various enums
|
|
# we are handling these as ints
|
|
} elsif ($args[$i] =~ m/^ *(ImGuiMouseButton|ImGuiTableBgTarget|ImGuiPopupFlags|ImGuiDataType|ImGuiDir|ImGuiCond|ImGuiFocusedFlags|ImGuiHoveredFlags|ImGuiWindowFlags|ImGuiCol|ImGuiStyleVar|ImGuiAlign|ImGuiColorEditMode|ImGuiMouseCursor|ImGuiSetCond|ImGuiInputTextFlags|ImGuiSelectableFlags|ImGuiSliderFlags|ImDrawFlags|ImGuiButtonFlags|ImGuiColorEditFlags|ImGuiComboFlags|ImGuiDockNodeFlags|ImGuiDragDropFlags|ImGuiPopupFlags|ImGuiTabBarFlags|ImGuiTabItemFlags|ImGuiTableColumnFlags|ImGuiTableFlags|ImGuiTableRowFlags|ImGuiTreeNodeFlags) ([^ =]*)( *= *[0-9]*|) *$/) {
|
|
#These are ints
|
|
my $name = $2;
|
|
if ($3 =~ m/^ *= *([0-9]+)$/) {
|
|
push(@before, "OPTIONAL_INT_ARG($name, $1)");
|
|
} else {
|
|
push(@before, "INT_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
#int with default value or not
|
|
} elsif ($args[$i] =~ m/^ *int ([^ =\[]*)( = [^ ]*|) *$/) {
|
|
my $name = $1;
|
|
if ($2 =~ m/^ = ([^ ]*)$/) {
|
|
push(@before, "OPTIONAL_INT_ARG($name, $1)");
|
|
} else {
|
|
push(@before, "INT_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
#unsigned int with default value or not
|
|
} elsif ($args[$i] =~ m/^ *(unsigned +int|ImGuiID|ImU32) ([^ =\[]*)( = [^ ]*|) *$/) {
|
|
my $name = $2;
|
|
if ($3 =~ m/^ = ([^ ]*)$/) {
|
|
push(@before, "OPTIONAL_UINT_ARG($name, $1)");
|
|
} else {
|
|
push(@before, "UINT_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
#ImTextureID or const ImTextureID&
|
|
# const ImTextureID& is the same thing as var
|
|
# as lua is concerned
|
|
} elsif ($args[$i] =~ m/^ *(ImTextureID|const ImTextureID&) ([^ =\[]*) *$/) {
|
|
my $name = $2;
|
|
push(@before, "IM_TEXTURE_ID_ARG($name)");
|
|
push(@funcArgs, $name);
|
|
# bool with default value or not
|
|
} elsif ($args[$i] =~ m/^ *bool ([^ =\[]*)( *= *true| *= *false|) *$/) {
|
|
my $name = $1;
|
|
if ($2 =~ m/^ *= *([^ ]*)$/) {
|
|
push(@before, "OPTIONAL_BOOL_ARG($name, $1)");
|
|
} else {
|
|
push(@before, "BOOL_ARG($name)");
|
|
}
|
|
push(@funcArgs, $name);
|
|
# int * x
|
|
} elsif ($args[$i] =~ m/^ *int *\* *([^ =\[]*)$/) {
|
|
my $name = $1;
|
|
push(@before, "INT_POINTER_ARG($name)");
|
|
push(@funcArgs, $name);
|
|
push(@after, "END_INT_POINTER($name)");
|
|
# unsigned int * x
|
|
} elsif ($args[$i] =~ m/^ *unsigned +int *\* *([^ =\[]*)$/) {
|
|
my $name = $1;
|
|
push(@before, "UINT_POINTER_ARG($name)");
|
|
push(@funcArgs, $name);
|
|
push(@after, "END_UINT_POINTER($name)");
|
|
# we don't support variadic functions yet but we let you use it without extra variables
|
|
} elsif ($args[$i] =~ m/^ *\.\.\. *$/) {
|
|
print "// Variadic functions aren't suppported but here it is anyway\n";
|
|
} else {
|
|
print "// Unsupported arg type " . $args[$i] . "\n";
|
|
$shouldPrint = 0;
|
|
}
|
|
}
|
|
if ($shouldPrint != 0) {
|
|
my $luaFunc = $funcName;
|
|
# Stupid way of implementing overriding
|
|
while($funcNames{$luaFunc}) {
|
|
$luaFunc .= "_" . scalar(@args);
|
|
}
|
|
$funcNames{$luaFunc} = 1;
|
|
|
|
|
|
print "IMGUI_FUNCTION${functionSuffix}($luaFunc)\n";
|
|
for (my $i = 0; $i < @before; $i++) {
|
|
print $before[$i] . "\n";
|
|
}
|
|
|
|
print $callMacro . "($funcName";
|
|
for (my $i = 0; $i < @funcArgs; $i++) {
|
|
print ", " . $funcArgs[$i];
|
|
}
|
|
print ")\n";
|
|
|
|
#for begin and end stack stuff
|
|
if ($funcName =~ m/^Begin(.*)$/ || defined($beginN{$funcName})) {
|
|
my $curEndType;
|
|
if (defined($beginN{$funcName})) {
|
|
$curEndType = $beginN{$funcName};
|
|
} else {
|
|
$curEndType = $1;
|
|
}
|
|
if (defined($endOverride{$curEndType})) {
|
|
$curEndType = $endOverride{$curEndType};
|
|
}
|
|
if (!defined($endTypeToInt{$curEndType})) {
|
|
$endTypeToInt{$curEndType} = scalar(@endTypes);
|
|
push(@endTypes, $curEndType);
|
|
}
|
|
my $curEndTypeInt = $endTypeToInt{$curEndType};
|
|
if ($hasRet) {
|
|
print "IF_RET_ADD_END_STACK($curEndTypeInt)\n";
|
|
} else {
|
|
print "ADD_END_STACK($curEndTypeInt)\n";
|
|
}
|
|
} elsif ($funcName =~ m/^End(.*)$/ || defined($endN{$funcName})) {
|
|
my $curEndType;
|
|
if (defined($endN{$funcName})) {
|
|
$curEndType = $endN{$funcName};
|
|
} else {
|
|
$curEndType = $1;
|
|
}
|
|
if (defined($endOverride{$curEndType})) {
|
|
$curEndType = $endOverride{$curEndType};
|
|
}
|
|
if (!defined($endTypeToInt{$curEndType})) {
|
|
$endTypeToInt{$curEndType} = scalar(@endTypes);
|
|
push(@endTypes, $curEndType);
|
|
}
|
|
my $curEndTypeInt = $endTypeToInt{$curEndType};
|
|
print "POP_END_STACK($curEndTypeInt)\n"
|
|
}
|
|
|
|
for (my $i = 0; $i < @after; $i++) {
|
|
print $after[$i] . "\n";
|
|
}
|
|
print "END_IMGUI_FUNC\n";
|
|
$numSupported += 1;
|
|
} else {
|
|
$numUnsupported += 1;
|
|
}
|
|
} elsif ($terminator) {
|
|
if ($line =~ m/^${terminator}$/) {
|
|
last;
|
|
}
|
|
}
|
|
}
|
|
#for end stack stuff
|
|
if ($doEndStackOptions)
|
|
{
|
|
print "END_STACK_START\n";
|
|
for (my $i = 0; $i < @endTypes; $i++) {
|
|
my $endFunc;
|
|
if (defined($changeN{$endTypes[$i]})) {
|
|
$endFunc = $changeN{$endTypes[$i]};
|
|
} else {
|
|
$endFunc = "End" . $endTypes[$i];
|
|
}
|
|
print "END_STACK_OPTION($i, " . $endFunc .")\n";
|
|
}
|
|
print "END_STACK_END\n";
|
|
}
|
|
|
|
#debug info
|
|
print STDERR "Supported: $numSupported Unsupported: $numUnsupported\n";
|
|
|
|
}
|
|
|
|
sub generateEnums {
|
|
my $enumName = shift;
|
|
my ($imguiCodeBlock) = @_;
|
|
|
|
my $lineCaptureRegex = qr"^ *(ImGui)([^, _]+)_([a-zA-Z0-9]+)\b";
|
|
|
|
print "START_ENUM($enumName)\n";
|
|
my $line;
|
|
foreach $line (split /\n/, $imguiCodeBlock) {
|
|
if ($line =~ m/$lineCaptureRegex/) {
|
|
die "Malformed enum at $enumName" unless ($2 eq $enumName);
|
|
|
|
print "//" . $line . "\n";
|
|
print "MAKE_ENUM($1$2_$3,$3)\n";
|
|
}
|
|
}
|
|
print "END_ENUM($enumName)\n";
|
|
}
|
|
|
|
|
|
my ($blocksref, $blocknamesref) = parse_blocks();
|
|
|
|
my @blocks = @$blocksref;
|
|
my @blocknames = @$blocknamesref;
|
|
|
|
# @spaderthomas 3/1/2020: ImGui also puts its deprecated functions in namespace ImGui,
|
|
# so we'll end up parsing a couple functions twice and causing compiler errors.
|
|
#
|
|
# This flag just means that we've parsed the main one, so don't parse the next one. If ImGui
|
|
# splits up its header to multiple instances of namespace ImGui, this would break.
|
|
my $alreadyParsedMainImguiNamespace = 0;
|
|
|
|
for (my $i=0; $i < scalar @blocks; $i++) {
|
|
print "//" . $blocknames[$i] . "\n";
|
|
if (($blocknames[$i] eq "namespace ImGui\n") and not $alreadyParsedMainImguiNamespace) {
|
|
$alreadyParsedMainImguiNamespace = 1;
|
|
generateNamespaceImgui($blocks[$i]);
|
|
}
|
|
if ($blocknames[$i] =~ m/enum ImGui(.*)_\n/) {
|
|
generateEnums($1, $blocks[$i]);
|
|
}
|
|
if ($blocknames[$i] eq "struct ImDrawList\n") {
|
|
generateDrawListFunctions($blocks[$i]);
|
|
}
|
|
}
|
|
|
|
|