blob: b55609077dfd99edd3addf2f904e8ad128c4f5d7 [file] [log] [blame]
#!/usr/bin/env perl
############################################################################
# __________ __ ___.
# Open \______ \ ____ ____ | | _\_ |__ _______ ___
# Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
# Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
# Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
# \/ \/ \/ \/ \/
# $Id$
#
# Copyright (C) 2019 by William Wilgus
#
# All files in this archive are subject to the GNU General Public License.
# See the file COPYING in the source tree root for full license agreement.
#
# This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
# KIND, either express or implied.
#
############################################################################
#rockbox to lua define generator, add names of constants to the array to include
if ($#ARGV + 1 != 1) {
warn "no definition type defined";
exit;
}
my $def_type = $ARGV[0];
#warn "$def_type\n";
my $lua_table;
my @rockbox_defines;
if ($def_type eq "rb_defines") {
$lua_table = "rb";
@rockbox_defines = (
'^HZ$',
'^LCD_(DEPTH|HEIGHT|WIDTH)$',
'^MODEL_NAME$',
'^SCREEN_MAIN$',
'^LCD_DEFAULT_(FG|BG)$',
'^LCD_REMOTE_(DEPTH|HEIGHT|WIDTH)$',
'^LCD_.+(BRIGHT|DARK)COLOR',
'^SCREEN_REMOTE$',
'^FONT_SYSFIXED$',
'^FONT_UI$',
'^PLAYBACK_EVENT_.*',
'^PLAYLIST_(INSERT|PREPEND|REPLACE)',
'^TOUCHSCREEN_(POINT|BUTTON)$',
'^SYS_CHARGER_(DIS|)CONNECTED$',
'^SYS_(TIMEOUT|POWEROFF)$',
'^SYS_USB_(DIS|)CONNECTED$',
'^HOME_DIR$',
'^PLUGIN_DIR$',
'^PLUGIN(_APPS_|_GAMES_|_)DATA_DIR$',
'^ROCKBOX_DIR$',
'^VIEWERS_DATA_DIR$');
}
elsif ($def_type eq "sound_defines") {
$lua_table = "rb.sound_settings";
@rockbox_defines = (
'^(?!.*LAST_SETTING)SOUND_');
}
my @captured_defines;
my @names_seen;
my @all_defines;
############# precompile regex for speed #############
my $def_regex = qr/#define\s+([^\s\r\n]+)\s+([^\r\n]+)/;
my $quot_regex = qr/.*([\"\']).*/;
my $num_regex = qr/.*([\+\-\*\\|&\d]).*/;
my $configh_regex = qr/^\s*#define\s*__CONFIG_H__\s*$/;
my $config_h = "?";
my $exclude_regex = qr/^#define\s*_?POSIX.*/;
my $exclude_enum_regex = qr/^#define\s*(_SC_|_PC_|_CS_).*/;
print <<EOF
#include <stdio.h>
#include <stdbool.h>
#define stringify(s) #s
/* auto generated defines */
EOF
;
while(my $line = <STDIN>)
{
if($config_h eq "?" && $line =~ $configh_regex) { $config_h = $line; }
next if($config_h eq "?"); #don't capture till we get to the config file
next if($line =~ $exclude_regex);
if($line =~ $def_regex) #does it begin with #define?
{
push(@all_defines, $line); #save all defines for possible ambiguous macros
$name = $1;
next if $name =~ /^(ATTRIBUTE_|print|__).*/; #don't add reserved
$value = $2;
if(grep($name =~ $_, @rockbox_defines))
{
push(@names_seen, $name);
push(@captured_defines, {'name' => $name, 'value' => $value});
print $line
}
else
{
$name =~ s{(\(.*\))}{}gx; #remove (...) on function type macros
#ifndef guard so we don't clash with the host
printf "#ifndef %s\n%s#endif\n\n", $name, $line;
}
}
elsif($line =~ /^(?!.*;)enum.*{/) #enum {
{
next if($line =~ /enum\s*__.*/); #don't add reserved
print $line;
next if($line =~ /.*};.*/);
do_enum($line)
}
elsif($line =~ /^(?!.*[;\(\)])enum.*/) #enum
{
next if($line =~ /enum\s*__.*/); #don't add reserved
#need to be careful might be a function returning enum
my $lastline = $line;
next if($line =~ /.*};.*/);
($line = <STDIN>);
if($line =~ /.*{.*/)
{
print $lastline;
print $line;
}
else { next; }
do_enum($line)
}
elsif($line =~ /^enum.*{[^;]+};.*/) #enum {
{
next if($line =~ /enum\s*__.*/); #don't add reserved
next if(do_enum($line));
}
}
#warn "total defines: ".scalar @all_defines;
#warn "captured defines: ".scalar @captured_defines;
#Sort the functions
my @sorted_defines = sort { @$a{'name'} cmp @$b{'name'} } @captured_defines;
printf "int main(void)\n{\n";
printf "\tprintf(\"--[[Autogenerated rockbox constants]]\\n\\n\");\n\n";
printf "\tprintf(\"%s = %s or {}\\n\");\n", $lua_table, $lua_table;
# Print the C array
foreach my $define (@sorted_defines)
{
if(@$define{'value'} =~ /^0[xX][0-9a-fA-F]+$/) #hex number
{
printf "\tprintf(\"%s[\\\"%%s\\\"] = 0x%%x\\n\", stringify(%s), %s);\n", $lua_table, @$define{'name'}, @$define{'name'};
}
elsif(@$define{'value'} =~ /^[0-9]+$/) #number
{
printf "\tprintf(\"%s[\\\"%%s\\\"] = %%d\\n\", stringify(%s), %s);\n", $lua_table, @$define{'name'}, @$define{'name'};
}
else #might be a string but we don't know since the macro isn't expanded far enough
{
my $max_depth = 10;
my $var = @$define{'value'};
while($max_depth > 0) #follow the chain of #defines until we can see what type
{
$max_depth--;
$var = "\\\s*#define\\s+$var\\s+.*";
#warn $var;
if(my ($x) = grep($_ =~ /($var)/, @all_defines)) #check all the defines
{
#warn $x;
if($x =~ $def_regex)
{
$var = $2;
#warn $var;
last if ($var =~ $quot_regex);
last if ($var =~ $num_regex);
next
}
#warn "end ".$var;
last
}
else
{
$var = @$define{'value'};
last
}
}
if ($var =~$quot_regex) #has a quote it is a string
{
#guard with empty literals "" so gcc throws an error if it isn't a string
printf "\tprintf(\"%s[\\\"%%s\\\"] = \\\"%%s\\\"\\n\", stringify(%s), \"\" %s \"\");\n", $lua_table, @$define{'name'}, @$define{'name'};
}
elsif ($var =~$num_regex) #it must be a number
{
printf "\tprintf(\"%s[\\\"%%s\\\"] = %%d\\n\", stringify(%s), %s);\n", $lua_table, @$define{'name'}, @$define{'name'};
}
else { warn "Skipping ".@$define{'name'}." indeterminate macro type\n"; }
}
}
print <<EOF
return 0;
}
EOF
;
sub do_enum {
my ($line) = @_;
if($line =~ /.*enum.*{(.*)};.*/) #single line enums
{
print $line;
$value = "0"; #enums are always integers
my $enum = $1;
$enum =~ s/\s+//g;;
my @values = split(',', $enum);
foreach my $name(@values) {
if(grep($name =~ $_, @rockbox_defines))
{
push(@names_seen, $name);
push(@captured_defines, {'name' => $name, 'value' => $value});
}
}
return 1;
}
while($line = <STDIN>)
{
next if($line =~ $exclude_enum_regex);
if($line =~ /.*};.*/)
{
print $line;
last
}
elsif($line =~ /([^\s,\t]+)\s*=?.*,?/)
{
$name = $1;
#printf "%s WHATTT?", $name;
$value = "0"; #enums are always integers
}
print $line;
if(grep($name =~ $_, @rockbox_defines))
{
push(@names_seen, $name);
push(@captured_defines, {'name' => $name, 'value' => $value});
}
}
return 0;
}