2008-11-07 15:11:08 +00:00
|
|
|
/* io.c - Alpine Package Keeper (APK)
|
|
|
|
*
|
|
|
|
* Copyright (C) 2005-2008 Natanael Copa <n@tanael.org>
|
|
|
|
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
2009-06-11 10:03:10 +00:00
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
2008-11-07 15:11:08 +00:00
|
|
|
* under the terms of the GNU General Public License version 2 as published
|
|
|
|
* by the Free Software Foundation. See http://www.gnu.org/ for details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <malloc.h>
|
2008-11-07 17:06:58 +00:00
|
|
|
#include <sys/mman.h>
|
2008-11-07 15:11:08 +00:00
|
|
|
|
|
|
|
#include "apk_defines.h"
|
|
|
|
#include "apk_io.h"
|
|
|
|
|
|
|
|
struct apk_fd_istream {
|
|
|
|
struct apk_istream is;
|
|
|
|
int fd;
|
|
|
|
};
|
|
|
|
|
2008-11-28 14:28:54 +00:00
|
|
|
static size_t fdi_read(void *stream, void *ptr, size_t size)
|
2008-11-07 15:11:08 +00:00
|
|
|
{
|
|
|
|
struct apk_fd_istream *fis =
|
|
|
|
container_of(stream, struct apk_fd_istream, is);
|
|
|
|
size_t i = 0, r;
|
|
|
|
|
|
|
|
if (ptr == NULL) {
|
|
|
|
if (lseek(fis->fd, size, SEEK_CUR) < 0)
|
|
|
|
return -1;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (i < size) {
|
|
|
|
r = read(fis->fd, ptr + i, size - i);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (r == 0)
|
|
|
|
return i;
|
|
|
|
i += r;
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2008-11-28 14:28:54 +00:00
|
|
|
static void fdi_close(void *stream)
|
2008-11-07 15:11:08 +00:00
|
|
|
{
|
|
|
|
struct apk_fd_istream *fis =
|
|
|
|
container_of(stream, struct apk_fd_istream, is);
|
|
|
|
|
|
|
|
close(fis->fd);
|
|
|
|
free(fis);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_istream *apk_istream_from_fd(int fd)
|
|
|
|
{
|
|
|
|
struct apk_fd_istream *fis;
|
|
|
|
|
2008-11-28 13:03:27 +00:00
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
fis = malloc(sizeof(struct apk_fd_istream));
|
|
|
|
if (fis == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
*fis = (struct apk_fd_istream) {
|
2008-11-28 14:28:54 +00:00
|
|
|
.is.read = fdi_read,
|
|
|
|
.is.close = fdi_close,
|
2008-11-07 15:11:08 +00:00
|
|
|
.fd = fd,
|
|
|
|
};
|
|
|
|
|
|
|
|
return &fis->is;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_istream *apk_istream_from_file(const char *file)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
fd = open(file, O_RDONLY);
|
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2008-11-28 13:03:27 +00:00
|
|
|
fcntl(fd, F_SETFD, FD_CLOEXEC);
|
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
return apk_istream_from_fd(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t apk_istream_skip(struct apk_istream *is, size_t size)
|
|
|
|
{
|
|
|
|
unsigned char buf[2048];
|
|
|
|
size_t done = 0, r, togo;
|
|
|
|
|
|
|
|
while (done < size) {
|
|
|
|
togo = size - done;
|
|
|
|
if (togo > sizeof(buf))
|
|
|
|
togo = sizeof(buf);
|
|
|
|
r = is->read(is, buf, togo);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
done += r;
|
|
|
|
if (r != togo)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return done;
|
|
|
|
}
|
|
|
|
|
2009-01-07 19:45:11 +00:00
|
|
|
size_t apk_istream_splice(void *stream, int fd, size_t size,
|
|
|
|
apk_progress_cb cb, void *cb_ctx)
|
2008-11-07 15:11:08 +00:00
|
|
|
{
|
|
|
|
struct apk_istream *is = (struct apk_istream *) stream;
|
2008-11-07 17:06:58 +00:00
|
|
|
unsigned char *buf;
|
|
|
|
size_t bufsz, done = 0, r, togo;
|
|
|
|
|
|
|
|
bufsz = size;
|
|
|
|
if (bufsz > 256*1024)
|
|
|
|
bufsz = 256*1024;
|
|
|
|
|
|
|
|
buf = malloc(bufsz);
|
|
|
|
if (buf == NULL)
|
|
|
|
return -1;
|
2008-11-07 15:11:08 +00:00
|
|
|
|
|
|
|
while (done < size) {
|
2009-01-07 19:45:11 +00:00
|
|
|
if (done != 0 && cb != NULL)
|
|
|
|
cb(cb_ctx, muldiv(APK_PROGRESS_SCALE, done, size));
|
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
togo = size - done;
|
2008-11-07 17:06:58 +00:00
|
|
|
if (togo > bufsz)
|
|
|
|
togo = bufsz;
|
2008-11-07 15:11:08 +00:00
|
|
|
r = is->read(is, buf, togo);
|
|
|
|
if (r < 0)
|
2008-11-07 17:06:58 +00:00
|
|
|
goto err;
|
|
|
|
if (write(fd, buf, r) != r) {
|
|
|
|
r = -1;
|
|
|
|
goto err;
|
|
|
|
}
|
2008-11-07 15:11:08 +00:00
|
|
|
done += r;
|
|
|
|
if (r != togo)
|
|
|
|
break;
|
|
|
|
}
|
2008-11-07 17:06:58 +00:00
|
|
|
r = done;
|
|
|
|
err:
|
|
|
|
free(buf);
|
|
|
|
return r;
|
2008-11-07 15:11:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_istream_bstream {
|
|
|
|
struct apk_bstream bs;
|
|
|
|
struct apk_istream *is;
|
2009-07-14 06:33:32 +00:00
|
|
|
apk_blob_t left;
|
|
|
|
char buffer[8*1024];
|
2008-11-28 14:28:54 +00:00
|
|
|
size_t size;
|
2008-11-07 15:11:08 +00:00
|
|
|
};
|
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
static apk_blob_t is_bs_read(void *stream, apk_blob_t token)
|
2008-11-07 15:11:08 +00:00
|
|
|
{
|
|
|
|
struct apk_istream_bstream *isbs =
|
|
|
|
container_of(stream, struct apk_istream_bstream, bs);
|
2009-07-14 06:33:32 +00:00
|
|
|
ssize_t size;
|
|
|
|
apk_blob_t ret;
|
|
|
|
|
|
|
|
/* If we have cached stuff, first check if it full fills the request */
|
|
|
|
if (isbs->left.len != 0) {
|
|
|
|
if (!APK_BLOB_IS_NULL(token)) {
|
|
|
|
/* If we have tokenized thingy left, return it */
|
|
|
|
if (apk_blob_split(isbs->left, token, &ret, &isbs->left))
|
|
|
|
goto ret;
|
|
|
|
} else
|
|
|
|
goto ret_all;
|
|
|
|
}
|
2008-11-07 15:11:08 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
/* If we've exchausted earlier, it's end of stream */
|
|
|
|
if (APK_BLOB_IS_NULL(isbs->left))
|
|
|
|
return APK_BLOB_NULL;
|
2008-11-07 15:11:08 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
/* We need more data */
|
|
|
|
if (isbs->left.len != 0)
|
|
|
|
memcpy(isbs->buffer, isbs->left.ptr, isbs->left.len);
|
|
|
|
isbs->left.ptr = isbs->buffer;
|
|
|
|
size = isbs->is->read(isbs->is, isbs->buffer + isbs->left.len,
|
|
|
|
sizeof(isbs->buffer) - isbs->left.len);
|
|
|
|
if (size > 0) {
|
|
|
|
isbs->size += size;
|
|
|
|
isbs->left.len += size;
|
|
|
|
} else if (size == 0) {
|
|
|
|
if (isbs->left.len == 0)
|
|
|
|
isbs->left = APK_BLOB_NULL;
|
|
|
|
goto ret_all;
|
|
|
|
}
|
2008-11-07 15:11:08 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
if (!APK_BLOB_IS_NULL(token)) {
|
|
|
|
/* If we have tokenized thingy left, return it */
|
|
|
|
if (apk_blob_split(isbs->left, token, &ret, &isbs->left))
|
|
|
|
goto ret;
|
|
|
|
/* No token found; just return the full buffer */
|
|
|
|
}
|
|
|
|
|
|
|
|
ret_all:
|
|
|
|
/* Return all that is in cache */
|
|
|
|
ret = isbs->left;
|
|
|
|
isbs->left.len = 0;
|
|
|
|
ret:
|
|
|
|
return ret;
|
2008-11-07 15:11:08 +00:00
|
|
|
}
|
|
|
|
|
2009-07-13 17:37:03 +00:00
|
|
|
static void is_bs_close(void *stream, size_t *size)
|
2008-11-07 15:11:08 +00:00
|
|
|
{
|
|
|
|
struct apk_istream_bstream *isbs =
|
|
|
|
container_of(stream, struct apk_istream_bstream, bs);
|
|
|
|
|
2008-11-28 14:28:54 +00:00
|
|
|
if (size != NULL)
|
|
|
|
*size = isbs->size;
|
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
isbs->is->close(isbs->is);
|
|
|
|
free(isbs);
|
|
|
|
}
|
2008-11-07 17:18:53 +00:00
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
struct apk_bstream *apk_bstream_from_istream(struct apk_istream *istream)
|
|
|
|
{
|
|
|
|
struct apk_istream_bstream *isbs;
|
|
|
|
|
|
|
|
isbs = malloc(sizeof(struct apk_istream_bstream));
|
|
|
|
if (isbs == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
isbs->bs = (struct apk_bstream) {
|
|
|
|
.read = is_bs_read,
|
|
|
|
.close = is_bs_close,
|
|
|
|
};
|
|
|
|
isbs->is = istream;
|
2009-07-14 06:33:32 +00:00
|
|
|
isbs->left = APK_BLOB_PTR_LEN(isbs->buffer, 0),
|
|
|
|
isbs->size = 0;
|
2008-11-07 15:11:08 +00:00
|
|
|
|
|
|
|
return &isbs->bs;
|
|
|
|
}
|
|
|
|
|
2008-11-07 17:06:58 +00:00
|
|
|
struct apk_mmap_bstream {
|
|
|
|
struct apk_bstream bs;
|
|
|
|
int fd;
|
|
|
|
size_t size;
|
|
|
|
unsigned char *ptr;
|
2009-07-14 06:33:32 +00:00
|
|
|
apk_blob_t left;
|
2008-11-07 17:06:58 +00:00
|
|
|
};
|
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
static apk_blob_t mmap_read(void *stream, apk_blob_t token)
|
2008-11-07 17:06:58 +00:00
|
|
|
{
|
|
|
|
struct apk_mmap_bstream *mbs =
|
|
|
|
container_of(stream, struct apk_mmap_bstream, bs);
|
2009-07-14 06:33:32 +00:00
|
|
|
apk_blob_t ret;
|
2008-11-07 17:06:58 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
if (!APK_BLOB_IS_NULL(token) && !APK_BLOB_IS_NULL(mbs->left)) {
|
|
|
|
if (apk_blob_split(mbs->left, token, &ret, &mbs->left))
|
|
|
|
return ret;
|
|
|
|
}
|
2008-11-07 17:06:58 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
ret = mbs->left;
|
|
|
|
mbs->left = APK_BLOB_NULL;
|
2008-11-07 17:06:58 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
return ret;
|
2008-11-07 17:06:58 +00:00
|
|
|
}
|
|
|
|
|
2009-07-13 17:37:03 +00:00
|
|
|
static void mmap_close(void *stream, size_t *size)
|
2008-11-07 17:06:58 +00:00
|
|
|
{
|
|
|
|
struct apk_mmap_bstream *mbs =
|
|
|
|
container_of(stream, struct apk_mmap_bstream, bs);
|
|
|
|
|
2008-11-28 14:28:54 +00:00
|
|
|
if (size != NULL)
|
|
|
|
*size = mbs->size;
|
|
|
|
|
2008-11-07 17:06:58 +00:00
|
|
|
munmap(mbs->ptr, mbs->size);
|
2009-06-11 10:03:10 +00:00
|
|
|
close(mbs->fd);
|
2008-11-07 17:06:58 +00:00
|
|
|
free(mbs);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct apk_bstream *apk_mmap_bstream_from_fd(int fd)
|
|
|
|
{
|
|
|
|
struct apk_mmap_bstream *mbs;
|
|
|
|
struct stat st;
|
|
|
|
void *ptr;
|
|
|
|
|
|
|
|
if (fstat(fd, &st) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
ptr = mmap(NULL, st.st_size, PROT_READ, MAP_SHARED, fd, 0);
|
2008-11-07 17:06:58 +00:00
|
|
|
if (ptr == MAP_FAILED)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
mbs = malloc(sizeof(struct apk_mmap_bstream));
|
|
|
|
if (mbs == NULL) {
|
|
|
|
munmap(ptr, st.st_size);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
mbs->bs = (struct apk_bstream) {
|
2009-07-17 12:56:09 +00:00
|
|
|
.flags = APK_BSTREAM_SINGLE_READ,
|
2008-11-07 17:06:58 +00:00
|
|
|
.read = mmap_read,
|
|
|
|
.close = mmap_close,
|
|
|
|
};
|
|
|
|
mbs->fd = fd;
|
|
|
|
mbs->size = st.st_size;
|
|
|
|
mbs->ptr = ptr;
|
2009-07-14 06:33:32 +00:00
|
|
|
mbs->left = APK_BLOB_PTR_LEN(ptr, mbs->size);
|
2008-11-07 17:06:58 +00:00
|
|
|
|
|
|
|
return &mbs->bs;
|
|
|
|
}
|
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
struct apk_bstream *apk_bstream_from_fd(int fd)
|
|
|
|
{
|
2008-11-07 17:06:58 +00:00
|
|
|
struct apk_bstream *bs;
|
|
|
|
|
2008-11-28 13:03:27 +00:00
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2008-11-07 17:06:58 +00:00
|
|
|
bs = apk_mmap_bstream_from_fd(fd);
|
|
|
|
if (bs != NULL)
|
|
|
|
return bs;
|
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
return apk_bstream_from_istream(apk_istream_from_fd(fd));
|
|
|
|
}
|
|
|
|
|
2008-11-28 13:03:27 +00:00
|
|
|
struct apk_bstream *apk_bstream_from_file(const char *file)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
fd = open(file, O_RDONLY);
|
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
fcntl(fd, F_SETFD, FD_CLOEXEC);
|
|
|
|
return apk_bstream_from_fd(fd);
|
|
|
|
}
|
|
|
|
|
2009-06-28 17:36:16 +00:00
|
|
|
|
|
|
|
struct apk_tee_bstream {
|
|
|
|
struct apk_bstream bs;
|
|
|
|
struct apk_bstream *inner_bs;
|
|
|
|
int fd;
|
|
|
|
size_t size;
|
|
|
|
};
|
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
static apk_blob_t tee_read(void *stream, apk_blob_t token)
|
2009-06-28 17:36:16 +00:00
|
|
|
{
|
|
|
|
struct apk_tee_bstream *tbs =
|
|
|
|
container_of(stream, struct apk_tee_bstream, bs);
|
2009-07-14 06:33:32 +00:00
|
|
|
apk_blob_t blob;
|
2009-06-28 17:36:16 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
blob = tbs->inner_bs->read(tbs->inner_bs, token);
|
|
|
|
if (!APK_BLOB_IS_NULL(blob))
|
|
|
|
tbs->size += write(tbs->fd, blob.ptr, blob.len);
|
2009-06-28 17:36:16 +00:00
|
|
|
|
2009-07-14 06:33:32 +00:00
|
|
|
return blob;
|
2009-06-28 17:36:16 +00:00
|
|
|
}
|
|
|
|
|
2009-07-13 17:37:03 +00:00
|
|
|
static void tee_close(void *stream, size_t *size)
|
2009-06-28 17:36:16 +00:00
|
|
|
{
|
|
|
|
struct apk_tee_bstream *tbs =
|
|
|
|
container_of(stream, struct apk_tee_bstream, bs);
|
|
|
|
|
2009-07-13 17:37:03 +00:00
|
|
|
tbs->inner_bs->close(tbs->inner_bs, NULL);
|
2009-06-28 17:36:16 +00:00
|
|
|
if (size != NULL)
|
|
|
|
*size = tbs->size;
|
|
|
|
close(tbs->fd);
|
|
|
|
free(tbs);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_bstream *apk_bstream_tee(struct apk_bstream *from, const char *to)
|
|
|
|
{
|
|
|
|
struct apk_tee_bstream *tbs;
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
fd = creat(to, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
|
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
tbs = malloc(sizeof(struct apk_tee_bstream));
|
|
|
|
if (tbs == NULL) {
|
|
|
|
close(fd);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
tbs->bs = (struct apk_bstream) {
|
|
|
|
.read = tee_read,
|
|
|
|
.close = tee_close,
|
|
|
|
};
|
|
|
|
tbs->inner_bs = from;
|
|
|
|
tbs->fd = fd;
|
2009-06-29 08:22:55 +00:00
|
|
|
tbs->size = 0;
|
2009-06-28 17:36:16 +00:00
|
|
|
|
|
|
|
return &tbs->bs;
|
|
|
|
}
|
|
|
|
|
2008-11-07 15:11:08 +00:00
|
|
|
apk_blob_t apk_blob_from_istream(struct apk_istream *is, size_t size)
|
|
|
|
{
|
|
|
|
void *ptr;
|
|
|
|
size_t rsize;
|
|
|
|
|
|
|
|
ptr = malloc(size);
|
|
|
|
if (ptr == NULL)
|
|
|
|
return APK_BLOB_NULL;
|
|
|
|
|
|
|
|
rsize = is->read(is, ptr, size);
|
|
|
|
if (rsize < 0) {
|
|
|
|
free(ptr);
|
|
|
|
return APK_BLOB_NULL;
|
|
|
|
}
|
|
|
|
if (rsize != size)
|
|
|
|
ptr = realloc(ptr, rsize);
|
|
|
|
|
|
|
|
return APK_BLOB_PTR_LEN(ptr, rsize);
|
|
|
|
}
|
|
|
|
|
2008-11-27 14:59:04 +00:00
|
|
|
apk_blob_t apk_blob_from_file(const char *file)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
struct stat st;
|
|
|
|
char *buf;
|
|
|
|
|
|
|
|
fd = open(file, O_RDONLY);
|
|
|
|
if (fd < 0)
|
|
|
|
return APK_BLOB_NULL;
|
|
|
|
|
|
|
|
if (fstat(fd, &st) < 0)
|
|
|
|
goto err_fd;
|
|
|
|
|
|
|
|
buf = malloc(st.st_size);
|
|
|
|
if (buf == NULL)
|
|
|
|
goto err_fd;
|
|
|
|
|
|
|
|
if (read(fd, buf, st.st_size) != st.st_size)
|
|
|
|
goto err_read;
|
|
|
|
|
|
|
|
close(fd);
|
|
|
|
return APK_BLOB_PTR_LEN(buf, st.st_size);
|
|
|
|
err_read:
|
|
|
|
free(buf);
|
|
|
|
err_fd:
|
|
|
|
close(fd);
|
|
|
|
return APK_BLOB_NULL;
|
|
|
|
}
|
|
|
|
|
2009-07-14 16:14:05 +00:00
|
|
|
int apk_file_get_info(const char *filename, int checksum, struct apk_file_info *fi)
|
2008-11-14 12:26:59 +00:00
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
struct apk_bstream *bs;
|
|
|
|
|
|
|
|
if (stat(filename, &st) != 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
*fi = (struct apk_file_info) {
|
|
|
|
.size = st.st_size,
|
|
|
|
.uid = st.st_uid,
|
|
|
|
.gid = st.st_gid,
|
|
|
|
.mode = st.st_mode,
|
|
|
|
.mtime = st.st_mtime,
|
|
|
|
.device = st.st_dev,
|
|
|
|
};
|
|
|
|
|
2009-07-14 16:14:05 +00:00
|
|
|
if (checksum == APK_CHECKSUM_NONE)
|
|
|
|
return 0;
|
|
|
|
|
2008-11-28 13:03:27 +00:00
|
|
|
bs = apk_bstream_from_file(filename);
|
2009-07-13 17:37:03 +00:00
|
|
|
if (bs != NULL) {
|
2009-07-14 16:14:05 +00:00
|
|
|
EVP_MD_CTX mdctx;
|
2009-07-14 06:33:32 +00:00
|
|
|
apk_blob_t blob;
|
2009-07-13 17:37:03 +00:00
|
|
|
|
2009-07-14 16:14:05 +00:00
|
|
|
EVP_DigestInit(&mdctx, apk_get_digest(checksum));
|
2009-07-17 12:56:09 +00:00
|
|
|
if (bs->flags & APK_BSTREAM_SINGLE_READ)
|
|
|
|
EVP_MD_CTX_set_flags(&mdctx, EVP_MD_CTX_FLAG_ONESHOT);
|
2009-07-14 06:33:32 +00:00
|
|
|
while (!APK_BLOB_IS_NULL(blob = bs->read(bs, APK_BLOB_NULL)))
|
2009-07-14 16:14:05 +00:00
|
|
|
EVP_DigestUpdate(&mdctx, (void*) blob.ptr, blob.len);
|
|
|
|
fi->csum.type = EVP_MD_CTX_size(&mdctx);
|
|
|
|
EVP_DigestFinal(&mdctx, fi->csum.data, NULL);
|
2009-07-13 17:37:03 +00:00
|
|
|
|
|
|
|
bs->close(bs, NULL);
|
|
|
|
}
|
2008-11-14 12:26:59 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2008-11-27 19:06:45 +00:00
|
|
|
|
|
|
|
struct apk_istream *apk_istream_from_file_gz(const char *file)
|
|
|
|
{
|
2009-07-16 10:47:26 +00:00
|
|
|
return apk_bstream_gunzip(apk_bstream_from_file(file));
|
2008-11-28 14:28:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_fd_ostream {
|
|
|
|
struct apk_ostream os;
|
|
|
|
int fd;
|
2009-03-17 12:17:35 +00:00
|
|
|
size_t bytes;
|
|
|
|
char buffer[1024];
|
2008-11-28 14:28:54 +00:00
|
|
|
};
|
|
|
|
|
2009-03-17 12:17:35 +00:00
|
|
|
static size_t safe_write(int fd, const void *ptr, size_t size)
|
2008-11-28 14:28:54 +00:00
|
|
|
{
|
|
|
|
size_t i = 0, r;
|
|
|
|
|
|
|
|
while (i < size) {
|
2009-03-17 12:17:35 +00:00
|
|
|
r = write(fd, ptr + i, size - i);
|
2008-11-28 14:28:54 +00:00
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (r == 0)
|
|
|
|
return i;
|
|
|
|
i += r;
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2009-03-17 12:17:35 +00:00
|
|
|
static int fdo_flush(struct apk_fd_ostream *fos)
|
|
|
|
{
|
|
|
|
if (fos->bytes == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (safe_write(fos->fd, fos->buffer, fos->bytes) != fos->bytes)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
fos->bytes = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t fdo_write(void *stream, const void *ptr, size_t size)
|
|
|
|
{
|
|
|
|
struct apk_fd_ostream *fos =
|
|
|
|
container_of(stream, struct apk_fd_ostream, os);
|
|
|
|
|
|
|
|
if (size + fos->bytes >= sizeof(fos->buffer)) {
|
|
|
|
if (fdo_flush(fos))
|
|
|
|
return -1;
|
|
|
|
if (size >= sizeof(fos->buffer) / 2)
|
|
|
|
return safe_write(fos->fd, ptr, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(&fos->buffer[fos->bytes], ptr, size);
|
|
|
|
fos->bytes += size;
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2008-11-28 14:28:54 +00:00
|
|
|
static void fdo_close(void *stream)
|
|
|
|
{
|
|
|
|
struct apk_fd_ostream *fos =
|
|
|
|
container_of(stream, struct apk_fd_ostream, os);
|
|
|
|
|
2009-03-17 12:17:35 +00:00
|
|
|
fdo_flush(fos);
|
2008-11-28 14:28:54 +00:00
|
|
|
close(fos->fd);
|
|
|
|
free(fos);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_ostream *apk_ostream_to_fd(int fd)
|
|
|
|
{
|
|
|
|
struct apk_fd_ostream *fos;
|
|
|
|
|
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
fos = malloc(sizeof(struct apk_fd_ostream));
|
|
|
|
if (fos == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
*fos = (struct apk_fd_ostream) {
|
|
|
|
.os.write = fdo_write,
|
|
|
|
.os.close = fdo_close,
|
|
|
|
.fd = fd,
|
|
|
|
};
|
|
|
|
|
|
|
|
return &fos->os;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_ostream *apk_ostream_to_file(const char *file, mode_t mode)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
fd = creat(file, mode);
|
|
|
|
if (fd < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
fcntl(fd, F_SETFD, FD_CLOEXEC);
|
|
|
|
|
|
|
|
return apk_ostream_to_fd(fd);
|
2008-11-27 19:06:45 +00:00
|
|
|
}
|
|
|
|
|
2009-07-16 12:16:05 +00:00
|
|
|
struct apk_counter_ostream {
|
|
|
|
struct apk_ostream os;
|
|
|
|
off_t *counter;
|
|
|
|
};
|
|
|
|
|
|
|
|
static size_t co_write(void *stream, const void *ptr, size_t size)
|
|
|
|
{
|
|
|
|
struct apk_counter_ostream *cos =
|
|
|
|
container_of(stream, struct apk_counter_ostream, os);
|
|
|
|
|
|
|
|
*cos->counter += size;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void co_close(void *stream)
|
|
|
|
{
|
|
|
|
struct apk_counter_ostream *cos =
|
|
|
|
container_of(stream, struct apk_counter_ostream, os);
|
|
|
|
|
|
|
|
free(cos);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct apk_ostream *apk_ostream_counter(off_t *counter)
|
|
|
|
{
|
|
|
|
struct apk_counter_ostream *cos;
|
|
|
|
|
|
|
|
cos = malloc(sizeof(struct apk_counter_ostream));
|
|
|
|
if (cos == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
*cos = (struct apk_counter_ostream) {
|
|
|
|
.os.write = co_write,
|
|
|
|
.os.close = co_close,
|
|
|
|
.counter = counter,
|
|
|
|
};
|
|
|
|
|
|
|
|
return &cos->os;
|
|
|
|
}
|
|
|
|
|
2009-04-16 14:05:27 +00:00
|
|
|
size_t apk_ostream_write_string(struct apk_ostream *os, const char *string)
|
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
len = strlen(string);
|
|
|
|
if (os->write(os, string, len) != len)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|