2019-07-14 12:54:39 -04:00
|
|
|
/* zstd encoding (ENCODING_ZSTD) backend */
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2019-11-15 11:09:31 -05:00
|
|
|
#include <stdlib.h>
|
2019-07-14 12:54:39 -04:00
|
|
|
#include <string.h>
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
#include <zstd.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "elinks.h"
|
|
|
|
|
|
|
|
#include "encoding/zstd.h"
|
|
|
|
#include "encoding/encoding.h"
|
|
|
|
#include "util/memory.h"
|
|
|
|
|
|
|
|
/* How many bytes of compressed data to read before decompressing.
|
|
|
|
*/
|
|
|
|
#define ELINKS_ZSTD_BUFFER_LENGTH 16384
|
|
|
|
|
|
|
|
struct zstd_enc_data {
|
|
|
|
ZSTD_DCtx *zstd_stream;
|
|
|
|
ZSTD_inBuffer input;
|
|
|
|
ZSTD_outBuffer output;
|
2019-07-14 16:25:03 -04:00
|
|
|
size_t sent_pos;
|
2019-07-14 12:54:39 -04:00
|
|
|
/* The file descriptor from which we read. */
|
|
|
|
int fdread;
|
2019-07-14 16:25:03 -04:00
|
|
|
int decoded:1;
|
2019-07-14 12:54:39 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
|
|
|
zstd_open(struct stream_encoded *stream, int fd)
|
|
|
|
{
|
2022-01-16 15:08:50 -05:00
|
|
|
struct zstd_enc_data *data = (struct zstd_enc_data *)mem_calloc(1, sizeof(*data));
|
2019-07-14 12:54:39 -04:00
|
|
|
|
|
|
|
stream->data = NULL;
|
|
|
|
if (!data) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->fdread = fd;
|
|
|
|
data->zstd_stream = ZSTD_createDCtx();
|
|
|
|
|
|
|
|
if (!data->zstd_stream) {
|
|
|
|
mem_free(data);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
stream->data = data;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-02 10:20:27 -05:00
|
|
|
static char *
|
|
|
|
zstd_decode_buffer(struct stream_encoded *st, char *data, int len, int *new_len)
|
2019-07-14 12:54:39 -04:00
|
|
|
{
|
|
|
|
struct zstd_enc_data *enc_data = (struct zstd_enc_data *)st->data;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
*new_len = 0; /* default, left there if an error occurs */
|
|
|
|
|
|
|
|
enc_data->input.src = data;
|
|
|
|
enc_data->input.pos = 0;
|
|
|
|
enc_data->input.size = len;
|
|
|
|
enc_data->output.pos = 0;
|
|
|
|
enc_data->output.size = 0;
|
|
|
|
enc_data->output.dst = NULL;
|
|
|
|
|
|
|
|
do {
|
2021-01-02 10:20:27 -05:00
|
|
|
char *new_buffer;
|
2019-07-14 12:54:39 -04:00
|
|
|
size_t size = enc_data->output.size + ELINKS_ZSTD_BUFFER_LENGTH;
|
|
|
|
|
2022-01-16 13:38:30 -05:00
|
|
|
new_buffer = (char *)mem_realloc(enc_data->output.dst, size);
|
2019-07-14 12:54:39 -04:00
|
|
|
if (!new_buffer) {
|
|
|
|
error = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
enc_data->output.dst = new_buffer;
|
|
|
|
enc_data->output.size += ELINKS_ZSTD_BUFFER_LENGTH;
|
|
|
|
|
|
|
|
error = ZSTD_decompressStream(enc_data->zstd_stream, &enc_data->output , &enc_data->input);
|
|
|
|
|
|
|
|
if (ZSTD_isError(error)) {
|
|
|
|
mem_free_if(enc_data->output.dst);
|
|
|
|
enc_data->output.dst = NULL;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
} while (enc_data->input.pos < enc_data->input.size);
|
|
|
|
|
|
|
|
*new_len = enc_data->output.pos;
|
|
|
|
return enc_data->output.dst;
|
|
|
|
}
|
|
|
|
|
2019-07-14 16:25:03 -04:00
|
|
|
static int
|
2021-01-02 10:20:27 -05:00
|
|
|
zstd_read(struct stream_encoded *stream, char *buf, int len)
|
2019-07-14 16:25:03 -04:00
|
|
|
{
|
|
|
|
struct zstd_enc_data *data = (struct zstd_enc_data *) stream->data;
|
|
|
|
|
|
|
|
if (!data) return -1;
|
|
|
|
|
|
|
|
assert(len > 0);
|
|
|
|
|
|
|
|
if (!data->decoded) {
|
|
|
|
size_t read_pos = 0;
|
2022-01-16 13:38:30 -05:00
|
|
|
char *tmp_buf = (char *)mem_alloc(len);
|
2019-07-14 16:25:03 -04:00
|
|
|
int new_len;
|
|
|
|
|
|
|
|
if (!tmp_buf) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
do {
|
|
|
|
int l = safe_read(data->fdread, tmp_buf + read_pos, len - read_pos);
|
|
|
|
|
|
|
|
if (!l) break;
|
|
|
|
|
|
|
|
if (l == -1 && errno == EAGAIN) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (l == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
read_pos += l;
|
|
|
|
} while (1);
|
|
|
|
|
|
|
|
if (zstd_decode_buffer(stream, tmp_buf, len, &new_len)) {
|
|
|
|
data->decoded = 1;
|
|
|
|
}
|
|
|
|
mem_free(tmp_buf);
|
|
|
|
}
|
|
|
|
if (data->decoded) {
|
|
|
|
int length = len < (data->output.pos - data->sent_pos) ? len : (data->output.pos - data->sent_pos);
|
|
|
|
|
|
|
|
if (length <= 0) {
|
|
|
|
mem_free(data->output.dst);
|
|
|
|
data->output.dst = NULL;
|
|
|
|
} else {
|
|
|
|
memcpy(buf, data->output.dst + data->sent_pos, length);
|
|
|
|
data->sent_pos += length;
|
|
|
|
}
|
|
|
|
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-14 12:54:39 -04:00
|
|
|
static void
|
|
|
|
zstd_close(struct stream_encoded *stream)
|
|
|
|
{
|
|
|
|
struct zstd_enc_data *data = (struct zstd_enc_data *) stream->data;
|
|
|
|
|
|
|
|
if (data) {
|
|
|
|
if (data->zstd_stream) {
|
|
|
|
ZSTD_freeDCtx(data->zstd_stream);
|
|
|
|
data->zstd_stream = NULL;
|
|
|
|
}
|
|
|
|
if (data->fdread != -1) {
|
|
|
|
close(data->fdread);
|
|
|
|
}
|
|
|
|
mem_free(data);
|
|
|
|
stream->data = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-27 11:02:58 -05:00
|
|
|
const char *
|
|
|
|
get_zstd_version(void)
|
|
|
|
{
|
|
|
|
return ZSTD_versionString();
|
|
|
|
}
|
|
|
|
|
2021-01-02 10:20:27 -05:00
|
|
|
static const char *const zstd_extensions[] = { ".zst", NULL };
|
2019-07-14 12:54:39 -04:00
|
|
|
|
|
|
|
const struct decoding_backend zstd_decoding_backend = {
|
|
|
|
"zstd",
|
|
|
|
zstd_extensions,
|
|
|
|
zstd_open,
|
|
|
|
zstd_read,
|
|
|
|
zstd_decode_buffer,
|
|
|
|
zstd_close,
|
|
|
|
};
|