istream, archive, db: convert db and tar function to use istream

cute-signatures
Timo Teräs 2020-01-11 03:23:22 +02:00
parent 7ca0d146ec
commit 6da3e8eb15
8 changed files with 224 additions and 156 deletions

View File

@ -635,14 +635,9 @@ int main(int argc, char **argv)
apk_blob_pull_deps(&b, &db, &db.world); apk_blob_pull_deps(&b, &db, &db.world);
} }
if (test_installed_db != NULL) { if (test_installed_db != NULL) {
struct apk_bstream *bs = apk_bstream_from_file(AT_FDCWD, test_installed_db); apk_db_index_read(&db, apk_istream_from_file(AT_FDCWD, test_installed_db), -1);
if (!IS_ERR_OR_NULL(bs)) {
apk_db_index_read(&db, bs, -1);
apk_bstream_close(bs);
}
} }
for (i = 0; i < test_repos->num; i++) { for (i = 0; i < test_repos->num; i++) {
struct apk_bstream *bs;
apk_blob_t spec = APK_BLOB_STR(test_repos->item[i]), name, tag; apk_blob_t spec = APK_BLOB_STR(test_repos->item[i]), name, tag;
int repo_tag = 0, repo = APK_REPOSITORY_FIRST_CONFIGURED + i; int repo_tag = 0, repo = APK_REPOSITORY_FIRST_CONFIGURED + i;
@ -659,14 +654,11 @@ int main(int argc, char **argv)
name = spec; name = spec;
} }
bs = apk_bstream_from_file(AT_FDCWD, name.ptr); if (apk_db_index_read(&db, apk_istream_from_file(AT_FDCWD, name.ptr), repo) != 0) {
if (IS_ERR_OR_NULL(bs)) {
apk_error("Failed to open repository: " BLOB_FMT, BLOB_PRINTF(name)); apk_error("Failed to open repository: " BLOB_FMT, BLOB_PRINTF(name));
goto err; goto err;
} }
apk_db_index_read(&db, bs, repo);
apk_bstream_close(bs);
if (repo != -2) { if (repo != -2) {
if (!(apk_flags & APK_NO_NETWORK)) if (!(apk_flags & APK_NO_NETWORK))
db.available_repos |= BIT(repo); db.available_repos |= BIT(repo);

View File

@ -234,7 +234,7 @@ struct apk_package *apk_db_pkg_add(struct apk_database *db, struct apk_package *
struct apk_package *apk_db_get_pkg(struct apk_database *db, struct apk_checksum *csum); struct apk_package *apk_db_get_pkg(struct apk_database *db, struct apk_checksum *csum);
struct apk_package *apk_db_get_file_owner(struct apk_database *db, apk_blob_t filename); struct apk_package *apk_db_get_file_owner(struct apk_database *db, apk_blob_t filename);
int apk_db_index_read(struct apk_database *db, struct apk_bstream *bs, int repo); int apk_db_index_read(struct apk_database *db, struct apk_istream *is, int repo);
int apk_db_index_read_file(struct apk_database *db, const char *file, int repo); int apk_db_index_read_file(struct apk_database *db, const char *file, int repo);
int apk_db_index_write(struct apk_database *db, struct apk_ostream *os); int apk_db_index_write(struct apk_database *db, struct apk_ostream *os);

View File

@ -81,6 +81,9 @@ struct apk_istream *apk_istream_from_fd(int fd);
struct apk_istream *apk_istream_from_fd_url_if_modified(int atfd, const char *url, time_t since); struct apk_istream *apk_istream_from_fd_url_if_modified(int atfd, const char *url, time_t since);
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_all(struct apk_istream *is);
apk_blob_t apk_istream_get_delim(struct apk_istream *is, apk_blob_t token);
#define APK_SPLICE_ALL 0xffffffff #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,
@ -107,6 +110,14 @@ static inline void apk_istream_close(struct apk_istream *is)
is->ops->close(is); is->ops->close(is);
} }
struct apk_segment_istream {
struct apk_istream is;
struct apk_istream *pis;
size_t bytes_left;
time_t mtime;
};
void apk_istream_segment(struct apk_segment_istream *sis, struct apk_istream *is, size_t len, time_t mtime);
#define APK_BSTREAM_SINGLE_READ 0x0001 #define APK_BSTREAM_SINGLE_READ 0x0001
#define APK_BSTREAM_EOF 0x0002 #define APK_BSTREAM_EOF 0x0002

View File

@ -73,55 +73,6 @@ static void put_octal(char *s, size_t l, size_t value)
*(ptr--) = '0'; *(ptr--) = '0';
} }
struct apk_tar_entry_istream {
struct apk_istream is;
struct apk_istream *tar_is;
size_t bytes_left;
time_t mtime;
};
static void tar_entry_get_meta(struct apk_istream *is, struct apk_file_meta *meta)
{
struct apk_tar_entry_istream *teis = container_of(is, struct apk_tar_entry_istream, is);
*meta = (struct apk_file_meta) {
.atime = teis->mtime,
.mtime = teis->mtime,
};
}
static ssize_t tar_entry_read(struct apk_istream *is, void *ptr, size_t size)
{
struct apk_tar_entry_istream *teis = container_of(is, struct apk_tar_entry_istream, is);
ssize_t r;
if (size > teis->bytes_left)
size = teis->bytes_left;
if (size == 0)
return 0;
r = apk_istream_read(teis->tar_is, ptr, size);
if (r <= 0) {
/* If inner stream returned zero (end-of-stream), we
* are getting short read, because tar header indicated
* more was to be expected. */
if (r == 0) return -ECONNABORTED;
return r;
}
teis->bytes_left -= r;
return r;
}
static void tar_entry_close(struct apk_istream *is)
{
}
static const struct apk_istream_ops tar_istream_ops = {
.get_meta = tar_entry_get_meta,
.read = tar_entry_read,
.close = tar_entry_close,
};
static int blob_realloc(apk_blob_t *b, size_t newsize) static int blob_realloc(apk_blob_t *b, size_t newsize)
{ {
char *tmp; char *tmp;
@ -180,12 +131,8 @@ int apk_tar_parse(struct apk_istream *is, apk_archive_entry_parser parser,
void *ctx, struct apk_id_cache *idc) void *ctx, struct apk_id_cache *idc)
{ {
struct apk_file_info entry; struct apk_file_info entry;
struct apk_tar_entry_istream teis = { struct apk_segment_istream segment;
.is.ops = &tar_istream_ops,
.tar_is = is,
};
struct tar_header buf; struct tar_header buf;
unsigned long offset = 0;
int end = 0, r; int end = 0, r;
size_t toskip, paxlen = 0; size_t toskip, paxlen = 0;
apk_blob_t pax = APK_BLOB_NULL, longname = APK_BLOB_NULL; apk_blob_t pax = APK_BLOB_NULL, longname = APK_BLOB_NULL;
@ -194,7 +141,6 @@ int apk_tar_parse(struct apk_istream *is, apk_archive_entry_parser parser,
memset(&entry, 0, sizeof(entry)); memset(&entry, 0, sizeof(entry));
entry.name = buf.name; entry.name = buf.name;
while ((r = apk_istream_read(is, &buf, 512)) == 512) { while ((r = apk_istream_read(is, &buf, 512)) == 512) {
offset += 512;
if (buf.name[0] == '\0') { if (buf.name[0] == '\0') {
if (end) break; if (end) break;
end++; end++;
@ -222,7 +168,6 @@ int apk_tar_parse(struct apk_istream *is, apk_archive_entry_parser parser,
} }
buf.mode[0] = 0; /* to nul terminate 100-byte buf.name */ buf.mode[0] = 0; /* to nul terminate 100-byte buf.name */
buf.magic[0] = 0; /* to nul terminate 100-byte buf.linkname */ buf.magic[0] = 0; /* to nul terminate 100-byte buf.linkname */
teis.mtime = entry.mtime;
apk_xattr_array_resize(&entry.xattrs, 0); apk_xattr_array_resize(&entry.xattrs, 0);
if (entry.size >= SSIZE_MAX-512) goto err; if (entry.size >= SSIZE_MAX-512) goto err;
@ -232,6 +177,7 @@ int apk_tar_parse(struct apk_istream *is, apk_archive_entry_parser parser,
apk_fileinfo_hash_xattr(&entry); apk_fileinfo_hash_xattr(&entry);
} }
toskip = (entry.size + 511) & -512;
switch (buf.typeflag) { switch (buf.typeflag) {
case 'L': /* GNU long name extension */ case 'L': /* GNU long name extension */
if ((r = blob_realloc(&longname, entry.size+1)) != 0 || if ((r = blob_realloc(&longname, entry.size+1)) != 0 ||
@ -239,8 +185,7 @@ int apk_tar_parse(struct apk_istream *is, apk_archive_entry_parser parser,
goto err; goto err;
entry.name = longname.ptr; entry.name = longname.ptr;
entry.name[entry.size] = 0; entry.name[entry.size] = 0;
offset += entry.size; toskip -= entry.size;
entry.size = 0;
break; break;
case 'K': /* GNU long link target extension - ignored */ case 'K': /* GNU long link target extension - ignored */
break; break;
@ -272,11 +217,10 @@ int apk_tar_parse(struct apk_istream *is, apk_archive_entry_parser parser,
break; break;
case 'x': /* file specific pax header */ case 'x': /* file specific pax header */
paxlen = entry.size; paxlen = entry.size;
entry.size = 0;
if ((r = blob_realloc(&pax, (paxlen + 511) & -512)) != 0 || if ((r = blob_realloc(&pax, (paxlen + 511) & -512)) != 0 ||
(r = apk_istream_read(is, pax.ptr, paxlen)) != paxlen) (r = apk_istream_read(is, pax.ptr, paxlen)) != paxlen)
goto err; goto err;
offset += paxlen; toskip -= entry.size;
break; break;
default: default:
break; break;
@ -288,27 +232,20 @@ int apk_tar_parse(struct apk_istream *is, apk_archive_entry_parser parser,
goto err; goto err;
} }
teis.bytes_left = entry.size;
if (entry.mode & S_IFMT) { if (entry.mode & S_IFMT) {
r = parser(ctx, &entry, &teis.is); apk_istream_segment(&segment, is, entry.size, entry.mtime);
r = parser(ctx, &entry, &segment.is);
if (r != 0) goto err; if (r != 0) goto err;
apk_istream_close(&segment.is);
entry.name = buf.name; entry.name = buf.name;
toskip -= entry.size;
paxlen = 0; paxlen = 0;
} }
offset += entry.size - teis.bytes_left; if (toskip && (r = apk_istream_read(is, NULL, toskip)) != toskip)
toskip = teis.bytes_left;
if ((offset + toskip) & 511)
toskip += 512 - ((offset + toskip) & 511);
offset += toskip;
if (toskip != 0) {
if ((r = apk_istream_read(is, NULL, toskip)) != toskip) {
r = -EIO;
goto err; goto err;
} }
}
}
/* Read remaining end-of-archive records, to ensure we read all of /* Read remaining end-of-archive records, to ensure we read all of
* the file. The underlying istream is likely doing checksumming. */ * the file. The underlying istream is likely doing checksumming. */

View File

@ -174,13 +174,11 @@ int apk_blob_split(apk_blob_t blob, apk_blob_t split, apk_blob_t *l, apk_blob_t
{ {
char *pos = blob.ptr, *end = blob.ptr + blob.len - split.len + 1; char *pos = blob.ptr, *end = blob.ptr + blob.len - split.len + 1;
if (end < pos) if (!pos || end < pos) return 0;
return 0;
while (1) { while (1) {
pos = memchr(pos, split.ptr[0], end - pos); pos = memchr(pos, split.ptr[0], end - pos);
if (pos == NULL) if (!pos) return 0;
return 0;
if (split.len > 1 && memcmp(pos, split.ptr, split.len) != 0) { if (split.len > 1 && memcmp(pos, split.ptr, split.len) != 0) {
pos++; pos++;

View File

@ -715,27 +715,26 @@ static struct apk_db_dir_instance *find_diri(struct apk_installed_package *ipkg,
return NULL; return NULL;
} }
int apk_db_read_overlay(struct apk_database *db, struct apk_bstream *bs) int apk_db_read_overlay(struct apk_database *db, struct apk_istream *is)
{ {
struct apk_db_dir_instance *diri = NULL; struct apk_db_dir_instance *diri = NULL;
struct hlist_node **diri_node = NULL, **file_diri_node = NULL; struct hlist_node **diri_node = NULL, **file_diri_node = NULL;
struct apk_package *pkg; struct apk_package *pkg;
struct apk_installed_package *ipkg; struct apk_installed_package *ipkg;
apk_blob_t token = APK_BLOB_STR("\n"), line, bdir, bfile; apk_blob_t token = APK_BLOB_STR("\n"), line, bdir, bfile;
int r = -1;
if (IS_ERR_OR_NULL(bs)) return -1; if (IS_ERR_OR_NULL(is)) return -1;
pkg = apk_pkg_new(); pkg = apk_pkg_new();
if (pkg == NULL) if (pkg == NULL) goto err;
return -1;
ipkg = apk_pkg_install(db, pkg); ipkg = apk_pkg_install(db, pkg);
if (ipkg == NULL) if (ipkg == NULL) goto err;
return -1;
diri_node = hlist_tail_ptr(&ipkg->owned_dirs); diri_node = hlist_tail_ptr(&ipkg->owned_dirs);
while (!APK_BLOB_IS_NULL(line = apk_bstream_read(bs, token))) { while (!APK_BLOB_IS_NULL(line = apk_istream_get_delim(is, token))) {
if (!apk_blob_rsplit(line, '/', &bdir, &bfile)) if (!apk_blob_rsplit(line, '/', &bdir, &bfile))
break; break;
@ -752,11 +751,13 @@ int apk_db_read_overlay(struct apk_database *db, struct apk_bstream *bs)
(void) apk_db_file_get(db, diri, bfile, &file_diri_node); (void) apk_db_file_get(db, diri, bfile, &file_diri_node);
} }
} }
r = 0;
return 0; err:
apk_istream_close(is);
return r;
} }
int apk_db_index_read(struct apk_database *db, struct apk_bstream *bs, int repo) int apk_db_index_read(struct apk_database *db, struct apk_istream *is, int repo)
{ {
struct apk_package *pkg = NULL; struct apk_package *pkg = NULL;
struct apk_installed_package *ipkg = NULL; struct apk_installed_package *ipkg = NULL;
@ -772,7 +773,9 @@ int apk_db_index_read(struct apk_database *db, struct apk_bstream *bs, int repo)
gid_t gid; gid_t gid;
int field, r, lineno = 0; int field, r, lineno = 0;
while (!APK_BLOB_IS_NULL(l = apk_bstream_read(bs, token))) { if (IS_ERR_OR_NULL(is)) return PTR_ERR(is);
while (!APK_BLOB_IS_NULL(l = apk_istream_get_delim(is, token))) {
lineno++; lineno++;
if (l.len < 2 || l.ptr[1] != ':') { if (l.len < 2 || l.ptr[1] != ':') {
@ -893,13 +896,16 @@ int apk_db_index_read(struct apk_database *db, struct apk_bstream *bs, int repo)
} }
if (APK_BLOB_IS_NULL(l)) goto bad_entry; if (APK_BLOB_IS_NULL(l)) goto bad_entry;
} }
apk_istream_close(is);
return 0; return 0;
old_apk_tools: old_apk_tools:
/* Installed db should not have unsupported fields */ /* Installed db should not have unsupported fields */
apk_error("This apk-tools is too old to handle installed packages"); apk_error("This apk-tools is too old to handle installed packages");
return -1; goto err;
bad_entry: bad_entry:
apk_error("FDB format error (line %d, entry '%c')", lineno, field); apk_error("FDB format error (line %d, entry '%c')", lineno, field);
err:
apk_istream_close(is);
return -1; return -1;
} }
@ -1114,14 +1120,16 @@ static void apk_db_triggers_write(struct apk_database *db, struct apk_ostream *o
} }
} }
static void apk_db_triggers_read(struct apk_database *db, struct apk_bstream *bs) static void apk_db_triggers_read(struct apk_database *db, struct apk_istream *is)
{ {
struct apk_checksum csum; struct apk_checksum csum;
struct apk_package *pkg; struct apk_package *pkg;
struct apk_installed_package *ipkg; struct apk_installed_package *ipkg;
apk_blob_t l; apk_blob_t l;
while (!APK_BLOB_IS_NULL(l = apk_bstream_read(bs, APK_BLOB_STR("\n")))) { if (IS_ERR_OR_NULL(is)) return;
while (!APK_BLOB_IS_NULL(l = apk_istream_get_delim(is, APK_BLOB_STR("\n")))) {
apk_blob_pull_csum(&l, &csum); apk_blob_pull_csum(&l, &csum);
apk_blob_pull_char(&l, ' '); apk_blob_pull_char(&l, ' ');
@ -1136,12 +1144,12 @@ static void apk_db_triggers_read(struct apk_database *db, struct apk_bstream *bs
list_add_tail(&ipkg->trigger_pkgs_list, list_add_tail(&ipkg->trigger_pkgs_list,
&db->installed.triggers); &db->installed.triggers);
} }
apk_istream_close(is);
} }
static int apk_db_read_state(struct apk_database *db, int flags) static int apk_db_read_state(struct apk_database *db, int flags)
{ {
struct apk_istream *is; struct apk_istream *is;
struct apk_bstream *bs;
apk_blob_t blob, world; apk_blob_t blob, world;
int r; int r;
@ -1153,26 +1161,16 @@ static int apk_db_read_state(struct apk_database *db, int flags)
*/ */
if (!(flags & APK_OPENF_NO_WORLD)) { if (!(flags & APK_OPENF_NO_WORLD)) {
blob = world = apk_blob_from_file(db->root_fd, apk_world_file); blob = world = apk_blob_from_file(db->root_fd, apk_world_file);
if (APK_BLOB_IS_NULL(blob)) if (APK_BLOB_IS_NULL(blob)) return -ENOENT;
return -ENOENT;
blob = apk_blob_trim(blob); blob = apk_blob_trim(blob);
apk_blob_pull_deps(&blob, db, &db->world); apk_blob_pull_deps(&blob, db, &db->world);
free(world.ptr); free(world.ptr);
} }
if (!(flags & APK_OPENF_NO_INSTALLED)) { if (!(flags & APK_OPENF_NO_INSTALLED)) {
bs = apk_bstream_from_file(db->root_fd, apk_installed_file); r = apk_db_index_read(db, apk_istream_from_file(db->root_fd, apk_installed_file), -1);
if (!IS_ERR_OR_NULL(bs)) {
r = apk_db_index_read(db, bs, -1);
apk_bstream_close(bs);
if (r != 0) return -1; if (r != 0) return -1;
} apk_db_triggers_read(db, apk_istream_from_file(db->root_fd, apk_triggers_file));
bs = apk_bstream_from_file(db->root_fd, apk_triggers_file);
if (!IS_ERR_OR_NULL(bs)) {
apk_db_triggers_read(db, bs);
apk_bstream_close(bs);
}
} }
if (!(flags & APK_OPENF_NO_SCRIPTS)) { if (!(flags & APK_OPENF_NO_SCRIPTS)) {
@ -1514,7 +1512,6 @@ int apk_db_open(struct apk_database *db, struct apk_db_options *dbopts)
{ {
const char *msg = NULL; const char *msg = NULL;
struct apk_repository_list *repo = NULL; struct apk_repository_list *repo = NULL;
struct apk_bstream *bs;
struct statfs stfs; struct statfs stfs;
apk_blob_t blob; apk_blob_t blob;
int r, fd, write_arch = FALSE; int r, fd, write_arch = FALSE;
@ -1662,8 +1659,7 @@ int apk_db_open(struct apk_database *db, struct apk_db_options *dbopts)
if (apk_flags & APK_OVERLAY_FROM_STDIN) { if (apk_flags & APK_OVERLAY_FROM_STDIN) {
apk_flags &= ~APK_OVERLAY_FROM_STDIN; apk_flags &= ~APK_OVERLAY_FROM_STDIN;
apk_db_read_overlay(db, apk_bstream_from_istream( apk_db_read_overlay(db, apk_istream_from_fd(STDIN_FILENO));
apk_istream_from_fd(STDIN_FILENO)));
} }
r = apk_db_read_state(db, dbopts->open_flags); r = apk_db_read_state(db, dbopts->open_flags);
@ -1682,11 +1678,7 @@ int apk_db_open(struct apk_database *db, struct apk_db_options *dbopts)
if (!(dbopts->open_flags & APK_OPENF_NO_INSTALLED_REPO)) { if (!(dbopts->open_flags & APK_OPENF_NO_INSTALLED_REPO)) {
if (apk_db_cache_active(db)) { if (apk_db_cache_active(db)) {
bs = apk_bstream_from_file(db->cache_fd, "installed"); apk_db_index_read(db, apk_istream_from_file(db->cache_fd, "installed"), -2);
if (!IS_ERR_OR_NULL(bs)) {
apk_db_index_read(db, bs, -2);
apk_bstream_close(bs);
}
} }
} }
@ -2165,7 +2157,6 @@ static int load_apkindex(void *sctx, const struct apk_file_info *fi,
struct apk_istream *is) struct apk_istream *is)
{ {
struct apkindex_ctx *ctx = (struct apkindex_ctx *) sctx; struct apkindex_ctx *ctx = (struct apkindex_ctx *) sctx;
struct apk_bstream *bs;
struct apk_repository *repo; struct apk_repository *repo;
int r; int r;
@ -2179,11 +2170,7 @@ static int load_apkindex(void *sctx, const struct apk_file_info *fi,
repo->description = apk_blob_from_istream(is, fi->size); repo->description = apk_blob_from_istream(is, fi->size);
} else if (strcmp(fi->name, "APKINDEX") == 0) { } else if (strcmp(fi->name, "APKINDEX") == 0) {
ctx->found = 1; ctx->found = 1;
bs = apk_bstream_from_istream(is); apk_db_index_read(ctx->db, is, ctx->repo);
if (!IS_ERR_OR_NULL(bs)) {
apk_db_index_read(ctx->db, bs, ctx->repo);
apk_bstream_close(bs);
}
} }
return 0; return 0;
@ -2212,11 +2199,7 @@ static int load_index(struct apk_database *db, struct apk_bstream *bs,
if (r >= 0 && ctx.found == 0) if (r >= 0 && ctx.found == 0)
r = -ENOMSG; r = -ENOMSG;
} else { } else {
bs = apk_bstream_from_istream(apk_bstream_gunzip(bs)); apk_db_index_read(db, apk_bstream_gunzip(bs), repo);
if (!IS_ERR_OR_NULL(bs)) {
apk_db_index_read(db, bs, repo);
apk_bstream_close(bs);
}
} }
return r; return r;
} }
@ -2440,9 +2423,9 @@ static int apk_db_install_archive_entry(void *_ctx,
if (ae->name[0] == '.') { if (ae->name[0] == '.') {
/* APK 2.0 format */ /* APK 2.0 format */
if (strcmp(ae->name, ".PKGINFO") == 0) { if (strcmp(ae->name, ".PKGINFO") == 0) {
apk_blob_t blob = apk_blob_from_istream(is, ae->size); apk_blob_t l, token = APK_BLOB_STR("\n");
apk_blob_for_each_segment(blob, "\n", read_info_line, ctx); while (!APK_BLOB_IS_NULL(l = apk_istream_get_delim(is, token)))
free(blob.ptr); read_info_line(ctx, l);
return 0; return 0;
} }
type = apk_script_type(&ae->name[1]); type = apk_script_type(&ae->name[1]);

164
src/io.c
View File

@ -35,7 +35,7 @@
#define HAVE_FGETGRENT_R #define HAVE_FGETGRENT_R
#endif #endif
size_t apk_io_bufsize = 2*1024; size_t apk_io_bufsize = 8*1024;
static void apk_file_meta_from_fd(int fd, struct apk_file_meta *meta) static void apk_file_meta_from_fd(int fd, struct apk_file_meta *meta)
{ {
@ -96,6 +96,157 @@ ssize_t apk_istream_read(struct apk_istream *is, void *ptr, size_t size)
return size - left; return size - left;
} }
static int __apk_istream_fill(struct apk_istream *is)
{
ssize_t sz;
if (is->err) return is->err;
if (is->ptr != is->buf) {
sz = is->end - is->ptr;
memmove(is->buf, is->ptr, sz);
is->ptr = is->buf;
is->end = is->buf + sz;
}
sz = is->ops->read(is, is->end, is->buf + is->buf_size - is->end);
if (sz <= 0) {
is->err = sz ?: 1;
return is->err;
}
is->end += sz;
return 0;
}
apk_blob_t apk_istream_get(struct apk_istream *is, size_t len)
{
apk_blob_t ret = APK_BLOB_NULL;
do {
if (is->end - is->ptr >= len) {
ret = APK_BLOB_PTR_LEN((char*)is->ptr, len);
break;
}
if (is->err>0 || is->end-is->ptr == is->buf_size) {
ret = APK_BLOB_PTR_LEN((char*)is->ptr, is->end - is->ptr);
break;
}
} while (!__apk_istream_fill(is));
if (!APK_BLOB_IS_NULL(ret)) {
is->ptr = (uint8_t*)ret.ptr + ret.len;
return ret;
}
return (struct apk_blob) { .len = is->err < 0 ? is->err : 0 };
}
apk_blob_t apk_istream_get_all(struct apk_istream *is)
{
if (is->ptr == is->end)
__apk_istream_fill(is);
if (is->ptr != is->end) {
apk_blob_t ret = APK_BLOB_PTR_LEN((char*)is->ptr, is->end - is->ptr);
is->ptr = is->end = 0;
return ret;
}
return (struct apk_blob) { .len = is->err < 0 ? is->err : 0 };
}
apk_blob_t apk_istream_get_delim(struct apk_istream *is, apk_blob_t token)
{
apk_blob_t ret = APK_BLOB_NULL, left = APK_BLOB_NULL;
do {
if (apk_blob_split(APK_BLOB_PTR_LEN((char*)is->ptr, is->end - is->ptr), token, &ret, &left))
break;
if (is->end - is->ptr == is->buf_size) {
is->err = -ENOBUFS;
break;
}
} while (!__apk_istream_fill(is));
/* Last segment before end-of-file. Return also zero length non-null
* blob if eof comes immediately after the delimiter. */
if (is->ptr && is->err > 0)
ret = APK_BLOB_PTR_LEN((char*)is->ptr, is->end - is->ptr);
if (!APK_BLOB_IS_NULL(ret)) {
is->ptr = (uint8_t*)left.ptr;
is->end = (uint8_t*)left.ptr + left.len;
return ret;
}
return (struct apk_blob) { .len = is->err < 0 ? is->err : 0 };
}
static void segment_get_meta(struct apk_istream *is, struct apk_file_meta *meta)
{
struct apk_segment_istream *sis = container_of(is, struct apk_segment_istream, is);
*meta = (struct apk_file_meta) {
.atime = sis->mtime,
.mtime = sis->mtime,
};
}
static ssize_t segment_read(struct apk_istream *is, void *ptr, size_t size)
{
struct apk_segment_istream *sis = container_of(is, struct apk_segment_istream, is);
ssize_t r;
if (size > sis->bytes_left) size = sis->bytes_left;
if (size == 0) return 0;
r = sis->pis->ops->read(sis->pis, ptr, size);
if (r <= 0) {
/* If inner stream returned zero (end-of-stream), we
* are getting short read, because tar header indicated
* more was to be expected. */
if (r == 0) r = -ECONNABORTED;
} else {
sis->bytes_left -= r;
}
return r;
}
static void segment_close(struct apk_istream *is)
{
struct apk_segment_istream *sis = container_of(is, struct apk_segment_istream, is);
if (sis->bytes_left) {
apk_istream_read(sis->pis, NULL, sis->bytes_left);
sis->bytes_left = 0;
}
}
static const struct apk_istream_ops segment_istream_ops = {
.get_meta = segment_get_meta,
.read = segment_read,
.close = segment_close,
};
void apk_istream_segment(struct apk_segment_istream *sis, struct apk_istream *is, size_t len, time_t mtime)
{
*sis = (struct apk_segment_istream) {
.is.ops = &segment_istream_ops,
.is.buf = is->buf,
.is.buf_size = is->buf_size,
.is.ptr = is->ptr,
.is.end = is->end,
.pis = is,
.bytes_left = len,
.mtime = mtime,
};
if (sis->is.end - sis->is.ptr > len) {
sis->is.end = sis->is.ptr + len;
is->ptr += len;
} else {
is->ptr = is->end = 0;
}
sis->bytes_left -= sis->is.end - sis->is.ptr;
}
struct apk_fd_istream { struct apk_fd_istream {
struct apk_istream is; struct apk_istream is;
int fd; int fd;
@ -630,7 +781,6 @@ int apk_fileinfo_get(int atfd, const char *filename, unsigned int flags,
struct apk_file_info *fi) struct apk_file_info *fi)
{ {
struct stat64 st; struct stat64 st;
struct apk_bstream *bs;
unsigned int checksum = flags & 0xff; unsigned int checksum = flags & 0xff;
unsigned int xattr_checksum = (flags >> 8) & 0xff; unsigned int xattr_checksum = (flags >> 8) & 0xff;
int atflags = 0; int atflags = 0;
@ -701,23 +851,23 @@ int apk_fileinfo_get(int atfd, const char *filename, unsigned int flags,
apk_checksum_evp(checksum), NULL); apk_checksum_evp(checksum), NULL);
fi->csum.type = checksum; fi->csum.type = checksum;
} else { } else {
bs = apk_bstream_from_file(atfd, filename); struct apk_istream *is = apk_istream_from_file(atfd, filename);
if (!IS_ERR_OR_NULL(bs)) { if (!IS_ERR_OR_NULL(is)) {
EVP_MD_CTX *mdctx; EVP_MD_CTX *mdctx;
apk_blob_t blob; apk_blob_t blob;
mdctx = EVP_MD_CTX_new(); mdctx = EVP_MD_CTX_new();
if (mdctx) { if (mdctx) {
EVP_DigestInit_ex(mdctx, apk_checksum_evp(checksum), NULL); EVP_DigestInit_ex(mdctx, apk_checksum_evp(checksum), NULL);
if (bs->flags & APK_BSTREAM_SINGLE_READ) if (is->flags & APK_ISTREAM_SINGLE_READ)
EVP_MD_CTX_set_flags(mdctx, EVP_MD_CTX_FLAG_ONESHOT); EVP_MD_CTX_set_flags(mdctx, EVP_MD_CTX_FLAG_ONESHOT);
while (!APK_BLOB_IS_NULL(blob = apk_bstream_read(bs, APK_BLOB_NULL))) while (!APK_BLOB_IS_NULL(blob = apk_istream_get_all(is)))
EVP_DigestUpdate(mdctx, (void*) blob.ptr, blob.len); EVP_DigestUpdate(mdctx, (void*) blob.ptr, blob.len);
fi->csum.type = EVP_MD_CTX_size(mdctx); fi->csum.type = EVP_MD_CTX_size(mdctx);
EVP_DigestFinal_ex(mdctx, fi->csum.data, NULL); EVP_DigestFinal_ex(mdctx, fi->csum.data, NULL);
EVP_MD_CTX_free(mdctx); EVP_MD_CTX_free(mdctx);
} }
apk_bstream_close(bs); apk_istream_close(is);
} }
} }

View File

@ -637,11 +637,9 @@ int apk_sign_ctx_verify_tar(void *sctx, const struct apk_file_info *fi,
return r; return r;
if (strcmp(fi->name, ".PKGINFO") == 0) { if (strcmp(fi->name, ".PKGINFO") == 0) {
apk_blob_t blob = apk_blob_from_istream(is, fi->size); apk_blob_t l, token = APK_BLOB_STR("\n");
apk_blob_for_each_segment( while (!APK_BLOB_IS_NULL(l = apk_istream_get_delim(is, token)))
blob, "\n", apk_sign_ctx_parse_pkginfo_line(ctx, l);
apk_sign_ctx_parse_pkginfo_line, ctx);
free(blob.ptr);
} }
return 0; return 0;
@ -883,15 +881,14 @@ static int read_info_entry(void *ctx, const struct apk_file_info *ae,
/* Meta info and scripts */ /* Meta info and scripts */
r = apk_sign_ctx_process_file(ri->sctx, ae, is); r = apk_sign_ctx_process_file(ri->sctx, ae, is);
if (r <= 0) if (r <= 0) return r;
return r;
if (ae->name[0] == '.') { if (ae->name[0] == '.') {
/* APK 2.0 format */ /* APK 2.0 format */
if (strcmp(ae->name, ".PKGINFO") == 0) { if (strcmp(ae->name, ".PKGINFO") == 0) {
apk_blob_t blob = apk_blob_from_istream(is, ae->size); apk_blob_t l, token = APK_BLOB_STR("\n");
apk_blob_for_each_segment(blob, "\n", read_info_line, ctx); while (!APK_BLOB_IS_NULL(l = apk_istream_get_delim(is, token)))
free(blob.ptr); read_info_line(ctx, l);
} else if (strcmp(ae->name, ".INSTALL") == 0) { } else if (strcmp(ae->name, ".INSTALL") == 0) {
apk_warning("Package '%s-%s' contains deprecated .INSTALL", apk_warning("Package '%s-%s' contains deprecated .INSTALL",
pkg->name->name, pkg->version); pkg->name->name, pkg->version);