2017-10-05 23:32:57 -07:00
|
|
|
/* mz_os.c -- System functions
|
2021-01-23 16:18:11 -08:00
|
|
|
part of the minizip-ng project
|
2017-10-05 23:32:57 -07:00
|
|
|
|
2023-02-16 13:14:21 -08:00
|
|
|
Copyright (C) Nathan Moinvaziri
|
2021-01-23 16:18:11 -08:00
|
|
|
https://github.com/zlib-ng/minizip-ng
|
2017-10-23 22:17:03 -07:00
|
|
|
Copyright (C) 1998-2010 Gilles Vollant
|
2018-10-17 22:39:01 +00:00
|
|
|
https://www.winimage.com/zLibDll/minizip.html
|
2017-10-05 23:32:57 -07:00
|
|
|
|
|
|
|
This program is distributed under the terms of the same license as zlib.
|
|
|
|
See the accompanying LICENSE file for the full text of the license.
|
|
|
|
*/
|
|
|
|
|
2017-10-16 07:37:11 -07:00
|
|
|
#include "mz.h"
|
2018-11-03 19:45:41 -07:00
|
|
|
#include "mz_crypt.h"
|
2017-10-05 23:32:57 -07:00
|
|
|
#include "mz_os.h"
|
2017-10-23 22:12:07 -07:00
|
|
|
#include "mz_strm.h"
|
2018-10-25 08:39:48 -07:00
|
|
|
#include "mz_strm_os.h"
|
2017-10-09 00:40:53 -07:00
|
|
|
|
2018-11-20 16:56:21 -08:00
|
|
|
#include <ctype.h> /* tolower */
|
2023-03-31 18:18:31 -07:00
|
|
|
#include <string.h>
|
2018-11-19 21:34:35 -08:00
|
|
|
|
2018-08-27 08:17:16 -07:00
|
|
|
/***************************************************************************/
|
2017-10-09 00:40:53 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_combine(char *path, const char *join, int32_t max_path) {
|
2017-10-16 14:50:31 -07:00
|
|
|
int32_t path_len = 0;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path || !join || !max_path)
|
2017-10-16 14:50:31 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2018-10-08 23:44:45 -07:00
|
|
|
path_len = (int32_t)strlen(path);
|
2017-10-16 14:50:31 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (path_len == 0) {
|
2018-09-06 22:12:49 +00:00
|
|
|
strncpy(path, join, max_path - 1);
|
2018-09-10 08:52:45 -07:00
|
|
|
path[max_path - 1] = 0;
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2019-05-08 23:41:47 -07:00
|
|
|
mz_path_append_slash(path, max_path, MZ_PATH_SLASH_PLATFORM);
|
2022-02-15 16:42:45 -05:00
|
|
|
path_len = (int32_t)strlen(path);
|
|
|
|
if (max_path > path_len)
|
|
|
|
strncat(path, join, max_path - path_len - 1);
|
2017-10-16 14:50:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
2017-10-23 22:12:07 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_append_slash(char *path, int32_t max_path, char slash) {
|
2019-05-02 21:07:39 -07:00
|
|
|
int32_t path_len = (int32_t)strlen(path);
|
2019-05-08 23:41:47 -07:00
|
|
|
if ((path_len + 2) >= max_path)
|
|
|
|
return MZ_BUF_ERROR;
|
2020-06-14 15:19:14 -07:00
|
|
|
if (path[path_len - 1] != '\\' && path[path_len - 1] != '/') {
|
2019-05-08 23:41:47 -07:00
|
|
|
path[path_len] = slash;
|
|
|
|
path[path_len + 1] = 0;
|
|
|
|
}
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_remove_slash(char *path) {
|
2019-05-02 21:07:39 -07:00
|
|
|
int32_t path_len = (int32_t)strlen(path);
|
2020-06-14 15:19:14 -07:00
|
|
|
while (path_len > 0) {
|
2019-05-05 20:13:58 -07:00
|
|
|
if (path[path_len - 1] == '\\' || path[path_len - 1] == '/')
|
2019-05-05 08:19:14 -07:00
|
|
|
path[path_len - 1] = 0;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
|
|
|
|
path_len -= 1;
|
|
|
|
}
|
2019-05-02 21:07:39 -07:00
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_has_slash(const char *path) {
|
2019-05-02 21:07:39 -07:00
|
|
|
int32_t path_len = (int32_t)strlen(path);
|
|
|
|
if (path[path_len - 1] != '\\' && path[path_len - 1] != '/')
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_convert_slashes(char *path, char slash) {
|
2019-05-08 19:24:09 -07:00
|
|
|
int32_t i = 0;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
for (i = 0; i < (int32_t)strlen(path); i += 1) {
|
2019-05-08 19:24:09 -07:00
|
|
|
if (path[i] == '\\' || path[i] == '/')
|
|
|
|
path[i] = slash;
|
|
|
|
}
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_compare_wc(const char *path, const char *wildcard, uint8_t ignore_case) {
|
|
|
|
while (*path != 0) {
|
|
|
|
switch (*wildcard) {
|
2018-07-31 12:48:31 -07:00
|
|
|
case '*':
|
|
|
|
|
|
|
|
if (*(wildcard + 1) == 0)
|
|
|
|
return MZ_OK;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
while (*path != 0) {
|
2018-07-31 12:48:31 -07:00
|
|
|
if (mz_path_compare_wc(path, (wildcard + 1), ignore_case) == MZ_OK)
|
|
|
|
return MZ_OK;
|
|
|
|
|
|
|
|
path += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
|
|
|
|
default:
|
2018-11-20 16:56:21 -08:00
|
|
|
/* Ignore differences in path slashes on platforms */
|
2018-07-31 12:48:31 -07:00
|
|
|
if ((*path == '\\' && *wildcard == '/') || (*path == '/' && *wildcard == '\\'))
|
|
|
|
break;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (ignore_case) {
|
2018-07-31 12:48:31 -07:00
|
|
|
if (tolower(*path) != tolower(*wildcard))
|
|
|
|
return MZ_EXIST_ERROR;
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2018-07-31 12:48:31 -07:00
|
|
|
if (*path != *wildcard)
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
path += 1;
|
|
|
|
wildcard += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((*wildcard != 0) && (*wildcard != '*'))
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_resolve(const char *path, char *output, int32_t max_output) {
|
2018-06-11 07:38:58 -07:00
|
|
|
const char *source = path;
|
2018-06-19 09:44:06 -07:00
|
|
|
const char *check = output;
|
2018-06-11 07:38:58 -07:00
|
|
|
char *target = output;
|
|
|
|
|
|
|
|
if (max_output <= 0)
|
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
while (*source != 0 && max_output > 1) {
|
2018-06-19 09:44:06 -07:00
|
|
|
check = source;
|
|
|
|
if ((*check == '\\') || (*check == '/'))
|
|
|
|
check += 1;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if ((source == path) || (target == output) || (check != source)) {
|
2018-11-20 16:56:21 -08:00
|
|
|
/* Skip double paths */
|
2020-06-14 15:19:14 -07:00
|
|
|
if ((*check == '\\') || (*check == '/')) {
|
2018-06-11 07:38:58 -07:00
|
|
|
source += 1;
|
2018-06-19 09:44:06 -07:00
|
|
|
continue;
|
2018-06-11 07:38:58 -07:00
|
|
|
}
|
2020-06-14 15:35:14 -07:00
|
|
|
if (*check == '.') {
|
2018-06-19 09:44:06 -07:00
|
|
|
check += 1;
|
|
|
|
|
2019-10-29 17:02:46 -07:00
|
|
|
/* Remove . if at end of string and not at the beginning */
|
2020-06-14 15:19:14 -07:00
|
|
|
if ((*check == 0) && (source != path && target != output)) {
|
2018-11-20 16:56:21 -08:00
|
|
|
/* Copy last slash */
|
2018-06-19 09:44:06 -07:00
|
|
|
*target = *source;
|
|
|
|
target += 1;
|
|
|
|
max_output -= 1;
|
|
|
|
source += (check - source);
|
|
|
|
continue;
|
2018-06-11 07:38:58 -07:00
|
|
|
}
|
2019-10-29 17:02:46 -07:00
|
|
|
/* Remove . if not at end of string */
|
2020-06-14 15:19:14 -07:00
|
|
|
else if ((*check == '\\') || (*check == '/')) {
|
2019-10-29 17:02:46 -07:00
|
|
|
source += (check - source);
|
|
|
|
/* Skip slash if at beginning of string */
|
|
|
|
if (target == output && *source != 0)
|
|
|
|
source += 1;
|
|
|
|
continue;
|
2018-06-19 09:44:06 -07:00
|
|
|
}
|
2018-11-20 16:56:21 -08:00
|
|
|
/* Go to parent directory .. */
|
2020-06-14 15:19:14 -07:00
|
|
|
else if (*check == '.') {
|
2018-06-19 09:44:06 -07:00
|
|
|
check += 1;
|
2020-06-14 15:19:14 -07:00
|
|
|
if ((*check == 0) || (*check == '\\' || *check == '/')) {
|
2018-06-19 09:44:06 -07:00
|
|
|
source += (check - source);
|
|
|
|
|
2018-11-20 16:56:21 -08:00
|
|
|
/* Search backwards for previous slash */
|
2020-06-14 15:19:14 -07:00
|
|
|
if (target != output) {
|
2018-06-19 09:44:06 -07:00
|
|
|
target -= 1;
|
2020-06-14 15:19:14 -07:00
|
|
|
do {
|
2018-06-19 09:44:06 -07:00
|
|
|
if ((*target == '\\') || (*target == '/'))
|
|
|
|
break;
|
|
|
|
|
|
|
|
target -= 1;
|
|
|
|
max_output += 1;
|
2020-06-14 15:19:14 -07:00
|
|
|
} while (target > output);
|
2018-06-19 09:44:06 -07:00
|
|
|
}
|
2018-09-04 15:06:53 +00:00
|
|
|
|
2018-06-19 09:44:06 -07:00
|
|
|
if ((target == output) && (*source != 0))
|
|
|
|
source += 1;
|
|
|
|
if ((*target == '\\' || *target == '/') && (*source == 0))
|
|
|
|
target += 1;
|
|
|
|
|
|
|
|
*target = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2018-06-11 07:38:58 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*target = *source;
|
|
|
|
|
|
|
|
source += 1;
|
|
|
|
target += 1;
|
|
|
|
max_output -= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*target = 0;
|
|
|
|
|
|
|
|
if (*path == 0)
|
|
|
|
return MZ_INTERNAL_ERROR;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_remove_filename(char *path) {
|
2018-05-06 16:59:31 -07:00
|
|
|
char *path_ptr = NULL;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2018-05-06 16:59:31 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
2018-09-04 15:06:53 +00:00
|
|
|
path_ptr = path + strlen(path) - 1;
|
2018-05-06 16:59:31 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
while (path_ptr > path) {
|
|
|
|
if ((*path_ptr == '/') || (*path_ptr == '\\')) {
|
2018-05-06 16:59:31 -07:00
|
|
|
*path_ptr = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
path_ptr -= 1;
|
|
|
|
}
|
2019-01-07 13:09:16 -08:00
|
|
|
|
|
|
|
if (path_ptr == path)
|
|
|
|
*path_ptr = 0;
|
|
|
|
|
2018-05-06 16:59:31 -07:00
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_remove_extension(char *path) {
|
2019-04-28 09:23:56 -07:00
|
|
|
char *path_ptr = NULL;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path)
|
2019-04-28 09:23:56 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
|
|
|
path_ptr = path + strlen(path) - 1;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
while (path_ptr > path) {
|
2019-04-28 09:23:56 -07:00
|
|
|
if ((*path_ptr == '/') || (*path_ptr == '\\'))
|
|
|
|
break;
|
2020-06-14 15:19:14 -07:00
|
|
|
if (*path_ptr == '.') {
|
2019-04-28 09:23:56 -07:00
|
|
|
*path_ptr = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
path_ptr -= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path_ptr == path)
|
|
|
|
*path_ptr = 0;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_path_get_filename(const char *path, const char **filename) {
|
2018-05-09 20:03:26 -07:00
|
|
|
const char *match = NULL;
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!path || !filename)
|
2018-05-09 20:03:26 -07:00
|
|
|
return MZ_PARAM_ERROR;
|
|
|
|
|
|
|
|
*filename = NULL;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
for (match = path; *match != 0; match += 1) {
|
2018-05-09 20:03:26 -07:00
|
|
|
if ((*match == '\\') || (*match == '/'))
|
|
|
|
*filename = match + 1;
|
|
|
|
}
|
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!*filename)
|
2018-05-09 20:03:26 -07:00
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_dir_make(const char *path) {
|
2018-08-27 08:17:16 -07:00
|
|
|
int32_t err = MZ_OK;
|
|
|
|
char *current_dir = NULL;
|
|
|
|
char *match = NULL;
|
|
|
|
char hold = 0;
|
|
|
|
|
2023-10-26 09:22:27 +01:00
|
|
|
if (!*path)
|
|
|
|
return MZ_OK;
|
|
|
|
|
2023-02-24 14:35:31 -08:00
|
|
|
current_dir = strdup(path);
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!current_dir)
|
2018-08-27 08:17:16 -07:00
|
|
|
return MZ_MEM_ERROR;
|
|
|
|
|
2019-05-08 19:24:09 -07:00
|
|
|
mz_path_remove_slash(current_dir);
|
2018-08-27 08:17:16 -07:00
|
|
|
|
|
|
|
err = mz_os_make_dir(current_dir);
|
2020-06-14 15:19:14 -07:00
|
|
|
if (err != MZ_OK) {
|
2018-08-27 08:17:16 -07:00
|
|
|
match = current_dir + 1;
|
2020-06-14 15:19:14 -07:00
|
|
|
while (1) {
|
2018-08-27 08:17:16 -07:00
|
|
|
while (*match != 0 && *match != '\\' && *match != '/')
|
|
|
|
match += 1;
|
|
|
|
hold = *match;
|
|
|
|
*match = 0;
|
|
|
|
|
|
|
|
err = mz_os_make_dir(current_dir);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
break;
|
|
|
|
if (hold == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
*match = hold;
|
|
|
|
match += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-24 14:23:04 -08:00
|
|
|
free(current_dir);
|
2018-08-27 08:17:16 -07:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_file_get_crc(const char *path, uint32_t *result_crc) {
|
2017-10-23 22:12:07 -07:00
|
|
|
void *stream = NULL;
|
2018-11-03 19:45:41 -07:00
|
|
|
uint32_t crc32 = 0;
|
2017-10-26 17:52:36 -07:00
|
|
|
int32_t read = 0;
|
2017-10-23 22:12:07 -07:00
|
|
|
int32_t err = MZ_OK;
|
2018-08-30 16:22:25 -07:00
|
|
|
uint8_t buf[16384];
|
2017-10-23 22:12:07 -07:00
|
|
|
|
2023-03-13 10:50:58 -07:00
|
|
|
stream = mz_stream_os_create();
|
|
|
|
if (!stream)
|
|
|
|
return MZ_MEM_ERROR;
|
2017-10-23 22:12:07 -07:00
|
|
|
|
|
|
|
err = mz_stream_os_open(stream, path, MZ_OPEN_MODE_READ);
|
2020-06-14 15:19:14 -07:00
|
|
|
if (err == MZ_OK) {
|
|
|
|
do {
|
2018-11-03 19:45:41 -07:00
|
|
|
read = mz_stream_os_read(stream, buf, sizeof(buf));
|
2017-10-23 22:12:07 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (read < 0) {
|
2017-10-23 22:12:07 -07:00
|
|
|
err = read;
|
|
|
|
break;
|
|
|
|
}
|
2018-11-03 19:45:41 -07:00
|
|
|
|
|
|
|
crc32 = mz_crypt_crc32_update(crc32, buf, read);
|
2020-06-14 15:19:14 -07:00
|
|
|
} while ((err == MZ_OK) && (read > 0));
|
2017-10-23 22:12:07 -07:00
|
|
|
|
|
|
|
mz_stream_os_close(stream);
|
|
|
|
}
|
|
|
|
|
2018-11-03 19:45:41 -07:00
|
|
|
*result_crc = crc32;
|
2017-10-23 22:12:07 -07:00
|
|
|
|
|
|
|
mz_stream_os_delete(&stream);
|
|
|
|
|
|
|
|
return err;
|
2018-05-02 19:59:38 +00:00
|
|
|
}
|
2018-08-27 08:17:16 -07:00
|
|
|
|
|
|
|
/***************************************************************************/
|