2012-01-31 01:40:22 +00:00
|
|
|
/*
|
|
|
|
* Rufus: The Reliable USB Formatting Utility
|
|
|
|
* ISO file extraction
|
|
|
|
* Copyright (c) 2011-2012 Pete Batard <pete@akeo.ie>
|
|
|
|
* Based on libcdio's iso & udf samples:
|
|
|
|
* Copyright (c) 2003-2011 Rocky Bernstein <rocky@gnu.org>
|
|
|
|
*
|
|
|
|
* 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Memory leaks detection - define _CRTDBG_MAP_ALLOC as preprocessor macro */
|
|
|
|
#ifdef _CRTDBG_MAP_ALLOC
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <crtdbg.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <malloc.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <direct.h>
|
|
|
|
|
|
|
|
#include <cdio/cdio.h>
|
|
|
|
#include <cdio/logging.h>
|
|
|
|
#include <cdio/iso9660.h>
|
|
|
|
#include <cdio/udf.h>
|
|
|
|
|
|
|
|
#include "rufus.h"
|
2012-02-01 14:26:36 +00:00
|
|
|
#include "msapi_utf8.h"
|
|
|
|
#include "resource.h"
|
2012-01-31 01:40:22 +00:00
|
|
|
|
|
|
|
#ifndef MIN
|
|
|
|
#define MIN(a,b) (((a) < (b)) ? (a) : (b))
|
|
|
|
#endif
|
2012-02-01 14:26:36 +00:00
|
|
|
#ifndef PBS_MARQUEE
|
|
|
|
#define PBS_MARQUEE 0x08
|
|
|
|
#endif
|
|
|
|
#ifndef PBM_SETMARQUEE
|
|
|
|
#define PBM_SETMARQUEE (WM_USER+10)
|
|
|
|
#endif
|
|
|
|
// How often should we update the progress bar (in 2K blocks) as updating
|
|
|
|
// the progress bar for every block will bring extraction to a crawl
|
|
|
|
#define PROGRESS_UPDATE 1024
|
|
|
|
#define FOUR_GIGABYTES 4294967296LL
|
2012-01-31 01:40:22 +00:00
|
|
|
|
2012-02-01 14:26:36 +00:00
|
|
|
// Needed for UDF ISO access
|
2012-01-31 01:40:22 +00:00
|
|
|
CdIo_t* cdio_open (const char *psz_source, driver_id_t driver_id) {return NULL;}
|
|
|
|
void cdio_destroy (CdIo_t *p_cdio) {}
|
|
|
|
|
2012-02-01 14:26:36 +00:00
|
|
|
RUFUS_ISO_REPORT iso_report;
|
|
|
|
static const char *psz_extract_dir;
|
|
|
|
static uint64_t total_blocks, nb_blocks;
|
|
|
|
static BOOL scan_only = FALSE;
|
2012-01-31 01:40:22 +00:00
|
|
|
|
2012-02-01 14:26:36 +00:00
|
|
|
// TODO: Unicode support, timestamp & permissions preservation
|
2012-01-31 01:40:22 +00:00
|
|
|
|
|
|
|
static int udf_extract_files(udf_t *p_udf, udf_dirent_t *p_udf_dirent, const char *psz_path)
|
|
|
|
{
|
|
|
|
FILE *fd = NULL;
|
|
|
|
int i_length;
|
|
|
|
char* psz_fullpath;
|
|
|
|
const char* psz_basename;
|
|
|
|
udf_dirent_t *p_udf_dirent2;
|
|
|
|
uint8_t buf[UDF_BLOCKSIZE];
|
|
|
|
int64_t i_read, i_file_length;
|
|
|
|
|
|
|
|
if ((p_udf_dirent == NULL) || (psz_path == NULL))
|
|
|
|
return 1;
|
|
|
|
|
2012-02-01 14:26:36 +00:00
|
|
|
while ((p_udf_dirent = udf_readdir(p_udf_dirent)) != NULL) {
|
|
|
|
if (FormatStatus) goto out;
|
2012-01-31 01:40:22 +00:00
|
|
|
psz_basename = udf_get_filename(p_udf_dirent);
|
|
|
|
i_length = (int)(3 + strlen(psz_path) + strlen(psz_basename) + strlen(psz_extract_dir));
|
|
|
|
psz_fullpath = (char*)calloc(sizeof(char), i_length);
|
|
|
|
if (psz_fullpath == NULL) {
|
|
|
|
uprintf("Error allocating file name\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
i_length = _snprintf(psz_fullpath, i_length, "%s%s/%s", psz_extract_dir, psz_path, psz_basename);
|
|
|
|
if (i_length < 0) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (udf_is_dir(p_udf_dirent)) {
|
2012-02-01 14:26:36 +00:00
|
|
|
if (!scan_only)
|
|
|
|
_mkdir(psz_fullpath);
|
2012-01-31 01:40:22 +00:00
|
|
|
p_udf_dirent2 = udf_opendir(p_udf_dirent);
|
|
|
|
if (p_udf_dirent2 != NULL) {
|
|
|
|
if (udf_extract_files(p_udf, p_udf_dirent2, &psz_fullpath[strlen(psz_extract_dir)]))
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
} else {
|
2012-02-01 14:26:36 +00:00
|
|
|
i_file_length = udf_get_file_length(p_udf_dirent);
|
|
|
|
if (scan_only) {
|
|
|
|
if (i_file_length >= FOUR_GIGABYTES)
|
|
|
|
iso_report.has_4GB_file = TRUE;
|
|
|
|
total_blocks += i_file_length/UDF_BLOCKSIZE;
|
|
|
|
if ((i_file_length != 0) && (i_file_length%UDF_BLOCKSIZE == 0))
|
|
|
|
total_blocks++;
|
|
|
|
safe_free(psz_fullpath);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
uprintf("Extracting: %s\n", psz_fullpath);
|
|
|
|
SetWindowTextU(hISOFileName, psz_fullpath);
|
2012-01-31 01:40:22 +00:00
|
|
|
fd = fopen(psz_fullpath, "wb");
|
|
|
|
if (fd == NULL) {
|
|
|
|
uprintf(" Unable to create file\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
while (i_file_length > 0) {
|
2012-02-01 14:26:36 +00:00
|
|
|
if (FormatStatus) goto out;
|
2012-01-31 01:40:22 +00:00
|
|
|
memset(buf, 0, UDF_BLOCKSIZE);
|
|
|
|
i_read = udf_read_block(p_udf_dirent, buf, 1);
|
|
|
|
if (i_read < 0) {
|
|
|
|
uprintf(" Error reading UDF file %s\n", &psz_fullpath[strlen(psz_extract_dir)]);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
fwrite(buf, (size_t)MIN(i_file_length, i_read), 1, fd);
|
|
|
|
if (ferror(fd)) {
|
|
|
|
uprintf(" Error writing file\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
i_file_length -= i_read;
|
2012-02-01 14:26:36 +00:00
|
|
|
if (nb_blocks++ % PROGRESS_UPDATE == 0)
|
|
|
|
SendMessage(hISOProgressBar, PBM_SETPOS, (WPARAM)((MAX_PROGRESS*nb_blocks)/total_blocks), 0);
|
2012-01-31 01:40:22 +00:00
|
|
|
}
|
|
|
|
fclose(fd);
|
|
|
|
fd = NULL;
|
|
|
|
}
|
2012-02-01 14:26:36 +00:00
|
|
|
safe_free(psz_fullpath);
|
2012-01-31 01:40:22 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out:
|
2012-02-01 14:26:36 +00:00
|
|
|
if (p_udf_dirent != NULL)
|
|
|
|
udf_dirent_free(p_udf_dirent);
|
2012-01-31 01:40:22 +00:00
|
|
|
if (fd != NULL)
|
|
|
|
fclose(fd);
|
2012-02-01 14:26:36 +00:00
|
|
|
safe_free(psz_fullpath);
|
2012-01-31 01:40:22 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int iso_extract_files(iso9660_t* p_iso, const char *psz_path)
|
|
|
|
{
|
|
|
|
FILE *fd = NULL;
|
|
|
|
int i_length, r = 1;
|
|
|
|
char psz_fullpath[4096], *psz_basename;
|
|
|
|
const char *psz_iso_name = &psz_fullpath[strlen(psz_extract_dir)];
|
|
|
|
unsigned char buf[ISO_BLOCKSIZE];
|
|
|
|
CdioListNode_t* p_entnode;
|
|
|
|
iso9660_stat_t *p_statbuf;
|
|
|
|
CdioList_t* p_entlist;
|
|
|
|
size_t i;
|
|
|
|
lsn_t lsn;
|
|
|
|
int64_t i_file_length;
|
|
|
|
|
|
|
|
if ((p_iso == NULL) || (psz_path == NULL))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
i_length = _snprintf(psz_fullpath, sizeof(psz_fullpath), "%s%s/", psz_extract_dir, psz_path);
|
|
|
|
if (i_length < 0)
|
|
|
|
return 1;
|
|
|
|
psz_basename = &psz_fullpath[i_length];
|
|
|
|
|
|
|
|
p_entlist = iso9660_ifs_readdir(p_iso, psz_path);
|
|
|
|
if (!p_entlist)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
_CDIO_LIST_FOREACH (p_entnode, p_entlist) {
|
2012-02-01 14:26:36 +00:00
|
|
|
if (FormatStatus) goto out;
|
2012-01-31 01:40:22 +00:00
|
|
|
p_statbuf = (iso9660_stat_t*) _cdio_list_node_data(p_entnode);
|
|
|
|
/* Eliminate . and .. entries */
|
|
|
|
if ( (strcmp(p_statbuf->filename, ".") == 0)
|
|
|
|
|| (strcmp(p_statbuf->filename, "..") == 0) )
|
|
|
|
continue;
|
|
|
|
iso9660_name_translate(p_statbuf->filename, psz_basename);
|
|
|
|
if (p_statbuf->type == _STAT_DIR) {
|
2012-02-01 14:26:36 +00:00
|
|
|
if (!scan_only)
|
|
|
|
_mkdir(psz_fullpath);
|
2012-01-31 01:40:22 +00:00
|
|
|
if (iso_extract_files(p_iso, psz_iso_name))
|
|
|
|
goto out;
|
|
|
|
} else {
|
2012-02-01 14:26:36 +00:00
|
|
|
i_file_length = p_statbuf->size;
|
|
|
|
if (scan_only) {
|
|
|
|
if (i_file_length >= FOUR_GIGABYTES)
|
|
|
|
iso_report.has_4GB_file = TRUE;
|
|
|
|
total_blocks += i_file_length/ISO_BLOCKSIZE;
|
|
|
|
if ((i_file_length != 0) && (i_file_length%ISO_BLOCKSIZE == 0))
|
|
|
|
total_blocks++;
|
|
|
|
continue;
|
|
|
|
}
|
2012-01-31 01:40:22 +00:00
|
|
|
uprintf("Extracting: %s\n", psz_fullpath);
|
|
|
|
fd = fopen(psz_fullpath, "wb");
|
|
|
|
if (fd == NULL) {
|
|
|
|
uprintf(" Unable to create file\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
for (i = 0; i_file_length > 0; i++) {
|
2012-02-01 14:26:36 +00:00
|
|
|
if (FormatStatus) goto out;
|
2012-01-31 01:40:22 +00:00
|
|
|
memset(buf, 0, ISO_BLOCKSIZE);
|
|
|
|
lsn = p_statbuf->lsn + (lsn_t)i;
|
|
|
|
if (iso9660_iso_seek_read(p_iso, buf, lsn, 1) != ISO_BLOCKSIZE) {
|
|
|
|
uprintf(" Error reading ISO9660 file %s at LSN %lu\n",
|
|
|
|
psz_iso_name, (long unsigned int)lsn);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
fwrite(buf, (size_t)MIN(i_file_length, ISO_BLOCKSIZE), 1, fd);
|
|
|
|
if (ferror(fd)) {
|
|
|
|
uprintf(" Error writing file\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
i_file_length -= ISO_BLOCKSIZE;
|
2012-02-01 14:26:36 +00:00
|
|
|
if (nb_blocks++ % PROGRESS_UPDATE == 0)
|
|
|
|
SendMessage(hISOProgressBar, PBM_SETPOS, (WPARAM)((MAX_PROGRESS*nb_blocks)/total_blocks), 0);
|
2012-01-31 01:40:22 +00:00
|
|
|
}
|
|
|
|
fclose(fd);
|
|
|
|
fd = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
r = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (fd != NULL)
|
|
|
|
fclose(fd);
|
|
|
|
_cdio_list_free(p_entlist, true);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2012-02-01 14:26:36 +00:00
|
|
|
BOOL ExtractISO(const char* src_iso, const char* dest_dir, bool scan)
|
2012-01-31 01:40:22 +00:00
|
|
|
{
|
|
|
|
BOOL r = FALSE;
|
|
|
|
iso9660_t* p_iso = NULL;
|
|
|
|
udf_t* p_udf = NULL;
|
|
|
|
udf_dirent_t* p_udf_root;
|
2012-02-01 14:26:36 +00:00
|
|
|
LONG progress_style;
|
|
|
|
const char* scan_text = "Scanning ISO image...\n";
|
|
|
|
|
|
|
|
if ((src_iso == NULL) || (dest_dir == NULL))
|
|
|
|
return FALSE;
|
2012-01-31 01:40:22 +00:00
|
|
|
|
2012-02-01 14:26:36 +00:00
|
|
|
scan_only = scan;
|
2012-01-31 01:40:22 +00:00
|
|
|
cdio_loglevel_default = CDIO_LOG_DEBUG;
|
2012-02-01 14:26:36 +00:00
|
|
|
psz_extract_dir = dest_dir;
|
|
|
|
progress_style = GetWindowLong(hISOProgressBar, GWL_STYLE);
|
|
|
|
if (scan_only) {
|
|
|
|
uprintf(scan_text);
|
|
|
|
total_blocks = 0;
|
|
|
|
iso_report.projected_size = 0;
|
|
|
|
iso_report.has_4GB_file = FALSE;
|
|
|
|
// Change the Window title and static text
|
|
|
|
SetWindowTextU(hISOProgressDlg, scan_text);
|
|
|
|
SetWindowTextU(hISOFileName, scan_text);
|
|
|
|
// Change progress style to marquee for scanning
|
|
|
|
SetWindowLong(hISOProgressBar, GWL_STYLE, progress_style | PBS_MARQUEE);
|
|
|
|
SendMessage(hISOProgressBar, PBM_SETMARQUEE, TRUE, 0);
|
|
|
|
} else {
|
|
|
|
uprintf("Extracting files...\n");
|
|
|
|
if (total_blocks == 0) {
|
|
|
|
uprintf("Error: ISO has not been properly scanned.\n");
|
|
|
|
FormatStatus = ERROR_SEVERITY_ERROR|FAC(FACILITY_STORAGE)|APPERR(ERROR_ISO_SCAN);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
nb_blocks = 0;
|
|
|
|
SetWindowLong(hISOProgressBar, GWL_STYLE, progress_style & (~PBS_MARQUEE));
|
|
|
|
SendMessage(hISOProgressBar, PBM_SETPOS, 0, 0);
|
|
|
|
}
|
|
|
|
ShowWindow(hISOProgressDlg, SW_SHOW);
|
|
|
|
UpdateWindow(hISOProgressDlg);
|
2012-01-31 01:40:22 +00:00
|
|
|
|
|
|
|
/* First try to open as UDF - fallback to ISO if it failed */
|
|
|
|
p_udf = udf_open(src_iso);
|
|
|
|
if (p_udf == NULL)
|
|
|
|
goto try_iso;
|
|
|
|
|
|
|
|
p_udf_root = udf_get_root(p_udf, true, 0);
|
|
|
|
if (p_udf_root == NULL) {
|
|
|
|
uprintf("Couldn't locate UDF root directory\n");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
r = udf_extract_files(p_udf, p_udf_root, "");
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
try_iso:
|
|
|
|
p_iso = iso9660_open(src_iso);
|
|
|
|
if (p_iso == NULL) {
|
|
|
|
uprintf("Unable to open image '%s'.\n", src_iso);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
r = iso_extract_files(p_iso, "");
|
|
|
|
|
|
|
|
out:
|
2012-02-01 14:26:36 +00:00
|
|
|
if (scan_only) {
|
|
|
|
// We use the fact that UDF_BLOCKSIZE and ISO_BLOCKSIZE are the same here
|
|
|
|
iso_report.projected_size = total_blocks * ISO_BLOCKSIZE;
|
|
|
|
}
|
|
|
|
SendMessage(hISOProgressDlg, UM_ISO_EXIT, 0, 0);
|
2012-01-31 01:40:22 +00:00
|
|
|
if (p_iso != NULL)
|
|
|
|
iso9660_close(p_iso);
|
|
|
|
if (p_udf != NULL)
|
|
|
|
udf_close(p_udf);
|
2012-02-01 14:26:36 +00:00
|
|
|
if ((r != 0) && (FormatStatus == 0))
|
|
|
|
FormatStatus = ERROR_SEVERITY_ERROR|FAC(FACILITY_STORAGE)|APPERR(scan_only?ERROR_ISO_SCAN:ERROR_ISO_EXTRACT);
|
2012-01-31 01:40:22 +00:00
|
|
|
return r;
|
|
|
|
}
|