899 lines
22 KiB
C
899 lines
22 KiB
C
/* @(#)isovfy.c 1.50 15/12/09 joerg */
|
|
#include <schily/mconfig.h>
|
|
#ifndef lint
|
|
static UConst char sccsid[] =
|
|
"@(#)isovfy.c 1.50 15/12/09 joerg";
|
|
#endif
|
|
/*
|
|
* File isovfy.c - verify consistency of iso9660 filesystem.
|
|
*
|
|
*
|
|
* Written by Eric Youngdale (1993).
|
|
*
|
|
* Copyright 1993 Yggdrasil Computing, Incorporated
|
|
* Copyright (c) 1999-2015 J. Schilling
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with
|
|
* this program; see the file COPYING. If not, write to the Free Software
|
|
* Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include <schily/mconfig.h>
|
|
#include <schily/stdlib.h>
|
|
#include <schily/unistd.h>
|
|
#include <schily/string.h>
|
|
#include <schily/utypes.h>
|
|
|
|
#include <schily/stdio.h>
|
|
#include <schily/standard.h>
|
|
#include <schily/signal.h>
|
|
#include <schily/schily.h>
|
|
#include <schily/nlsdefs.h>
|
|
|
|
#include "../iso9660.h"
|
|
#include "../rock.h"
|
|
#include "../scsi.h"
|
|
#include "cdrdeflt.h"
|
|
#include "../../cdrecord/version.h"
|
|
|
|
/*
|
|
* XXX JS: Some structures have odd lengths!
|
|
* Some compilers (e.g. on Sun3/mc68020) padd the structures to even length.
|
|
* For this reason, we cannot use sizeof (struct iso_path_table) or
|
|
* sizeof (struct iso_directory_record) to compute on disk sizes.
|
|
* Instead, we use offsetof(..., name) and add the name size.
|
|
* See iso9660.h
|
|
*/
|
|
#ifndef offsetof
|
|
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
|
|
#endif
|
|
|
|
/*
|
|
* Note: always use these macros to avoid problems.
|
|
*
|
|
* ISO_ROUND_UP(X) may cause an integer overflow and thus give
|
|
* incorrect results. So avoid it if possible.
|
|
*
|
|
* ISO_BLOCKS(X) is overflow safe. Prefer this when ever it is possible.
|
|
*/
|
|
#define SECTOR_SIZE (2048)
|
|
#define ISO_ROUND_UP(X) (((X) + (SECTOR_SIZE - 1)) & ~(SECTOR_SIZE - 1))
|
|
#define ISO_BLOCKS(X) (((X) / SECTOR_SIZE) + (((X)%SECTOR_SIZE)?1:0))
|
|
|
|
#define infile in_image
|
|
EXPORT FILE *infile = NULL;
|
|
EXPORT BOOL ignerr = FALSE;
|
|
LOCAL int su_version;
|
|
LOCAL int rr_version;
|
|
LOCAL int aa_version;
|
|
LOCAL int use_rock = TRUE;
|
|
LOCAL int blocksize;
|
|
|
|
#define PAGE sizeof (buffer)
|
|
|
|
struct diraddr {
|
|
struct diraddr *next;
|
|
int addr;
|
|
};
|
|
|
|
LOCAL int isonum_721 __PR((char * p));
|
|
LOCAL int isonum_723 __PR((char * p));
|
|
LOCAL int isonum_711 __PR((char * p));
|
|
LOCAL int isonum_731 __PR((char * p));
|
|
#if 0
|
|
LOCAL int isonum_722 __PR((char * p));
|
|
#endif
|
|
LOCAL int isonum_732 __PR((char * p));
|
|
LOCAL int isonum_733 __PR((char * p));
|
|
LOCAL int parse_rr __PR((unsigned char * pnt, int len, int cont_flag));
|
|
LOCAL void find_rr __PR((struct iso_directory_record * idr, Uchar **pntp, int *lenp));
|
|
LOCAL int dump_rr __PR((struct iso_directory_record * idr));
|
|
LOCAL void check_tree __PR((struct diraddr *dirs, off_t file_addr, int file_size, off_t parent_addr));
|
|
#if 0
|
|
LOCAL void check_path_tables __PR((int typel_extent, int typem_extent, int path_table_size));
|
|
#endif
|
|
LOCAL void usage __PR((int excode));
|
|
EXPORT int main __PR((int argc, char *argv[]));
|
|
|
|
LOCAL int
|
|
isonum_721(p)
|
|
char *p;
|
|
{
|
|
return ((p[0] & 0xff) | ((p[1] & 0xff) << 8));
|
|
}
|
|
|
|
LOCAL int
|
|
isonum_723(p)
|
|
char *p;
|
|
{
|
|
#if 0
|
|
if (p[0] != p[3] || p[1] != p[2]) {
|
|
fprintf(stderr, "invalid format 7.2.3 number\n");
|
|
exit(1);
|
|
}
|
|
#endif
|
|
return (isonum_721(p));
|
|
}
|
|
|
|
LOCAL int
|
|
isonum_711(p)
|
|
char *p;
|
|
{
|
|
return (*p & 0xff);
|
|
}
|
|
|
|
LOCAL int
|
|
isonum_731(p)
|
|
char *p;
|
|
{
|
|
return ((p[0] & 0xff)
|
|
| ((p[1] & 0xff) << 8)
|
|
| ((p[2] & 0xff) << 16)
|
|
| ((p[3] & 0xff) << 24));
|
|
}
|
|
|
|
#if 0
|
|
LOCAL int
|
|
isonum_722(p)
|
|
char *p;
|
|
{
|
|
return ((p[1] & 0xff)
|
|
| ((p[0] & 0xff) << 8));
|
|
}
|
|
#endif
|
|
|
|
LOCAL int
|
|
isonum_732(p)
|
|
char *p;
|
|
{
|
|
return ((p[3] & 0xff)
|
|
| ((p[2] & 0xff) << 8)
|
|
| ((p[1] & 0xff) << 16)
|
|
| ((p[0] & 0xff) << 24));
|
|
}
|
|
|
|
LOCAL int
|
|
isonum_733(p)
|
|
char *p;
|
|
{
|
|
return (isonum_731(p));
|
|
}
|
|
|
|
char lbuffer[1024];
|
|
int iline;
|
|
int rr_goof;
|
|
|
|
|
|
LOCAL int
|
|
parse_rr(pnt, len, cont_flag)
|
|
unsigned char *pnt;
|
|
int len;
|
|
int cont_flag;
|
|
{
|
|
int slen;
|
|
int ncount;
|
|
int flag1;
|
|
int flag2;
|
|
int extent;
|
|
unsigned char *pnts;
|
|
off_t cont_extent;
|
|
int cont_offset;
|
|
int cont_size;
|
|
char symlinkname[1024];
|
|
|
|
sprintf(lbuffer+iline, " RRlen=%d ", len);
|
|
iline += strlen(lbuffer+iline);
|
|
|
|
cont_extent = (off_t)0;
|
|
cont_offset = cont_size = 0;
|
|
|
|
symlinkname[0] = 0;
|
|
|
|
ncount = 0;
|
|
flag1 = -1;
|
|
flag2 = 0;
|
|
while (len >= 4) {
|
|
if (ncount)
|
|
sprintf(lbuffer+iline, ",");
|
|
else
|
|
sprintf(lbuffer+iline, "[");
|
|
iline += strlen(lbuffer + iline);
|
|
sprintf(lbuffer+iline, "%c%c", pnt[0], pnt[1]);
|
|
iline += strlen(lbuffer + iline);
|
|
if (pnt[0] < 'A' || pnt[0] > 'Z' || pnt[1] < 'A' ||
|
|
pnt[1] > 'Z') {
|
|
sprintf(lbuffer+iline, _("**BAD SUSP %d %d]"),
|
|
pnt[0], pnt[1]);
|
|
rr_goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
return (flag2);
|
|
}
|
|
|
|
if (pnt[3] != 1 && pnt[3] != 2) {
|
|
sprintf(lbuffer+iline, _("**BAD RRVERSION (%d)\n"), pnt[3]);
|
|
rr_goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
return (flag2);
|
|
}
|
|
if (pnt[2] < 4) {
|
|
sprintf(lbuffer+iline,
|
|
_("**BAD RRLEN (%d) in '%2.2s' field %2.2X %2.2X.\n"),
|
|
pnt[2], pnt, pnt[0], pnt[1]);
|
|
rr_goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
return (flag2);
|
|
}
|
|
ncount++;
|
|
if (pnt[0] == 'R' && pnt[1] == 'R') flag1 = pnt[4] & 0xff;
|
|
if (strncmp((char *)pnt, "PX", 2) == 0) flag2 |= RR_FLAG_PX;
|
|
if (strncmp((char *)pnt, "PN", 2) == 0) flag2 |= RR_FLAG_PN;
|
|
if (strncmp((char *)pnt, "SL", 2) == 0) flag2 |= RR_FLAG_SL;
|
|
if (strncmp((char *)pnt, "NM", 2) == 0) flag2 |= RR_FLAG_NM;
|
|
if (strncmp((char *)pnt, "CL", 2) == 0) flag2 |= RR_FLAG_CL;
|
|
if (strncmp((char *)pnt, "PL", 2) == 0) flag2 |= RR_FLAG_PL;
|
|
if (strncmp((char *)pnt, "RE", 2) == 0) flag2 |= RR_FLAG_RE;
|
|
if (strncmp((char *)pnt, "TF", 2) == 0) flag2 |= RR_FLAG_TF;
|
|
|
|
if (strncmp((char *)pnt, "SP", 2) == 0) {
|
|
flag2 |= RR_FLAG_SP; /* SUSP record */
|
|
su_version = pnt[3] & 0xff;
|
|
}
|
|
if (strncmp((char *)pnt, "AA", 2) == 0) { /* Neither SUSP nor RR */
|
|
flag2 |= RR_FLAG_AA; /* Apple Signature record */
|
|
aa_version = pnt[3] & 0xff;
|
|
}
|
|
if (strncmp((char *)pnt, "ER", 2) == 0) {
|
|
flag2 |= RR_FLAG_ER; /* ER record */
|
|
rr_version = pnt[7] & 0xff; /* Ext Version */
|
|
/* strlcpy(er_id, (char *)&pnt[8], (pnt[4] & 0xFF) + 1);*/
|
|
}
|
|
|
|
if (strncmp((char *)pnt, "CE", 2) == 0) {
|
|
cont_extent = (off_t)isonum_733((char *)pnt+4);
|
|
cont_offset = isonum_733((char *)pnt+12);
|
|
cont_size = isonum_733((char *)pnt+20);
|
|
sprintf(lbuffer+iline, "=[%x,%x,%d]",
|
|
(int)cont_extent, cont_offset, cont_size);
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
|
|
if (strncmp((char *)pnt, "ST", 2) == 0) { /* Terminate SUSP */
|
|
break;
|
|
}
|
|
|
|
if (strncmp((char *)pnt, "PL", 2) == 0 || strncmp((char *)pnt, "CL", 2) == 0) {
|
|
extent = isonum_733((char *)pnt+4);
|
|
sprintf(lbuffer+iline, "=%x", extent);
|
|
iline += strlen(lbuffer + iline);
|
|
if (extent == 0)
|
|
rr_goof++;
|
|
}
|
|
if (strncmp((char *)pnt, "SL", 2) == 0) {
|
|
pnts = pnt+5;
|
|
slen = pnt[2] - 5;
|
|
while (slen >= 1) {
|
|
switch (pnts[0] & 0xfe) {
|
|
case 0:
|
|
strncat(symlinkname, (char *)(pnts+2), pnts[1]);
|
|
break;
|
|
case 2:
|
|
strcat(symlinkname, ".");
|
|
break;
|
|
case 4:
|
|
strcat(symlinkname, "..");
|
|
break;
|
|
case 8:
|
|
strcat(symlinkname, "/");
|
|
break;
|
|
case 16:
|
|
strcat(symlinkname, "/mnt");
|
|
sprintf(lbuffer+iline, _("Warning - mount point requested"));
|
|
iline += strlen(lbuffer + iline);
|
|
break;
|
|
case 32:
|
|
strcat(symlinkname, "kafka");
|
|
sprintf(lbuffer+iline, _("Warning - host_name requested"));
|
|
iline += strlen(lbuffer + iline);
|
|
break;
|
|
default:
|
|
sprintf(lbuffer+iline, _("Reserved bit setting in symlink"));
|
|
rr_goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
break;
|
|
}
|
|
if ((pnts[0] & 0xfe) && pnts[1] != 0) {
|
|
sprintf(lbuffer+iline, _("Incorrect length in symlink component"));
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
if ((pnts[0] & 1) == 0)
|
|
strcat(symlinkname, "/");
|
|
slen -= (pnts[1] + 2);
|
|
pnts += (pnts[1] + 2);
|
|
}
|
|
if (symlinkname[0] != 0) {
|
|
sprintf(lbuffer+iline, "=%s", symlinkname);
|
|
iline += strlen(lbuffer + iline);
|
|
symlinkname[0] = 0;
|
|
}
|
|
}
|
|
|
|
len -= pnt[2];
|
|
pnt += pnt[2];
|
|
}
|
|
if (cont_extent) {
|
|
unsigned char sector[2048];
|
|
#ifdef USE_SCG
|
|
readsecs(cont_extent * blocksize / 2048, sector, ISO_BLOCKS(sizeof (sector)));
|
|
#else
|
|
lseek(fileno(infile), cont_extent * blocksize, SEEK_SET);
|
|
read(fileno(infile), sector, sizeof (sector));
|
|
#endif
|
|
flag2 |= parse_rr(§or[cont_offset], cont_size, 1);
|
|
}
|
|
if (ncount) {
|
|
sprintf(lbuffer+iline, "]");
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
if (!cont_flag && flag1 != -1 && flag1 != (flag2 & 0xFF)) {
|
|
sprintf(lbuffer+iline, _("Flag %x != %x"), flag1, flag2);
|
|
rr_goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
return (flag2);
|
|
}
|
|
|
|
LOCAL void
|
|
find_rr(idr, pntp, lenp)
|
|
struct iso_directory_record *idr;
|
|
Uchar **pntp;
|
|
int *lenp;
|
|
{
|
|
struct iso_xa_dir_record *xadp;
|
|
int len;
|
|
unsigned char * pnt;
|
|
|
|
len = idr->length[0] & 0xff;
|
|
len -= offsetof(struct iso_directory_record, name[0]);
|
|
len -= idr->name_len[0];
|
|
|
|
pnt = (unsigned char *) idr;
|
|
pnt += offsetof(struct iso_directory_record, name[0]);
|
|
pnt += idr->name_len[0];
|
|
if ((idr->name_len[0] & 1) == 0) {
|
|
pnt++;
|
|
len--;
|
|
}
|
|
if (len >= 14) {
|
|
xadp = (struct iso_xa_dir_record *)pnt;
|
|
|
|
if (xadp->signature[0] == 'X' && xadp->signature[1] == 'A' &&
|
|
xadp->reserved[0] == '\0') {
|
|
len -= 14;
|
|
pnt += 14;
|
|
}
|
|
}
|
|
*pntp = pnt;
|
|
*lenp = len;
|
|
}
|
|
|
|
LOCAL int
|
|
dump_rr(idr)
|
|
struct iso_directory_record *idr;
|
|
{
|
|
int len;
|
|
unsigned char * pnt;
|
|
|
|
rr_goof = 0;
|
|
find_rr(idr, &pnt, &len);
|
|
return (parse_rr(pnt, len, 0));
|
|
}
|
|
|
|
|
|
LOCAL int dir_count = 0;
|
|
LOCAL int dir_size_count = 0;
|
|
LOCAL int ngoof = 0;
|
|
|
|
LOCAL void
|
|
check_tree(dirs, file_addr, file_size, parent_addr)
|
|
struct diraddr *dirs;
|
|
off_t file_addr;
|
|
int file_size;
|
|
off_t parent_addr;
|
|
{
|
|
unsigned char buffer[2048];
|
|
unsigned int k;
|
|
int rflag = 0;
|
|
int i;
|
|
int i1;
|
|
int j;
|
|
int goof;
|
|
int extent;
|
|
int size;
|
|
off_t orig_file_addr;
|
|
off_t parent_file_addr;
|
|
struct iso_directory_record *idr;
|
|
|
|
i1 = 0;
|
|
|
|
orig_file_addr = file_addr / blocksize; /* Actual extent of this directory */
|
|
parent_file_addr = parent_addr / blocksize;
|
|
|
|
if ((dir_count % 100) == 0)
|
|
printf("[%d %d]\n", dir_count, dir_size_count);
|
|
#if 0
|
|
if (sizeof (file_addr) > sizeof (long)) {
|
|
printf("Starting directory %ld %d %lld\n",
|
|
file_addr, file_size,
|
|
(Llong)parent_addr);
|
|
} else {
|
|
printf("Starting directory %ld %d %ld\n", file_addr, file_size, parent_addr);
|
|
}
|
|
#endif
|
|
|
|
dir_count++;
|
|
dir_size_count += file_size / blocksize;
|
|
|
|
if (file_size & 0x3ff)
|
|
printf(_("********Directory has unusual size\n"));
|
|
|
|
for (k = 0; k < (file_size / sizeof (buffer)); k++) {
|
|
#ifdef USE_SCG
|
|
readsecs(file_addr / 2048, buffer, ISO_BLOCKS(sizeof (buffer)));
|
|
#else
|
|
lseek(fileno(infile), file_addr, SEEK_SET);
|
|
read(fileno(infile), buffer, sizeof (buffer));
|
|
#endif
|
|
i = 0;
|
|
while (1 == 1) {
|
|
goof = iline = 0;
|
|
idr = (struct iso_directory_record *) &buffer[i];
|
|
if (idr->length[0] == 0) break;
|
|
sprintf(&lbuffer[iline], "%3d ", idr->length[0]);
|
|
iline += strlen(lbuffer + iline);
|
|
extent = isonum_733(idr->extent);
|
|
size = isonum_733(idr->size);
|
|
sprintf(&lbuffer[iline], "%5x ", extent);
|
|
iline += strlen(lbuffer + iline);
|
|
sprintf(&lbuffer[iline], "%8d ", size);
|
|
iline += strlen(lbuffer + iline);
|
|
sprintf(&lbuffer[iline], "%c", (idr->flags[0] & 2) ? '*' : ' ');
|
|
iline += strlen(lbuffer + iline);
|
|
|
|
if (idr->name_len[0] > 33) {
|
|
sprintf(&lbuffer[iline], _("File name length=(%d)"),
|
|
idr->name_len[0]);
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
} else if (idr->name_len[0] == 1 && idr->name[0] == 0) {
|
|
sprintf(&lbuffer[iline], ". ");
|
|
iline += strlen(lbuffer + iline);
|
|
rflag = 0;
|
|
#ifdef Eric_seems_to_be_wrong
|
|
if (orig_file_addr != (off_t)(isonum_733(idr->extent) +
|
|
isonum_711((char *) idr->ext_attr_length))) {
|
|
#else
|
|
if (orig_file_addr != (off_t)isonum_733(idr->extent)) {
|
|
#endif
|
|
sprintf(&lbuffer[iline], _("***** Directory has null extent."));
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
if (i1) {
|
|
sprintf(&lbuffer[iline], _("***** . not first entry."));
|
|
rr_goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
} else if (idr->name_len[0] == 1 && idr->name[0] == 1) {
|
|
sprintf(&lbuffer[iline], ".. ");
|
|
iline += strlen(lbuffer + iline);
|
|
rflag = 0;
|
|
#ifdef Eric_seems_to_be_wrong
|
|
if (parent_file_addr != (off_t)(isonum_733(idr->extent) +
|
|
isonum_711((char *) idr->ext_attr_length))) {
|
|
#else
|
|
if (parent_file_addr != (off_t)isonum_733(idr->extent)) {
|
|
#endif
|
|
sprintf(&lbuffer[iline], _("***** Directory has null extent."));
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
if (i1 != 1) {
|
|
sprintf(&lbuffer[iline], _("***** .. not second entry."));
|
|
rr_goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
} else {
|
|
if (i1 < 2) {
|
|
sprintf(&lbuffer[iline], _(" Improper sorting."));
|
|
rr_goof++;
|
|
}
|
|
for (j = 0; j < (int)idr->name_len[0]; j++) {
|
|
sprintf(&lbuffer[iline], "%c", idr->name[j]);
|
|
}
|
|
for (j = 0; j < (14 - (int) idr->name_len[0]); j++) {
|
|
sprintf(&lbuffer[iline], " ");
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
rflag = 1;
|
|
}
|
|
|
|
if (size && extent == 0) {
|
|
sprintf(&lbuffer[iline], _("****Extent==0, size != 0"));
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
#if 0
|
|
/* This is apparently legal. */
|
|
if (size == 0 && extent) {
|
|
sprintf(&lbuffer[iline], _("****Extent!=0, size == 0"));
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
#endif
|
|
|
|
if (idr->flags[0] & 0xf5) {
|
|
sprintf(&lbuffer[iline], _("Flags=(%x) "), idr->flags[0]);
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
if (idr->interleave[0]) {
|
|
sprintf(&lbuffer[iline], _("Interleave=(%d) "), idr->interleave[0]);
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
|
|
if (idr->file_unit_size[0]) {
|
|
sprintf(&lbuffer[iline], _("File unit size=(%d) "), idr->file_unit_size[0]);
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
|
|
if (idr->volume_sequence_number[0] != 1) {
|
|
sprintf(&lbuffer[iline], _("Volume sequence number=(%d) "), idr->volume_sequence_number[0]);
|
|
goof++;
|
|
iline += strlen(lbuffer + iline);
|
|
}
|
|
|
|
if (use_rock) {
|
|
dump_rr(idr);
|
|
goof += rr_goof;
|
|
}
|
|
sprintf(&lbuffer[iline], "\n");
|
|
iline += strlen(lbuffer + iline);
|
|
|
|
|
|
if (goof) {
|
|
ngoof++;
|
|
lbuffer[iline++] = 0;
|
|
if (sizeof (orig_file_addr) > sizeof (long)) {
|
|
printf("%llx: %s", (Llong)orig_file_addr, lbuffer);
|
|
} else {
|
|
printf("%lx: %s", (long)orig_file_addr, lbuffer);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
if (rflag && (idr->flags[0] & 2)) {
|
|
struct diraddr dir;
|
|
struct diraddr *dp;
|
|
int addr = isonum_733(idr->extent);
|
|
BOOL isloop = FALSE;
|
|
|
|
dir.next = dirs;
|
|
dir.addr = addr;
|
|
|
|
for (dp = dirs; dp; dp = dp->next) {
|
|
if (addr == dp->addr) {
|
|
isloop = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
if (!isloop) {
|
|
#ifdef Eric_seems_to_be_wrong
|
|
check_tree(&dir, (off_t)(adr + isonum_711((char *)idr->ext_attr_length)) * blocksize,
|
|
isonum_733(idr->size),
|
|
orig_file_addr * blocksize);
|
|
#else
|
|
check_tree(&dir, (off_t)addr * blocksize,
|
|
isonum_733(idr->size),
|
|
orig_file_addr * blocksize);
|
|
#endif
|
|
}
|
|
}
|
|
i += buffer[i];
|
|
i1++;
|
|
if (i > 2048 - offsetof(struct iso_directory_record, name[0]))
|
|
break;
|
|
}
|
|
file_addr += sizeof (buffer);
|
|
}
|
|
fflush(stdout);
|
|
}
|
|
|
|
|
|
/*
|
|
* This function simply dumps the contents of the path tables. No
|
|
* consistency checking takes place, although this would proably be a good
|
|
* idea.
|
|
*/
|
|
struct path_table_info {
|
|
char *name;
|
|
unsigned int extent;
|
|
unsigned short index;
|
|
unsigned short parent;
|
|
};
|
|
|
|
#if 0
|
|
LOCAL void
|
|
check_path_tables(typel_extent, typem_extent, path_table_size)
|
|
int typel_extent;
|
|
int typem_extent;
|
|
int path_table_size;
|
|
{
|
|
int count;
|
|
int j;
|
|
char *pnt;
|
|
char *typel;
|
|
char *typem;
|
|
|
|
/* Now read in the path tables */
|
|
|
|
typel = (char *) malloc(ISO_ROUND_UP(path_table_size));
|
|
#ifdef USE_SCG
|
|
readsecs(typel_extent * blocksize / 2048, typel, ISO_BLOCKS(path_table_size));
|
|
#else
|
|
lseek(fileno(infile), (off_t)((off_t)typel_extent) * blocksize, SEEK_SET);
|
|
read(fileno(infile), typel, path_table_size);
|
|
#endif
|
|
typem = (char *) malloc(path_table_size);
|
|
lseek(fileno(infile), (off_t)((off_t)typem_extent) * blocksize, SEEK_SET);
|
|
read(fileno(infile), typem, path_table_size);
|
|
|
|
j = path_table_size;
|
|
pnt = typel;
|
|
count = 1;
|
|
while (j) {
|
|
int namelen;
|
|
int extent;
|
|
int idx;
|
|
char name[32];
|
|
|
|
namelen = *pnt++; pnt++;
|
|
extent = isonum_731(pnt);
|
|
pnt += 4;
|
|
idx = isonum_721(pnt);
|
|
pnt += 2;
|
|
j -= 8 + namelen;
|
|
memset(name, 0, sizeof (name));
|
|
|
|
strncpy(name, pnt, namelen);
|
|
pnt += namelen;
|
|
if (j & 1) {
|
|
j--;
|
|
pnt++;
|
|
}
|
|
printf("%4.4d %4.4d %8.8x %s\n", count++, idx, extent, name);
|
|
}
|
|
|
|
j = path_table_size;
|
|
pnt = typem;
|
|
count = 1;
|
|
|
|
while (j) {
|
|
int namelen;
|
|
int extent;
|
|
int idx;
|
|
char name[32];
|
|
|
|
namelen = *pnt++; pnt++;
|
|
extent = isonum_732(pnt);
|
|
pnt += 4;
|
|
idx = isonum_722(pnt);
|
|
pnt += 2;
|
|
j -= 8+namelen;
|
|
memset(name, 0, sizeof (name));
|
|
|
|
strncpy(name, pnt, namelen);
|
|
pnt += namelen;
|
|
if (j & 1) {
|
|
j--;
|
|
pnt++;
|
|
}
|
|
printf("%4.4d %4.4d %8.8x %s\n", count++, idx, extent, name);
|
|
}
|
|
free(typel);
|
|
free(typem);
|
|
}
|
|
#endif
|
|
|
|
LOCAL void
|
|
usage(excode)
|
|
int excode;
|
|
{
|
|
errmsgno(EX_BAD, _("Usage: %s [options] image\n"),
|
|
get_progname());
|
|
|
|
error(_("Options:\n"));
|
|
error(_("\t-help, -h Print this help\n"));
|
|
error(_("\t-version Print version info and exit\n"));
|
|
error(_("\t-ignore-error Ignore errors\n"));
|
|
error(_("\t-i filename Filename to read ISO-9660 image from\n"));
|
|
error(_("\tdev=target SCSI target to use as CD/DVD-Recorder\n"));
|
|
error(_("\nIf neither -i nor dev= are specified, <image> is needed.\n"));
|
|
exit(excode);
|
|
}
|
|
|
|
EXPORT int
|
|
main(argc, argv)
|
|
int argc;
|
|
char *argv[];
|
|
{
|
|
int cac;
|
|
char * const *cav;
|
|
int c;
|
|
char *opts = "help,h,version,ignore-error,i*,dev*";
|
|
BOOL help = FALSE;
|
|
BOOL prvers = FALSE;
|
|
char *filename = NULL;
|
|
char *sdevname = NULL;
|
|
#if defined(USE_NLS)
|
|
char *dir;
|
|
#endif
|
|
off_t file_addr;
|
|
int file_size;
|
|
struct iso_primary_descriptor ipd;
|
|
struct iso_directory_record *idr;
|
|
int typel_extent;
|
|
int typem_extent;
|
|
int path_table_size;
|
|
char buffer[2048];
|
|
|
|
save_args(argc, argv);
|
|
|
|
#if defined(USE_NLS)
|
|
setlocale(LC_ALL, "");
|
|
#if !defined(TEXT_DOMAIN) /* Should be defined by cc -D */
|
|
#define TEXT_DOMAIN "isoinfo" /* Use this only if it weren't */
|
|
#endif
|
|
dir = searchfileinpath("share/locale", F_OK,
|
|
SIP_ANY_FILE|SIP_NO_PATH, NULL);
|
|
if (dir)
|
|
(void) bindtextdomain(TEXT_DOMAIN, dir);
|
|
else
|
|
#if defined(PROTOTYPES) && defined(INS_BASE)
|
|
(void) bindtextdomain(TEXT_DOMAIN, INS_BASE "/share/locale");
|
|
#else
|
|
(void) bindtextdomain(TEXT_DOMAIN, "/usr/share/locale");
|
|
#endif
|
|
(void) textdomain(TEXT_DOMAIN);
|
|
#endif
|
|
|
|
cac = argc - 1;
|
|
cav = argv + 1;
|
|
if (getallargs(&cac, &cav, opts, &help, &help, &prvers, &ignerr,
|
|
&filename, &sdevname) < 0) {
|
|
errmsgno(EX_BAD, _("Bad Option: '%s'\n"), cav[0]);
|
|
usage(EX_BAD);
|
|
}
|
|
if (help)
|
|
usage(0);
|
|
if (prvers) {
|
|
printf(_("isovfy %s (%s-%s-%s) Copyright (C) 1993-1999 %s (C) 1999-2015 %s\n"),
|
|
VERSION,
|
|
HOST_CPU, HOST_VENDOR, HOST_OS,
|
|
_("Eric Youngdale"),
|
|
_("Joerg Schilling"));
|
|
exit(0);
|
|
}
|
|
cac = argc - 1;
|
|
cav = argv + 1;
|
|
if (filename == NULL && sdevname == NULL) {
|
|
if (getfiles(&cac, &cav, opts) != 0) {
|
|
filename = cav[0];
|
|
cac--, cav++;
|
|
}
|
|
}
|
|
if (getfiles(&cac, &cav, opts) != 0) {
|
|
errmsgno(EX_BAD, _("Bad Argument: '%s'\n"), cav[0]);
|
|
usage(EX_BAD);
|
|
}
|
|
if (filename != NULL && sdevname != NULL) {
|
|
errmsgno(EX_BAD, _("Only one of -i or dev= allowed\n"));
|
|
usage(EX_BAD);
|
|
}
|
|
#ifdef USE_SCG
|
|
if (filename == NULL && sdevname == NULL)
|
|
cdr_defaults(&sdevname, NULL, NULL, NULL, NULL);
|
|
#endif
|
|
if (filename == NULL && sdevname == NULL) {
|
|
fprintf(stderr, _("ISO-9660 image not specified\n"));
|
|
usage(EX_BAD);
|
|
}
|
|
|
|
if (filename != NULL)
|
|
infile = fopen(filename, "rb");
|
|
else
|
|
filename = sdevname;
|
|
|
|
if (infile != NULL) {
|
|
/* EMPTY */;
|
|
#ifdef USE_SCG
|
|
} else if (scsidev_open(filename) < 0) {
|
|
#else
|
|
} else {
|
|
#endif
|
|
fprintf(stderr, _("Cannot open '%s'\n"), filename);
|
|
exit(1);
|
|
}
|
|
|
|
|
|
file_addr = (off_t)32768;
|
|
#ifdef USE_SCG
|
|
readsecs(file_addr / 2048, &ipd, ISO_BLOCKS(sizeof (ipd)));
|
|
#else
|
|
lseek(fileno(infile), file_addr, SEEK_SET);
|
|
read(fileno(infile), &ipd, sizeof (ipd));
|
|
#endif
|
|
|
|
idr = (struct iso_directory_record *)ipd.root_directory_record;
|
|
|
|
blocksize = isonum_723((char *)ipd.logical_block_size);
|
|
if (blocksize != 512 && blocksize != 1024 && blocksize != 2048) {
|
|
blocksize = 2048;
|
|
}
|
|
|
|
#ifdef Eric_seems_to_be_wrong
|
|
file_addr = (off_t)isonum_733(idr->extent) + isonum_711((char *)idr->ext_attr_length);
|
|
#else
|
|
file_addr = (off_t)isonum_733(idr->extent);
|
|
#endif
|
|
file_size = isonum_733(idr->size);
|
|
|
|
#ifdef USE_SCG
|
|
readsecs(file_addr, buffer, ISO_BLOCKS(sizeof (buffer)));
|
|
#else
|
|
lseek(fileno(infile), file_addr * blocksize, SEEK_SET);
|
|
read(fileno(infile), buffer, sizeof (buffer));
|
|
#endif
|
|
c = dump_rr((struct iso_directory_record *) buffer);
|
|
if (c == 0 ||
|
|
(c & (RR_FLAG_SP | RR_FLAG_ER)) == 0 || su_version < 1 || rr_version < 1) {
|
|
use_rock = FALSE;
|
|
}
|
|
|
|
if (sizeof (file_addr) > sizeof (long)) {
|
|
printf(_("Root at extent %llx, %d bytes\n"), (Llong)file_addr, file_size);
|
|
} else {
|
|
printf(_("Root at extent %lx, %d bytes\n"), (long)file_addr, file_size);
|
|
}
|
|
file_addr = file_addr * blocksize;
|
|
|
|
check_tree((struct diraddr *)NULL, file_addr, file_size, file_addr);
|
|
|
|
typel_extent = isonum_731((char *)ipd.type_l_path_table);
|
|
typem_extent = isonum_732((char *)ipd.type_m_path_table);
|
|
path_table_size = isonum_733(ipd.path_table_size);
|
|
|
|
/* Enable this to get the dump of the path tables */
|
|
#if 0
|
|
check_path_tables(typel_extent, typem_extent, path_table_size);
|
|
#endif
|
|
|
|
if (infile != NULL)
|
|
fclose(infile);
|
|
|
|
if (!ngoof)
|
|
printf(_("No errors found\n"));
|
|
return (0);
|
|
}
|