2003-09-17 12:13:07 +00:00
|
|
|
/*
|
2003-11-10 03:47:37 +00:00
|
|
|
$Id: cd-read.c,v 1.15 2003/11/10 03:47:37 rocky Exp $
|
2003-09-17 12:13:07 +00:00
|
|
|
|
|
|
|
|
Copyright (C) 2003 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 debug read routines audio, mode1, mode2 forms 1 & 2. */
|
|
|
|
|
|
2003-09-21 04:21:39 +00:00
|
|
|
#include "util.h"
|
2003-09-21 03:35:39 +00:00
|
|
|
|
2003-09-27 23:29:29 +00:00
|
|
|
#ifdef HAVE_SYS_STAT_H
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_FCNTL_H
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_ERRNO_H
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
/* Configuration option codes */
|
|
|
|
|
enum {
|
|
|
|
|
|
|
|
|
|
/* These correspond to driver_id_t in cdio.h and have to MATCH! */
|
|
|
|
|
OP_SOURCE_UNDEF = DRIVER_UNKNOWN,
|
|
|
|
|
OP_SOURCE_AUTO,
|
|
|
|
|
OP_SOURCE_BIN,
|
|
|
|
|
OP_SOURCE_CUE,
|
|
|
|
|
OP_SOURCE_NRG = DRIVER_NRG,
|
|
|
|
|
OP_SOURCE_DEVICE = DRIVER_DEVICE,
|
|
|
|
|
|
|
|
|
|
/* These are the remaining configuration options */
|
|
|
|
|
OP_READ_MODE,
|
|
|
|
|
OP_VERSION,
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
typedef enum
|
|
|
|
|
{
|
|
|
|
|
READ_MODE_UNINIT,
|
|
|
|
|
READ_AUDIO,
|
|
|
|
|
READ_M1F1,
|
|
|
|
|
READ_M1F2,
|
|
|
|
|
READ_M2F1,
|
|
|
|
|
READ_M2F2
|
2003-10-15 01:59:51 +00:00
|
|
|
#if AUTO_FINISHED
|
|
|
|
|
READ_AUTO
|
|
|
|
|
#endif
|
2003-09-21 03:35:39 +00:00
|
|
|
} read_mode_t;
|
|
|
|
|
|
|
|
|
|
/* Structure used so we can binary sort and set the --mode switch. */
|
|
|
|
|
typedef struct
|
|
|
|
|
{
|
|
|
|
|
char name[30];
|
|
|
|
|
read_mode_t read_mode;
|
|
|
|
|
} subopt_entry_t;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Sub-options for --mode. Note: entries must be sorted! */
|
|
|
|
|
subopt_entry_t modes_sublist[] = {
|
|
|
|
|
{"audio", READ_AUDIO},
|
|
|
|
|
{"m1f1", READ_M1F1},
|
|
|
|
|
{"m1f2", READ_M1F2},
|
|
|
|
|
{"m2f1", READ_M2F1},
|
|
|
|
|
{"m2f2", READ_M2F2},
|
|
|
|
|
{"mode1form1", READ_M1F1},
|
|
|
|
|
{"mode1form2", READ_M1F2},
|
|
|
|
|
{"mode2form1", READ_M2F1},
|
|
|
|
|
{"mode2form2", READ_M2F2},
|
|
|
|
|
{"red", READ_AUDIO},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* Used by `main' to communicate with `parse_opt'. And global options
|
|
|
|
|
*/
|
|
|
|
|
struct arguments
|
|
|
|
|
{
|
2003-09-27 23:29:29 +00:00
|
|
|
char *access_mode; /* Access method driver should use for control */
|
|
|
|
|
char *output_file; /* file to output blocks if not NULL. */
|
2003-09-21 03:35:39 +00:00
|
|
|
int debug_level;
|
|
|
|
|
read_mode_t read_mode;
|
|
|
|
|
int version_only;
|
|
|
|
|
int no_header;
|
|
|
|
|
int print_iso9660;
|
|
|
|
|
source_image_t source_image;
|
|
|
|
|
lsn_t start_lsn;
|
2003-09-21 18:43:36 +00:00
|
|
|
lsn_t end_lsn;
|
|
|
|
|
int num_sectors;
|
2003-09-21 03:35:39 +00:00
|
|
|
} opts;
|
|
|
|
|
|
2003-09-17 12:13:07 +00:00
|
|
|
static void
|
|
|
|
|
hexdump (uint8_t * buffer, unsigned int len)
|
|
|
|
|
{
|
|
|
|
|
unsigned int i;
|
|
|
|
|
for (i = 0; i < len; i++, buffer++)
|
|
|
|
|
{
|
|
|
|
|
if (i % 16 == 0)
|
|
|
|
|
printf ("0x%04x: ", i);
|
|
|
|
|
printf ("%02x", *buffer);
|
|
|
|
|
if (i % 2 == 1)
|
|
|
|
|
printf (" ");
|
|
|
|
|
if (i % 16 == 15) {
|
|
|
|
|
uint8_t *p;
|
2003-10-02 02:59:57 +00:00
|
|
|
printf (" ");
|
2003-09-17 12:13:07 +00:00
|
|
|
for (p=buffer-15; p <= buffer; p++) {
|
|
|
|
|
printf("%c", isprint(*p) ? *p : '.');
|
|
|
|
|
}
|
|
|
|
|
printf ("\n");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
printf ("\n");
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-21 04:21:39 +00:00
|
|
|
/* Comparison function called by bearch() to find sub-option record. */
|
|
|
|
|
static int
|
|
|
|
|
compare_subopts(const void *key1, const void *key2)
|
|
|
|
|
{
|
|
|
|
|
subopt_entry_t *a = (subopt_entry_t *) key1;
|
|
|
|
|
subopt_entry_t *b = (subopt_entry_t *) key2;
|
|
|
|
|
return (strncmp(a->name, b->name, 30));
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
/* Do processing of a --mode sub option.
|
|
|
|
|
Basically we find the option in the array, set it's corresponding
|
|
|
|
|
flag variable to true as well as the "show.all" false.
|
|
|
|
|
*/
|
|
|
|
|
static void
|
|
|
|
|
process_suboption(const char *subopt, subopt_entry_t *sublist, const int num,
|
|
|
|
|
const char *subopt_name)
|
|
|
|
|
{
|
|
|
|
|
subopt_entry_t *subopt_rec =
|
|
|
|
|
bsearch(subopt, sublist, num, sizeof(subopt_entry_t),
|
|
|
|
|
&compare_subopts);
|
|
|
|
|
if (subopt_rec != NULL) {
|
|
|
|
|
opts.read_mode = subopt_rec->read_mode;
|
|
|
|
|
return;
|
|
|
|
|
} else {
|
|
|
|
|
unsigned int i;
|
|
|
|
|
bool is_help=strcmp(subopt, "help")==0;
|
|
|
|
|
if (is_help) {
|
|
|
|
|
fprintf (stderr, "The list of sub options for \"%s\" are:\n",
|
|
|
|
|
subopt_name);
|
|
|
|
|
} else {
|
|
|
|
|
fprintf (stderr, "Invalid option following \"%s\": %s.\n",
|
|
|
|
|
subopt_name, subopt);
|
|
|
|
|
fprintf (stderr, "Should be one of: ");
|
|
|
|
|
}
|
|
|
|
|
for (i=0; i<num-1; i++) {
|
|
|
|
|
fprintf(stderr, "%s, ", sublist[i].name);
|
|
|
|
|
}
|
|
|
|
|
fprintf(stderr, "or %s.\n", sublist[num-1].name);
|
|
|
|
|
exit (is_help ? EXIT_SUCCESS : EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Parse a options. */
|
|
|
|
|
static bool
|
|
|
|
|
parse_options (int argc, const char *argv[])
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
int opt;
|
|
|
|
|
char *opt_arg;
|
|
|
|
|
|
|
|
|
|
/* Command-line options */
|
|
|
|
|
struct poptOption optionsTable[] = {
|
|
|
|
|
|
2003-09-27 23:29:29 +00:00
|
|
|
{"access-mode", 'a', POPT_ARG_STRING, &opts.access_mode, 0,
|
|
|
|
|
"Set CD control access mode"},
|
|
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
{"mode", 'm',
|
|
|
|
|
POPT_ARG_STRING, &opt_arg,
|
|
|
|
|
OP_READ_MODE,
|
|
|
|
|
"set CD-ROM read mode (audio, m1f1, m1f2, m2mf1, m2f2)",
|
|
|
|
|
"MODE-TYPE"},
|
|
|
|
|
|
|
|
|
|
{"debug", 'd',
|
|
|
|
|
POPT_ARG_INT, &opts.debug_level, 0,
|
|
|
|
|
"Set debugging to LEVEL"},
|
|
|
|
|
|
|
|
|
|
{"start", 's',
|
|
|
|
|
POPT_ARG_INT, &opts.start_lsn, 0,
|
|
|
|
|
"Set LBA to start reading from"},
|
|
|
|
|
|
2003-09-21 18:43:36 +00:00
|
|
|
{"end", 'e',
|
|
|
|
|
POPT_ARG_INT, &opts.end_lsn, 0,
|
|
|
|
|
"Set LBA to end reading from"},
|
|
|
|
|
|
|
|
|
|
{"number", 'n',
|
|
|
|
|
POPT_ARG_INT, &opts.num_sectors, 0,
|
|
|
|
|
"Set number of sectors to read"},
|
|
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
{"bin-file", 'b', POPT_ARG_STRING|POPT_ARGFLAG_OPTIONAL, &source_name,
|
|
|
|
|
OP_SOURCE_BIN, "set \"bin\" CD-ROM disk image file as source", "FILE"},
|
|
|
|
|
|
|
|
|
|
{"cue-file", 'c', POPT_ARG_STRING|POPT_ARGFLAG_OPTIONAL, &source_name,
|
|
|
|
|
OP_SOURCE_CUE, "set \"cue\" CD-ROM disk image file as source", "FILE"},
|
|
|
|
|
|
|
|
|
|
{"input", 'i', POPT_ARG_STRING|POPT_ARGFLAG_OPTIONAL, &source_name,
|
|
|
|
|
OP_SOURCE_AUTO,
|
|
|
|
|
"set source and determine if \"bin\" image or device", "FILE"},
|
|
|
|
|
|
|
|
|
|
{"cdrom-device", 'C', POPT_ARG_STRING|POPT_ARGFLAG_OPTIONAL, &source_name,
|
|
|
|
|
OP_SOURCE_DEVICE,
|
|
|
|
|
"set CD-ROM device as source", "DEVICE"},
|
|
|
|
|
|
|
|
|
|
{"nrg-file", 'N', POPT_ARG_STRING|POPT_ARGFLAG_OPTIONAL, &source_name,
|
|
|
|
|
OP_SOURCE_NRG, "set Nero CD-ROM disk image file as source", "FILE"},
|
|
|
|
|
|
2003-09-27 23:29:29 +00:00
|
|
|
{"output-file", 'o', POPT_ARG_STRING, &opts.output_file, 0,
|
|
|
|
|
"Output blocks to file rather than give a hexdump."},
|
|
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
{"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);
|
|
|
|
|
|
2003-09-21 18:43:36 +00:00
|
|
|
program_name = strrchr(argv[0],'/');
|
|
|
|
|
program_name = program_name ? program_name+1 : strdup(argv[0]);
|
|
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
while ((opt = poptGetNextOpt (optCon)) != -1)
|
|
|
|
|
switch (opt)
|
|
|
|
|
{
|
|
|
|
|
case OP_SOURCE_AUTO:
|
|
|
|
|
case OP_SOURCE_BIN:
|
|
|
|
|
case OP_SOURCE_CUE:
|
|
|
|
|
case OP_SOURCE_NRG:
|
|
|
|
|
case OP_SOURCE_DEVICE:
|
|
|
|
|
if (opts.source_image != IMAGE_UNKNOWN) {
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%s: another source type option given before.\n",
|
|
|
|
|
program_name);
|
|
|
|
|
fprintf(stderr, "%s: give only one source type option.\n",
|
|
|
|
|
program_name);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
switch (opt) {
|
|
|
|
|
case OP_SOURCE_BIN:
|
|
|
|
|
opts.source_image = IMAGE_BIN;
|
|
|
|
|
break;
|
|
|
|
|
case OP_SOURCE_CUE:
|
|
|
|
|
opts.source_image = IMAGE_CUE;
|
|
|
|
|
break;
|
|
|
|
|
case OP_SOURCE_NRG:
|
|
|
|
|
opts.source_image = IMAGE_NRG;
|
|
|
|
|
break;
|
|
|
|
|
case OP_SOURCE_AUTO:
|
|
|
|
|
opts.source_image = IMAGE_AUTO;
|
|
|
|
|
break;
|
|
|
|
|
case OP_SOURCE_DEVICE:
|
|
|
|
|
opts.source_image = IMAGE_DEVICE;
|
|
|
|
|
source_name = fillout_device_name(source_name);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case OP_READ_MODE:
|
|
|
|
|
process_suboption(opt_arg, modes_sublist,
|
|
|
|
|
sizeof(modes_sublist) / sizeof(subopt_entry_t),
|
|
|
|
|
"--mode");
|
|
|
|
|
break;
|
|
|
|
|
case OP_VERSION:
|
2003-09-21 04:21:39 +00:00
|
|
|
print_version(program_name, VERSION, 0, true);
|
2003-09-21 03:35:39 +00:00
|
|
|
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");
|
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
const char *remaining_arg = poptGetArg(optCon);
|
|
|
|
|
if ( remaining_arg != NULL) {
|
|
|
|
|
if (opts.source_image != IMAGE_UNKNOWN) {
|
|
|
|
|
fprintf (stderr,
|
|
|
|
|
"%s: Source specified in option %s and as %s\n",
|
|
|
|
|
program_name, source_name, remaining_arg);
|
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (opts.source_image == OP_SOURCE_DEVICE)
|
|
|
|
|
source_name = fillout_device_name(remaining_arg);
|
2003-09-21 18:43:36 +00:00
|
|
|
else
|
|
|
|
|
source_name = strdup(remaining_arg);
|
2003-09-21 03:35:39 +00:00
|
|
|
|
|
|
|
|
if ( (poptGetArgs(optCon)) != NULL) {
|
|
|
|
|
fprintf (stderr,
|
|
|
|
|
"%s: Source specified in previously %s and %s\n",
|
|
|
|
|
program_name, source_name, remaining_arg);
|
|
|
|
|
exit (EXIT_FAILURE);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2003-11-10 03:47:37 +00:00
|
|
|
if (opts.debug_level == 3) {
|
|
|
|
|
cdio_loglevel_default = CDIO_LOG_INFO;
|
|
|
|
|
} else if (opts.debug_level >= 4) {
|
|
|
|
|
cdio_loglevel_default = CDIO_LOG_DEBUG;
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
if (opts.read_mode == READ_MODE_UNINIT) {
|
|
|
|
|
fprintf(stderr,
|
2003-10-16 13:21:56 +00:00
|
|
|
"%s: Need to give a read mode (audio, m1f1, m1f2, m2f1 or m2f2)\n",
|
2003-09-21 03:35:39 +00:00
|
|
|
program_name);
|
|
|
|
|
exit(10);
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-21 18:43:36 +00:00
|
|
|
/* Check consistency between start_lsn, end_lsn and num_sectors. */
|
2003-09-22 01:00:10 +00:00
|
|
|
|
|
|
|
|
if (opts.start_lsn == CDIO_INVALID_LSN) {
|
|
|
|
|
/* Maybe we derive the start from the end and num sectors. */
|
|
|
|
|
if (opts.end_lsn == CDIO_INVALID_LSN) {
|
|
|
|
|
/* No start or end LSN, so use 0 for the start */
|
|
|
|
|
opts.start_lsn = 0;
|
|
|
|
|
if (opts.num_sectors == 0) opts.num_sectors = 1;
|
|
|
|
|
} else if (opts.num_sectors != 0) {
|
|
|
|
|
if (opts.end_lsn <= opts.num_sectors) {
|
|
|
|
|
fprintf(stderr,
|
2003-10-15 02:35:19 +00:00
|
|
|
"%s: end LSN (%lu) needs to be greater than "
|
|
|
|
|
" the sector to read (%lu)\n",
|
|
|
|
|
program_name, (unsigned long) opts.end_lsn,
|
|
|
|
|
(unsigned long) opts.num_sectors);
|
2003-09-22 01:00:10 +00:00
|
|
|
exit(12);
|
|
|
|
|
}
|
|
|
|
|
opts.start_lsn = opts.end_lsn - opts.num_sectors + 1;
|
2003-09-21 18:43:36 +00:00
|
|
|
}
|
2003-09-22 01:00:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* opts.start_lsn has been set somehow or we've aborted. */
|
|
|
|
|
|
|
|
|
|
if (opts.end_lsn == CDIO_INVALID_LSN) {
|
|
|
|
|
if (0 == opts.num_sectors) opts.num_sectors = 1;
|
2003-09-21 18:43:36 +00:00
|
|
|
opts.end_lsn = opts.start_lsn + opts.num_sectors - 1;
|
|
|
|
|
} else {
|
|
|
|
|
/* We were given an end lsn. */
|
|
|
|
|
if (opts.end_lsn < opts.start_lsn) {
|
|
|
|
|
fprintf(stderr,
|
2003-10-15 02:35:19 +00:00
|
|
|
"%s: end LSN (%lu) needs to be less than start LSN (%lu)\n",
|
|
|
|
|
program_name, (unsigned long) opts.start_lsn,
|
|
|
|
|
(unsigned long) opts.end_lsn);
|
2003-09-22 01:00:10 +00:00
|
|
|
exit(13);
|
2003-09-21 18:43:36 +00:00
|
|
|
}
|
|
|
|
|
if (opts.num_sectors != opts.end_lsn - opts.start_lsn + 1)
|
|
|
|
|
if (opts.num_sectors != 0) {
|
|
|
|
|
fprintf(stderr,
|
2003-10-15 02:35:19 +00:00
|
|
|
"%s: inconsistency between start LSN (%lu), end (%lu), "
|
2003-09-21 18:43:36 +00:00
|
|
|
"and count (%d)\n",
|
2003-10-15 02:35:19 +00:00
|
|
|
program_name, (unsigned long) opts.start_lsn,
|
|
|
|
|
(unsigned long) opts.end_lsn, opts.num_sectors);
|
2003-09-22 01:00:10 +00:00
|
|
|
exit(14);
|
2003-09-21 18:43:36 +00:00
|
|
|
}
|
|
|
|
|
opts.num_sectors = opts.end_lsn - opts.start_lsn + 1;
|
2003-09-21 03:35:39 +00:00
|
|
|
}
|
2003-09-21 18:43:36 +00:00
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2003-09-21 04:21:39 +00:00
|
|
|
log_handler (cdio_log_level_t level, const char message[])
|
2003-09-21 03:35:39 +00:00
|
|
|
{
|
2003-09-21 04:21:39 +00:00
|
|
|
if (level == CDIO_LOG_DEBUG && opts.debug_level < 2)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (level == CDIO_LOG_INFO && opts.debug_level < 1)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (level == CDIO_LOG_WARN && opts.debug_level < 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
gl_default_cdio_log_handler (level, message);
|
2003-09-21 03:35:39 +00:00
|
|
|
}
|
|
|
|
|
|
2003-09-21 04:21:39 +00:00
|
|
|
|
2003-09-21 18:43:36 +00:00
|
|
|
static void
|
|
|
|
|
init(void)
|
|
|
|
|
{
|
|
|
|
|
opts.debug_level = 0;
|
2003-09-22 01:00:10 +00:00
|
|
|
opts.start_lsn = CDIO_INVALID_LSN;
|
2003-09-21 18:43:36 +00:00
|
|
|
opts.end_lsn = CDIO_INVALID_LSN;
|
|
|
|
|
opts.num_sectors = 0;
|
|
|
|
|
opts.read_mode = READ_MODE_UNINIT;
|
|
|
|
|
opts.source_image = IMAGE_UNKNOWN;
|
|
|
|
|
|
|
|
|
|
gl_default_cdio_log_handler = cdio_log_set_handler (log_handler);
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-17 12:13:07 +00:00
|
|
|
int
|
|
|
|
|
main(int argc, const char *argv[])
|
|
|
|
|
{
|
|
|
|
|
uint8_t buffer[CDIO_CD_FRAMESIZE_RAW] = { 0, };
|
2003-09-19 04:36:06 +00:00
|
|
|
unsigned int blocklen=CDIO_CD_FRAMESIZE_RAW;
|
2003-09-21 03:35:39 +00:00
|
|
|
CdIo *cdio=NULL;
|
2003-09-27 23:29:29 +00:00
|
|
|
int output_fd=-1;
|
2003-09-17 12:13:07 +00:00
|
|
|
|
2003-09-21 18:43:36 +00:00
|
|
|
init();
|
2003-09-19 04:11:23 +00:00
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
/* Parse our arguments; every option seen by `parse_opt' will
|
|
|
|
|
be reflected in `arguments'. */
|
|
|
|
|
parse_options(argc, argv);
|
|
|
|
|
|
|
|
|
|
/* end of local declarations */
|
|
|
|
|
|
|
|
|
|
switch (opts.source_image) {
|
|
|
|
|
case IMAGE_UNKNOWN:
|
|
|
|
|
case IMAGE_AUTO:
|
|
|
|
|
cdio = cdio_open (source_name, DRIVER_UNKNOWN);
|
|
|
|
|
if (cdio==NULL) {
|
2003-10-03 01:33:15 +00:00
|
|
|
err_exit("Error in automatically selecting driver with input %s\n",
|
|
|
|
|
source_name);
|
2003-09-21 03:35:39 +00:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case IMAGE_DEVICE:
|
|
|
|
|
cdio = cdio_open (source_name, DRIVER_DEVICE);
|
|
|
|
|
if (cdio==NULL) {
|
2003-10-03 01:33:15 +00:00
|
|
|
err_exit("Error in automatically selecting device with input %s\n",
|
|
|
|
|
source_name);
|
2003-09-27 23:29:29 +00:00
|
|
|
|
2003-09-21 03:35:39 +00:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case IMAGE_BIN:
|
|
|
|
|
cdio = cdio_open (source_name, DRIVER_BINCUE);
|
|
|
|
|
if (cdio==NULL) {
|
2003-09-27 23:29:29 +00:00
|
|
|
err_exit("Error in opening bin/cue file %s\n",
|
|
|
|
|
source_name);
|
2003-09-21 03:35:39 +00:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case IMAGE_CUE:
|
|
|
|
|
cdio = cdio_open_cue(source_name);
|
|
|
|
|
if (cdio==NULL) {
|
2003-09-27 23:29:29 +00:00
|
|
|
err_exit("Error in opening cue/bin file %s with input\n",
|
|
|
|
|
source_name);
|
2003-09-21 03:35:39 +00:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case IMAGE_NRG:
|
|
|
|
|
cdio = cdio_open (source_name, DRIVER_NRG);
|
|
|
|
|
if (cdio==NULL) {
|
2003-09-27 23:29:29 +00:00
|
|
|
err_exit("Error in opening NRG file %s for input\n",
|
|
|
|
|
source_name);
|
2003-09-21 03:35:39 +00:00
|
|
|
}
|
|
|
|
|
break;
|
2003-09-17 12:13:07 +00:00
|
|
|
}
|
|
|
|
|
|
2003-09-27 23:29:29 +00:00
|
|
|
if (opts.access_mode!=NULL) {
|
|
|
|
|
cdio_set_arg(cdio, "access-mode", opts.access_mode);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (opts.output_file!=NULL) {
|
|
|
|
|
output_fd = open(opts.output_file, O_WRONLY|O_CREAT|O_TRUNC, 0644);
|
|
|
|
|
if (-1 == output_fd) {
|
|
|
|
|
err_exit("Error opening output file %s: %s\n",
|
|
|
|
|
opts.output_file, strerror(errno));
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-21 18:43:36 +00:00
|
|
|
|
|
|
|
|
for ( ; opts.start_lsn <= opts.end_lsn; opts.start_lsn++ ) {
|
|
|
|
|
switch (opts.read_mode) {
|
|
|
|
|
case READ_AUDIO:
|
2003-10-20 04:28:38 +00:00
|
|
|
if (cdio_read_audio_sector(cdio, &buffer, opts.start_lsn)) {
|
|
|
|
|
fprintf (stderr, "error reading block %u\n",
|
|
|
|
|
(unsigned int) opts.start_lsn);
|
|
|
|
|
blocklen = 0;
|
|
|
|
|
}
|
2003-09-21 18:43:36 +00:00
|
|
|
break;
|
|
|
|
|
case READ_M1F1:
|
2003-10-20 04:28:38 +00:00
|
|
|
if (cdio_read_mode1_sector(cdio, &buffer, opts.start_lsn, false)) {
|
|
|
|
|
fprintf (stderr, "error reading block %u\n",
|
|
|
|
|
(unsigned int) opts.start_lsn);
|
|
|
|
|
blocklen = 0;
|
|
|
|
|
} else
|
|
|
|
|
blocklen=CDIO_CD_FRAMESIZE;
|
2003-09-21 18:43:36 +00:00
|
|
|
break;
|
|
|
|
|
case READ_M1F2:
|
2003-10-20 04:28:38 +00:00
|
|
|
if (cdio_read_mode1_sector(cdio, &buffer, opts.start_lsn, true)) {
|
|
|
|
|
fprintf (stderr, "error reading block %u\n",
|
|
|
|
|
(unsigned int) opts.start_lsn);
|
|
|
|
|
blocklen = 0;
|
|
|
|
|
} else
|
|
|
|
|
blocklen=M2RAW_SECTOR_SIZE;
|
2003-09-21 18:43:36 +00:00
|
|
|
break;
|
|
|
|
|
case READ_M2F1:
|
2003-10-20 04:28:38 +00:00
|
|
|
if (cdio_read_mode2_sector(cdio, &buffer, opts.start_lsn, false)) {
|
|
|
|
|
fprintf (stderr, "error reading block %u\n",
|
|
|
|
|
(unsigned int) opts.start_lsn);
|
|
|
|
|
blocklen=0;
|
|
|
|
|
} else
|
|
|
|
|
blocklen=CDIO_CD_FRAMESIZE;
|
2003-09-21 18:43:36 +00:00
|
|
|
break;
|
|
|
|
|
case READ_M2F2:
|
2003-10-20 04:28:38 +00:00
|
|
|
if (cdio_read_mode2_sector(cdio, &buffer, opts.start_lsn, true)) {
|
|
|
|
|
fprintf (stderr, "error reading block %u\n",
|
|
|
|
|
(unsigned int) opts.start_lsn);
|
|
|
|
|
blocklen=0;
|
|
|
|
|
} else
|
|
|
|
|
blocklen=M2F2_SECTOR_SIZE;
|
2003-09-21 18:43:36 +00:00
|
|
|
break;
|
2003-10-15 01:59:51 +00:00
|
|
|
#if AUTO_FINISHED
|
|
|
|
|
case READ_AUTO:
|
|
|
|
|
/* Find what track lsn is in. Then
|
|
|
|
|
switch cdio_get_track_format(cdio, i)
|
|
|
|
|
and also test using is_green
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
2003-09-21 18:43:36 +00:00
|
|
|
case READ_MODE_UNINIT:
|
|
|
|
|
err_exit("%s: Reading mode not set\n", program_name);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-27 23:29:29 +00:00
|
|
|
if (opts.output_file) {
|
|
|
|
|
write(output_fd, buffer, blocklen);
|
|
|
|
|
} else {
|
|
|
|
|
hexdump(buffer, blocklen);
|
|
|
|
|
}
|
2003-09-17 12:13:07 +00:00
|
|
|
}
|
|
|
|
|
|
2003-09-27 23:29:29 +00:00
|
|
|
if (opts.output_file) close(output_fd);
|
|
|
|
|
|
2003-09-17 12:13:07 +00:00
|
|
|
cdio_destroy(cdio);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|