diff options
Diffstat (limited to 'genpacksizetables.pl')
-rwxr-xr-x | genpacksizetables.pl | 142 |
1 files changed, 62 insertions, 80 deletions
diff --git a/genpacksizetables.pl b/genpacksizetables.pl index 0fffe9b959..5872e86a9c 100755 --- a/genpacksizetables.pl +++ b/genpacksizetables.pl @@ -4,103 +4,85 @@ use strict; use Encode; -my @lines = grep { - s/#.*//; - /\S/; -} <DATA>; +sub make_text { + my ($chrmap, $letter, $unpredictable, $nocsum, $size, $condition) = @_; + my $text = " /* $letter */ $size"; + $text .= " | PACK_SIZE_UNPREDICTABLE" if $unpredictable; + $text .= " | PACK_SIZE_CANNOT_CSUM" if $nocsum; + $text .= ","; -sub addline { - my ($arrays, $chrmap, $letter, $arrayname, $unpredictable, $nocsum, $size, - $condition) = @_; - my $line = "/* $letter */ $size"; - $line .= " | PACK_SIZE_UNPREDICTABLE" if $unpredictable; - $line .= " | PACK_SIZE_CANNOT_CSUM" if $nocsum; - $line .= ","; - # And then the hack - $line = [$condition, $line] if $condition; - $arrays->{$arrayname}->[ord $chrmap->{$letter}] = $line; - # print ord $chrmap->{$letter}, " $line\n"; + if ($condition) { + $condition = join " && ", map {"defined($_)"} split ' ', $condition; + $text = "#if $condition +$text +#else + 0, +#endif"; + } + return $text; } -sub output_tables { - my %arrays; - - my $chrmap = shift; - foreach (@_) { - my ($letter, $shriek, $unpredictable, $nocsum, $size, $condition) - = /^([A-Za-z])(!?)\t(\S*)\t(\S*)\t([^\t\n]+)(?:\t+(.*))?$/; - die "Can't parse '$_'" unless $size; +sub make_tables { + my %arrays; - if (defined $condition) { - $condition = join " && ", map {"defined($_)"} split ' ', $condition; - } - unless ($size =~ s/^=//) { - $size = "sizeof($size)"; - } + my $chrmap = shift; + foreach (@_) { + my ($letter, $shriek, $unpredictable, $nocsum, $size, $condition) = + /^([A-Za-z])(!?)\t(\S*)\t(\S*)\t([^\t\n]+)(?:\t+(.*))?$/ or + die "Can't parse '$_'"; - addline (\%arrays, $chrmap, $letter, $shriek ? 'shrieking' : 'normal', - $unpredictable, $nocsum, $size, $condition); - } + $size = "sizeof($size)" unless $size =~ s/^=//; - my %earliest; - foreach my $arrayname (sort keys %arrays) { - my $array = $arrays{$arrayname}; - die "No defined entries in $arrayname" unless $array->[$#$array]; - # Find the first used entry - my $earliest = 0; - $earliest++ while (!$array->[$earliest]); - # Remove all the empty elements. - splice @$array, 0, $earliest; - print "unsigned char size_${arrayname}[", scalar @$array, "] = {\n"; - my @lines; - foreach (@$array) { - # Remove the assumption here that the last entry isn't conditonal - if (ref $_) { - push @lines, - ["#if $_->[0]", " $_->[1]", "#else", " 0,", "#endif"]; - } else { - push @lines, $_ ? " $_" : " 0,"; - } + $arrays{$shriek ? 'shrieking' : 'normal'}{ord $chrmap->{$letter}} = + make_text($chrmap, $letter, + $unpredictable, $nocsum, $size, $condition); } - # remove the last, annoying, comma - my $last = $lines[$#lines]; - my $got; - foreach (ref $last ? @$last : $last) { - $got += s/,$//; + + my $text = "const packprops_t packprops[512] = {\n"; + foreach my $arrayname (qw(normal shrieking)) { + my $array = $arrays{$arrayname} || + die "No defined entries in $arrayname"; + $text .= " /* $arrayname */\n"; + for my $ch (0..255) { + $text .= $array->{$ch} || " 0,"; + $text .= "\n"; + } } - die "Last entry had no commas" unless $got; - print map {"$_\n"} ref $_ ? @$_ : $_ foreach @lines; - print "};\n"; - $earliest{$arrayname} = $earliest; - } + # Join "0," entries together + 1 while $text =~ s/\b0,\s*\n\s*0,/0, 0,/g; + # But split them up again if the sequence gets too long + $text =~ s/((?:\b0, ){15}0,) /$1\n /g; + # Clean up final , + $text =~ s/,$//; + $text .= "};"; + return $text; +} - print "struct packsize_t packsize[2] = {\n"; +my @lines = grep { + s/#.*//; + /\S/; +} <DATA>; - my @lines; - foreach (qw(normal shrieking)) { - my $array = $arrays{$_}; - push @lines, " {size_$_, $earliest{$_}, " . (scalar @$array) . "},"; - } - # remove the last, annoying, comma - chop $lines[$#lines]; - print "$_\n" foreach @lines; - print "};\n"; -} +my %asciimap = map {chr $_, chr $_} 0..255; +my %ebcdicmap = map {chr $_, Encode::encode("posix-bc", chr $_)} 0..255; -my %asciimap = (map {chr $_, chr $_} 0..255); -my %ebcdicmap = (map {chr $_, Encode::encode ("posix-bc", chr $_)} 0..255); +print <<"EOC"; +/* These tables are regenerated by genpacksizetables.pl (and then hand pasted + in). You're unlikely ever to need to regenerate them. */ -print <<'EOC'; +#if TYPE_IS_SHRIEKING != 0x100 + ++++shriek offset should be 256 +#endif + +typedef U8 packprops_t; #if 'J'-'I' == 1 /* ASCII */ -EOC -output_tables (\%asciimap, @lines); -print <<'EOC'; +@{[make_tables (\%asciimap, @lines)]} #else /* EBCDIC (or bust) */ +@{[make_tables (\%ebcdicmap, @lines)]} +#endif EOC -output_tables (\%ebcdicmap, @lines); -print "#endif\n"; __DATA__ #Symbol unpredictable |