2017-10-09 23:36:30 -07:00
|
|
|
/* mz_strm_split.c -- Stream for split files
|
2021-01-23 16:18:11 -08:00
|
|
|
part of the minizip-ng project
|
2017-10-09 23:36:30 -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-09 23:36:30 -07:00
|
|
|
|
2017-10-16 07:37:11 -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-09 23:36:30 -07:00
|
|
|
*/
|
|
|
|
|
2017-10-16 07:37:11 -07:00
|
|
|
#include "mz.h"
|
2017-10-26 20:10:02 -07:00
|
|
|
#include "mz_os.h"
|
2017-10-09 23:36:30 -07:00
|
|
|
#include "mz_strm.h"
|
|
|
|
#include "mz_strm_split.h"
|
|
|
|
|
2018-11-21 15:05:58 -08:00
|
|
|
#include <stdio.h> /* snprintf */
|
2023-03-31 18:18:31 -07:00
|
|
|
#include <string.h>
|
2018-11-21 15:05:58 -08:00
|
|
|
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER < 1900)
|
|
|
|
# define snprintf _snprintf
|
|
|
|
#endif
|
|
|
|
|
2017-10-09 23:36:30 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
2018-11-07 20:11:29 -08:00
|
|
|
#define MZ_ZIP_MAGIC_DISKHEADER (0x08074b50)
|
2017-11-26 13:43:38 -05:00
|
|
|
|
2017-10-17 23:20:14 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
2018-05-02 21:34:22 +00:00
|
|
|
static mz_stream_vtbl mz_stream_split_vtbl = {
|
2024-11-01 16:30:39 -07:00
|
|
|
mz_stream_split_open, mz_stream_split_is_open, mz_stream_split_read, mz_stream_split_write,
|
|
|
|
mz_stream_split_tell, mz_stream_split_seek, mz_stream_split_close, mz_stream_split_error,
|
|
|
|
mz_stream_split_create, mz_stream_split_delete, mz_stream_split_get_prop_int64, mz_stream_split_set_prop_int64};
|
2017-10-09 23:36:30 -07:00
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
typedef struct mz_stream_split_s {
|
2024-11-01 16:30:39 -07:00
|
|
|
mz_stream stream;
|
|
|
|
int32_t is_open;
|
|
|
|
int64_t disk_size;
|
|
|
|
int64_t total_in;
|
|
|
|
int64_t total_in_disk;
|
|
|
|
int64_t total_out;
|
|
|
|
int64_t total_out_disk;
|
|
|
|
int32_t mode;
|
|
|
|
char *path_cd;
|
|
|
|
char *path_disk;
|
|
|
|
uint32_t path_disk_size;
|
|
|
|
int32_t number_disk;
|
|
|
|
int32_t current_disk;
|
|
|
|
int64_t current_disk_size;
|
|
|
|
int32_t reached_end;
|
2017-10-09 23:36:30 -07:00
|
|
|
} mz_stream_split;
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
2018-11-07 20:11:29 -08:00
|
|
|
#if 0
|
|
|
|
# define mz_stream_split_print printf
|
|
|
|
#else
|
2023-02-19 10:41:34 -08:00
|
|
|
# define mz_stream_split_print(fmt, ...)
|
2018-11-07 20:11:29 -08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
static int32_t mz_stream_split_open_disk(void *stream, int32_t number_disk) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2017-10-10 19:18:05 -07:00
|
|
|
uint32_t magic = 0;
|
2020-01-04 17:42:51 -08:00
|
|
|
int64_t position = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
int32_t i = 0;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2017-10-26 21:09:49 -07:00
|
|
|
int16_t disk_part = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
|
2018-11-20 16:56:21 -08:00
|
|
|
/* Check if we are reading or writing a disk part or the cd disk */
|
2020-06-14 15:19:14 -07:00
|
|
|
if (number_disk >= 0) {
|
2017-10-26 21:09:49 -07:00
|
|
|
if ((split->mode & MZ_OPEN_MODE_WRITE) == 0)
|
|
|
|
disk_part = MZ_OPEN_MODE_READ;
|
|
|
|
else if (split->disk_size > 0)
|
|
|
|
disk_part = MZ_OPEN_MODE_WRITE;
|
|
|
|
}
|
|
|
|
|
2018-11-20 16:56:21 -08:00
|
|
|
/* Construct disk path */
|
2020-06-14 15:19:14 -07:00
|
|
|
if (disk_part > 0) {
|
|
|
|
for (i = (int32_t)strlen(split->path_disk) - 1; i >= 0; i -= 1) {
|
2017-10-10 19:18:05 -07:00
|
|
|
if (split->path_disk[i] != '.')
|
2017-10-09 23:36:30 -07:00
|
|
|
continue;
|
2024-11-01 16:30:39 -07:00
|
|
|
snprintf(&split->path_disk[i], split->path_disk_size - (uint32_t)i, ".z%02" PRId32, number_disk + 1);
|
2017-10-09 23:36:30 -07:00
|
|
|
break;
|
|
|
|
}
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2018-09-06 22:12:49 +00:00
|
|
|
strncpy(split->path_disk, split->path_cd, split->path_disk_size - 1);
|
2018-09-17 07:46:25 -07:00
|
|
|
split->path_disk[split->path_disk_size - 1] = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
2017-10-10 19:18:05 -07:00
|
|
|
|
2019-09-16 11:07:43 -07:00
|
|
|
mz_stream_split_print("Split - Goto disk - %s (disk %" PRId32 ")\n", split->path_disk, number_disk);
|
2018-11-07 20:11:29 -08:00
|
|
|
|
2018-11-20 16:56:21 -08:00
|
|
|
/* If disk part doesn't exist during reading then return MZ_EXIST_ERROR */
|
2024-09-12 16:29:03 +08:00
|
|
|
if (disk_part == MZ_OPEN_MODE_READ) {
|
2024-10-22 10:29:39 -07:00
|
|
|
if (strcmp(split->path_disk, split->path_cd) == 0) {
|
2024-09-12 16:29:03 +08:00
|
|
|
err = MZ_EXIST_ERROR;
|
|
|
|
} else {
|
|
|
|
err = mz_os_file_exists(split->path_disk);
|
|
|
|
}
|
|
|
|
}
|
2017-10-26 21:09:49 -07:00
|
|
|
|
2017-10-26 20:10:02 -07:00
|
|
|
if (err == MZ_OK)
|
|
|
|
err = mz_stream_open(split->stream.base, split->path_disk, split->mode);
|
2017-10-09 23:36:30 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (err == MZ_OK) {
|
2017-10-10 19:18:05 -07:00
|
|
|
split->total_in_disk = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
split->total_out_disk = 0;
|
2017-10-10 19:18:05 -07:00
|
|
|
split->current_disk = number_disk;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (split->mode & MZ_OPEN_MODE_WRITE) {
|
|
|
|
if ((split->current_disk == 0) && (split->disk_size > 0)) {
|
2017-10-16 07:37:11 -07:00
|
|
|
err = mz_stream_write_uint32(split->stream.base, MZ_ZIP_MAGIC_DISKHEADER);
|
2020-01-03 16:58:16 -08:00
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
split->total_out_disk += 4;
|
|
|
|
split->total_out += split->total_out_disk;
|
|
|
|
}
|
2020-06-14 15:19:14 -07:00
|
|
|
} else if (split->mode & MZ_OPEN_MODE_READ) {
|
|
|
|
if (split->current_disk == 0) {
|
2017-10-10 19:18:05 -07:00
|
|
|
err = mz_stream_read_uint32(split->stream.base, &magic);
|
2017-10-16 07:37:11 -07:00
|
|
|
if (magic != MZ_ZIP_MAGIC_DISKHEADER)
|
2017-10-10 19:18:05 -07:00
|
|
|
err = MZ_FORMAT_ERROR;
|
|
|
|
}
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
}
|
2017-10-10 19:18:05 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (err == MZ_OK) {
|
2020-01-04 17:42:51 -08:00
|
|
|
/* Get the size of the current disk we are on */
|
|
|
|
position = mz_stream_tell(split->stream.base);
|
|
|
|
mz_stream_seek(split->stream.base, 0, MZ_SEEK_END);
|
|
|
|
split->current_disk_size = mz_stream_tell(split->stream.base);
|
|
|
|
mz_stream_seek(split->stream.base, position, MZ_SEEK_SET);
|
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
split->is_open = 1;
|
2020-01-04 17:42:51 -08:00
|
|
|
}
|
2017-10-10 19:18:05 -07:00
|
|
|
|
2017-10-09 23:36:30 -07:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
static int32_t mz_stream_split_close_disk(void *stream) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2017-10-10 19:18:05 -07:00
|
|
|
|
|
|
|
if (mz_stream_is_open(split->stream.base) != MZ_OK)
|
|
|
|
return MZ_OK;
|
|
|
|
|
2018-11-07 20:11:29 -08:00
|
|
|
mz_stream_split_print("Split - Close disk\n");
|
2017-10-09 23:36:30 -07:00
|
|
|
return mz_stream_close(split->stream.base);
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
static int32_t mz_stream_split_goto_disk(void *stream, int32_t number_disk) {
|
2017-10-10 19:18:05 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2018-11-07 20:11:29 -08:00
|
|
|
int32_t err_is_open = MZ_OK;
|
|
|
|
|
|
|
|
err_is_open = mz_stream_is_open(split->stream.base);
|
2017-10-10 19:18:05 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if ((split->disk_size == 0) && (split->mode & MZ_OPEN_MODE_WRITE)) {
|
2018-11-07 20:11:29 -08:00
|
|
|
if (err_is_open != MZ_OK)
|
2017-10-18 16:51:10 -07:00
|
|
|
err = mz_stream_split_open_disk(stream, number_disk);
|
2020-06-14 15:19:14 -07:00
|
|
|
} else if ((number_disk != split->current_disk) || (err_is_open != MZ_OK)) {
|
2017-10-10 19:18:05 -07:00
|
|
|
err = mz_stream_split_close_disk(stream);
|
2020-06-14 15:19:14 -07:00
|
|
|
if (err == MZ_OK) {
|
2017-10-10 19:18:05 -07:00
|
|
|
err = mz_stream_split_open_disk(stream, number_disk);
|
|
|
|
if (err == MZ_OK)
|
|
|
|
split->number_disk = number_disk;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_open(void *stream, const char *path, int32_t mode) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2017-10-10 19:18:05 -07:00
|
|
|
int32_t number_disk = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
|
|
|
|
split->mode = mode;
|
2023-02-24 14:35:31 -08:00
|
|
|
split->path_cd = strdup(path);
|
2018-08-30 16:22:25 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!split->path_cd)
|
2018-08-30 16:22:25 -07:00
|
|
|
return MZ_MEM_ERROR;
|
|
|
|
|
2019-09-16 11:07:43 -07:00
|
|
|
mz_stream_split_print("Split - Open - %s (disk %" PRId32 ")\n", split->path_cd, number_disk);
|
2018-11-07 20:11:29 -08:00
|
|
|
|
2018-10-08 22:40:38 -07:00
|
|
|
split->path_disk_size = (uint32_t)strlen(path) + 10;
|
2023-02-24 14:23:04 -08:00
|
|
|
split->path_disk = (char *)malloc(split->path_disk_size);
|
2018-08-30 16:22:25 -07:00
|
|
|
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!split->path_disk) {
|
2023-02-24 14:23:04 -08:00
|
|
|
free(split->path_cd);
|
2018-08-30 16:22:25 -07:00
|
|
|
return MZ_MEM_ERROR;
|
|
|
|
}
|
|
|
|
|
2018-09-06 22:12:49 +00:00
|
|
|
strncpy(split->path_disk, path, split->path_disk_size - 1);
|
2018-09-17 07:46:25 -07:00
|
|
|
split->path_disk[split->path_disk_size - 1] = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if ((mode & MZ_OPEN_MODE_WRITE) && ((mode & MZ_OPEN_MODE_APPEND) == 0)) {
|
2017-10-10 19:18:05 -07:00
|
|
|
number_disk = 0;
|
|
|
|
split->current_disk = -1;
|
2020-06-14 15:19:14 -07:00
|
|
|
} else {
|
2017-10-10 19:18:05 -07:00
|
|
|
number_disk = -1;
|
|
|
|
split->current_disk = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mz_stream_split_goto_disk(stream, number_disk);
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_is_open(void *stream) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2018-10-28 16:15:13 -07:00
|
|
|
if (split->is_open != 1)
|
|
|
|
return MZ_OPEN_ERROR;
|
|
|
|
return MZ_OK;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_read(void *stream, void *buf, int32_t size) {
|
2017-10-10 19:18:05 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
|
|
|
int32_t bytes_left = size;
|
|
|
|
int32_t read = 0;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2017-10-10 19:18:05 -07:00
|
|
|
uint8_t *buf_ptr = (uint8_t *)buf;
|
|
|
|
|
|
|
|
err = mz_stream_split_goto_disk(stream, split->number_disk);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
return err;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
while (bytes_left > 0) {
|
2017-10-10 19:18:05 -07:00
|
|
|
read = mz_stream_read(split->stream.base, buf_ptr, bytes_left);
|
2018-11-07 20:11:29 -08:00
|
|
|
|
2019-09-16 11:07:43 -07:00
|
|
|
mz_stream_split_print("Split - Read disk - %" PRId32 "\n", read);
|
2018-11-07 20:11:29 -08:00
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
if (read < 0)
|
|
|
|
return read;
|
2020-06-14 15:19:14 -07:00
|
|
|
if (read == 0) {
|
2018-11-20 16:56:21 -08:00
|
|
|
if (split->current_disk < 0) /* No more disks to goto */
|
2017-10-16 14:50:31 -07:00
|
|
|
break;
|
2024-10-30 09:15:29 -07:00
|
|
|
if (size != bytes_left) /* Report read from previous disk before switching */
|
|
|
|
break;
|
2017-10-10 19:18:05 -07:00
|
|
|
err = mz_stream_split_goto_disk(stream, split->current_disk + 1);
|
2024-10-30 09:15:29 -07:00
|
|
|
if (err == MZ_EXIST_ERROR)
|
2018-11-07 20:11:29 -08:00
|
|
|
split->current_disk = -1;
|
2017-10-10 19:18:05 -07:00
|
|
|
if (err != MZ_OK)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes_left -= read;
|
|
|
|
buf_ptr += read;
|
|
|
|
split->total_in += read;
|
|
|
|
split->total_in_disk += read;
|
|
|
|
}
|
|
|
|
return size - bytes_left;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_write(void *stream, const void *buf, int32_t size) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2020-01-03 16:58:16 -08:00
|
|
|
int64_t position = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
int32_t written = 0;
|
|
|
|
int32_t bytes_left = size;
|
2017-10-10 19:18:05 -07:00
|
|
|
int32_t bytes_to_write = 0;
|
2017-10-09 23:36:30 -07:00
|
|
|
int32_t bytes_avail = 0;
|
2017-10-10 19:18:05 -07:00
|
|
|
int32_t number_disk = -1;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2018-05-09 09:42:31 -07:00
|
|
|
const uint8_t *buf_ptr = (const uint8_t *)buf;
|
2017-10-09 23:36:30 -07:00
|
|
|
|
2020-01-03 16:58:16 -08:00
|
|
|
position = mz_stream_tell(split->stream.base);
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
while (bytes_left > 0) {
|
2017-10-10 19:18:05 -07:00
|
|
|
bytes_to_write = bytes_left;
|
2020-02-12 08:25:33 -08:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (split->disk_size > 0) {
|
2017-10-18 16:51:10 -07:00
|
|
|
if ((split->total_out_disk == split->disk_size && split->total_out > 0) ||
|
2020-06-14 15:19:14 -07:00
|
|
|
(split->number_disk == -1 && split->number_disk != split->current_disk)) {
|
2017-10-18 16:51:10 -07:00
|
|
|
if (split->number_disk != -1)
|
|
|
|
number_disk = split->current_disk + 1;
|
|
|
|
|
|
|
|
err = mz_stream_split_goto_disk(stream, number_disk);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
return err;
|
2023-02-12 15:57:00 -08:00
|
|
|
|
|
|
|
position = 0;
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if (split->number_disk != -1) {
|
2017-10-18 16:51:10 -07:00
|
|
|
bytes_avail = (int32_t)(split->disk_size - split->total_out_disk);
|
|
|
|
if (bytes_to_write > bytes_avail)
|
|
|
|
bytes_to_write = bytes_avail;
|
|
|
|
}
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
2018-04-24 10:02:39 +00:00
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
written = mz_stream_write(split->stream.base, buf_ptr, bytes_to_write);
|
|
|
|
if (written != bytes_to_write)
|
2018-10-28 16:15:13 -07:00
|
|
|
return MZ_WRITE_ERROR;
|
2019-07-04 10:32:02 -07:00
|
|
|
|
2019-09-16 11:07:43 -07:00
|
|
|
mz_stream_split_print("Split - Write disk - %" PRId32 "\n", written);
|
2018-04-24 10:02:39 +00:00
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
bytes_left -= written;
|
|
|
|
buf_ptr += written;
|
2020-01-03 16:58:16 -08:00
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
split->total_out += written;
|
|
|
|
split->total_out_disk += written;
|
2020-01-03 16:58:16 -08:00
|
|
|
|
2023-02-12 15:57:00 -08:00
|
|
|
position += written;
|
|
|
|
if (position > split->current_disk_size)
|
|
|
|
split->current_disk_size = position;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return size - bytes_left;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int64_t mz_stream_split_tell(void *stream) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2017-10-10 19:18:05 -07:00
|
|
|
err = mz_stream_split_goto_disk(stream, split->number_disk);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
return err;
|
2017-10-09 23:36:30 -07:00
|
|
|
return mz_stream_tell(split->stream.base);
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_seek(void *stream, int64_t offset, int32_t origin) {
|
2017-10-10 19:18:05 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2018-11-23 10:07:35 -08:00
|
|
|
int64_t disk_left = 0;
|
|
|
|
int64_t position = 0;
|
2017-10-10 19:18:05 -07:00
|
|
|
int32_t err = MZ_OK;
|
2018-11-23 10:07:35 -08:00
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
err = mz_stream_split_goto_disk(stream, split->number_disk);
|
2018-11-23 10:07:35 -08:00
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
if (err != MZ_OK)
|
|
|
|
return err;
|
2018-11-23 10:07:35 -08:00
|
|
|
|
2019-09-16 11:07:43 -07:00
|
|
|
mz_stream_split_print("Split - Seek disk - %" PRId64 " (origin %" PRId32 ")\n", offset, origin);
|
2018-11-23 10:07:35 -08:00
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
if ((origin == MZ_SEEK_CUR) && (split->number_disk != -1)) {
|
2018-11-23 10:07:35 -08:00
|
|
|
position = mz_stream_tell(split->stream.base);
|
|
|
|
disk_left = split->current_disk_size - position;
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
while (offset > disk_left) {
|
2018-11-23 10:07:35 -08:00
|
|
|
err = mz_stream_split_goto_disk(stream, split->current_disk + 1);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
offset -= disk_left;
|
|
|
|
disk_left = split->current_disk_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-10 19:18:05 -07:00
|
|
|
return mz_stream_seek(split->stream.base, offset, origin);
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_close(void *stream) {
|
2017-10-10 19:18:05 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2017-10-10 19:18:05 -07:00
|
|
|
|
|
|
|
err = mz_stream_split_close_disk(stream);
|
|
|
|
split->is_open = 0;
|
|
|
|
return err;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_error(void *stream) {
|
2017-10-10 19:18:05 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
|
|
|
return mz_stream_error(split->stream.base);
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_get_prop_int64(void *stream, int32_t prop, int64_t *value) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2020-06-14 15:19:14 -07:00
|
|
|
switch (prop) {
|
2017-10-09 23:36:30 -07:00
|
|
|
case MZ_STREAM_PROP_TOTAL_OUT:
|
|
|
|
*value = split->total_out;
|
2018-07-11 15:34:45 -07:00
|
|
|
break;
|
2017-10-09 23:36:30 -07:00
|
|
|
case MZ_STREAM_PROP_DISK_NUMBER:
|
|
|
|
*value = split->number_disk;
|
2018-07-11 15:34:45 -07:00
|
|
|
break;
|
2017-10-09 23:36:30 -07:00
|
|
|
case MZ_STREAM_PROP_DISK_SIZE:
|
|
|
|
*value = split->disk_size;
|
2018-07-11 15:34:45 -07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return MZ_EXIST_ERROR;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
2018-07-11 15:34:45 -07:00
|
|
|
return MZ_OK;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
int32_t mz_stream_split_set_prop_int64(void *stream, int32_t prop, int64_t value) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = (mz_stream_split *)stream;
|
2020-06-14 15:19:14 -07:00
|
|
|
switch (prop) {
|
2017-10-10 19:18:05 -07:00
|
|
|
case MZ_STREAM_PROP_DISK_NUMBER:
|
|
|
|
split->number_disk = (int32_t)value;
|
2018-07-11 15:34:45 -07:00
|
|
|
break;
|
2017-10-09 23:36:30 -07:00
|
|
|
case MZ_STREAM_PROP_DISK_SIZE:
|
|
|
|
split->disk_size = value;
|
2018-07-11 15:34:45 -07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return MZ_EXIST_ERROR;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
2018-07-11 15:34:45 -07:00
|
|
|
return MZ_OK;
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
|
2023-03-13 10:50:58 -07:00
|
|
|
void *mz_stream_split_create(void) {
|
|
|
|
mz_stream_split *split = (mz_stream_split *)calloc(1, sizeof(mz_stream_split));
|
2023-02-24 14:35:31 -08:00
|
|
|
if (split)
|
2017-10-09 23:36:30 -07:00
|
|
|
split->stream.vtbl = &mz_stream_split_vtbl;
|
|
|
|
return split;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
void mz_stream_split_delete(void **stream) {
|
2017-10-09 23:36:30 -07:00
|
|
|
mz_stream_split *split = NULL;
|
2023-02-19 11:17:54 -08:00
|
|
|
if (!stream)
|
2017-10-09 23:36:30 -07:00
|
|
|
return;
|
|
|
|
split = (mz_stream_split *)*stream;
|
2023-02-19 11:17:54 -08:00
|
|
|
if (split) {
|
2017-10-10 19:18:05 -07:00
|
|
|
if (split->path_cd)
|
2023-02-24 14:23:04 -08:00
|
|
|
free(split->path_cd);
|
2017-10-10 19:18:05 -07:00
|
|
|
if (split->path_disk)
|
2023-02-24 14:23:04 -08:00
|
|
|
free(split->path_disk);
|
2017-10-10 10:05:20 -07:00
|
|
|
|
2023-02-24 14:23:04 -08:00
|
|
|
free(split);
|
2017-10-09 23:36:30 -07:00
|
|
|
}
|
|
|
|
*stream = NULL;
|
|
|
|
}
|
|
|
|
|
2020-06-14 15:19:14 -07:00
|
|
|
void *mz_stream_split_get_interface(void) {
|
2017-10-09 23:36:30 -07:00
|
|
|
return (void *)&mz_stream_split_vtbl;
|
2017-10-10 11:59:09 -07:00
|
|
|
}
|