// filters.h - originally written and placed in the public domain by Wei Dai /// \file filters.h /// \brief Implementation of BufferedTransformation's attachment interface. #ifndef CRYPTOPP_FILTERS_H #define CRYPTOPP_FILTERS_H #include "config.h" #if CRYPTOPP_MSC_VERSION # pragma warning(push) # pragma warning(disable: 4127 4189 4231 4275 4514) #endif #include "cryptlib.h" #include "simple.h" #include "secblock.h" #include "misc.h" #include "smartptr.h" #include "queue.h" #include "algparam.h" #include "stdcpp.h" NAMESPACE_BEGIN(CryptoPP) /// \brief Implementation of BufferedTransformation's attachment interface /// \details Filter is a cornerstone of the Pipeline trinity. Data flows from /// Sources, through Filters, and then terminates in Sinks. The difference /// between a Source and Filter is a Source \a pumps data, while a Filter does /// not. The difference between a Filter and a Sink is a Filter allows an /// attached transformation, while a Sink does not. /// \details See the discussion of BufferedTransformation in cryptlib.h for /// more details. class CRYPTOPP_DLL CRYPTOPP_NO_VTABLE Filter : public BufferedTransformation, public NotCopyable { public: virtual ~Filter() {} /// \name ATTACHMENT //@{ /// \brief Construct a Filter /// \param attachment an optional attached transformation /// \details attachment can be NULL. Filter(BufferedTransformation *attachment = NULLPTR); /// \brief Determine if attachable /// \return true if the object allows attached transformations, false otherwise. /// \note Source and Filter offer attached transformations; while Sink does not. bool Attachable() {return true;} /// \brief Retrieve attached transformation /// \return pointer to a BufferedTransformation if there is an attached transformation, NULL otherwise. BufferedTransformation *AttachedTransformation(); /// \brief Retrieve attached transformation /// \return pointer to a BufferedTransformation if there is an attached transformation, NULL otherwise. const BufferedTransformation *AttachedTransformation() const; /// \brief Replace an attached transformation /// \param newAttachment an optional attached transformation /// \details newAttachment can be a single filter, a chain of filters or NULL. /// Pass NULL to remove an existing BufferedTransformation or chain of filters void Detach(BufferedTransformation *newAttachment = NULLPTR); //@} /// \name RETRIEVAL OF ONE MESSAGE //@{ // BufferedTransformation in cryptlib.h size_t TransferTo2(BufferedTransformation &target, lword &transferBytes, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true); size_t CopyRangeTo2(BufferedTransformation &target, lword &begin, lword end=LWORD_MAX, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true) const; //@} /// \name SIGNALS //@{ // BufferedTransformation in cryptlib.h void Initialize(const NameValuePairs ¶meters=g_nullNameValuePairs, int propagation=-1); bool Flush(bool hardFlush, int propagation=-1, bool blocking=true); bool MessageSeriesEnd(int propagation=-1, bool blocking=true); //@} protected: virtual BufferedTransformation * NewDefaultAttachment() const; void Insert(Filter *nextFilter); // insert filter after this one virtual bool ShouldPropagateMessageEnd() const {return true;} virtual bool ShouldPropagateMessageSeriesEnd() const {return true;} void PropagateInitialize(const NameValuePairs ¶meters, int propagation); /// \brief Forward processed data on to attached transformation /// \param outputSite unknown, system crash between keyboard and chair... /// \param inString the byte buffer to process /// \param length the size of the string, in bytes /// \param messageEnd means how many filters to signal MessageEnd() to, including this one /// \param blocking specifies whether the object should block when processing input /// \param channel the channel to process the data /// \return the number of bytes that remain to be processed (i.e., bytes not processed). /// 0 indicates all bytes were processed. size_t Output(int outputSite, const byte *inString, size_t length, int messageEnd, bool blocking, const std::string &channel=DEFAULT_CHANNEL); /// \brief Output multiple bytes that may be modified by callee. /// \param outputSite unknown, system crash between keyboard and chair... /// \param inString the byte buffer to process /// \param length the size of the string, in bytes /// \param messageEnd means how many filters to signal MessageEnd() to, including this one /// \param blocking specifies whether the object should block when processing input /// \param channel the channel to process the data /// \return the number of bytes that remain to be processed (i.e., bytes not processed). /// 0 indicates all bytes were processed. size_t OutputModifiable(int outputSite, byte *inString, size_t length, int messageEnd, bool blocking, const std::string &channel=DEFAULT_CHANNEL); /// \brief Signals the end of messages to the object /// \param outputSite unknown, system crash between keyboard and chair... /// \param propagation the number of attached transformations the MessageEnd() signal should be passed /// \param blocking specifies whether the object should block when processing input /// \param channel the channel to process the data /// \return true is the MessageEnd signal was successful, false otherwise. /// \details propagation count includes this object. Setting propagation to 1 means this /// object only. Setting propagation to -1 means unlimited propagation. bool OutputMessageEnd(int outputSite, int propagation, bool blocking, const std::string &channel=DEFAULT_CHANNEL); /// \brief Flush buffered input and/or output, with signal propagation /// \param outputSite unknown, system crash between keyboard and chair... /// \param hardFlush is used to indicate whether all data should be flushed /// \param propagation the number of attached transformations the Flush() signal should be passed /// \param blocking specifies whether the object should block when processing input /// \param channel the channel to process the data /// \return true is the Flush signal was successful, false otherwise. /// \details propagation count includes this object. Setting propagation to 1 means this /// object only. Setting propagation to -1 means unlimited propagation. /// \note Hard flushes must be used with care. It means try to process and output everything, even if /// there may not be enough data to complete the action. For example, hard flushing a HexDecoder /// would cause an error if you do it after inputing an odd number of hex encoded characters. /// \note For some types of filters, like ZlibDecompressor, hard flushes can only /// be done at "synchronization points". These synchronization points are positions in the data /// stream that are created by hard flushes on the corresponding reverse filters, in this /// example ZlibCompressor. This is useful when zlib compressed data is moved across a /// network in packets and compression state is preserved across packets, as in the SSH2 protocol. bool OutputFlush(int outputSite, bool hardFlush, int propagation, bool blocking, const std::string &channel=DEFAULT_CHANNEL); /// \brief Marks the end of a series of messages, with signal propagation /// \param outputSite unknown, system crash between keyboard and chair... /// \param propagation the number of attached transformations the MessageSeriesEnd() signal should be passed /// \param blocking specifies whether the object should block when processing input /// \param channel the channel to process the data /// \return true is the MessageEnd signal was successful, false otherwise. /// \details Each object that receives the signal will perform its processing, decrement /// propagation, and then pass the signal on to attached transformations if the value is not 0. /// \details propagation count includes this object. Setting propagation to 1 means this /// object only. Setting propagation to -1 means unlimited propagation. /// \note There should be a MessageEnd() immediately before MessageSeriesEnd(). bool OutputMessageSeriesEnd(int outputSite, int propagation, bool blocking, const std::string &channel=DEFAULT_CHANNEL); private: member_ptr m_attachment; protected: size_t m_inputPosition; int m_continueAt; }; /// \brief Create a working space in a BufferedTransformation struct CRYPTOPP_DLL FilterPutSpaceHelper { virtual ~FilterPutSpaceHelper() {} /// \brief Create a working space in a BufferedTransformation /// \param target BufferedTransformation for the working space /// \param channel channel for the working space /// \param minSize minimum size of the allocation, in bytes /// \param desiredSize preferred size of the allocation, in bytes /// \param bufferSize actual size of the allocation, in bytes /// \pre desiredSize >= minSize and bufferSize >= minSize. /// \details bufferSize is an IN and OUT parameter. If HelpCreatePutSpace() returns a non-NULL value, then /// bufferSize is valid and provides the size of the working space created for the caller. /// \details Internally, HelpCreatePutSpace() calls \ref BufferedTransformation::ChannelCreatePutSpace /// "ChannelCreatePutSpace()" using desiredSize. If the target returns desiredSize with a size less /// than minSize (i.e., the request could not be fulfilled), then an internal SecByteBlock /// called m_tempSpace is resized and used for the caller. byte *HelpCreatePutSpace(BufferedTransformation &target, const std::string &channel, size_t minSize, size_t desiredSize, size_t &bufferSize) { CRYPTOPP_ASSERT(desiredSize >= minSize && bufferSize >= minSize); if (m_tempSpace.size() < minSize) { byte *result = target.ChannelCreatePutSpace(channel, desiredSize); if (desiredSize >= minSize) { bufferSize = desiredSize; return result; } m_tempSpace.New(bufferSize); } bufferSize = m_tempSpace.size(); return m_tempSpace.begin(); } /// \brief Create a working space in a BufferedTransformation /// \param target the BufferedTransformation for the working space /// \param channel channel for the working space /// \param minSize minimum size of the allocation, in bytes /// \return pointer to the created space /// \details Internally, the overload calls HelpCreatePutSpace() using minSize for missing arguments. /// \details The filter will delete the space. The caller does not need to delete the space. byte *HelpCreatePutSpace(BufferedTransformation &target, const std::string &channel, size_t minSize) {return HelpCreatePutSpace(target, channel, minSize, minSize, minSize);} /// \brief Create a working space in a BufferedTransformation /// \param target the BufferedTransformation for the working space /// \param channel channel for the working space /// \param minSize minimum size of the allocation, in bytes /// \param bufferSize the actual size of the allocation, in bytes /// \details Internally, the overload calls HelpCreatePutSpace() using minSize for missing arguments. /// \details The filter will delete the space. The caller does not need to delete the space. byte *HelpCreatePutSpace(BufferedTransformation &target, const std::string &channel, size_t minSize, size_t bufferSize) {return HelpCreatePutSpace(target, channel, minSize, minSize, bufferSize);} /// \brief Temporary working space SecByteBlock m_tempSpace; }; /// \brief Measure how many bytes and messages pass through the filter /// \details measure how many bytes and messages pass through the filter. The filter also serves as valve by /// maintaining a list of ranges to skip during processing. class CRYPTOPP_DLL MeterFilter : public Bufferless { public: virtual ~MeterFilter() {} /// \brief Construct a MeterFilter /// \param attachment an optional attached transformation /// \param transparent flag indicating if the filter should function transparently /// \details attachment can be NULL. The filter is transparent by default. If the filter is /// transparent, then PutMaybeModifiable() does not process a request and always returns 0. MeterFilter(BufferedTransformation *attachment=NULLPTR, bool transparent=true) : m_transparent(transparent), m_currentMessageBytes(0), m_totalBytes(0) , m_currentSeriesMessages(0), m_totalMessages(0), m_totalMessageSeries(0) , m_begin(NULLPTR), m_length(0) {Detach(attachment); ResetMeter();} /// \brief Set or change the transparent mode of this object /// \param transparent the new transparent mode void SetTransparent(bool transparent) {m_transparent = transparent;} /// \brief Adds a range to skip during processing /// \param message the message to apply the range /// \param position the 0-based index in the current stream /// \param size the length of the range /// \param sortNow flag indicating whether the range should be sorted /// \details Internally, MeterFilter maitains a deque of ranges to skip. As messages are processed, /// ranges of bytes are skipped according to the list of ranges. void AddRangeToSkip(unsigned int message, lword position, lword size, bool sortNow = true); /// \brief Resets the meter /// \details ResetMeter() reinitializes the meter by setting counters to 0 and removing previous /// skip ranges. void ResetMeter(); // BufferedTransformation in cryptlib.h void IsolatedInitialize(const NameValuePairs ¶meters) {CRYPTOPP_UNUSED(parameters); ResetMeter();} /// \brief Number of bytes in the current message /// \return the number of bytes in the current message lword GetCurrentMessageBytes() const {return m_currentMessageBytes;} /// \brief Number of bytes processed by the filter /// \return the number of bytes processed by the filter lword GetTotalBytes() const {return m_totalBytes;} /// \brief Message number in the series /// \return the message number in the series unsigned int GetCurrentSeriesMessages() const {return m_currentSeriesMessages;} /// \brief Number of messages in the message series /// \return the number of messages in the message series unsigned int GetTotalMessages() const {return m_totalMessages;} /// \brief Number of messages processed by the filter /// \return the number of messages processed by the filter unsigned int GetTotalMessageSeries() const {return m_totalMessageSeries;} // BufferedTransformation in cryptlib.h byte * CreatePutSpace(size_t &size) {return AttachedTransformation()->CreatePutSpace(size);} size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking); size_t PutModifiable2(byte *inString, size_t length, int messageEnd, bool blocking); bool IsolatedMessageSeriesEnd(bool blocking); private: size_t PutMaybeModifiable(byte *inString, size_t length, int messageEnd, bool blocking, bool modifiable); bool ShouldPropagateMessageEnd() const {return m_transparent;} bool ShouldPropagateMessageSeriesEnd() const {return m_transparent;} struct MessageRange { inline bool operator<(const MessageRange &b) const // BCB2006 workaround: this has to be a member function {return message < b.message || (message == b.message && position < b.position);} unsigned int message; lword position; lword size; }; bool m_transparent; lword m_currentMessageBytes, m_totalBytes; unsigned int m_currentSeriesMessages, m_totalMessages, m_totalMessageSeries; std::deque m_rangesToSkip; byte *m_begin; size_t m_length; }; /// \brief A transparent MeterFilter /// \sa MeterFilter, OpaqueFilter class CRYPTOPP_DLL TransparentFilter : public MeterFilter { public: /// \brief Construct a TransparentFilter /// \param attachment an optional attached transformation TransparentFilter(BufferedTransformation *attachment=NULLPTR) : MeterFilter(attachment, true) {} }; /// \brief A non-transparent MeterFilter /// \sa MeterFilter, TransparentFilter class CRYPTOPP_DLL OpaqueFilter : public MeterFilter { public: /// \brief Construct an OpaqueFilter /// \param attachment an optional attached transformation OpaqueFilter(BufferedTransformation *attachment=NULLPTR) : MeterFilter(attachment, false) {} }; /// \brief Divides an input stream into discrete blocks /// \details FilterWithBufferedInput divides the input stream into a first block, a number of /// middle blocks, and a last block. First and last blocks are optional, and middle blocks may /// be a stream instead (i.e. blockSize == 1). /// \sa AuthenticatedEncryptionFilter, AuthenticatedDecryptionFilter, HashVerificationFilter, /// SignatureVerificationFilter, StreamTransformationFilter class CRYPTOPP_DLL FilterWithBufferedInput : public Filter { public: virtual ~FilterWithBufferedInput() {} /// \brief Construct a FilterWithBufferedInput with an attached transformation /// \param attachment an attached transformation FilterWithBufferedInput(BufferedTransformation *attachment); /// \brief Construct a FilterWithBufferedInput with an attached transformation /// \param firstSize the size of the first block /// \param blockSize the size of middle blocks /// \param lastSize the size of the last block /// \param attachment an attached transformation /// \details firstSize and lastSize may be 0. blockSize must be at least 1. FilterWithBufferedInput(size_t firstSize, size_t blockSize, size_t lastSize, BufferedTransformation *attachment); void IsolatedInitialize(const NameValuePairs ¶meters); size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking) { return PutMaybeModifiable(const_cast(inString), length, messageEnd, blocking, false); } size_t PutModifiable2(byte *inString, size_t length, int messageEnd, bool blocking) { return PutMaybeModifiable(inString, length, messageEnd, blocking, true); } /// \brief Flushes data buffered by this object, without signal propagation /// \param hardFlush indicates whether all data should be flushed /// \param blocking specifies whether the object should block when processing input /// \return true if the Flush was successful, false otherwise /// \details IsolatedFlush() calls ForceNextPut() if hardFlush is true /// \note hardFlush must be used with care bool IsolatedFlush(bool hardFlush, bool blocking); /// \brief Flushes data buffered by this object /// \details The input buffer may contain more than blockSize bytes if lastSize != 0. /// ForceNextPut() forces a call to NextPut() if this is the case. void ForceNextPut(); protected: virtual bool DidFirstPut() const {return m_firstInputDone;} virtual size_t GetFirstPutSize() const {return m_firstSize;} virtual size_t GetBlockPutSize() const {return m_blockSize;} virtual size_t GetLastPutSize() const {return m_lastSize;} virtual void InitializeDerivedAndReturnNewSizes(const NameValuePairs ¶meters, size_t &firstSize, size_t &blockSize, size_t &lastSize) {CRYPTOPP_UNUSED(parameters); CRYPTOPP_UNUSED(firstSize); CRYPTOPP_UNUSED(blockSize); CRYPTOPP_UNUSED(lastSize); InitializeDerived(parameters);} virtual void InitializeDerived(const NameValuePairs ¶meters) {CRYPTOPP_UNUSED(parameters);} // FirstPut() is called if (firstSize != 0 and totalLength >= firstSize) // or (firstSize == 0 and (totalLength > 0 or a MessageEnd() is received)). // inString is m_firstSize in length. virtual void FirstPut(const byte *inString) =0; // NextPut() is called if totalLength >= firstSize+blockSize+lastSize virtual void NextPutSingle(const byte *inString) {CRYPTOPP_UNUSED(inString); CRYPTOPP_ASSERT(false);} // Same as NextPut() except length can be a multiple of blockSize // Either NextPut() or NextPutMultiple() must be overridden virtual void NextPutMultiple(const byte *inString, size_t length); // Same as NextPutMultiple(), but inString can be modified virtual void NextPutModifiable(byte *inString, size_t length) {NextPutMultiple(inString, length);} /// \brief Input the last block of data /// \param inString the input byte buffer /// \param length the size of the input buffer, in bytes /// \details LastPut() processes the last block of data and signals attached filters to do the same. /// LastPut() is always called. The pseudo algorithm for the logic is: ///
	///     if totalLength < firstSize then length == totalLength
	///     else if totalLength <= firstSize+lastSize then length == totalLength-firstSize
	///     else lastSize <= length < lastSize+blockSize
	/// 
