RetroArch/tasks/task_decompress.c

339 lines
9.2 KiB
C
Raw Normal View History

2015-11-27 20:43:49 +00:00
/* RetroArch - A frontend for libretro.
2016-01-10 03:06:50 +00:00
* Copyright (C) 2011-2016 - Daniel De Matteis
2016-10-27 15:47:10 +00:00
* Copyright (C) 2016 - Brad Parker
2015-11-27 20:43:49 +00:00
*
* RetroArch 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 Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* RetroArch 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 RetroArch.
* If not, see <http://www.gnu.org/licenses/>.
*/
#include <lists/string_list.h>
2015-12-26 06:23:13 +00:00
#include <string/stdstring.h>
2015-11-27 20:43:49 +00:00
#include <file/file_path.h>
#include <file/archive_file.h>
#include <retro_miscellaneous.h>
#include <retro_stat.h>
#include <compat/strl.h>
2015-11-27 20:43:49 +00:00
2016-02-09 16:55:15 +00:00
#include "tasks_internal.h"
2015-11-27 20:43:49 +00:00
#include "../verbosity.h"
#include "../msg_hash.h"
2015-11-27 20:43:49 +00:00
2016-02-03 16:41:04 +00:00
static int file_decompressed_target_file(const char *name,
const char *valid_exts,
const uint8_t *cdata,
unsigned cmode, uint32_t csize, uint32_t size,
2016-09-18 19:20:27 +00:00
uint32_t crc32, struct archive_extract_userdata *userdata)
{
/* TODO/FIXME */
return 0;
}
2016-02-03 16:41:04 +00:00
static int file_decompressed_subdir(const char *name,
const char *valid_exts,
const uint8_t *cdata,
unsigned cmode, uint32_t csize,uint32_t size,
2016-09-18 19:20:27 +00:00
uint32_t crc32, struct archive_extract_userdata *userdata)
{
2016-10-15 23:31:06 +00:00
char path_dir[PATH_MAX_LENGTH];
char path[PATH_MAX_LENGTH];
path_dir[0] = path[0] = '\0';
/* Ignore directories. */
2016-01-21 20:33:52 +00:00
if (name[strlen(name) - 1] == '/' || name[strlen(name) - 1] == '\\')
goto next_file;
2016-09-18 20:05:33 +00:00
if (strstr(name, userdata->dec->subdir) != name)
return 1;
2016-09-18 20:05:33 +00:00
name += strlen(userdata->dec->subdir) + 1;
2016-09-18 20:05:33 +00:00
fill_pathname_join(path, userdata->dec->target_dir, name, sizeof(path));
fill_pathname_basedir(path_dir, path, sizeof(path_dir));
/* Make directory */
if (!path_mkdir(path_dir))
goto error;
2016-01-24 06:42:46 +00:00
if (!file_archive_perform_mode(path, valid_exts,
cdata, cmode, csize, size, crc32, userdata))
goto error;
RARCH_LOG("[deflate subdir] Path: %s, CRC32: 0x%x\n", name, crc32);
next_file:
return 1;
error:
2016-09-18 20:05:33 +00:00
userdata->dec->callback_error = (char*)malloc(PATH_MAX_LENGTH);
snprintf(userdata->dec->callback_error,
2016-02-03 16:41:04 +00:00
PATH_MAX_LENGTH, "Failed to deflate %s.\n", path);
return 0;
}
2015-11-27 20:43:49 +00:00
static int file_decompressed(const char *name, const char *valid_exts,
const uint8_t *cdata, unsigned cmode, uint32_t csize, uint32_t size,
2016-09-18 19:20:27 +00:00
uint32_t crc32, struct archive_extract_userdata *userdata)
2015-11-27 20:43:49 +00:00
{
char path[PATH_MAX_LENGTH];
2016-09-18 20:05:33 +00:00
decompress_state_t *dec = userdata->dec;
2015-11-27 20:43:49 +00:00
path[0] = '\0';
2015-11-27 20:43:49 +00:00
/* Ignore directories. */
2016-01-21 20:33:52 +00:00
if (name[strlen(name) - 1] == '/' || name[strlen(name) - 1] == '\\')
2015-11-27 20:43:49 +00:00
goto next_file;
/* Make directory */
fill_pathname_join(path, dec->target_dir, name, sizeof(path));
path_basedir(path);
if (!path_mkdir(path))
goto error;
fill_pathname_join(path, dec->target_dir, name, sizeof(path));
2016-01-24 06:42:46 +00:00
if (!file_archive_perform_mode(path, valid_exts,
2015-11-27 20:43:49 +00:00
cdata, cmode, csize, size, crc32, userdata))
goto error;
RARCH_LOG("[deflate] Path: %s, CRC32: 0x%x\n", name, crc32);
next_file:
return 1;
error:
dec->callback_error = (char*)malloc(PATH_MAX_LENGTH);
2016-02-03 16:41:04 +00:00
snprintf(dec->callback_error, PATH_MAX_LENGTH,
"Failed to deflate %s.\n", path);
2015-11-27 20:43:49 +00:00
return 0;
}
2016-05-13 08:19:53 +00:00
static void task_decompress_handler_finished(retro_task_t *task,
decompress_state_t *dec)
2015-11-27 20:43:49 +00:00
{
task->finished = true;
if (!task->error && task->cancelled)
2015-11-27 20:43:49 +00:00
task->error = strdup("Task canceled");
2016-01-23 23:06:50 +00:00
if (task->error)
free(dec->source_file);
else
2015-11-27 20:43:49 +00:00
{
decompress_task_data_t *data =
(decompress_task_data_t*)calloc(1, sizeof(*data));
2015-11-27 20:43:49 +00:00
data->source_file = dec->source_file;
2016-01-23 23:06:50 +00:00
task->task_data = data;
2015-11-27 20:43:49 +00:00
}
if (dec->subdir)
free(dec->subdir);
2015-11-27 20:43:49 +00:00
if (dec->valid_ext)
free(dec->valid_ext);
free(dec->target_dir);
free(dec);
}
2016-05-13 08:19:53 +00:00
static void task_decompress_handler(retro_task_t *task)
{
2016-09-19 16:29:10 +00:00
int ret;
2016-09-29 10:02:45 +00:00
bool retdec = false;
struct archive_extract_userdata userdata = {{0}};
2016-09-29 17:24:18 +00:00
decompress_state_t *dec = (decompress_state_t*)task->state;
2016-09-19 16:29:10 +00:00
userdata.dec = dec;
strlcpy(userdata.archive_path, dec->source_file, sizeof(userdata.archive_path));
2016-09-18 20:05:33 +00:00
2016-09-19 16:29:10 +00:00
ret = file_archive_parse_file_iterate(&dec->archive,
2016-02-03 16:41:04 +00:00
&retdec, dec->source_file,
dec->valid_ext, file_decompressed, &userdata);
task->progress = file_archive_parse_file_progress(&dec->archive);
if (task->cancelled || ret != 0)
{
task->error = dec->callback_error;
file_archive_parse_file_iterate_stop(&dec->archive);
2016-05-13 08:19:53 +00:00
task_decompress_handler_finished(task, dec);
}
}
2016-05-13 08:19:53 +00:00
static void task_decompress_handler_target_file(retro_task_t *task)
{
2016-02-03 16:41:04 +00:00
bool retdec;
2016-09-20 01:01:20 +00:00
int ret;
struct archive_extract_userdata userdata = {{0}};
2016-09-29 17:24:18 +00:00
decompress_state_t *dec = (decompress_state_t*)task->state;
2016-09-20 01:01:20 +00:00
2016-09-29 10:02:45 +00:00
strlcpy(userdata.archive_path,
dec->source_file, sizeof(userdata.archive_path));
2016-09-20 00:58:31 +00:00
2016-09-20 01:01:20 +00:00
ret = file_archive_parse_file_iterate(&dec->archive,
2016-02-03 16:41:04 +00:00
&retdec, dec->source_file,
2016-09-18 19:20:27 +00:00
dec->valid_ext, file_decompressed_target_file, &userdata);
task->progress = file_archive_parse_file_progress(&dec->archive);
if (task->cancelled || ret != 0)
{
task->error = dec->callback_error;
file_archive_parse_file_iterate_stop(&dec->archive);
2016-05-13 08:19:53 +00:00
task_decompress_handler_finished(task, dec);
}
}
2016-05-13 08:19:53 +00:00
static void task_decompress_handler_subdir(retro_task_t *task)
{
2016-09-19 16:29:10 +00:00
int ret;
2016-02-03 16:41:04 +00:00
bool retdec;
decompress_state_t *dec = (decompress_state_t*)task->state;
struct archive_extract_userdata userdata = {{0}};
2016-09-19 16:29:10 +00:00
userdata.dec = dec;
strlcpy(userdata.archive_path, dec->source_file, sizeof(userdata.archive_path));
2016-09-18 19:20:27 +00:00
2016-09-19 16:29:10 +00:00
ret = file_archive_parse_file_iterate(&dec->archive,
2016-02-03 16:41:04 +00:00
&retdec, dec->source_file,
dec->valid_ext, file_decompressed_subdir, &userdata);
2016-09-19 16:29:10 +00:00
task->progress = file_archive_parse_file_progress(&dec->archive);
if (task->cancelled || ret != 0)
{
task->error = dec->callback_error;
file_archive_parse_file_iterate_stop(&dec->archive);
2016-05-13 08:19:53 +00:00
task_decompress_handler_finished(task, dec);
}
}
2016-05-13 08:19:53 +00:00
static bool task_decompress_finder(
2016-02-09 16:47:04 +00:00
retro_task_t *task, void *user_data)
{
decompress_state_t *dec = (decompress_state_t*)task->state;
2016-05-13 08:19:53 +00:00
if (task->handler != task_decompress_handler)
return false;
2016-01-20 03:07:24 +00:00
return string_is_equal(dec->source_file, (const char*)user_data);
}
2016-05-27 16:14:47 +00:00
bool task_check_decompress(const char *source_file)
{
task_finder_data_t find_data;
/* Prepare find parameters */
2016-05-13 08:19:53 +00:00
find_data.func = task_decompress_finder;
find_data.userdata = (void *)source_file;
/* Return whether decompressing is in progress or not */
return task_queue_ctl(TASK_QUEUE_CTL_FIND, &find_data);
}
2016-05-27 16:14:47 +00:00
bool task_push_decompress(
const char *source_file,
const char *target_dir,
const char *target_file,
const char *subdir,
const char *valid_ext,
2016-02-09 16:47:04 +00:00
retro_task_callback_t cb,
void *user_data)
2015-11-27 20:43:49 +00:00
{
2016-10-15 23:31:06 +00:00
char tmp[PATH_MAX_LENGTH];
2016-01-21 02:16:48 +00:00
decompress_state_t *s = NULL;
2016-02-09 16:47:04 +00:00
retro_task_t *t = NULL;
2015-11-27 20:43:49 +00:00
2016-10-15 23:31:06 +00:00
tmp[0] = '\0';
2016-01-21 02:16:48 +00:00
if (string_is_empty(target_dir) || string_is_empty(source_file))
{
2016-02-03 16:41:04 +00:00
RARCH_WARN("[decompress] Empty or null source file or"
" target directory arguments.\n");
2015-11-27 20:43:49 +00:00
return false;
}
2015-11-27 20:43:49 +00:00
/* ZIP or APK only */
if (!path_file_exists(source_file) ||
msg_hash_to_file_type(msg_hash_calculate(path_get_extension(source_file)))
2016-06-20 13:50:37 +00:00
!= FILE_TYPE_COMPRESSED)
{
RARCH_WARN("[decompress] File '%s' does not exist or is not a compressed file.\n",
source_file);
2015-11-27 20:43:49 +00:00
return false;
}
2015-11-27 20:43:49 +00:00
if (!valid_ext || !valid_ext[0])
2016-01-21 02:16:48 +00:00
valid_ext = NULL;
2015-11-27 20:43:49 +00:00
2016-05-27 16:14:47 +00:00
if (task_check_decompress(source_file))
{
2016-02-03 16:41:04 +00:00
RARCH_LOG("[decompress] File '%s' already being decompressed.\n",
source_file);
return false;
}
2016-01-10 12:16:55 +00:00
RARCH_LOG("[decompress] File '%s.\n", source_file);
2016-01-21 02:16:48 +00:00
s = (decompress_state_t*)calloc(1, sizeof(*s));
2015-11-27 20:43:49 +00:00
if (!s)
goto error;
2015-11-27 20:43:49 +00:00
s->source_file = strdup(source_file);
s->target_dir = strdup(target_dir);
2016-01-21 02:16:48 +00:00
s->valid_ext = valid_ext ? strdup(valid_ext) : NULL;
s->archive.type = ARCHIVE_TRANSFER_INIT;
2015-11-27 20:43:49 +00:00
2016-02-09 16:47:04 +00:00
t = (retro_task_t*)calloc(1, sizeof(*t));
if (!t)
goto error;
2016-01-21 02:16:48 +00:00
t->state = s;
2016-05-13 08:19:53 +00:00
t->handler = task_decompress_handler;
2015-12-26 06:23:13 +00:00
if (!string_is_empty(subdir))
{
s->subdir = strdup(subdir);
2016-05-13 08:19:53 +00:00
t->handler = task_decompress_handler_subdir;
}
else if (!string_is_empty(target_file))
{
2016-01-26 03:29:19 +00:00
s->target_file = strdup(target_file);
2016-05-13 08:19:53 +00:00
t->handler = task_decompress_handler_target_file;
}
2015-11-27 20:43:49 +00:00
2016-01-21 02:16:48 +00:00
t->callback = cb;
t->user_data = user_data;
2015-11-27 20:43:49 +00:00
2016-02-03 16:41:04 +00:00
snprintf(tmp, sizeof(tmp), "%s '%s'",
msg_hash_to_str(MSG_EXTRACTING), path_basename(source_file));
2016-01-21 02:16:48 +00:00
t->title = strdup(tmp);
2016-02-09 17:26:27 +00:00
task_queue_ctl(TASK_QUEUE_CTL_PUSH, t);
2015-11-27 20:43:49 +00:00
return true;
error:
if (s)
free(s);
return false;
2015-11-27 20:43:49 +00:00
}