axh@philabs.philips.com (Abid Hussain) (02/26/91)
Hi everyone, Over the past few days I have received too many requests for the vmsbackup version 4.0 program. I thought the best solution is to post it on the net, as the program itself isn't that big and would also save my time. I got this program from David Kinney, and would thank him again for the help. - Abid. #--------------------------------CUT HERE------------------------------------- #!/bin/sh # # This is a shell archive. Save this into a file, edit it and delete all # lines through the line "CUT HERE" (above). Then give this file to sh by # executing the command "sh file". The files will be extracted into the # current directory owned by you with default permissions. # # The files contained herein are: # # -rw-rw-r-- 1 dkinney 299 May 22 1990 Makefile # -rw-r--r-- 1 dkinney 605 May 22 1990 Makefile.orig # -rw-rw-r-- 1 dkinney 573 May 22 1990 README # -rw-rw-r-- 1 dkinney 2048 May 22 1990 getopt.1 # -rw-rw-r-- 1 dkinney 3561 May 22 1990 getopt.c # -rw-rw-r-- 1 dkinney 8100 May 22 1990 match.c # -rw-rw-r-- 1 dkinney 3528 May 22 1990 vmsbackup.1 # -rw-rw-r-- 1 dkinney 24995 May 22 1990 vmsbackup.c # -rw-rw-r-- 1 dkinney 4304 Jun 19 1990 vmsbackup.doc # echo 'x - Makefile' if test -f Makefile; then echo 'shar: not overwriting Makefile'; else sed 's/^X//' << '________This_Is_The_END________' > Makefile X# X# XREMOTE= # -DREMOTE use remote tape XSWAP= # -DSWAP swap bytes XCFLAGS= $(SWAP) $(REMOTE) XLFLAGS= XLIBS= # remote magtape library XCC = cc X# Xvmsbackup: vmsbackup.o match.o X $(CC) $(LFLAGS) -o vmsbackup vmsbackup.o match.o $(LIBS) ________This_Is_The_END________ if test `wc -l < Makefile` -ne 11; then echo 'shar: Makefile was damaged during transit (should have been 11 bytes)' fi fi ; : end of overwriting check echo 'x - Makefile.orig' if test -f Makefile.orig; then echo 'shar: not overwriting Makefile.orig'; else sed 's/^X//' << '________This_Is_The_END________' > Makefile.orig X# X# XREMOTE=-DREMOTE # -DREMOTE use remote tape XSWAP= # -DSWAP swap bytes XCFLAGS= $(SWAP) $(REMOTE) XLFLAGS= XLIBS= -lrmt # remote magtape library XOWNER=tar # user for remote tape access XMODE=4755 XBINDIR=/usr/local/bin XMANSEC=l XMANDIR=/usr/man/man$(MANSEC) X X# Xvmsbackup: vmsbackup.o getopt.o match.o X cc $(LFLAGS) -o vmsbackup vmsbackup.o match.o getopt.o $(LIBS) Xinstall: X install -m $(MODE) -o $(OWNER) -s vmsbackup $(BINDIR) X cp vmsbackup.1 $(MANDIR)/vmsbackup.$(MANSEC) Xclean: X rm -f vmsbackup *.o core Xshar: X shar -a README vmsbackup.1 Makefile vmsbackup.c match.c \ X > vmsbackup.shar ________This_Is_The_END________ if test `wc -l < Makefile.orig` -ne 24; then echo 'shar: Makefile.orig was damaged during transit (should have been 24 bytes)' fi fi ; : end of overwriting check echo 'x - README' if test -f README; then echo 'shar: not overwriting README'; else sed 's/^X//' << '________This_Is_The_END________' > README XThis progam reads a VMS backuptape. X XThe tape program is orginally written by John Douglas Carey and Xthe pattern matching routine by some unknown on the net. X XThe remote tape option use the rmtlib from mod.sources. X XA good way to archive remotetape access for users with only Xa local account is to create a "netwide" user tar and let Xthe remote tape programs do suid to user tar. X XThe program is tested on vax and sun. X X XSven-Ove Westberg XLulea University of Technology XS-951 87 Lulea, Sweden XUUCP: sow@luthcad.UUCP XUUCP: {decvax,philabs,seismo}!mcvax!enea!luthcad!sow ________This_Is_The_END________ if test `wc -l < README` -ne 19; then echo 'shar: README was damaged during transit (should have been 19 bytes)' fi fi ; : end of overwriting check echo 'x - getopt.1' if test -f getopt.1; then echo 'shar: not overwriting getopt.1'; else sed 's/^X//' << '________This_Is_The_END________' > getopt.1 X.TH GETOPT 1 LOCAL X.SH NAME Xgetopt \- format flags for shell scripts X.SH SYNOPSIS X.B getopt Xflag_spec argument ... X.SH DESCRIPTION X.I Getopt Xis a program intended to be called by scripts to ``canonicalize'' their Xarguments before processing them, just as the X.IR getopt (3) Xroutine does for C programs. X(This need for scripts is usually most noticeable in the way X.IR lint (1) Xhandles the X.B \-n Xflag.) X.PP XThe following two examples provide the initial parsing for a script Xwhich takes two flags, X.B \-a Xand X.BR \-b , Xthe second of which takes an argument. X.RS X.ta +4n +4n +4n +4n X.nf X# For /bin/csh... Xset argv = (`getopt "ab:" $*`) Xif ( $status ) then X echo "Read the documentation and try again." >/dev/tty X exit 1 Xendif Xset Aflag=0 Xset Name=NONE Xwhile "$1" != "--" X switch ("$1") X case '-a': X set Aflag=1 X breaksw X case '-b': X shift X set Name=$1 X breaksw X endsw X shift Xend Xshift Xecho Aflag=$Aflag / Name=$Name / Remaining args are $* X X# For /bin/sh... Xset -- `getopt "d:s" $@` Xif test $? != 0 ; then X echo "Read the documentation and try again." X exit 1 Xfi XAflag=0 XName=NONE Xfor f Xdo X case "$f" in X -a) Aflag=1 X ;; X -b) shift X Name=$2 X ;; X --) break X ;; X esac X shift Xdone Xshift Xecho Aflag=$Aflag / Name=$Name / Remaining args are $* X.fi X.RE X.SH DIAGNOSTICS XThe program burps the standard X.IR getopt (3) Xdiagnostics to standard error, and exits with a non-zero status if an Xerror occurs. XIt is arguable wrong that the diagnostics imply that the program Xis named ``getopt'' rather than the real name of the script. XIt is undeniably AT&T\-compatible to do this, however. X.SH "SEE ALSO" Xcsh(1), sh(1), getopt(3) X.SH AUTHOR X.nf XRich $alz XMirror Systems X(mirror!rs, rs@mirror.TMC.COM) ________This_Is_The_END________ if test `wc -l < getopt.1` -ne 90; then echo 'shar: getopt.1 was damaged during transit (should have been 90 bytes)' fi fi ; : end of overwriting check echo 'x - getopt.c' if test -f getopt.c; then echo 'shar: not overwriting getopt.c'; else sed 's/^X//' << '________This_Is_The_END________' > getopt.c X/* X** GETOPT PROGRAM AND LIBRARY ROUTINE X** X** I wrote main() and AT&T wrote getopt() and we both put our efforts into X** the public domain via mod.sources. X** Rich $alz X** Mirror Systems X** (mirror!rs, rs@mirror.TMC.COM) X** August 10, 1986 X*/ X X#include <stdio.h> X X X#ifndef INDEX X#define INDEX index X#endif X X Xextern char *INDEX(); Xextern int optind; Xextern char *optarg; X X Xmain(ac, av) X register int ac; X register char *av[]; X{ X register char *flags; X register int c; X X /* Check usage. */ X if (ac < 2) { X fprintf(stderr, "usage: %s flag-specification arg-list\n", av[0]); X exit(2); X } X X /* Play games; remember the flags (first argument), then splice X them out so it looks like a "standard" command vector. */ X flags = av[1]; X av[1] = av[0]; X av++; X ac--; X X /* Print flags. */ X while ((c = getopt(ac, av, flags)) != EOF) { X if (c == '?') X exit(1); X /* We assume that shells collapse multiple spaces in `` expansion. */ X printf("-%c %s ", c, INDEX(flags, c)[1] == ':' ? optarg : ""); X } X X /* End of flags; print rest of options. */ X printf("-- "); X for (av += optind; *av; av++) X printf("%s ", *av); X exit(0); X} X X X X X X X X X X/* X** This is the public-domain AT&T getopt(3) code. I added the X** #ifndef stuff because I include <stdio.h> for the program; X** getopt, per se, doesn't need it. I also added the INDEX/index X** hack (the original used strchr, of course). And, note that X** technically the casts in the write(2) calls shouldn't be there. X*/ X X#ifndef NULL X#define NULL 0 X#endif X#ifndef EOF X#define EOF (-1) X#endif X#ifndef INDEX X#define INDEX index X#endif X X X#define ERR(s, c) if(opterr){\ X extern int strlen(), write();\ X char errbuf[2];\ X errbuf[0] = c; errbuf[1] = '\n';\ X (void) write(2, argv[0], (unsigned)strlen(argv[0]));\ X (void) write(2, s, (unsigned)strlen(s));\ X (void) write(2, errbuf, 2);} X Xextern int strcmp(); Xextern char *INDEX(); X Xint opterr = 1; Xint optind = 1; Xint optopt; Xchar *optarg; X Xint Xgetopt(argc, argv, opts) Xint argc; Xchar **argv, *opts; X{ X static int sp = 1; X register int c; X register char *cp; X X if(sp == 1) X if(optind >= argc || X argv[optind][0] != '-' || argv[optind][1] == '\0') X return(EOF); X else if(strcmp(argv[optind], "--") == NULL) { X optind++; X return(EOF); X } X optopt = c = argv[optind][sp]; X if(c == ':' || (cp=INDEX(opts, c)) == NULL) { X ERR(": illegal option -- ", c); X if(argv[optind][++sp] == '\0') { X optind++; X sp = 1; X } X return('?'); X } X if(*++cp == ':') { X if(argv[optind][sp+1] != '\0') X optarg = &argv[optind++][sp+1]; X else if(++optind >= argc) { X ERR(": option requires an argument -- ", c); X sp = 1; X return('?'); X } else X optarg = argv[optind++]; X sp = 1; X } else { X if(argv[optind][++sp] == '\0') { X sp = 1; X optind++; X } X optarg = NULL; X } X return(c); X} ________This_Is_The_END________ if test `wc -l < getopt.c` -ne 147; then echo 'shar: getopt.c was damaged during transit (should have been 147 bytes)' fi fi ; : end of overwriting check echo 'x - match.c' if test -f match.c; then echo 'shar: not overwriting match.c'; else sed 's/^X//' << '________This_Is_The_END________' > match.c X#include <stdio.h> X#include <sys/types.h> X X#define ASTERISK '*' /* The '*' metacharacter */ X#define QUESTION '?' /* The '?' metacharacter */ X#define LEFT_BRACKET '[' /* The '[' metacharacter */ X#define RIGHT_BRACKET ']' /* The ']' metacharacter */ X X#define IS_OCTAL(ch) (ch >= '0' && ch <= '7') X Xtypedef int BOOLEAN; X#define VOID void X#define TRUE 1 X#define FALSE 0 X#define EOS '\000' X Xstatic BOOLEAN do_list (); Xstatic char nextch (); Xstatic VOID list_parse (); X X X X X X X X X X/* X * FUNCTION X * X * match test string for wildcard match X * X * SYNOPSIS X * X * BOOLEAN match (string, pattern) X * register char *string; X * register char *pattern; X * X * DESCRIPTION X * X * Test string for match using pattern. The pattern may X * contain the normal shell metacharacters for pattern X * matching. The '*' character matches any string, X * including the null string. The '?' character matches X * any single character. A list of characters enclosed X * in '[' and ']' matches any character in the list. X * If the first character following the beginning '[' X * is a '!' then any character not in the list is matched. X * X */ X X X X X X X X X X/* X * PSEUDO CODE X * X * Begin match X * Switch on type of pattern character X * Case ASTERISK: X * Attempt to match asterisk X * Break X * Case QUESTION MARK: X * Attempt to match question mark X * Break X * Case EOS: X * Match is result of EOS on string test X * Break X * Case default: X * If explicit match then X * Match is result of submatch X * Else X * Match is FALSE X * End if X * Break X * End switch X * Return result of match test X * End match X * X */ X XBOOLEAN match (string, pattern) Xregister char *string; Xregister char *pattern; X{ X register BOOLEAN ismatch; X X ismatch = FALSE; X switch (*pattern) { X case ASTERISK: X pattern++; X do { X ismatch = match (string, pattern); X } while (!ismatch && *string++ != EOS); X break; X case QUESTION: X if (*string != EOS) { X ismatch = match (++string, ++pattern); X } X break; X case EOS: X if (*string == EOS) { X ismatch = TRUE; X } X break; X case LEFT_BRACKET: X if (*string != EOS) { X ismatch = do_list (string, pattern); X } X break; X default: X if (*string++ == *pattern++) { X ismatch = match (string, pattern); X } else { X ismatch = FALSE; X } X break; X } X return (ismatch); X} X X X X X X X X X X/* X * FUNCTION X * X * do_list process a list and following substring X * X * SYNOPSIS X * X * static BOOLEAN do_list (string, pattern) X * register char *string; X * register char *pattern; X * X * DESCRIPTION X * X * Called when a list is found in the pattern. Returns X * TRUE if the current character matches the list and X * the remaining substring matches the remaining pattern. X * X * Returns FALSE if either the current character fails to X * match the list or the list matches but the remaining X * substring and subpattern's don't. X * X * RESTRICTIONS X * X * The mechanism used to match characters in an inclusive X * pair (I.E. [a-d]) may not be portable to machines X * in which the native character set is not ASCII. X * X * The rules implemented here are: X * X * (1) The backslash character may be X * used to quote any special character. X * I.E. "\]" and "\-" anywhere in list, X * or "\!" at start of list. X * X * (2) The sequence \nnn becomes the character X * given by nnn (in octal). X * X * (3) Any non-escaped ']' marks the end of list. X * X * (4) A list beginning with the special character X * '!' matches any character NOT in list. X * The '!' character is only special if it X * is the first character in the list. X * X */ X X X X X X X X X X/* X * PSEUDO CODE X * X * Begin do_list X * Default result is no match X * Skip over the opening left bracket X * If the next pattern character is a '!' then X * List match gives FALSE X * Skip over the '!' character X * Else X * List match gives TRUE X * End if X * While not at closing bracket or EOS X * Get lower and upper bounds X * If character in bounds then X * Result is same as sense flag. X * Skip over rest of list X * End if X * End while X * If match found then X * If not at end of pattern then X * Call match with rest of pattern X * End if X * End if X * Return match result X * End do_list X * X */ X Xstatic BOOLEAN do_list (string, pattern) Xregister char *string; Xchar *pattern; X{ X register BOOLEAN ismatch; X register BOOLEAN if_found; X register BOOLEAN if_not_found; X auto char lower; X auto char upper; X X pattern++; X if (*pattern == '!') { X if_found = FALSE; X if_not_found = TRUE; X pattern++; X } else { X if_found = TRUE; X if_not_found = FALSE; X } X ismatch = if_not_found; X while (*pattern != ']' && *pattern != EOS) { X list_parse (&pattern, &lower, &upper); X if (*string >= lower && *string <= upper) { X ismatch = if_found; X while (*pattern != ']' && *pattern != EOS) {pattern++;} X } X } X if (*pattern++ != ']') { X fprintf (stderr, "warning - character class error\n"); X } else { X if (ismatch) { X ismatch = match (++string, pattern); X } X } X return (ismatch); X} X X X X X X X X X X/* X * FUNCTION X * X * list_parse parse part of list into lower and upper bounds X * X * SYNOPSIS X * X * static VOID list_parse (patp, lowp, highp) X * char **patp; X * char *lowp; X * char *highp; X * X * DESCRIPTION X * X * Given pointer to a pattern pointer (patp), pointer to X * a place to store lower bound (lowp), and pointer to a X * place to store upper bound (highp), parses part of X * the list, updating the pattern pointer in the process. X * X * For list characters which are not part of a range, X * the lower and upper bounds are set to that character. X * X */ X Xstatic VOID list_parse (patp, lowp, highp) Xchar **patp; Xchar *lowp; Xchar *highp; X{ X *lowp = nextch (patp); X if (**patp == '-') { X (*patp)++; X *highp = nextch (patp); X } else { X *highp = *lowp; X } X} X X X X X X X X X X/* X * FUNCTION X * X * nextch determine next character in a pattern X * X * SYNOPSIS X * X * static char nextch (patp) X * char **patp; X * X * DESCRIPTION X * X * Given pointer to a pointer to a pattern, uses the pattern X * pointer to determine the next character in the pattern, X * subject to translation of backslash-char and backslash-octal X * sequences. X * X * The character pointer is updated to point at the next pattern X * character to be processed. X * X */ X Xstatic char nextch (patp) Xchar **patp; X{ X register char ch; X register char chsum; X register int count; X X ch = *(*patp)++; X if (ch == '\\') { X ch = *(*patp)++; X if (IS_OCTAL (ch)) { X chsum = 0; X for (count = 0; count < 3 && IS_OCTAL (ch); count++) { X chsum *= 8; X chsum += ch - '0'; X ch = *(*patp)++; X } X (*patp)--; X ch = chsum; X } X } X return (ch); X} ________This_Is_The_END________ if test `wc -l < match.c` -ne 354; then echo 'shar: match.c was damaged during transit (should have been 354 bytes)' fi fi ; : end of overwriting check echo 'x - vmsbackup.1' if test -f vmsbackup.1; then echo 'shar: not overwriting vmsbackup.1'; else sed 's/^X//' << '________This_Is_The_END________' > vmsbackup.1 X.TH VMSBACKUP 1 X.SH NAME Xvmsbackup \- read a VMS backup tape X.SH SYNOPSIS X.B vmsbackup X.B \-{tx}[bcdevw][s setnumber][f tapefile] X[ name ... ] X.SH DESCRIPTION X.I vmsbackup Xreads a VMS generated backup tape, converting the files Xto Unix format and writing the files to disc. XThe default operation of the program is to go through an entire Xtape, extracting every file and writing it to disc. XThis may be modified by the following options. X.TP 8 X.B b blocksize XSet block size of saveset file. Default is 32256 bytes per record. X.TP 8 X.B c XUse complete filenames, including the version number. XA colon and the octal version number will be appended to all filenames. XA colon, rather than a semicolon, is used since the Unix Shell Xuses the semicolon as the line separator. XUsing a colon prevents the user from having to escape the semicolon Xwhen referencing the filename. XThis option is useful only when multiple versions of the same file Xare on a single tape or when a file of the same name already Xexists in the destination directory. XThe default is to ignore version numbers. X.TP 8 X.B d Xuse the directory structure from VMS, the default value is off. X.TP 8 X.B e XProcess all filename extensions. XSince this program is mainly intended to move source code and possibly Xdata from a DEC system to a Unix system, the default is to ignore Xall files whose filename extension specifies system dependent data. XThe file types which will be ignored, unless the X.B e Xoption is specified, are X.IP "" 10 Xexe VMS executable file X.br Xlib VMS object library file X.br Xobj RSX object file X.br Xodl RSX overlay description file X.br Xolb RSX object library file X.br Xpmd RSX post mortem dump file X.br Xstb RSX task symbol table file X.br Xsys RSX bootable system file X.br Xtsk RSX executable task file X.PP X.TP 8 X.B f XUse the next argument in the command line as the tape device to Xbe used, rather than the default. X.sp XIf vmsbackup is compiled with the remote tape option Xand the file name has the form X.IR system [. user ]:/dev/??? X.I vmsbackup Xwill use the tape drive /dev/??? on the remote system X.IR system , Xvia X.IR rsh (1), Xand X.IR rmt (8). XThe optional X.I user Xportion of the pathname specifies the login name to use on the Xremote system. XIf it is not supplied, the current user's login name will be used. XIn all the cases, the user must have the appropriate Xpermissions on the remote machine, in order to use this facility. XThe default is X.I /dev/rmt8 X(drive 0, raw mode, 1600 bpi). XThis must be a raw mode tape device. X.TP 8 X.B s saveset XProcess only the given saveset number. X.TP 8 X.B t XProduce a table of contents (a directory listing) on the standard output Xof the files on tape. X.TP 8 X.B v XVerbose output. XNormally X.I vmsbackup Xdoes its work silently. XThe verbose option will cause the filenames of the files being read from Xtape to disk to be output on the standard output. X.TP 8 X.B w X.I vmsbackup Xprints the action to be taken followed by file name, then Xwait for user confirmation. If a word beginning with `y' Xis given, the action is done. Any other input means don't do it. X.TP 8 X.B x Xextract the named files from the tape. X.TP 8 XThe optional X.I name Xargument specifies one or more filenames to be Xsearched for specifically on the tape and only those files are to be processed. XThe name may contain the usal sh(1) meta-characters *?![] \nnn. X.SH FILES X/dev/rmt\fIx\fP X.SH SEE ALSO Xrmtops(3) X.SH BUGS XThe filename match uses the complete VMS file names. X X.SH AUTHOR XJohn Douglas Carey X.br XSven-Ove Westberg ________This_Is_The_END________ if test `wc -l < vmsbackup.1` -ne 127; then echo 'shar: vmsbackup.1 was damaged during transit (should have been 127 bytes)' fi fi ; : end of overwriting check echo 'x - vmsbackup.c' if test -f vmsbackup.c; then echo 'shar: not overwriting vmsbackup.c'; else sed 's/^X//' << '________This_Is_The_END________' > vmsbackup.c X/* X * X * Title: X * Backup X * X * Decription: X * Program to read VMS backup tape X * X * Author: X * John Douglas CAREY. X * Sven-Ove Westberg (version 3.0) X * X * Net-addess: X * john%monu1.oz@seismo.ARPA X * luthcad!sow@enea.UUCP X * X * History: X * Version 1.0 - September 1984 X * Can only read variable length records X * Version 1.1 X * Cleaned up the program from the original hack X * Can now read stream files X * Version 1.2 X * Now convert filename from VMS to UNIX X * and creates sub-directories X * Version 1.3 X * Works on the Pyramid if SWAP is defined X * Version 1.4 X * Reads files spanning multiple tape blocks X * Version 1.5 X * Always reset reclen = 0 on file open X * Now output fixed length records X * X * Version 2.0 - July 1985 X * VMS Version 4.0 causes a rethink !! X * Now use mtio operations instead of opening and closing file X * Blocksize now grabed from the label X * X * Version 2.1 - September 1985 X * Handle variable length records of zero length. X * X * Version 2.2 - July 1986 X * Handle FORTRAN records of zero length. X * Inserted exit(0) at end of program. X * Distributed program in aus.sources X * X * Version 2.3 - August 1986 X * Handle FORTRAN records with record length fields X * at the end of a block X * Put debug output to a file. X * Distributed program in net.sources X * X * Version 3.0 - December 1986 X * Handle multiple saveset X * Remote tape X * Interactive mode X * File name selection with meta-characters X * Convert ; to : in VMS filenames X * Flag for usage of VMS directory structure X * Flag for "useless" files eg. *.exe X * Flag for use VMS version in file names X * Flag for verbose mode X * Flag to list the contents of the tape X * Distributed to mod.sources X * X * Version 4.0 - November 1988 by Timothy Stark (11TSTARK@GALLUA) X * Handle disk saveset X * Add -b option to set block size for disk saveset X * X * Installation: X * X * Computer Centre X * Monash University X * Wellington Road X * Clayton X * Victoria 3168 X * AUSTRALIA X * X */ X#include <stdio.h> X#include <ctype.h> X X#include <sys/ioctl.h> X#include <sys/types.h> X#ifdef REMOTE X#include <local/rmt.h> X#include <sys/stat.h> X#endif X#include <sys/mtio.h> X#include <sys/file.h> X X#ifdef pyr X#define SWAP X#endif pyr X X#ifdef sun X#define SWAP X#endif X Xstruct bbh { X short bbh_dol_w_size; X short bbh_dol_w_opsys; X short bbh_dol_w_subsys; X short bbh_dol_w_applic; X long bbh_dol_l_number; X char bbh_dol_t_spare_1[20]; X short bbh_dol_w_struclev; X short bbh_dol_w_volnum; X long bbh_dol_l_crc; X long bbh_dol_l_blocksize; X long bbh_dol_l_flags; X char bbh_dol_t_ssname[32]; X short bbh_dol_w_fid[3]; X short bbh_dol_w_did[3]; X char bbh_dol_t_filename[128]; X char bbh_dol_b_rtype; X char bbh_dol_b_rattrib; X short bbh_dol_w_rsize; X char bbh_dol_b_bktsize; X char bbh_dol_b_vfcsize; X short bbh_dol_w_maxrec; X long bbh_dol_l_filesize; X char bbh_dol_t_spare_2[22]; X short bbh_dol_w_checksum; X} *block_header; X Xstruct brh { X short brh_dol_w_rsize; X short brh_dol_w_rtype; X long brh_dol_l_flags; X long brh_dol_l_address; X long brh_dol_l_spare; X} *record_header; X X/* define record types */ X X#define brh_dol_k_null 0 X#define brh_dol_k_summary 1 X#define brh_dol_k_volume 2 X#define brh_dol_k_file 3 X#define brh_dol_k_vbn 4 X#define brh_dol_k_physvol 5 X#define brh_dol_k_lbn 6 X#define brh_dol_k_fid 7 X Xstruct bsa { X short bsa_dol_w_size; X short bsa_dol_w_type; X char bsa_dol_t_text[1]; X} *data_item; X X#ifdef STREAM Xchar *def_tapefile = "/dev/rts8"; X#else Xchar *def_tapefile = "/dev/rmt8"; X#endif Xchar *tapefile; X Xchar filename[128]; Xint filesize; X Xchar recfmt; /* record format */ X X#define FAB_dol_C_UDF 0 /* undefined */ X#define FAB_dol_C_FIX 1 /* fixed-length record */ X#define FAB_dol_C_VAR 2 /* variable-length record */ X#define FAB_dol_C_VFC 3 /* variable-length with fixed-le Xngth control record */ X#define FAB_dol_C_STM 4 /* RMS-11 stream record (valid o Xnly for sequential org X ) */ X#define FAB_dol_C_STMLF 5 /* stream record delimited by LF X (sequential org onl X y) */ X#define FAB_dol_C_STMCR 6 /* stream record delimited by CR X (sequential org onl X y) */ X#define FAB_dol_C_MAXRFM 6 /* maximum rfm supported X */ X Xchar recatt; /* record attributes */ X X#define FAB_dol_V_FTN 0 /* FORTRAN carriage control char Xacter */ X#define FAB_dol_V_CR 1 /* line feed - record -carriage Xreturn */ X#define FAB_dol_V_PRN 2 /* print-file carriage control * X/ X#define FAB_dol_V_BLK 3 /* records don't cross block bou Xndaries */ X X#define FANO 20 X X#ifdef pyr Xstatic struct bsa *file_table[FANO]; X#else Xstruct bsa *file_table[FANO]; X#endif X XFILE *f = NULL; Xint file_count; Xshort reclen; Xshort fix; Xshort recsize; Xint vfcsize; X X#ifdef NEWD XFILE *lf; X#endif NEWD X Xint fd; /* tape file descriptor */ Xint cflag, dflag, eflag, fflag, sflag, tflag, vflag, wflag, xflag; Xint setnr; Xchar **gargv; Xint goptind, gargc; X X#define LABEL_SIZE 80 Xchar label[LABEL_SIZE]; X Xchar *block; Xint blocksize; X Xstruct mtop op; X XFILE * Xopenfile(fn) Xchar *fn; X{ X char ufn[256]; X char ans[80]; X char *p, *q, s, *ext; X int procf; X X procf = 1; X /* copy fn to ufn and convert to lower case */ X p = fn; X q = ufn; X while (*p) { X if (isupper(*p)) X *q = *p - 'A' + 'a'; X else X *q = *p; X p++; X q++; X } X *q = '\0'; X X /* convert the VMS to UNIX and make the directory path */ X p = ufn; X q = ++p; X while (*q) { X if (*q == '.' || *q == ']') { X s = *q; X *q = '\0'; X if(procf && dflag) mkdir(p, 0777); X *q = '/'; X if (s == ']') X break; X } X *q++; X } X *q++; X if(!dflag) p=q; X /* strip off the version number */ X while (*q && *q != ';') { X if( *q == '.') ext = q; X q++; X } X if (cflag) { X *q = ':'; X } X else { X *q = '\0'; X } X if(!eflag && procf) procf = typecmp(++ext); X if(procf && wflag) { X printf("extract %s [ny]",filename); X fflush(stdout); X gets(ans); X if(*ans != 'y') procf = NULL; X } X if(procf) X /* open the file for writing */ X return(fopen(p, "w")); X else X return(NULL); X} X Xtypecmp(str) /* Compare the filename type in str with our list X of file type to be ignored. Return 0 if the X file is to be ignored, return 1 if the X file is not in our list and should not be ignored. */ Xregister char *str; X{ X static char *type[] = { X "exe", /* vms executable image */ X "lib", /* vms object library */ X "obj", /* rsx object file */ X "odl", /* rsx overlay description file */ X "olb", /* rsx object library */ X "pmd", /* rsx post mortem dump */ X "stb", /* rsx symbol table */ X "sys", /* rsx bootable system image */ X "tsk", /* rsx executable image */ X "dir", X "upd", X "tlo", X "tlb", X "" /* null string terminates list */ X }; X register int i; X X i = -1; X while (*type[++i]) X if (strncmp(str, type[i],3) == 0) X return(0); /* found a match, file to be ignored */ X return(1); /* no match found */ X} X Xprocess_file(buffer) Xchar *buffer; X{ X int i, n; X char *p, *q; X short dsize, nblk, lnch; X X int c; X short *s; X X int procf; X X s = (short *) buffer; X X /* check the header word */ X if (*s != 257) { X printf("Snark: invalid data header\n"); X exit(1); X } X X c = 2; X for (i = 0; i < FANO; i++) { X file_table[i] = (struct bsa *) &buffer[c]; X#ifndef SWAP X dsize = file_table[i]->bsa_dol_w_size; X#else X swap(&file_table[i]->bsa_dol_w_size, &dsize, sizeof(short)); X#endif X c += dsize + 4; X } X X /* extract file name */ X#ifndef SWAP X dsize = file_table[0]->bsa_dol_w_size; X#else X swap(&file_table[0]->bsa_dol_w_size, &dsize, sizeof(short)); X#endif X p = file_table[0]->bsa_dol_t_text; X q = filename; X for (i = 0; i < dsize; i++) X *q++ = *p++; X *q = '\0'; X X /* extract file's record attributes */ X#ifndef SWAP X dsize = file_table[5]->bsa_dol_w_size; X#else X swap(&file_table[5]->bsa_dol_w_size, &dsize, sizeof(short)); X#endif X p = file_table[5]->bsa_dol_t_text; X recfmt = p[0]; X recatt = p[1]; X#ifndef SWAP X bcopy(&p[2], &recsize, sizeof(short)); X#else X swap(&p[2], &recsize, sizeof(short)); X#endif X vfcsize = p[15]; X if (vfcsize == 0) X vfcsize = 2; X#ifdef DEBUG X printf("recfmt = %d\n", recfmt); X printf("recatt = %d\n", recatt); X printf("reclen = %d\n", recsize); X printf("vfcsize = %d\n", vfcsize); X#endif X#ifndef SWAP X bcopy(&p[10], &nblk, sizeof(short)); X bcopy(&p[12], &lnch, sizeof(short)); X#else X swap(&p[10], &nblk, sizeof(short)); X swap(&p[12], &lnch, sizeof(short)); X#endif X filesize = (nblk-1)*512 + lnch; X#ifdef DEBUG X printf("nbk = %d, lnch = %d\n", nblk, lnch); X printf("filesize = 0x%x\n", filesize); X#endif X X /* open the file */ X if (f != NULL) { X fclose(f); X file_count = 0; X reclen = 0; X } X procf = 0; X if (goptind < gargc) X for(i=goptind; i < gargc; i++) { X procf |= match(filename,gargv[i]); X } X else X procf = 1; X if (tflag && procf) X printf( " %-35s %8d \n",filename,filesize); X if (xflag && procf) { X /* open file */ X f = openfile(filename); X if(f != NULL && vflag) printf("extracting %s\n", filename); X } X} X/* X * X * process a virtual block record (file record) X * X */ Xprocess_vbn(buffer, rsize) Xchar *buffer; Xunsigned short rsize; X{ X int c, i; X X if (f == NULL) { X return; X } X i = 0; X while (file_count+i < filesize && i < rsize) { X switch (recfmt) { X case FAB_dol_C_FIX: X if (reclen == 0) { X reclen = recsize; X } X fputc(buffer[i], f); X i++; X reclen--; X break; X X case FAB_dol_C_VAR: X case FAB_dol_C_VFC: X if (reclen == 0) { X reclen = *((short *) &buffer[i]); X#ifdef SWAP X swap(&reclen, &reclen, sizeof(short)); X#endif X#ifdef NEWD X fprintf(lf, "---\n"); X fprintf(lf, "reclen = %d\n", reclen); X fprintf(lf, "i = %d\n", i); X fprintf(lf, "rsize = %d\n", rsize); X#endif NEWD X fix = reclen; X i += 2; X if (recfmt == FAB_dol_C_VFC) { X i += vfcsize; X reclen -= vfcsize; X } X } else if (reclen == fix X && recatt == (1 << FAB_dol_V_FTN)) { X /**** X if (buffer[i] == '0') X fputc('\n', f); X else if (buffer[i] == '1') X fputc('\f', f); X *** sow ***/ X fputc(buffer[i],f); /** sow **/ X i++; X reclen--; X } else { X fputc(buffer[i], f); X i++; X reclen--; X } X if (reclen == 0) { X fputc('\n', f); X if (i & 1) X i++; X } X break; X X case FAB_dol_C_STM: X case FAB_dol_C_STMLF: X if (reclen < 0) { X printf("SCREAM\n"); X } X if (reclen == 0) { X reclen = 512; X } X c = buffer[i++]; X reclen--; X if (c == '\n') { X reclen = 0; X } X fputc(c, f); X break; X X case FAB_dol_C_STMCR: X c = buffer[i++]; X if (c == '\r') X fputc('\n', f); X else X fputc(c, f); X break; X X default: X fclose(f); X unlink(filename); X fprintf(stderr, "Invalid record format = %d\n", recfmt); X return; X } X } X file_count += i; X} X#ifdef SWAP X/* X * X * do swapping for Motorola type architectures X * X */ Xswap(from, to, nbytes) Xchar *from, *to; Xint nbytes; X{ X int i, j; X char temp[100]; X X for (i = 0; i < nbytes; i++) X temp[i] = from[i]; X for (i = 0, j = nbytes-1; i < nbytes; i++, j--) X to[i] = temp[j]; X} X#endif X/* X * X * process a backup block X * X */ Xprocess_block(block, blocksize) Xchar *block; Xint blocksize; X{ X X unsigned short bhsize, rsize, rtype; X unsigned long bsize, i; X X i = 0; X X /* read the backup block header */ X block_header = (struct bbh *) &block[i]; X i += sizeof(struct bbh); X X bhsize = block_header->bbh_dol_w_size; X bsize = block_header->bbh_dol_l_blocksize; X#ifdef SWAP X swap(&bhsize, &bhsize, sizeof(short)); X swap(&bsize, &bsize, sizeof(long)); X#endif X X /* check the validity of the header block */ X if (bhsize != sizeof(struct bbh)) { X fprintf(stderr, "Snark: Invalid header block size\n"); X exit(1); X } X if (bsize != 0 && bsize != blocksize) { X fprintf(stderr, "Snark: Invalid block size\n"); X exit(1); X } X#ifdef DEBUG X printf("new block: i = %d, bsize = %d\n", i, bsize); X#endif X X /* read the records */ X while (i < bsize) { X /* read the backup record header */ X record_header = (struct brh *) &block[i]; X i += sizeof(struct brh); X X rtype = record_header->brh_dol_w_rtype; X rsize = record_header->brh_dol_w_rsize; X#ifdef SWAP X swap(&rtype, &rtype, sizeof(short)); X swap(&rsize, &rsize, sizeof(short)); X#endif X#ifdef DEBUG X printf("rtype = %d\n", rtype); X printf("rsize = %d\n", rsize); X printf("flags = 0x%x\n", record_header->brh_dol_l_flags); X printf("addr = 0x%x\n", record_header->brh_dol_l_address); X printf("i = %d\n", i); X#endif X X switch (rtype) { X X case brh_dol_k_null: X#ifdef DEBUG X printf("rtype = null\n"); X#endif X break; X X case brh_dol_k_summary: X#ifdef DEBUG X printf("rtype = summary\n"); X#endif X break; X X case brh_dol_k_file: X#ifdef DEBUG X printf("rtype = file\n"); X#endif X process_file(&block[i]); X break; X X case brh_dol_k_vbn: X#ifdef DEBUG X printf("rtype = vbn\n"); X#endif X process_vbn(&block[i], rsize); X break; X X case brh_dol_k_physvol: X#ifdef DEBUG X printf("rtype = physvol\n"); X#endif X break; X X case brh_dol_k_lbn: X#ifdef DEBUG X printf("rtype = lbn\n"); X#endif X break; X X case brh_dol_k_fid: X#ifdef DEBUG X printf("rtype = fid\n"); X#endif X break; X X default: X fprintf(stderr, " Snark: invalid record type\n"); X fprintf(stderr, " record type = %d\n", rtype); X exit(1); X } X#ifdef pyr X i = i + rsize; X#else X i += rsize; X#endif X } X} X Xrdhead() X{ X int i, nfound; X char name[80]; X nfound = 1; X /* read the tape label - 4 records of 80 bytes */ X while ((i = read(fd, label, LABEL_SIZE)) != 0) { X if (i != LABEL_SIZE) { X fprintf(stderr, "Snark: bad label record\n"); X exit(1); X } X if (strncmp(label, "VOL1",4) == 0) { X sscanf(label+4, "%14s", name); X if(vflag || tflag) printf("Volume: %s\n",name); X } X if (strncmp(label, "HDR1",4) == 0) { X sscanf(label+4, "%14s", name); X sscanf(label+31, "%4d", &setnr); X } X /* get the block size */ X if (strncmp(label, "HDR2", 4) == 0) { X nfound = 0; X sscanf(label+5, "%5d", &blocksize); X#ifdef DEBUG X printf("blocksize = %d\n", blocksize); X#endif X } X } X if((vflag || tflag) && !nfound) X printf("Saveset name: %s number: %d\n",name,setnr); X /* get the block buffer */ X block = (char *) malloc(blocksize); X if (block == (char *) 0) { X fprintf(stderr, "memory allocation for block failed\n"); X exit(1); X } X return(nfound); X} X Xrdtail() X{ X int i; X char name[80]; X /* read the tape label - 4 records of 80 bytes */ X while ((i = read(fd, label, LABEL_SIZE)) != 0) { X if (i != LABEL_SIZE) { X fprintf(stderr, "Snark: bad label record\n"); X exit(1); X } X if (strncmp(label, "EOF1",4) == 0) { X sscanf(label+4, "%14s", name); X if(vflag || tflag) X printf("End of saveset: %s\n\n\n",name); X } X } X} X Xusage(progname) Xchar *progname; X{ X fprintf(stderr, "Usage: %s -{tx}[cdevw][-b blocksize][-s setnumber][-f tapefile]\n",progname); X} X X X X Xmain(argc, argv) Xint argc; Xchar *argv[]; X{ X X char *progname; X int c, i, eoffl; X int selset; X extern int optind; X extern char *optarg; X X progname = argv[0]; X if(argc < 2){ X usage(progname); X exit(1); X } X gargv = argv; X gargc = argc; X blocksize = 32256; X tapefile = def_tapefile; X cflag=dflag=eflag=fflag=sflag=tflag=vflag=wflag=xflag=0; X while((c=getopt(argc,argv,"bcdef:s:tvwx")) != EOF) X switch(c){ X case 'b': X sscanf(optarg,"%d",&blocksize); X break; X case 'c': X cflag++; X break; X case 'd': X dflag++; X break; X case 'e': X eflag++; X break; X case 'f': X fflag++; X tapefile = optarg; X break; X case 's': X sflag++; X sscanf(optarg,"%d",&selset); X break; X case 't': X tflag++; X break; X case 'v': X vflag++; X break; X case 'w': X wflag++; X break; X case 'x': X xflag++; X break; X case '?': X usage(progname); X exit(1); X break; X }; X if(!tflag && !xflag) { X usage(progname); X exit(1); X } X goptind = optind; X X#ifdef NEWD X /* open debug file */ X lf = fopen("log", "w"); X if (lf == NULL) { X perror("log"); X exit(1); X } X#endif X X /* open the tape/disk file */ X fd = open(tapefile, O_RDONLY); X if (fd < 0) { X perror(tapefile); X exit(1); X } X X if(!fflag) { X /* rewind the tape */ X op.mt_op = MTREW; X op.mt_count = 1; X i = ioctl(fd, MTIOCTOP, &op); X if (i < 0) { X perror(tapefile); X exit(1); X } X eoffl = rdhead(); X } X else { X eoffl = 0; X block = (char *) malloc(blocksize); X } X X /* read the backup tape blocks until end of tape */ X while (!eoffl) { X if(sflag && setnr != selset && !fflag) { X op.mt_op = MTFSF; X op.mt_count = 1; X i = ioctl(fd, MTIOCTOP, &op); X if (i < 0) { X perror(tapefile); X exit(1); X } X i = 0; X } X else X i = read(fd, block, blocksize); X if(i == 0 && !fflag) { X rdtail(); X eoffl=rdhead(); X } X else if(i == 0) break; X X if (i != blocksize) { X fprintf(stderr, "bad block read i = %d\n", i); X exit(1); X } X else{ X eoffl = 0; X process_block(block, blocksize); X } X } X if(vflag || tflag) printf("End of tape\n"); X X /* close the tape */ X close(fd); X X#ifdef NEWD X /* close debug file */ X fclose(lf); X#endif NEWD X X /* exit cleanly */ X exit(0); X} ________This_Is_The_END________ if test `wc -l < vmsbackup.c` -ne 863; then echo 'shar: vmsbackup.c was damaged during transit (should have been 863 bytes)' fi fi ; : end of overwriting check echo 'x - vmsbackup.doc' if test -f vmsbackup.doc; then echo 'shar: not overwriting vmsbackup.doc'; else sed 's/^X//' << '________This_Is_The_END________' > vmsbackup.doc X X Xvmsbackup.exe X To read a tape written in VMS BACKUP format on X a UNIX tape drive: X X /usr/local/vmsbackup.exe -xvdf /dev/rmt0 X X This will correctly extract the contents of the tape. X For example, if one of the files on the tape was X KINNEY:[QA.BDL.68020]68020.CKT;1 this would be X extracted as ./kinney/qa/bdl/68020/68020.ckt X X One time I was having a problem reading a VMS BACKUP X tape that I wrote last year. It would mount, without X telling me the label name. When I tried to read the X tape using "backup/lis mub0:", VMS would complain X "Tape not in valid ANSI format". The solution was X to use HHB's tape drive and read the VMS BACKUP tape X under UNIX. X X VMSBACKUP X XNAME X vmsbackup Read a VMS backup tape on a UNIX system. X X XSYNOPSIS Xvmsbackup -{tx}[b blocksize][cdevw][s setnumber][f tapefile] [ name ... ] X X XDESCRIPTION X Xvmsbackup reads a VMS generated backup tape, converting the files Xto UNIX format and writing the files to disk. The default operation of Xthe program is to go through an entire tape, extracting every file and Xwriting it to disc. This may be modified by the following options. X X -b blocksize Set block size of saveset file. Default is 32256 X bytes per record. X X -c Use complete filenames, including the version X number. A colon and the octal version number X will be appended to all filenames. A colon, X rather than a semicolon, is used since the UNIX X Shell uses the semicolon as the line separator. X Using a colon prevents the user from having to X escape the semicolon when referencing the filename. X This option is useful only when multiple versions X of the same file are on a single tape or when a X file of the same name already exists in the dest- X ination directory. The default is to ignore X version numbers. X X X -d Use the directory structure from VMS. The default X value is off. X X X -e Process all filename extensions. Since this X program is mainly intended to move source code X and possibly data from a DEC system to a UNIX X system, the default is to ignore all files whose X filename extension specifies system dependent X data. The file types which will be ignored, X unless the -e option is specified, are X X exe VMS executable file X X lib VMS object library file X X obj RSX object file X X odl RSX overlay description file X X olb RSX object library file X X pmd RSX post mortem dump file X X stb RSX task symbol table file X X sys RSX bootable system file X X tsk RSX executable task file X X X -f Use the next argument in the command line as X the tape device to be used, rather than the X default. X X If vmsbackup is compiled with the remote tape X option and the file name has the form system user X vmsbackup will use the tape drive /dev/??? on X the remote system system , via rsh and rmt. The X optional user portion of the pathname specifies X the login name to use on the remote system. If X it is not supplied, the current user's login name X will be used. In all the cases, the user must X have the appropriate permissions on the remote X machine, in order to use this facility. The X default is /dev/rmt8 (drive 0, raw mode, 1600 X bpi). This must be a raw mode tape device. X X X -s saveset Process only the given saveset number. X X X -t Produce a table of contents (a directory listing) X on the standard output of the files on tape. X X X -v Verbose output. Normally vmsbackup does its work X silently. The verbose option will cause the file- X names of the files being read from tape to disk X to be output on the standard output. X X X -w vmsbackup prints the action to be taken followed X by file name, then wait for user confirmation. If X a word beginning with `y' is given, the action is X done. Any other input means don't do it. X X X -x Extract the named files from the tape. The opt- X ional name argument specifies one or more file- X names to be searched for specifically on the tape X and only those files are to be processed. The X name may contain the usal sh(1) meta-characters *?![] X X X XFILES X /dev/rmt x X X XSEE ALSO X rmtops(3) X X XBUGS X The filename match uses the complete VMS file names. X X XAUTHOR X John Douglas Carey X Sven-Ove Westberg ________This_Is_The_END________ if test `wc -l < vmsbackup.doc` -ne 146; then echo 'shar: vmsbackup.doc was damaged during transit (should have been 146 bytes)' fi fi ; : end of overwriting check exit 0 -- Abid Hussain axh@philabs.philips.com Philips Laboratories - or - (914) 945-6490 {uunet|ihnp4|decvax}!philabs!axh