virtual void LastPut(const byte *inString, size_t length) =0; virtual void FlushDerived() {} protected: size_t PutMaybeModifiable(byte *begin, size_t length, int messageEnd, bool blocking, bool modifiable); void NextPutMaybeModifiable(byte *inString, size_t length, bool modifiable) { if (modifiable) NextPutModifiable(inString, length); else NextPutMultiple(inString, length); } // This function should no longer be used, put this here to cause a compiler error // if someone tries to override NextPut(). virtual int NextPut(const byte *inString, size_t length) {CRYPTOPP_UNUSED(inString); CRYPTOPP_UNUSED(length); CRYPTOPP_ASSERT(false); return 0;} class BlockQueue { public: void ResetQueue(size_t blockSize, size_t maxBlocks); byte *GetBlock(); byte *GetContigousBlocks(size_t &numberOfBytes); size_t GetAll(byte *outString); void Put(const byte *inString, size_t length); size_t CurrentSize() const {return m_size;} size_t MaxSize() const {return m_buffer.size();} private: SecByteBlock m_buffer; size_t m_blockSize, m_maxBlocks, m_size; byte *m_begin; }; size_t m_firstSize, m_blockSize, m_lastSize; bool m_firstInputDone; BlockQueue m_queue; }; /// \brief A filter that buffers input using a ByteQueue /// \details FilterWithInputQueue will buffer input using a ByteQueue. When the filter receives /// a \ref BufferedTransformation::MessageEnd() "MessageEnd()" signal it will pass the data /// on to its attached transformation. class CRYPTOPP_DLL FilterWithInputQueue : public Filter { public: virtual ~FilterWithInputQueue() {} /// \brief Construct a FilterWithInputQueue /// \param attachment an optional attached transformation FilterWithInputQueue(BufferedTransformation *attachment=NULLPTR) : Filter(attachment) {} size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking) { if (!blocking) throw BlockingInputOnly("FilterWithInputQueue"); m_inQueue.Put(inString, length); if (messageEnd) { IsolatedMessageEnd(blocking); Output(0, NULLPTR, 0, messageEnd, blocking); } return 0; } protected: virtual bool IsolatedMessageEnd(bool blocking) =0; void IsolatedInitialize(const NameValuePairs ¶meters) {CRYPTOPP_UNUSED(parameters); m_inQueue.Clear();} ByteQueue m_inQueue; }; /// \struct BlockPaddingSchemeDef /// \brief Padding schemes used for block ciphers /// \since Crypto++ 5.0 struct BlockPaddingSchemeDef { /// \enum BlockPaddingScheme /// \brief Padding schemes used for block ciphers. /// \details DEFAULT_PADDING means PKCS_PADDING if cipher.MandatoryBlockSize() > 1 && /// cipher.MinLastBlockSize() == 0, which holds for ECB or CBC mode. Otherwise, /// NO_PADDING for modes like OFB, CFB, CTR, CBC-CTS. /// \sa Block Cipher Padding for /// additional details. /// \since Crypto++ 5.0 enum BlockPaddingScheme { /// \brief No padding added to a block /// \since Crypto++ 5.0 NO_PADDING, /// \brief 0's padding added to a block /// \since Crypto++ 5.0 ZEROS_PADDING, /// \brief PKCS padding added to a block /// \since Crypto++ 5.0 PKCS_PADDING, /// \brief 1 and 0's padding added to a block /// \since Crypto++ 5.0 ONE_AND_ZEROS_PADDING, /// \brief W3C padding added to a block /// \sa XML /// Encryption Syntax and Processing /// \since Crypto++ 6.0 W3C_PADDING, /// \brief Default padding scheme /// \since Crypto++ 5.0 DEFAULT_PADDING }; }; /// \brief Filter wrapper for StreamTransformation /// \details StreamTransformationFilter() is a filter wrapper for StreamTransformation(). It is used when /// pipelining data for stream ciphers and confidentiality-only block ciphers. The filter will optionally /// handle padding and unpadding when needed. If you are using an authenticated encryption mode of operation, /// then use AuthenticatedEncryptionFilter() and AuthenticatedDecryptionFilter() /// \since Crypto++ 5.0 class CRYPTOPP_DLL StreamTransformationFilter : public FilterWithBufferedInput, public BlockPaddingSchemeDef, private FilterPutSpaceHelper { public: virtual ~StreamTransformationFilter() {} /// \brief Construct a StreamTransformationFilter /// \param c reference to a StreamTransformation /// \param attachment an optional attached transformation /// \param padding the \ref BlockPaddingSchemeDef "padding scheme" /// \details This constructor creates a StreamTransformationFilter() for stream ciphers and /// confidentiality-only block cipher modes of operation. If you are using an authenticated /// encryption mode of operation, then use either AuthenticatedEncryptionFilter() or /// AuthenticatedDecryptionFilter(). /// \sa AuthenticatedEncryptionFilter() and AuthenticatedDecryptionFilter() StreamTransformationFilter(StreamTransformation &c, BufferedTransformation *attachment = NULLPTR, BlockPaddingScheme padding = DEFAULT_PADDING); std::string AlgorithmName() const {return m_cipher.AlgorithmName();} protected: friend class AuthenticatedEncryptionFilter; friend class AuthenticatedDecryptionFilter; /// \brief Construct a StreamTransformationFilter /// \param c reference to a StreamTransformation /// \param attachment an optional attached transformation /// \param padding the \ref BlockPaddingSchemeDef "padding scheme" /// \param authenticated flag indicating whether the filter should allow authenticated encryption schemes /// \details This constructor is used for authenticated encryption mode of operation and by /// AuthenticatedEncryptionFilter() and AuthenticatedDecryptionFilter(). StreamTransformationFilter(StreamTransformation &c, BufferedTransformation *attachment, BlockPaddingScheme padding, bool authenticated); void InitializeDerivedAndReturnNewSizes(const NameValuePairs ¶meters, size_t &firstSize, size_t &blockSize, size_t &lastSize); void FirstPut(const byte *inString); void NextPutMultiple(const byte *inString, size_t length); void NextPutModifiable(byte *inString, size_t length); void LastPut(const byte *inString, size_t length); static size_t LastBlockSize(StreamTransformation &c, BlockPaddingScheme padding); StreamTransformation &m_cipher; BlockPaddingScheme m_padding; unsigned int m_mandatoryBlockSize; unsigned int m_optimalBufferSize; unsigned int m_reservedBufferSize; bool m_isSpecial; }; /// \brief Filter wrapper for HashTransformation /// \since Crypto++ 1.0 class CRYPTOPP_DLL HashFilter : public Bufferless, private FilterPutSpaceHelper { public: virtual ~HashFilter() {} /// \brief Construct a HashFilter /// \param hm reference to a HashTransformation /// \param attachment an optional attached transformation /// \param putMessage flag indicating whether the original message should be passed to an attached transformation /// \param truncatedDigestSize the size of the digest /// \param messagePutChannel the channel on which the message should be output /// \param hashPutChannel the channel on which the digest should be output HashFilter(HashTransformation &hm, BufferedTransformation *attachment = NULLPTR, bool putMessage=false, int truncatedDigestSize=-1, const std::string &messagePutChannel=DEFAULT_CHANNEL, const std::string &hashPutChannel=DEFAULT_CHANNEL); std::string AlgorithmName() const {return m_hashModule.AlgorithmName();} void IsolatedInitialize(const NameValuePairs ¶meters); size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking); byte * CreatePutSpace(size_t &size) {return m_hashModule.CreateUpdateSpace(size);} private: HashTransformation &m_hashModule; bool m_putMessage; unsigned int m_digestSize; byte *m_space; std::string m_messagePutChannel, m_hashPutChannel; }; /// \brief Filter wrapper for HashTransformation /// \since Crypto++ 4.0 class CRYPTOPP_DLL HashVerificationFilter : public FilterWithBufferedInput { public: virtual ~HashVerificationFilter() {} /// \brief Exception thrown when a data integrity check failure is encountered class HashVerificationFailed : public Exception { public: HashVerificationFailed() : Exception(DATA_INTEGRITY_CHECK_FAILED, "HashVerificationFilter: message hash or MAC not valid") {} }; /// \enum Flags /// \brief Flags controlling filter behavior. /// \details The flags are a bitmask and can be OR'd together. enum Flags { /// \brief The hash is at the end of the message (i.e., concatenation of message+hash) HASH_AT_END=0, /// \brief The hash is at the beginning of the message (i.e., concatenation of hash+message) HASH_AT_BEGIN=1, /// \brief The message should be passed to an attached transformation PUT_MESSAGE=2, /// \brief The hash should be passed to an attached transformation PUT_HASH=4, /// \brief The result of the verification should be passed to an attached transformation PUT_RESULT=8, /// \brief The filter should throw a HashVerificationFailed if a failure is encountered THROW_EXCEPTION=16, /// \brief Default flags using HASH_AT_BEGIN and PUT_RESULT DEFAULT_FLAGS = HASH_AT_BEGIN | PUT_RESULT }; /// \brief Construct a HashVerificationFilter /// \param hm reference to a HashTransformation /// \param attachment an optional attached transformation /// \param flags flags indicating behaviors for the filter /// \param truncatedDigestSize the size of the digest /// \details truncatedDigestSize = -1 indicates \ref HashTransformation::DigestSize() "DigestSize" should be used. HashVerificationFilter(HashTransformation &hm, BufferedTransformation *attachment = NULLPTR, word32 flags = DEFAULT_FLAGS, int truncatedDigestSize=-1); std::string AlgorithmName() const {return m_hashModule.AlgorithmName();} bool GetLastResult() const {return m_verified;} protected: void InitializeDerivedAndReturnNewSizes(const NameValuePairs ¶meters, size_t &firstSize, size_t &blockSize, size_t &lastSize); void FirstPut(const byte *inString); void NextPutMultiple(const byte *inString, size_t length); void LastPut(const byte *inString, size_t length); private: friend class AuthenticatedDecryptionFilter; HashTransformation &m_hashModule; word32 m_flags; unsigned int m_digestSize; bool m_verified; SecByteBlock m_expectedHash; }; /// \brief Filter wrapper for encrypting with AuthenticatedSymmetricCipher /// \details AuthenticatedEncryptionFilter() is a wrapper for encrypting with /// AuthenticatedSymmetricCipher(), optionally handling padding/unpadding when needed. /// \details AuthenticatedDecryptionFilter() for Crypto++ 8.2 and earlier /// had a bug where a FileSource() would cause an exception, but a StringSource() /// was OK. Also see Issue 817 and Commit ff110c6e183e. /// \sa AuthenticatedSymmetricCipher, AuthenticatedDecryptionFilter, EAX, CCM, GCM, /// and AadSource on the /// Crypto++ wiki. /// \since Crypto++ 5.6.0 class CRYPTOPP_DLL AuthenticatedEncryptionFilter : public StreamTransformationFilter { public: virtual ~AuthenticatedEncryptionFilter() {} /// \brief Construct a AuthenticatedEncryptionFilter /// \param c reference to a AuthenticatedSymmetricCipher /// \param attachment an optional attached transformation /// \param putAAD flag indicating whether the AAD should be passed to an attached transformation /// \param truncatedDigestSize the size of the digest /// \param macChannel the channel on which the MAC should be output /// \param padding the \ref BlockPaddingSchemeDef "padding scheme" /// \details truncatedDigestSize = -1 indicates \ref HashTransformation::DigestSize() "DigestSize" should be used. /// \since Crypto++ 5.6.0 AuthenticatedEncryptionFilter(AuthenticatedSymmetricCipher &c, BufferedTransformation *attachment = NULLPTR, bool putAAD=false, int truncatedDigestSize=-1, const std::string &macChannel=DEFAULT_CHANNEL, BlockPaddingScheme padding = DEFAULT_PADDING); void IsolatedInitialize(const NameValuePairs ¶meters); byte * ChannelCreatePutSpace(const std::string &channel, size_t &size); size_t ChannelPut2(const std::string &channel, const byte *begin, size_t length, int messageEnd, bool blocking); /// \brief Input the last block of data /// \param inString the input byte buffer /// \param length the size of the input buffer, in bytes /// \details LastPut() processes the last block of data and signals attached filters to do the same. /// LastPut() is always called. The pseudo algorithm for the logic is: ///
	///     if totalLength < firstSize then length == totalLength
	///     else if totalLength <= firstSize+lastSize then length == totalLength-firstSize
	///     else lastSize <= length < lastSize+blockSize
	/// 
