2001-02-23 04:17:06 +05:30
|
|
|
#!/usr/bin/perl -w
|
|
|
|
|
|
|
|
use strict;
|
2001-02-23 08:03:28 +05:30
|
|
|
use Getopt::Long;
|
|
|
|
|
|
|
|
# collect lines continued with a '\' into an array
|
|
|
|
sub continuation {
|
|
|
|
my $fh = shift;
|
|
|
|
my @line;
|
2001-02-23 04:17:06 +05:30
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
while (<$fh>) {
|
|
|
|
my $s = $_;
|
|
|
|
$s =~ s/\\\s*$//;
|
|
|
|
$s =~ s/#.*$//;
|
|
|
|
push @line, $s;
|
|
|
|
last unless (/\\\s*$/);
|
|
|
|
}
|
|
|
|
return @line;
|
|
|
|
}
|
|
|
|
|
|
|
|
# regex && eval away unwanted strings from documentation
|
|
|
|
sub beautify {
|
|
|
|
my $text = shift;
|
2001-02-23 23:25:03 +05:30
|
|
|
$text =~ s/USAGE_NOT\w+\(.*?"\s*\)//sxg;
|
2001-02-23 21:45:34 +05:30
|
|
|
$text =~ s/USAGE_\w+\(\s*?(.*?)"\s*\)/$1"/sxg;
|
|
|
|
$text =~ s/"\s*"//sg;
|
2001-02-23 08:03:28 +05:30
|
|
|
my @line = split("\n", $text);
|
|
|
|
$text = join('',
|
|
|
|
map {
|
|
|
|
s/^\s*//;
|
|
|
|
s/"//g;
|
2001-02-24 20:07:48 +05:30
|
|
|
s/%/%%/g;
|
2001-02-26 08:20:11 +05:30
|
|
|
s/\$/\\\$/g;
|
2001-02-24 20:07:48 +05:30
|
|
|
eval qq[ sprintf(qq#$_#) ]
|
|
|
|
} @line
|
2001-02-23 08:03:28 +05:30
|
|
|
);
|
|
|
|
return $text;
|
|
|
|
}
|
|
|
|
|
|
|
|
# generate POD for an applet
|
|
|
|
sub pod_for_usage {
|
|
|
|
my $name = shift;
|
|
|
|
my $usage = shift;
|
|
|
|
|
2001-02-23 23:11:41 +05:30
|
|
|
# make options bold
|
2001-02-23 08:03:28 +05:30
|
|
|
my $trivial = $usage->{trivial};
|
2001-02-26 08:20:11 +05:30
|
|
|
$trivial =~ s/(?<!\w)(-\w+)/B<$1>/sxg;
|
2001-02-23 23:11:41 +05:30
|
|
|
my @f0 =
|
2001-02-23 08:03:28 +05:30
|
|
|
map { $_ !~ /^\s/ && s/(?<!\w)(-\w+)/B<$1>/g; $_ }
|
2001-02-23 23:11:41 +05:30
|
|
|
split("\n", $usage->{full});
|
|
|
|
|
|
|
|
# add "\n" prior to certain lines to make indented
|
|
|
|
# lines look right
|
2001-02-24 20:07:48 +05:30
|
|
|
my @f1;
|
2001-02-23 23:11:41 +05:30
|
|
|
my $len = @f0;
|
|
|
|
for (my $i = 0; $i < $len; $i++) {
|
|
|
|
push @f1, $f0[$i];
|
|
|
|
if (($i+1) != $len && $f0[$i] !~ /^\s/ && $f0[$i+1] =~ /^\s/) {
|
|
|
|
next if ($f0[$i] =~ /^$/);
|
|
|
|
push(@f1, "") unless ($f0[$i+1] =~ /^\s*$/s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
my $full = join("\n", @f1);
|
2001-02-26 08:20:11 +05:30
|
|
|
|
2001-03-15 23:44:25 +05:30
|
|
|
# prepare notes if they exists
|
|
|
|
my $notes = (defined $usage->{notes})
|
|
|
|
? "$usage->{notes}\n\n"
|
|
|
|
: "";
|
|
|
|
|
2001-02-26 08:20:11 +05:30
|
|
|
# prepare example if one exists
|
|
|
|
my $example = (defined $usage->{example})
|
2001-03-16 02:38:01 +05:30
|
|
|
?
|
|
|
|
"Example:\n\n" .
|
|
|
|
join ("\n",
|
|
|
|
map { "\t$_" }
|
|
|
|
split("\n", $usage->{example})) . "\n\n"
|
2001-02-26 08:20:11 +05:30
|
|
|
: "";
|
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
return
|
2001-04-06 01:05:17 +05:30
|
|
|
"=item B<$name>".
|
2001-02-26 08:20:11 +05:30
|
|
|
"\n\n" .
|
2001-02-23 08:03:28 +05:30
|
|
|
"$name $trivial".
|
2001-02-26 08:20:11 +05:30
|
|
|
"\n\n" .
|
|
|
|
$full .
|
|
|
|
"\n\n" .
|
2001-03-15 23:44:25 +05:30
|
|
|
$notes .
|
2001-02-26 08:20:11 +05:30
|
|
|
$example.
|
|
|
|
"-------------------------------".
|
2001-02-23 08:03:28 +05:30
|
|
|
"\n\n"
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
2001-02-24 20:07:48 +05:30
|
|
|
# FIXME | generate SGML for an applet
|
2001-02-23 08:03:28 +05:30
|
|
|
sub sgml_for_usage {
|
|
|
|
my $name = shift;
|
|
|
|
my $usage = shift;
|
|
|
|
return
|
2001-02-23 23:21:08 +05:30
|
|
|
"<fixme>\n".
|
|
|
|
" $name\n".
|
|
|
|
"</fixme>\n"
|
|
|
|
;
|
2001-02-23 08:03:28 +05:30
|
|
|
}
|
|
|
|
|
2001-02-23 08:24:31 +05:30
|
|
|
# the keys are applet names, and
|
|
|
|
# the values will contain hashrefs of the form:
|
|
|
|
#
|
2001-02-23 08:03:28 +05:30
|
|
|
# {
|
|
|
|
# trivial => "...",
|
|
|
|
# full => "...",
|
2001-03-07 00:55:25 +05:30
|
|
|
# example => "...",
|
2001-02-23 08:03:28 +05:30
|
|
|
# }
|
|
|
|
my %docs;
|
|
|
|
|
2001-02-24 20:07:48 +05:30
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
# get command-line options
|
2001-02-24 20:07:48 +05:30
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
my %opt;
|
|
|
|
|
|
|
|
GetOptions(
|
|
|
|
\%opt,
|
|
|
|
"help|h",
|
|
|
|
"sgml|s",
|
|
|
|
"pod|p",
|
|
|
|
"verbose|v",
|
|
|
|
);
|
|
|
|
|
|
|
|
if (defined $opt{help}) {
|
|
|
|
print
|
|
|
|
"$0 [OPTION]... [FILE]...\n",
|
|
|
|
"\t--help\n",
|
|
|
|
"\t--sgml\n",
|
|
|
|
"\t--pod\n",
|
|
|
|
"\t--verbose\n",
|
|
|
|
;
|
|
|
|
exit 1;
|
|
|
|
}
|
|
|
|
|
2001-02-24 20:07:48 +05:30
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
# collect documenation into %docs
|
2001-02-24 20:07:48 +05:30
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
foreach (@ARGV) {
|
2001-02-26 08:20:11 +05:30
|
|
|
open(USAGE, $_) || die("$0: $_: $!");
|
2001-02-23 08:03:28 +05:30
|
|
|
my $fh = *USAGE;
|
|
|
|
my ($applet, $type, @line);
|
|
|
|
while (<$fh>) {
|
|
|
|
if (/^#define (\w+)_(\w+)_usage/) {
|
|
|
|
$applet = $1;
|
|
|
|
$type = $2;
|
|
|
|
@line = continuation($fh);
|
|
|
|
my $doc = $docs{$applet} ||= { };
|
|
|
|
my $text = join("\n", @line);
|
|
|
|
$doc->{$type} = beautify($text);
|
2001-02-23 04:17:06 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2001-02-23 08:03:28 +05:30
|
|
|
|
2001-02-24 20:07:48 +05:30
|
|
|
|
|
|
|
# generate structured documentation
|
|
|
|
|
2001-02-23 23:21:08 +05:30
|
|
|
my $generator = \&pod_for_usage;
|
|
|
|
if (defined $opt{sgml}) {
|
2001-02-24 20:07:48 +05:30
|
|
|
$generator = \&sgml_for_usage;
|
2001-02-23 23:21:08 +05:30
|
|
|
}
|
2001-02-23 08:03:28 +05:30
|
|
|
|
2001-02-24 20:07:48 +05:30
|
|
|
foreach my $applet (sort keys %docs) {
|
|
|
|
print $generator->($applet, $docs{$applet});
|
2001-02-23 08:03:28 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
exit 0;
|
|
|
|
|
|
|
|
__END__
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
autodocifier.pl - generate docs for busybox based on usage.h
|
|
|
|
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
|
|
|
|
autodocifier.pl usage.h > something
|
|
|
|
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
|
|
|
|
The purpose of this script is to automagically generate documentation
|
|
|
|
for busybox using its usage.h as the original source for content.
|
|
|
|
Currently, the same content has to be duplicated in 3 places in
|
|
|
|
slightly different formats -- F<usage.h>, F<docs/busybox.pod>, and
|
2001-02-23 08:42:45 +05:30
|
|
|
F<docs/busybox.sgml>. This is tedious, so Perl has come to the rescue.
|
2001-02-23 08:03:28 +05:30
|
|
|
|
2001-02-23 08:42:45 +05:30
|
|
|
This script was based on a script by Erik Andersen (andersen@lineo.com).
|
2001-02-23 08:03:28 +05:30
|
|
|
|
|
|
|
=head1 OPTIONS
|
|
|
|
|
2001-02-23 23:21:08 +05:30
|
|
|
=over 4
|
2001-02-23 08:03:28 +05:30
|
|
|
|
2001-04-06 01:05:17 +05:30
|
|
|
=item B<--help>
|
2001-02-23 08:03:28 +05:30
|
|
|
|
|
|
|
This displays the help message.
|
|
|
|
|
2001-04-06 01:05:17 +05:30
|
|
|
=item B<--pod>
|
2001-02-23 23:21:08 +05:30
|
|
|
|
|
|
|
Generate POD (this is the default)
|
|
|
|
|
2001-04-06 01:05:17 +05:30
|
|
|
=item B<--sgml>
|
2001-02-23 23:21:08 +05:30
|
|
|
|
|
|
|
Generate SGML
|
|
|
|
|
2001-04-06 01:05:17 +05:30
|
|
|
=item B<--verbose>
|
2001-02-23 23:21:08 +05:30
|
|
|
|
|
|
|
Be verbose (not implemented)
|
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
=back
|
|
|
|
|
2001-04-06 01:05:17 +05:30
|
|
|
=head1 FORMAT
|
|
|
|
|
|
|
|
The following is an example of some data this script might parse.
|
|
|
|
|
|
|
|
#define length_trivial_usage \
|
|
|
|
"STRING"
|
|
|
|
#define length_full_usage \
|
|
|
|
"Prints out the length of the specified STRING."
|
|
|
|
#define length_example_usage \
|
|
|
|
"$ length "Hello"\n" \
|
|
|
|
"5\n"
|
|
|
|
|
|
|
|
Each entry is a cpp macro that defines a string. The macros are
|
|
|
|
named systematically in the form:
|
|
|
|
|
|
|
|
$name_$type_usage
|
|
|
|
|
|
|
|
$name is the name of the applet. $type can be "trivial", "full", "notes",
|
|
|
|
or "example". Every documentation macro must end with "_usage".
|
|
|
|
|
|
|
|
The definition of the types is as follows:
|
|
|
|
|
|
|
|
=over 4
|
|
|
|
|
|
|
|
=item B<trivial>
|
|
|
|
|
|
|
|
This should be a brief, one-line description of parameters that
|
|
|
|
the command expects. This will be displayed when B<-h> is issued to
|
|
|
|
a command. I<REQUIRED>
|
|
|
|
|
|
|
|
=item B<full>
|
|
|
|
|
|
|
|
This should contain descriptions of each option. This will also
|
|
|
|
be displayed along with the trivial help if BB_FEATURE_TRIVIAL_HELP
|
|
|
|
is disabled. I<REQUIRED>
|
|
|
|
|
|
|
|
=item B<notes>
|
|
|
|
|
|
|
|
This is documentation that is intended to go in the POD or SGML, but
|
|
|
|
not be output when a B<-h> is given to a command. To see an example
|
|
|
|
of notes being used, see init_notes_usage. I<OPTIONAL>
|
|
|
|
|
|
|
|
=item B<example>
|
|
|
|
|
|
|
|
This should be an example of how the command is acutally used.
|
|
|
|
I<OPTIONAL>
|
|
|
|
|
|
|
|
=back
|
|
|
|
|
2001-02-23 08:03:28 +05:30
|
|
|
=head1 FILES
|
|
|
|
|
2001-02-23 23:21:08 +05:30
|
|
|
F<usage.h>
|
2001-02-23 08:03:28 +05:30
|
|
|
|
|
|
|
=head1 COPYRIGHT
|
|
|
|
|
|
|
|
Copyright (c) 2001 John BEPPU. All rights reserved. This program is
|
|
|
|
free software; you can redistribute it and/or modify it under the same
|
|
|
|
terms as Perl itself.
|
|
|
|
|
|
|
|
=head1 AUTHOR
|
|
|
|
|
|
|
|
John BEPPU <beppu@lineo.com>
|
|
|
|
|
|
|
|
=cut
|
|
|
|
|
2001-04-06 01:05:17 +05:30
|
|
|
# $Id: autodocifier.pl,v 1.18 2001/04/05 19:35:17 beppu Exp $
|