/* io.c - Alpine Package Keeper (APK) * * Copyright (C) 2005-2008 Natanael Copa * Copyright (C) 2008-2011 Timo Teräs * All rights reserved. * * This program is free software; you can redistribute it and/or modify it * 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 #include #include #include #include #include #include #include #include #include #include #include "apk_defines.h" #include "apk_io.h" #include "apk_hash.h" struct apk_fd_istream { struct apk_istream is; int fd; pid_t pid; int (*translate_status)(int status); }; static ssize_t fdi_read(void *stream, void *ptr, size_t size) { struct apk_fd_istream *fis = container_of(stream, struct apk_fd_istream, is); ssize_t i = 0, r; if (ptr == NULL) { if (lseek(fis->fd, size, SEEK_CUR) < 0) return -errno; return size; } while (i < size) { r = read(fis->fd, ptr + i, size - i); if (r < 0) return -errno; if (r == 0) break; i += r; } if (i == 0 && fis->pid != 0) { int status; if (waitpid(fis->pid, &status, 0) == fis->pid) i = fis->translate_status(status); fis->pid = 0; } return i; } static void fdi_close(void *stream) { struct apk_fd_istream *fis = container_of(stream, struct apk_fd_istream, is); int status; close(fis->fd); if (fis->pid != 0) waitpid(fis->pid, &status, 0); free(fis); } struct apk_istream *apk_istream_from_fd_pid(int fd, pid_t pid, int (*translate_status)(int)) { struct apk_fd_istream *fis; if (fd < 0) return NULL; fis = malloc(sizeof(struct apk_fd_istream)); if (fis == NULL) { close(fd); return NULL; } *fis = (struct apk_fd_istream) { .is.read = fdi_read, .is.close = fdi_close, .fd = fd, .pid = pid, .translate_status = translate_status, }; return &fis->is; } struct apk_istream *apk_istream_from_file(int atfd, const char *file) { int fd; fd = openat(atfd, file, O_RDONLY | O_CLOEXEC); if (fd < 0) return NULL; 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; } size_t apk_istream_splice(void *stream, int fd, size_t size, apk_progress_cb cb, void *cb_ctx) { static void *splice_buffer = NULL; struct apk_istream *is = (struct apk_istream *) stream; unsigned char *buf, *mmapbase = MAP_FAILED; size_t bufsz, done = 0, r, togo; bufsz = size; if (size > 128 * 1024) { if (size != APK_SPLICE_ALL && ftruncate(fd, size) == 0) mmapbase = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); if (bufsz > 2*1024*1024) bufsz = 2*1024*1024; buf = mmapbase; } if (mmapbase == MAP_FAILED) { if (splice_buffer == NULL) splice_buffer = malloc(256*1024); buf = splice_buffer; if (buf == NULL) return -ENOMEM; if (bufsz > 256*1024) bufsz = 256*1024; } while (done < size) { if (done != 0 && cb != NULL) cb(cb_ctx, muldiv(APK_PROGRESS_SCALE, done, size)); togo = size - done; if (togo > bufsz) togo = bufsz; r = is->read(is, buf, togo); if (r < 0) goto err; if (r == 0) break; if (mmapbase == MAP_FAILED) { if (write(fd, buf, r) != r) { if (r < 0) r = -errno; goto err; } } else buf += r; done += r; if (r != togo) break; } r = done; err: if (mmapbase != MAP_FAILED) munmap(mmapbase, size); return r; } struct apk_istream_bstream { struct apk_bstream bs; struct apk_istream *is; apk_blob_t left; char buffer[8*1024]; size_t size; }; static apk_blob_t is_bs_read(void *stream, apk_blob_t token) { struct apk_istream_bstream *isbs = container_of(stream, struct apk_istream_bstream, bs); 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; } /* If we've exchausted earlier, it's end of stream or error */ if (APK_BLOB_IS_NULL(isbs->left)) return isbs->left; /* 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; } else { /* cache and return error */ isbs->left = ret = APK_BLOB_ERROR(size); goto ret; } 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; } static void is_bs_close(void *stream, size_t *size) { struct apk_istream_bstream *isbs = container_of(stream, struct apk_istream_bstream, bs); if (size != NULL) *size = isbs->size; isbs->is->close(isbs->is); free(isbs); } 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; isbs->left = APK_BLOB_PTR_LEN(isbs->buffer, 0), isbs->size = 0; return &isbs->bs; } struct apk_mmap_bstream { struct apk_bstream bs; int fd; size_t size; unsigned char *ptr; apk_blob_t left; }; static apk_blob_t mmap_read(void *stream, apk_blob_t token) { struct apk_mmap_bstream *mbs = container_of(stream, struct apk_mmap_bstream, bs); apk_blob_t ret; if (!APK_BLOB_IS_NULL(token) && !APK_BLOB_IS_NULL(mbs->left)) { if (apk_blob_split(mbs->left, token, &ret, &mbs->left)) return ret; } ret = mbs->left; mbs->left = APK_BLOB_NULL; mbs->bs.flags |= APK_BSTREAM_EOF; return ret; } static void mmap_close(void *stream, size_t *size) { struct apk_mmap_bstream *mbs = container_of(stream, struct apk_mmap_bstream, bs); if (size != NULL) *size = mbs->size; munmap(mbs->ptr, mbs->size); close(mbs->fd); 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; ptr = mmap(NULL, st.st_size, PROT_READ, MAP_SHARED, fd, 0); 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) { .flags = APK_BSTREAM_SINGLE_READ, .read = mmap_read, .close = mmap_close, }; mbs->fd = fd; mbs->size = st.st_size; mbs->ptr = ptr; mbs->left = APK_BLOB_PTR_LEN(ptr, mbs->size); return &mbs->bs; } struct apk_bstream *apk_bstream_from_fd_pid(int fd, pid_t pid, int (*translate_status)(int)) { struct apk_bstream *bs; if (fd < 0) return NULL; if (pid == 0) { bs = apk_mmap_bstream_from_fd(fd); if (bs != NULL) return bs; } return apk_bstream_from_istream(apk_istream_from_fd_pid(fd, pid, translate_status)); } struct apk_bstream *apk_bstream_from_file(int atfd, const char *file) { int fd; fd = openat(atfd, file, O_RDONLY | O_CLOEXEC); if (fd < 0) return NULL; return apk_bstream_from_fd(fd); } struct apk_tee_bstream { struct apk_bstream bs; struct apk_bstream *inner_bs; int fd; size_t size; }; static apk_blob_t tee_read(void *stream, apk_blob_t token) { struct apk_tee_bstream *tbs = container_of(stream, struct apk_tee_bstream, bs); apk_blob_t blob; blob = tbs->inner_bs->read(tbs->inner_bs, token); if (!APK_BLOB_IS_NULL(blob)) tbs->size += write(tbs->fd, blob.ptr, blob.len); return blob; } static void tee_close(void *stream, size_t *size) { struct apk_tee_bstream *tbs = container_of(stream, struct apk_tee_bstream, bs); tbs->inner_bs->close(tbs->inner_bs, NULL); if (size != NULL) *size = tbs->size; close(tbs->fd); free(tbs); } struct apk_bstream *apk_bstream_tee(struct apk_bstream *from, int atfd, const char *to) { struct apk_tee_bstream *tbs; int fd; fd = openat(atfd, to, O_CREAT | O_RDWR | O_TRUNC | O_CLOEXEC, 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; tbs->size = 0; return &tbs->bs; } 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); } apk_blob_t apk_blob_from_file(int atfd, const char *file) { int fd; struct stat st; char *buf; fd = openat(atfd, file, O_RDONLY | O_CLOEXEC); 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; } int apk_blob_to_file(int atfd, const char *file, apk_blob_t b, unsigned int flags) { int fd, r, len; fd = openat(atfd, file, O_CREAT | O_WRONLY | O_CLOEXEC, 0644); if (fd < 0) return -errno; len = b.len; r = write(fd, b.ptr, len); if ((r == len) && (flags & APK_BTF_ADD_EOL) && (b.len == 0 || b.ptr[b.len-1] != '\n')) { len = 1; r = write(fd, "\n", len); } if (r < 0) r = -errno; else if (r != len) r = -ENOSPC; else r = 0; close(fd); if (r != 0) unlinkat(atfd, file, 0); return r; } int apk_file_get_info(int atfd, const char *filename, unsigned int flags, struct apk_file_info *fi) { struct stat64 st; struct apk_bstream *bs; int checksum = flags & 0xffff, atflags = 0; if (flags & APK_FI_NOFOLLOW) atflags |= AT_SYMLINK_NOFOLLOW; if (fstatat64(atfd, filename, &st, atflags) != 0) return -errno; *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, }; if (checksum == APK_CHECKSUM_NONE || S_ISDIR(st.st_mode)) return 0; if ((flags & APK_FI_NOFOLLOW) && S_ISLNK(st.st_mode)) { char *target = alloca(st.st_size); if (target == NULL) return -ENOMEM; if (readlinkat(atfd, filename, target, st.st_size) < 0) return -errno; EVP_Digest(target, st.st_size, fi->csum.data, NULL, apk_checksum_evp(checksum), NULL); fi->csum.type = checksum; return 0; } bs = apk_bstream_from_file(atfd, filename); if (bs != NULL) { EVP_MD_CTX mdctx; apk_blob_t blob; EVP_DigestInit(&mdctx, apk_checksum_evp(checksum)); if (bs->flags & APK_BSTREAM_SINGLE_READ) EVP_MD_CTX_set_flags(&mdctx, EVP_MD_CTX_FLAG_ONESHOT); while (!APK_BLOB_IS_NULL(blob = bs->read(bs, APK_BLOB_NULL))) EVP_DigestUpdate(&mdctx, (void*) blob.ptr, blob.len); fi->csum.type = EVP_MD_CTX_size(&mdctx); EVP_DigestFinal(&mdctx, fi->csum.data, NULL); bs->close(bs, NULL); } return 0; } int apk_dir_foreach_file(int dirfd, apk_dir_file_cb cb, void *ctx) { struct dirent *de; DIR *dir; if (dirfd < 0) return -1; dir = fdopendir(dirfd); if (dir == NULL) return -1; /* We get called here with dup():ed fd. Since they all refer to * same object, we need to rewind so subsequent calls work. */ rewinddir(dir); while ((de = readdir(dir)) != NULL) { if (de->d_name[0] == '.') { if (de->d_name[1] == 0 || (de->d_name[1] == '.' && de->d_name[2] == 0)) continue; } cb(ctx, dirfd, de->d_name); } closedir(dir); return 0; } struct apk_istream *apk_istream_from_file_gz(int atfd, const char *file) { return apk_bstream_gunzip(apk_bstream_from_file(atfd, file)); } struct apk_fd_ostream { struct apk_ostream os; int fd, rc; const char *file, *tmpfile; int atfd; size_t bytes; char buffer[1024]; }; static ssize_t safe_write(int fd, const void *ptr, size_t size) { ssize_t i = 0, r; while (i < size) { r = write(fd, ptr + i, size - i); if (r < 0) return -errno; if (r == 0) return i; i += r; } return i; } static ssize_t fdo_flush(struct apk_fd_ostream *fos) { ssize_t r; if (fos->bytes == 0) return 0; if ((r = safe_write(fos->fd, fos->buffer, fos->bytes)) != fos->bytes) { fos->rc = r < 0 ? r : -EIO; return r; } fos->bytes = 0; return 0; } static ssize_t fdo_write(void *stream, const void *ptr, size_t size) { struct apk_fd_ostream *fos = container_of(stream, struct apk_fd_ostream, os); ssize_t r; if (size + fos->bytes >= sizeof(fos->buffer)) { r = fdo_flush(fos); if (r != 0) return r; if (size >= sizeof(fos->buffer) / 2) { r = safe_write(fos->fd, ptr, size); if (r != size) fos->rc = r < 0 ? r : -EIO; return r; } } memcpy(&fos->buffer[fos->bytes], ptr, size); fos->bytes += size; return size; } static int fdo_close(void *stream) { struct apk_fd_ostream *fos = container_of(stream, struct apk_fd_ostream, os); int rc = fos->rc; fdo_flush(fos); if (fos->fd > STDERR_FILENO && close(fos->fd) < 0) rc = -errno; if (fos->tmpfile != NULL) { if (rc == 0) renameat(fos->atfd, fos->tmpfile, fos->atfd, fos->file); else unlinkat(fos->atfd, fos->tmpfile, 0); } free(fos); return rc; } 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) { close(fd); 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(int atfd, const char *file, const char *tmpfile, mode_t mode) { struct apk_ostream *os; int fd; fd = openat(atfd, tmpfile ?: file, O_CREAT | O_RDWR | O_TRUNC | O_CLOEXEC, mode); if (fd < 0) return NULL; fcntl(fd, F_SETFD, FD_CLOEXEC); os = apk_ostream_to_fd(fd); if (os == NULL) return NULL; if (tmpfile != NULL) { struct apk_fd_ostream *fos = container_of(os, struct apk_fd_ostream, os); fos->file = file; fos->tmpfile = tmpfile; fos->atfd = atfd; } return os; } struct apk_counter_ostream { struct apk_ostream os; off_t *counter; }; static ssize_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 int co_close(void *stream) { struct apk_counter_ostream *cos = container_of(stream, struct apk_counter_ostream, os); free(cos); return 0; } 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; } 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; } int apk_move_file(int atfd, const char *from, const char *to) { struct apk_istream *is; struct stat64 st; int rc, tofd; if (renameat(atfd, from, atfd, to) == 0) return 0; if (fstatat64(atfd, from, &st, 0) != 0) return -errno; is = apk_istream_from_file(atfd, from); if (is == NULL) return -ENOENT; tofd = openat(atfd, to, O_CREAT | O_RDWR | O_TRUNC | O_CLOEXEC, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH); if (tofd < 0) { rc = -errno; goto close_is; } rc = apk_istream_splice(is, tofd, st.st_size, NULL, NULL); close(tofd); unlinkat(atfd, from, 0); close_is: is->close(is); return rc; } struct cache_item { apk_hash_node hash_node; unsigned int genid; union { uid_t uid; gid_t gid; }; unsigned short len; char name[]; }; static apk_blob_t cache_item_get_key(apk_hash_item item) { struct cache_item *ci = (struct cache_item *) item; return APK_BLOB_PTR_LEN(ci->name, ci->len); } static const struct apk_hash_ops id_hash_ops = { .node_offset = offsetof(struct cache_item, hash_node), .get_key = cache_item_get_key, .hash_key = apk_blob_hash, .compare = apk_blob_compare, .delete_item = (apk_hash_delete_f) free, }; static struct cache_item *resolve_cache_item(struct apk_hash *hash, apk_blob_t name) { struct cache_item *ci; unsigned long h; h = id_hash_ops.hash_key(name); ci = (struct cache_item *) apk_hash_get_hashed(hash, name, h); if (ci != NULL) return ci; ci = calloc(1, sizeof(struct cache_item) + name.len); if (ci == NULL) return NULL; ci->len = name.len; memcpy(ci->name, name.ptr, name.len); apk_hash_insert_hashed(hash, ci, h); return ci; } void apk_id_cache_init(struct apk_id_cache *idc, int root_fd) { idc->root_fd = root_fd; idc->genid = 1; apk_hash_init(&idc->uid_cache, &id_hash_ops, 256); apk_hash_init(&idc->gid_cache, &id_hash_ops, 256); } void apk_id_cache_free(struct apk_id_cache *idc) { apk_hash_free(&idc->uid_cache); apk_hash_free(&idc->gid_cache); } void apk_id_cache_reset(struct apk_id_cache *idc) { idc->genid++; if (idc->genid == 0) idc->genid = 1; } uid_t apk_resolve_uid(struct apk_id_cache *idc, const char *username, uid_t default_uid) { struct cache_item *ci; struct passwd pwent, *pwd; char buf[1024]; FILE *in; ci = resolve_cache_item(&idc->uid_cache, APK_BLOB_STR(username)); if (ci == NULL) return default_uid; if (ci->genid != idc->genid) { ci->genid = idc->genid; ci->uid = -1; in = fdopen(openat(idc->root_fd, "etc/passwd", O_RDONLY|O_CLOEXEC), "r"); if (in != NULL) { do { fgetpwent_r(in, &pwent, buf, sizeof(buf), &pwd); if (pwd == NULL) break; if (strcmp(pwd->pw_name, username) == 0) { ci->uid = pwd->pw_uid; break; } } while (1); fclose(in); } } if (ci->uid != -1) return ci->uid; return default_uid; } uid_t apk_resolve_gid(struct apk_id_cache *idc, const char *groupname, uid_t default_gid) { struct cache_item *ci; struct group grent, *grp; char buf[1024]; FILE *in; ci = resolve_cache_item(&idc->gid_cache, APK_BLOB_STR(groupname)); if (ci == NULL) return default_gid; if (ci->genid != idc->genid) { ci->genid = idc->genid; ci->gid = -1; in = fdopen(openat(idc->root_fd, "etc/group", O_RDONLY|O_CLOEXEC), "r"); if (in != NULL) { do { fgetgrent_r(in, &grent, buf, sizeof(buf), &grp); if (grp == NULL) break; if (strcmp(grp->gr_name, groupname) == 0) { ci->gid = grp->gr_gid; break; } } while (1); fclose(in); } } if (ci->gid != -1) return ci->gid; return default_gid; }