void LastPut(const byte *inString, size_t length); protected: HashFilter m_hf; }; /// \brief Filter wrapper for decrypting with AuthenticatedSymmetricCipher /// \details AuthenticatedDecryptionFilter() is a wrapper for decrypting with /// AuthenticatedSymmetricCipher(), optionally handling padding/unpadding when /// needed. /// \details AuthenticatedDecryptionFilter() for Crypto++ 8.2 and earlier /// had a bug where a FileSource() would cause an exception, but a StringSource() /// was OK. Also see Issue 817 and Commit ff110c6e183e. /// \sa AuthenticatedSymmetricCipher, AuthenticatedEncryptionFilter, EAX, CCM, GCM, /// and AadSource on the /// Crypto++ wiki. /// \since Crypto++ 5.6.0 class CRYPTOPP_DLL AuthenticatedDecryptionFilter : public FilterWithBufferedInput, public BlockPaddingSchemeDef { public: /// \enum Flags /// \brief Flags controlling filter behavior. /// \details The flags are a bitmask and can be OR'd together. enum Flags { /// \brief The MAC is at the end of the message (i.e., concatenation of message+mac) MAC_AT_END=0, /// \brief The MAC is at the beginning of the message (i.e., concatenation of mac+message) MAC_AT_BEGIN=1, /// \brief The filter should throw a HashVerificationFailed if a failure is encountered THROW_EXCEPTION=16, /// \brief Default flags using THROW_EXCEPTION DEFAULT_FLAGS = THROW_EXCEPTION }; virtual ~AuthenticatedDecryptionFilter() {} /// \brief Construct a AuthenticatedDecryptionFilter /// \param c reference to a AuthenticatedSymmetricCipher /// \param attachment an optional attached transformation /// \param flags flags indicating behaviors for the filter /// \param truncatedDigestSize the size of the digest /// \param padding the \ref BlockPaddingSchemeDef "padding scheme" /// \details Additional authenticated data should be given in channel "AAD". /// \details truncatedDigestSize = -1 indicates \ref HashTransformation::DigestSize() "DigestSize" should be used. /// \since Crypto++ 5.6.0 AuthenticatedDecryptionFilter(AuthenticatedSymmetricCipher &c, BufferedTransformation *attachment = NULLPTR, word32 flags = DEFAULT_FLAGS, int truncatedDigestSize=-1, BlockPaddingScheme padding = DEFAULT_PADDING); std::string AlgorithmName() const {return m_hashVerifier.AlgorithmName();} byte * ChannelCreatePutSpace(const std::string &channel, size_t &size); size_t ChannelPut2(const std::string &channel, const byte *begin, size_t length, int messageEnd, bool blocking); size_t ChannelPutModifiable2(const std::string &channel, byte *begin, size_t length, int messageEnd, bool blocking) { return ChannelPut2(channel, begin, length, messageEnd, blocking); } /// \brief Get verifier result /// \return true if the digest on the previosus message was valid, false otherwise bool GetLastResult() const {return m_hashVerifier.GetLastResult();} protected: void InitializeDerivedAndReturnNewSizes(const NameValuePairs ¶meters, size_t &firstSize, size_t &blockSize, size_t &lastSize); void FirstPut(const byte *inString); void NextPutMultiple(const byte *inString, size_t length); /// \brief Input the last block of data /// \param inString the input byte buffer /// \param length the size of the input buffer, in bytes /// \details LastPut() processes the last block of data and signals attached filters to do the same. /// LastPut() is always called. The pseudo algorithm for the logic is: ///
	///     if totalLength < firstSize then length == totalLength
	///     else if totalLength <= firstSize+lastSize then length == totalLength-firstSize
	///     else lastSize <= length < lastSize+blockSize
	/// 
