2015-11-05 06:59:46 +00:00
// simple.h - written and placed in the public domain by Wei Dai
2015-11-18 20:32:28 +00:00
//! \file simple.h
2015-12-17 00:39:29 +00:00
//! \brief Classes providing basic library services.
2015-11-05 06:59:46 +00:00
# ifndef CRYPTOPP_SIMPLE_H
# define CRYPTOPP_SIMPLE_H
# include "config.h"
# if CRYPTOPP_MSC_VERSION
# pragma warning(push)
# pragma warning(disable: 4127 4189)
# endif
# include "cryptlib.h"
# include "misc.h"
NAMESPACE_BEGIN ( CryptoPP )
2015-11-18 20:32:28 +00:00
//! \class ClonableImpl
//! \brief Base class for identifying alogorithm
//! \tparam BASE base class from which to derive
//! \tparam DERIVED class which to clone
2015-11-05 06:59:46 +00:00
template < class DERIVED , class BASE >
class CRYPTOPP_NO_VTABLE ClonableImpl : public BASE
{
public :
Clonable * Clone ( ) const { return new DERIVED ( * static_cast < const DERIVED * > ( this ) ) ; }
} ;
2015-11-18 20:32:28 +00:00
//! \class AlgorithmImpl
//! \brief Base class for identifying alogorithm
//! \tparam BASE an Algorithm derived class
//! \tparam ALGORITHM_INFO an Algorithm derived class
//! \details AlgorithmImpl provides StaticAlgorithmName from the template parameter BASE
2015-11-05 06:59:46 +00:00
template < class BASE , class ALGORITHM_INFO = BASE >
class CRYPTOPP_NO_VTABLE AlgorithmImpl : public BASE
{
public :
static std : : string CRYPTOPP_API StaticAlgorithmName ( ) { return ALGORITHM_INFO : : StaticAlgorithmName ( ) ; }
std : : string AlgorithmName ( ) const { return ALGORITHM_INFO : : StaticAlgorithmName ( ) ; }
} ;
2015-11-18 20:32:28 +00:00
//! \class InvalidKeyLength
//! \brief Exception thrown when an invalid key length is encountered
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL InvalidKeyLength : public InvalidArgument
{
public :
explicit InvalidKeyLength ( const std : : string & algorithm , size_t length ) : InvalidArgument ( algorithm + " : " + IntToString ( length ) + " is not a valid key length " ) { }
} ;
2015-11-18 20:32:28 +00:00
//! \class InvalidRounds
//! \brief Exception thrown when an invalid number of rounds is encountered
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL InvalidRounds : public InvalidArgument
{
public :
explicit InvalidRounds ( const std : : string & algorithm , unsigned int rounds ) : InvalidArgument ( algorithm + " : " + IntToString ( rounds ) + " is not a valid number of rounds " ) { }
} ;
2016-04-17 07:06:28 +00:00
//! \class InvalidPersonalizationLength
//! \brief Exception thrown when an invalid personalization string length is encountered
class CRYPTOPP_DLL InvalidPersonalizationLength : public InvalidArgument
{
public :
explicit InvalidPersonalizationLength ( const std : : string & algorithm , size_t length ) : InvalidArgument ( algorithm + " : " + IntToString ( length ) + " is not a valid salt length " ) { }
} ;
//! \class InvalidSaltLength
//! \brief Exception thrown when an invalid salt length is encountered
class CRYPTOPP_DLL InvalidSaltLength : public InvalidArgument
{
public :
explicit InvalidSaltLength ( const std : : string & algorithm , size_t length ) : InvalidArgument ( algorithm + " : " + IntToString ( length ) + " is not a valid salt length " ) { }
} ;
2015-11-05 06:59:46 +00:00
// *****************************
2015-11-18 20:32:28 +00:00
//! \class Bufferless
//! \brief Base class for bufferless filters
//! \tparam T the class or type
2015-11-05 06:59:46 +00:00
template < class T >
class CRYPTOPP_NO_VTABLE Bufferless : public T
{
public :
bool IsolatedFlush ( bool hardFlush , bool blocking )
{ CRYPTOPP_UNUSED ( hardFlush ) ; CRYPTOPP_UNUSED ( blocking ) ; return false ; }
} ;
2015-11-18 20:32:28 +00:00
//! \class Unflushable
//! \brief Base class for unflushable filters
//! \tparam T the class or type
2015-11-05 06:59:46 +00:00
template < class T >
class CRYPTOPP_NO_VTABLE Unflushable : public T
{
public :
bool Flush ( bool completeFlush , int propagation = - 1 , bool blocking = true )
{ return ChannelFlush ( DEFAULT_CHANNEL , completeFlush , propagation , blocking ) ; }
bool IsolatedFlush ( bool hardFlush , bool blocking )
{ CRYPTOPP_UNUSED ( hardFlush ) ; CRYPTOPP_UNUSED ( blocking ) ; assert ( false ) ; return false ; }
bool ChannelFlush ( const std : : string & channel , bool hardFlush , int propagation = - 1 , bool blocking = true )
{
if ( hardFlush & & ! InputBufferIsEmpty ( ) )
throw CannotFlush ( " Unflushable<T>: this object has buffered input that cannot be flushed " ) ;
2016-09-10 08:57:48 +00:00
else
2015-11-05 06:59:46 +00:00
{
BufferedTransformation * attached = this - > AttachedTransformation ( ) ;
return attached & & propagation ? attached - > ChannelFlush ( channel , hardFlush , propagation - 1 , blocking ) : false ;
}
}
protected :
virtual bool InputBufferIsEmpty ( ) const { return false ; }
} ;
2015-11-18 20:32:28 +00:00
//! \class InputRejecting
//! \brief Base class for input rejecting filters
//! \tparam T the class or type
//! \details T should be a BufferedTransformation derived class
2015-11-05 06:59:46 +00:00
template < class T >
class CRYPTOPP_NO_VTABLE InputRejecting : public T
{
public :
struct InputRejected : public NotImplemented
{ InputRejected ( ) : NotImplemented ( " BufferedTransformation: this object doesn't allow input " ) { } } ;
2015-11-18 20:32:28 +00:00
//! \name INPUT
//@{
//! \brief Input a byte array for processing
//! \param inString the byte array 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
//! \throws InputRejected
//! \returns the number of bytes that remain in the block (i.e., bytes not processed)
//! \details Internally, the default implmentation throws InputRejected.
size_t Put2 ( const byte * inString , size_t length , int messageEnd , bool blocking )
{ CRYPTOPP_UNUSED ( inString ) ; CRYPTOPP_UNUSED ( length ) ; CRYPTOPP_UNUSED ( messageEnd ) ; CRYPTOPP_UNUSED ( blocking ) ; throw InputRejected ( ) ; }
//@}
//! \name SIGNALS
//@{
2015-11-05 06:59:46 +00:00
bool IsolatedFlush ( bool hardFlush , bool blocking )
{ CRYPTOPP_UNUSED ( hardFlush ) ; CRYPTOPP_UNUSED ( blocking ) ; return false ; }
bool IsolatedMessageSeriesEnd ( bool blocking )
{ CRYPTOPP_UNUSED ( blocking ) ; throw InputRejected ( ) ; }
2015-11-18 20:32:28 +00:00
size_t ChannelPut2 ( const std : : string & channel , const byte * inString , size_t length , int messageEnd , bool blocking )
{ CRYPTOPP_UNUSED ( channel ) ; CRYPTOPP_UNUSED ( inString ) ; CRYPTOPP_UNUSED ( length ) ; CRYPTOPP_UNUSED ( messageEnd ) ; CRYPTOPP_UNUSED ( blocking ) ; throw InputRejected ( ) ; }
2015-11-05 06:59:46 +00:00
bool ChannelMessageSeriesEnd ( const std : : string & channel , int messageEnd , bool blocking )
{ CRYPTOPP_UNUSED ( channel ) ; CRYPTOPP_UNUSED ( messageEnd ) ; CRYPTOPP_UNUSED ( blocking ) ; throw InputRejected ( ) ; }
2015-11-18 20:32:28 +00:00
//@}
2015-11-05 06:59:46 +00:00
} ;
2015-11-18 20:32:28 +00:00
//! \class CustomFlushPropagation
//! \brief Provides interface for custom flush signals
//! \tparam T the class or type
//! \details T should be a BufferedTransformation derived class
2015-11-05 06:59:46 +00:00
template < class T >
class CRYPTOPP_NO_VTABLE CustomFlushPropagation : public T
{
public :
2015-11-18 20:32:28 +00:00
//! \name SIGNALS
//@{
2015-11-05 06:59:46 +00:00
virtual bool Flush ( bool hardFlush , int propagation = - 1 , bool blocking = true ) = 0 ;
2015-11-18 20:32:28 +00:00
//@}
2015-11-05 06:59:46 +00:00
private :
bool IsolatedFlush ( bool hardFlush , bool blocking )
{ CRYPTOPP_UNUSED ( hardFlush ) ; CRYPTOPP_UNUSED ( blocking ) ; assert ( false ) ; return false ; }
} ;
2015-11-18 20:32:28 +00:00
//! \class CustomSignalPropagation
//! \brief Provides interface for initialization of derived filters
//! \tparam T the class or type
//! \details T should be a BufferedTransformation derived class
2015-11-05 06:59:46 +00:00
template < class T >
class CRYPTOPP_NO_VTABLE CustomSignalPropagation : public CustomFlushPropagation < T >
{
public :
virtual void Initialize ( const NameValuePairs & parameters = g_nullNameValuePairs , int propagation = - 1 ) = 0 ;
private :
void IsolatedInitialize ( const NameValuePairs & parameters )
{ CRYPTOPP_UNUSED ( parameters ) ; assert ( false ) ; }
} ;
2015-11-18 20:32:28 +00:00
//! \class Multichannel
//! \brief Provides multiple channels support for custom flush signal processing
//! \tparam T the class or type
//! \details T should be a BufferedTransformation derived class
2015-11-05 06:59:46 +00:00
template < class T >
class CRYPTOPP_NO_VTABLE Multichannel : public CustomFlushPropagation < T >
{
public :
bool Flush ( bool hardFlush , int propagation = - 1 , bool blocking = true )
{ return this - > ChannelFlush ( DEFAULT_CHANNEL , hardFlush , propagation , blocking ) ; }
bool MessageSeriesEnd ( int propagation = - 1 , bool blocking = true )
{ return this - > ChannelMessageSeriesEnd ( DEFAULT_CHANNEL , propagation , blocking ) ; }
byte * CreatePutSpace ( size_t & size )
{ return this - > ChannelCreatePutSpace ( DEFAULT_CHANNEL , size ) ; }
2015-11-18 20:32:28 +00:00
size_t Put2 ( const byte * inString , size_t length , int messageEnd , bool blocking )
{ return this - > ChannelPut2 ( DEFAULT_CHANNEL , inString , length , messageEnd , blocking ) ; }
2015-11-05 06:59:46 +00:00
size_t PutModifiable2 ( byte * inString , size_t length , int messageEnd , bool blocking )
{ return this - > ChannelPutModifiable2 ( DEFAULT_CHANNEL , inString , length , messageEnd , blocking ) ; }
// void ChannelMessageSeriesEnd(const std::string &channel, int propagation=-1)
// {PropagateMessageSeriesEnd(propagation, channel);}
byte * ChannelCreatePutSpace ( const std : : string & channel , size_t & size )
{ CRYPTOPP_UNUSED ( channel ) ; size = 0 ; return NULL ; }
bool ChannelPutModifiable ( const std : : string & channel , byte * inString , size_t length )
{ this - > ChannelPut ( channel , inString , length ) ; return false ; }
virtual size_t ChannelPut2 ( const std : : string & channel , const byte * begin , size_t length , int messageEnd , bool blocking ) = 0 ;
size_t ChannelPutModifiable2 ( const std : : string & channel , byte * begin , size_t length , int messageEnd , bool blocking )
{ return ChannelPut2 ( channel , begin , length , messageEnd , blocking ) ; }
virtual bool ChannelFlush ( const std : : string & channel , bool hardFlush , int propagation = - 1 , bool blocking = true ) = 0 ;
} ;
2015-11-18 20:32:28 +00:00
//! \class AutoSignaling
//! \brief Provides auto signaling support
//! \tparam T the class or type
//! \details T should be a BufferedTransformation derived class
2015-11-05 06:59:46 +00:00
template < class T >
class CRYPTOPP_NO_VTABLE AutoSignaling : public T
{
public :
AutoSignaling ( int propagation = - 1 ) : m_autoSignalPropagation ( propagation ) { }
void SetAutoSignalPropagation ( int propagation )
{ m_autoSignalPropagation = propagation ; }
int GetAutoSignalPropagation ( ) const
{ return m_autoSignalPropagation ; }
private :
int m_autoSignalPropagation ;
} ;
2015-11-18 20:32:28 +00:00
//! \class Store
//! \brief Acts as a Source for pre-existing, static data
//! \tparam T the class or type
//! \details A BufferedTransformation that only contains pre-existing data as "output"
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL CRYPTOPP_NO_VTABLE Store : public AutoSignaling < InputRejecting < BufferedTransformation > >
{
public :
2015-12-17 00:39:29 +00:00
//! \brief Construct a Store
2015-11-05 06:59:46 +00:00
Store ( ) : m_messageEnd ( false ) { }
void IsolatedInitialize ( const NameValuePairs & parameters )
{
m_messageEnd = false ;
StoreInitialize ( parameters ) ;
}
unsigned int NumberOfMessages ( ) const { return m_messageEnd ? 0 : 1 ; }
bool GetNextMessage ( ) ;
unsigned int CopyMessagesTo ( BufferedTransformation & target , unsigned int count = UINT_MAX , const std : : string & channel = DEFAULT_CHANNEL ) const ;
protected :
virtual void StoreInitialize ( const NameValuePairs & parameters ) = 0 ;
bool m_messageEnd ;
} ;
2015-11-18 20:32:28 +00:00
//! \class Sink
//! \brief Implementation of BufferedTransformation's attachment interface
//! \details Sink 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 A Sink doesnot produce any retrievable output.
//! \details See the discussion of BufferedTransformation in cryptlib.h for
//! more details.
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL CRYPTOPP_NO_VTABLE Sink : public BufferedTransformation
{
public :
size_t TransferTo2 ( BufferedTransformation & target , lword & transferBytes , const std : : string & channel = DEFAULT_CHANNEL , bool blocking = true )
{ CRYPTOPP_UNUSED ( target ) ; CRYPTOPP_UNUSED ( transferBytes ) ; CRYPTOPP_UNUSED ( channel ) ; CRYPTOPP_UNUSED ( blocking ) ; transferBytes = 0 ; return 0 ; }
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 ) ; return 0 ; }
} ;
2015-11-18 20:32:28 +00:00
//! \class BitBucket
//! \brief Acts as an input discarding Filter or Sink
//! \tparam T the class or type
//! \details The BitBucket discards all input and returns 0 to the caller
//! to indicate all data was processed.
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL BitBucket : public Bufferless < Sink >
{
public :
std : : string AlgorithmName ( ) const { return " BitBucket " ; }
void IsolatedInitialize ( const NameValuePairs & params )
{ CRYPTOPP_UNUSED ( params ) ; }
2015-11-18 20:32:28 +00:00
size_t Put2 ( const byte * inString , size_t length , int messageEnd , bool blocking )
{ CRYPTOPP_UNUSED ( inString ) ; CRYPTOPP_UNUSED ( length ) ; CRYPTOPP_UNUSED ( messageEnd ) ; CRYPTOPP_UNUSED ( blocking ) ; return 0 ; }
2015-11-05 06:59:46 +00:00
} ;
NAMESPACE_END
# if CRYPTOPP_MSC_VERSION
# pragma warning(pop)
# endif
# endif