Michal Marek 411abb
#! /usr/bin/perl -w
Michal Marek 411abb
Michal Marek 411abb
use File::Basename;
Michal Marek 411abb
use File::Path;
Michal Marek 411abb
use File::Find;
Michal Marek 411abb
use Getopt::Long;
Michal Marek 411abb
use strict;
Michal Marek 411abb
Michal Marek 411abb
my %symbol_type_name = (
Michal Marek ee6da8
    n => 'normal', t => 'typedef', e => 'enum', s => 'struct', u => 'union',
Michal Marek ee6da8
    E => 'enum constant'
Michal Marek 411abb
);
Michal Marek 411abb
Michal Marek 411abb
my %definitions;
Michal Marek 411abb
my %override;
Michal Marek 411abb
my %override_locally;
Michal Marek 411abb
my %locally_unknown;
Michal Marek 411abb
my %locally_defined;
Michal Marek 411abb
Michal Marek 411abb
sub expand_types($);
Michal Marek 411abb
sub expand_types($) {
Michal Marek 411abb
    my ($definition) = @_;
Michal Marek 411abb
    local ($_, $1, $2);
Michal Marek 411abb
Michal Marek 411abb
    my @defn = split ' ', $definition;
Michal Marek 411abb
    for (@defn[1..$#defn]) {
Michal Marek 411abb
	if (/^(.)#(.*)/) {
Michal Marek 411abb
	    #print "<<$defn[0] : $_>>\n";
Michal Marek 411abb
	    next if exists $locally_defined{$_};
Michal Marek 411abb
	    $locally_defined{$_} = 1;
Michal Marek 411abb
Michal Marek 411abb
	    if ($locally_unknown{$_}) {
Michal Marek 411abb
                print "override " if $override_locally{$_};
Michal Marek 411abb
		print "$_ $symbol_type_name{$1} $2 { UNKNOWN } \n";
Michal Marek 411abb
	    } else {
Michal Marek 411abb
		if (!exists $definitions{$_}) {
Michal Marek 411abb
		    die "Missing definition of $symbol_type_name{$1} $2\n";
Michal Marek 411abb
		}
Michal Marek 411abb
		expand_types("$_ $definitions{$_}");
Michal Marek 411abb
	    }
Michal Marek 411abb
	}
Michal Marek 411abb
    }
Michal Marek 411abb
    print "override " if $override_locally{$defn[0]};
Michal Marek 411abb
    print "$definition\n";
Michal Marek 411abb
}
Michal Marek 411abb
Michal Marek 411abb
sub pack_dump($$) {
Michal Marek 411abb
    my ($dir, $ext) = @_;
Michal Marek 411abb
    my @files;
Michal Marek 411abb
Michal Marek 411abb
    $ext = ".symtypes" unless defined $ext;
Michal Marek 411abb
    $dir =~ s/\/+$//;
Michal Marek 411abb
Michal Marek 411abb
    find(sub ($) { /\Q$ext\E$/ && push @files, $File::Find::name}, $dir);
Michal Marek 411abb
    map { s/^\Q$dir\E\/(.*)\Q$ext\E$/$1/ } @files;
Michal Marek 411abb
Michal Marek 411abb
    foreach my $file (sort @files) {
Michal Marek 411abb
	print "/* $file.o */\n";
Michal Marek 411abb
Michal Marek 411abb
	local *FD;
Michal Marek 411abb
	open FD, "< $dir/$file$ext"
Michal Marek 411abb
	    or die "$dir/$file$ext: $!\n";
Michal Marek 411abb
	while (<FD>) {
Michal Marek 411abb
	    chomp;
Michal Marek 411abb
Michal Marek 411abb
	    my $override = "";
Michal Marek 411abb
	    if (s/^override //) {
Michal Marek 411abb
		$override = $&;
Michal Marek 411abb
	    }
Michal Marek 411abb
Michal Marek 411abb
	    if (/^(\S)#(\S+)\s*(.*)/) {
Michal Marek 411abb
		my $sym = "$1#$2";
Michal Marek 411abb
		my $definition = $3;
Michal Marek 411abb
Michal Marek 411abb
		if (/^$sym\s+$symbol_type_name{$1}\s+$2\s+{\s+UNKNOWN\s+}\s*$/) {
Michal Marek 411abb
		    $_ = $override . substr($sym, 0, 1) . "##" . substr($sym, 2);
Michal Marek 411abb
		} else {
Michal Marek 411abb
		    if (exists $definitions{$sym} && $definitions{$sym} eq $definition) {
Michal Marek 411abb
			if (($override ne "") == (exists $override{$sym})) {
Michal Marek 411abb
			    next;
Michal Marek 411abb
			}
Michal Marek 411abb
			$_ = "$override$sym";
Michal Marek 411abb
		    } else {
Michal Marek 411abb
			$definitions{$sym} = $definition;
Michal Marek 411abb
			if ($override eq "") {
Michal Marek 411abb
			    delete $override{$sym};
Michal Marek 411abb
			} else {
Michal Marek 411abb
			    $override{$sym} = 1;
Michal Marek 411abb
			    $_ = "$override$_";
Michal Marek 411abb
			}
Michal Marek 411abb
		    }
Michal Marek 411abb
		}
Michal Marek 411abb
	    } elsif ($override) {
Michal Marek 411abb
		    $_ = "$override$_";
Michal Marek 411abb
	    }
Michal Marek 411abb
	    print "$_\n";
Michal Marek 411abb
	}
Michal Marek 411abb
	close FD;
Michal Marek 411abb
	print "\n";
Michal Marek 411abb
    }
Michal Marek 411abb
}
Michal Marek 411abb
Michal Marek 411abb
sub unpack_dump($$) {
Michal Marek 411abb
    my ($dir, $ext) = @_;
Michal Marek 411abb
Michal Marek 411abb
    $ext = ".symref" unless defined $ext;
Michal Marek 411abb
Michal Marek 411abb
    while (<STDIN>) {
Michal Marek 411abb
	next if /^$/;
Michal Marek 411abb
	chomp;
Michal Marek 411abb
Michal Marek 411abb
	if (/^\/\* (.*)\.o \*\//) {
Michal Marek 411abb
	    close STDOUT;
Michal Marek 411abb
	    mkpath(dirname("$dir/$1$ext"));
Michal Marek 411abb
	    open STDOUT, "> $dir/$1$ext"
Michal Marek 411abb
		or die "$dir/$1$ext: $!\n";
Michal Marek 411abb
	    %locally_defined = ();
Michal Marek 411abb
	    %locally_unknown = ();
Michal Marek 411abb
	    %override_locally = %override;
Michal Marek 411abb
	    next;
Michal Marek 411abb
	}
Michal Marek 411abb
Michal Marek 411abb
	my $override = /^override\s/;
Michal Marek 411abb
	s/^override\s//;
Michal Marek 411abb
Michal Marek 411abb
	if (/^([^ ])#(#?)([^ ]+) *(.*)$/) {
Michal Marek 411abb
	    my $sym = "$1#$3";
Michal Marek 411abb
Michal Marek 411abb
	    if ($4 ne "") {
Michal Marek 411abb
		if (/\s+{\s+UNKNOWN\s+}\s*$/) {
Michal Marek 411abb
		    $locally_unknown{$sym} = 1;
Michal Marek 411abb
		    $override_locally{$sym} = $override;
Michal Marek 411abb
		} else {
Michal Marek 411abb
		    $definitions{$sym} = $4;
Michal Marek 411abb
		    $locally_unknown{$sym} = 0;
Michal Marek 411abb
		    $override{$sym} = $override;
Michal Marek 411abb
		    $override_locally{$sym} = $override;
Michal Marek 411abb
		}
Michal Marek 411abb
	    } else {
Michal Marek 411abb
		$locally_unknown{$sym} = ($2 ne "");
Michal Marek 411abb
		$override_locally{$sym} = $override;
Michal Marek 411abb
	    }
Michal Marek 411abb
	    next;
Michal Marek 411abb
	} elsif (/^([^# ]*)/) {
Michal Marek 411abb
		$override_locally{$1} = $override;
Michal Marek 411abb
	}
Michal Marek 411abb
	expand_types($_);
Michal Marek 411abb
    }
Michal Marek 411abb
}
Michal Marek 411abb
Michal Marek 411abb
my ($pack, $unpack, $ext);
Michal Marek 411abb
GetOptions("pack" => \$pack, "unpack" => \$unpack, "ext:s" => \$ext)
Michal Marek 411abb
    && ($pack || $unpack) && @ARGV == 1
Michal Marek 411abb
    or die "USAGE:\t$0 [--ext extension] --pack {dir} > file\n" .
Michal Marek 411abb
		 "\t$0 [--ext extension] --unpack {dir} < file\n";
Michal Marek 411abb
Michal Marek 411abb
pack_dump($ARGV[0], $ext) if $pack;
Michal Marek 411abb
unpack_dump($ARGV[0], $ext) if $unpack;