2002-02-23 01:24:02 +00:00
|
|
|
#!/usr/bin/perl -w
|
|
|
|
#-
|
2003-06-01 12:54:03 +00:00
|
|
|
# Copyright (c) 2002-2003 Networks Associates Technology, Inc.
|
2011-12-18 17:08:40 +00:00
|
|
|
# Copyright (c) 2004-2011 Dag-Erling Smørgrav
|
2002-02-23 01:24:02 +00:00
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# This software was developed for the FreeBSD Project by ThinkSec AS and
|
2002-06-30 21:30:05 +00:00
|
|
|
# Network Associates Laboratories, the Security Research Division of
|
|
|
|
# Network Associates, Inc. under DARPA/SPAWAR contract N66001-01-C-8035
|
|
|
|
# ("CBOSS"), as part of the DARPA CHATS research program.
|
2002-02-23 01:24:02 +00:00
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions
|
|
|
|
# are met:
|
|
|
|
# 1. Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
# 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer in the
|
|
|
|
# documentation and/or other materials provided with the distribution.
|
|
|
|
# 3. The name of the author may not be used to endorse or promote
|
|
|
|
# products derived from this software without specific prior written
|
|
|
|
# permission.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
|
|
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
# SUCH DAMAGE.
|
|
|
|
#
|
2011-12-18 17:08:40 +00:00
|
|
|
# $Id: gendoc.pl 465 2011-11-02 20:34:26Z des $
|
2002-02-23 01:24:02 +00:00
|
|
|
#
|
|
|
|
|
|
|
|
use strict;
|
2003-07-15 07:18:26 +00:00
|
|
|
use locale;
|
2002-02-23 01:24:02 +00:00
|
|
|
use Fcntl;
|
2002-06-30 21:30:05 +00:00
|
|
|
use Getopt::Std;
|
2003-07-15 07:18:26 +00:00
|
|
|
use POSIX qw(locale_h strftime);
|
2011-12-18 17:08:40 +00:00
|
|
|
use vars qw($COPYRIGHT %AUTHORS $TODAY %FUNCTIONS %PAMERR);
|
2002-02-23 01:24:02 +00:00
|
|
|
|
2002-06-30 21:30:05 +00:00
|
|
|
$COPYRIGHT = ".\\\"-
|
2003-06-01 12:54:03 +00:00
|
|
|
.\\\" Copyright (c) 2001-2003 Networks Associates Technology, Inc.
|
2011-12-18 17:08:40 +00:00
|
|
|
.\\\" Copyright (c) 2004-2011 Dag-Erling Smørgrav
|
2002-06-30 21:30:05 +00:00
|
|
|
.\\\" All rights reserved.
|
|
|
|
.\\\"
|
|
|
|
.\\\" This software was developed for the FreeBSD Project by ThinkSec AS and
|
|
|
|
.\\\" Network Associates Laboratories, the Security Research Division of
|
|
|
|
.\\\" Network Associates, Inc. under DARPA/SPAWAR contract N66001-01-C-8035
|
|
|
|
.\\\" (\"CBOSS\"), as part of the DARPA CHATS research program.
|
|
|
|
.\\\"
|
|
|
|
.\\\" Redistribution and use in source and binary forms, with or without
|
|
|
|
.\\\" modification, are permitted provided that the following conditions
|
|
|
|
.\\\" are met:
|
|
|
|
.\\\" 1. Redistributions of source code must retain the above copyright
|
|
|
|
.\\\" notice, this list of conditions and the following disclaimer.
|
|
|
|
.\\\" 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
.\\\" notice, this list of conditions and the following disclaimer in the
|
|
|
|
.\\\" documentation and/or other materials provided with the distribution.
|
|
|
|
.\\\" 3. The name of the author may not be used to endorse or promote
|
|
|
|
.\\\" products derived from this software without specific prior written
|
|
|
|
.\\\" permission.
|
|
|
|
.\\\"
|
|
|
|
.\\\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
.\\\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
.\\\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
.\\\" ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
|
|
.\\\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
.\\\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
.\\\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
.\\\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
.\\\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
.\\\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
.\\\" SUCH DAMAGE.
|
|
|
|
.\\\"
|
2011-12-18 17:08:40 +00:00
|
|
|
.\\\" \$" . "Id" . "\$
|
2002-06-30 21:30:05 +00:00
|
|
|
.\\\"";
|
|
|
|
|
2011-12-18 17:08:40 +00:00
|
|
|
%AUTHORS = (
|
|
|
|
THINKSEC => "ThinkSec AS and Network Associates Laboratories, the
|
|
|
|
Security Research Division of Network Associates, Inc.\\& under
|
|
|
|
DARPA/SPAWAR contract N66001-01-C-8035
|
|
|
|
.Pq Dq CBOSS ,
|
|
|
|
as part of the DARPA CHATS research program.",
|
|
|
|
DES => ".An Dag-Erling Sm\\(/orgrav Aq des\@FreeBSD.org .",
|
|
|
|
);
|
|
|
|
|
2002-02-23 01:24:02 +00:00
|
|
|
%PAMERR = (
|
|
|
|
PAM_SUCCESS => "Success",
|
|
|
|
PAM_OPEN_ERR => "Failed to load module",
|
|
|
|
PAM_SYMBOL_ERR => "Invalid symbol",
|
|
|
|
PAM_SERVICE_ERR => "Error in service module",
|
|
|
|
PAM_SYSTEM_ERR => "System error",
|
|
|
|
PAM_BUF_ERR => "Memory buffer error",
|
|
|
|
PAM_CONV_ERR => "Conversation failure",
|
|
|
|
PAM_PERM_DENIED => "Permission denied",
|
|
|
|
PAM_MAXTRIES => "Maximum number of tries exceeded",
|
|
|
|
PAM_AUTH_ERR => "Authentication error",
|
|
|
|
PAM_NEW_AUTHTOK_REQD => "New authentication token required",
|
|
|
|
PAM_CRED_INSUFFICIENT => "Insufficient credentials",
|
|
|
|
PAM_AUTHINFO_UNAVAIL => "Authentication information is unavailable",
|
|
|
|
PAM_USER_UNKNOWN => "Unknown user",
|
|
|
|
PAM_CRED_UNAVAIL => "Failed to retrieve user credentials",
|
|
|
|
PAM_CRED_EXPIRED => "User credentials have expired",
|
|
|
|
PAM_CRED_ERR => "Failed to set user credentials",
|
2005-02-01 10:16:17 +00:00
|
|
|
PAM_ACCT_EXPIRED => "User account has expired",
|
2002-02-23 01:24:02 +00:00
|
|
|
PAM_AUTHTOK_EXPIRED => "Password has expired",
|
|
|
|
PAM_SESSION_ERR => "Session failure",
|
|
|
|
PAM_AUTHTOK_ERR => "Authentication token failure",
|
|
|
|
PAM_AUTHTOK_RECOVERY_ERR => "Failed to recover old authentication token",
|
|
|
|
PAM_AUTHTOK_LOCK_BUSY => "Authentication token lock busy",
|
|
|
|
PAM_AUTHTOK_DISABLE_AGING => "Authentication token aging disabled",
|
|
|
|
PAM_NO_MODULE_DATA => "Module data not found",
|
|
|
|
PAM_IGNORE => "Ignore this module",
|
|
|
|
PAM_ABORT => "General failure",
|
|
|
|
PAM_TRY_AGAIN => "Try again",
|
|
|
|
PAM_MODULE_UNKNOWN => "Unknown module type",
|
|
|
|
PAM_DOMAIN_UNKNOWN => "Unknown authentication domain",
|
|
|
|
);
|
|
|
|
|
|
|
|
sub parse_source($) {
|
|
|
|
my $fn = shift;
|
|
|
|
|
|
|
|
local *FILE;
|
|
|
|
my $source;
|
|
|
|
my $func;
|
|
|
|
my $descr;
|
|
|
|
my $type;
|
|
|
|
my $args;
|
|
|
|
my $argnames;
|
|
|
|
my $man;
|
|
|
|
my $inlist;
|
2011-12-18 17:08:40 +00:00
|
|
|
my $intaglist;
|
2002-02-23 01:24:02 +00:00
|
|
|
my $inliteral;
|
|
|
|
my %xref;
|
|
|
|
my @errors;
|
2011-12-18 17:08:40 +00:00
|
|
|
my $author;
|
2002-02-23 01:24:02 +00:00
|
|
|
|
|
|
|
if ($fn !~ m,\.c$,) {
|
|
|
|
warn("$fn: not C source, ignoring\n");
|
2002-06-30 21:30:05 +00:00
|
|
|
return undef;
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
|
2011-12-18 17:08:40 +00:00
|
|
|
open(FILE, "<", "$fn")
|
2002-02-23 01:24:02 +00:00
|
|
|
or die("$fn: open(): $!\n");
|
|
|
|
$source = join('', <FILE>);
|
|
|
|
close(FILE);
|
|
|
|
|
2002-06-30 21:30:05 +00:00
|
|
|
return undef
|
|
|
|
if ($source =~ m/^ \* NOPARSE\s*$/m);
|
2002-02-23 01:24:02 +00:00
|
|
|
|
2011-12-18 17:08:40 +00:00
|
|
|
$author = 'THINKSEC';
|
|
|
|
if ($source =~ s/^ \* AUTHOR\s+(.*?)\s*$//m) {
|
|
|
|
$author = $1;
|
|
|
|
}
|
|
|
|
|
2002-02-23 01:24:02 +00:00
|
|
|
$func = $fn;
|
|
|
|
$func =~ s,^(?:.*/)?([^/]+)\.c$,$1,;
|
|
|
|
if ($source !~ m,\n \* ([\S ]+)\n \*/\n\n([\S ]+)\n$func\((.*?)\)\n\{,s) {
|
|
|
|
warn("$fn: can't find $func\n");
|
2002-06-30 21:30:05 +00:00
|
|
|
return undef;
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
($descr, $type, $args) = ($1, $2, $3);
|
|
|
|
$descr =~ s,^([A-Z][a-z]),lc($1),e;
|
|
|
|
$descr =~ s,[\.\s]*$,,;
|
|
|
|
while ($args =~ s/^((?:[^\(]|\([^\)]*\))*),\s*/$1\" \"/g) {
|
|
|
|
# nothing
|
|
|
|
}
|
|
|
|
$args =~ s/,\s+/, /gs;
|
|
|
|
$args = "\"$args\"";
|
|
|
|
|
|
|
|
%xref = (
|
2005-02-01 10:16:17 +00:00
|
|
|
3 => { 'pam' => 1 },
|
2002-02-23 01:24:02 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
if ($type eq "int") {
|
|
|
|
foreach (split("\n", $source)) {
|
|
|
|
next unless (m/^ \*\s+(!?PAM_[A-Z_]+|=[a-z_]+)\s*$/);
|
|
|
|
push(@errors, $1);
|
|
|
|
}
|
2005-02-01 10:16:17 +00:00
|
|
|
++$xref{3}->{'pam_strerror'};
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$argnames = $args;
|
2005-02-01 10:16:17 +00:00
|
|
|
# extract names of regular arguments
|
2002-02-23 01:24:02 +00:00
|
|
|
$argnames =~ s/\"[^\"]+\*?\b(\w+)\"/\"$1\"/g;
|
2005-02-01 10:16:17 +00:00
|
|
|
# extract names of function pointer arguments
|
|
|
|
$argnames =~ s/\"([\w\s\*]+)\(\*?(\w+)\)\([^\)]+\)\"/\"$2\"/g;
|
|
|
|
# escape metacharacters (there shouldn't be any, but...)
|
2002-02-23 01:24:02 +00:00
|
|
|
$argnames =~ s/([\|\[\]\(\)\.\*\+\?])/\\$1/g;
|
2005-02-01 10:16:17 +00:00
|
|
|
# separate argument names with |
|
2002-02-23 01:24:02 +00:00
|
|
|
$argnames =~ s/\" \"/|/g;
|
2005-02-01 10:16:17 +00:00
|
|
|
# and surround with ()
|
2002-02-23 01:24:02 +00:00
|
|
|
$argnames =~ s/^\"(.*)\"$/($1)/;
|
2005-02-01 10:16:17 +00:00
|
|
|
# $argnames is now a regexp that matches argument names
|
2011-12-18 17:08:40 +00:00
|
|
|
$inliteral = $inlist = $intaglist = 0;
|
2002-02-23 01:24:02 +00:00
|
|
|
foreach (split("\n", $source)) {
|
|
|
|
s/\s*$//;
|
|
|
|
if (!defined($man)) {
|
|
|
|
if (m/^\/\*\*$/) {
|
|
|
|
$man = "";
|
|
|
|
}
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
last if (m/^ \*\/$/);
|
|
|
|
s/^ \* ?//;
|
|
|
|
s/\\(.)/$1/gs;
|
|
|
|
if (m/^$/) {
|
2011-12-18 17:08:40 +00:00
|
|
|
# paragraph separator
|
2002-02-23 01:24:02 +00:00
|
|
|
if ($man ne "" && $man !~ m/\.Pp\n$/s) {
|
|
|
|
if ($inliteral) {
|
|
|
|
$man .= "\0\n";
|
2011-12-18 17:08:40 +00:00
|
|
|
} elsif ($inlist || $intaglist) {
|
2002-06-30 21:30:05 +00:00
|
|
|
$man .= ".El\n.Pp\n";
|
2011-12-18 17:08:40 +00:00
|
|
|
$inlist = $intaglist = 0;
|
2002-02-23 01:24:02 +00:00
|
|
|
} else {
|
|
|
|
$man .= ".Pp\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
next;
|
|
|
|
}
|
2005-02-01 10:16:17 +00:00
|
|
|
if (m/^>(\w+)(\s+\d)?$/) {
|
2011-12-18 17:08:40 +00:00
|
|
|
# "see also" cross-reference
|
2005-02-01 10:16:17 +00:00
|
|
|
my ($page, $sect) = ($1, $2 ? int($2) : 3);
|
|
|
|
++$xref{$sect}->{$page};
|
2002-02-23 01:24:02 +00:00
|
|
|
next;
|
|
|
|
}
|
2011-12-18 17:08:40 +00:00
|
|
|
if (s/^\s+-\s+//) {
|
|
|
|
# item in bullet list
|
2002-02-23 01:24:02 +00:00
|
|
|
if ($inliteral) {
|
|
|
|
$man .= ".Ed\n";
|
|
|
|
$inliteral = 0;
|
|
|
|
}
|
2011-12-18 17:08:40 +00:00
|
|
|
if ($intaglist) {
|
|
|
|
$man .= ".El\n.Pp\n";
|
|
|
|
$intaglist = 0;
|
|
|
|
}
|
2002-02-23 01:24:02 +00:00
|
|
|
if (!$inlist) {
|
|
|
|
$man =~ s/\.Pp\n$//s;
|
2011-12-18 17:08:40 +00:00
|
|
|
$man .= ".Bl -bullet\n";
|
2002-02-23 01:24:02 +00:00
|
|
|
$inlist = 1;
|
|
|
|
}
|
2011-12-18 17:08:40 +00:00
|
|
|
$man .= ".It\n";
|
|
|
|
# fall through
|
|
|
|
} elsif (s/^\s+(\S+):\s*/.It $1/) {
|
|
|
|
# item in tag list
|
|
|
|
if ($inliteral) {
|
|
|
|
$man .= ".Ed\n";
|
|
|
|
$inliteral = 0;
|
|
|
|
}
|
|
|
|
if ($inlist) {
|
|
|
|
$man .= ".El\n.Pp\n";
|
|
|
|
$inlist = 0;
|
|
|
|
}
|
|
|
|
if (!$intaglist) {
|
|
|
|
$man =~ s/\.Pp\n$//s;
|
|
|
|
$man .= ".Bl -tag -width 18n\n";
|
|
|
|
$intaglist = 1;
|
|
|
|
}
|
2002-02-23 01:24:02 +00:00
|
|
|
s/^\.It =([A-Z][A-Z_]+)$/.It Dv $1/gs;
|
|
|
|
$man .= "$_\n";
|
|
|
|
next;
|
2011-12-18 17:08:40 +00:00
|
|
|
} elsif (($inlist || $intaglist) && m/^\S/) {
|
|
|
|
# regular text after list
|
2002-06-30 21:30:05 +00:00
|
|
|
$man .= ".El\n.Pp\n";
|
2011-12-18 17:08:40 +00:00
|
|
|
$inlist = $intaglist = 0;
|
2002-02-23 01:24:02 +00:00
|
|
|
} elsif ($inliteral && m/^\S/) {
|
2011-12-18 17:08:40 +00:00
|
|
|
# regular text after literal section
|
2002-02-23 01:24:02 +00:00
|
|
|
$man .= ".Ed\n";
|
2002-05-02 04:40:21 +00:00
|
|
|
$inliteral = 0;
|
2002-02-23 01:24:02 +00:00
|
|
|
} elsif ($inliteral) {
|
2011-12-18 17:08:40 +00:00
|
|
|
# additional text within literal section
|
2002-02-23 01:24:02 +00:00
|
|
|
$man .= "$_\n";
|
|
|
|
next;
|
2011-12-18 17:08:40 +00:00
|
|
|
} elsif ($inlist || $intaglist) {
|
|
|
|
# additional text within list
|
2002-02-23 01:24:02 +00:00
|
|
|
s/^\s+//;
|
|
|
|
} elsif (m/^\s+/) {
|
2011-12-18 17:08:40 +00:00
|
|
|
# new literal section
|
2002-02-23 01:24:02 +00:00
|
|
|
$man .= ".Bd -literal\n";
|
|
|
|
$inliteral = 1;
|
|
|
|
$man .= "$_\n";
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
s/\s*=$func\b\s*/\n.Nm\n/gs;
|
2005-02-01 10:16:17 +00:00
|
|
|
s/\s*=$argnames\b\s*/\n.Fa $1\n/gs;
|
2002-02-23 01:24:02 +00:00
|
|
|
s/\s*=(struct \w+(?: \*)?)\b\s*/\n.Vt $1\n/gs;
|
2002-05-24 13:18:43 +00:00
|
|
|
s/\s*:([a-z_]+)\b\s*/\n.Va $1\n/gs;
|
|
|
|
s/\s*;([a-z_]+)\b\s*/\n.Dv $1\n/gs;
|
2003-06-01 12:54:03 +00:00
|
|
|
while (s/\s*=([a-z_]+)\b\s*/\n.Xr $1 3\n/s) {
|
2005-02-01 10:16:17 +00:00
|
|
|
++$xref{3}->{$1};
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
s/\s*\"(?=\w)/\n.Do\n/gs;
|
|
|
|
s/\"(?!\w)\s*/\n.Dc\n/gs;
|
|
|
|
s/\s*=([A-Z][A-Z_]+)\b\s*(?![\.,:;])/\n.Dv $1\n/gs;
|
|
|
|
s/\s*=([A-Z][A-Z_]+)\b([\.,:;]+)\s*/\n.Dv $1 $2\n/gs;
|
|
|
|
s/\s*{([A-Z][a-z] .*?)}\s*/\n.$1\n/gs;
|
|
|
|
$man .= "$_\n";
|
|
|
|
}
|
|
|
|
if (defined($man)) {
|
2011-12-18 17:08:40 +00:00
|
|
|
if ($inlist || $intaglist) {
|
2002-05-02 04:40:21 +00:00
|
|
|
$man .= ".El\n";
|
2011-12-18 17:08:40 +00:00
|
|
|
$inlist = $intaglist = 0;
|
2002-05-02 04:40:21 +00:00
|
|
|
}
|
|
|
|
if ($inliteral) {
|
|
|
|
$man .= ".Ed\n";
|
2011-12-18 17:08:40 +00:00
|
|
|
$inliteral = 0;
|
2002-05-02 04:40:21 +00:00
|
|
|
}
|
2011-12-18 17:08:40 +00:00
|
|
|
$man =~ s/\%/\\&\%/gs;
|
2002-02-23 01:24:02 +00:00
|
|
|
$man =~ s/(\n\.[A-Z][a-z] [\w ]+)\n([\.,:;-]\S*)\s*/$1 $2\n/gs;
|
|
|
|
$man =~ s/\s*$/\n/gm;
|
|
|
|
$man =~ s/\n+/\n/gs;
|
|
|
|
$man =~ s/\0//gs;
|
2002-05-02 04:40:21 +00:00
|
|
|
$man =~ s/\n\n\./\n\./gs;
|
2002-02-23 01:24:02 +00:00
|
|
|
chomp($man);
|
|
|
|
} else {
|
|
|
|
$man = "No description available.";
|
|
|
|
}
|
|
|
|
|
|
|
|
$FUNCTIONS{$func} = {
|
2002-06-30 21:30:05 +00:00
|
|
|
'source' => $fn,
|
2002-02-23 01:24:02 +00:00
|
|
|
'name' => $func,
|
|
|
|
'descr' => $descr,
|
|
|
|
'type' => $type,
|
|
|
|
'args' => $args,
|
|
|
|
'man' => $man,
|
|
|
|
'xref' => \%xref,
|
|
|
|
'errors' => \@errors,
|
2011-12-18 17:08:40 +00:00
|
|
|
'author' => $author,
|
2002-02-23 01:24:02 +00:00
|
|
|
};
|
|
|
|
if ($source =~ m/^ \* NODOC\s*$/m) {
|
|
|
|
$FUNCTIONS{$func}->{'nodoc'} = 1;
|
|
|
|
}
|
|
|
|
if ($source !~ m/^ \* XSSO \d/m) {
|
|
|
|
$FUNCTIONS{$func}->{'openpam'} = 1;
|
|
|
|
}
|
2002-06-30 21:30:05 +00:00
|
|
|
expand_errors($FUNCTIONS{$func});
|
|
|
|
return $FUNCTIONS{$func};
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sub expand_errors($);
|
|
|
|
sub expand_errors($) {
|
|
|
|
my $func = shift; # Ref to function hash
|
|
|
|
|
|
|
|
my %errors;
|
2002-06-30 21:30:05 +00:00
|
|
|
my $ref;
|
|
|
|
my $fn;
|
2002-02-23 01:24:02 +00:00
|
|
|
|
|
|
|
if (defined($func->{'recursed'})) {
|
|
|
|
warn("$func->{'name'}(): loop in error spec\n");
|
|
|
|
return qw();
|
|
|
|
}
|
|
|
|
$func->{'recursed'} = 1;
|
|
|
|
|
|
|
|
foreach (@{$func->{'errors'}}) {
|
|
|
|
if (m/^(PAM_[A-Z_]+)$/) {
|
|
|
|
if (!defined($PAMERR{$1})) {
|
|
|
|
warn("$func->{'name'}(): unrecognized error: $1\n");
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
$errors{$1} = 1;
|
|
|
|
} elsif (m/^!(PAM_[A-Z_]+)$/) {
|
|
|
|
# treat negations separately
|
|
|
|
} elsif (m/^=([a-z_]+)$/) {
|
2002-06-30 21:30:05 +00:00
|
|
|
$ref = $1;
|
|
|
|
if (!defined($FUNCTIONS{$ref})) {
|
|
|
|
$fn = $func->{'source'};
|
|
|
|
$fn =~ s/$func->{'name'}/$ref/;
|
|
|
|
parse_source($fn);
|
|
|
|
}
|
|
|
|
if (!defined($FUNCTIONS{$ref})) {
|
|
|
|
warn("$func->{'name'}(): reference to unknown $ref()\n");
|
2002-02-23 01:24:02 +00:00
|
|
|
next;
|
|
|
|
}
|
2002-06-30 21:30:05 +00:00
|
|
|
foreach (@{$FUNCTIONS{$ref}->{'errors'}}) {
|
2002-02-23 01:24:02 +00:00
|
|
|
$errors{$_} = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
warn("$func->{'name'}(): invalid error specification: $_\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
foreach (@{$func->{'errors'}}) {
|
|
|
|
if (m/^!(PAM_[A-Z_]+)$/) {
|
|
|
|
delete($errors{$1});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete($func->{'recursed'});
|
2002-06-30 21:30:05 +00:00
|
|
|
$func->{'errors'} = [ sort(keys(%errors)) ];
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
|
2005-06-17 08:11:43 +00:00
|
|
|
sub dictionary_order($$) {
|
|
|
|
my ($a, $b) = @_;
|
|
|
|
|
|
|
|
$a =~ s/[^[:alpha:]]//g;
|
|
|
|
$b =~ s/[^[:alpha:]]//g;
|
|
|
|
$a cmp $b;
|
|
|
|
}
|
|
|
|
|
2005-02-01 10:16:17 +00:00
|
|
|
sub genxref($) {
|
|
|
|
my $xref = shift; # References
|
|
|
|
|
|
|
|
my $mdoc = '';
|
|
|
|
my @refs = ();
|
|
|
|
foreach my $sect (sort(keys(%{$xref}))) {
|
2005-06-17 08:11:43 +00:00
|
|
|
foreach my $page (sort(dictionary_order keys(%{$xref->{$sect}}))) {
|
2005-02-01 10:16:17 +00:00
|
|
|
push(@refs, "$page $sect");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while ($_ = shift(@refs)) {
|
|
|
|
$mdoc .= ".Xr $_" .
|
|
|
|
(@refs ? " ,\n" : "\n");
|
|
|
|
}
|
|
|
|
return $mdoc;
|
|
|
|
}
|
|
|
|
|
2002-02-23 01:24:02 +00:00
|
|
|
sub gendoc($) {
|
|
|
|
my $func = shift; # Ref to function hash
|
|
|
|
|
|
|
|
local *FILE;
|
|
|
|
my $mdoc;
|
|
|
|
my $fn;
|
|
|
|
|
|
|
|
return if defined($func->{'nodoc'});
|
|
|
|
|
|
|
|
$mdoc = "$COPYRIGHT
|
|
|
|
.Dd $TODAY
|
|
|
|
.Dt " . uc($func->{'name'}) . " 3
|
|
|
|
.Os
|
|
|
|
.Sh NAME
|
|
|
|
.Nm $func->{'name'}
|
|
|
|
.Nd $func->{'descr'}
|
|
|
|
.Sh LIBRARY
|
|
|
|
.Lb libpam
|
|
|
|
.Sh SYNOPSIS
|
2003-01-06 16:07:51 +00:00
|
|
|
.In sys/types.h
|
2002-02-23 01:24:02 +00:00
|
|
|
.In security/pam_appl.h
|
|
|
|
";
|
|
|
|
if ($func->{'name'} =~ m/_sm_/) {
|
|
|
|
$mdoc .= ".In security/pam_modules.h\n"
|
|
|
|
}
|
2002-05-02 04:40:21 +00:00
|
|
|
if ($func->{'name'} =~ m/openpam/) {
|
|
|
|
$mdoc .= ".In security/openpam.h\n"
|
|
|
|
}
|
2003-07-15 07:18:26 +00:00
|
|
|
$mdoc .= ".Ft \"$func->{'type'}\"
|
2002-02-23 01:24:02 +00:00
|
|
|
.Fn $func->{'name'} $func->{'args'}
|
|
|
|
.Sh DESCRIPTION
|
|
|
|
$func->{'man'}
|
|
|
|
";
|
|
|
|
if ($func->{'type'} eq "int") {
|
|
|
|
$mdoc .= ".Sh RETURN VALUES
|
|
|
|
The
|
|
|
|
.Nm
|
|
|
|
function returns one of the following values:
|
|
|
|
.Bl -tag -width 18n
|
|
|
|
";
|
2002-06-30 21:30:05 +00:00
|
|
|
my @errors = @{$func->{'errors'}};
|
2002-02-23 01:24:02 +00:00
|
|
|
warn("$func->{'name'}(): no error specification\n")
|
|
|
|
unless(@errors);
|
|
|
|
foreach (@errors) {
|
|
|
|
$mdoc .= ".It Bq Er $_\n$PAMERR{$_}.\n";
|
|
|
|
}
|
|
|
|
$mdoc .= ".El\n";
|
|
|
|
} else {
|
|
|
|
if ($func->{'type'} =~ m/\*$/) {
|
|
|
|
$mdoc .= ".Sh RETURN VALUES
|
|
|
|
The
|
|
|
|
.Nm
|
|
|
|
function returns
|
|
|
|
.Dv NULL
|
|
|
|
on failure.
|
|
|
|
";
|
|
|
|
}
|
|
|
|
}
|
2005-02-01 10:16:17 +00:00
|
|
|
$mdoc .= ".Sh SEE ALSO\n" . genxref($func->{'xref'});
|
2002-02-23 01:24:02 +00:00
|
|
|
$mdoc .= ".Sh STANDARDS\n";
|
|
|
|
if ($func->{'openpam'}) {
|
|
|
|
$mdoc .= "The
|
|
|
|
.Nm
|
|
|
|
function is an OpenPAM extension.
|
|
|
|
";
|
|
|
|
} else {
|
|
|
|
$mdoc .= ".Rs
|
|
|
|
.%T \"X/Open Single Sign-On Service (XSSO) - Pluggable Authentication Modules\"
|
|
|
|
.%D \"June 1997\"
|
|
|
|
.Re
|
|
|
|
";
|
|
|
|
}
|
|
|
|
$mdoc .= ".Sh AUTHORS
|
|
|
|
The
|
|
|
|
.Nm
|
2003-07-15 07:18:26 +00:00
|
|
|
function and this manual page were developed for the
|
|
|
|
.Fx
|
2011-12-18 17:08:40 +00:00
|
|
|
Project by\n" . $AUTHORS{$func->{'author'} // 'THINKSEC_DARPA'} . "\n";
|
2002-02-23 01:24:02 +00:00
|
|
|
$fn = "$func->{'name'}.3";
|
2011-12-18 17:08:40 +00:00
|
|
|
if (open(FILE, ">", $fn)) {
|
2002-05-24 13:18:43 +00:00
|
|
|
print(FILE $mdoc);
|
|
|
|
close(FILE);
|
2002-05-02 04:40:21 +00:00
|
|
|
} else {
|
|
|
|
warn("$fn: open(): $!\n");
|
|
|
|
}
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
|
2002-06-30 21:30:05 +00:00
|
|
|
sub readproto($) {
|
|
|
|
my $fn = shift; # File name
|
|
|
|
|
|
|
|
local *FILE;
|
|
|
|
my %func;
|
|
|
|
|
2011-12-18 17:08:40 +00:00
|
|
|
open(FILE, "<", "$fn")
|
2002-06-30 21:30:05 +00:00
|
|
|
or die("$fn: open(): $!\n");
|
|
|
|
while (<FILE>) {
|
|
|
|
if (m/^\.Nm ((?:open)?pam_.*?)\s*$/) {
|
|
|
|
$func{'Nm'} = $func{'Nm'} || $1;
|
|
|
|
} elsif (m/^\.Ft (\S.*?)\s*$/) {
|
|
|
|
$func{'Ft'} = $func{'Ft'} || $1;
|
|
|
|
} elsif (m/^\.Fn (\S.*?)\s*$/) {
|
|
|
|
$func{'Fn'} = $func{'Fn'} || $1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close(FILE);
|
|
|
|
if ($func{'Nm'}) {
|
|
|
|
$FUNCTIONS{$func{'Nm'}} = \%func;
|
|
|
|
} else {
|
|
|
|
warn("No function found\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub gensummary($) {
|
|
|
|
my $page = shift; # Which page to produce
|
2002-02-23 01:24:02 +00:00
|
|
|
|
2002-06-30 21:30:05 +00:00
|
|
|
local *FILE;
|
|
|
|
my $upage;
|
2002-02-23 01:24:02 +00:00
|
|
|
my $func;
|
2002-06-30 21:30:05 +00:00
|
|
|
my %xref;
|
|
|
|
|
2011-12-18 17:08:40 +00:00
|
|
|
open(FILE, ">", "$page.3")
|
2002-06-30 21:30:05 +00:00
|
|
|
or die("$page.3: $!\n");
|
2002-02-23 01:24:02 +00:00
|
|
|
|
2011-12-18 17:08:40 +00:00
|
|
|
$page =~ m/(\w+)$/;
|
|
|
|
$upage = uc($1);
|
2002-06-30 21:30:05 +00:00
|
|
|
print FILE "$COPYRIGHT
|
2002-02-23 01:24:02 +00:00
|
|
|
.Dd $TODAY
|
2002-06-30 21:30:05 +00:00
|
|
|
.Dt $upage 3
|
2002-02-23 01:24:02 +00:00
|
|
|
.Os
|
|
|
|
.Sh NAME
|
|
|
|
";
|
|
|
|
my @funcs = sort(keys(%FUNCTIONS));
|
|
|
|
while ($func = shift(@funcs)) {
|
2002-06-30 21:30:05 +00:00
|
|
|
print FILE ".Nm $FUNCTIONS{$func}->{'Nm'}";
|
|
|
|
print FILE " ,"
|
|
|
|
if (@funcs);
|
|
|
|
print FILE "\n";
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
2002-06-30 21:30:05 +00:00
|
|
|
print FILE ".Nd Pluggable Authentication Modules Library
|
2002-02-23 01:24:02 +00:00
|
|
|
.Sh LIBRARY
|
|
|
|
.Lb libpam
|
2002-06-30 21:30:05 +00:00
|
|
|
.Sh SYNOPSIS\n";
|
|
|
|
if ($page eq 'pam') {
|
|
|
|
print FILE ".In security/pam_appl.h\n";
|
|
|
|
} else {
|
|
|
|
print FILE ".In security/openpam.h\n";
|
|
|
|
}
|
2002-02-23 01:24:02 +00:00
|
|
|
foreach $func (sort(keys(%FUNCTIONS))) {
|
2002-06-30 21:30:05 +00:00
|
|
|
print FILE ".Ft $FUNCTIONS{$func}->{'Ft'}\n";
|
|
|
|
print FILE ".Fn $FUNCTIONS{$func}->{'Fn'}\n";
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
2002-06-30 21:30:05 +00:00
|
|
|
while (<STDIN>) {
|
|
|
|
if (m/^\.Xr (\S+)\s*(\d)\s*$/) {
|
2005-02-01 10:16:17 +00:00
|
|
|
++$xref{int($2)}->{$1};
|
2003-06-01 12:54:03 +00:00
|
|
|
}
|
2002-06-30 21:30:05 +00:00
|
|
|
print FILE $_;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($page eq 'pam') {
|
|
|
|
print FILE ".Sh RETURN VALUES
|
|
|
|
The following return codes are defined by
|
2005-02-01 10:16:17 +00:00
|
|
|
.In security/pam_constants.h :
|
2002-02-23 01:24:02 +00:00
|
|
|
.Bl -tag -width 18n
|
|
|
|
";
|
2002-06-30 21:30:05 +00:00
|
|
|
foreach (sort(keys(%PAMERR))) {
|
|
|
|
print FILE ".It Bq Er $_\n$PAMERR{$_}.\n";
|
|
|
|
}
|
|
|
|
print FILE ".El\n";
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
2002-06-30 21:30:05 +00:00
|
|
|
print FILE ".Sh SEE ALSO
|
2002-02-23 01:24:02 +00:00
|
|
|
";
|
2005-02-01 10:16:17 +00:00
|
|
|
if ($page eq 'pam') {
|
|
|
|
++$xref{3}->{'openpam'};
|
2002-06-30 21:30:05 +00:00
|
|
|
}
|
2005-02-01 10:16:17 +00:00
|
|
|
foreach $func (keys(%FUNCTIONS)) {
|
|
|
|
++$xref{3}->{$func};
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
2005-02-01 10:16:17 +00:00
|
|
|
print FILE genxref(\%xref);
|
2002-06-30 21:30:05 +00:00
|
|
|
print FILE ".Sh STANDARDS
|
2002-02-23 01:24:02 +00:00
|
|
|
.Rs
|
|
|
|
.%T \"X/Open Single Sign-On Service (XSSO) - Pluggable Authentication Modules\"
|
|
|
|
.%D \"June 1997\"
|
|
|
|
.Re
|
|
|
|
.Sh AUTHORS
|
|
|
|
The OpenPAM library and this manual page were developed for the
|
2003-07-15 07:18:26 +00:00
|
|
|
.Fx
|
|
|
|
Project by ThinkSec AS and Network Associates Laboratories, the
|
2005-06-17 14:37:52 +00:00
|
|
|
Security Research Division of Network Associates, Inc.\\& under
|
2002-06-30 21:30:05 +00:00
|
|
|
DARPA/SPAWAR contract N66001-01-C-8035
|
2002-02-23 01:24:02 +00:00
|
|
|
.Pq Dq CBOSS ,
|
|
|
|
as part of the DARPA CHATS research program.
|
2002-06-30 21:30:05 +00:00
|
|
|
";
|
|
|
|
close(FILE);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub usage() {
|
|
|
|
|
2011-12-18 17:08:40 +00:00
|
|
|
print(STDERR "usage: gendoc [-op] source [...]\n");
|
2002-06-30 21:30:05 +00:00
|
|
|
exit(1);
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MAIN:{
|
2002-06-30 21:30:05 +00:00
|
|
|
my %opts;
|
|
|
|
|
|
|
|
usage()
|
|
|
|
unless (@ARGV && getopts("op", \%opts));
|
2011-12-18 17:08:40 +00:00
|
|
|
setlocale(LC_ALL, "en_US.UTF-8");
|
2002-02-23 01:24:02 +00:00
|
|
|
$TODAY = strftime("%B %e, %Y", localtime(time()));
|
|
|
|
$TODAY =~ s,\s+, ,g;
|
2002-06-30 21:30:05 +00:00
|
|
|
if ($opts{'o'} || $opts{'p'}) {
|
|
|
|
foreach my $fn (@ARGV) {
|
|
|
|
readproto($fn);
|
|
|
|
}
|
|
|
|
gensummary('openpam')
|
|
|
|
if ($opts{'o'});
|
|
|
|
gensummary('pam')
|
|
|
|
if ($opts{'p'});
|
|
|
|
} else {
|
|
|
|
foreach my $fn (@ARGV) {
|
|
|
|
my $func = parse_source($fn);
|
|
|
|
gendoc($func)
|
|
|
|
if (defined($func));
|
|
|
|
}
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|
2002-06-30 21:30:05 +00:00
|
|
|
exit(0);
|
2002-02-23 01:24:02 +00:00
|
|
|
}
|