void LastPut(const byte *inString, size_t length); HashVerificationFilter m_hashVerifier; StreamTransformationFilter m_streamFilter; }; /// \brief Filter wrapper for PK_Signer /// \since Crypto++ 4.0 class CRYPTOPP_DLL SignerFilter : public Unflushable { public: virtual ~SignerFilter() {} /// \brief Construct a SignerFilter /// \param rng a RandomNumberGenerator derived class /// \param signer a PK_Signer derived class /// \param attachment an optional attached transformation /// \param putMessage flag indicating whether the original message should be passed to an attached transformation SignerFilter(RandomNumberGenerator &rng, const PK_Signer &signer, BufferedTransformation *attachment = NULLPTR, bool putMessage=false) : m_rng(rng), m_signer(signer), m_messageAccumulator(signer.NewSignatureAccumulator(rng)), m_putMessage(putMessage) {Detach(attachment);} std::string AlgorithmName() const {return m_signer.AlgorithmName();} void IsolatedInitialize(const NameValuePairs ¶meters); size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking); private: RandomNumberGenerator &m_rng; const PK_Signer &m_signer; member_ptr m_messageAccumulator; bool m_putMessage; SecByteBlock m_buf; }; /// \brief Filter wrapper for PK_Verifier /// \details This filter was formerly named VerifierFilter. The name changed at Crypto++ 5.0. /// \since Crypto++ 4.0 class CRYPTOPP_DLL SignatureVerificationFilter : public FilterWithBufferedInput { public: /// \brief Exception thrown when an invalid signature is encountered class SignatureVerificationFailed : public Exception { public: SignatureVerificationFailed() : Exception(DATA_INTEGRITY_CHECK_FAILED, "VerifierFilter: digital signature not valid") {} }; /// \enum Flags /// \brief Flags controlling filter behavior. /// \details The flags are a bitmask and can be OR'd together. enum Flags { /// \brief The signature is at the end of the message (i.e., concatenation of message+signature) SIGNATURE_AT_END=0, /// \brief The signature is at the beginning of the message (i.e., concatenation of signature+message) SIGNATURE_AT_BEGIN=1, /// \brief The message should be passed to an attached transformation PUT_MESSAGE=2, /// \brief The signature should be passed to an attached transformation PUT_SIGNATURE=4, /// \brief The result of the verification should be passed to an attached transformation PUT_RESULT=8, /// \brief The filter should throw a HashVerificationFailed if a failure is encountered THROW_EXCEPTION=16, /// \brief Default flags using SIGNATURE_AT_BEGIN and PUT_RESULT DEFAULT_FLAGS = SIGNATURE_AT_BEGIN | PUT_RESULT }; virtual ~SignatureVerificationFilter() {} /// \brief Construct a SignatureVerificationFilter /// \param verifier a PK_Verifier derived class /// \param attachment an optional attached transformation /// \param flags flags indicating behaviors for the filter SignatureVerificationFilter(const PK_Verifier &verifier, BufferedTransformation *attachment = NULLPTR, word32 flags = DEFAULT_FLAGS); std::string AlgorithmName() const {return m_verifier.AlgorithmName();} /// \brief Retrieves the result of the last verification /// \return true if the signature on the previosus message was valid, false otherwise bool GetLastResult() const {return m_verified;} protected: void InitializeDerivedAndReturnNewSizes(const NameValuePairs ¶meters, size_t &firstSize, size_t &blockSize, size_t &lastSize); void FirstPut(const byte *inString); void NextPutMultiple(const byte *inString, size_t length); void LastPut(const byte *inString, size_t length); private: const PK_Verifier &m_verifier; member_ptr m_messageAccumulator; word32 m_flags; SecByteBlock m_signature; bool m_verified; }; /// \brief Redirect input to another BufferedTransformation without owning it /// \since Crypto++ 4.0 class CRYPTOPP_DLL Redirector : public CustomSignalPropagation { public: /// \enum Behavior /// \brief Controls signal propagation behavior enum Behavior { /// \brief Pass data only DATA_ONLY = 0x00, /// \brief Pass signals PASS_SIGNALS = 0x01, /// \brief Pass wait events PASS_WAIT_OBJECTS = 0x02, /// \brief Pass everything /// \details PASS_EVERYTHING is default PASS_EVERYTHING = PASS_SIGNALS | PASS_WAIT_OBJECTS }; virtual ~Redirector() {} /// \brief Construct a Redirector Redirector() : m_target(NULLPTR), m_behavior(PASS_EVERYTHING) {} /// \brief Construct a Redirector /// \param target the destination BufferedTransformation /// \param behavior Behavior "flags" specifying signal propagation Redirector(BufferedTransformation &target, Behavior behavior=PASS_EVERYTHING) : m_target(&target), m_behavior(behavior) {} /// \brief Redirect input to another BufferedTransformation /// \param target the destination BufferedTransformation void Redirect(BufferedTransformation &target) {m_target = ⌖} /// \brief Stop redirecting input void StopRedirection() {m_target = NULLPTR;} /// \brief Retrieve signal propagation behavior /// \return the current signal propagation behavior Behavior GetBehavior() {return static_cast(m_behavior);} /// \brief Set signal propagation behavior /// \param behavior the new signal propagation behavior void SetBehavior(Behavior behavior) {m_behavior=behavior;} /// \brief Retrieve signal propagation behavior /// \return true if the Redirector passes signals, false otherwise. bool GetPassSignals() const {return (m_behavior & PASS_SIGNALS) != 0;} /// \brief Set signal propagation behavior /// \param pass flag indicating if the Redirector should pass signals void SetPassSignals(bool pass) { if (pass) m_behavior |= PASS_SIGNALS; else m_behavior &= ~static_cast(PASS_SIGNALS); } /// \brief Retrieve signal propagation behavior /// \return true if the Redirector passes wait objects, false otherwise. bool GetPassWaitObjects() const {return (m_behavior & PASS_WAIT_OBJECTS) != 0;} /// \brief Set signal propagation behavior /// \param pass flag indicating if the Redirector should pass wait objects void SetPassWaitObjects(bool pass) { if (pass) m_behavior |= PASS_WAIT_OBJECTS; else m_behavior &= ~static_cast(PASS_WAIT_OBJECTS); } bool CanModifyInput() const {return m_target ? m_target->CanModifyInput() : false;} void Initialize(const NameValuePairs ¶meters, int propagation); byte * CreatePutSpace(size_t &size) { if (m_target) return m_target->CreatePutSpace(size); else { size = 0; return NULLPTR; } } size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking) {return m_target ? m_target->Put2(inString, length, GetPassSignals() ? messageEnd : 0, blocking) : 0;} bool Flush(bool hardFlush, int propagation=-1, bool blocking=true) {return m_target && GetPassSignals() ? m_target->Flush(hardFlush, propagation, blocking) : false;} bool MessageSeriesEnd(int propagation=-1, bool blocking=true) {return m_target && GetPassSignals() ? m_target->MessageSeriesEnd(propagation, blocking) : false;} byte * ChannelCreatePutSpace(const std::string &channel, size_t &size) { if (m_target) return m_target->ChannelCreatePutSpace(channel, size); else { size = 0; return NULLPTR; } } size_t ChannelPut2(const std::string &channel, const byte *begin, size_t length, int messageEnd, bool blocking) {return m_target ? m_target->ChannelPut2(channel, begin, length, GetPassSignals() ? messageEnd : 0, blocking) : 0;} size_t ChannelPutModifiable2(const std::string &channel, byte *begin, size_t length, int messageEnd, bool blocking) {return m_target ? m_target->ChannelPutModifiable2(channel, begin, length, GetPassSignals() ? messageEnd : 0, blocking) : 0;} bool ChannelFlush(const std::string &channel, bool completeFlush, int propagation=-1, bool blocking=true) {return m_target && GetPassSignals() ? m_target->ChannelFlush(channel, completeFlush, propagation, blocking) : false;} bool ChannelMessageSeriesEnd(const std::string &channel, int propagation=-1, bool blocking=true) {return m_target && GetPassSignals() ? m_target->ChannelMessageSeriesEnd(channel, propagation, blocking) : false;} unsigned int GetMaxWaitObjectCount() const { return m_target && GetPassWaitObjects() ? m_target->GetMaxWaitObjectCount() : 0; } void GetWaitObjects(WaitObjectContainer &container, CallStack const& callStack) { if (m_target && GetPassWaitObjects()) m_target->GetWaitObjects(container, callStack); } private: BufferedTransformation *m_target; word32 m_behavior; }; /// \brief Filter class that is a proxy for a sink /// \details Used By ProxyFilter /// \since Crypto++ 4.0 class CRYPTOPP_DLL OutputProxy : public CustomSignalPropagation { public: virtual ~OutputProxy() {} /// \brief Construct an OutputProxy /// \param owner the owning transformation /// \param passSignal flag indicating if signals should be passed OutputProxy(BufferedTransformation &owner, bool passSignal) : m_owner(owner), m_passSignal(passSignal) {} /// \brief Retrieve passSignal flag /// \return flag indicating if signals should be passed bool GetPassSignal() const {return m_passSignal;} /// \brief Set passSignal flag /// \param passSignal flag indicating if signals should be passed void SetPassSignal(bool passSignal) {m_passSignal = passSignal;} byte * CreatePutSpace(size_t &size) {return m_owner.AttachedTransformation()->CreatePutSpace(size);} size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking) {return m_owner.AttachedTransformation()->Put2(inString, length, m_passSignal ? messageEnd : 0, blocking);} size_t PutModifiable2(byte *begin, size_t length, int messageEnd, bool blocking) {return m_owner.AttachedTransformation()->PutModifiable2(begin, length, m_passSignal ? messageEnd : 0, blocking);} void Initialize(const NameValuePairs ¶meters=g_nullNameValuePairs, int propagation=-1) {if (m_passSignal) m_owner.AttachedTransformation()->Initialize(parameters, propagation);} bool Flush(bool hardFlush, int propagation=-1, bool blocking=true) {return m_passSignal ? m_owner.AttachedTransformation()->Flush(hardFlush, propagation, blocking) : false;} bool MessageSeriesEnd(int propagation=-1, bool blocking=true) {return m_passSignal ? m_owner.AttachedTransformation()->MessageSeriesEnd(propagation, blocking) : false;} byte * ChannelCreatePutSpace(const std::string &channel, size_t &size) {return m_owner.AttachedTransformation()->ChannelCreatePutSpace(channel, size);} size_t ChannelPut2(const std::string &channel, const byte *begin, size_t length, int messageEnd, bool blocking) {return m_owner.AttachedTransformation()->ChannelPut2(channel, begin, length, m_passSignal ? messageEnd : 0, blocking);} size_t ChannelPutModifiable2(const std::string &channel, byte *begin, size_t length, int messageEnd, bool blocking) {return m_owner.AttachedTransformation()->ChannelPutModifiable2(channel, begin, length, m_passSignal ? messageEnd : 0, blocking);} bool ChannelFlush(const std::string &channel, bool completeFlush, int propagation=-1, bool blocking=true) {return m_passSignal ? m_owner.AttachedTransformation()->ChannelFlush(channel, completeFlush, propagation, blocking) : false;} bool ChannelMessageSeriesEnd(const std::string &channel, int propagation=-1, bool blocking=true) {return m_passSignal ? m_owner.AttachedTransformation()->ChannelMessageSeriesEnd(channel, propagation, blocking) : false;} private: BufferedTransformation &m_owner; bool m_passSignal; }; /// \brief Base class for Filter classes that are proxies for a chain of other filters /// \since Crypto++ 4.0 class CRYPTOPP_DLL ProxyFilter : public FilterWithBufferedInput { public: virtual ~ProxyFilter() {} /// \brief Construct a ProxyFilter /// \param filter an output filter /// \param firstSize the first Put size /// \param lastSize the last Put size /// \param attachment an attached transformation ProxyFilter(BufferedTransformation *filter, size_t firstSize, size_t lastSize, BufferedTransformation *attachment); bool IsolatedFlush(bool hardFlush, bool blocking); /// \brief Sets the OutputProxy filter /// \param filter an OutputProxy filter void SetFilter(Filter *filter); void NextPutMultiple(const byte *s, size_t len); void NextPutModifiable(byte *inString, size_t length); protected: member_ptr m_filter; }; /// \brief Proxy filter that doesn't modify the underlying filter's input or output /// \since Crypto++ 5.0 class CRYPTOPP_DLL SimpleProxyFilter : public ProxyFilter { public: /// \brief Construct a SimpleProxyFilter /// \param filter an output filter /// \param attachment an attached transformation SimpleProxyFilter(BufferedTransformation *filter, BufferedTransformation *attachment) : ProxyFilter(filter, 0, 0, attachment) {} void FirstPut(const byte * inString) {CRYPTOPP_UNUSED(inString);} /// \brief Input the last block of data /// \param inString the input byte buffer /// \param length the size of the input buffer, in bytes /// \details LastPut() processes the last block of data and signals attached filters to do the same. /// LastPut() is always called. The pseudo algorithm for the logic is: ///
	///     if totalLength < firstSize then length == totalLength
	///     else if totalLength <= firstSize+lastSize then length == totalLength-firstSize
	///     else lastSize <= length < lastSize+blockSize
	/// 
