Files
org-hyperion-cules/cckdutil.c
Jan Jaeger 439bdbca92 Remove subversion $Id$ tag
Update copyright statement to 2012
2012-03-24 18:41:51 +11:00

2932 lines
110 KiB
C

/* CCKDUTIL.C (c) Copyright Roger Bowler, 1999-2012 */
/* ESA/390 Compressed CKD Common routines */
/* */
/* Released under "The Q Public License Version 1" */
/* (http://www.hercules-390.org/herclic.html) as modifications to */
/* Hercules. */
/*-------------------------------------------------------------------*/
/* This module contains functions for compressed CKD devices */
/* used by more than 1 main program. */
/*-------------------------------------------------------------------*/
#include "hstdinc.h"
#define _CCKDUTIL_C_
#define _HDASD_DLL_
#include "hercules.h"
#include "opcode.h"
/*-------------------------------------------------------------------*/
typedef struct _SPCTAB { /* Space table */
BYTE typ; /* Type of space */
int val; /* Value for space */
U32 pos; /* Space offset */
U32 len; /* Space length */
U32 siz; /* Space size */
} SPCTAB;
#define SPCTAB_NONE 0 /* Ignore this space entry */
#define SPCTAB_DEVHDR 1 /* Space is device header */
#define SPCTAB_CDEVHDR 2 /* Space is compressed hdr */
#define SPCTAB_L1 3 /* Space is level 1 table */
#define SPCTAB_L2 4 /* Space is level 2 table */
#define SPCTAB_TRK 5 /* Space is track image */
#define SPCTAB_BLKGRP 6 /* Space is blkgrp image */
#define SPCTAB_FREE 7 /* Space is free block */
#define SPCTAB_EOF 8 /* Space is end-of-file */
/*-------------------------------------------------------------------*/
/* Internal functions */
/*-------------------------------------------------------------------*/
static int comp_spctab_sort(const void *a, const void *b);
static int cdsk_spctab_sort(const void *a, const void *b);
static int cdsk_build_free_space(SPCTAB *spctab, int s);
static int cdsk_valid_trk (int trk, BYTE *buf, int heads, int len);
/*-------------------------------------------------------------------*/
/* Static data areas */
/*-------------------------------------------------------------------*/
static BYTE eighthexFF[] = {0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff};
static char *spaces[] = { "none", "devhdr", "cdevhdr", "l1", "l2",
"trk", "blkgrp", "free", "eof" };
static char *comps[] = { "none", "zlib", "bzip2" };
/*-------------------------------------------------------------------*/
/* EXTERNALGUI support */
/*-------------------------------------------------------------------*/
#ifdef EXTERNALGUI
#define gui_fprintf if (extgui) fprintf
#else
#define gui_fprintf(...)
#endif
/*-------------------------------------------------------------------*/
/* Change the endianess of a compressed file */
/*-------------------------------------------------------------------*/
DLL_EXPORT int cckd_swapend (DEVBLK *dev)
{
CCKDDASD_EXT *cckd; /* -> cckd extension */
int fd; /* File descriptor */
int rc; /* Return code */
struct stat fst; /* File status buffer */
int i; /* Index */
int swapend; /* 1=swap space */
int len; /* Length */
off_t off, lopos, hipos; /* File offsets */
CCKD_DEVHDR cdevhdr; /* Compressed ckd header */
CCKD_L1ENT *l1 = NULL; /* Level 1 table */
CCKD_L2ENT l2[256]; /* Level 2 table */
CCKD_FREEBLK freeblk; /* Free block */
/* Get fd */
cckd = dev->cckd_ext;
if (cckd == NULL)
fd = dev->fd;
else
fd = cckd->fd[cckd->sfn];
/* Get file size */
if (fstat (fd, &fst) < 0)
goto cswp_fstat_error;
gui_fprintf (stderr, "SIZE=%"I64_FMT"u\n", (U64) fst.st_size);
hipos = fst.st_size;
/* Device header */
off = CCKD_DEVHDR_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_DEVHDR_SIZE;
if ((rc = read (fd, &cdevhdr, len)) != len)
goto cswp_read_error;
swapend = (cdevhdr.options & CCKD_BIGENDIAN) != cckd_endian();
cckd_swapend_chdr (&cdevhdr);
cdevhdr.options |= CCKD_ORDWR;
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, &cdevhdr, len)) != len)
goto cswp_write_error;
if (!swapend) cckd_swapend_chdr (&cdevhdr);
/* l1 table */
len = cdevhdr.numl1tab * CCKD_L1ENT_SIZE;
if ((l1 = malloc (len)) == NULL)
goto cswp_malloc_error;
off = CCKD_L1TAB_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, l1, len)) != len)
goto cswp_read_error;
cckd_swapend_l1 (l1, (int)cdevhdr.numl1tab);
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, l1, len)) != len)
goto cswp_write_error;
if (!swapend) cckd_swapend_l1 (l1, (int)cdevhdr.numl1tab);
lopos = CCKD_L1TAB_POS + len;
/* l2 tables */
for (i = 0; i < cdevhdr.numl1tab; i++)
{
if (l1[i] == 0 || l1[i] == 0xffffffff
|| l1[i] < lopos || l1[i] > hipos - CCKD_L2TAB_SIZE)
continue;
off = (off_t)l1[i];
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_L2TAB_SIZE;
if ((rc = read (fd, l2, len)) != len)
goto cswp_read_error;
cckd_swapend_l2 (l2);
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, l2, len)) != len)
goto cswp_write_error;
}
free (l1);
l1 = NULL;
/* free space */
if (cdevhdr.free && cdevhdr.free >= lopos
&& cdevhdr.free <= hipos - CCKD_FREEBLK_SIZE)
{
off = (off_t)cdevhdr.free;
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_FREEBLK_SIZE;
if ((rc = read (fd, &freeblk, len)) != len)
goto cswp_read_error;
if (memcmp(&freeblk, "FREE_BLK", 8) == 0)
{
/* New format free space */
for (i = 0; i < cdevhdr.free_number; i++)
{
off += CCKD_FREEBLK_SIZE;
if (off > hipos - CCKD_FREEBLK_SIZE)
break;
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, &freeblk, len)) != len)
goto cswp_read_error;
cckd_swapend_free (&freeblk);
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, &freeblk, len)) != len)
goto cswp_write_error;
} /* for each free space */
} /* if new format free space */
else
{
/* Old format free space */
for (i = 0; i < cdevhdr.free_number; i++)
{
if (off < lopos || off > hipos - CCKD_FREEBLK_SIZE)
break;
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, &freeblk, len)) != len)
goto cswp_read_error;
cckd_swapend_free (&freeblk);
if (lseek (fd, off, SEEK_SET) < 0)
goto cswp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, &freeblk, len)) != len)
goto cswp_write_error;
if (!swapend) cckd_swapend_free (&freeblk);
off = (off_t)freeblk.pos;
} /* for each free space */
} /* else old format free space */
} /* if free space */
return 0;
/* error exits */
cswp_fstat_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"fstat()", strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"fstat()", strerror(errno));
goto cswp_error;
cswp_lseek_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"lseek()", off, strerror(errno)));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"lseek()", off, strerror(errno));
goto cswp_error;
cswp_read_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"read()", off, rc < 0 ? strerror(errno) : "incomplete"));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"read()", off, rc < 0 ? strerror(errno) : "incomplete");
goto cswp_error;
cswp_write_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"write()", off, rc < 0 ? strerror(errno) : "incomplete"));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"write()", off, rc < 0 ? strerror(errno) : "incomplete");
goto cswp_error;
cswp_malloc_error:
{
char buf[64];
MSGBUF( buf, "malloc(%d)", len);
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno));
goto cswp_error;
}
cswp_error:
if (l1) free(l1);
return -1;
}
/*-------------------------------------------------------------------*/
/* Swap endian - compressed device header */
/*-------------------------------------------------------------------*/
DLL_EXPORT void cckd_swapend_chdr (CCKD_DEVHDR *cdevhdr)
{
/* fix the compressed ckd header */
cdevhdr->options ^= CCKD_BIGENDIAN;
cckd_swapend4 ((char *) &cdevhdr->numl1tab);
cckd_swapend4 ((char *) &cdevhdr->numl2tab);
cckd_swapend4 ((char *) &cdevhdr->size);
cckd_swapend4 ((char *) &cdevhdr->used);
cckd_swapend4 ((char *) &cdevhdr->free);
cckd_swapend4 ((char *) &cdevhdr->free_total);
cckd_swapend4 ((char *) &cdevhdr->free_largest);
cckd_swapend4 ((char *) &cdevhdr->free_number);
cckd_swapend4 ((char *) &cdevhdr->free_imbed);
cckd_swapend2 ((char *) &cdevhdr->compress_parm);
}
/*-------------------------------------------------------------------*/
/* Swap endian - level 1 table */
/*-------------------------------------------------------------------*/
DLL_EXPORT void cckd_swapend_l1 (CCKD_L1ENT *l1, int n)
{
int i; /* Index */
for (i = 0; i < n; i++)
cckd_swapend4 ((char *) &l1[i]);
}
/*-------------------------------------------------------------------*/
/* Swap endian - level 2 table */
/*-------------------------------------------------------------------*/
DLL_EXPORT void cckd_swapend_l2 (CCKD_L2ENT *l2)
{
int i; /* Index */
for (i = 0; i < 256; i++)
{
cckd_swapend4 ((char *) &l2[i].pos);
cckd_swapend2 ((char *) &l2[i].len);
cckd_swapend2 ((char *) &l2[i].size);
}
}
/*-------------------------------------------------------------------*/
/* Swap endian - free space entry */
/*-------------------------------------------------------------------*/
DLL_EXPORT void cckd_swapend_free (CCKD_FREEBLK *fb)
{
cckd_swapend4 ((char *) &fb->pos);
cckd_swapend4 ((char *) &fb->len);
}
/*-------------------------------------------------------------------*/
/* Swap endian - 4 bytes */
/*-------------------------------------------------------------------*/
DLL_EXPORT void cckd_swapend4 (char *c)
{
char temp[4];
memcpy (&temp, c, 4);
c[0] = temp[3];
c[1] = temp[2];
c[2] = temp[1];
c[3] = temp[0];
}
/*-------------------------------------------------------------------*/
/* Swap endian - 2 bytes */
/*-------------------------------------------------------------------*/
DLL_EXPORT void cckd_swapend2 (char *c)
{
char temp[2];
memcpy (&temp, c, 2);
c[0] = temp[1];
c[1] = temp[0];
}
/*-------------------------------------------------------------------*/
/* Are we little or big endian? From Harbison&Steele. */
/*-------------------------------------------------------------------*/
DLL_EXPORT int cckd_endian()
{
union
{
long l;
char c[sizeof (long)];
} u;
u.l = 1;
return u.c[sizeof (long) - 1] == 1 ? CCKD_BIGENDIAN : 0;
}
/*-------------------------------------------------------------------
* Remove all free space from a compressed ckd file
*-------------------------------------------------------------------*/
DLL_EXPORT int cckd_comp (DEVBLK *dev)
{
CCKDDASD_EXT *cckd; /* -> cckd extension */
int fd; /* File descriptor */
struct stat fst; /* File status buffer */
int rc; /* Return code */
off_t off; /* File offset */
off_t l2area; /* Boundary for l2 tables */
int len; /* Length */
int i, j, l, n; /* Work variables */
int relocate = 0; /* 1=spaces will be relocated*/
int l1size; /* l1 table size */
U32 next; /* offset of next space */
int s; /* space table index */
CKDDASD_DEVHDR devhdr; /* CKD device header */
CCKD_DEVHDR cdevhdr; /* CCKD device header */
CCKD_L1ENT *l1=NULL; /* -> l1 table */
CCKD_L2ENT **l2=NULL; /* -> l2 table array */
SPCTAB *spctab=NULL; /* -> space table */
BYTE *rbuf=NULL; /* Relocation buffer */
BYTE *p; /* -> relocation buffer */
int rlen=0; /* Relocation buffer length */
CCKD_L2ENT zero_l2[256]; /* Empty l2 table (zeros) */
CCKD_L2ENT ff_l2[256]; /* Empty l2 table (0xff's) */
BYTE buf[65536*4]; /* Buffer */
/*---------------------------------------------------------------
* Get fd
*---------------------------------------------------------------*/
cckd = dev->cckd_ext;
if (cckd == NULL)
fd = dev->fd;
else
fd = cckd->fd[cckd->sfn];
/*---------------------------------------------------------------
* Get file statistics
*---------------------------------------------------------------*/
if (fstat (fd, &fst) < 0)
goto comp_fstat_error;
gui_fprintf (stderr, "SIZE=%"I64_FMT"u\n", (U64) fst.st_size);
/*---------------------------------------------------------------
* Read device header
*---------------------------------------------------------------*/
off = 0;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CKDDASD_DEVHDR_SIZE;
if ((rc = read (fd, &devhdr, len)) != len)
goto comp_read_error;
if (memcmp (devhdr.devid, "CKD_C370", 8) != 0
&& memcmp (devhdr.devid, "CKD_S370", 8) != 0
&& memcmp (devhdr.devid, "FBA_C370", 8) != 0
&& memcmp (devhdr.devid, "FBA_S370", 8) != 0)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00356, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00356, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
goto comp_error;
}
comp_restart:
/*---------------------------------------------------------------
* Read compressed device header
*---------------------------------------------------------------*/
off = CCKD_DEVHDR_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_DEVHDR_SIZE;
if ((rc = read (fd, &cdevhdr, len)) != len)
goto comp_read_error;
/*---------------------------------------------------------------
* Check the endianess of the file
*---------------------------------------------------------------*/
if ((cdevhdr.options & CCKD_BIGENDIAN) != cckd_endian())
{
if(dev->batch)
fprintf(stdout, MSG(HHC00357, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
cckd_endian() ? "big-endian" : "little-endian"));
else
WRMSG(HHC00357, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
cckd_endian() ? "big-endian" : "little-endian");
if (cckd_swapend (dev) < 0)
goto comp_error;
else
goto comp_restart;
}
/*---------------------------------------------------------------
* Some header checks
*---------------------------------------------------------------*/
if ((off_t)cdevhdr.size != fst.st_size
|| cdevhdr.size != cdevhdr.used || cdevhdr.free != 0
|| cdevhdr.free_total != 0 || cdevhdr.free_largest != 0
|| cdevhdr.free_number != 0 || cdevhdr.free_imbed != 0)
relocate = 1;
/*---------------------------------------------------------------
* Build empty l2 tables
*---------------------------------------------------------------*/
memset( &zero_l2, 0, CCKD_L2TAB_SIZE );
if (cdevhdr.nullfmt != 0)
for (i = 0; i < 256; i++)
zero_l2[i].len = zero_l2[i].size = cdevhdr.nullfmt;
memset (&ff_l2, 0xff, CCKD_L2TAB_SIZE);
/*---------------------------------------------------------------
* Read the l1 table
*---------------------------------------------------------------*/
l1size = len = cdevhdr.numl1tab * CCKD_L1ENT_SIZE;
if ((l1 = malloc (len)) == NULL)
goto comp_malloc_error;
off = CCKD_L1TAB_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, l1, len)) != len)
goto comp_read_error;
/*---------------------------------------------------------------
* Build the space table
*---------------------------------------------------------------*/
n = 1 + 1 + 1 + cdevhdr.numl1tab + 1;
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l1[i] != 0 && l1[i] != 0xffffffff)
n += 256;
len = sizeof(SPCTAB);
if ((spctab = calloc (n, len)) == NULL)
goto comp_calloc_error;
s = 0;
spctab[s].typ = SPCTAB_DEVHDR;
spctab[s].val = -1;
spctab[s].pos = 0;
spctab[s].len =
spctab[s].siz = CKDDASD_DEVHDR_SIZE;
s++;
spctab[s].typ = SPCTAB_CDEVHDR;
spctab[s].val = -1;
spctab[s].pos = CCKD_DEVHDR_POS;
spctab[s].len =
spctab[s].siz = CCKD_DEVHDR_SIZE;
s++;
spctab[s].typ = SPCTAB_L1;
spctab[s].val = -1;
spctab[s].pos = CCKD_L1TAB_POS;
spctab[s].len =
spctab[s].siz = l1size;
s++;
spctab[s].typ = SPCTAB_EOF;
spctab[s].val = -1;
spctab[s].pos = fst.st_size;
spctab[s].len =
spctab[s].siz = 0;
s++;
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l1[i] != 0 && l1[i] != 0xffffffff)
{
spctab[s].typ = SPCTAB_L2;
spctab[s].val = i;
spctab[s].pos = l1[i];
spctab[s].len =
spctab[s].siz = CCKD_L2TAB_SIZE;
s++;
}
qsort (spctab, s, sizeof(SPCTAB), comp_spctab_sort);
/*---------------------------------------------------------------
* Read level 2 tables
*---------------------------------------------------------------*/
n = cdevhdr.numl1tab;
len = sizeof (void *);
if ((l2 = calloc (n, len)) == NULL)
goto comp_calloc_error;
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
{
if (spctab[i].typ != SPCTAB_L2) continue;
l = spctab[i].val;
len = CCKD_L2TAB_SIZE;
if ((l2[l] = malloc (len)) == NULL)
goto comp_malloc_error;
off = (off_t)spctab[i].pos;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, l2[l], len)) != len)
goto comp_read_error;
for (j = 0; j < 256; j++)
{
if (l2[l][j].pos == 0 || l2[l][j].pos == 0xffffffff)
continue;
spctab[s].typ = SPCTAB_TRK;
spctab[s].val = spctab[i].val*256 + j;
spctab[s].pos = l2[l][j].pos;
spctab[s].len = l2[l][j].len;
spctab[s].siz = l2[l][j].size;
s++;
} /* for each l2 entry */
/* check if empty l2 table */
if (memcmp (l2[l], &zero_l2, CCKD_L2TAB_SIZE) == 0
|| memcmp (l2[l], &ff_l2, CCKD_L2TAB_SIZE) == 0)
{
l1[l] = l2[l][0].pos; /* 0x00000000 or 0xffffffff */
spctab[i].typ = SPCTAB_NONE;
free (l2[l]);
l2[l] = NULL;
relocate = 1;
}
} /* for each space */
qsort (spctab, s, sizeof(SPCTAB), comp_spctab_sort);
while (spctab[s-1].typ == SPCTAB_NONE) s--;
/* set relocate flag if last space is free space */
if (spctab[s-2].pos + spctab[s-2].len != spctab[s-1].pos)
relocate = 1;
/*---------------------------------------------------------------
* relocate l2 tables in order
*---------------------------------------------------------------*/
/* determine l2 area */
l2area = CCKD_L1TAB_POS + l1size;
for (i = 0; i < cdevhdr.numl1tab; i++)
{
if (l1[i] == 0 || l1[i] == 0xffffffff) continue;
if (l1[i] != l2area)
relocate = 1;
l2area += CCKD_L2TAB_SIZE;
}
/* quick return if all l2 tables are orderered and no free space */
if (!relocate)
{
for (i = 1; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i-1].pos + spctab[i-1].len != spctab[i].pos)
break;
if (spctab[i].typ == SPCTAB_EOF)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00358, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00358, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
goto comp_return_ok;
}
}
/* file will be updated */
cdevhdr.options |= CCKD_ORDWR;
/* calculate track size within the l2 area */
for (i = rlen = 0; spctab[i].pos < l2area; i++)
if (spctab[i].typ == SPCTAB_TRK)
rlen += sizeof(spctab[i].val) + sizeof(spctab[i].len)
+ spctab[i].len;
/* read any tracks in the l2area into rbuf */
if ((len = rlen) > 0)
{
if ((rbuf = malloc (len)) == NULL)
goto comp_malloc_error;
for (i = 0, p = rbuf; spctab[i].pos < l2area; i++)
{
if (spctab[i].typ != SPCTAB_TRK) continue;
memcpy (p, &spctab[i].val, sizeof(spctab[i].val));
p += sizeof(spctab[i].val);
memcpy (p, &spctab[i].len, sizeof(spctab[i].len));
p += sizeof(spctab[i].len);
off = (off_t)spctab[i].pos;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = spctab[i].len;
if ((rc = read (fd, p, len)) != len)
goto comp_read_error;
p += len;
spctab[i].typ = SPCTAB_NONE;
} /* for each space in the l2 area */
qsort (spctab, s, sizeof(SPCTAB), comp_spctab_sort);
while (spctab[s-1].typ == SPCTAB_NONE) s--;
} /* if any tracks to relocate */
/* remove all l2 tables from the space table */
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].typ == SPCTAB_L2)
spctab[i].typ = SPCTAB_NONE;
qsort (spctab, s, sizeof(SPCTAB), comp_spctab_sort);
while (spctab[s-1].typ == SPCTAB_NONE) s--;
/* add all l2 tables at their ordered offsets */
off = CCKD_L1TAB_POS + l1size;
for (i = 0; i < cdevhdr.numl1tab; i++)
{
if (l1[i] == 0 || l1[i] == 0xffffffff) continue;
spctab[s].typ = SPCTAB_L2;
spctab[s].val = i;
spctab[s].pos = (U32)off;
spctab[s].len =
spctab[s].siz = CCKD_L2TAB_SIZE;
s++;
off += CCKD_L2TAB_SIZE;
}
qsort (spctab, s, sizeof(SPCTAB), comp_spctab_sort);
/* set end-of-file position */
spctab[s-1].pos = spctab[s-2].pos + spctab[s-2].len;
/*---------------------------------------------------------------
* Perform compression
*---------------------------------------------------------------*/
/* move spaces left */
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
{
/* ignore contiguous spaces */
if (spctab[i].pos + spctab[i].len == spctab[i+1].pos)
continue;
/* found a gap */
off = (off_t)spctab[i+1].pos;
/* figure out how much we can read */
for (len = 0, j = i + 1; spctab[j].typ != SPCTAB_EOF; j++)
{
if (len + spctab[j].len > sizeof(buf))
break;
next = spctab[j].pos + spctab[j].len;
spctab[j].pos = spctab[i].pos + spctab[i].len + len;
spctab[j].siz = spctab[j].len;
len += spctab[j].len;
if (next != spctab[j+1].pos)
break;
} /* search for contiguous spaces */
/* this can happen if the next space is end-of-file */
if (len == 0)
continue;
/* read the image(s) to be relocated */
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, buf, len)) != len)
goto comp_write_error;
/* write the images */
off = (off_t)spctab[i].pos + spctab[i].len;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, buf, len)) != len)
goto comp_write_error;
}
/* adjust the size of the file */
spctab[s-1].pos = spctab[s-2].pos + spctab[s-2].len;
/*---------------------------------------------------------------
* Write spaces relocated from the l2area to the end of the file
*---------------------------------------------------------------*/
off = (off_t)spctab[s-1].pos;
p = rbuf;
while (rlen)
{
spctab[s].typ = SPCTAB_TRK;
spctab[s].pos = (U32)off;
memcpy (&spctab[s].val, p, sizeof(spctab[s].val));
p += sizeof(spctab[s].val);
memcpy (&spctab[s].len, p, sizeof(spctab[s].len));
spctab[s].siz = spctab[s].len;
p += sizeof(spctab[s].len);
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = spctab[s].len;
if ((rc = write (fd, p, len)) != len)
goto comp_write_error;
p += len;
off += len;
rlen -= len + sizeof(spctab[s].val) + sizeof(spctab[s].len);
s++;
} /* for each relocated space in l2area */
/* adjust the space table */
if (rbuf)
{
free (rbuf); rbuf = NULL;
qsort (spctab, s, sizeof(SPCTAB), comp_spctab_sort);
spctab[s-1].pos = spctab[s-2].pos + spctab[s-2].len;
}
/*---------------------------------------------------------------
* Update the device header
*---------------------------------------------------------------*/
cdevhdr.size =
cdevhdr.used = spctab[s-1].pos;
cdevhdr.free =
cdevhdr.free_total =
cdevhdr.free_largest =
cdevhdr.free_number =
cdevhdr.free_imbed = 0;
cdevhdr.vrm[0] = CCKD_VERSION;
cdevhdr.vrm[1] = CCKD_RELEASE;
cdevhdr.vrm[2] = CCKD_MODLVL;
/*---------------------------------------------------------------
* Update the lookup tables
*---------------------------------------------------------------*/
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].typ == SPCTAB_L2)
l1[spctab[i].val] = spctab[i].pos;
else if (spctab[i].typ == SPCTAB_TRK)
{
l = spctab[i].val / 256;
j = spctab[i].val % 256;
l2[l][j].pos = spctab[i].pos;
l2[l][j].len =
l2[l][j].size = spctab[i].len;
}
/*---------------------------------------------------------------
* Write the cdevhdr, l1 table and l2 tables
*---------------------------------------------------------------*/
/* write cdevhdr */
off = CCKD_DEVHDR_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_DEVHDR_SIZE;
if ((rc = write (fd, &cdevhdr, len)) != len)
goto comp_write_error;
/* write l1 table */
off = CCKD_L1TAB_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = l1size;
if ((rc = write (fd, l1, len)) != len)
goto comp_write_error;
/* write l2 tables */
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l1[i] != 0 && l1[i] != 0xffffffff)
{
off = (off_t)l1[i];
if (lseek (fd, off, SEEK_SET) < 0)
goto comp_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_L2TAB_SIZE;
if ((rc = write (fd, l2[i], len)) != len)
goto comp_lseek_error;
}
/* truncate the file */
off = (off_t)spctab[s-1].pos;
if (off < fst.st_size)
{
ftruncate (fd, off);
if(dev->batch)
fprintf(stdout, MSG(HHC00359, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
fst.st_size - off));
else
WRMSG(HHC00359, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, fst.st_size - off);
}
else
{
if(dev->batch)
fprintf(stdout, MSG(HHC00360, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00360, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
}
/*---------------------------------------------------------------
* Return
*---------------------------------------------------------------*/
comp_return_ok:
rc = 0;
comp_return:
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if (rbuf) free(rbuf);
if (l2)
{
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l2[i])
free (l2[i]);
free (l2);
}
if (l1) free (l1);
if (spctab) free (spctab);
return rc;
/*---------------------------------------------------------------
* Error exits
*---------------------------------------------------------------*/
comp_fstat_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"fstat()", strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"fstat()", strerror(errno));
goto comp_error;
comp_lseek_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"lseek()", off, strerror(errno)));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"lseek()", off, strerror(errno));
goto comp_error;
comp_read_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"read()", off, rc < 0 ? strerror(errno) : "incomplete"));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"read()", off, rc < 0 ? strerror(errno) : "incomplete");
goto comp_error;
comp_write_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"write()", off, rc < 0 ? strerror(errno) : "incomplete"));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"write()", off, rc < 0 ? strerror(errno) : "incomplete");
goto comp_error;
comp_malloc_error:
{
char buf[64];
MSGBUF( buf, "malloc(%d)", len);
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno));
goto comp_error;
}
comp_calloc_error:
{
char buf[64];
MSGBUF( buf, "calloc(%d)", n * len);
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno));
goto comp_error;
}
comp_error:
rc = -1;
goto comp_return;
} /* cckd_comp() */
/*-------------------------------------------------------------------
* cckd_comp() space table sort
*-------------------------------------------------------------------*/
static int comp_spctab_sort(const void *a, const void *b)
{
const SPCTAB *x = a, *y = b;
if (x->typ == SPCTAB_NONE) return 1;
else if (y->typ == SPCTAB_NONE) return -1;
else if (x->typ == SPCTAB_EOF) return 1;
else if (y->typ == SPCTAB_EOF) return -1;
else if (x->pos < y->pos) return -1;
else return 1;
}
/*-------------------------------------------------------------------
* Perform check function on a compressed ckd file
*
* check levels
* -1 devhdr, cdevhdr, l1 table
* 0 devhdr, cdevhdr, l1 table, l2 tables
* 1 devhdr, cdevhdr, l1 table, l2 tables, free spaces
* 2 devhdr, cdevhdr, l1 table, l2 tables, free spaces, trkhdrs
* 3 devhdr, cdevhdr, l1 table, l2 tables, free spaces, trkimgs
* 4 devhdr, cdevhdr. Build everything else from recovery
*-------------------------------------------------------------------*/
DLL_EXPORT int cckd_chkdsk(DEVBLK *dev, int level)
{
CCKDDASD_EXT *cckd; /* -> ckd extension */
int fd; /* file descriptor */
struct stat fst; /* file status information */
int fdflags; /* file descriptor flags */
S64 maxsize; /* max cckd file size */
int ro; /* 1=file opened read-only */
int f, i, j, l, n; /* work integers */
int l1x, l2x; /* l1, l2 table indexes */
BYTE compmask[256]; /* compression byte mask
00 - supported
0x - valid, not supported
ff - invalid */
off_t off; /* file offset */
int len; /* length to read */
int rc; /* function return code */
int comp; /* trkhdr compression byte[0]*/
int cyl; /* trkhdr cyl bytes[1-2]*/
int head; /* trkhdr head bytes[3-4]*/
int trk; /* trkhdr calculated trk */
int cyls; /* number cylinders */
int heads; /* number heads/cylinder */
int trks; /* number tracks */
unsigned int trksz; /* track size */
int blks; /* number fba blocks */
int blkgrp; /* current block group nbr */
int blkgrps; /* number fba block groups */
unsigned int blkgrpsz; /* fba block group size */
int trktyp; /* track type (TRK, BLKGRP) */
int ckddasd=0; /* 1=ckd */
int fbadasd=0; /* 1= fba */
int shadow=0; /* 0xff=shadow file */
int hdrerr=0; /* non-zero: header errors */
int fsperr=0; /* 1=rebuild free space */
int comperrs=0; /* 1=unsupported comp found */
int recovery=0; /* 1=perform track recovery */
int valid; /* 1=valid trk recovered */
int l1size; /* size of l1 table */
int swapend=0; /* 1=call cckd_swapend */
U32 lopos, hipos; /* low/high file positions */
int pass; /* recovery pass number (fba)*/
int s; /* space table index */
SPCTAB *spctab=NULL; /* -> space table */
BYTE *l2errs=NULL; /* l2 error table */
BYTE *rcvtab=NULL; /* recovered tracks */
CKDDASD_DEVHDR devhdr; /* device header */
CCKD_DEVHDR cdevhdr; /* compressed device header */
CCKD_DEVHDR cdevhdr2; /* compressed device header 2*/
CCKD_L1ENT *l1=NULL; /* -> level 1 table */
CCKD_L2ENT l2ent; /* level 2 entry */
CCKD_L2ENT l2tab[256]; /* level 2 table */
CCKD_L2ENT **l2=NULL; /* -> level 2 table array */
CCKD_L2ENT empty_l2[256]; /* Empty l2 table */
CCKD_FREEBLK freeblk; /* free block */
CCKD_FREEBLK *fsp=NULL; /* free blocks (new format) */
BYTE buf[4*65536]; /* buffer */
/* Get fd */
cckd = dev->cckd_ext;
if (cckd == NULL)
fd = dev->fd;
else
fd = cckd->fd[cckd->sfn];
/* Get some file information */
if ( fstat (fd, &fst) < 0 )
goto cdsk_fstat_error;
gui_fprintf (stderr, "SIZE=%"I64_FMT"u\n", (U64) fst.st_size);
hipos = fst.st_size;
maxsize = sizeof(off_t) == 4 ? 0x7fffffffll : 0xffffffffll;
fdflags = get_file_accmode_flags(fd);
ro = (fdflags & O_RDWR) == 0;
/* Build table for compression byte test */
memset (compmask, 0xff, 256);
compmask[0] = 0;
#if defined(HAVE_LIBZ)
compmask[CCKD_COMPRESS_ZLIB] = 0;
#else
compmask[CCKD_COMPRESS_ZLIB] = 1;
#endif
#if defined(CCKD_BZIP2)
compmask[CCKD_COMPRESS_BZIP2] = 0;
#else
compmask[CCKD_COMPRESS_BZIP2] = 2;
#endif
/*---------------------------------------------------------------
* Header checks
*---------------------------------------------------------------*/
/* Read the device header */
off = 0;
if ( lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CKDDASD_DEVHDR_SIZE;
if ((rc = read (fd, &devhdr, len)) != len)
goto cdsk_read_error;
/* Device header checks */
if (memcmp(devhdr.devid, "CKD_C370", 8) == 0
|| memcmp(devhdr.devid, "CKD_S370", 8) == 0
)
ckddasd = 1;
else if (memcmp(devhdr.devid, "FBA_C370", 8) == 0
|| memcmp(devhdr.devid, "FBA_S370", 8) == 0
)
fbadasd = 1;
else
{
if(dev->batch)
fprintf(stdout, MSG(HHC00356, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00356, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
goto cdsk_error;
}
if (memcmp(devhdr.devid, "CKD_S370", 8) == 0
|| memcmp(devhdr.devid, "FBA_S370", 8) == 0
)
shadow = 0xff;
trktyp = ckddasd ? SPCTAB_TRK : SPCTAB_BLKGRP;
/* Read the cckd device header */
off = CCKD_DEVHDR_POS;
if ( lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_DEVHDR_SIZE;
if ((rc = read (fd, &cdevhdr, len)) != len)
goto cdsk_read_error;
/* Endianess check */
if ((cdevhdr.options & CCKD_BIGENDIAN) != cckd_endian())
{
if (!ro)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00357, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
cckd_endian() ? "big-endian" : "little-endian"));
else
WRMSG(HHC00357, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
cckd_endian() ? "big-endian" : "little-endian");
if (cckd_swapend (dev) < 0)
goto cdsk_error;
if (level < 0) level = 0;
swapend = 0;
}
else
swapend = 1;
cckd_swapend_chdr (&cdevhdr);
}
/* ckd checks */
if (ckddasd)
{
CKDDEV *ckd;
heads = (devhdr.heads[3] << 24)
+ (devhdr.heads[2] << 16)
+ (devhdr.heads[1] << 8)
+ (devhdr.heads[0]);
cyls = (cdevhdr.cyls[3] << 24)
+ (cdevhdr.cyls[2] << 16)
+ (cdevhdr.cyls[1] << 8)
+ (cdevhdr.cyls[0]);
trks = heads * cyls;
trksz = (devhdr.trksize[3] << 24)
+ (devhdr.trksize[2] << 16)
+ (devhdr.trksize[1] << 8)
+ (devhdr.trksize[0]);
/* ckd dasd lookup */
ckd = dasd_lookup (DASD_CKDDEV, NULL, devhdr.devtype, 0);
if (ckd == NULL)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00361, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
devhdr.devtype, cyls));
else
WRMSG(HHC00361, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
devhdr.devtype, cyls);
goto cdsk_error;
}
/* track size check */
n = sizeof(CKDDASD_TRKHDR)
+ sizeof(CKDDASD_RECHDR) + 8 /* r0 length */
+ sizeof(CKDDASD_RECHDR) + ckd->r1 /* max data length */
+ sizeof(eighthexFF);
n = ((n+511)/512)*512;
if ((int)trksz != n)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"track size", trksz, n));
else
WRMSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"track size", trksz, n);
goto cdsk_error;
}
/* number of heads check */
if (heads != ckd->heads)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"number of heads", heads, ckd->heads));
else
WRMSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"number of heads", heads, ckd->heads);
goto cdsk_error;
}
} /* if (ckddasd) */
/* fba checks */
else
{
/* Note: cyls & heads are setup for ckd type hdr checks */
blks = (cdevhdr.cyls[3] << 24)
+ (cdevhdr.cyls[2] << 16)
+ (cdevhdr.cyls[1] << 8)
+ (cdevhdr.cyls[0]);
trks = blks / CFBA_BLOCK_NUM;
if (blks % CFBA_BLOCK_NUM) trks++;
trksz = CFBA_BLOCK_SIZE + CKDDASD_TRKHDR_SIZE;
heads = 65536;
cyls = trks / heads;
if (trks % heads) cyls++;
}
/* fba variables */
blkgrps = trks;
blkgrpsz = trksz;
/* `numl1tab' check */
n = trks / 256;
if (trks % 256) n++;
if (cdevhdr.numl1tab != n && cdevhdr.numl1tab != n + 1)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"numl1tab", cdevhdr.numl1tab, n));
else
WRMSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"numl1tab", cdevhdr.numl1tab, n);
goto cdsk_error;
}
l1size = cdevhdr.numl1tab * CCKD_L1ENT_SIZE;
if (CCKD_L1TAB_POS + l1size > fst.st_size)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"file length to contain L1 table", (int)fst.st_size, (int)CCKD_L1TAB_POS + l1size));
else
WRMSG(HHC00362, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"file length to contain L1 table", (int)fst.st_size, (int)CCKD_L1TAB_POS + l1size);
goto cdsk_error;
}
/* check level 2 if SPERRS bit on */
if (!ro && level < 2 && (cdevhdr.options & CCKD_SPERRS))
{
level = 2;
if(dev->batch)
fprintf(stdout, MSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level));
else
WRMSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level);
}
/* cdevhdr inconsistencies check */
hdrerr = 0;
hdrerr |= fst.st_size != (off_t)cdevhdr.size && cdevhdr.size != cdevhdr.free ? 0x0001 : 0;
hdrerr |= cdevhdr.size != cdevhdr.used + cdevhdr.free_total ? 0x0002 : 0;
hdrerr |= cdevhdr.free_largest > cdevhdr.free_total - cdevhdr.free_imbed ? 0x0004 : 0;
hdrerr |= cdevhdr.free == 0 && cdevhdr.free_number != 0 ? 0x0008 : 0;
hdrerr |= cdevhdr.free == 0 && cdevhdr.free_total != cdevhdr.free_imbed ? 0x0010 : 0;
hdrerr |= cdevhdr.free != 0 && cdevhdr.free_total == 0 ? 0x0020 : 0;
hdrerr |= cdevhdr.free != 0 && cdevhdr.free_number == 0 ? 0x0040 : 0;
hdrerr |= cdevhdr.free_number == 0 && cdevhdr.free_total != cdevhdr.free_imbed ? 0x0080 : 0;
hdrerr |= cdevhdr.free_number != 0 && cdevhdr.free_total <= cdevhdr.free_imbed ? 0x0100 : 0;
hdrerr |= cdevhdr.free_imbed > cdevhdr.free_total ? 0x0200 : 0;
/* Additional checking if header errors */
if (hdrerr != 0)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00363, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, hdrerr));
else
WRMSG(HHC00363, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, hdrerr);
if (level < 1)
{
level = 1;
if(dev->batch)
fprintf(stdout, MSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level));
else
WRMSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level);
}
}
/* Additional checking if not properly closed */
if (level < 1 && (cdevhdr.options & CCKD_OPENED))
{
level = 1;
if(dev->batch)
fprintf(stdout, MSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level));
else
WRMSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level);
}
/* Additional checking if last opened for read/write */
if (level < 0 && (cdevhdr.options & CCKD_ORDWR))
level = 0;
/* Set check level -1 */
if (level == 0 && !dev->batch && !hdrerr
&& (cdevhdr.options & (CCKD_OPENED|CCKD_SPERRS)) == 0
&& ((cdevhdr.options & (CCKD_ORDWR)) == 0 || ro))
level = -1;
/* Build empty l2 table */
memset (&empty_l2, shadow, CCKD_L2TAB_SIZE);
if (shadow == 0 && cdevhdr.nullfmt != 0)
for (i = 0; i < 256; i++)
empty_l2[i].len = empty_l2[i].size = cdevhdr.nullfmt;
/*---------------------------------------------------------------
* read the level 1 table
*---------------------------------------------------------------*/
len = l1size;
if ((l1 = malloc (len)) == NULL)
goto cdsk_error;
off = CCKD_L1TAB_POS;
if ( lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, l1, len)) != len)
goto cdsk_read_error;
if (swapend) cckd_swapend_l1 (l1, (int)cdevhdr.numl1tab);
lopos = CCKD_L1TAB_POS + l1size;
/*---------------------------------------------------------------
* initialize the space table
*---------------------------------------------------------------*/
/* find number of non-null l1 entries */
for (i = n = 0; i < cdevhdr.numl1tab; i++)
if (l1[i] != 0 && l1[i] != 0xffffffff)
n++;
if (level >= 4) n = cdevhdr.numl1tab;
/* calculate max possible space table entries */
n = 1 + 1 + 1 // devhdr, cdevhdr, l1tab
+ n // l2tabs
+ (n * 256) // trk/blk images
+ (1 + n + (n * 256) + 1) // max possible free spaces
+ 1; // end-of-file
/* obtain the space table */
len = sizeof(SPCTAB);
if ((spctab = calloc (n, len)) == NULL)
goto cdsk_calloc_error;
/* populate the table with what we have */
s = 0;
/* devhdr */
spctab[s].typ = SPCTAB_DEVHDR;
spctab[s].val = -1;
spctab[s].pos = 0;
spctab[s].len =
spctab[s].siz = CKDDASD_DEVHDR_SIZE;
s++;
/* cdevhdr */
spctab[s].typ = SPCTAB_CDEVHDR;
spctab[s].val = -1;
spctab[s].pos = CCKD_DEVHDR_POS;
spctab[s].len =
spctab[s].siz = CCKD_DEVHDR_SIZE;
s++;
/* l1 table */
spctab[s].typ = SPCTAB_L1;
spctab[s].val = -1;
spctab[s].pos = CCKD_L1TAB_POS;
spctab[s].len =
spctab[s].siz = l1size;
s++;
/* l2 tables */
for (i = 0; i < cdevhdr.numl1tab && level < 4; i++)
{
if (l1[i] == 0 || l1[i] == 0xffffffff) continue;
spctab[s].typ = SPCTAB_L2;
spctab[s].val = i;
spctab[s].pos = l1[i];
spctab[s].len =
spctab[s].siz = CCKD_L2TAB_SIZE;
s++;
}
/* end-of-file */
spctab[s].typ = SPCTAB_EOF;
spctab[s].val = -1;
spctab[s].pos = (U32)fst.st_size;
spctab[s].len =
spctab[s].siz = 0;
s++;
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
/*---------------------------------------------------------------
* Quick return if level -1
*---------------------------------------------------------------*/
if (level < 0)
{
int err = 0;
/* check for overlaps */
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].pos + spctab[i].siz > spctab[i+1].pos)
err = 1;
/* exit if no errors */
if (!err) goto cdsk_return_ok;
}
/*---------------------------------------------------------------
* obtain the l2errs table and recovery table
*---------------------------------------------------------------*/
len = sizeof(BYTE);
n = cdevhdr.numl1tab;
if ((l2errs = calloc (n, len)) == NULL)
goto cdsk_calloc_error;
n = trks;
if ((rcvtab = calloc (n, len)) == NULL)
goto cdsk_calloc_error;
/*---------------------------------------------------------------
* Special processing for level 4 (recover everything)
*---------------------------------------------------------------*/
if (level == 4)
{
memset (l2errs, 1, cdevhdr.numl1tab);
memset (rcvtab, 1, trks);
goto cdsk_recovery;
}
/*---------------------------------------------------------------
* Read the level 2 tables
*---------------------------------------------------------------*/
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
{
if (spctab[i].typ != SPCTAB_L2
|| spctab[i].pos < lopos || spctab[i].pos > hipos)
continue;
off = spctab[i].pos;
if ( lseek (fd, off, SEEK_SET) < 0 )
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_L2TAB_SIZE;
if ((rc = read (fd, l2tab, len)) != len)
goto cdsk_read_error;
if (swapend) cckd_swapend_l2 (l2tab);
/* add trks/blkgrps to the space table */
for (j = 0; j < 256; j++)
{
if (l2tab[j].pos != 0 && l2tab[j].pos != 0xffffffff)
{
spctab[s].typ = trktyp;
spctab[s].val = spctab[i].val * 256 + j;
spctab[s].pos = l2tab[j].pos;
spctab[s].len = l2tab[j].len;
spctab[s].siz = l2tab[j].size;
s++;
}
}
}
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
/*---------------------------------------------------------------
* Consistency checks.
*
* The space table is now populated with everything but free
* space. Therefore we can infer what the free space should
* be (ie gaps between allocated spaces).
*---------------------------------------------------------------*/
lopos = CCKD_L1TAB_POS + l1size;
hipos = fst.st_size;
/* Make adjustment if new format free space is at the end */
len = spctab[s-1].pos - (spctab[s-2].pos + spctab[s-2].siz);
if (len > 0
&& cdevhdr.size == cdevhdr.free
&& cdevhdr.size + len == spctab[s-1].pos)
{
spctab[s-1].pos -= len;
hipos -= len;
}
memset( &cdevhdr2, 0, CCKD_DEVHDR_SIZE );
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
{
/* Calculate gap size */
len = spctab[i+1].pos - (spctab[i].pos + spctab[i].siz);
/* Update space statistics */
cdevhdr2.size += spctab[i].siz + len;
cdevhdr2.used += spctab[i].len;
if (len > 0)
{
cdevhdr2.free_number++;
cdevhdr2.free_total += len;
if (cdevhdr2.free_largest < (U32)len)
cdevhdr2.free_largest = (U32)len;
}
if (spctab[i].typ == trktyp)
{
cdevhdr2.free_total += spctab[i].siz - spctab[i].len;
cdevhdr2.free_imbed += spctab[i].siz - spctab[i].len;
}
/* ignore devhdr, cdevhdr and l1 (these are `out of bounds') */
if (spctab[i].typ == SPCTAB_DEVHDR
|| spctab[i].typ == SPCTAB_CDEVHDR
|| spctab[i].typ == SPCTAB_L1
)
continue;
/* check if the space is out of bounds */
valid = (off_t)spctab[i].pos >= lopos
&& (off_t)spctab[i].pos + spctab[i].siz <= hipos;
/* Overlap check */
if (len < 0 || !valid)
{
char space1[32], space2[32];
recovery = 1;
/* issue error message */
j = MSGBUF(space1, "%s", spaces[spctab[i].typ]);
if (spctab[i].val >= 0)
sprintf(space1+j, "[%d]", spctab[i].val);
j = MSGBUF(space2, "%s", spaces[spctab[i+1].typ]);
if (spctab[i+1].val >= 0)
sprintf(space2+j, "[%d]", spctab[i+1].val);
if (!valid)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00365, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
space1, spctab[i].pos, spctab[i].siz));
else
WRMSG(HHC00365, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
space1, spctab[i].pos, spctab[i].siz);
}
else
{
if(dev->batch)
fprintf(stdout, MSG(HHC00366, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
space1, spctab[i].pos, spctab[i].siz, space2, spctab[i+1].pos));
else
WRMSG(HHC00366, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
space1, spctab[i].pos, spctab[i].siz, space2, spctab[i+1].pos);
}
/* setup recovery */
if (spctab[i].typ == SPCTAB_L2)
{
l2errs[spctab[i].val] = 1;
/* Mark all tracks for the l2 for recovery */
memset (rcvtab + (spctab[i].val*256), 1, 256);
}
else if (spctab[i].typ == trktyp)
rcvtab[spctab[i].val] = 1;
if (spctab[i+1].typ == SPCTAB_L2 && valid)
{
l2errs[spctab[i+1].val] = 1;
memset (rcvtab + (spctab[i+1].val*256), 1, 256);
}
else if (spctab[i+1].typ == trktyp && valid)
rcvtab[spctab[i+1].val] = 1;
} /* if overlap or out of bounds */
/* Check image l2 entry consistency */
else if (spctab[i].typ == trktyp
&& (spctab[i].len < CKDDASD_TRKHDR_SIZE
|| spctab[i].len > spctab[i].siz
|| spctab[i].len > trksz))
{
recovery = 1;
/* issue error message */
if(dev->batch)
fprintf(stdout, MSG(HHC00367, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], spctab[i].val, spctab[i].len, spctab[i].siz));
else
WRMSG(HHC00367, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], spctab[i].val, spctab[i].len, spctab[i].siz);
/* setup recovery */
rcvtab[spctab[i].val] = 1;
} /* if inconsistent l2 */
} /* for each space */
/* remove any l2 tables or tracks in error from the space table */
for (i = 0; recovery && spctab[i].typ != SPCTAB_EOF; i++)
if ((spctab[i].typ == SPCTAB_L2 && l2errs[spctab[i].val])
|| (spctab[i].typ == trktyp && rcvtab[spctab[i].val]))
spctab[i].typ = SPCTAB_NONE;
/* overlaps are serious */
if (recovery && level < 3)
{
level = 3;
if(dev->batch)
fprintf(stdout, MSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level));
else
WRMSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level);
}
/* Rebuild free space if any errors */
if (recovery || hdrerr
|| cdevhdr.size != cdevhdr2.size
|| cdevhdr.used != cdevhdr2.used
|| cdevhdr.free_number != cdevhdr2.free_number
|| cdevhdr.free_largest != cdevhdr2.free_largest
|| cdevhdr.free_total != cdevhdr2.free_total
|| cdevhdr.free_imbed != cdevhdr2.free_imbed
)
fsperr = 1;
/*---------------------------------------------------------------
* read the free space
*---------------------------------------------------------------*/
lopos = CCKD_L1TAB_POS + l1size;
hipos = fst.st_size;
if (level >= 1 && !fsperr)
{
while (cdevhdr.free) // `while' so code can break
{
fsperr = 1; // be pessimistic
fsp = NULL;
/* Read the free space */
off = (off_t)cdevhdr.free;
len = CCKD_FREEBLK_SIZE;
if (off < lopos || off + CCKD_FREEBLK_SIZE > hipos
|| lseek (fd, off, SEEK_SET) < 0
|| (rc = read (fd, &freeblk, len)) != len)
break;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if (memcmp (&freeblk, "FREE_BLK", 8) == 0)
{
/* new format free space */
len = cdevhdr.free_number * CCKD_FREEBLK_SIZE;
if ((fsp = malloc(len)) == NULL
|| (rc = read (fd, fsp, len)) != len)
break;
for (i = 0; i < cdevhdr.free_number; i++)
{
if (swapend) cckd_swapend_free (&fsp[i]);
spctab[s].typ = SPCTAB_FREE;
spctab[s].val = -1;
spctab[s].pos = fsp[i].pos;
spctab[s].len =
spctab[s].siz = fsp[i].len;
/* Free space should be ascending */
if (spctab[s].pos < lopos
|| spctab[s].pos + spctab[s].siz > hipos)
break;
lopos = spctab[s].pos + spctab[s].siz;
s++;
} /* for each free space */
if (i >= cdevhdr.free_number)
fsperr = 0;
} /* new format free space */
else
{
/* old format free space */
off = (off_t)cdevhdr.free;
len = CCKD_FREEBLK_SIZE;
for (i = 0; i < cdevhdr.free_number; i++)
{
if (off < lopos || off > hipos) break;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, &freeblk, len)) != len)
goto cdsk_read_error;
if (swapend) cckd_swapend_free (&freeblk);
spctab[s].typ = SPCTAB_FREE;
spctab[s].val = -1;
spctab[s].pos = (U32)off;
spctab[s].len =
spctab[s].siz = freeblk.len;
s++;
lopos = off + freeblk.len;
off = (off_t)freeblk.pos;
}
if (i >= cdevhdr.free_number && freeblk.pos == 0)
fsperr = 0;
} /* if old format free space */
if (fsp) free(fsp);
fsp = NULL;
/* Check for gaps/overlaps */
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
for (i = 0; !fsperr && spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].pos + spctab[i].siz != spctab[i+1].pos)
fsperr = 1;
break;
} /* while (cdevhdr.free) */
} /* if (level >= 1 && !fsperr) */
if (fsperr)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00368, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00368, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
}
/*---------------------------------------------------------------
* Read track headers/images
*---------------------------------------------------------------*/
cdsk_space_check:
if (level >= 2)
{
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
{
if (spctab[i].typ != trktyp) continue;
/* read the header or image depending on the check level */
off = spctab[i].pos;
if ( lseek (fd, off, SEEK_SET) < 0 )
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = level < 3 ? CKDDASD_TRKHDR_SIZE : spctab[i].len;
if ((rc = read (fd, buf, len)) != len)
goto cdsk_read_error;
/* Extract header info */
comp = buf[0];
cyl = fetch_hw (buf + 1);
head = fetch_hw (buf + 3);
trk = cyl * heads + head;
/* Validate header info */
if (compmask[comp] == 0xff
|| cyl >= cyls || head >= heads
|| trk != spctab[i].val)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00369, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], spctab[i].val, off,
buf[0],buf[1],buf[2],buf[3],buf[4]));
else
WRMSG(HHC00369, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], spctab[i].val, off,
buf[0],buf[1],buf[2],buf[3],buf[4]);
/* recover this track */
rcvtab[spctab[i].val] = recovery = 1;
spctab[i].typ = SPCTAB_NONE;
/* Force level 3 checking */
if (level < 3)
{
level = 3;
if(dev->batch)
fprintf(stdout, MSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum,
dev->filename, level));
else
WRMSG(HHC00364, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename, level);
goto cdsk_space_check;
}
continue;
} /* if invalid header info */
/* Check if compression supported */
if (compmask[comp])
{
comperrs = 1;
if(dev->batch)
fprintf(stdout, MSG(HHC00370, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], trk, comps[compmask[comp]]));
else
WRMSG(HHC00370, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], trk, comps[compmask[comp]]);
continue;
}
/* Validate the space if check level 3 */
if (level > 2)
{
if (!cdsk_valid_trk (trk, buf, heads, len))
{
if(dev->batch)
fprintf(stdout, MSG(HHC00371, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], trk, off, len));
else
WRMSG(HHC00371, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], trk, off, len);
/* recover this track */
rcvtab[trk] = recovery = 1;
spctab[i].typ = SPCTAB_NONE;
} /* if invalid space */
else
rcvtab[trk] = 0;
} /* if level > 2 */
} /* for each space */
} /* if (level >= 2) */
/*---------------------------------------------------------------
* Recovery
*---------------------------------------------------------------*/
cdsk_recovery:
if (recovery || level == 4)
{
U32 flen, fpos;
/*-----------------------------------------------------------
* Phase 1 -- recover trk/blkgrp images
*-----------------------------------------------------------*/
/*
* Reset the end-of-file pos to the file size
* It might have been changed if new format free space
* occurred at the end of the file.
*/
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
while (spctab[s-1].typ == SPCTAB_NONE) s--;
spctab[s-1].pos = fst.st_size;
/* count number tracks to be recovered */
for (i = n = 0; i < trks; i++)
if (rcvtab[i] == 1)
n++;
/*-----------------------------------------------------------
* ckd recovery
*-----------------------------------------------------------*/
if (ckddasd)
{
/* recovery loop */
s = cdsk_build_free_space (spctab, s);
for (f = 0; spctab[f].typ != SPCTAB_EOF && n; )
{
/* next free space if too small */
if (spctab[f].typ != SPCTAB_FREE
|| spctab[f].siz <= CKDDASD_TRKHDR_SIZE+8)
{
for (f = f + 1; spctab[f].typ != SPCTAB_EOF; f++)
if (spctab[f].typ == SPCTAB_FREE)
break;
continue;
}
fpos = spctab[f].pos;
flen = spctab[f].siz;
/* length to read */
len = flen < sizeof(buf) ? flen : sizeof(buf);
/* read the free space */
off = (off_t)fpos;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, buf, len)) != len)
goto cdsk_read_error;
/* Scan the space for a trkhdr */
for (i = 0; i < len - (CKDDASD_TRKHDR_SIZE+8); i++)
{
/* Check compression byte */
if (compmask[buf[i]])
continue;
/* Fetch possible trkhdr */
comp = buf[i];
cyl = fetch_hw (buf + i + 1);
head = fetch_hw (buf + i + 3);
trk = cyl * heads + head;
/* Validate possible trkhdr */
if (cyl >= cyls || head >= heads || rcvtab[trk] != 1)
continue;
/* Quick validation for compress none */
if (comp == CCKD_COMPRESS_NONE
&& (fetch_hw (buf + i + 5) != cyl // r0 cyl
|| fetch_hw (buf + i + 7) != head // r0 head
|| buf[i + 9] != 0 // r0 record
|| buf[i + 10] != 0 // r0 key length
|| fetch_hw (buf + i + 11) != 8 // r0 data length
)
)
continue;
/* Quick validation for zlib */
else if (comp == CCKD_COMPRESS_ZLIB
&& fetch_hw(buf + i + 5) % 31 != 0)
continue;
/* Quick validation for bzip2 */
else if (comp == CCKD_COMPRESS_BZIP2
&& (buf[i+5] != 'B' || buf[i+6] != 'Z'))
continue;
/*
* If we are in `borrowed space' then start over
* with the current position at the beginning
*/
if (flen > (U32)len && i > len - (int)trksz)
break;
/* Checks for comp none */
if (comp == CCKD_COMPRESS_NONE)
{
l = len - i;
if ((l = cdsk_valid_trk (trk, buf+i, heads, -l)))
goto cdsk_ckd_recover;
else
continue;
}
/* Check short `length' */
if (flen == (U32)len && (l = len - i) <= 1024)
{
if (cdsk_valid_trk (trk, buf+i, heads, l))
{
while (cdsk_valid_trk (trk, buf+i, heads, --l));
l++;
goto cdsk_ckd_recover;
}
}
/* Scan for next trkhdr */
for (j = i + CKDDASD_TRKHDR_SIZE+8;
j <= len - (CKDDASD_TRKHDR_SIZE+8);
j++)
{
if (j - i > (int)trksz) break;
if (compmask[buf[j]] != 0
|| fetch_hw(buf+j+1) >= cyls
|| fetch_hw(buf+j+3) >= heads)
continue;
/* check uncompressed hdr */
if (buf[j] == CCKD_COMPRESS_NONE
&& (fetch_hw (buf+j+5) != fetch_hw(buf+j+1)
|| fetch_hw (buf+j+7) != fetch_hw(buf+j+3)
|| buf[j+9] != 0 || buf[j+10] != 0
|| fetch_hw(buf+j+11) != 8))
continue;
/* check zlib compressed header */
else if (buf[j] == CCKD_COMPRESS_ZLIB
&& fetch_hw(buf + j + 5) % 31 != 0)
continue;
/* check bzip2 compressed header */
else if (buf[j] == CCKD_COMPRESS_BZIP2
&& (buf[j+5] != 'B' || buf[j+6] != 'Z'))
continue;
/* check to possible trkhdr */
l = j - i;
if (cdsk_valid_trk (trk, buf+i, heads, l))
{
#if 0
while (cdsk_valid_trk (trk, buf+i, heads, --l));
l++;
#endif
goto cdsk_ckd_recover;
}
} /* scan for next trkhdr */
/* Check `length' */
if (flen == (U32)len && (l = len - i) <= (int)trksz)
{
if (cdsk_valid_trk (trk, buf+i, heads, l))
{
while (cdsk_valid_trk (trk, buf+i, heads, --l));
l++;
goto cdsk_ckd_recover;
}
}
/* Scan all lengths */
for (l = CKDDASD_TRKHDR_SIZE+8; i + l <= len; l++)
{
if (l > (int)trksz)
break;
if (cdsk_valid_trk (trk, buf+i, heads, l))
goto cdsk_ckd_recover;
} /* for all lengths */
continue;
cdsk_ckd_recover:
if(dev->batch)
fprintf(stdout, MSG(HHC00372, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], trk, off + i, l));
else
WRMSG(HHC00372, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], trk, off + i, l);
n--;
rcvtab[trk] = 2;
/* add recovered track to the space table */
spctab[s].typ = trktyp;
spctab[s].val = trk;
spctab[s].pos = fpos + i;
spctab[s].len =
spctab[s].siz = l;
s++;
/*
* adjust `i' knowing it will be incremented
* in the `for' loop above.
*/
i += l - 1;
} /* for each byte in the free space */
/* Adjust the free space for what we processed */
spctab[f].pos += i;
spctab[f].len -= i;
spctab[f].siz -= i;
} /* for each free space */
} /* if ckddasd */
/*-----------------------------------------------------------
* fba recovery
*-----------------------------------------------------------*/
/*
* FBA blkgrps are harder to recover than CKD tracks because
* there is not any information within the blkgrp itself to
* validate (unlike a track, which has count fields that
* terminate in an end-of-track marker).
*
* On the first pass we recover all compressed blkgrps since
* these are readily validated (they must uncompress to a
* certain size, CFBA_BLOCK_SIZE+CKDDASD_TRKHDR_SIZE). We
* also recover uncompressed blkgrps if they are followed by
* a valid trkhdr (and don't occur to close to the beginning
* of the file).
*
* On the second pass we recover all uncompressed blkgrps
* that weren't recovered in the first pass. The only
* criteria is that the compression byte is zero and the
* 4 byte blkgrp number is in range and there are at least
* CFBA_BLOCK_SIZE bytes following.
*/
for (pass = 0; fbadasd && pass < 2; pass++)
{
lopos = CCKD_L1TAB_POS + (cdevhdr.numl1tab * 4);
if (pass == 0)
lopos += (cdevhdr.numl1tab * CCKD_L2TAB_SIZE);
/* recovery loop */
s = cdsk_build_free_space (spctab, s);
for (f = 0; spctab[f].typ != SPCTAB_EOF && n > 0; )
{
U32 flen, fpos;
/* next free space if too small */
if (spctab[f].typ != SPCTAB_FREE
|| spctab[f].siz <= CKDDASD_TRKHDR_SIZE+8
|| (pass == 1 && spctab[f].siz < blkgrpsz))
{
for (f = f + 1; spctab[f].typ != SPCTAB_EOF; f++)
if (spctab[f].typ == SPCTAB_FREE)
break;
continue;
}
fpos = spctab[f].pos;
flen = spctab[f].siz;
/*
* calculate length to read
* if flen > len then we only read part of the space
*/
len = flen < sizeof(buf) ? flen : sizeof(buf);
/* read the free space */
off = (off_t)fpos;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = read (fd, buf, len)) != len)
goto cdsk_read_error;
/* Scan the space */
for (i = 0; i < len - (CKDDASD_TRKHDR_SIZE+8); i++)
{
/* For pass 1 the size left must be at least blkgrpsz */
if (pass == 1 && len - i < (int)blkgrpsz)
break;
/* Check compression byte */
if ((pass == 0 && compmask[buf[i]])
|| (pass == 1 && buf[i] != CCKD_COMPRESS_NONE))
continue;
/* Fetch possible trkhdr */
comp = buf[i];
blkgrp = fetch_fw (buf + i + 1);
/* Validate possible trkhdr */
if (blkgrp < 0 || blkgrp >= blkgrps
|| rcvtab[blkgrp] != 1)
continue;
/* Validation for compress none */
if (comp == CCKD_COMPRESS_NONE
&& flen == (U32)len && len - i < (int)blkgrpsz)
continue;
/* Quick validation for zlib */
else if (comp == CCKD_COMPRESS_ZLIB
&& fetch_hw(buf + i + 5) % 31 != 0)
continue;
/* Quick validation for bzip2 */
else if (comp == CCKD_COMPRESS_BZIP2
&& (buf[i+5] != 'B' || buf[i+6] != 'Z'))
continue;
/*
* If we are in `borrowed space' then start over
* with the current position at the beginning
*/
if (flen > (U32)len && i > len - (int)blkgrpsz)
break;
/* Checks for comp none */
if (comp == CCKD_COMPRESS_NONE)
{
l = blkgrpsz;
if (len - i < (int)blkgrpsz || fpos + i < lopos)
continue;
if (len - i == (int)blkgrpsz && flen == (U32)len)
goto cdsk_fba_recover;
/* Pass 0 checks */
if (pass == 0
&& (len - i - l < CKDDASD_TRKHDR_SIZE+8
|| compmask[buf[i+l]]
|| fetch_fw (buf+i+l+1) >= (unsigned int)blkgrps)
)
continue;
goto cdsk_fba_recover;
}
/* The tests below are for pass 0 only */
if (pass == 1)
continue;
/* Check short `length' */
if (flen == (U32)len && (l = len - i) <= 1024)
{
if (cdsk_valid_trk (blkgrp, buf+i, heads, l))
{
while (cdsk_valid_trk (blkgrp, buf+i, heads, --l));
l++;
goto cdsk_fba_recover;
}
}
/* Scan for next trkhdr */
for (j = i + CKDDASD_TRKHDR_SIZE+8;
j <= len - (CKDDASD_TRKHDR_SIZE+8);
j++)
{
if (j - i > (int)blkgrpsz) break;
if (compmask[buf[j]] != 0
|| fetch_fw(buf+j+1) >= (unsigned int)blkgrps)
continue;
/* check zlib compressed header */
if (buf[j] == CCKD_COMPRESS_ZLIB
&& fetch_hw(buf + j + 5) % 31 != 0)
continue;
/* check bzip2 compressed header */
else if (buf[j] == CCKD_COMPRESS_BZIP2
&& (buf[j+5] != 'B' || buf[j+6] != 'Z'))
continue;
/* check to possible trkhdr */
l = j - i;
if (cdsk_valid_trk (blkgrp, buf+i, heads, l))
{
#if 0
while (cdsk_valid_trk (blkgrp, buf+i, heads, --l));
l++;
#endif
goto cdsk_fba_recover;
}
} /* scan for next trkhdr */
/* Check `length' */
l = len - i;
if (flen == (U32)len && l <= (int)blkgrpsz)
{
if (cdsk_valid_trk (blkgrp, buf+i, heads, l))
{
while (cdsk_valid_trk (blkgrp, buf+i, heads, --l));
l++;
goto cdsk_fba_recover;
}
}
/* Scan all lengths */
for (l = CKDDASD_TRKHDR_SIZE+8; i + l <= len; l++)
{
if (l > (int)blkgrpsz)
break;
if (cdsk_valid_trk (blkgrp, buf+i, heads, l))
goto cdsk_fba_recover;
} /* for all lengths */
continue;
cdsk_fba_recover:
if(dev->batch)
fprintf(stdout, MSG(HHC00372, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], blkgrp, off + i, l));
else
WRMSG(HHC00372, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
spaces[trktyp], blkgrp, off + i, l);
n--;
rcvtab[blkgrp] = 2;
/* Enable recovery of comp 0 blkgrps for pass 0 */
if (fpos + i < lopos)
lopos = fpos + i;
/* add recovered block group to the space table */
spctab[s].typ = trktyp;
spctab[s].val = blkgrp;
spctab[s].pos = fpos + i;
spctab[s].len =
spctab[s].siz = l;
s++;
/*
* adjust `i' knowing it will be incremented
* in the `for' loop above.
*/
i += l - 1;
} /* for each byte in the free space */
/* Adjust the free space for what we processed */
spctab[f].pos += i;
spctab[f].len -= i;
spctab[f].siz -= i;
} /* for each free space */
} /* if fbadasd */
for (i = n = 0; i < trks; i++)
if (rcvtab[i] == 2)
n++;
if(dev->batch)
fprintf(stdout, MSG(HHC00373, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
n, spaces[trktyp]));
else
WRMSG(HHC00373, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
n, spaces[trktyp]);
/*-----------------------------------------------------------
* Phase 2 -- rebuild affected l2 tables
*-----------------------------------------------------------*/
/*
* Make sure there's at least one non-zero `rcvtab' entry
* for l2 tables in `l2errs'. Space validation may have
* turned off all `rcvtab' entries for an l2.
*/
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l2errs[i])
rcvtab[i*256] = 1;
/* Get storage for the l2 table array */
n = cdevhdr.numl1tab;
len = sizeof(void *);
if ((l2 = calloc (n, len)) == NULL)
goto cdsk_calloc_error;
/* Get storage for the rebuilt l2 tables */
len = CCKD_L2TAB_SIZE;
for (i = 0; i < trks; i++)
{
l1x = i / 256;
if (rcvtab[i] != 0 && l2[l1x] == NULL)
{
if ((l2[l1x] = malloc (len)) == NULL)
goto cdsk_malloc_error;
l1[l1x] = shadow ? 0xffffffff : 0;
memcpy (l2[l1x], &empty_l2, len);
}
}
/* Rebuild the l2 tables */
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
{
if (spctab[i].typ == SPCTAB_L2 && l2[spctab[i].val])
spctab[i].typ = SPCTAB_NONE;
else if (spctab[i].typ == trktyp && l2[spctab[i].val/256])
{
l1x = spctab[i].val / 256;
l2x = spctab[i].val % 256;
l2[l1x][l2x].pos = spctab[i].pos;
l2[l1x][l2x].len = spctab[i].len;
l2[l1x][l2x].size = spctab[i].siz;
}
} /* for each space */
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
while (spctab[s-1].typ == SPCTAB_NONE) s--;
/* Look for empty l2 tables */
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l2[i] != NULL
&& memcmp (l2[i], &empty_l2, CCKD_L2TAB_SIZE) == 0)
{
free (l2[i]);
l2[i] = NULL;
}
/*
* `s-1' indexes the SPCTAB_EOF space table entry.
* Set its `pos' to the maximum allowed value to ensure
* there will be free space for the rebuilt l2 tables.
*/
spctab[s-1].pos = (U32)maxsize;
/* Build the free space */
s = cdsk_build_free_space (spctab, s);
/* Find space for the rebuilt l2 tables */
for (i = j = 0; i < cdevhdr.numl1tab; i++)
{
if (l2[i] == NULL) continue;
/* find a free space */
for ( ; spctab[j].typ != SPCTAB_EOF; j++)
if (spctab[j].typ == SPCTAB_FREE
&& spctab[j].siz >= CCKD_L2TAB_SIZE)
break;
/* weird error if no space */
if (spctab[j].typ == SPCTAB_EOF)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00374, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00374, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
goto cdsk_error;
}
/* add l2 space */
l1[i] = spctab[j].pos;
spctab[s].typ = SPCTAB_L2;
spctab[s].val = i;
spctab[s].pos = spctab[j].pos;
spctab[s].len =
spctab[s].siz = CCKD_L2TAB_SIZE;
s++;
/* adjust the free space */
spctab[j].pos += CCKD_L2TAB_SIZE;
spctab[j].len -= CCKD_L2TAB_SIZE;
spctab[j].siz -= CCKD_L2TAB_SIZE;
} /* for each l2 table */
/*-----------------------------------------------------------
* Phase 3 -- write l1 and l2 tables
*-----------------------------------------------------------*/
if (ro)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00375, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"file opened read-only"));
else
WRMSG(HHC00375, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"file opened read-only");
goto cdsk_error;
}
if (comperrs)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00375, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"missing compression"));
else
WRMSG(HHC00375, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"missing compression");
goto cdsk_error;
}
/* Write the l1 table */
off = CCKD_L1TAB_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = l1size;
if ((rc = write (fd, l1, len)) != len)
goto cdsk_write_error;
/* Write l2 tables */
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
{
l1x = spctab[i].val;
if (spctab[i].typ != SPCTAB_L2 || l2[l1x] == NULL)
continue;
off = (off_t)l1[l1x];
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_L2TAB_SIZE;
if ((rc = write (fd, l2[l1x], len)) != len)
goto cdsk_write_error;
free (l2[l1x]);
l2[l1x] = NULL;
} /* for each space */
/* Free recovery related storage */
if (l2)
{
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l2[i])
free (l2[i]);
free (l2); l2 = NULL;
}
free (l2errs); l2errs = NULL;
free (rcvtab); rcvtab = NULL;
/* Ensure we do free space recovery */
fsperr = 1;
} /* if (recovery || level >= 4) */
/*---------------------------------------------------------------
* Rebuild free space
*---------------------------------------------------------------*/
if (fsperr && ro)
{
if(dev->batch)
fprintf(stdout, MSG(HHC00376, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00376, "W", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
}
else if (fsperr)
{
/*-----------------------------------------------------------
* Phase 1 -- build the free space
* make sure the last space isn't free space and
* that each free space is long enough (8 bytes).
*-----------------------------------------------------------*/
cdsk_fsperr_retry:
s = cdsk_build_free_space (spctab, s);
/*
* spctab[s-1] is the SPCTAB_EOF entry.
* if spctab[s-2] is SPCTAB_FREE then discard it
*/
if (spctab[s-2].typ == SPCTAB_FREE)
{
spctab[s-1].typ = SPCTAB_NONE;
spctab[s-2].typ = SPCTAB_EOF;
spctab[s-2].val = -1;
spctab[s-2].len =
spctab[s-2].siz = 0;
s--;
}
/*
* Check for short free spaces.
* If found, shift left until the next free space or eof.
*/
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].typ == SPCTAB_FREE
&& spctab[i].siz < CCKD_FREEBLK_SIZE)
break;
if (spctab[i].typ != SPCTAB_EOF)
{
/* Shift following space left */
l = spctab[i++].siz;
while (spctab[i].typ != SPCTAB_FREE && spctab[i].typ != SPCTAB_EOF)
{
/* Read the space and write shifted to the left */
off = (off_t)spctab[i].pos;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = spctab[i].siz;
if ((rc = read (fd, buf, len)) != len)
goto cdsk_read_error;
off -= l;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, buf, len)) != len)
goto cdsk_write_error;
spctab[i].pos -= l;
/* Update the l2 or l1 table entry */
if (spctab[i].typ == trktyp)
{
l1x = spctab[i].val/256;
l2x = spctab[i].val%256;
off = (off_t)l1[l1x] + l2x * CCKD_L2ENT_SIZE;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_L2ENT_SIZE;
if ((rc = read (fd, &l2ent, len)) != len)
goto cdsk_read_error;
l2ent.pos -= l;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, &l2ent, len)) != len)
goto cdsk_write_error;
} /* trk/blkgrp relocated */
else if (spctab[i].typ == SPCTAB_L2)
l1[spctab[i].val] -= l;
i++;
} /* while not FREE space or EOF */
goto cdsk_fsperr_retry;
} /* if short free space found */
/*-----------------------------------------------------------
* Phase 2 -- rebuild free space statistics
*-----------------------------------------------------------*/
cdevhdr.vrm[0] = CCKD_VERSION;
cdevhdr.vrm[1] = CCKD_RELEASE;
cdevhdr.vrm[2] = CCKD_MODLVL;
cdevhdr.size = cdevhdr.used = cdevhdr.free =
cdevhdr.free_total = cdevhdr.free_largest =
cdevhdr.free_number = cdevhdr.free_imbed = 0;
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].typ == SPCTAB_FREE)
{
cdevhdr.size += spctab[i].siz;
if (spctab[i].siz > cdevhdr.free_largest)
cdevhdr.free_largest = spctab[i].siz;
cdevhdr.free_total += spctab[i].siz;
cdevhdr.free_number++;
}
else
{
cdevhdr.size += spctab[i].siz;
cdevhdr.used += spctab[i].len;
cdevhdr.free_total += spctab[i].siz - spctab[i].len;
cdevhdr.free_imbed += spctab[i].siz - spctab[i].len;
}
/*-----------------------------------------------------------
* Phase 3 -- write the free space
*-----------------------------------------------------------*/
if (cdevhdr.free_number)
{
/* size needed for new format free space */
len = (cdevhdr.free_number+1) * CCKD_FREEBLK_SIZE;
/* look for existing free space to fit new format free space */
for (i = off = 0; !off && spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].typ == SPCTAB_FREE && len <= (int)spctab[i].siz)
off = (off_t)spctab[i].pos;
/* if no applicable space see if we can append to the file */
if (!off && maxsize - cdevhdr.size >= len)
off = (off_t)cdevhdr.size;
/* get free space buffer */
if (off && (fsp = malloc (len)) == NULL)
off = 0;
if (off)
{
/* new format free space */
memcpy (fsp, "FREE_BLK", 8);
for (i = 0, j = 1; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].typ == SPCTAB_FREE)
{
fsp[j].pos = spctab[i].pos;
fsp[j++].len = spctab[i].siz;
}
/* Write the free space */
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if ((rc = write (fd, fsp, len)) != len)
goto cdsk_write_error;
cdevhdr.free = (U32)off;
free (fsp);
fsp = NULL;
} /* new format free space */
else
{
/* old format free space */
len = CCKD_FREEBLK_SIZE;
for (i = 0; spctab[i].typ != SPCTAB_FREE; i++);
cdevhdr.free = spctab[i].pos;
off = (off_t)spctab[i].pos;
freeblk.pos = 0;
freeblk.len = spctab[i].siz;
for (i = i + 1; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].typ == SPCTAB_FREE)
{
freeblk.pos = spctab[i].pos;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if (write (fd, &freeblk, len) != len)
goto cdsk_write_error;
off = (off_t)spctab[i].pos;
freeblk.pos = 0;
freeblk.len = spctab[i].len;
}
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
if (write (fd, &freeblk, len) != len)
goto cdsk_write_error;
} /* old format free space */
} /* if (cdevhdr.free_number) */
/* Write cdevhdr and l1 table */
off = CCKD_DEVHDR_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = CCKD_DEVHDR_SIZE;
if (write (fd, &cdevhdr, len) != len)
goto cdsk_write_error;
off = CCKD_L1TAB_POS;
if (lseek (fd, off, SEEK_SET) < 0)
goto cdsk_lseek_error;
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
len = l1size;
if (write (fd, l1, len) != len)
goto cdsk_write_error;
/* Truncate the file */
off = (off_t)cdevhdr.size;
if (cdevhdr.free == cdevhdr.size)
off += (cdevhdr.free_number+1) * CCKD_FREEBLK_SIZE;
rc = ftruncate (fd, off);
if(dev->batch)
fprintf(stdout, MSG(HHC00377, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename));
else
WRMSG(HHC00377, "I", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename);
} /* if (fsperr) */
/*---------------------------------------------------------------
* Return
*---------------------------------------------------------------*/
cdsk_return_ok:
rc = recovery ? 2 : fsperr ? 1 : 0;
if (!ro && (cdevhdr.options & (CCKD_ORDWR|CCKD_OPENED|CCKD_SPERRS)))
{
/*
* Leave the ORDWR bit on for now. This will prevent
* old-format free space releases from doing a -1 check
* on a file that has new-format free space
*/
#if 0
cdevhdr.options &= ~(CCKD_ORDWR|CCKD_OPENED|CCKD_SPERRS);
#else
cdevhdr.options &= ~(CCKD_OPENED|CCKD_SPERRS);
#endif
/* Set version.release.modlvl */
cdevhdr.vrm[0] = CCKD_VERSION;
cdevhdr.vrm[1] = CCKD_RELEASE;
cdevhdr.vrm[2] = CCKD_MODLVL;
off = CCKD_DEVHDR_POS;
if (lseek (fd, CCKD_DEVHDR_POS, SEEK_SET) >= 0)
write (fd, &cdevhdr, CCKD_DEVHDR_SIZE);
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
}
cdsk_return:
gui_fprintf (stderr, "POS=%"I64_FMT"u\n", (U64) lseek( fd, 0, SEEK_CUR ));
/* free all space */
if (l1) free (l1);
if (spctab) free (spctab);
if (l2errs) free (l2errs);
if (rcvtab) free (rcvtab);
if (fsp) free (fsp);
if (l2)
{
for (i = 0; i < cdevhdr.numl1tab; i++)
if (l2[i]) free (l2[i]);
free (l2);
}
return rc;
/*---------------------------------------------------------------
* Error exits
*---------------------------------------------------------------*/
cdsk_fstat_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"fstat()", strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"fstat()", strerror(errno));
goto cdsk_error;
cdsk_lseek_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"lseek()", off, strerror(errno)));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"lseek()", off, strerror(errno));
goto cdsk_error;
cdsk_read_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"read()", off, rc < 0 ? strerror(errno) : "incomplete"));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"read()", off, rc < 0 ? strerror(errno) : "incomplete");
goto cdsk_error;
cdsk_write_error:
if(dev->batch)
fprintf(stdout, MSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"write()", off, rc < 0 ? strerror(errno) : "incomplete"));
else
WRMSG(HHC00355, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
"write()", off, rc < 0 ? strerror(errno) : "incomplete");
goto cdsk_error;
cdsk_malloc_error:
{
char buf[64];
MSGBUF( buf, "malloc(%d)", len);
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno));
}
goto cdsk_error;
cdsk_calloc_error:
{
char buf[64];
MSGBUF( buf, "calloc(%d)", n * len);
if(dev->batch)
fprintf(stdout, MSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno)));
else
WRMSG(HHC00354, "E", SSID_TO_LCSS(dev->ssid), dev->devnum, dev->filename,
buf, strerror(errno));
}
goto cdsk_error;
cdsk_error:
rc = -1;
goto cdsk_return;
} /* end function cckd_chkdsk */
/*-------------------------------------------------------------------
* cckd_chkdsk() space table sort
*-------------------------------------------------------------------*/
static int cdsk_spctab_sort(const void *a, const void *b)
{
const SPCTAB *x = a, *y = b;
if (x->typ == SPCTAB_NONE) return 1;
else if (y->typ == SPCTAB_NONE) return -1;
else if (x->typ == SPCTAB_EOF) return 1;
else if (y->typ == SPCTAB_EOF) return -1;
else if (x->pos < y->pos) return -1;
else return 1;
} /* end function cdsk_spctab_sort */
/*-------------------------------------------------------------------*/
/* Build free space in the space table */
/*-------------------------------------------------------------------*/
static int cdsk_build_free_space(SPCTAB *spctab, int s)
{
int i;
for (i = 0; i < s; i++)
if (spctab[i].typ == SPCTAB_FREE)
spctab[i].typ = SPCTAB_NONE;
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
while (spctab[s-1].typ == SPCTAB_NONE) s--;
for (i = 0; spctab[i].typ != SPCTAB_EOF; i++)
if (spctab[i].pos + spctab[i].siz < spctab[i+1].pos)
{
spctab[s].typ = SPCTAB_FREE;
spctab[s].val = -1;
spctab[s].pos = spctab[i].pos + spctab[i].siz;
spctab[s].len =
spctab[s].siz = spctab[i+1].pos - spctab[s].pos;
s++;
}
qsort (spctab, s, sizeof(SPCTAB), cdsk_spctab_sort);
return s;
}
/*-------------------------------------------------------------------*/
/* Validate a track image */
/* */
/* If `len' is negative and compression is CCKD_COMPRESS_NONE then */
/* `len' indicates a buffer size containing the track image and the */
/* value returned is the actual track image length */
/*-------------------------------------------------------------------*/
static int cdsk_valid_trk (int trk, BYTE *buf, int heads, int len)
{
int i; /* Index */
int len2; /* Positive `len' */
int kl, dl; /* Key/Data lengths */
BYTE *bufp; /* Buffer pointer */
int bufl; /* Buffer length */
#ifdef HAVE_LIBZ
uLongf zlen;
#endif
#ifdef CCKD_BZIP2
unsigned int bz2len;
#endif
#if defined(HAVE_LIBZ) || defined(CCKD_BZIP2)
int rc; /* Return code */
BYTE buf2[65536]; /* Uncompressed buffer */
#endif
/* Negative len only allowed for comp none */
len2 = len > 0 ? len : -len;
if (len2 < CKDDASD_TRKHDR_SIZE + 8)
return 0;
/* Uncompress the track/block image */
switch (buf[0]) {
case CCKD_COMPRESS_NONE:
bufp = buf;
bufl = len2;
break;
#ifdef HAVE_LIBZ
case CCKD_COMPRESS_ZLIB:
if (len < 0) return 0;
bufp = (BYTE *)buf2;
memcpy (buf2, buf, CKDDASD_TRKHDR_SIZE);
zlen = sizeof(buf2) - CKDDASD_TRKHDR_SIZE;
rc = uncompress (buf2 + CKDDASD_TRKHDR_SIZE, &zlen,
buf + CKDDASD_TRKHDR_SIZE, len - CKDDASD_TRKHDR_SIZE);
if (rc != Z_OK)
return 0;
bufl = (int)zlen + CKDDASD_TRKHDR_SIZE;
break;
#endif
#ifdef CCKD_BZIP2
case CCKD_COMPRESS_BZIP2:
if (len < 0) return 0;
bufp = (BYTE *)buf2;
memcpy (buf2, buf, CKDDASD_TRKHDR_SIZE);
bz2len = sizeof(buf2) - CKDDASD_TRKHDR_SIZE;
rc = BZ2_bzBuffToBuffDecompress ( (char *)&buf2[CKDDASD_TRKHDR_SIZE], &bz2len,
(char *)&buf[CKDDASD_TRKHDR_SIZE], len - CKDDASD_TRKHDR_SIZE, 0, 0);
if (rc != BZ_OK)
return 0;
bufl = (int)bz2len + CKDDASD_TRKHDR_SIZE;
break;
#endif
default:
return 0;
} /* switch (buf[0]) */
/* fba check */
if (heads == 65536)
{
if (bufl != CFBA_BLOCK_SIZE + CKDDASD_TRKHDR_SIZE)
return 0;
else
return len > 0 ? len : bufl;
}
/* Check length */
if (bufl <= 5 + 8 + 8 + 8 + 8) return 0;
/* Check ha */
if (fetch_hw(bufp + 1) != trk / heads
|| fetch_hw(bufp + 3) != trk % heads)
return 0;
/* Check r0 */
if (fetch_hw(bufp + 1) != fetch_hw(bufp + 5)
|| fetch_hw(bufp + 3) != fetch_hw(bufp + 7)
|| bufp[9] != 0 || bufp[10] != 0
|| fetch_hw(bufp +11) != 8)
return 0;
/* Check user records */
for (i = 21; i < bufl - 8; i += 8 + kl + dl)
{
if (fetch_hw(bufp + i + 2) >= heads || bufp[i + 4] == 0)
break;
kl = bufp[i + 5];
dl = fetch_hw(bufp + i + 6);
}
if (len < 0) bufl = i + 8;
if (i != bufl - 8 || memcmp(bufp + i, eighthexFF, 8))
return 0;
return len > 0 ? len : bufl;
} /* end function cdsk_valid_trk */