/* apk_io.h - Alpine Package Keeper (APK) * * Copyright (C) 2008-2011 Timo Teräs * All rights reserved. * * SPDX-License-Identifier: GPL-2.0-only */ #ifndef APK_IO #define APK_IO #include #include #include #include "apk_defines.h" #include "apk_blob.h" #include "apk_atom.h" #include "apk_crypto.h" struct apk_id_hash { int empty; struct hlist_head by_id[16], by_name[16]; }; struct apk_id_cache { int root_fd; struct apk_id_hash uid_cache; struct apk_id_hash gid_cache; }; struct apk_xattr { const char *name; apk_blob_t value; }; APK_ARRAY(apk_xattr_array, struct apk_xattr); struct apk_file_meta { time_t mtime, atime; }; void apk_file_meta_to_fd(int fd, struct apk_file_meta *meta); struct apk_file_info { char *name; char *link_target; char *uname; char *gname; off_t size; uid_t uid; gid_t gid; mode_t mode; time_t mtime; dev_t device; struct apk_digest digest; struct apk_digest xattr_digest; struct apk_xattr_array *xattrs; }; extern size_t apk_io_bufsize; struct apk_istream; struct apk_ostream; struct apk_istream_ops { void (*get_meta)(struct apk_istream *is, struct apk_file_meta *meta); ssize_t (*read)(struct apk_istream *is, void *ptr, size_t size); int (*close)(struct apk_istream *is); }; #define APK_ISTREAM_SINGLE_READ 0x0001 struct apk_istream { uint8_t *ptr, *end, *buf; size_t buf_size; int err; unsigned int flags; const struct apk_istream_ops *ops; }; #define APK_IO_ALL ((size_t)-1) #define APK_ISTREAM_FORCE_REFRESH ((time_t) -1) 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_fd(int fd); struct apk_istream *apk_istream_from_fd_url_if_modified(int atfd, const char *url, time_t since); static inline int apk_istream_error(struct apk_istream *is, int err) { if (!is->err) is->err = err; return err; } 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_max(struct apk_istream *is, size_t size); 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); } ssize_t apk_istream_splice(struct apk_istream *is, int fd, size_t size, 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, struct apk_digest_ctx *dctx); static inline struct apk_istream *apk_istream_from_url(const char *url, time_t since) { return apk_istream_from_fd_url_if_modified(AT_FDCWD, url, since); } static inline struct apk_istream *apk_istream_from_fd_url(int atfd, const char *url, time_t since) { return apk_istream_from_fd_url_if_modified(atfd, url, since); } static inline void apk_istream_get_meta(struct apk_istream *is, struct apk_file_meta *meta) { is->ops->get_meta(is, meta); } static inline int apk_istream_close(struct apk_istream *is) { return is->ops->close(is); } #define APK_MPART_DATA 1 /* data processed so far */ #define APK_MPART_BOUNDARY 2 /* final part of data, before boundary */ #define APK_MPART_END 3 /* signals end of stream */ typedef int (*apk_multipart_cb)(void *ctx, int part, apk_blob_t data); struct apk_istream *apk_istream_gunzip_mpart(struct apk_istream *, apk_multipart_cb cb, void *ctx); static inline struct apk_istream *apk_istream_gunzip(struct apk_istream *is) { return apk_istream_gunzip_mpart(is, NULL, NULL); } struct apk_segment_istream { struct apk_istream is; struct apk_istream *pis; size_t bytes_left; time_t mtime; }; struct apk_istream *apk_istream_segment(struct apk_segment_istream *sis, struct apk_istream *is, size_t len, time_t mtime); struct apk_istream *apk_istream_tee(struct apk_istream *from, int atfd, const char *to, int copy_meta, apk_progress_cb cb, void *cb_ctx); struct apk_ostream_ops { ssize_t (*write)(struct apk_ostream *os, const void *buf, size_t size); int (*close)(struct apk_ostream *os); }; struct apk_ostream { const struct apk_ostream_ops *ops; int rc; }; struct apk_ostream *apk_ostream_gzip(struct apk_ostream *); struct apk_ostream *apk_ostream_counter(off_t *); struct apk_ostream *apk_ostream_to_fd(int fd); struct apk_ostream *apk_ostream_to_file(int atfd, const char *file, mode_t mode); struct apk_ostream *apk_ostream_to_file_gz(int atfd, const char *file, const char *tmpfile, mode_t mode); size_t apk_ostream_write_string(struct apk_ostream *ostream, const char *string); static inline int apk_ostream_error(struct apk_ostream *os) { return os->rc; } static inline int apk_ostream_cancel(struct apk_ostream *os, int rc) { if (!os->rc) os->rc = rc; return rc; } static inline ssize_t apk_ostream_write(struct apk_ostream *os, const void *buf, size_t size) { return os->ops->write(os, buf, size); } static inline int apk_ostream_close(struct apk_ostream *os) { int rc = os->rc; return os->ops->close(os) ?: rc; } apk_blob_t apk_blob_from_istream(struct apk_istream *istream, size_t size); apk_blob_t apk_blob_from_file(int atfd, const char *file); #define APK_BTF_ADD_EOL 0x00000001 int apk_blob_to_file(int atfd, const char *file, apk_blob_t b, unsigned int flags); #define APK_FI_NOFOLLOW 0x80000000 #define APK_FI_XATTR_DIGEST(x) (((x) & 0xff) << 8) #define APK_FI_XATTR_CSUM(x) APK_FI_XATTR_DIGEST(apk_digest_alg_from_csum(x)) #define APK_FI_DIGEST(x) (((x) & 0xff)) #define APK_FI_CSUM(x) APK_FI_DIGEST(apk_digest_alg_from_csum(x)) int apk_fileinfo_get(int atfd, const char *filename, unsigned int flags, struct apk_file_info *fi, struct apk_atom_pool *atoms); void apk_fileinfo_hash_xattr(struct apk_file_info *fi, uint8_t alg); void apk_fileinfo_free(struct apk_file_info *fi); typedef int apk_dir_file_cb(void *ctx, int dirfd, const char *entry); int apk_dir_foreach_file(int dirfd, apk_dir_file_cb cb, void *ctx); const char *apk_url_local_file(const char *url); void apk_id_cache_init(struct apk_id_cache *idc, int root_fd); void apk_id_cache_free(struct apk_id_cache *idc); void apk_id_cache_reset(struct apk_id_cache *idc); uid_t apk_id_cache_resolve_uid(struct apk_id_cache *idc, apk_blob_t username, uid_t default_uid); gid_t apk_id_cache_resolve_gid(struct apk_id_cache *idc, apk_blob_t groupname, gid_t default_gid); apk_blob_t apk_id_cache_resolve_user(struct apk_id_cache *idc, uid_t uid); apk_blob_t apk_id_cache_resolve_group(struct apk_id_cache *idc, gid_t gid); #endif