Mail Archives: djgpp/1995/06/19/12:57:43
Xref: | news-dnh.mv.net comp.os.msdos.djgpp:443
|
Path: | news-dnh.mv.net!mv!news.sprintlink.net!cs.utexas.edu!convex!not-for-mail
|
From: | rosenkra AT convex DOT com (William Rosenkranz)
|
Newsgroups: | comp.os.msdos.djgpp
|
Subject: | man(1) system (part01/01)
|
Date: | 19 Jun 1995 00:11:31 -0500
|
Organization: | Engineering, Convex Computer Corporation, Richardson, Tx USA
|
Lines: | 2659
|
Nntp-Posting-Host: | convex1.convex.com
|
Summary: | unix man, apropos, whatis
|
Keywords: | unix manual man whatis apropos
|
To: | djgpp AT sun DOT soe DOT clarkson DOT edu
|
Dj-Gateway: | from newsgroup comp.os.msdos.djgpp
|
man (part01/01)
this is a man(1) system. it has been run successfully under both unix
and with gcc on the atari ST. untested under dos and djgpp, but it
should port trivially. someone had asked for this, and i had it
laying around. written circa 1991-1992.
i am also posting whatis(1) (in 2 parts) that goes along with this.
see the readme files for instructions.
-bill rosenkranz
rosenkra AT convex DOT com
-------------------------- cut here --------------------------
# This is a shell archive. Remove anything before this line,
# then unpack it by saving it in a file and typing "sh file".
#
# Wrapped by William Rosenkranz <rosenkra AT det_hp> on Mon Jun 19 00:31:44 1995
#
# This archive contains:
# man
#
# Error checking via sum(1) will be performed.
LANG=""; export LANG
PATH=/bin:/usr/bin:$PATH; export PATH
if sum -r </dev/null >/dev/null 2>&1
then
sumopt='-r'
else
sumopt=''
fi
echo mkdir - man
mkdir man
echo x - man/README
cat >man/README <<'@EOF'
man v3.0.2 92/10/11
-------------------
This is a man(1) program, to be used in conjunction with manpager (which
I concurrently posted). It is fairly robust, patterned after BSD unix man.
It requires ul(1) and cat(1), if the "-" and "-ul" options are used,
whatis(1) if the "-f" option is used, apropos(1) for the "-k" option, and
always needs either less (setenv PAGER c:\bin\less.ttp) or manpager
(setenv MANPAGER c:\bin\manpager.ttp). One of these is required, or you
can setenv PAGER to your own prefered pager. Personally, I think less
is superior over pg or more. And this version is really oriented toward
using manpager, for reading nroff v1.10p4 manpages (with font changes).
NOTE: the manpage posted here, "man.1", uses special escape sequences
for changing fonts, so it must be viewed with manpager to get the full
effect. You can also use less or emacs, in a pinch.
What is nice about this man, is that it can deal with files compressed
with compress(1), using the "-z" switch in manpager (less does not support
this directly) without resorting to something like "zcat file | less".
I tested man and manpager with the compress on atari.archive.umich.edu
(terminator) which is v4.3, 16-bit. For more on this, see manpager docs.
Incidently, when I say "man(1)", the "(1)" refers to the section of the
unix programmer's manual (more on this below). Section 1 is user commands.
It is just the way things are refered to in unix-land.
I was going to post apropos(1), whatis(1), and whatisin(1), but they need
more work before I do. No timeframe for this. Sorry.
Anyway, it is pretty simple to use. To set up a manpage directory, do this:
1) mkdir c:\man (this is the root of the manpage tree, the default
in the program, too, but you can use anything you like).
2) cd c:\man
mkdir man0 man1 man2 man3 man4 man5 man6 man7 man8
mkdir manl mano mann. You will place formatted manpages in this
tree (those already run through nroff). Normally, these would be
called "catpages" and placed in .\cat*. man does NOT invoke nroff!
3) copy man.ttp to your bin directory and rehash your shell. man can
also be run from the desktop, but you need an environment if you
want to override the default man directory.
4) setenv MANDIR c:\man (or whatever).
5) setenv MANPAGER c:\bin\manpager.ttp.
6) put some manpages in the c:\man\man* directories as follows (note
that file extensions ARE important and MUST correspond to the
last char of the subdirectory):
subdir filename description
man0 *.0 general information
man1 *.1 commands
man2 *.2? system calls, my convention is
*.2 normal unix system calls
*.2g gemdos
*.2b bios
*.2x xbios
man3 *.3? libraries, normal unix convention is
*.3 general libc
*.3s stdio (fopen, etc)
*.3m math (sin, etc)
*.3c compatibility
*.3x others
man4 *.4 devices, special files (for dev drivers
on unix, i use it for things like
sound.4, dma.4, etc)
man5 *.5 file formats (e.g. arc.5, ar.5, tar.5)
man6 *.6 games
man7 *.7 misc
man8 *.8 system administration (e.g. ramdisk.8)
manl *.l local commands specific to your setup
mano *.o old manpages (replaced with newer)
mann *.n new manpages (for testing prior to
install)
You can start with man's manpage (cp man.1 c:\man\man1).
7) run man (man man) if you copied man.1 to c:\man\man1. You do not
specify the section in the file name. note that
if you have 2 manpages called "xyz" (i.e. xyz.1 and xyz.2), man
will ALWAYS find xyz.1 and never the other. if you want xyz.2,
you should specify the section as in "man 2 xyz".
8) if you want to use the -k and -f options, you will have to build
a whatis(1) database, which is simply the NAME text from manpages,
one per single line. the whatis entry for man is:
man - read online documentation
"whatis" is a file which goes in c:\lib. a sample is included
here. Each time you add a manpage to the system, add the NAME line
to whatis in the format specified (see whatis file). my own file
has about 600 entries!
You don't have to stick with these (BSD) conventions, but it will make life
simpler. At any rate, the program requires the directory stucture and searches
there. See the manpage (or source) for the search order. Try to put only
unix commands in man1 (putting other commands like uniterm) in manl. I have
hundreds of files in man1, and it gets to be a pain after a while.
Note that it is not necessarily to have a manpage for 2 commands or functions
which share the same manpage. You can "source" a manpage from another by
placing a single line in the file with a line such as:
.so man3\fclose.3s
This is the manpage for fflush(3S). It uses the manpage for fclose(3S).
Someday, I may post manpages for sections 2 (bios, xbios, gemdos) and possibly
section 3 (dlibs, gemlib, etc). Don't hold your breath, though :-).
Enjoy...
-Bill Rosenkranz
rosenkra AT convex DOT com
@EOF
set `sum $sumopt <man/README`; if test $1 -ne 8878
then
echo ERROR: man/README checksum is $1 should be 8878
fi
chmod 644 man/README
echo x - man/README.all
cat >man/README.all <<'@EOF'
README (man, whatis, 95/06/18)
------------------------------
someone (stanb AT netcom DOT com) had asked for a man(1) command for reading
manpages. i am posting such a system which includes:
man - read online manpages
apropos - list relevant manpages on a given topic (same as "man -k word")
whatis - give 1-line description of a command (same as "man -f word")
whatisin - list information about what is in sections of the manual
mkwhatis - tool to help build whatis database
note that man does not have its own pager. man calls the pager and
gives it a filename. what man does is find the file (in known places)
and figures out what to run (pager, whatis, apropos, etc.). in this
regard it is really pretty simple minded. the hard part is getting,
organizing, formatting, and maintaining the manpages. for example, i
have hundreds of manpages on my systems. the typical unix system might
have over a thousand manpages.
these programs function more or less the same as the unix (BSD) commands
with some exceptions:
- man does not invoke nroff. your manpages should be preformatted.
- every unix flavor has its own way of doing man so some of the
switches on these commands may not correspond with all unix
(or posix) ways of dealing with this.
- the whatis database itself is different than how unix does it.
but it is similar, and no two unix systems do it the same
anyway.
as it stands, these programs assumes the following directory structure:
c:\usr\man main location of manpages, whatis db, etc
c:\usr\man\man[0-9lno] dirs containing *unformatted* manpages)
c:\usr\man\cat[0-9lno] dirs containing *formatted* manpages)
c:\usr\man\whatis._?_ whatis databases for each section (you
have to build these based on your collection
of manpages)
c:\usr\man\whatisin.___ whatisin database
man will ONLY look in the cat* directories, and does not format the manpages.
you will need nroff for this (cawf, groff, etc). i also have an nroff
that i wrote years ago that is PD. or you can format the manpages on a unix
system and download them to your PC.
everything is configurable, both in the source and via the environment.
for example, the c:\usr\man path can be overridded with:
- setting environment variable MANPATH
- using the commandline switch "-M path" on each command
- changing the executables with a binary editor (i have left
space in the source for such patches)
- changing the source and recompiling
caveats:
- i am providing this free, with no restrictions. it is 100% my
code (does not contain any unix code).
- i will not help in any way to fix anything or add new features.
this system runs fine under both Unix and on my atari ST (with
gcc) so it should port easily, though i have not tried it under
DOS or with djgpp.
- i will not reformat the source to correspond to someone's notion
of style. this is my prefered style, after 10 years of C coding
and 20 years of fortran coding. :-)
- if you don't like it, either a) fix it yourself, or b) don't use
it. it is free, after all.
Install:
- Unpack everything into separate directories (./man, ./whatis).
- The files are all ascii (no binary files). there are no CF-LF
(just LF) so if you need CR-LF, you have to fix this yourself.
- go to each dir (./man and ./whatis) and look at the makefile.
change (at least) the following for your particular setup:
CC how to invoke compiler
LD how to link .o files
MANTOP main location to place manpages (install)
MANDIR specific location for manpage source
CATDIR specific location for formatted manpages
BINDIR location to install executables (man,
whatis, etc)
DEFS in man, you should specify what you have
available (eg, if you have ul(1) and want
the -ul switch to function, define HAS_UL).
you probably want to define MSDOS, __GNUC__,
and possibly undefine unix.
in general, check all the macros in upper case in the
makefile. note that i could have used sed in the makefile
to patch names in the program, but i don't know how good
make is on the PC.
- man invokes the PAGER and other programs via the system(3) call.
on unix (and on my atari), system will search PATH for executables
so you don't need a full path. if this is not the case with djgpp,
then you will have to alter ./man/man.c to build a full path
for executing PAGER, cat, ul, whatis, apropos, etc.
- type:
make all will build the executables
make install will install the execs and manpages
make clobber will clean up after the make all.
you may also consider doing "make dirs" before make install
when building man. this creates the directory structure.
you should do man before whatis.
- if successful, you should be all set. i have provided formatted
manpages (man.1, whatis.1, whatis.5, apropos.1, whatisin.1) from
the manpage sources (man.man, whatis.man, whatis.ma5, etc). these
were piped through "ul -t dumb" to remove any bold/underline
escape sequences. so the formatted manpages are pure ascii. if
you have a PAGER that groks how to handle nroff bold, etc, then
you may want to reformat the manpages.
you should at a minimum build man. it will allow you to read manpages.
if you are up for it, the whatis capability might be desirable. whatis
contains a 1-line description of every manpage you have on your system.
for more information, see the file ./whatis/whatis.5 and whatis.1.
note: i have not included any manpages here other than those for these
programs themselves. don't ask me for a manpage on ls(1).
Environment:
these programs recognize several environment variables:
MANPAGER man if set, this pager is used by man. on my
atari, i have a special pager that resets
fonts, based on what -T does in nroff on
my system. this gives me bold, etc.
PAGER man if MANPAGER is not set, this pager is used.
the default is less.
MANPATH man, location of the manpages, top level. the
whatis default is c:\usr\man. at the moment, MANPATH
apropos is a single path, not a list.
whatisin
for more, grep the sources for "getenv" or read the manpages.
note that man will invoke other programs as follows (using system(3)):
man file PAGER
man -k word apropos
man -f name whatis
you can also invoke apropos and whatis by themselves, of course.
that's about all i have time for. have fun...
-bill rosenkranz
rosenkra AT convex DOT com
@EOF
set `sum $sumopt <man/README.all`; if test $1 -ne 9384
then
echo ERROR: man/README.all checksum is $1 should be 9384
fi
chmod 644 man/README.all
echo x - man/manifest
cat >man/manifest <<'@EOF'
README.all readme for man and whatis (top level)
README readme for man
makefile makefile for man
man.1 formatted manpage for man
man.c man source
man.man unformatted manpage for man (nroff -man format)
manifest this file
@EOF
set `sum $sumopt <man/manifest`; if test $1 -ne 13953
then
echo ERROR: man/manifest checksum is $1 should be 13953
fi
chmod 644 man/manifest
echo x - man/makefile
cat >man/makefile <<'@EOF'
# makefile for man
#
# date: Sat Jan 19 04:08:46 1991
# version: 2.0
#
#
# set this to executable file extension, if necessary (eg, .exe for MSDOS,
# .ttp for atari)
#
EXE =
SECT = 1
TARGET = man$(EXE)
MANPAGE = man.$(SECT)
MANSRC = man.man
HEADERS =
SRCS = $(HEADERS) man.c
OTHERS = readme makefile $(MANSRC)
DISTFILES = $(OTHERS) $(SRCS)
OBJS = man.o
# set CC to the command needed to compile a file (to a .o file).
# set LD to the command needed to link .o files.
#
#CC = gcc -v -z -Wall -Ic:/u/gcc/include
#LD = gcc -v -z -Wall -Lc:/u/gcc/lib -nostdlib c:/u/gcc/lib/crt0.o
CC = cc -Aa
LD = cc -Aa
# defines based on what you have:
# -DHAS_CAT yes, i have cat$(EXE)
# -DHAS_UL yes, i have ul$(EXE)
# -DHAS_WHATIS yes, i have whatis$(EXE)
# -DHAS_APROPOS yes, i have apropos$(EXE)
# -DLONG_SECT use longer list for things like "local"
# -DUSE_INDEX if you need index/rindex instead of strchr/strrchr
# -Datarist for Atari ST
# -DMSDOS for PC
# -Dunix for unix systems (or unix-like)
#
DEFS = -DHAS_CAT -DHAS_UL -DHAS_WHATIS -DHAS_APROPOS -DLONG_SECT
# set ARCH for things like -mshort.
# set OPT potentially to -O.
# set libs to list of libraries needed in link (eg, -lc).
#
ARCH =
OPT =
CFLAGS = $(ARCH) $(OPT) $(DEFS)
LDFLAGS = $(ARCH)
LIBS =
# set BINDIR to the place where you want the executable installed.
# set MANDIR to the place where man's manpage will be installed (the
# directory must exist; consider doing "make dirs" before "make install")
# include a trailing / (or \) for both BINDIR and MANDIR.
#
#SLASH =\\
#BINDIR = c:\usr\bin
#MANTOP = c:\usr\man
#MANDIR = $(MANTOP)$(SLASH)man$(SECT)
#CATDIR = $(MANTOP)$(SLASH)cat$(SECT)
SLASH =/
BINDIR = /mnt/rosenkra/test/bin
MANTOP = /mnt/rosenkra/test/man
MANDIR = $(MANTOP)$(SLASH)man$(SECT)
CATDIR = $(MANTOP)$(SLASH)cat$(SECT)
# install commands
#
INST_BIN = cp -p
INST_MAN = cp -p
RM = rm -f
# directions...
#
directions:
@echo type "make all" to built $(TARGET)
@echo type "make dirs" to create $(MANDIR) directories
@echo type "make install" to build/install $(TARGET)
@echo type "make clean" to remove objects
@echo type "make clobber" to remove objects and $(TARGET)
@echo these should be done in this order
# main target...
#
all: $(TARGET)
$(TARGET): $(OBJS)
$(LD) $(LDFLAGS) -o $(TARGET) $(OBJS) $(LIBS)
# manpage (use built-in .man.cat rule)...
#
manpage: $(MANPAGE)
$(MANPAGE): $(MANSRC)
nroff -man $(MANSRC)|ul -t dumb|cat -s >$(MANPAGE)
# create directories
#
dirs: install_dirs
install_dirs:
mkdir $(MANTOP)
mkdir $(MANTOP)$(SLASH)cat0 $(MANTOP)$(SLASH)man0
mkdir $(MANTOP)$(SLASH)cat1 $(MANTOP)$(SLASH)man1
mkdir $(MANTOP)$(SLASH)cat2 $(MANTOP)$(SLASH)man2
mkdir $(MANTOP)$(SLASH)cat3 $(MANTOP)$(SLASH)man3
mkdir $(MANTOP)$(SLASH)cat4 $(MANTOP)$(SLASH)man4
mkdir $(MANTOP)$(SLASH)cat5 $(MANTOP)$(SLASH)man5
mkdir $(MANTOP)$(SLASH)cat6 $(MANTOP)$(SLASH)man6
mkdir $(MANTOP)$(SLASH)cat7 $(MANTOP)$(SLASH)man7
mkdir $(MANTOP)$(SLASH)cat8 $(MANTOP)$(SLASH)man8
mkdir $(MANTOP)$(SLASH)cat9 $(MANTOP)$(SLASH)man9
touch install_dirs
# to install it...
#
install: install_dirs install_bin install_man
install_bin: $(TARGET)
$(INST_BIN) $(TARGET) $(BINDIR)$(SLASH)$(TARGET)
touch install_bin
install_man: $(MANPAGE)
$(INST_MAN) $(MANSRC) $(MANDIR)$(SLASH)$(MANPAGE)
$(INST_MAN) $(MANPAGE) $(CATDIR)$(SLASH)$(MANPAGE)
touch install_man
# others...
#
clean:
$(RM) core $(OBJS) errs
clobber: clean
$(RM) $(TARGET)
# dependencies...
#
man.o: man.c
@EOF
set `sum $sumopt <man/makefile`; if test $1 -ne 56249
then
echo ERROR: man/makefile checksum is $1 should be 56249
fi
chmod 644 man/makefile
echo x - man/man.1
cat >man/man.1 <<'@EOF'
MAN(1) MAN(1)
NAME
man - read online documentation
SYNOPSIS
man [- [-ul]] [-s sect[subs]] [-M path] [-m path] [-e] [sect[subs]]
name...
man -f file...
man -k keyword...
DESCRIPTION
The man command finds and displays information in the online
documentation set. When the standard output is a terminal, man pipes
its output through less(1) or the name of the command specified with
the MANPAGER or PAGER environment variables.
OPTIONS
The following command line options are recognized:
- Pipes the output through cat(1) rather than less(1) or PAGER.
- -ul
Use ul(1) rather than cat(1).
-M path
Specifies a directory tree in which to search. By default, man
searches in c:\usr\man under MSDOS or on the Atari ST or /usr/man
under Unix, or the directory specified by the MANPATH environment
variable, the standard location for online documentation. The
MANPATH variable should have a single path name at the present
time (eventually, it will support a colon-separated list of
locations). This option assumes path is the root of a normal
directory structure. For compatibility with some Unix man
systems, you can use -P as well as -M.
-m path
Specifies a directory in which to search. This option does not
use a directory tree. Files are searched only in the directory
specified by path. This option is useful for debugging new
manpages. For example, assuming you are working on a manpage
named mycmd.1, you can do this:
man -m . 1 mycmd
-e Find every manpage concerning the given name regardless of
section. It is expected that a normal exit from the pager will
result in an exit status of 0 (in which case the search is
continued in other sections). An abnormal exit should result in
a nonzero exit status and the search is then terminated.
-s sect[subsect]
Specifies a section and subsection similar to the 4.2 BSD UNIX
- 1 - Formatted: June 18, 1995
MAN(1) MAN(1)
Programmer's Manual (as explained below) in which to search. If
no section is specified, man searches through all of the
sections. Example: man -s 3s fopen.
sect As an alternative to -s, you can specify the section (and
optional subsection) as the first argument. Example: man 3s
fopen.
name Searches for documentation files associated with the specified
name.
-f name
Displays a one line synopsis of each online documentation file
whose names match the specified word (equivalent to the whatis(1)
command). For example, man -f chmod will give entries for both
chmod(1) and chmod(2). Note that man invokes whatis(1) in this
case, so it should be found in the path (see ENVIRONMENT).
-k keyword
Displays a one line synopsis of each online documentation file
whose name or description contains the specified keyword
(equivalent to the apropos(1) command). Note that man invokes
apropos(1) in this case, so it should be found in the path (see
ENVIRONMENT).
ENVIRONMENT
The following environment variables are recognized:
MANSECT
This can be used to alter the search order and to customize
subsections. The default is equivalent to:
setenv MANSECT 108234576,tcgesla,,s,mgbxl,msvcxgl,dkvscm,,,
This means the major sections (man1, etc.) are searched in the
order man1, man0, man8, etc. For each section, you can also
provide a list of subsections which must correspond in the
remaining list. In this case man1 subsections are t,c,g,e,s,l,
and a; there are no subsections for man0; etc. The first entry
(the section list) must consist of numbers from 0 to 9 only.
Numbers can be skipped.
The default search order is oriented toward users. To speed up
the search, a casual programmer might perfer:
325481760,gsmcvxl,mgbxl,,dkvscm,s,tcgesla,,,
while a Unix programmer might prefer:
235481760,mgbxl,csmxvlg,,dkvscm,s,tcgesla,,,
- 2 - Formatted: June 18, 1995
MAN(1) MAN(1)
MANPATH,MANDIR
Location of main manual direcotry, where subdirectories are to be
found. Man uses (in order of priority) the path specified with
-M, the MANPATH specified, or the built-in default (c:\usr\man or
/usr/man).
MANPAGER
Name of prefered pager. Use this pager if you have compressed
manpages (see manpager(1)). Man will use first MANPAGER, if
found, then PAGER, if found. If neither are found, it uses
c:\usr\bin\less.
PAGER
Name of secondary pager to replace less(1). Use this if you do
not have manpages generated with font changes (see nroff(1)).
Either MANPAGER or PAGER should be defined. The search order is
MANPAGER then PAGER.
Note that BINDIR and TMPDIR are not currently used.
SECTION NAMES
By convention, the "sections" of the manual are (see whatisin(1) as
well):
Sect Section Name Description
---- ------------- --------------------------------------
0 General overview of features and documentation
1 Commands user commands
2 System Calls low-level system library calls (C)
3 Library Calls standard user calls (C)
4 Special Files special system files and hardware
5 File formats things like arc(1) file formats
6 Games games manual
7 Miscellaneous miscellaneous information
8 Administration system administration commands
In addition, this man recognizes the following section names:
local files specific to local system
new files added since current software release
old files from previous software release
gnu GNU files (gcc typically)
paper misc formal papers
doc misc documentation
The search order, if no section is specified, is:
108234576
In addition to sections, there are a number of subsections possible,
though these files do not reside in a special directory. The
- 3 - Formatted: June 18, 1995
MAN(1) MAN(1)
subsection name is simply appended to the file name. Example:
fopen.3s resides in $MANPATH\man3, the section is 3 (libraries) and
the subsection is s, stdio. Here is the subsection search order for
subsections in each section (the square brackets mean a choice of each
character contained in them, in that order):
man1\*.1[tcgesla] util & text,com,graphics,edit,shell,lang
archival
man0\*.0
man8\*.8[s] util & system
man2\*.2[mgbxl] system & mint,gemdos,bios,xbios,lineA
man3\*.3[msvcxgl] C lib & math,stdio,sysV,compat,extra
gem(aes/vdi),local
man4\*.4[dkvscm] general & disk,keyboard,video,sound,chips
memory
man5\*.5
man7\*.7
man6\*.6
NOTES
To save disk space, you can "source" other files by including the
nroff(1) directive .so in a file as the first line. In this case, the
first line should start with .so followed by at least one space, then
the file to use instead of the actual manpage. For example, the
manpage for feof.3s (in $MANPATH\man3) contains the single line:
.so man3\ferror.3s
which will cause man to display the contents of ferror.3s instead.
Note the relative path (relative to MANPATH, by default c:\usr\man).
In this way, numerous manpages can refer to a single (larger) file.
Do not compress files with source lines as man only checks for this
line if it finds a file which is not compressed. Note that you should
only compress files for viewing with a pager that supports reading
compressed files (e.g., manpager(1)) and must set the MANPAGER
environment variable as well.
You can also specify just a file name for sourcing (without relative
path):
.so ferror.3s
and man will use the relative path of the file containing this line.
FILES
c:\usr\man root of standard manual page directory tree
c:\usr\man\man*\* manual entries
c:\usr\man\whatis table of contents and keyword database
SEE ALSO
apropos(1), cat(1), less(1), manpager(1), nroff(1), what(1),
- 4 - Formatted: June 18, 1995
MAN(1) MAN(1)
whatis(1), whatisin(1), whereis(1), man(7)
AUTHOR
Bill Rosenkranz
rosenkra AT convex DOT com
VERSION
man v3.0.2 92/10/11 rosenkra
- 5 - Formatted: June 18, 1995
@EOF
set `sum $sumopt <man/man.1`; if test $1 -ne 25753
then
echo ERROR: man/man.1 checksum is $1 should be 25753
fi
chmod 644 man/man.1
echo x - man/man.c
cat >man/man.c <<'@EOF'
static char *rcsid = "$Id: man.c,v 1.4 1992/10/12 09:59:02 rosenkra Exp $";
static char *version = "man v3.0.2 92/10/11 rosenkra AT convex DOT com";
static char *myname = "man";
/*
* man - read on-line documentation
*
* This version uses "manpager" or "less" for paging. It does NOT
* call nroff (ie, manpages must be preformatted -- see SUBDIRBASE
* macro below).
*
* search order (based on my manpage nomenclature):
*
* man1/*.1[tcgesla] cmds util & text,com,graphics,
* edit,shell,lang, archival
* man0/*.0 gen
* man8/*.8[s] admin util & system
* man2/*.2[b] syslib system & bios
* man3/*.3[msvcxl] libs C lib & math,stdio,sysV,
* compat,extra,local
* man4/*.4[dkvscm] h/w gen & disk,keyboard,video,
* sound,chips,memory
* man5/*.5 files
* man7/*.7 misc
* man6/*.6 games
*
* these are seached only if given sections "local", "new", "old":
*
* manl/*.l local
* mann/*.n new
* mano/*.o old
*/
/*
* $Log: man.c,v $
* Revision 1.4 1992/10/12 09:59:02 rosenkra
* basically same as last version (which was premature). fix a couple
* of bugs.
*
* Revision 1.3 1992/10/11 16:21:24 rosenkra
* this is v3.0.2. it adds a -P alias for -M, ability to source files
* with either .so manx/file.x or .so file.x; also added was MANSECT
* env var. help (usage) was beefed up. all reading of env now done in
* env_inquiry().
*
* Revision 1.2 1992/07/27 10:03:38 rosenkra
* added ability to review all manpages of given name regardless
* of section (-e). also do env checks before arg parsing.
*
* Revision 1.1 1992/07/27 02:03:58 rosenkra
* Initial revision
*
*/
#ifdef __hpux
/* HPUX for some reason needs this with <sys/stat.h> */
#define _INCLUDE_POSIX_SOURCE 1
#define unix 1
#endif
#include <stdio.h>
#include <ctype.h>
#if defined(unix) || defined(__unix)
#include <sys/types.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#elif defined(__GNUC__) && defined(MSDOS)
#include <types.h>
#include <stat.h>
#include <stdlib.h>
#include <unistd.h>
#else
#include <sys\types.h>
#include <sys\stat.h>
extern char *index ();
extern char *getenv ();
#endif
#ifdef USE_INDEX
#include <strings.h>
#define strchr index
#define strrchr rindex
#endif
/*
* deal with MSDOS
*/
#if defined(unix) || defined(__unix)
#define SLASH_CHR '/'
#define SLASH_STR "/"
#else
#define SLASH_CHR '\\'
#define SLASH_STR "\\"
#endif
/*
* common (default) places. note the patch space...
*/
#if defined(atarist) || defined(MSDOS)
# define MANDIR "c:\\usr\\man\0\0\0<------MANDIR_patch_space------->\0"
# define BINDIR "c:\\usr\\bin\0\0\0<------BINDIR_patch_space------->\0"
# define TMPDIR "c:\\tmp\0\0\0<---------TMPDIR_patch_space--------->\0"
#else
/* assume unix */
# define MANDIR "/usr/man"
# define BINDIR "/usr/bin"
# define TMPDIR "/tmp"
#endif
/*
* default pager (we invoke this with system()):
*/
#if defined(unix) || defined(__unix)
# define MORE "less" /* could use -m -s also */
#elif defined(atarist)
# define MORE "less.ttp\0\0\0<--------MORE_patch_space----------->\0"
#elif defined(MSDOS)
# define MORE "less.exe\0\0\0<--------MORE_patch_space----------->\0"
#endif
/*
* other commands we may run
*/
#ifdef HAS_CAT
# if defined(unix) || defined(__unix)
# define CAT "cat -s" /* for '-' opt, use "cat -s" */
# elif defined(atarist)
# define CAT "cat.ttp\0\0\0<---------CAT_patch_space------------>\0"
# elif defined(MSDOS)
# define CAT "cat.exe\0\0\0<---------CAT_patch_space------------>\0"
# else
# undef HAS_CAT
# endif
#endif
#ifdef HAS_UL
# if defined(unix) || defined(__unix)
# define UL "ul -t dumb" /* for '- -ul' opt (don't use pager) */
# elif defined(atarist)
# define UL "ul.ttp\0\0\0<----------UL_patch_space------------->\0"
# elif defined(MSDOS)
# define UL "ul.exe\0\0\0<----------UL_patch_space------------->\0"
# else
# undef HAS_UL
# endif
#endif
#ifdef HAS_APROPOS
# if defined(unix) || defined(__unix)
# define APROPOS "apropos"
# elif defined(atarist)
# define APROPOS "apropos.ttp\0\0<--------APROPOS_patch_space------->\0"
# elif defined(MSDOS)
# define APROPOS "apropos.exe\0\0<--------APROPOS_patch_space------->\0"
# else
# undef HAS_APROPOS
# endif
#endif
#ifdef HAS_WHATIS
# if defined(unix) || defined(__unix)
# define WHATIS "whatis"
# elif defined(atarist)
# define WHATIS "whatis.ttp\0\0<--------WHATIS_patch_space--------->\0"
# elif defined(MSDOS)
# define WHATIS "whatis.ttp\0\0<--------WHATIS_patch_space--------->\0"
# else
# undef HAS_WHATIS
# endif
#endif
/*
* define sections and subsections. sections are dirs like "man1" and
* "cat1". subsections are used in names of manpages within the dir
* (eg, "foo.1t").
*/
#define ALLSECT "108234576" /* order to look through sections */
#define SUBSEC0 "" /* subsec to try in each section */
#define SUBSEC1 "tcgesla"
#define SUBSEC2 "g"
#define SUBSEC3 "msvcxl"
#define SUBSEC4 "dkvscm"
#define SUBSEC5 ""
#define SUBSEC6 ""
#define SUBSEC7 ""
#define SUBSEC8 "s"
#define SUBSEC9 ""
/*
* deal with compressed files
*/
#if defined(MSDOS)
#define ZSUFFIX "Z" /* for compressed files */
#else
#define ZSUFFIX ".Z" /* for compressed files */
#endif
#define ZSWITCH "-z" /* for MANPAGER to read compressed file */
/*
* files come from MANPATH/cat*. normally, unix man will first
* check for preformatted manpages in /usr/man/cat* dirs. if they
* are not found, nroff is run on /usr/man/man* files. with this
* man, however, we are assuming that manpages have already been
* formatted. by convention, this means that the dirs to search
* should be named 'cat[1-8]' and not 'man[1-8]'. if you don't
* care about this, make SUBDIRBASE "man".
*/
#define SUBDIRBASE "cat"
#ifdef dbg
# undef dbg
#endif
#define dbg(x) if(debugging_G)printf x , fflush(stdout)
/*
* misc macros
*/
#ifdef lastchar
# undef lastchar
#endif
#ifdef plastchar
# undef plastchar
#endif
#define lastchar(s) ((s)[strlen(s)-1]) /* last char in a string */
#define plastchar(s) &((s)[strlen(s)-1]) /* ptr to it */
#define PLEN 256 /* max path length */
/*
* globals:
*/
#ifdef HAS_UL
int ul4cat_G = 0; /* use ul instead of cat */
#endif
int every_G = 0; /* every section */
int debugging_G = 0;
int hasmanpager_G = 0; /* if pager is MANPAGER */
int usermanpath_G = 0; /* if -M */
int nopager_G = 0; /* if no pager of any kind */
int pthpager_G = 0; /* if pager has a full path */
int envmansect_G = 0; /* if MANSECT in env */
char *manpath_G = 0L;
char mpbuf[PLEN];
char *binpath_G = 0L;
char bpbuf[PLEN];
char *tmppath_G = 0L;
char tpbuf[PLEN];
char *pager_G = 0L;
char pgbuf[PLEN];
char cmdbuf_G[512];
int section_G = 0; /* section, ascii char */
int subsec_G = 0; /* subsection, ascii char */
/*
* set up default search lists. these can be overridden with the env:
*
* setenv MANSECT 108234576,tcgesla,,s,mgbxl,msvcxgl,dkvscm,,,
*/
char *allsect = ALLSECT;
char *subsec0 = SUBSEC0;
char *subsec1 = SUBSEC1;
char *subsec2 = SUBSEC2;
char *subsec3 = SUBSEC3;
char *subsec4 = SUBSEC4;
char *subsec5 = SUBSEC5;
char *subsec6 = SUBSEC6;
char *subsec7 = SUBSEC7;
char *subsec8 = SUBSEC8;
char *subsec9 = SUBSEC9;
#ifdef LONG_SECT
char *longsect[] = {"local", "new", "old", "gnu", "paper",
"doc", (char *) 0};
#endif
/*
* functions:
*/
int set_section (char *);
int find_manpage (char *);
int do_it (char *, int);
char *so_line (char *);
void env_inquiry (void);
void usage (int);
/*------------------------------*/
/* main */
/*------------------------------*/
void main (int argc, char *argv[])
{
int retcode;
char **av;
int ac;
/*
* quick scan of args to find debug since we need it first.
* note that argc and argv are not changed...
*/
ac = argc;
av = argv;
for (ac--, av++; ac > 0 && **av == '-'; ac--, av++)
{
if (*(*av+1) == 'd')
{
debugging_G = 1;
break;
}
}
/*
* check environment (cmdline overrules things)
*/
env_inquiry ();
/*
* must have an arg
*/
argc--, argv++;
if (argc < 1)
{
usage (1);
}
/*
* parse cmdline. it overrides defaults and environment...
*/
while (argc > 0 && **argv == '-')
{
switch (*(*argv+1))
{
case 0: /* use cat (just a "-") */
#ifdef HAS_CAT
nopager_G++;
#else
fprintf (stderr, "%s: cat and not supported\n");
usage (1);
#endif
break;
case 'e': /* every section */
every_G++;
break;
case 'P':
case 'M': /* reset path (-M path) */
/* this is manpath/man[1...] */
argc--, argv++;
manpath_G = *argv;
break;
case 'm': /* reset path (-m path) */
/* this is manpath alone */
/* use "-m ." for cwd */
argc--, argv++;
manpath_G = *argv;
usermanpath_G = 1;
break;
case 'u': /* use ul(1) for cat */
#ifdef HAS_UL
ul4cat_G++;
#else
fprintf (stderr, "%s: ul not supported\n");
usage (1);
#endif
break;
case 's': /* specific section */
switch (--argc)
{
case 0: /* Nothing follows -s */
fprintf (stderr,
"%s: must specify a section with -s\n",
myname);
usage (1);
case 1: /* Section, but no man page */
fprintf (stderr,
"%s: must specify an item from section %s\n",
myname, *argv);
usage (1);
default:
/*
* if valid section, set_section sets
* section_G
*/
if (set_section (*++argv) == 0)
{
fprintf (stderr,
"%s: unknown section: %s\n",
myname, *argv);
usage (1);
}
break;
}
break;
case 'f': /* whatis */
case 'w':
#ifdef HAS_WHATIS
#if defined(__GNUC__) || defined(unix) || defined(__unix)
strcpy (cmdbuf_G, WHATIS);
#else
strcpy (cmdbuf_G, binpath_G);
if (lastchar (cmdbuf_G) != SLASH_CHR)
strcat (cmdbuf_G, SLASH_STR);
strcat (cmdbuf_G, WHATIS);
#endif /*__GNUC__*/
argc--, argv++;
while (argc)
{
strcat (cmdbuf_G, " ");
strcat (cmdbuf_G, *argv);
argc--, argv++;
}
dbg(("whatis, system (%s)\n",cmdbuf_G));
retcode = system (cmdbuf_G);
exit (retcode);
#else /*!HAS_WHATIS*/
fprintf (stderr, "%s: whatis not supported\n");
usage (1);
#endif /*HAS_WHATIS*/
break;
case 'k': /* apropos */
case 'a':
#ifdef HAS_APROPOS
strcpy (cmdbuf_G, APROPOS);
#if defined(__GNUC__) || defined(unix) || defined(__unix)
strcpy (cmdbuf_G, APROPOS);
#else
strcpy (cmdbuf_G, binpath_G);
if (lastchar (cmdbuf_G) != SLASH_CHR)
strcat (cmdbuf_G, SLASH_STR);
strcat (cmdbuf_G, APROPOS);
#endif /*__GNUC__*/
argc--, argv++;
while (argc)
{
strcat (cmdbuf_G, " ");
strcat (cmdbuf_G, *argv);
argc--, argv++;
}
dbg(("apropos, system (%s)\n",cmdbuf_G));
retcode = system (cmdbuf_G);
exit (retcode);
#else /*!HAS_APROPOS*/
fprintf (stderr, "%s: apropos not supported\n");
usage (1);
#endif /*HAS_APROPOS*/
break;
case 'h': /* help */
usage (0);
break;
case 'd': /* turn on debugging */
debugging_G = 1;
break;
case 'v': /* version */
printf ("%s\n", version);
exit (0);
break;
case '-':
switch (*(*argv+2))
{
case 'v': /* --version */
printf ("%s\n", version);
exit (0);
break;
case 'h': /* --help */
usage (0);
break;
}
break;
}
argc--, argv++;
}
/*
* If there are multiple words, no -s seen above, and the first
* word is a valid section name, read the subsequent man pages
* from that section. set_section will set section_G if *argv is a
* bona fide section.
*
* this also should make sure arg is a digit and only <sect>[<sub>]
* but will fail on actual manpages named something like "1.1".
* hopefully these are non-existent.
*/
if ((argc > 1) && (section_G == 0)
&& (set_section (*argv) || (isdigit(**argv) && (strlen(*argv) < 3))))
{
argc--;
argv++;
}
/*
* move to correct directory. no need to move if it is cwd...
*/
if (strcmp (manpath_G, ".") && (chdir (manpath_G) < 0))
{
fprintf (stderr, "%s: can't chdir to %s.\n",
myname, manpath_G);
exit (1);
}
/*
* check if stdout not a tty an no pager.
*/
#if 0
if (nopager_G == 0 && !isatty (1))
nopager_G++;
#endif
dbg(("nopager_G = %d, isatty(1) = %d\n", nopager_G, (int) isatty(1)));
/*
* do it for all remaining tokens
*/
while (argc-- > 0)
{
retcode = find_manpage (*argv++);
if (retcode)
{
fprintf (stderr, "%s: failed!!!\n", myname);
exit (retcode);
}
}
exit (0);
}
/*------------------------------*/
/* set_section */
/*------------------------------*/
int set_section (char *cptr)
{
/*
* routine to set section_G and subsec_G from a given section name,
* returning the letter of the section used or a zero if an unfamiliar
* keyword is used.
*/
int sec;
int ret;
int i;
if (strlen (cptr) > 2)
{
/*
* section is 3 or more char
*/
#ifdef LONG_SECT
for (i = 0; longsect[i]; i++)
{
if (!strcmp (longsect[i], cptr))
return (section_G = (int) (*longsect[i]));
}
#else
if (!strcmp (cptr, "local"))
return ((int) (section_G = 'l'));
if (!strcmp (cptr, "new"))
return ((int) (section_G = 'n'));
if (!strcmp (cptr, "old"))
return ((int) (section_G = 'o'));
#endif
/*
* If we reach here, none of the special names match,
* and we'll return a zero (error) on the other side of
* the else.
*/
section_G = '\0';
subsec_G = '\0';
ret = 0;
}
else
{
/*
* section is 2 or less char. subsec could be 0
*/
section_G = (int) cptr[0];
subsec_G = (int) cptr[1];
ret = section_G;
/*
* crude check for legality: section must be 0-8, subsect
* must be a letter
*/
sec = section_G - '0';
/*!!!*/ if (sec < 0 || sec > 9)
ret = 0;
if (subsec_G)
{
if (!isalpha (subsec_G))
ret = 0;
}
}
/*
* ret is 0 if invalid section, else section number
*/
return (ret);
}
struct stat sbuf;
/*------------------------------*/
/* find_manpage */
/*------------------------------*/
int find_manpage (char *name)
{
/*
* find the manpage file
*/
char c_sectn;
char work[PLEN];
int stflag;
int last;
char seclist[30];
char sublist[30];
char *psec;
char *psub;
char *ps;
char *cp;
int iscompressed = 0;
int found = 0;
/*
* set up template (we removed any trailing '/' already)
*
* if usermanpath is set, we read from that dir and don't assume
* any "normal" dir structure (-m option). otherwise we do normal
* (or -M option).
*/
if (usermanpath_G)
sprintf (work, "%s%s%s.x", manpath_G, SLASH_STR, name);
else
sprintf (work, "%s%s%sx%s%s.x",
manpath_G, SLASH_STR, SUBDIRBASE, SLASH_STR, name);
/*
* index of last char in template. this should be section.
* work[last+1] is subsection, and work[last+2] will be ZSUFFIX
* if compressed and we are using MANPAGER.
*/
last = strlen (work) - 1;
dbg(("template, work = |%s|, last = %d\n", work, last));
/*
* set up list of sections to search. either use section_G or
* allsect
*/
if (section_G)
{
seclist[0] = (char) section_G;
seclist[1] = '\0';
}
else
{
strcpy (seclist, allsect);
}
psec = seclist;
if (subsec_G)
{
sublist[0] = (char) subsec_G;
sublist[1] = '\0';
psub = sublist;
}
else
psub = (char *) NULL;
dbg(("psec = |%s|, psub = |%s|\n", psec, (psub ? psub : "NULL")));
/*
* find manx string in template. ps points to the '/'
*/
if (!usermanpath_G)
{
char tmpstringx[PLEN];
tmpstringx[0] = SLASH_CHR;
strcpy(&tmpstringx[1], SUBDIRBASE);
for (ps = work; *ps; ps++)
{
if (!strncmp (ps, tmpstringx, 4)
&& *(ps+5) == SLASH_CHR)
break;
}
}
/*
* loop thru all sections. could be only specified section
*/
for (c_sectn = *psec; c_sectn; c_sectn = *++psec)
{
/*
* fill in template: man_/xxx._\0\0 (no sub yet)
*/
if (!usermanpath_G)
ps[4] = c_sectn;
work[last] = c_sectn;
work[last + 1] = 0;
work[last + 2] = 0;
work[last + 3] = 0;
/*
* if a subsection was specified, skip past this...
*/
if (subsec_G)
goto SEARCH;
/*
* does this file exist?
*/
dbg(("try |%s|...\n", work));
if ((stflag = stat (work, &sbuf)) >= 0)
{
/*
* yes...exit this for loop
*/
if (every_G)
{
/*
* PAGER should support ^C exit. it should
* exit with status 1 in that case (and
* do_it() would also return 1). it is not
* an error, however, to exit this way but
* we return (with 0) so next item will
* be searched.
*/
found++;
if (do_it (work, iscompressed))
return (0);
goto NEXT_LOOP;
}
else
goto FOUND_ONE;
}
/*
* no...try looking for compressed version
*/
if (hasmanpager_G)
{
dbg(("not found, try compressed file...\n"));
#if 0
work[last + 1] = ZSUFFIX;
work[last + 2] = '\0';
work[last + 3] = '\0';
#endif
strcpy(&work[last+1], ZSUFFIX);
dbg(("try |%s|...\n", work));
if ((stflag = stat (work, &sbuf)) >= 0)
{
/*
* yes...
*/
iscompressed = 1;
if (every_G)
{
found++;
if (do_it (work, iscompressed))
return (0);
goto NEXT_LOOP;
}
else
goto FOUND_ONE;
}
work[last + 1] = '\0'; /* reset! */
dbg(("not found...\n"));
}
SEARCH: ;
/*
* still not found...look for subsections in manx
*
* set up subsect string
*/
if (subsec_G)
cp = psub;
else
{
/*
* take the default subsections for a section...
*/
switch (work[last])
{
case '0': cp = subsec0; break;
case '1': cp = subsec1; break;
case '2': cp = subsec2; break;
case '3': cp = subsec3; break;
case '4': cp = subsec4; break;
case '5': cp = subsec5; break;
case '6': cp = subsec6; break;
case '7': cp = subsec7; break;
case '8': cp = subsec8; break;
case '9': cp = subsec9; break;
default: cp = ""; break;
}
}
dbg(("not found, try subsect, cp = |%s|\n", cp));
/*
* cycle thru subsections
*/
while (*cp)
{
/*
* do we find one now?
*/
work[last + 1] = *cp++;
work[last + 2] = '\0';
dbg(("try |%s|...\n", work));
if ((stflag = stat (work, &sbuf)) >= 0)
{
/*
* yes...
*/
if (every_G)
{
found++;
if (do_it (work, iscompressed))
return (0);
goto NEXT_LOOP;
}
else
goto FOUND_ONE;
}
/*
* still no...try compressed...
*/
if (hasmanpager_G)
{
dbg(("not found, try compressed file...\n"));
#if 0
work[last + 2] = ZSUFFIX;
work[last + 3] = '\0';
#endif
strcpy(&work[last+2], ZSUFFIX);
dbg(("try |%s|...\n", work));
if ((stflag = stat (work, &sbuf)) >= 0)
{
/*
* yes...
*/
iscompressed = 1;
if (every_G)
{
found++;
if (do_it (work, iscompressed))
return (0);
goto NEXT_LOOP;
}
else
goto FOUND_ONE;
}
work[last + 2] = '\0'; /* reset! */
dbg(("not found...\n"));
}
/* we give up. try next subsection, if any... */
}
/* we give up. try next section, if any... */
NEXT_LOOP: ;
}
if (every_G)
{
if (found)
return (0);
else
printf ("%s: no manual entry for %s. try man local %s\n",
myname, name, name);
}
else
{
if (section_G == 0)
printf ("%s: no manual entry for %s. try man local %s\n",
myname, name, name);
else
printf ("%s: no entry for %s in section %c%c of the manual.\n",
myname, name,
(char) section_G,
(subsec_G ? (char) subsec_G : ' '));
}
return (1);
FOUND_ONE: ;
dbg(("Found: %s(%x), iscompressed = %d\n",work,stflag,iscompressed));
return ((int) do_it (work, iscompressed));
}
/*------------------------------*/
/* do_it */
/*------------------------------*/
int do_it (char *cp, int iscompressed)
{
char cmdpath[256];
char cmd[256];
char *so;
/*
* open the file, look for ".so file" as first line. use that file
* instead. only one level of this nesting is possible (the
* included file can't include, too). do this only if file not
* compressed!
*/
dbg(("enter do_it, cp = |%s|...\n", cp));
if (!iscompressed)
{
so = so_line (cp);
if (so)
{
cp = so;
dbg(("so_line found something, new cp = |%s|...\n",cp));
}
}
/*
* set up command path (for paging)
*/
#if defined(__GNUC__) || defined(unix) || defined(__unix)
cmdpath[0] = '\0';
#else
strcpy (cmdpath, binpath_G);
if (lastchar (cmdpath) != SLASH_CHR)
strcat (cmdpath, SLASH_STR);
#endif
if (!nopager_G)
{
/*
* use pager. if compressed, add correct switch (note:
* iscompressed set only if hasmanpager is true)
*/
if (iscompressed)
{
sprintf (cmd, "%s%s %s %s",
(pthpager_G ? "" : cmdpath),
pager_G,
ZSWITCH,
cp);
}
else
{
sprintf (cmd, "%s%s %s",
(pthpager_G ? "" : cmdpath),
pager_G,
cp);
}
}
#ifdef HAS_UL
else if (ul4cat_G)
{
/*
* use ul rather than cat if no pager
*/
sprintf (cmd, "%s%s %s", cmdpath, UL, cp);
}
#endif
#ifdef HAS_CAT
else
{
/*
* use cat if no pager
*/
sprintf (cmd, "%s%s %s", cmdpath, CAT, cp);
}
#endif
/*
* invoke the pager (or cat or ul)
*/
dbg(("system (%s)\n", cmd));
return ((int) system (cmd));
}
/*------------------------------*/
/* so_line */
/*------------------------------*/
char *so_line (char *fname)
{
/*
* read first line from found file, looking for ".so " followed by
* alternate sourced file. if found, returns ptr to the new name.
* otherwise, null.
*
* the file name should include the path of the subsection, e.g.
* the file for degas.5 (in man5) contains:
*
* .so man5/picture.5
*
* the rest of the file (man5/degas.5) is ignored.
*
* new: now can also handle
*
* .so picture.5
*
* fname comes in with man5/degas.5 so use the leading path.
*/
static char buf[256];
char tbuf[256];
char *so;
FILE *stream;
char *ps;
so = (char *) 0;
if ((stream = fopen (fname, "r")) == (FILE *) 0)
{
printf ("%s: could not open %s to look for .so\n",
myname, fname);
return (so);
}
/*
* grab first line, delete newline
*/
fgets (buf, 255, stream);
fclose (stream);
so = buf;
while (*++so)
;
*--so = '\0';
/*
* is it .so?
*/
so = buf;
if (!strncmp (so, ".so ", 4))
{
so += 4;
while (*so == ' ' || *so == '\t')
so++;
dbg(("so_line read |%s|\n", so));
if ((ps = strrchr (so, '\\')) == (char *) 0
&& (ps = strrchr (so, '/')) == (char *) 0)
{
/*
* the .so line does not contain manx/name.x so
* use path from fname.
*/
strcpy (tbuf, so);
strcpy (so, fname);
if (ps = strrchr (so, SLASH_CHR))
{
ps++;
strcpy (ps, tbuf);
}
else if (ps = strrchr (so, '/'))
{
ps++;
strcpy (ps, tbuf);
}
}
dbg(("so_line returning |%s|\n", so));
return (so);
}
else
{
dbg(("so_line returning NULL\n"));
return ((char *) 0);
}
}
/*------------------------------*/
/* env_inquiry */
/*------------------------------*/
void env_inquiry (void)
{
/*
* look up things in environment:
*
* MANPAGER,PAGER
* TEMP,TMPDIR
* MANPATH,MANDIR
* BINDIR
* MANSECT
*
* setenv MANSECT 108234576
* setenv MANSECT 108234576,tcgesla,,s,mgbxl,msvcxgl,dkvscm,,,
*/
char *ps;
char *pl;
char *pd;
static char sectbuf[256];
/*
* alternate pager from environment. if env has variable MANPAGER
* defined, use it as pager (this allows dealing with bold and
* italics in manpage, for example). if not, look for PAGER. if
* neither is set, use default as defined by macro MORE here.
*/
if (pager_G == (char *) NULL)
{
if ((ps = getenv ("MANPAGER")) != (char *) NULL)
{
strcpy (pgbuf, ps);
pager_G = pgbuf;
hasmanpager_G = 1;
}
else if ((ps = getenv ("PAGER")) != (char *) NULL)
{
strcpy (pgbuf, ps);
pager_G = pgbuf;
}
else
pager_G = MORE;
}
if (pager_G && (strchr (pager_G, ':') || strchr (pager_G, SLASH_CHR)))
pthpager_G = 1;
dbg(("pager_G = |%s|, pthpager_G = %d\n",(pager_G ? pager_G : "NULL"), pthpager_G));
/*
* place to put temp files, if needed (not needed yet...)
*/
if (tmppath_G == (char *) NULL)
{
if ((ps = getenv ("TEMP")) != (char *) NULL)
{
strcpy (tpbuf, ps);
tmppath_G = tpbuf;
}
else if ((ps = getenv ("TMPDIR")) != (char *) NULL)
{
strcpy (tpbuf, ps);
tmppath_G = tpbuf;
}
else
tmppath_G = TMPDIR;
}
if (tmppath_G && lastchar (tmppath_G) == SLASH_CHR)/* rem trail '/' */
lastchar (tmppath_G) = '\0';
dbg(("tmppath_G = |%s|\n", tmppath_G ? tmppath_G : "NULL"));
/*
* where man pages are (e.g. /usr/man)
*/
if (manpath_G == (char *) NULL)
{
if ((ps = getenv ("MANPATH")) != (char *) NULL)
{
strcpy (mpbuf, ps);
manpath_G = mpbuf;
}
else if ((ps = getenv ("MANDIR")) != (char *) NULL)
{
strcpy (mpbuf, ps);
manpath_G = mpbuf;
}
else
manpath_G = MANDIR;
}
if (manpath_G && lastchar (manpath_G) == SLASH_CHR)/* rem trail '/' */
lastchar (manpath_G) = '\0';
dbg(("manpath_G = |%s|\n", manpath_G ? manpath_G : "NULL"));
/*
* place to find pagers
*/
if (binpath_G == (char *) NULL)
{
if ((ps = getenv ("BINDIR")) != (char *) NULL)
{
strcpy (bpbuf, ps);
binpath_G = bpbuf;
}
else
binpath_G = BINDIR;
}
if (binpath_G && lastchar (binpath_G) == SLASH_CHR)/* rem trail '/' */
lastchar (binpath_G) = '\0';
dbg(("binpath_G = |%s|\n", binpath_G ? binpath_G : "NULL"));
/*
* finally, deal with MANSECT. this should be last section
* since this returns.
*/
if ((ps = getenv ("MANSECT")) != (char *) NULL)
{
strcpy (sectbuf, ps);
dbg(("MANSECT = |%s|\n", sectbuf));
envmansect_G = 1;
pl = ps = sectbuf;
while (*ps && *ps != ',') /* find first , */
ps++;
if (*ps == '\0') /* if no more list, return */
return;
*ps++ = '\0'; /* terminate allsect */
allsect = pl;
dbg(("allsect = |%s|\n", allsect));
while (*pl)
{
dbg(("for sect %c, ", (int) *pl));
pd = ps;
switch (*pl)
{
case '0': subsec0 = ps; goto find_comma;
case '1': subsec1 = ps; goto find_comma;
case '2': subsec2 = ps; goto find_comma;
case '3': subsec3 = ps; goto find_comma;
case '4': subsec4 = ps; goto find_comma;
case '5': subsec5 = ps; goto find_comma;
case '6': subsec6 = ps; goto find_comma;
case '7': subsec7 = ps; goto find_comma;
case '8': subsec8 = ps; goto find_comma;
case '9': subsec9 = ps;
find_comma: ;
if (*ps == ',')
{
*ps++ = '\0';
if (*ps == '\0')
{
dbg(("subsec = empty\n"));
return;
}
dbg(("subsec = empty\n"));
break;
}
while (*ps && *ps != ',')
ps++;
if (*ps == '\0')
{
dbg(("subsec = empty\n"));
return;
}
*ps++ = '\0';
dbg(("subsec = |%s|\n", pd));
break;
default:
break;
}
pl++;
}
}
}
#define FP fprintf
/*------------------------------*/
/* usage */
/*------------------------------*/
void usage (int excode)
{
char *ps;
int i;
#if 0
FP (stderr, "Usage: %s [-s sec[sub]] [-M dir | -m dir] [- [-ul]] [-k key] [-f file] [sec[sub]] name\n", myname);
#endif
FP (stderr, "\nUsage: %s [options] [sec[sub]] name\n", myname);
FP (stderr, "Opts: -s sec[sub] section/subsect (e.g. 2g)\n");
FP (stderr, " -M dir specify search dir (for man tree)\n");
FP (stderr, " -m dir specify search dir (dir, no tree, eg man -m .)\n");
#ifdef HAS_CAT
FP (stderr, " - use cat rather than more or less\n");
# ifdef HAS_UL
FP (stderr, " - -ul use ul rather than cat\n");
# endif
#endif
#ifdef HAS_APROPOS
FP (stderr, " -k key same as \"apropos keyword\"\n");
#endif
#ifdef HAS_WHATIS
FP (stderr, " -f file same as \"whatis file\"\n");
#endif
FP (stderr, "Args: sec section, alternate for -s\n");
FP (stderr, " sub subsection, alternate for -s\n");
FP (stderr, " name desired entry (command, etc)\n");
FP (stderr, "Search order:\n");
if (envmansect_G)
{
for (ps = allsect; *ps; ps++)
{
switch (*ps)
{
case '0':
FP (stderr, " %s0%s*.0", SUBDIRBASE, SLASH_STR);
if (*subsec0)
FP (stderr, "[%s]%s", subsec0, (strlen(subsec0) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "general\n");
break;
case '1':
FP (stderr, " %s1%s*.1", SUBDIRBASE, SLASH_STR);
if (*subsec1)
FP (stderr, "[%s]%s", subsec1, (strlen(subsec1) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "commands\n");
break;
case '2':
FP (stderr, " %s2%s*.2", SUBDIRBASE, SLASH_STR);
if (*subsec2)
FP (stderr, "[%s]%s", subsec2, (strlen(subsec2) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "system calls\n");
break;
case '3':
FP (stderr, " %s3%s*.3", SUBDIRBASE, SLASH_STR);
if (*subsec3)
FP (stderr, "[%s]%s", subsec3, (strlen(subsec3) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "library calls\n");
break;
case '4':
FP (stderr, " %s4%s*.4", SUBDIRBASE, SLASH_STR);
if (*subsec4)
FP (stderr, "[%s]%s", subsec4, (strlen(subsec4) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "hardware\n");
break;
case '5':
FP (stderr, " %s5%s*.5", SUBDIRBASE, SLASH_STR);
if (*subsec5)
FP (stderr, "[%s]%s", subsec5, (strlen(subsec5) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "special files\n");
break;
case '6':
FP (stderr, " %s6%s*.6", SUBDIRBASE, SLASH_STR);
if (*subsec6)
FP (stderr, "[%s]%s", subsec6, (strlen(subsec6) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "games\n");
break;
case '7':
FP (stderr, " %s7%s*.7", SUBDIRBASE, SLASH_STR);
if (*subsec7)
FP (stderr, "[%s]%s", subsec7, (strlen(subsec7) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "misc\n");
break;
case '8':
FP (stderr, " %s8%s*.8", SUBDIRBASE, SLASH_STR);
if (*subsec8)
FP (stderr, "[%s]%s", subsec8, (strlen(subsec8) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "admin commands\n");
break;
case '9':
FP (stderr, " %s9%s*.9", SUBDIRBASE, SLASH_STR);
if (*subsec9)
FP (stderr, "[%s]%s", subsec9, (strlen(subsec9) < 7) ? "\t " : " ");
else
FP (stderr, "\t\t ");
FP (stderr, "other\n");
break;
}
}
}
else
{
FP (stderr, " %s1%s*.1[tcgesla] cmds (util & text,com,graph,edit,shell,lang,archive)\n", SUBDIRBASE, SLASH_STR);
FP (stderr, " %s0%s*.0 general\n", SUBDIRBASE, SLASH_STR);
FP (stderr, " %s8%s*.8[s] admin (util & system)\n", SUBDIRBASE, SLASH_STR);
FP (stderr, " %s2%s*.2[b] syslib (unix & bios)\n", SUBDIRBASE,SLASH_STR);
FP (stderr, " %s3%s*.3[msvcxl] libs (C lib & math,stdio,sysV,compat,misc,local)\n", SUBDIRBASE, SLASH_STR);
FP (stderr, " %s4%s*.4[dkvscm] h/w (gen & disk,keyboard,video,sound,chips,memory)\n", SUBDIRBASE, SLASH_STR);
FP (stderr, " %s5%s*.5 special files\n", SUBDIRBASE,SLASH_STR);
FP (stderr, " %s7%s*.7 misc\n", SUBDIRBASE, SLASH_STR);
FP (stderr, " %s6%s*.6 games\n", SUBDIRBASE, SLASH_STR);
}
#ifdef LONG_SECT
FP (stderr, "Other sections: ");
for (i = 0; i < 10; i++)
{
if (longsect[i] == (char *) NULL)
break;
FP (stderr, "%s", longsect[i]);
if (longsect[i+1])
FP (stderr, ",");
}
FP (stderr, "\n");
#else
FP (stderr, "Other sections: local,new,old\n");
#endif
FP (stderr, "Environment: MANPAGER,PAGER,MANPATH,MANSECT,BINDIR,TMPDIR\n");
exit (excode);
}
@EOF
set `sum $sumopt <man/man.c`; if test $1 -ne 9208
then
echo ERROR: man/man.c checksum is $1 should be 9208
fi
chmod 644 man/man.c
echo x - man/man.man
sed 's/^@//' >man/man.man <<'@EOF'
@.\" man v3.0.2 92/10/11 rosenkra
@.TH MAN 1
@.SH NAME
man \- read online documentation
@.SH SYNOPSIS
@.\" man [- [-ul]] [-s sect[subs]] [-M path] [-m path] [-e] [sect[subs]] name ...
@.B man
@.RB [ \-
@.RB [ \-ul ]]
@.RB [ \-s
@.IR sect [ subs ]]
@.RB [ \-M
@.IR path ]
@.RB [ \-m
@.IR path ]
@.RB [ \-e ]
@.RI [ sect [ subs ]]
@.IR name ...
@.br
@.\" man -f file
@.B man
@.B \-f
@.IR file ...
@.br
@.\" man -k keyword...
@.B man
@.B \-k
@.IR keyword ...
@.SH DESCRIPTION
The
@.B man
command finds and displays information in the online documentation set.
When the standard output is a terminal,
@.B man
pipes its output through
@.IR less (1)
or the name of the command specified with the MANPAGER or PAGER environment
variables.
@.SH OPTIONS
The following command line options are recognized:
@.IP \fB\-\fR
Pipes the output through
@.IR cat (1)
rather than
@.IR less (1)
or PAGER.
@.IP "\fB\- \-ul\fR"
Use
@.IR ul (1)
rather than
@.IR cat (1).
@.IP "\fB\-M\fR \fIpath\fR"
Specifies a directory tree in which to search.
By default,
@.B man
searches in c:\\usr\\man under MSDOS or on the Atari ST or
/usr/man under Unix,
or the directory specified by the
MANPATH environment variable, the standard location for online
documentation.
The MANPATH variable should have a single path name at the
present time (eventually, it will support a colon-separated
list of locations).
This option assumes
@.I path
is the root of a \*(lqnormal\*(rq directory structure.
For compatibility with some Unix man systems, you can use
@.B \-P
as well as
@.BR \-M .
@.IP "\fB\-m\fR \fIpath\fR"
Specifies a directory in which to search.
This option does not use a directory tree.
Files are searched only in the directory specified by
@.IR path .
This option is useful for debugging new manpages.
For example, assuming you are working on a manpage
named \*(lqmycmd.1\*(rq, you can do this:
@.sp
@.nf
man -m . 1 mycmd
@.fi
@.IP \fB\-e\fR
Find every manpage concerning the given
@.I name
regardless of section.
It is expected that a \*(lqnormal\*(rq exit from the pager
will result in an exit status of 0 (in which case the search is continued
in other sections).
An \*(lqabnormal\*(rq exit should result in a nonzero exit status
and the search is then terminated.
@.IP "\fB\-s\fR \fIsect\fR[\fIsubsect\fR]"
Specifies a
@.I section
and
@.I subsection
similar to the 4.2 BSD UNIX Programmer's Manual (as explained below)
in which to search.
If no
@.I section
is specified,
@.B man
searches through all of the sections.
Example: \*(lqman -s 3s fopen\*(rq.
@.IP \fIsect\fR
As an alternative to
@.BR \-s ,
you can specify the section (and optional subsection) as the first
argument.
Example: \*(lqman 3s fopen\*(rq.
@.IP \fIname\fR
Searches for documentation files associated with the specified
@.IR name .
@.IP "\fB\-f\fR \fIname\fR"
Displays a one line synopsis of each online documentation file whose
names match the specified word (equivalent to the
@.IR whatis (1)
command).
For example, \*(lqman -f chmod\*(rq will give entries for both
@.IR chmod (1)
and
@.IR chmod (2).
Note that
@.B man
invokes
@.IR whatis (1)
in this case, so it should be found in the path (see ENVIRONMENT).
@.IP "\fB\-k\fR \fIkeyword\fR"
Displays a one line synopsis of each online documentation file whose name
or description contains the specified
@.I keyword
(equivalent to the
@.IR apropos (1)
command).
Note that
@.B man
invokes
@.IR apropos (1)
in this case, so it should be found in the path (see ENVIRONMENT).
@.SH ENVIRONMENT
The following environment variables are recognized:
@.IP MANSECT
This can be used to alter the search order and to customize subsections.
The default is equivalent to:
@.sp
@.nf
setenv MANSECT 108234576,tcgesla,,s,mgbxl,msvcxgl,dkvscm,,,
@.fi
@.sp
This means the major sections (man1, etc.) are searched in the order
man1, man0, man8, etc.
For each section, you can also provide a list of subsections which must
correspond in the remaining list.
In this case man1 subsections are t,c,g,e,s,l, and a;
there are no subsections for man0;
etc.
The first entry (the section list) must consist of numbers from 0 to 9
@.IR only .
Numbers can be skipped.
@.sp
The default search order is oriented toward users.
To speed up the search, a casual programmer might perfer:
@.sp
@.nf
325481760,gsmcvxl,mgbxl,,dkvscm,s,tcgesla,,,
@.fi
@.sp
while a Unix programmer might prefer:
@.sp
@.nf
235481760,mgbxl,csmxvlg,,dkvscm,s,tcgesla,,,
@.fi
@.IP MANPATH,MANDIR
Location of main manual direcotry, where subdirectories are to be found.
@.B Man
uses (in order of priority) the path specified with
@.BR \-M ,
the MANPATH specified, or the built-in default (c:\\usr\\man or /usr/man).
@.IP MANPAGER
Name of prefered pager.
Use this pager if you have compressed manpages (see
@.IR manpager (1)).
@.B Man
will use first MANPAGER, if found, then PAGER, if found.
If neither are found, it uses c:\\usr\\bin\\less.
@.IP PAGER
Name of secondary pager to replace
@.IR less (1).
Use this if you do not have manpages generated with font changes (see
@.IR nroff (1)).
Either MANPAGER or PAGER should be defined.
The search order is MANPAGER then PAGER.
@.\" .IP BINDIR
@.\" Alternate location to find all necessary executables.
@.\" The default is c:\\bin.
@.\" You need
@.\" .BR less (1),
@.\" .BR cat (1),
@.\" and
@.\" .BR ul (1)
@.\" at a minimum.
@.\" .IP TMPDIR
@.\" Place to put temporary files, if necessary.
@.\" Default is g:\\tmp.
@.\" Make it a ramdisk partition, if you have one.
@.\" So far this is not used.
@.PP
Note that BINDIR and TMPDIR are not currently used.
@.SH "SECTION NAMES"
By convention, the "sections" of the manual are (see
@.IR whatisin (1)
as well):
@.sp
@.nf
Sect Section Name Description
---- ------------- --------------------------------------
0 General overview of features and documentation
1 Commands user commands
2 System Calls low-level system library calls (C)
3 Library Calls standard user calls (C)
4 Special Files special system files and hardware
5 File formats things like arc(1) file formats
6 Games games manual
7 Miscellaneous miscellaneous information
8 Administration system administration commands
@.fi
@.sp
In addition, this
@.B man
recognizes the following section names:
@.sp
@.nf
local files specific to local system
new files added since current software release
old files from previous software release
gnu GNU files (gcc typically)
paper misc formal papers
doc misc documentation
@.fi
@.PP
The search order, if no section is specified, is:
@.sp
@.nf
108234576
@.fi
@.PP
In addition to sections, there are a number of subsections possible,
though these files do not reside in a special directory.
The subsection name is simply appended to the file name.
Example:
\*(lqfopen.3s\*(rq resides in $MANPATH\\man3, the section is \*(lq3\*(rq
(libraries) and the subsection is \*(lqs\*(rq, stdio.
Here is the subsection search order for subsections in each section
(the square brackets mean a choice of each character contained in them,
in that order):
@.sp
@.nf
man1\\*.1[tcgesla] util & text,com,graphics,edit,shell,lang
archival
man0\\*.0
man8\\*.8[s] util & system
man2\\*.2[mgbxl] system & mint,gemdos,bios,xbios,lineA
man3\\*.3[msvcxgl] C lib & math,stdio,sysV,compat,extra
gem(aes/vdi),local
man4\\*.4[dkvscm] general & disk,keyboard,video,sound,chips
memory
man5\\*.5
man7\\*.7
man6\\*.6
@.fi
@.SH NOTES
To save disk space, you can "source" other files by including the
@.BR nroff (1)
directive \*(lq.so\*(rq in a file as the first line.
In this case, the first line should start with .so followed by at least
one space, then the file to use instead of the actual manpage.
For example, the manpage for feof.3s (in $MANPATH\\man3) contains the single
line:
@.sp
@.nf
\&.so man3\\ferror.3s
@.fi
@.sp
which will cause
@.B man
to display the contents of \*(lqferror.3s\*(rq instead.
Note the relative path (relative to MANPATH, by default c:\\usr\\man).
In this way, numerous manpages can refer to a single (larger) file.
Do not compress files with source lines as
@.B man
only checks for this line if it finds a file which is not compressed.
Note that you should only compress files for viewing with a pager
that supports reading compressed files (e.g.,
@.BR manpager (1))
and must set the MANPAGER environment variable as well.
@.PP
You can also specify just a file name for sourcing (without relative path):
@.sp
@.nf
\&.so ferror.3s
@.fi
@.sp
and
@.B man
will use the relative path of the file containing this line.
@.SH FILES
@.nf
c:\\usr\\man root of standard manual page directory tree
c:\\usr\\man\\man*\\* manual entries
c:\\usr\\man\\whatis table of contents and keyword database
@.fi
@.SH "SEE ALSO"
apropos(1), cat(1), less(1), manpager(1), nroff(1), what(1), whatis(1), whatisin(1), whereis(1), man(7)
@.SH AUTHOR
Bill Rosenkranz
@.br
rosenkra AT convex DOT com
@.SH VERSION
man v3.0.2 92/10/11 rosenkra
@EOF
set `sum $sumopt <man/man.man`; if test $1 -ne 53179
then
echo ERROR: man/man.man checksum is $1 should be 53179
fi
chmod 644 man/man.man
chmod 755 man
exit 0
--
Bill Rosenkranz |UUCP: {uunet,texsun}!convex!rosenkra
Convex Computer Corp. |ARPA: rosenkra AT convex DOT com
- Raw text -