|
Packit |
dd8086 |
/*
|
|
Packit |
dd8086 |
Copyright (C) 2005, 2006, 2008-2011, 2017
|
|
Packit |
dd8086 |
Rocky Bernstein <rocky@gnu.org>
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
This program is free software: you can redistribute it and/or modify
|
|
Packit |
dd8086 |
it under the terms of the GNU General Public License as published by
|
|
Packit |
dd8086 |
the Free Software Foundation, either version 3 of the License, or
|
|
Packit |
dd8086 |
(at your option) any later version.
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
This program is distributed in the hope that it will be useful,
|
|
Packit |
dd8086 |
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
dd8086 |
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit |
dd8086 |
GNU General Public License for more details.
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
You should have received a copy of the GNU General Public License
|
|
Packit |
dd8086 |
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
Packit |
dd8086 |
*/
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
/* Simple program to show using libudf to extract a file.
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
This program can be compiled with either a C or C++ compiler. In
|
|
Packit |
dd8086 |
the distribution we prefer C++ just to make sure we haven't broken
|
|
Packit |
dd8086 |
things on the C++ side.
|
|
Packit |
dd8086 |
*/
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
/* config.h has to come first else _FILE_OFFSET_BITS are redefined in
|
|
Packit |
dd8086 |
say opensolaris. */
|
|
Packit |
dd8086 |
#ifdef HAVE_CONFIG_H
|
|
Packit |
dd8086 |
#include "config.h"
|
|
Packit |
dd8086 |
#define __CDIO_CONFIG_H__ 1
|
|
Packit |
dd8086 |
#endif
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
/* This is the UDF image. */
|
|
Packit |
dd8086 |
#define UDF_IMAGE_PATH "../"
|
|
Packit |
dd8086 |
#define UDF_IMAGE "../test/data/udf102.iso"
|
|
Packit |
dd8086 |
#define UDF_FILENAME "/COPYING"
|
|
Packit |
dd8086 |
#define LOCAL_FILENAME "copying"
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
#ifdef HAVE_STDIO_H
|
|
Packit |
dd8086 |
#include <stdio.h>
|
|
Packit |
dd8086 |
#endif
|
|
Packit |
dd8086 |
#ifdef HAVE_ERRNO_H
|
|
Packit |
dd8086 |
#include <errno.h>
|
|
Packit |
dd8086 |
#endif
|
|
Packit |
dd8086 |
#ifdef HAVE_STDLIB_H
|
|
Packit |
dd8086 |
#include <stdlib.h>
|
|
Packit |
dd8086 |
#endif
|
|
Packit |
dd8086 |
#ifdef HAVE_STRING_H
|
|
Packit |
dd8086 |
#include <string.h>
|
|
Packit |
dd8086 |
#endif
|
|
Packit |
dd8086 |
#ifdef HAVE_UNISTD_H
|
|
Packit |
dd8086 |
#include <unistd.h>
|
|
Packit |
dd8086 |
#endif
|
|
Packit |
dd8086 |
#ifdef HAVE_SYS_TYPES_H
|
|
Packit |
dd8086 |
#include <sys/types.h>
|
|
Packit |
dd8086 |
#endif
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
#include <cdio/cdio.h>
|
|
Packit |
dd8086 |
#include <cdio/udf.h>
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
#define CEILING(x, y) ((x+(y-1))/y)
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
#define udf_PATH_DELIMITERS "/\\"
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
int
|
|
Packit |
dd8086 |
main(int argc, const char *argv[])
|
|
Packit |
dd8086 |
{
|
|
Packit |
dd8086 |
udf_t *p_udf;
|
|
Packit |
dd8086 |
FILE *p_outfd;
|
|
Packit |
dd8086 |
char const *psz_udf_image;
|
|
Packit |
dd8086 |
char const *psz_udf_fname;
|
|
Packit |
dd8086 |
char const *psz_local_fname;
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
if (argc > 1)
|
|
Packit |
dd8086 |
psz_udf_image = argv[1];
|
|
Packit |
dd8086 |
else
|
|
Packit |
dd8086 |
psz_udf_image = UDF_IMAGE;
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
if (argc > 2)
|
|
Packit |
dd8086 |
psz_udf_fname = argv[2];
|
|
Packit |
dd8086 |
else
|
|
Packit |
dd8086 |
psz_udf_fname = UDF_FILENAME;
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
if (argc > 3)
|
|
Packit |
dd8086 |
psz_local_fname = argv[3];
|
|
Packit |
dd8086 |
else
|
|
Packit |
dd8086 |
psz_local_fname = LOCAL_FILENAME;
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
p_udf = udf_open (psz_udf_image);
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
if (NULL == p_udf) {
|
|
Packit |
dd8086 |
fprintf(stderr, "Sorry, couldn't open %s as something using UDF\n",
|
|
Packit |
dd8086 |
psz_udf_image);
|
|
Packit |
dd8086 |
return 1;
|
|
Packit |
dd8086 |
} else {
|
|
Packit |
dd8086 |
udf_dirent_t *p_udf_root = udf_get_root(p_udf, true, 0);
|
|
Packit |
dd8086 |
udf_dirent_t *p_udf_file = NULL;
|
|
Packit |
dd8086 |
if (NULL == p_udf_root) {
|
|
Packit |
dd8086 |
fprintf(stderr, "Sorry, couldn't find / in %s\n",
|
|
Packit |
dd8086 |
psz_udf_image);
|
|
Packit |
dd8086 |
udf_close(p_udf);
|
|
Packit |
dd8086 |
return 1;
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
p_udf_file = udf_fopen(p_udf_root, psz_udf_fname);
|
|
Packit |
dd8086 |
if (!p_udf_file) {
|
|
Packit |
dd8086 |
fprintf(stderr, "Sorry, couldn't find %s in %s\n",
|
|
Packit |
dd8086 |
psz_udf_fname, psz_udf_image);
|
|
Packit |
dd8086 |
udf_close(p_udf);
|
|
Packit |
dd8086 |
return 2;
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
if (!(p_outfd = fopen (psz_local_fname, "wb")))
|
|
Packit |
dd8086 |
{
|
|
Packit |
dd8086 |
perror ("fopen()");
|
|
Packit |
dd8086 |
udf_close(p_udf);
|
|
Packit |
dd8086 |
udf_dirent_free(p_udf_file);
|
|
Packit |
dd8086 |
return 3;
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
{
|
|
Packit |
dd8086 |
uint64_t i_file_length = udf_get_file_length(p_udf_file);
|
|
Packit |
dd8086 |
const unsigned int i_blocks = (unsigned int) CEILING(i_file_length, UDF_BLOCKSIZE);
|
|
Packit |
dd8086 |
unsigned int i;
|
|
Packit |
dd8086 |
for (i = 0; i < i_blocks ; i++) {
|
|
Packit |
dd8086 |
char buf[UDF_BLOCKSIZE] = {'\0',};
|
|
Packit |
dd8086 |
ssize_t i_read = udf_read_block(p_udf_file, buf, 1);
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
if ( i_read < 0 ) {
|
|
Packit |
dd8086 |
fprintf(stderr, "Error reading UDF file %s at block %u\n",
|
|
Packit |
dd8086 |
psz_local_fname, i);
|
|
Packit |
dd8086 |
return 4;
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
fwrite (buf, i_read, 1, p_outfd);
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
if (ferror (p_outfd)) {
|
|
Packit |
dd8086 |
perror ("fwrite()");
|
|
Packit |
dd8086 |
return 5;
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
fflush (p_outfd);
|
|
Packit |
dd8086 |
udf_dirent_free(p_udf_file);
|
|
Packit |
dd8086 |
udf_dirent_free(p_udf_root);
|
|
Packit |
dd8086 |
udf_close(p_udf);
|
|
Packit |
dd8086 |
/* Make sure the file size has the exact same byte size. Without the
|
|
Packit |
dd8086 |
truncate below, the file will a multiple of UDF_BLOCKSIZE.
|
|
Packit |
dd8086 |
*/
|
|
Packit |
dd8086 |
if (ftruncate (fileno (p_outfd), i_file_length))
|
|
Packit |
dd8086 |
perror ("ftruncate()");
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
printf("Extraction of file '%s' from %s successful.\n",
|
|
Packit |
dd8086 |
psz_local_fname, psz_udf_image);
|
|
Packit |
dd8086 |
|
|
Packit |
dd8086 |
return 0;
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
}
|
|
Packit |
dd8086 |
}
|