Files
org-hyperion-cules/tapemap.c
2013-09-19 07:21:32 -07:00

237 lines
7.7 KiB
C

/* TAPEMAP.C (c) Copyright Jay Maynard, 2000-2012 */
/* Map AWSTAPE format tape image */
/* */
/* Released under "The Q Public License Version 1" */
/* (http://www.hercules-390.org/herclic.html) as modifications to */
/* Hercules. */
/*-------------------------------------------------------------------*/
/* This program reads an AWSTAPE format tape image file and produces */
/* a map of the tape, printing any standard label records it finds. */
/*-------------------------------------------------------------------*/
#include "hstdinc.h"
#include "hercules.h"
#include "tapedev.h"
#define UTILITY_NAME "tapemap"
/*-------------------------------------------------------------------*/
/* Static data areas */
/*-------------------------------------------------------------------*/
static BYTE vollbl[] = "\xE5\xD6\xD3"; /* EBCDIC characters "VOL" */
static BYTE hdrlbl[] = "\xC8\xC4\xD9"; /* EBCDIC characters "HDR" */
static BYTE eoflbl[] = "\xC5\xD6\xC6"; /* EBCDIC characters "EOF" */
static BYTE eovlbl[] = "\xC5\xD6\xE5"; /* EBCDIC characters "EOV" */
static BYTE buf[ MAX_BLKLEN ];
#ifdef EXTERNALGUI
/* Report progress every this many bytes */
#define PROGRESS_MASK (~0x3FFFF /* 256K */)
/* How many bytes we've read so far. */
long curpos = 0;
long prevpos = 0;
#endif /*EXTERNALGUI*/
/*-------------------------------------------------------------------*/
/* TAPEMAP main entry point */
/*-------------------------------------------------------------------*/
int main (int argc, char *argv[])
{
char *pgmname; /* prog name in host format */
char *pgm; /* less any extension (.ext) */
char msgbuf[512]; /* message build work area */
int i; /* Array subscript */
int len; /* Block length */
char *filename; /* -> Input file name */
int infd = -1; /* Input file descriptor */
int fileno; /* Tape file number */
int blkcount; /* Block count */
int curblkl; /* Current block length */
int minblksz; /* Minimum block size */
int maxblksz; /* Maximum block size */
int64_t file_bytes; /* File byte count */
BYTE labelrec[81]; /* Standard label (ASCIIZ) */
AWSTAPE_BLKHDR awshdr; /* AWSTAPE block header */
char pathname[MAX_PATH]; /* file path in host format */
char *strtok_str = NULL;
/* Set program name */
if ( argc > 0 )
{
if ( strlen(argv[0]) == 0 )
{
pgmname = strdup( UTILITY_NAME );
}
else
{
char path[MAX_PATH];
#if defined( _MSVC_ )
GetModuleFileName( NULL, path, MAX_PATH );
#else
strncpy( path, argv[0], sizeof( path ) );
#endif
pgmname = strdup(basename(path));
#if !defined( _MSVC_ )
strncpy( path, argv[0], sizeof(path) );
#endif
}
}
else
{
pgmname = strdup( UTILITY_NAME );
}
pgm = strtok_r( strdup(pgmname), ".", &strtok_str);
INITIALIZE_UTILITY( pgmname );
/* Display the program identification message */
MSGBUF( msgbuf, MSG_C( HHC02499, "I", pgm, "tape map" ) );
display_version (stderr, msgbuf+10, FALSE);
/* The only argument is the tape image file name */
if (argc == 2 && argv[1] != NULL)
{
filename = argv[1];
}
else
{
printf ( MSG( HHC02726, "I", pgm ) );
exit (1);
}
/* Open the tape device */
hostpath(pathname, filename, sizeof(pathname));
infd = HOPEN (pathname, O_RDONLY | O_BINARY);
if (infd < 0)
{
printf( MSG( HHC02715, "E", filename, errno, strerror(errno) ) );
exit (2);
}
/* Read blocks from the input file and report on them */
fileno = 1;
blkcount = 0;
minblksz = 0;
maxblksz = 0;
file_bytes = 0;
len = 0;
while (1)
{
#ifdef EXTERNALGUI
if (extgui)
{
/* Report progress every nnnK */
if( ( curpos & PROGRESS_MASK ) != ( prevpos & PROGRESS_MASK ) )
{
prevpos = curpos;
fprintf( stderr, "IPOS=%ld\n", curpos );
}
}
#endif /*EXTERNALGUI*/
/* Read a block from the tape */
len = read (infd, buf, sizeof(AWSTAPE_BLKHDR));
#ifdef EXTERNALGUI
if (extgui) curpos += len;
#endif /*EXTERNALGUI*/
if (len < 0)
{
printf( MSG( HHC02707, "E", filename, "AWSTAPE", len, errno, strerror(errno) ) );
exit (3);
}
/* Did we finish too soon? */
if ((len > 0) && (len < (int)sizeof(AWSTAPE_BLKHDR)))
{
printf( MSG( HHC02741, "E", filename, "AWSTAPE" ) );
exit(4);
}
/* Check for end of tape. */
if (len == 0)
{
printf( MSG( HHC02704, "I" ) );
break;
}
/* Parse the block header */
memcpy(&awshdr, buf, sizeof(AWSTAPE_BLKHDR));
/* Tapemark? */
if ((awshdr.flags1 & AWSTAPE_FLAG1_TAPEMARK) != 0)
{
/* Print summary of current file */
printf( MSG( HHC02721, "I", fileno, blkcount, file_bytes,
minblksz, maxblksz,
(blkcount ? ((int)file_bytes/blkcount) : 0)) );
/* Reset counters for next file */
fileno++;
minblksz = 0;
maxblksz = 0;
blkcount = 0;
file_bytes = 0;
}
else /* if(tapemark) */
{
/* Count blocks and block sizes */
blkcount++;
curblkl = awshdr.curblkl[0] + (awshdr.curblkl[1] << 8);
if (curblkl > maxblksz) maxblksz = curblkl;
if (minblksz == 0 || curblkl < minblksz) minblksz = curblkl;
file_bytes += curblkl;
/* Read the data block. */
len = read (infd, buf, curblkl);
#ifdef EXTERNALGUI
if (extgui) curpos += len;
#endif /*EXTERNALGUI*/
if (len < 0)
{
printf ( MSG( HHC02709, "E", filename, "AWSTAPE", len, errno, strerror(errno) ) );
exit (5);
}
/* Did we finish too soon? */
if ((len > 0) && (len < curblkl))
{
printf (MSG( HHC02742, "E", filename, curblkl, len ) );
exit(6);
}
/* Check for end of tape */
if (len == 0)
{
printf ( MSG( HHC02743, "E", filename, "AWSTAPE" ) );
exit(7);
}
/* Print standard labels */
if (len == 80 && blkcount < 4
&& (memcmp(buf, vollbl, 3) == 0
|| memcmp(buf, hdrlbl, 3) == 0
|| memcmp(buf, eoflbl, 3) == 0
|| memcmp(buf, eovlbl, 3) == 0))
{
for (i=0; i < 80; i++)
labelrec[i] = guest_to_host(buf[i]);
labelrec[i] = '\0';
printf ( MSG( HHC02722, "I", labelrec ) );
}
} /* end if(tapemark) */
} /* end while */
/* Close files and exit */
close (infd);
return 0;
} /* end function main */