mirror of
git://projects.qi-hardware.com/eda-tools.git
synced 2025-01-12 16:00:14 +02:00
211 lines
4.7 KiB
Perl
Executable File
211 lines
4.7 KiB
Perl
Executable File
#!/usr/bin/perl
|
|
|
|
use re 'eval';
|
|
|
|
|
|
#
|
|
# "sub" populates the following global variables:
|
|
#
|
|
# $end[rule-number] = 0 / 1
|
|
# $match[rule-number]{field}[0] = original-pattern
|
|
# $match[rule-number]{field}[1] = RE1
|
|
# $match[rule-number]{field}[2] = RE2
|
|
# $action[rule-number]{field} = value
|
|
#
|
|
# $match_stack[depth]{field}[0] = original-pattern
|
|
# $match_stack[depth]{field}[1] = RE1
|
|
# $match_stack[depth]{field}[2] = RE2
|
|
# $action_stack[depth]{field} = value
|
|
# $may_cont = 0 / 1
|
|
# $last
|
|
# $last_action
|
|
#
|
|
|
|
#
|
|
# $cvn_from{internal-handle} = index
|
|
# $cvn_to{internal-handle} = index
|
|
# $cvn_unit{internal-handle} = unit-name
|
|
# $cvn_num = internal-handle
|
|
# $found{field-or-subfield} = string
|
|
|
|
|
|
#
|
|
# We convert each input pattern into two regular expressions: the first matches
|
|
# units in the nXn notation, e.g., 4u7 or 100R. The second matches them in SI
|
|
# notation (sans space).
|
|
#
|
|
# When matching (sub_match), we first apply the first expression. Each time we
|
|
# encounter a unit ($R, $F, etc.), __cvn is called. __cvn stores the index of
|
|
# the unit in %cvn_from and %cvn_to.
|
|
#
|
|
# We then pick these substrings from the input string and convert the units to
|
|
# SI notation. At the same time, we normalize the mantissa. Once done, we run
|
|
# the second expression. This one always matches (hopefully :-)
|
|
#
|
|
# All (...) ranges in the original pattern have been replaced with named
|
|
# capture buffers in the second expression, so all these subfields are now
|
|
# gathered in the $+ array. (The same also happened in the first pass, but we
|
|
# ignore it.)
|
|
#
|
|
# Finally, when expanding a value (sub_expand), we look for $field and
|
|
# $field:index, and expand accordingly.
|
|
#
|
|
|
|
|
|
sub __cvn
|
|
{
|
|
local ($num) = @_;
|
|
|
|
$cvn_from{$num} = $-[$#-];
|
|
$cvn_to{$num} = $+[$#+];
|
|
}
|
|
|
|
|
|
sub sub_match
|
|
{
|
|
local ($s, $field, $m1, $m2) = @_;
|
|
|
|
#
|
|
# Perform the first match and record where we saw $<unit> patterns.
|
|
#
|
|
undef %cvn_from;
|
|
undef %cvn_to;
|
|
return undef unless $s =~ $m1;
|
|
|
|
#
|
|
# Convert the unit patterns to almost-SI notation. (We don't put a space
|
|
# after the number, but the rest is SI-compliant.)
|
|
#
|
|
my $off = 0;
|
|
for (keys %cvn_from) {
|
|
my $unit = $cvn_unit{$_};
|
|
my $from = $cvn_from{$_}+$off;
|
|
my $len = $cvn_to{$_}-$cvn_from{$_};
|
|
die unless substr($s, $from, $len) =~
|
|
/(\d+)$unit(\d*)|(\d+)([GMkmunpf])(\d*)/;
|
|
|
|
#
|
|
# Normalize to \d+.\d*
|
|
#
|
|
my $v = "$1$3.$2$5";
|
|
my $exp = $4 eq "" ? " " : $4;
|
|
|
|
#
|
|
# Remove leading zeroes.
|
|
#
|
|
$v =~ s/^0*(\d+)/\1/;
|
|
|
|
#
|
|
# Mantissa must be < 1000.
|
|
# Do the math as string operation to avoid rounding errors.
|
|
#
|
|
while ($v =~ /(\d+)(\d{3})\./) {
|
|
$v = "$1.$2$'";
|
|
$exp =~ tr/GMk munpf/TGMk munp/;
|
|
}
|
|
|
|
#
|
|
# Mantissa must be >= 1.
|
|
#
|
|
while ($v =~ /\b0\.(\d+)/) {
|
|
if (length $1 < 3) {
|
|
$v = $1.("0" x (3-length $1)).".";
|
|
} else {
|
|
$v = substr($1, 0, 3).".".substr($1, 3);
|
|
}
|
|
$exp =~ tr/GMk munpf/Mk munpa/;
|
|
}
|
|
|
|
#
|
|
# Remove trailing zeroes
|
|
#
|
|
$v =~ s/(\.[1-9]*)0*/\1/;
|
|
|
|
$exp =~ s/ //;
|
|
$v =~ s/\.$//;
|
|
$v = $v.$exp.$unit;
|
|
$off += length($v)-$len;
|
|
substr($s, $from, $len, $v);
|
|
}
|
|
|
|
#
|
|
# Run the second match on the string to process any (...) patterns
|
|
#
|
|
$found{$field} = $s;
|
|
die $m2 unless $s =~ $m2;
|
|
for (keys %+) {
|
|
$found{$_} = $+{$_};
|
|
}
|
|
return $s;
|
|
}
|
|
|
|
|
|
sub sub_expand
|
|
{
|
|
local ($s) = @_;
|
|
|
|
while ($s =~ /^([^\$]*)\$([A-Za-z_]\w*)(:(\d+))?|^([^\$]*)\${([A-Za-z_]\w*)(:(\d+))?}/) {
|
|
my $name = "$2$6";
|
|
$name .= "__$4$8" if defined($4) || defined($8);
|
|
if (!defined $found{$name}) {
|
|
die "don't know \"$name\"".
|
|
(defined $__match_error ?
|
|
" (processing \"$__match_error\")" : "");
|
|
}
|
|
$s = $1.$5.$found{$name}.$';
|
|
}
|
|
return $s;
|
|
}
|
|
|
|
|
|
#
|
|
# return 0 if all rules have been exhausted, 1 if there was an explicit halt.
|
|
#
|
|
|
|
sub apply_rules
|
|
{
|
|
RULE: for (my $i = 0; $i <= $#match; $i++) {
|
|
print STDERR "RULE #$i\n" if $debug;
|
|
%found = %field;
|
|
FIELD: for my $f (keys %{ $match[$i] }) {
|
|
my @f = $f ne "FN" ? ($f) :
|
|
("F1", "F2", "F3", "F4", "F5", "F6", "F7", "F8", "F9");
|
|
for (@f) {
|
|
print STDERR " MATCH $_=$match[$i]{$f}[0] " if $debug;
|
|
if (!defined $found{$_}) {
|
|
print STDERR "NO FIELD\n" if $debug;
|
|
next;
|
|
}
|
|
print STDERR "FIELD $found{$_} " if $debug;
|
|
if (!defined &sub_match($found{$_}, $f,
|
|
$match[$i]{$f}[1], $match[$i]{$f}[2])) {
|
|
print STDERR "MISS\n" if $debug;
|
|
next;
|
|
}
|
|
print STDERR "MATCH\n" if $debug;
|
|
next FIELD;
|
|
}
|
|
next RULE;
|
|
}
|
|
for (keys %{ $action[$i] }) {
|
|
my $s = &sub_expand($action[$i]{$_});
|
|
print STDERR " SET $_=$action[$i]{$_} => $s\n" if $debug;
|
|
$field{$_} = $s;
|
|
}
|
|
if ($end[$i]) {
|
|
print STDERR " END\n" if $debug;
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
sub match_set_error
|
|
{
|
|
$__match_error = $_[0];
|
|
}
|
|
|
|
|
|
return 1;
|