| # |
| # Copyright 1999, 2000, 2001 Patrik Stridvall |
| # |
| # This library is free software; you can redistribute it and/or |
| # modify it under the terms of the GNU Lesser General Public |
| # License as published by the Free Software Foundation; either |
| # version 2.1 of the License, or (at your option) any later version. |
| # |
| # This library is distributed in the hope that it will be useful, |
| # but WITHOUT ANY WARRANTY; without even the implied warranty of |
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| # Lesser General Public License for more details. |
| # |
| # You should have received a copy of the GNU Lesser General Public |
| # License along with this library; if not, write to the Free Software |
| # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| # |
| |
| package options; |
| |
| use strict; |
| |
| use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); |
| require Exporter; |
| |
| @ISA = qw(Exporter); |
| @EXPORT = qw(); |
| @EXPORT_OK = qw($options &parse_comma_list &parse_value); |
| |
| use vars qw($options); |
| |
| use output qw($output); |
| |
| sub parse_comma_list { |
| my $prefix = shift; |
| my $value = shift; |
| |
| if(defined($prefix) && $prefix eq "no") { |
| return { active => 0, filter => 0, hash => {} }; |
| } elsif(defined($value)) { |
| my %names; |
| for my $name (split /,/, $value) { |
| $names{$name} = 1; |
| } |
| return { active => 1, filter => 1, hash => \%names }; |
| } else { |
| return { active => 1, filter => 0, hash => {} }; |
| } |
| } |
| |
| sub parse_value { |
| my $prefix = shift; |
| my $value = shift; |
| |
| return $value; |
| } |
| |
| package _options; |
| |
| use strict; |
| |
| use output qw($output); |
| |
| sub new { |
| my $proto = shift; |
| my $class = ref($proto) || $proto; |
| my $self = {}; |
| bless ($self, $class); |
| |
| my $options_long = \%{$self->{_OPTIONS_LONG}}; |
| my $options_short = \%{$self->{_OPTIONS_SHORT}}; |
| my $options_usage = \${$self->{_OPTIONS_USAGE}}; |
| |
| my $refoptions_long = shift; |
| my $refoptions_short = shift; |
| $$options_usage = shift; |
| |
| %$options_long = %{$refoptions_long}; |
| %$options_short = %{$refoptions_short}; |
| |
| $self->options_set("default"); |
| |
| my $arguments = \@{$self->{_ARGUMENTS}}; |
| @$arguments = (); |
| |
| my $end_of_options = 0; |
| while(defined($_ = shift @ARGV)) { |
| if(/^--$/) { |
| $end_of_options = 1; |
| next; |
| } elsif($end_of_options) { |
| # Nothing |
| } elsif(/^--(all|none)$/) { |
| $self->options_set("$1"); |
| next; |
| } elsif(/^-([^=]*)(=(.*))?$/) { |
| my $name; |
| my $value; |
| if(defined($2)) { |
| $name = $1; |
| $value = $3; |
| } else { |
| $name = $1; |
| } |
| |
| if($name =~ /^([^-].*)$/) { |
| $name = $$options_short{$1}; |
| } else { |
| $name =~ s/^-(.*)$/$1/; |
| } |
| |
| my $prefix; |
| if(defined($name) && $name =~ /^no-(.*)$/) { |
| $name = $1; |
| $prefix = "no"; |
| if(defined($value)) { |
| $output->write("options with prefix 'no' can't take parameters\n"); |
| |
| return undef; |
| } |
| } |
| |
| my $option; |
| if(defined($name)) { |
| $option = $$options_long{$name}; |
| } |
| |
| if(defined($option)) { |
| my $key = $$option{key}; |
| my $parser = $$option{parser}; |
| my $refvalue = \${$self->{$key}}; |
| my @parents = (); |
| |
| if(defined($$option{parent})) { |
| if(ref($$option{parent}) eq "ARRAY") { |
| @parents = @{$$option{parent}}; |
| } else { |
| @parents = $$option{parent}; |
| } |
| } |
| |
| if(defined($parser)) { |
| if(!defined($value)) { |
| $value = shift @ARGV; |
| } |
| $$refvalue = &$parser($prefix,$value); |
| } else { |
| if(defined($value)) { |
| $$refvalue = $value; |
| } elsif(!defined($prefix)) { |
| $$refvalue = 1; |
| } else { |
| $$refvalue = 0; |
| } |
| } |
| |
| if((ref($$refvalue) eq "HASH" && $$refvalue->{active}) || $$refvalue) { |
| while($#parents >= 0) { |
| my @old_parents = @parents; |
| @parents = (); |
| foreach my $parent (@old_parents) { |
| my $parentkey = $$options_long{$parent}{key}; |
| my $refparentvalue = \${$self->{$parentkey}}; |
| |
| $$refparentvalue = 1; |
| |
| if(defined($$options_long{$parent}{parent})) { |
| if(ref($$options_long{$parent}{parent}) eq "ARRAY") { |
| push @parents, @{$$options_long{$parent}{parent}}; |
| } else { |
| push @parents, $$options_long{$parent}{parent}; |
| } |
| } |
| } |
| } |
| } |
| next; |
| } |
| } |
| |
| if(!$end_of_options && /^-(.*)$/) { |
| $output->write("unknown option: $_\n"); |
| $output->write($$options_usage); |
| exit 1; |
| } else { |
| push @$arguments, $_; |
| } |
| } |
| |
| if($self->help) { |
| $output->write($$options_usage); |
| $self->show_help; |
| exit 0; |
| } |
| |
| return $self; |
| } |
| |
| sub DESTROY { |
| } |
| |
| sub parse_files { |
| my $self = shift; |
| |
| my $arguments = \@{$self->{_ARGUMENTS}}; |
| my $directories = \@{$self->{_DIRECTORIES}}; |
| my $c_files = \@{$self->{_C_FILES}}; |
| my $h_files = \@{$self->{_H_FILES}}; |
| |
| my $error = 0; |
| my @files = (); |
| foreach (@$arguments) { |
| if(!-e $_) { |
| $output->write("$_: no such file or directory\n"); |
| $error = 1; |
| } else { |
| push @files, $_; |
| } |
| } |
| if($error) { |
| exit 1; |
| } |
| |
| my @paths = (); |
| my @c_files = (); |
| my @h_files = (); |
| foreach my $file (@files) { |
| if($file =~ /\.c$/) { |
| push @c_files, $file; |
| } elsif($file =~ /\.h$/) { |
| push @h_files, $file; |
| } else { |
| push @paths, $file; |
| } |
| } |
| |
| if($#c_files == -1 && $#h_files == -1 && $#paths == -1) |
| { |
| @paths = "."; |
| } |
| |
| if($#paths != -1 || $#c_files != -1) { |
| my $c_command = "find " . join(" ", @paths, @c_files) . " -name \\*.c"; |
| my %found; |
| @$c_files = sort(map { |
| s/^\.\/(.*)$/$1/; |
| if(defined($found{$_})) { |
| (); |
| } else { |
| $found{$_}++; |
| $_; |
| } |
| } split(/\n/, `$c_command`)); |
| } |
| |
| if($#paths != -1 || $#h_files != -1) { |
| my $h_command = "find " . join(" ", @paths, @h_files) . " -name \\*.h"; |
| my %found; |
| |
| @$h_files = sort(map { |
| s/^\.\/(.*)$/$1/; |
| if(defined($found{$_})) { |
| (); |
| } else { |
| $found{$_}++; |
| $_; |
| } |
| } split(/\n/, `$h_command`)); |
| } |
| |
| my %dirs; |
| foreach my $file (@$c_files, @$h_files) { |
| my $dir = $file; |
| $dir =~ s%/?[^/]+$%%; |
| if(!$dir) { $dir = "."; } |
| $dirs{$dir}++ |
| } |
| |
| @$directories = sort(keys(%dirs)); |
| } |
| |
| sub options_set { |
| my $self = shift; |
| |
| my $options_long = \%{$self->{_OPTIONS_LONG}}; |
| my $options_short = \%{$self->{_OPTIONS_SHORT}}; |
| |
| local $_ = shift; |
| for my $name (sort(keys(%$options_long))) { |
| my $option = $$options_long{$name}; |
| my $key = uc($name); |
| $key =~ tr/-/_/; |
| $$option{key} = $key; |
| my $refvalue = \${$self->{$key}}; |
| |
| if(/^default$/) { |
| $$refvalue = $$option{default}; |
| } elsif(/^all$/) { |
| if($name !~ /^help|debug|verbose|module$/) { |
| if(ref($$refvalue) ne "HASH") { |
| $$refvalue = 1; |
| } else { |
| $$refvalue = { active => 1, filter => 0, hash => {} }; |
| } |
| } |
| } elsif(/^none$/) { |
| if($name !~ /^help|debug|verbose|module$/) { |
| if(ref($$refvalue) ne "HASH") { |
| $$refvalue = 0; |
| } else { |
| $$refvalue = { active => 0, filter => 0, hash => {} }; |
| } |
| } |
| } |
| } |
| } |
| |
| sub show_help { |
| my $self = shift; |
| |
| my $options_long = \%{$self->{_OPTIONS_LONG}}; |
| my $options_short = \%{$self->{_OPTIONS_SHORT}}; |
| |
| my $maxname = 0; |
| for my $name (sort(keys(%$options_long))) { |
| if(length($name) > $maxname) { |
| $maxname = length($name); |
| } |
| } |
| |
| for my $name (sort(keys(%$options_long))) { |
| my $option = $$options_long{$name}; |
| my $description = $$option{description}; |
| my $default = $$option{default}; |
| my $current = ${$self->{$$option{key}}}; |
| |
| my $value = $current; |
| |
| my $command; |
| if(ref($value) ne "HASH") { |
| if($value) { |
| $command = "--no-$name"; |
| } else { |
| $command = "--$name"; |
| } |
| } else { |
| if($value->{active}) { |
| $command = "--[no-]$name\[=<value>]"; |
| } else { |
| $command = "--$name\[=<value>]"; |
| } |
| } |
| |
| $output->write($command); |
| for (0..(($maxname - length($name) + 17) - (length($command) - length($name) + 1))) { $output->write(" "); } |
| if(ref($value) ne "HASH") { |
| if($value) { |
| $output->write("Disable "); |
| } else { |
| $output->write("Enable "); |
| } |
| } else { |
| if($value->{active}) { |
| $output->write("(Disable) "); |
| } else { |
| $output->write("Enable "); |
| } |
| } |
| if($default == $current) { |
| $output->write("$description (default)\n"); |
| } else { |
| $output->write("$description\n"); |
| } |
| } |
| } |
| |
| sub AUTOLOAD { |
| my $self = shift; |
| |
| my $name = $_options::AUTOLOAD; |
| $name =~ s/^.*::(.[^:]*)$/\U$1/; |
| |
| my $refvalue = $self->{$name}; |
| if(!defined($refvalue)) { |
| die "<internal>: options.pm: member $name does not exists\n"; |
| } |
| |
| if(ref($$refvalue) ne "HASH") { |
| return $$refvalue; |
| } else { |
| return $$refvalue->{active}; |
| } |
| } |
| |
| sub arguments { |
| my $self = shift; |
| |
| my $arguments = \@{$self->{_ARGUMENTS}}; |
| |
| return @$arguments; |
| } |
| |
| sub c_files { |
| my $self = shift; |
| |
| my $c_files = \@{$self->{_C_FILES}}; |
| |
| if(!defined(@$c_files)) { |
| $self->parse_files; |
| } |
| |
| return @$c_files; |
| } |
| |
| sub h_files { |
| my $self = shift; |
| |
| my $h_files = \@{$self->{_H_FILES}}; |
| |
| if(!defined(@$h_files)) { |
| $self->parse_files; |
| } |
| |
| return @$h_files; |
| } |
| |
| sub directories { |
| my $self = shift; |
| |
| my $directories = \@{$self->{_DIRECTORIES}}; |
| |
| if(!defined(@$directories)) { |
| $self->parse_files; |
| } |
| |
| return @$directories; |
| } |
| |
| 1; |