2004-02-25 10:03:27 +00:00
|
|
|
/*
|
2004-03-11 01:01:15 +00:00
|
|
|
$Id: iso-read.c,v 1.4 2004/03/11 01:01:15 rocky Exp $
|
2004-02-25 10:03:27 +00:00
|
|
|
|
|
|
|
|
Copyright (C) 2004 Rocky Bernstein <rocky@panix.com>
|
|
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
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; if not, write to the Free Software
|
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* Program to read ISO-9660 images. */
|
|
|
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
|
# include "config.h"
|
|
|
|
|
#endif
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <cdio/cdio.h>
|
|
|
|
|
#include <cdio/iso9660.h>
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_ERRNO_H
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_STRING_H
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2004-03-11 01:01:15 +00:00
|
|
|
#if defined ( WIN32 )
|
|
|
|
|
#define ftruncate chsize
|
|
|
|
|
#endif
|
|
|
|
|
|
2004-02-25 10:03:27 +00:00
|
|
|
/* Used by `main' to communicate with `parse_opt'. And global options
|
|
|
|
|
*/
|
|
|
|
|
struct arguments
|
|
|
|
|
{
|
|
|
|
|
char *file_name;
|
|
|
|
|
char *output_file;
|
|
|
|
|
char *iso9660_image;
|
|
|
|
|
int debug_level;
|
|
|
|
|
} opts;
|
|
|
|
|
|
|
|
|
|
/* Parse a options. */
|
|
|
|
|
static bool
|
|
|
|
|
parse_options (int argc, const char *argv[])
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
int opt;
|
|
|
|
|
|
|
|
|
|
/* Configuration option codes */
|
|
|
|
|
enum {
|
|
|
|
|
OP_VERSION=1
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Command-line options */
|
|
|
|
|
struct poptOption optionsTable[] = {
|
|
|
|
|
|
|
|
|
|
{"debug", 'd',
|
|
|
|
|
POPT_ARG_INT, &opts.debug_level, 0,
|
|
|
|
|
"Set debugging to LEVEL"},
|
|
|
|
|
|
|
|
|
|
{"image", 'i', POPT_ARG_STRING, &opts.iso9660_image, 0,
|
|
|
|
|
"read from ISO-9660 image", "FILE"},
|
|
|
|
|
|
|
|
|
|
{"extract", 'e', POPT_ARG_STRING, &opts.file_name, 0,
|
|
|
|
|
"extract FILE from ISO-9660 image", "FILE"},
|
|
|
|
|
|
|
|
|
|
{"output-file", 'o', POPT_ARG_STRING, &opts.output_file, 0,
|
|
|
|
|
"Output file."},
|
|
|
|
|
|
|
|
|
|
{"version", 'V', POPT_ARG_NONE, NULL, OP_VERSION,
|
|
|
|
|
"display version and copyright information and exit"},
|
|
|
|
|
POPT_AUTOHELP {NULL, 0, 0, NULL, 0}
|
|
|
|
|
};
|
|
|
|
|
poptContext optCon = poptGetContext (NULL, argc, argv, optionsTable, 0);
|
|
|
|
|
|
|
|
|
|
program_name = strrchr(argv[0],'/');
|
|
|
|
|
program_name = program_name ? strdup(program_name+1) : strdup(argv[0]);
|
|
|
|
|
|
|
|
|
|
while ((opt = poptGetNextOpt (optCon)) != -1)
|
|
|
|
|
switch (opt)
|
|
|
|
|
{
|
|
|
|
|
case OP_VERSION:
|
|
|
|
|
print_version(program_name, VERSION, 0, true);
|
|
|
|
|
poptFreeContext(optCon);
|
|
|
|
|
free(program_name);
|
|
|
|
|
exit (EXIT_SUCCESS);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
fprintf (stderr, "%s: %s\n",
|
|
|
|
|
poptBadOption(optCon, POPT_BADOPTION_NOALIAS),
|
|
|
|
|
poptStrerror(opt));
|
|
|
|
|
fprintf (stderr, "error while parsing command line - try --help\n");
|
|
|
|
|
poptFreeContext(optCon);
|
|
|
|
|
free(program_name);
|
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
const char *remaining_arg = poptGetArg(optCon);
|
|
|
|
|
if ( remaining_arg != NULL) {
|
|
|
|
|
if (opts.iso9660_image != NULL) {
|
|
|
|
|
fprintf (stderr,
|
|
|
|
|
"%s: Source specified as --image %s and as %s\n",
|
|
|
|
|
program_name, opts.iso9660_image, remaining_arg);
|
|
|
|
|
poptFreeContext(optCon);
|
|
|
|
|
free(program_name);
|
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opts.iso9660_image = strdup(remaining_arg);
|
|
|
|
|
|
|
|
|
|
if ( (poptGetArgs(optCon)) != NULL) {
|
|
|
|
|
fprintf (stderr,
|
|
|
|
|
"%s: use only one unnamed argument for the ISO 9660 image name\n",
|
|
|
|
|
program_name);
|
|
|
|
|
poptFreeContext(optCon);
|
|
|
|
|
free(program_name);
|
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
poptFreeContext(optCon);
|
|
|
|
|
|
|
|
|
|
if (opts.iso9660_image == NULL) {
|
|
|
|
|
fprintf (stderr,
|
|
|
|
|
"%s: you need to specify an ISO-9660 image name. Try --help\n",
|
|
|
|
|
program_name);
|
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
init(void)
|
|
|
|
|
{
|
|
|
|
|
opts.debug_level = 0;
|
|
|
|
|
opts.file_name = NULL;
|
|
|
|
|
opts.output_file = NULL;
|
|
|
|
|
opts.iso9660_image = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
main(int argc, const char *argv[])
|
|
|
|
|
{
|
|
|
|
|
iso9660_stat_t *statbuf;
|
|
|
|
|
FILE *outfd;
|
|
|
|
|
int i;
|
2004-02-29 16:02:48 +00:00
|
|
|
iso9660_t *iso;
|
2004-02-25 10:03:27 +00:00
|
|
|
|
|
|
|
|
init();
|
|
|
|
|
|
|
|
|
|
/* Parse our arguments; every option seen by `parse_opt' will
|
|
|
|
|
be reflected in `arguments'. */
|
|
|
|
|
parse_options(argc, argv);
|
|
|
|
|
|
2004-02-29 16:02:48 +00:00
|
|
|
iso = iso9660_open (opts.iso9660_image);
|
2004-02-25 10:03:27 +00:00
|
|
|
|
|
|
|
|
if (NULL == iso) {
|
|
|
|
|
fprintf(stderr, "Sorry, couldn't open ISO-9660 image %s\n",
|
|
|
|
|
opts.iso9660_image);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2004-02-26 00:13:24 +00:00
|
|
|
statbuf = iso9660_ifs_stat_translate (iso, opts.file_name);
|
2004-02-25 10:03:27 +00:00
|
|
|
|
|
|
|
|
if (NULL == statbuf)
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"Could not get ISO-9660 file information for file %s\n",
|
|
|
|
|
opts.file_name);
|
|
|
|
|
return 2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(outfd = fopen (opts.output_file, "wb")))
|
|
|
|
|
{
|
|
|
|
|
perror ("fopen()");
|
|
|
|
|
return 3;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Copy the blocks from the ISO-9660 filesystem to the local filesystem. */
|
|
|
|
|
for (i = 0; i < statbuf->size; i += ISO_BLOCKSIZE)
|
|
|
|
|
{
|
|
|
|
|
char buf[ISO_BLOCKSIZE];
|
|
|
|
|
|
|
|
|
|
memset (buf, 0, ISO_BLOCKSIZE);
|
|
|
|
|
|
|
|
|
|
if ( ISO_BLOCKSIZE != iso9660_iso_seek_read (iso, buf, statbuf->lsn
|
|
|
|
|
+ (i / ISO_BLOCKSIZE),
|
|
|
|
|
1) )
|
|
|
|
|
{
|
|
|
|
|
fprintf(stderr, "Error reading ISO 9660 file at lsn %lu\n",
|
|
|
|
|
(long unsigned int) statbuf->lsn + (i / ISO_BLOCKSIZE));
|
|
|
|
|
return 4;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
fwrite (buf, ISO_BLOCKSIZE, 1, outfd);
|
|
|
|
|
|
|
|
|
|
if (ferror (outfd))
|
|
|
|
|
{
|
|
|
|
|
perror ("fwrite()");
|
|
|
|
|
return 5;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fflush (outfd);
|
|
|
|
|
|
|
|
|
|
/* Make sure the file size has the exact same byte size. Without the
|
|
|
|
|
truncate below, the file will a multiple of ISO_BLOCKSIZE.
|
|
|
|
|
*/
|
|
|
|
|
if (ftruncate (fileno (outfd), statbuf->size))
|
|
|
|
|
perror ("ftruncate()");
|
|
|
|
|
|
|
|
|
|
fclose (outfd);
|
|
|
|
|
iso9660_close(iso);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|