io: add stream copy helper
parent
d60477751f
commit
72be813930
|
@ -74,6 +74,8 @@ struct apk_istream {
|
||||||
const struct apk_istream_ops *ops;
|
const struct apk_istream_ops *ops;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#define APK_IO_ALL ((size_t)-1)
|
||||||
|
|
||||||
struct apk_istream *apk_istream_from_file(int atfd, const char *file);
|
struct apk_istream *apk_istream_from_file(int atfd, const char *file);
|
||||||
struct apk_istream *apk_istream_from_file_gz(int atfd, const char *file);
|
struct apk_istream *apk_istream_from_file_gz(int atfd, const char *file);
|
||||||
struct apk_istream *apk_istream_from_fd(int fd);
|
struct apk_istream *apk_istream_from_fd(int fd);
|
||||||
|
@ -81,12 +83,13 @@ struct apk_istream *apk_istream_from_fd_url_if_modified(int atfd, const char *ur
|
||||||
struct apk_istream *apk_istream_from_url_gz(const char *url);
|
struct apk_istream *apk_istream_from_url_gz(const char *url);
|
||||||
ssize_t apk_istream_read(struct apk_istream *is, void *ptr, size_t size);
|
ssize_t apk_istream_read(struct apk_istream *is, void *ptr, size_t size);
|
||||||
apk_blob_t apk_istream_get(struct apk_istream *is, size_t len);
|
apk_blob_t apk_istream_get(struct apk_istream *is, size_t len);
|
||||||
apk_blob_t apk_istream_get_all(struct apk_istream *is);
|
apk_blob_t apk_istream_get_max(struct apk_istream *is, size_t size);
|
||||||
apk_blob_t apk_istream_get_delim(struct apk_istream *is, apk_blob_t token);
|
apk_blob_t apk_istream_get_delim(struct apk_istream *is, apk_blob_t token);
|
||||||
|
static inline apk_blob_t apk_istream_get_all(struct apk_istream *is) { return apk_istream_get_max(is, APK_IO_ALL); }
|
||||||
#define APK_SPLICE_ALL 0xffffffff
|
|
||||||
ssize_t apk_istream_splice(struct apk_istream *is, int fd, size_t size,
|
ssize_t apk_istream_splice(struct apk_istream *is, int fd, size_t size,
|
||||||
apk_progress_cb cb, void *cb_ctx);
|
apk_progress_cb cb, void *cb_ctx);
|
||||||
|
ssize_t apk_stream_copy(struct apk_istream *is, struct apk_ostream *os, size_t size,
|
||||||
|
apk_progress_cb cb, void *cb_ctx, EVP_MD_CTX *mdctx);
|
||||||
|
|
||||||
static inline struct apk_istream *apk_istream_from_url(const char *url)
|
static inline struct apk_istream *apk_istream_from_url(const char *url)
|
||||||
{
|
{
|
||||||
|
|
|
@ -663,7 +663,7 @@ int apk_cache_download(struct apk_database *db, struct apk_repository *repo,
|
||||||
|
|
||||||
if (fd >= 0) {
|
if (fd >= 0) {
|
||||||
struct apk_file_meta meta;
|
struct apk_file_meta meta;
|
||||||
r = apk_istream_splice(is, fd, APK_SPLICE_ALL, cb, cb_ctx);
|
r = apk_istream_splice(is, fd, APK_IO_ALL, cb, cb_ctx);
|
||||||
if (!autoupdate) {
|
if (!autoupdate) {
|
||||||
apk_istream_get_meta(is, &meta);
|
apk_istream_get_meta(is, &meta);
|
||||||
apk_file_meta_to_fd(fd, &meta);
|
apk_file_meta_to_fd(fd, &meta);
|
||||||
|
|
34
src/io.c
34
src/io.c
|
@ -140,13 +140,13 @@ apk_blob_t apk_istream_get(struct apk_istream *is, size_t len)
|
||||||
return (struct apk_blob) { .len = is->err < 0 ? is->err : 0 };
|
return (struct apk_blob) { .len = is->err < 0 ? is->err : 0 };
|
||||||
}
|
}
|
||||||
|
|
||||||
apk_blob_t apk_istream_get_all(struct apk_istream *is)
|
apk_blob_t apk_istream_get_max(struct apk_istream *is, size_t max)
|
||||||
{
|
{
|
||||||
if (is->ptr == is->end)
|
if (is->ptr == is->end)
|
||||||
__apk_istream_fill(is);
|
__apk_istream_fill(is);
|
||||||
|
|
||||||
if (is->ptr != is->end) {
|
if (is->ptr != is->end) {
|
||||||
apk_blob_t ret = APK_BLOB_PTR_LEN((char*)is->ptr, is->end - is->ptr);
|
apk_blob_t ret = APK_BLOB_PTR_LEN((char*)is->ptr, min((size_t)(is->end - is->ptr), max));
|
||||||
is->ptr = is->end = 0;
|
is->ptr = is->end = 0;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -486,6 +486,32 @@ struct apk_istream *apk_istream_from_file(int atfd, const char *file)
|
||||||
return apk_istream_from_fd(fd);
|
return apk_istream_from_fd(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ssize_t apk_stream_copy(struct apk_istream *is, struct apk_ostream *os, size_t size,
|
||||||
|
apk_progress_cb cb, void *cb_ctx, EVP_MD_CTX *mdctx)
|
||||||
|
{
|
||||||
|
size_t done = 0;
|
||||||
|
apk_blob_t d;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
while (done < size) {
|
||||||
|
if (cb != NULL) cb(cb_ctx, done);
|
||||||
|
|
||||||
|
d = apk_istream_get_max(is, size - done);
|
||||||
|
if (APK_BLOB_IS_NULL(d)) {
|
||||||
|
if (d.len) return d.len;
|
||||||
|
if (size != APK_IO_ALL) return -EBADMSG;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (mdctx) EVP_DigestUpdate(mdctx, d.ptr, d.len);
|
||||||
|
|
||||||
|
r = apk_ostream_write(os, d.ptr, d.len);
|
||||||
|
if (r < 0) return r;
|
||||||
|
|
||||||
|
done += d.len;
|
||||||
|
}
|
||||||
|
return done;
|
||||||
|
}
|
||||||
|
|
||||||
ssize_t apk_istream_splice(struct apk_istream *is, int fd, size_t size,
|
ssize_t apk_istream_splice(struct apk_istream *is, int fd, size_t size,
|
||||||
apk_progress_cb cb, void *cb_ctx)
|
apk_progress_cb cb, void *cb_ctx)
|
||||||
{
|
{
|
||||||
|
@ -496,7 +522,7 @@ ssize_t apk_istream_splice(struct apk_istream *is, int fd, size_t size,
|
||||||
|
|
||||||
bufsz = size;
|
bufsz = size;
|
||||||
if (size > 128 * 1024) {
|
if (size > 128 * 1024) {
|
||||||
if (size != APK_SPLICE_ALL) {
|
if (size != APK_IO_ALL) {
|
||||||
r = posix_fallocate(fd, 0, size);
|
r = posix_fallocate(fd, 0, size);
|
||||||
if (r == 0)
|
if (r == 0)
|
||||||
mmapbase = mmap(NULL, size, PROT_READ | PROT_WRITE,
|
mmapbase = mmap(NULL, size, PROT_READ | PROT_WRITE,
|
||||||
|
@ -521,7 +547,7 @@ ssize_t apk_istream_splice(struct apk_istream *is, int fd, size_t size,
|
||||||
r = apk_istream_read(is, buf, togo);
|
r = apk_istream_read(is, buf, togo);
|
||||||
if (r <= 0) {
|
if (r <= 0) {
|
||||||
if (r) goto err;
|
if (r) goto err;
|
||||||
if (size != APK_SPLICE_ALL && done != size) {
|
if (size != APK_IO_ALL && done != size) {
|
||||||
r = -EBADMSG;
|
r = -EBADMSG;
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue