/* * Copyright (c) 2017-2020, [Ribose Inc](https://www.ribose.com). * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef STREAM_COMMON_H_ #define STREAM_COMMON_H_ #include #include #include #include "types.h" #define PGP_INPUT_CACHE_SIZE 32768 #define PGP_OUTPUT_CACHE_SIZE 32768 #define PGP_PARTIAL_PKT_FIRST_PART_MIN_SIZE 512 typedef enum { PGP_STREAM_NULL, PGP_STREAM_FILE, PGP_STREAM_MEMORY, PGP_STREAM_STDIN, PGP_STREAM_STDOUT, PGP_STREAM_PACKET, PGP_STREAM_PARLEN_PACKET, PGP_STREAM_LITERAL, PGP_STREAM_COMPRESSED, PGP_STREAM_ENCRYPTED, PGP_STREAM_SIGNED, PGP_STREAM_ARMORED, PGP_STREAM_CLEARTEXT } pgp_stream_type_t; typedef struct pgp_source_t pgp_source_t; typedef struct pgp_dest_t pgp_dest_t; typedef bool pgp_source_read_func_t(pgp_source_t *src, void *buf, size_t len, size_t *read); typedef rnp_result_t pgp_source_finish_func_t(pgp_source_t *src); typedef void pgp_source_close_func_t(pgp_source_t *src); typedef rnp_result_t pgp_dest_write_func_t(pgp_dest_t *dst, const void *buf, size_t len); typedef rnp_result_t pgp_dest_finish_func_t(pgp_dest_t *src); typedef void pgp_dest_close_func_t(pgp_dest_t *dst, bool discard); /* statically preallocated cache for sources */ typedef struct pgp_source_cache_t { uint8_t buf[PGP_INPUT_CACHE_SIZE]; unsigned pos; /* current position in cache */ unsigned len; /* number of bytes available in cache */ bool readahead; /* whether read-ahead with larger chunks allowed */ } pgp_source_cache_t; typedef struct pgp_source_t { pgp_source_read_func_t * read; pgp_source_finish_func_t *finish; pgp_source_close_func_t * close; pgp_stream_type_t type; uint64_t size; /* size of the data if available, see knownsize */ uint64_t readb; /* number of bytes read from the stream via src_read. Do not confuse with number of bytes as returned via the read since data may be cached */ pgp_source_cache_t *cache; /* cache if used */ void * param; /* source-specific additional data */ unsigned eof : 1; /* end of data as reported by read and empty cache */ unsigned knownsize : 1; /* whether size of the data is known */ unsigned error : 1; /* there were reading error */ } pgp_source_t; /** @brief helper function to allocate memory for source's cache and param * Also fills src and param with zeroes * @param src pointer to the source structure * @param paramsize number of bytes required for src->param * @return true on success or false if memory allocation failed. **/ bool init_src_common(pgp_source_t *src, size_t paramsize); /** @brief read up to len bytes from the source * While this function tries to read as much bytes as possible however it may return * less then len bytes. Then src->eof can be checked if it's end of data. * * @param src source structure * @param buf preallocated buffer which can store up to len bytes * @param len number of bytes to read * @param read number of read bytes will be stored here. Cannot be NULL. * @return true on success or false otherwise **/ bool src_read(pgp_source_t *src, void *buf, size_t len, size_t *read); /** @brief shortcut to read exactly len bytes from source. See src_read for parameters. * @return true if len bytes were read or false otherwise (i.e. less then len were read or * read error occurred) */ bool src_read_eq(pgp_source_t *src, void *buf, size_t len); /** @brief read up to len bytes and keep them in the cache/do not process * Works only for streams with cache * @param src source structure * @param buf preallocated buffer which can store up to len bytes, or NULL if data should be * discarded, just making sure that needed input is available in source * @param len number of bytes to read. Must be less then PGP_INPUT_CACHE_SIZE. * @param read number of bytes read will be stored here. Cannot be NULL. * @return true on success or false otherwise **/ bool src_peek(pgp_source_t *src, void *buf, size_t len, size_t *read); /** @brief shortcut to read exactly len bytes and keep them in the cache/do not process * Works only for streams with cache * @return true if len bytes were read or false otherwise (i.e. less then len were read or * read error occurred) */ bool src_peek_eq(pgp_source_t *src, void *buf, size_t len); /** @brief skip up to len bytes. * Note: use src_read() if you want to check error condition/get number of bytes *skipped. * @param src source structure * @param len number of bytes to skip **/ void src_skip(pgp_source_t *src, size_t len); /** @brief notify source that all reading is done, so final data processing may be started, * i.e. signature reading and verification and so on. Do not misuse with src_close. * @param src allocated and initialized source structure * @return RNP_SUCCESS or error code. If source doesn't have finish handler then also * RNP_SUCCESS is returned */ rnp_result_t src_finish(pgp_source_t *src); /** @brief check whether there were reading error on source * @param allocated and initialized source structure * @return true if there were reading error or false otherwise */ bool src_error(const pgp_source_t *src); /** @brief check whether there is no more input on source * @param src allocated and initialized source structure * @return true if there is no more input or false otherwise. * On read error false will be returned. */ bool src_eof(pgp_source_t *src); /** @brief close the source and deallocate all internal resources if any */ void src_close(pgp_source_t *src); /** @brief skip end of line on the source (\r\n or \n, depending on input) * @param src allocated and initialized source * @return true if eol was found and skipped or false otherwise */ bool src_skip_eol(pgp_source_t *src); /** @brief peek the line on the source * @param src allocated and initialized source with data * @param buf preallocated buffer to store the result. Result include NULL character and * doesn't include the end of line sequence. * @param len maximum length of data to store in buf, including terminating NULL * @param read on success here will be stored number of bytes in the string, without the NULL * character. * @return true on success * false is returned if there were eof, read error or eol was not found within the * len. Supported eol sequences are \r\n and \n */ bool src_peek_line(pgp_source_t *src, char *buf, size_t len, size_t *read); /** @brief init file source * @param src pre-allocated source structure * @param path path to the file * @return RNP_SUCCESS or error code **/ rnp_result_t init_file_src(pgp_source_t *src, const char *path); /** @brief init stdin source * @param src pre-allocated source structure * @return RNP_SUCCESS or error code **/ rnp_result_t init_stdin_src(pgp_source_t *src); /** @brief init memory source * @param src pre-allocated source structure * @param mem memory to read from * @param len number of bytes in input * @param free free the memory pointer on stream close or not * @return RNP_SUCCESS or error code **/ rnp_result_t init_mem_src(pgp_source_t *src, const void *mem, size_t len, bool free); /** @brief init NULL source, which doesn't allow to read anything and always returns an error. * @param src pre-allocated source structure * @return always RNP_SUCCESS **/ rnp_result_t init_null_src(pgp_source_t *src); /** @brief init memory source with contents of other source * @param src pre-allocated source structure * @param readsrc opened source with data * @return RNP_SUCCESS or error code **/ rnp_result_t read_mem_src(pgp_source_t *src, pgp_source_t *readsrc); /** @brief init memory source with contents of the specified file * @param src pre-allocated source structure * @param filename name of the file * @return RNP_SUCCESS or error code **/ rnp_result_t file_to_mem_src(pgp_source_t *src, const char *filename); /** @brief get memory from the memory source * @param src initialized memory source * @param own transfer ownership of the memory * @return pointer to the memory or NULL if it is not a memory source **/ const void *mem_src_get_memory(pgp_source_t *src, bool own = false); typedef struct pgp_dest_t { pgp_dest_write_func_t * write; pgp_dest_finish_func_t *finish; pgp_dest_close_func_t * close; pgp_stream_type_t type; rnp_result_t werr; /* write function may set this to some error code */ size_t writeb; /* number of bytes written */ void * param; /* source-specific additional data */ bool no_cache; /* disable write caching */ uint8_t cache[PGP_OUTPUT_CACHE_SIZE]; unsigned clen; /* number of bytes in cache */ bool finished; /* whether dst_finish was called on dest or not */ } pgp_dest_t; /** @brief helper function to allocate memory for dest's param. * Initializes dst and param with zeroes as well. * @param dst dest structure * @param paramsize number of bytes required for dst->param * @return true on success, or false if memory allocation failed **/ bool init_dst_common(pgp_dest_t *dst, size_t paramsize); /** @brief write buffer to the destination * * @param dst destination structure * @param buf buffer with data * @param len number of bytes to write * @return true on success or false otherwise **/ void dst_write(pgp_dest_t *dst, const void *buf, size_t len); /** @brief printf formatted string to the destination * * @param dst destination structure * @param format format string, which is the same as printf() uses * @param ... additional arguments */ void dst_printf(pgp_dest_t *dst, const char *format, ...); /** @brief do all finalization tasks after all writing is done, i.e. calculate and write * mdc, signatures and so on. Do not misuse with dst_close. If was not called then will be * called from the dst_close * * @param dst destination structure * @return RNP_SUCCESS or error code if something went wrong **/ rnp_result_t dst_finish(pgp_dest_t *dst); /** @brief close the destination * * @param dst destination structure to be closed * @param discard if this is true then all produced output should be discarded * @return void **/ void dst_close(pgp_dest_t *dst, bool discard); /** @brief flush cached data if any. dst_write caches small writes, so data does not * immediately go to stream write function. * * @param dst destination structure * @return void **/ void dst_flush(pgp_dest_t *dst); /** @brief init file destination * @param dst pre-allocated dest structure * @param path path to the file * @param overwrite overwrite existing file * @return RNP_SUCCESS or error code **/ rnp_result_t init_file_dest(pgp_dest_t *dst, const char *path, bool overwrite); /** @brief init file destination, using the temporary file name, based on path. * Once writing is over, dst_finish() will attempt to rename to the desired name. * @param dst pre-allocated dest structure * @param path path to the file * @param overwrite overwrite existing file on rename * @return RNP_SUCCESS or error code **/ rnp_result_t init_tmpfile_dest(pgp_dest_t *dst, const char *path, bool overwrite); /** @brief init stdout destination * @param dst pre-allocated dest structure * @return RNP_SUCCESS or error code **/ rnp_result_t init_stdout_dest(pgp_dest_t *dst); /** @brief init memory destination * @param dst pre-allocated dest structure * @param mem pointer to the pre-allocated memory buffer, or NULL if it should be allocated * @param len number of bytes which mem can keep, or maximum amount of memory to allocate if * mem is NULL. If len is zero in later case then allocation is not limited. * @return RNP_SUCCESS or error code **/ rnp_result_t init_mem_dest(pgp_dest_t *dst, void *mem, unsigned len); /** @brief set whether to silently discard bytes which overflow memory of the dst. * @param dst pre-allocated and initialized memory dest * @param discard true to discard or false to return an error on overflow. **/ void mem_dest_discard_overflow(pgp_dest_t *dst, bool discard); /** @brief get the pointer to the memory where data is written. * Do not retain the result, it may change between calls due to realloc * @param dst pre-allocated and initialized memory dest * @return pointer to the memory area or NULL if memory was not allocated **/ void *mem_dest_get_memory(pgp_dest_t *dst); /** @brief get ownership on the memory dest's contents. This must be called only before * closing the dest * @param dst pre-allocated and initialized memory dest * @return pointer to the memory area or NULL if memory was not allocated (i.e. nothing was * written to the destination). Also NULL will be returned on possible (re-)allocation * failure, this case can be identified by non-zero dst->writeb. **/ void *mem_dest_own_memory(pgp_dest_t *dst); /** @brief mark memory dest as secure, so it will be deallocated securely * @param dst pre-allocated and initialized memory dest * @param secure whether memory should be considered as secure or not * @return void **/ void mem_dest_secure_memory(pgp_dest_t *dst, bool secure); /** @brief init null destination which silently discards all the output * @param dst pre-allocated dest structure * @return RNP_SUCCESS or error code **/ rnp_result_t init_null_dest(pgp_dest_t *dst); /** @brief reads from source and writes to destination * @param src initialized source * @param dst initialized destination * @param limit sets the maximum amount of bytes to be read, * returning an error if the source hasn't come to eof after that amount * if 0, no limit is imposed * @return RNP_SUCCESS or error code **/ rnp_result_t dst_write_src(pgp_source_t *src, pgp_dest_t *dst, uint64_t limit = 0); namespace rnp { /* Temporary wrapper to destruct stack-based pgp_source_t */ class Source { protected: pgp_source_t src_; public: Source(const Source &) = delete; Source(Source &&) = delete; Source() : src_({}) { } virtual ~Source() { src_close(&src_); } virtual pgp_source_t & src() { return src_; } size_t size() { return src().size; } size_t readb() { return src().readb; } bool eof() { return src_eof(&src()); } bool error() { return src_error(&src()); } }; class MemorySource : public Source { public: MemorySource(const MemorySource &) = delete; MemorySource(MemorySource &&) = delete; /** * @brief Construct memory source object. * * @param mem source memory. Must be valid for the whole lifetime of the object. * @param len size of the memory. * @param free free memory once processing is finished. */ MemorySource(const void *mem, size_t len, bool free) : Source() { auto res = init_mem_src(&src_, mem, len, free); if (res) { throw std::bad_alloc(); } } /** * @brief Construct memory source object * * @param vec vector with data. Must be valid for the whole lifetime of the object. */ MemorySource(const std::vector &vec) : MemorySource(vec.data(), vec.size(), false) { } MemorySource(pgp_source_t &src) : Source() { auto res = read_mem_src(&src_, &src); if (res) { throw rnp::rnp_exception(res); } } const void * memory(bool own = false) { return mem_src_get_memory(&src_, own); } }; /* Temporary wrapper to destruct stack-based pgp_dest_t */ class Dest { protected: pgp_dest_t dst_; bool discard_; public: Dest(const Dest &) = delete; Dest(Dest &&) = delete; Dest() : dst_({}), discard_(false) { } virtual ~Dest() { dst_close(&dst_, discard_); } void write(const void *buf, size_t len) { dst_write(&dst_, buf, len); } void set_discard(bool discard) { discard_ = discard; } pgp_dest_t & dst() { return dst_; } size_t writeb() { return dst_.writeb; } rnp_result_t werr() { return dst_.werr; } }; class MemoryDest : public Dest { public: MemoryDest(const MemoryDest &) = delete; MemoryDest(MemoryDest &&) = delete; MemoryDest(void *mem = NULL, size_t len = 0) : Dest() { auto res = init_mem_dest(&dst_, mem, len); if (res) { throw std::bad_alloc(); } discard_ = true; } void * memory() { return mem_dest_get_memory(&dst_); } void set_secure(bool secure) { mem_dest_secure_memory(&dst_, secure); } std::vector to_vector() { uint8_t *mem = (uint8_t *) memory(); return std::vector(mem, mem + writeb()); } }; } // namespace rnp #endif