void LastPut(const byte *inString, size_t length) {CRYPTOPP_UNUSED(inString), CRYPTOPP_UNUSED(length); m_filter->MessageEnd();} }; /// \brief Filter wrapper for PK_Encryptor /// \details PK_DecryptorFilter is a proxy for the filter created by PK_Encryptor::CreateEncryptionFilter. /// This class provides symmetry with VerifierFilter. /// \since Crypto++ 5.0 class CRYPTOPP_DLL PK_EncryptorFilter : public SimpleProxyFilter { public: /// \brief Construct a PK_EncryptorFilter /// \param rng a RandomNumberGenerator derived class /// \param encryptor a PK_Encryptor derived class /// \param attachment an optional attached transformation PK_EncryptorFilter(RandomNumberGenerator &rng, const PK_Encryptor &encryptor, BufferedTransformation *attachment = NULLPTR) : SimpleProxyFilter(encryptor.CreateEncryptionFilter(rng), attachment) {} }; /// \brief Filter wrapper for PK_Decryptor /// \details PK_DecryptorFilter is a proxy for the filter created by PK_Decryptor::CreateDecryptionFilter. /// This class provides symmetry with SignerFilter. /// \since Crypto++ 5.0 class CRYPTOPP_DLL PK_DecryptorFilter : public SimpleProxyFilter { public: /// \brief Construct a PK_DecryptorFilter /// \param rng a RandomNumberGenerator derived class /// \param decryptor a PK_Decryptor derived class /// \param attachment an optional attached transformation PK_DecryptorFilter(RandomNumberGenerator &rng, const PK_Decryptor &decryptor, BufferedTransformation *attachment = NULLPTR) : SimpleProxyFilter(decryptor.CreateDecryptionFilter(rng), attachment) {} }; /// \brief Append input to a string object /// \tparam T std::basic_string type /// \details StringSinkTemplate is a StringSinkTemplate typedef /// \since Crypto++ 5.0 template class StringSinkTemplate : public Bufferless { public: typedef typename T::value_type value_type; virtual ~StringSinkTemplate() {} /// \brief Construct a StringSinkTemplate /// \param output std::basic_string or std::vector type StringSinkTemplate(T &output) : m_output(&output) {CRYPTOPP_ASSERT(sizeof(value_type)==1);} void IsolatedInitialize(const NameValuePairs ¶meters) {if (!parameters.GetValue("OutputStringPointer", m_output)) throw InvalidArgument("StringSink: OutputStringPointer not specified");} size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking) { CRYPTOPP_UNUSED(messageEnd); CRYPTOPP_UNUSED(blocking); if (length > 0) { typename T::size_type size = m_output->size(); if (length < size && size + length > m_output->capacity()) m_output->reserve(2*size); m_output->insert(m_output->end(), (const value_type *)inString, (const value_type *)inString+length); } return 0; } private: T *m_output; }; /// \brief Append input to a string object /// \details StringSink is a typedef for StringSinkTemplate. /// \sa ArraySink, ArrayXorSink /// \since Crypto++ 4.0 DOCUMENTED_TYPEDEF(StringSinkTemplate, StringSink); CRYPTOPP_DLL_TEMPLATE_CLASS StringSinkTemplate; /// \brief Append input to a std::vector object /// \details VectorSink is a typedef for StringSinkTemplate >. /// \since Crypto++ 8.0 DOCUMENTED_TYPEDEF(StringSinkTemplate >, VectorSink); CRYPTOPP_DLL_TEMPLATE_CLASS StringSinkTemplate >; /// \brief Incorporates input into RNG as additional entropy /// \since Crypto++ 4.0 class RandomNumberSink : public Bufferless { public: virtual ~RandomNumberSink() {} /// \brief Construct a RandomNumberSink RandomNumberSink() : m_rng(NULLPTR) {} /// \brief Construct a RandomNumberSink /// \param rng a RandomNumberGenerator derived class RandomNumberSink(RandomNumberGenerator &rng) : m_rng(&rng) {} void IsolatedInitialize(const NameValuePairs ¶meters); size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking); private: RandomNumberGenerator *m_rng; }; /// \brief Copy input to a memory buffer /// \details ArraySink wraps a fixed size buffer. The buffer is full once Put returns non-0. /// When used in a pipeline, ArraySink silently discards input if the buffer is full. /// AvailableSize() can be used to determine how much space remains in the buffer. /// TotalPutLength() can be used to determine how many bytes were processed. /// \sa StringSink, ArrayXorSink /// \since Crypto++ 4.0 class CRYPTOPP_DLL ArraySink : public Bufferless { public: virtual ~ArraySink() {} /// \brief Construct an ArraySink /// \param parameters a set of NameValuePairs to initialize this object /// \details Name::OutputBuffer() is a mandatory parameter using this constructor. ArraySink(const NameValuePairs ¶meters = g_nullNameValuePairs) : m_buf(NULLPTR), m_size(0), m_total(0) {IsolatedInitialize(parameters);} /// \brief Construct an ArraySink /// \param buf pointer to a memory buffer /// \param size length of the memory buffer ArraySink(byte *buf, size_t size) : m_buf(buf), m_size(size), m_total(0) {} /// \brief Provides the size remaining in the Sink /// \return size remaining in the Sink, in bytes size_t AvailableSize() {return SaturatingSubtract(m_size, m_total);} /// \brief Provides the number of bytes written to the Sink /// \return number of bytes written to the Sink, in bytes lword TotalPutLength() {return m_total;} void IsolatedInitialize(const NameValuePairs ¶meters); byte * CreatePutSpace(size_t &size); size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking); protected: byte *m_buf; size_t m_size; lword m_total; }; /// \brief Xor input to a memory buffer /// \details ArrayXorSink wraps a fixed size buffer. The buffer is full once Put returns non-0. /// When used in a pipeline, ArrayXorSink silently discards input if the buffer is full. /// AvailableSize() can be used to determine how much space remains in the buffer. /// TotalPutLength() can be used to determine how many bytes were processed. /// \sa StringSink, ArraySink /// \since Crypto++ 4.0 class CRYPTOPP_DLL ArrayXorSink : public ArraySink { public: virtual ~ArrayXorSink() {} /// \brief Construct an ArrayXorSink /// \param buf pointer to a memory buffer /// \param size length of the memory buffer ArrayXorSink(byte *buf, size_t size) : ArraySink(buf, size) {} size_t Put2(const byte *inString, size_t length, int messageEnd, bool blocking); byte * CreatePutSpace(size_t &size) {return BufferedTransformation::CreatePutSpace(size);} }; /// \brief String-based implementation of Store interface /// \since Crypto++ 4.0 class StringStore : public Store { public: /// \brief Construct a StringStore /// \param string pointer to a C-String StringStore(const char *string = NULLPTR) {StoreInitialize(MakeParameters("InputBuffer", ConstByteArrayParameter(string)));} /// \brief Construct a StringStore /// \param string pointer to a memory buffer /// \param length size of the memory buffer StringStore(const byte *string, size_t length) {StoreInitialize(MakeParameters("InputBuffer", ConstByteArrayParameter(string, length)));} /// \brief Construct a StringStore /// \tparam T std::basic_string type /// \param string reference to a std::basic_string type template StringStore(const T &string) {StoreInitialize(MakeParameters("InputBuffer", ConstByteArrayParameter(string)));} CRYPTOPP_DLL size_t TransferTo2(BufferedTransformation &target, lword &transferBytes, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true); CRYPTOPP_DLL size_t CopyRangeTo2(BufferedTransformation &target, lword &begin, lword end=LWORD_MAX, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true) const; private: CRYPTOPP_DLL void StoreInitialize(const NameValuePairs ¶meters); const byte *m_store; size_t m_length, m_count; }; /// \brief RNG-based implementation of Source interface /// \since Crypto++ 4.0 class CRYPTOPP_DLL RandomNumberStore : public Store { public: virtual ~RandomNumberStore() {} RandomNumberStore() : m_rng(NULLPTR), m_length(0), m_count(0) {} RandomNumberStore(RandomNumberGenerator &rng, lword length) : m_rng(&rng), m_length(length), m_count(0) {} bool AnyRetrievable() const {return MaxRetrievable() != 0;} lword MaxRetrievable() const {return m_length-m_count;} size_t TransferTo2(BufferedTransformation &target, lword &transferBytes, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true); size_t CopyRangeTo2(BufferedTransformation &target, lword &begin, lword end=LWORD_MAX, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true) const { CRYPTOPP_UNUSED(target); CRYPTOPP_UNUSED(begin); CRYPTOPP_UNUSED(end); CRYPTOPP_UNUSED(channel); CRYPTOPP_UNUSED(blocking); throw NotImplemented("RandomNumberStore: CopyRangeTo2() is not supported by this store"); } private: void StoreInitialize(const NameValuePairs ¶meters); RandomNumberGenerator *m_rng; lword m_length, m_count; }; /// \brief Empty store /// \since Crypto++ 5.0 class CRYPTOPP_DLL NullStore : public Store { public: NullStore(lword size = ULONG_MAX) : m_size(size) {} void StoreInitialize(const NameValuePairs ¶meters) {CRYPTOPP_UNUSED(parameters);} lword MaxRetrievable() const {return m_size;} size_t TransferTo2(BufferedTransformation &target, lword &transferBytes, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true); size_t CopyRangeTo2(BufferedTransformation &target, lword &begin, lword end=LWORD_MAX, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true) const; private: lword m_size; }; /// \brief Implementation of BufferedTransformation's attachment interface /// \details Source is a cornerstone of the Pipeline trinitiy. Data flows from /// Sources, through Filters, and then terminates in Sinks. The difference /// between a Source and Filter is a Source \a pumps data, while a Filter does /// not. The difference between a Filter and a Sink is a Filter allows an /// attached transformation, while a Sink does not. /// \details See the discussion of BufferedTransformation in cryptlib.h for /// more details. /// \sa Store and SourceTemplate /// \since Crypto++ 1.0 class CRYPTOPP_DLL CRYPTOPP_NO_VTABLE Source : public InputRejecting { public: virtual ~Source() {} /// \brief Construct a Source /// \param attachment an optional attached transformation Source(BufferedTransformation *attachment = NULLPTR) {Source::Detach(attachment);} /// \name PIPELINE //@{ /// \brief Pump data to attached transformation /// \param pumpMax the maximum number of bytes to pump /// \return the number of bytes that remain to be processed (i.e., bytes not processed). /// 0 indicates all bytes were processed. /// \details Internally, Pump() calls Pump2(). /// \note pumpMax is a lword, which is a 64-bit value that typically uses /// LWORD_MAX. The default argument is SIZE_MAX, and it can be /// 32-bits or 64-bits. /// \sa Pump2, PumpAll, AnyRetrievable, MaxRetrievable lword Pump(lword pumpMax=SIZE_MAX) {Pump2(pumpMax); return pumpMax;} /// \brief Pump messages to attached transformation /// \param count the maximum number of messages to pump /// \return TODO /// \details Internally, PumpMessages() calls PumpMessages2(). unsigned int PumpMessages(unsigned int count=UINT_MAX) {PumpMessages2(count); return count;} /// \brief Pump all data to attached transformation /// \details Pumps all data to the attached transformation and signal the end of the current /// message. To avoid the MessageEnd() signal call \ref Pump "Pump(LWORD_MAX)" or \ref Pump2 /// "Pump2(LWORD_MAX, bool)". /// \details Internally, PumpAll() calls PumpAll2(), which calls PumpMessages(). /// \sa Pump, Pump2, AnyRetrievable, MaxRetrievable void PumpAll() {PumpAll2();} /// \brief Pump data to attached transformation /// \param byteCount the maximum number of bytes to pump /// \param blocking specifies whether the object should block when processing input /// \return the number of bytes that remain to be processed (i.e., bytes not processed). /// 0 indicates all bytes were processed. /// \details byteCount is an \a IN and \a OUT parameter. When the call is made, byteCount is the /// requested size of the pump. When the call returns, byteCount is the number of bytes that /// were pumped. /// \sa Pump, PumpAll, AnyRetrievable, MaxRetrievable virtual size_t Pump2(lword &byteCount, bool blocking=true) =0; /// \brief Pump messages to attached transformation /// \param messageCount the maximum number of messages to pump /// \param blocking specifies whether the object should block when processing input /// \details messageCount is an IN and OUT parameter. virtual size_t PumpMessages2(unsigned int &messageCount, bool blocking=true) =0; /// \brief Pump all data to attached transformation /// \param blocking specifies whether the object should block when processing input /// \return the number of bytes that remain to be processed (i.e., bytes not processed). /// 0 indicates all bytes were processed. /// \sa Pump, Pump2, AnyRetrievable, MaxRetrievable virtual size_t PumpAll2(bool blocking=true); /// \brief Determines if the Source is exhausted /// \return true if the source has been exhausted virtual bool SourceExhausted() const =0; //@} protected: void SourceInitialize(bool pumpAll, const NameValuePairs ¶meters) { IsolatedInitialize(parameters); if (pumpAll) PumpAll(); } }; /// \brief Transform a Store into a Source /// \tparam T the class or type /// \since Crypto++ 5.0 template class SourceTemplate : public Source { public: virtual ~SourceTemplate() {} /// \brief Construct a SourceTemplate /// \param attachment an attached transformation SourceTemplate(BufferedTransformation *attachment) : Source(attachment) {} void IsolatedInitialize(const NameValuePairs ¶meters) {m_store.IsolatedInitialize(parameters);} size_t Pump2(lword &byteCount, bool blocking=true) {return m_store.TransferTo2(*AttachedTransformation(), byteCount, DEFAULT_CHANNEL, blocking);} size_t PumpMessages2(unsigned int &messageCount, bool blocking=true) {return m_store.TransferMessagesTo2(*AttachedTransformation(), messageCount, DEFAULT_CHANNEL, blocking);} size_t PumpAll2(bool blocking=true) {return m_store.TransferAllTo2(*AttachedTransformation(), DEFAULT_CHANNEL, blocking);} bool SourceExhausted() const {return !m_store.AnyRetrievable() && !m_store.AnyMessages();} void SetAutoSignalPropagation(int propagation) {m_store.SetAutoSignalPropagation(propagation);} int GetAutoSignalPropagation() const {return m_store.GetAutoSignalPropagation();} protected: T m_store; }; /// \brief String-based implementation of the Source interface /// \since Crypto++ 4.0 class CRYPTOPP_DLL StringSource : public SourceTemplate { public: /// \brief Construct a StringSource /// \param attachment an optional attached transformation StringSource(BufferedTransformation *attachment = NULLPTR) : SourceTemplate(attachment) {} /// \brief Construct a StringSource /// \param string C-String /// \param pumpAll flag indicating if source data should be pumped to its attached transformation /// \param attachment an optional attached transformation StringSource(const char *string, bool pumpAll, BufferedTransformation *attachment = NULLPTR) : SourceTemplate(attachment) {SourceInitialize(pumpAll, MakeParameters("InputBuffer", ConstByteArrayParameter(string)));} /// \brief Construct a StringSource /// \param string binary byte array /// \param length size of the byte array /// \param pumpAll flag indicating if source data should be pumped to its attached transformation /// \param attachment an optional attached transformation StringSource(const byte *string, size_t length, bool pumpAll, BufferedTransformation *attachment = NULLPTR) : SourceTemplate(attachment) {SourceInitialize(pumpAll, MakeParameters("InputBuffer", ConstByteArrayParameter(string, length)));} /// \brief Construct a StringSource /// \param string std::string /// \param pumpAll flag indicating if source data should be pumped to its attached transformation /// \param attachment an optional attached transformation StringSource(const std::string &string, bool pumpAll, BufferedTransformation *attachment = NULLPTR) : SourceTemplate(attachment) {SourceInitialize(pumpAll, MakeParameters("InputBuffer", ConstByteArrayParameter(string)));} }; /// \brief Pointer-based implementation of the Source interface /// \details ArraySource is a typedef for StringSource. Use the third constructor for an array source. /// The third constructor takes a pointer and length. /// \since Crypto++ 5.6.0 DOCUMENTED_TYPEDEF(StringSource, ArraySource); /// \brief std::vector-based implementation of the Source interface /// \since Crypto++ 8.0 class CRYPTOPP_DLL VectorSource : public SourceTemplate { public: /// \brief Construct a VectorSource /// \param attachment an optional attached transformation VectorSource(BufferedTransformation *attachment = NULLPTR) : SourceTemplate(attachment) {} /// \brief Construct a VectorSource /// \param vec vector of bytes /// \param pumpAll flag indicating if source data should be pumped to its attached transformation /// \param attachment an optional attached transformation VectorSource(const std::vector &vec, bool pumpAll, BufferedTransformation *attachment = NULLPTR) : SourceTemplate(attachment) {SourceInitialize(pumpAll, MakeParameters("InputBuffer", ConstByteArrayParameter(vec)));} }; /// \brief RNG-based implementation of Source interface /// \since Crypto++ 4.0 class CRYPTOPP_DLL RandomNumberSource : public SourceTemplate { public: RandomNumberSource(RandomNumberGenerator &rng, int length, bool pumpAll, BufferedTransformation *attachment = NULLPTR) : SourceTemplate(attachment) {SourceInitialize(pumpAll, MakeParameters("RandomNumberGeneratorPointer", &rng)("RandomNumberStoreSize", length));} }; NAMESPACE_END #if CRYPTOPP_MSC_VERSION # pragma warning(pop) #endif #endif