mirror of
				https://git.suyu.dev/suyu/suyu.git
				synced 2025-10-25 11:56:42 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			2307 lines
		
	
	
		
			93 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			2307 lines
		
	
	
		
			93 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
| //  Boost CRC library crc.hpp header file  -----------------------------------//
 | |
| 
 | |
| //  Copyright 2001, 2004, 2011 Daryle Walker.
 | |
| //  Distributed under the Boost Software License, Version 1.0.  (See the
 | |
| //  accompanying file LICENSE_1_0.txt or a copy at
 | |
| //  <http://www.boost.org/LICENSE_1_0.txt>.)
 | |
| 
 | |
| //  See <http://www.boost.org/libs/crc/> for the library's home page.
 | |
| 
 | |
| /** \file
 | |
|     \brief  A collection of function templates and class templates that compute
 | |
|       various forms of Cyclic Redundancy Codes (CRCs).
 | |
| 
 | |
|     \author  Daryle Walker
 | |
| 
 | |
|     \version  1.5
 | |
| 
 | |
|     \copyright  Boost Software License, version 1.0
 | |
| 
 | |
|     Contains the declarations (and definitions) of various kinds of CRC
 | |
|     computation functions, function object types, and encapsulated policy types.
 | |
| 
 | |
|     \warning  The sample CRC-computer types were just checked against the
 | |
|       <a href="http://regregex.bbcmicro.net/crc-catalogue.htm">Catalogue of
 | |
|       parametrised CRC algorithms</a>.  New type aliases were added where I got
 | |
|       a standard wrong.  However, the mistaken <code>typedef</code>s are still
 | |
|       there for backwards compatibility.
 | |
|     \note  There are references to the <i>Rocksoft™ Model CRC
 | |
|       Algorithm</i>, as described within \"A Painless Guide to CRC Error
 | |
|       Detection Algorithms,\" linked from \"<a
 | |
|       href="http://www.ross.net/crc/crcpaper.html">CRC: A Paper On CRCs</a>\" by
 | |
|       Ross Williams.  It will be abbreviated \"RMCA\" in other documentation
 | |
|       blocks.
 | |
|  */
 | |
| 
 | |
| #ifndef BOOST_CRC_HPP
 | |
| #define BOOST_CRC_HPP
 | |
| 
 | |
| #include <boost/array.hpp>           // for boost::array
 | |
| #include <boost/config.hpp>          // for BOOST_STATIC_CONSTANT, etc.
 | |
| #include <boost/cstdint.hpp>         // for UINTMAX_C, boost::uintmax_t
 | |
| #include <boost/integer.hpp>         // for boost::uint_t
 | |
| #include <boost/type_traits/conditional.hpp>
 | |
| #include <boost/type_traits/integral_constant.hpp>
 | |
| 
 | |
| #include <climits>  // for CHAR_BIT, etc.
 | |
| #include <cstddef>  // for std::size_t
 | |
| 
 | |
| #include <boost/limits.hpp>  // for std::numeric_limits
 | |
| 
 | |
| 
 | |
| // The type of CRC parameters that can go in a template should be related
 | |
| // on the CRC's bit count.  This macro expresses that type in a compact
 | |
| // form, but also allows an alternate type for compilers that don't support
 | |
| // dependent types (in template value-parameters).
 | |
| #if !(defined(BOOST_NO_DEPENDENT_TYPES_IN_TEMPLATE_VALUE_PARAMETERS))
 | |
| #define BOOST_CRC_PARM_TYPE  typename ::boost::uint_t<Bits>::fast
 | |
| #else
 | |
| #define BOOST_CRC_PARM_TYPE  unsigned long
 | |
| #endif
 | |
| 
 | |
| namespace boost
 | |
| {
 | |
| 
 | |
| 
 | |
| //  Forward declarations  ----------------------------------------------------//
 | |
| 
 | |
| //! Bit-wise CRC computer
 | |
| template < std::size_t Bits >
 | |
|     class crc_basic;
 | |
| 
 | |
| //! Table-driven CRC computer, usable as a function object
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly = 0u,
 | |
|            BOOST_CRC_PARM_TYPE InitRem = 0u,
 | |
|            BOOST_CRC_PARM_TYPE FinalXor = 0u, bool ReflectIn = false,
 | |
|            bool ReflectRem = false >
 | |
|     class crc_optimal;
 | |
| 
 | |
| //! Compute the (unaugmented) CRC of a memory block
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
|     typename uint_t<Bits>::fast  crc( void const *buffer,
 | |
|      std::size_t byte_count);
 | |
| 
 | |
| //! Compute the CRC of a memory block, with any augmentation provided by user
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly >
 | |
|     typename uint_t<Bits>::fast  augmented_crc( void const *buffer,
 | |
|      std::size_t byte_count,
 | |
|      typename uint_t<Bits>::fast initial_remainder = 0u);
 | |
| 
 | |
| //! Computation type for ARC|CRC-16|CRC-IBM|CRC-16/ARC|CRC-16/LHA standard
 | |
| typedef crc_optimal<16, 0x8005, 0, 0, true, true>         crc_16_type;
 | |
| //! Computation type for CRC-16/CCITT-FALSE standard
 | |
| typedef crc_optimal<16, 0x1021, 0xFFFF, 0, false, false>  crc_ccitt_false_t;
 | |
| //! Computation type for the CRC mistakenly called the CCITT standard
 | |
| typedef crc_ccitt_false_t                                 crc_ccitt_type;
 | |
| //! Computation type for the actual
 | |
| //! KERMIT|CRC-16/CCITT|CRC-16/CCITT-TRUE|CRC-CCITT standard
 | |
| typedef crc_optimal<16, 0x1021, 0, 0, true, true>         crc_ccitt_true_t;
 | |
| //! Computation type that I mistakenly called the XMODEM standard; it inverts
 | |
| //! both reflection parameters and reflects the truncated divisor (Don't use?!)
 | |
| typedef crc_optimal<16, 0x8408, 0, 0, true, true>         crc_xmodem_type;
 | |
| //! Computation type for the actual XMODEM|ZMODEM|CRC-16/ACORN standard
 | |
| typedef crc_optimal<16, 0x1021, 0, 0, false, false>       crc_xmodem_t;
 | |
| 
 | |
| //! Computation type for CRC-32|CRC-32/ADCCP|PKZIP standard
 | |
| typedef crc_optimal<32, 0x04C11DB7, 0xFFFFFFFF, 0xFFFFFFFF, true, true>
 | |
|   crc_32_type;
 | |
| 
 | |
| 
 | |
| //  Forward declarations for implementation detail stuff  --------------------//
 | |
| //  (Just for the stuff that will be needed for the next two sections)
 | |
| 
 | |
| //! \cond
 | |
| namespace detail
 | |
| {
 | |
|     //! Mix-in class to add a possibly-reflecting member function
 | |
|     template < int BitLength, bool DoIt, int Id = 0 >
 | |
|         class possible_reflector;
 | |
| 
 | |
|     //! Mix-in class for byte-fed, table-driven CRC algorithms
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial, bool Reflect,
 | |
|      int Id = 0 >
 | |
|     class crc_driver;
 | |
| 
 | |
| }  // namespace detail
 | |
| //! \endcond
 | |
| 
 | |
| 
 | |
| //  Simple cyclic redundancy code (CRC) class declaration  -------------------//
 | |
| 
 | |
| /** Objects of this type compute the CRC checksum of submitted data, where said
 | |
|     data can be entered piecemeal through several different kinds of groupings.
 | |
|     Modulo-2 polynomial division steps are always performed bit-wise, without
 | |
|     the use of pre-computation tables.  Said division uses the altered
 | |
|     algorithm, so any data has to be unaugmented.
 | |
| 
 | |
|     \pre  0 \< \a Bits \<= \c std\::numeric_limits\<uintmax_t\>\::digits
 | |
| 
 | |
|     \tparam Bits  The order of the modulo-2 polynomial divisor.  (\e Width from
 | |
|       the RMCA)
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| class crc_basic
 | |
| {
 | |
| public:
 | |
|     // Type
 | |
|     /** \brief  The register type used for computations
 | |
| 
 | |
|         This type is used for CRC calculations and is the type for any returned
 | |
|         checksums and returned or submitted remainders, (truncated) divisors, or
 | |
|         XOR masks.  It is a built-in unsigned integer type.
 | |
|      */
 | |
|     typedef typename boost::uint_t<Bits>::fast  value_type;
 | |
| 
 | |
|     // Constant for the template parameter
 | |
|     //! A copy of \a Bits provided for meta-programming purposes
 | |
|     BOOST_STATIC_CONSTANT( std::size_t, bit_count = Bits );
 | |
| 
 | |
|     // Constructor (use the automatic copy-ctr, move-ctr, and dtr)
 | |
|     //! Create a computer, separately listing each needed parameter
 | |
|     explicit  crc_basic( value_type truncated_polynomial,
 | |
|                value_type initial_remainder = 0, value_type final_xor_value = 0,
 | |
|                bool reflect_input = false, bool reflect_remainder = false );
 | |
| 
 | |
|     // Internal Operations
 | |
|     //! Return the (truncated) polynomial divisor
 | |
|     value_type  get_truncated_polynominal() const;
 | |
|     //! Return what the polynomial remainder was set to during construction
 | |
|     value_type  get_initial_remainder() const;
 | |
|     //! Return the XOR-mask used during output processing
 | |
|     value_type  get_final_xor_value() const;
 | |
|     //! Check if input-bytes will be reflected before processing
 | |
|     bool        get_reflect_input() const;
 | |
|     //! Check if the remainder will be reflected during output processing
 | |
|     bool        get_reflect_remainder() const;
 | |
| 
 | |
|     //! Return the remainder based from already-processed bits
 | |
|     value_type  get_interim_remainder() const;
 | |
|     //! Change the interim remainder to a new value
 | |
|     void        reset( value_type new_rem );
 | |
|     //! Change the interim remainder back to the initial value
 | |
|     void        reset();
 | |
| 
 | |
|     // External Operations
 | |
|     //! Submit a single bit for input processing
 | |
|     void  process_bit( bool bit );
 | |
|     //! Submit the lowest \a bit_length bits of a byte for input processing
 | |
|     void  process_bits( unsigned char bits, std::size_t bit_length );
 | |
|     //! Submit a single byte for input processing
 | |
|     void  process_byte( unsigned char byte );
 | |
|     //! Submit a memory block for input processing, iterator-pair style
 | |
|     void  process_block( void const *bytes_begin, void const *bytes_end );
 | |
|     //! Submit a memory block for input processing, pointer-and-size style
 | |
|     void  process_bytes( void const *buffer, std::size_t byte_count );
 | |
| 
 | |
|     //! Return the checksum of the already-processed bits
 | |
|     value_type  checksum() const;
 | |
| 
 | |
| private:
 | |
|     // Member data
 | |
|     value_type  rem_;
 | |
|     value_type  poly_, init_, final_;  // non-const to allow assignability
 | |
|     bool        rft_in_, rft_out_;     // non-const to allow assignability
 | |
| 
 | |
| };  // boost::crc_basic
 | |
| 
 | |
| 
 | |
| //  Optimized cyclic redundancy code (CRC) class declaration  ----------------//
 | |
| 
 | |
| /** Objects of this type compute the CRC checksum of submitted data, where said
 | |
|     data can be entered piecemeal through several different kinds of groupings.
 | |
|     Modulo-2 polynomial division steps are performed byte-wise, aided by the use
 | |
|     of pre-computation tables.  Said division uses the altered algorithm, so any
 | |
|     data has to be unaugmented.
 | |
| 
 | |
|     \pre  0 \< \a Bits \<= \c std\::numeric_limits\<uintmax_t\>\::digits
 | |
| 
 | |
|     \tparam Bits  The order of the modulo-2 polynomial divisor.  (\e Width from
 | |
|       the RMCA)
 | |
|     \tparam TruncPoly  The lowest coefficients of the divisor polynomial.  The
 | |
|       highest-order coefficient is omitted and always assumed to be 1.  Defaults
 | |
|       to \c 0, i.e. the only non-zero term is the implicit one for
 | |
|       x<sup><var>Bits</var></sup>.  (\e Poly from the RMCA)
 | |
|     \tparam InitRem  The (unaugmented) initial state of the polynomial
 | |
|       remainder.  Defaults to \c 0 if omitted.  (\e Init from the RMCA)
 | |
|     \tparam FinalXor  The (XOR) bit-mask to be applied to the output remainder,
 | |
|       after possible reflection but before returning.  Defaults to \c 0 (i.e. no
 | |
|       bit changes) if omitted.  (\e XorOut from the RMCA)
 | |
|     \tparam ReflectIn  If \c true, input bytes are read lowest-order bit first,
 | |
|       otherwise highest-order bit first.  Defaults to \c false if omitted.
 | |
|       (\e RefIn from the RMCA)
 | |
|     \tparam ReflectRem  If \c true, the output remainder is reflected before the
 | |
|       XOR-mask.  Defaults to \c false if omitted.  (\e RefOut from the RMCA)
 | |
| 
 | |
|     \todo  Get rid of the default value for \a TruncPoly.  Choosing a divisor is
 | |
|       an important decision with many factors, so a default is never useful,
 | |
|       especially a bad one.
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| class crc_optimal
 | |
| {
 | |
| public:
 | |
|     // Type
 | |
|     //! \copydoc  boost::crc_basic::value_type
 | |
|     typedef typename boost::uint_t<Bits>::fast  value_type;
 | |
| 
 | |
|     // Constants for the template parameters
 | |
|     //! \copydoc  boost::crc_basic::bit_count
 | |
|     BOOST_STATIC_CONSTANT( std::size_t, bit_count = Bits );
 | |
|     //! A copy of \a TruncPoly provided for meta-programming purposes
 | |
|     BOOST_STATIC_CONSTANT( value_type, truncated_polynominal = TruncPoly );
 | |
|     //! A copy of \a InitRem provided for meta-programming purposes
 | |
|     BOOST_STATIC_CONSTANT( value_type, initial_remainder = InitRem );
 | |
|     //! A copy of \a FinalXor provided for meta-programming purposes
 | |
|     BOOST_STATIC_CONSTANT( value_type, final_xor_value = FinalXor );
 | |
|     //! A copy of \a ReflectIn provided for meta-programming purposes
 | |
|     BOOST_STATIC_CONSTANT( bool, reflect_input = ReflectIn );
 | |
|     //! A copy of \a ReflectRem provided for meta-programming purposes
 | |
|     BOOST_STATIC_CONSTANT( bool, reflect_remainder = ReflectRem );
 | |
| 
 | |
|     // Constructor (use the automatic copy-ctr, move-ctr, and dtr)
 | |
|     //! Create a computer, giving an initial remainder if desired
 | |
|     explicit  crc_optimal( value_type init_rem = initial_remainder );
 | |
| 
 | |
|     // Internal Operations
 | |
|     //! \copybrief  boost::crc_basic::get_truncated_polynominal
 | |
|     value_type  get_truncated_polynominal() const;
 | |
|     //! \copybrief  boost::crc_basic::get_initial_remainder
 | |
|     value_type  get_initial_remainder() const;
 | |
|     //! \copybrief  boost::crc_basic::get_final_xor_value
 | |
|     value_type  get_final_xor_value() const;
 | |
|     //! \copybrief  boost::crc_basic::get_reflect_input
 | |
|     bool        get_reflect_input() const;
 | |
|     //! \copybrief  boost::crc_basic::get_reflect_remainder
 | |
|     bool        get_reflect_remainder() const;
 | |
| 
 | |
|     //! \copybrief  boost::crc_basic::get_interim_remainder
 | |
|     value_type  get_interim_remainder() const;
 | |
|     //! Change the interim remainder to either a given value or the initial one
 | |
|     void        reset( value_type new_rem = initial_remainder );
 | |
| 
 | |
|     // External Operations
 | |
|     //! \copybrief  boost::crc_basic::process_byte
 | |
|     void  process_byte( unsigned char byte );
 | |
|     //! \copybrief  boost::crc_basic::process_block
 | |
|     void  process_block( void const *bytes_begin, void const *bytes_end );
 | |
|     //! \copybrief  boost::crc_basic::process_bytes
 | |
|     void  process_bytes( void const *buffer, std::size_t byte_count );
 | |
| 
 | |
|     //! \copybrief  boost::crc_basic::checksum
 | |
|     value_type  checksum() const;
 | |
| 
 | |
|     // Operators
 | |
|     //! Submit a single byte for input processing, suitable for the STL
 | |
|     void        operator ()( unsigned char byte );
 | |
|     //! Return the checksum of the already-processed bits, suitable for the STL
 | |
|     value_type  operator ()() const;
 | |
| 
 | |
| private:
 | |
|     // Implementation types
 | |
|     // (Processing for reflected input gives reflected remainders, so you only
 | |
|     // have to apply output-reflection if Reflect-Remainder doesn't match
 | |
|     // Reflect-Input.)
 | |
|     typedef detail::possible_reflector<Bits, ReflectIn>     reflect_i_type;
 | |
|     typedef detail::crc_driver<Bits, TruncPoly, ReflectIn>  crc_table_type;
 | |
|     typedef detail::possible_reflector<Bits, ReflectRem != ReflectIn>
 | |
|       reflect_o_type;
 | |
| 
 | |
|     // Member data
 | |
|     value_type  rem_;
 | |
| 
 | |
| };  // boost::crc_optimal
 | |
| 
 | |
| 
 | |
| //  Implementation detail stuff  ---------------------------------------------//
 | |
| 
 | |
| //! \cond
 | |
| namespace detail
 | |
| {
 | |
|     /** \brief  Meta-programming integral constant for a single-bit bit-mask
 | |
| 
 | |
|         Generates a compile-time constant for a bit-mask that affects a single
 | |
|         bit.  The \c value will be 2<sup><var>BitIndex</var></sup>.  The \c type
 | |
|         will be the smallest built-in unsigned integer type that can contain the
 | |
|         value, unless there's a built-in type that the system can handle easier,
 | |
|         then the \c type will be smallest fast-handled unsigned integer type.
 | |
| 
 | |
|         \pre  0 \<= BitIndex \< \c std\::numeric_limits\<uintmax_t\>\::digits
 | |
| 
 | |
|         \tparam BitIndex  The place of the sole set bit.
 | |
|      */
 | |
|     template < int BitIndex >
 | |
|     struct high_bit_mask_c
 | |
|         : boost::integral_constant<typename boost::uint_t< BitIndex + 1 >::fast,
 | |
|            ( UINTMAX_C(1) << BitIndex )>
 | |
|     {};
 | |
| 
 | |
|     /** \brief  Meta-programming integral constant for a lowest-bits bit-mask
 | |
| 
 | |
|         Generates a compile-time constant for a bit-mask that affects the lowest
 | |
|         bits.  The \c value will be 2<sup><var>BitCount</var></sup> - 1.  The
 | |
|         \c type will be the smallest built-in unsigned integer type that can
 | |
|         contain the value, unless there's a built-in type that the system can
 | |
|         handle easier, then the \c type will be smallest fast-handled unsigned
 | |
|         integer type.
 | |
| 
 | |
|         \pre  0 \<= BitCount \<= \c std\::numeric_limits\<uintmax_t\>\::digits
 | |
| 
 | |
|         \tparam BitCount  The number of lowest-placed bits set.
 | |
|      */
 | |
|     template < int BitCount >
 | |
|     struct low_bits_mask_c
 | |
|         : boost::integral_constant<typename boost::uint_t< BitCount >::fast, (
 | |
|            BitCount ? (( (( UINTMAX_C(1) << (BitCount - 1) ) - 1u) << 1 ) |
 | |
|            UINTMAX_C( 1 )) : 0u )>
 | |
|     {};
 | |
| 
 | |
|     /** \brief  Reflects the bits of a number
 | |
| 
 | |
|         Reverses the order of the given number of bits within a value.  For
 | |
|         instance, if the given reflect count is 5, then the bit values for the
 | |
|         16- and 1-place will switch and the 8- and 2-place will switch, leaving
 | |
|         the other bits alone.  (The 4-place bit is in the middle, so it wouldn't
 | |
|         change.)
 | |
| 
 | |
|         \pre  \a Unsigned is a built-in unsigned integer type
 | |
|         \pre  0 \< word_length \<= \c std\::numeric_limits\<Unsigned\>\::digits
 | |
| 
 | |
|         \tparam Unsigned  The type of \a x.
 | |
| 
 | |
|         \param x  The value to be (partially) reflected.
 | |
|         \param word_length  The number of low-order bits to reflect.  Defaults
 | |
|           to the total number of value bits in \a Unsigned.
 | |
| 
 | |
|         \return  The (partially) reflected value.
 | |
| 
 | |
|         \todo  Check if this is the fastest way.
 | |
|      */
 | |
|     template < typename Unsigned >
 | |
|     Unsigned  reflect_unsigned( Unsigned x, int word_length
 | |
|      = std::numeric_limits<Unsigned>::digits )
 | |
|     {
 | |
|         for ( Unsigned  l = 1u, h = l << (word_length - 1) ; h > l ; h >>= 1, l
 | |
|          <<= 1 )
 | |
|         {
 | |
|             Unsigned const  m = h | l, t = x & m;
 | |
| 
 | |
|             if ( (t == h) || (t == l) )
 | |
|                 x ^= m;
 | |
|         }
 | |
| 
 | |
|         return x;
 | |
|     }
 | |
| 
 | |
|     /** \brief  Make a byte-to-byte-reflection map
 | |
| 
 | |
|         Creates a mapping array so the results can be cached.  Uses
 | |
|         #reflect_unsigned to generate the element values.
 | |
| 
 | |
|         \return  An array <var>a</var> such that, for a given byte value
 | |
|           <var>i</var>, <code><var>a</var>[ <var>i</var> ]</code> resolves to
 | |
|           the reflected value of <var>i</var>.
 | |
|      */
 | |
|     boost::array< unsigned char, (UINTMAX_C( 1 ) << CHAR_BIT) >
 | |
|     inline make_byte_reflection_table()
 | |
|     {
 | |
|         boost::array<unsigned char, ( UINTMAX_C(1) << CHAR_BIT )>  result;
 | |
|         unsigned char                                              i = 0u;
 | |
| 
 | |
|         do
 | |
|             result[ i ] = reflect_unsigned( i );
 | |
|         while ( ++i );
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /** \brief  Reflects the bits of a single byte
 | |
| 
 | |
|         Reverses the order of all the bits within a value.  For instance, the
 | |
|         bit values for the 2<sup><code>CHAR_BIT</code> - 1</sup>- and 1-place
 | |
|         will switch and the 2<sup><code>CHAR_BIT</code> - 2</sup>- and 2-place
 | |
|         will switch, etc.
 | |
| 
 | |
|         \param x  The byte value to be reflected.
 | |
| 
 | |
|         \return  The reflected value.
 | |
| 
 | |
|         \note  Since this could be the most common type of reflection, and the
 | |
|           number of states is relatively small, the implementation pre-computes
 | |
|           and uses a table of all the results.
 | |
|      */
 | |
|     inline unsigned char  reflect_byte( unsigned char x )
 | |
|     {
 | |
|         static  boost::array<unsigned char, ( UINTMAX_C(1) << CHAR_BIT )> const
 | |
|           table = make_byte_reflection_table();
 | |
| 
 | |
|         return table[ x ];
 | |
|     }
 | |
| 
 | |
|     /** \brief  Reflects some bits within a single byte
 | |
| 
 | |
|         Like #reflect_unsigned, except it takes advantage of any (long-term)
 | |
|         speed gains #reflect_byte may bring.
 | |
| 
 | |
|         \pre  0 \< \a word_length \<= \c CHAR_BIT
 | |
| 
 | |
|         \param x  The value to be (partially) reflected.
 | |
|         \param word_length  The number of low-order bits to reflect.
 | |
| 
 | |
|         \return  The (partially) reflected value.
 | |
|      */
 | |
|     inline  unsigned char  reflect_sub_byte( unsigned char x, int word_length )
 | |
|     { return reflect_byte(x) >> (CHAR_BIT - word_length); }
 | |
| 
 | |
|     /** \brief  Possibly reflects the bits of a number
 | |
| 
 | |
|         Reverses the order of the given number of bits within a value.  For
 | |
|         instance, if the given reflect count is 5, then the bit values for the
 | |
|         16- and 1-place will switch and the 8- and 2-place will switch, leaving
 | |
|         the other bits alone.  (The 4-place bit is in the middle, so it wouldn't
 | |
|         change.)  This variant function allows the reflection be controlled by
 | |
|         an extra parameter, in case the decision to use reflection is made at
 | |
|         run-time.
 | |
| 
 | |
|         \pre  \a Unsigned is a built-in unsigned integer type
 | |
|         \pre  0 \< word_length \<= \c std\::numeric_limits\<Unsigned\>\::digits
 | |
| 
 | |
|         \tparam Unsigned  The type of \a x.
 | |
| 
 | |
|         \param x  The value to be (partially) reflected.
 | |
|         \param reflect  Controls whether \a x is actually reflected (\c true) or
 | |
|           left alone (\c false).
 | |
|         \param word_length  The number of low-order bits to reflect.  Defaults
 | |
|           to the total number of value bits in \a Unsigned.
 | |
| 
 | |
|         \return  The possibly (partially) reflected value.
 | |
|      */
 | |
|     template < typename Unsigned >
 | |
|     inline
 | |
|     Unsigned  reflect_optionally( Unsigned x, bool reflect, int word_length
 | |
|      = std::numeric_limits<Unsigned>::digits )
 | |
|     { return reflect ? reflect_unsigned(x, word_length) : x; }
 | |
| 
 | |
|     /** \brief  Possibly reflects the bits of a single byte
 | |
| 
 | |
|         Uses #reflect_byte (if \a reflect is \c true).
 | |
| 
 | |
|         \param x  The byte value to be (possibly) reflected.
 | |
|         \param reflect  Whether (\c true) or not (\c false) \a x is reflected.
 | |
| 
 | |
|         \return  <code><var>reflect</var> ? reflect_byte(<var>x</var>) :
 | |
|           <var>x</var></code>
 | |
|      */
 | |
|     inline
 | |
|     unsigned char  reflect_byte_optionally( unsigned char x, bool reflect )
 | |
|     { return reflect ? reflect_byte(x) : x; }
 | |
| 
 | |
|     /** \brief  Update a CRC remainder by several bits, assuming a non-augmented
 | |
|           message
 | |
| 
 | |
|         Performs several steps of division required by the CRC algorithm, giving
 | |
|         a new remainder polynomial based on the divisor polynomial and the
 | |
|         synthesized dividend polynomial (from the old remainder and the
 | |
|         newly-provided input).  The computations assume that the CRC is directly
 | |
|         exposed from the remainder, without any zero-valued bits augmented to
 | |
|         the message bits.
 | |
| 
 | |
|         \pre  \a Register and \a Word are both built-in unsigned integer types
 | |
|         \pre  0 \< \a register_length \<= std\::numeric_limits\<\a Register\>
 | |
|           \::digits
 | |
|         \pre  0 \< \a word_length \<= std\::numeric_limits\<\a Word\>\::digits
 | |
| 
 | |
|         \tparam Register  The type used for representing the remainder and
 | |
|           divisor modulo-2 polynomials.  The bit at <code>2<sup>i</sup></code>
 | |
|           is used as the coefficient of <i>x<sup>i</sup></i>.
 | |
|         \tparam Word  The type used for storing the incoming terms of the
 | |
|           dividend modulo-2 polynomial.  The bit at <code>2<sup>i</sup></code>
 | |
|           is used as the coefficient of <i>x<sup>i</sup></i> when \a reflect is
 | |
|           \c false, and the coefficient of <i>x<sup><var>word_length</var> - 1 -
 | |
|           i</sup></i> otherwise.
 | |
| 
 | |
|         \param[in]     register_length  The number of significant low-order bits
 | |
|           to be used from \a Register values.  It is the order of the modulo-2
 | |
|           polynomial remainder and one less than the divisor's order.
 | |
|         \param[in,out] remainder  The upper part of the dividend polynomial
 | |
|           before division, and the remainder polynomial after.
 | |
|         \param[in]     new_dividend_bits  The coefficients for the next
 | |
|           \a word_length lowest terms of the dividend polynomial.
 | |
|         \param[in]     truncated_divisor  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|         \param[in]     word_length  The number of lowest-order bits to read from
 | |
|           \a new_dividend_bits.
 | |
|         \param[in]     reflect  If \c false, read from the highest-order marked
 | |
|           bit from \a new_dividend_bits and go down, as normal.  Otherwise,
 | |
|           proceed from the lowest-order bit and go up.
 | |
| 
 | |
|         \note  This routine performs a modulo-2 polynomial division variant.
 | |
|           The exclusive-or operations are applied in a different order, since
 | |
|           that kind of operation is commutative and associative.  It also
 | |
|           assumes that the zero-valued augment string was applied before this
 | |
|           step, which means that the updated remainder can be directly used as
 | |
|           the final CRC.
 | |
|      */
 | |
|     template < typename Register, typename Word >
 | |
|     void  crc_modulo_word_update( int register_length, Register &remainder, Word
 | |
|      new_dividend_bits, Register truncated_divisor, int word_length, bool
 | |
|      reflect )
 | |
|     {
 | |
|         // Create this masking constant outside the loop.
 | |
|         Register const  high_bit_mask = UINTMAX_C(1) << (register_length - 1);
 | |
| 
 | |
|         // The natural reading order for division is highest digit/bit first.
 | |
|         // The "reflect" parameter switches this.  However, building a bit mask
 | |
|         // for the lowest bit is the easiest....
 | |
|         new_dividend_bits = reflect_optionally( new_dividend_bits, !reflect,
 | |
|          word_length );
 | |
| 
 | |
|         // Perform modulo-2 division for each new dividend input bit
 | |
|         for ( int  i = word_length ; i ; --i, new_dividend_bits >>= 1 )
 | |
|         {
 | |
|             // compare the new bit with the remainder's highest
 | |
|             remainder ^= ( new_dividend_bits & 1u ) ? high_bit_mask : 0u;
 | |
| 
 | |
|             // perform modulo-2 division
 | |
|             bool const  quotient = remainder & high_bit_mask;
 | |
| 
 | |
|             remainder <<= 1;
 | |
|             remainder ^= quotient ? truncated_divisor : 0u;
 | |
| 
 | |
|             // The quotient isn't used for anything, so don't keep it.
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /** \brief  Update a CRC remainder by a single bit, assuming a non-augmented
 | |
|           message
 | |
| 
 | |
|         Performs the next step of division required by the CRC algorithm, giving
 | |
|         a new remainder polynomial based on the divisor polynomial and the
 | |
|         synthesized dividend polynomial (from the old remainder and the
 | |
|         newly-provided input).  The computations assume that the CRC is directly
 | |
|         exposed from the remainder, without any zero-valued bits augmented to
 | |
|         the message bits.
 | |
| 
 | |
|         \pre  \a Register is a built-in unsigned integer type
 | |
|         \pre  0 \< \a register_length \<= std\::numeric_limits\<\a Register\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam Register  The type used for representing the remainder and
 | |
|           divisor modulo-2 polynomials.  The bit at <code>2<sup>i</sup></code>
 | |
|           is used as the coefficient of <i>x<sup>i</sup></i>.
 | |
| 
 | |
|         \param[in]     register_length  The number of significant low-order bits
 | |
|           to be used from \a Register values.  It is the order of the modulo-2
 | |
|           polynomial remainder and one less than the divisor's order.
 | |
|         \param[in,out] remainder  The upper part of the dividend polynomial
 | |
|           before division, and the remainder polynomial after.
 | |
|         \param[in]     new_dividend_bit  The coefficient for the constant term
 | |
|           of the dividend polynomial.
 | |
|         \param[in]     truncated_divisor  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
| 
 | |
|         \note  This routine performs a modulo-2 polynomial division variant.
 | |
|           The exclusive-or operations are applied in a different order, since
 | |
|           that kind of operation is commutative and associative.  It also
 | |
|           assumes that the zero-valued augment string was applied before this
 | |
|           step, which means that the updated remainder can be directly used as
 | |
|           the final CRC.
 | |
|      */
 | |
|     template < typename Register >
 | |
|     inline  void  crc_modulo_update( int register_length, Register &remainder,
 | |
|      bool new_dividend_bit, Register truncated_divisor )
 | |
|     {
 | |
|         crc_modulo_word_update( register_length, remainder,
 | |
|          static_cast<unsigned>(new_dividend_bit), truncated_divisor, 1, false );
 | |
|     }
 | |
| 
 | |
|     /** \brief  Update a CRC remainder by several bits, assuming an augmented
 | |
|           message
 | |
| 
 | |
|         Performs several steps of division required by the CRC algorithm, giving
 | |
|         a new remainder polynomial based on the divisor polynomial and the
 | |
|         synthesized dividend polynomial (from the old remainder and the
 | |
|         newly-provided input).  The computations assume that a zero-valued
 | |
|         string of bits will be appended to the message before extracting the
 | |
|         CRC.
 | |
| 
 | |
|         \pre  \a Register and \a Word are both built-in unsigned integer types
 | |
|         \pre  0 \< \a register_length \<= std\::numeric_limits\<\a Register\>
 | |
|           \::digits
 | |
|         \pre  0 \< \a word_length \<= std\::numeric_limits\<\a Word\>\::digits
 | |
| 
 | |
|         \tparam Register  The type used for representing the remainder and
 | |
|           divisor modulo-2 polynomials.  The bit at <code>2<sup>i</sup></code>
 | |
|           is used as the coefficient of <i>x<sup>i</sup></i>.
 | |
|         \tparam Word  The type used for storing the incoming terms of the
 | |
|           dividend modulo-2 polynomial.  The bit at <code>2<sup>i</sup></code>
 | |
|           is used as the coefficient of <i>x<sup>i</sup></i> when \a reflect is
 | |
|           \c false, and the coefficient of <i>x<sup><var>word_length</var> - 1 -
 | |
|           i</sup></i> otherwise.
 | |
| 
 | |
|         \param[in]     register_length  The number of significant low-order bits
 | |
|           to be used from \a Register values.  It is the order of the modulo-2
 | |
|           polynomial remainder and one less than the divisor's order.
 | |
|         \param[in,out] remainder  The upper part of the dividend polynomial
 | |
|           before division, and the remainder polynomial after.
 | |
|         \param[in]     new_dividend_bits  The coefficients for the next
 | |
|           \a word_length lowest terms of the dividend polynomial.
 | |
|         \param[in]     truncated_divisor  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|         \param[in]     word_length  The number of lowest-order bits to read from
 | |
|           \a new_dividend_bits.
 | |
|         \param[in]     reflect  If \c false, read from the highest-order marked
 | |
|           bit from \a new_dividend_bits and go down, as normal.  Otherwise,
 | |
|           proceed from the lowest-order bit and go up.
 | |
| 
 | |
|         \note  This routine performs straight-forward modulo-2 polynomial
 | |
|           division.  It assumes that an augment string will be processed at the
 | |
|           end of the message bits before doing CRC analysis.
 | |
|         \todo  Use this function somewhere so I can test it.
 | |
|      */
 | |
|     template < typename Register, typename Word >
 | |
|     void  augmented_crc_modulo_word_update( int register_length, Register
 | |
|      &remainder, Word new_dividend_bits, Register truncated_divisor, int
 | |
|      word_length, bool reflect )
 | |
|     {
 | |
|         // Create this masking constant outside the loop.
 | |
|         Register const  high_bit_mask = UINTMAX_C(1) << (register_length - 1);
 | |
| 
 | |
|         // The natural reading order for division is highest digit/bit first.
 | |
|         // The "reflect" parameter switches this.  However, building a bit mask
 | |
|         // for the lowest bit is the easiest....
 | |
|         new_dividend_bits = reflect_optionally( new_dividend_bits, not reflect,
 | |
|          word_length );
 | |
| 
 | |
|         // Perform modulo-2 division for each new dividend input bit
 | |
|         for ( int  i = word_length ; i ; --i, new_dividend_bits >>= 1 )
 | |
|         {
 | |
|             bool const  quotient = remainder & high_bit_mask;
 | |
| 
 | |
|             remainder <<= 1;
 | |
|             remainder |= new_dividend_bits & 1u;
 | |
|             remainder ^= quotient ? truncated_divisor : 0u;
 | |
| 
 | |
|             // The quotient isn't used for anything, so don't keep it.
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /** \brief  Update a CRC remainder by a single bit, assuming an augmented
 | |
|           message
 | |
| 
 | |
|         Performs the next step of division required by the CRC algorithm, giving
 | |
|         a new remainder polynomial based on the divisor polynomial and the
 | |
|         synthesized dividend polynomial (from the old remainder and the
 | |
|         newly-provided input).  The computations assume that a zero-valued
 | |
|         string of bits will be appended to the message before extracting the
 | |
|         CRC.
 | |
| 
 | |
|         \pre  \a Register is a built-in unsigned integer type
 | |
|         \pre  0 \< \a register_length \<= std\::numeric_limits\<\a Register\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam Register  The type used for representing the remainder and
 | |
|           divisor modulo-2 polynomials.  The bit at <code>2<sup>i</sup></code>
 | |
|           is used as the coefficient of <i>x<sup>i</sup></i>.
 | |
| 
 | |
|         \param[in]     register_length  The number of significant low-order bits
 | |
|           to be used from \a Register values.  It is the order of the modulo-2
 | |
|           polynomial remainder and one less than the divisor's order.
 | |
|         \param[in,out] remainder  The upper part of the dividend polynomial
 | |
|           before division, and the remainder polynomial after.
 | |
|         \param[in]     new_dividend_bit  The coefficient for the constant term
 | |
|           of the dividend polynomial.
 | |
|         \param[in]     truncated_divisor  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
| 
 | |
|         \note  This routine performs straight-forward modulo-2 polynomial
 | |
|           division.  It assumes that an augment string will be processed at the
 | |
|           end of the message bits before doing CRC analysis.
 | |
|         \todo  Use this function somewhere so I can test it.
 | |
|      */
 | |
|     template < typename Register >
 | |
|     inline  void  augmented_crc_modulo_update( int register_length, Register
 | |
|      &remainder, bool new_dividend_bit, Register truncated_divisor )
 | |
|     {
 | |
|         augmented_crc_modulo_word_update( register_length, remainder,
 | |
|          static_cast<unsigned>(new_dividend_bit), truncated_divisor, 1, false );
 | |
|     }
 | |
| 
 | |
|     /** \brief  A mix-in class that returns its argument
 | |
| 
 | |
|         This class template makes a function object that returns its argument
 | |
|         as-is.  It's one case for #possible_reflector.
 | |
| 
 | |
|         \pre  0 \< \a BitLength \<= \c std\::numeric_limits\<uintmax_t\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam BitLength  How many significant bits arguments have.
 | |
|      */
 | |
|     template < int BitLength >
 | |
|     class non_reflector
 | |
|     {
 | |
|     public:
 | |
|         /** \brief  The type to check for specialization
 | |
| 
 | |
|             This is a Boost integral constant indicating that this class
 | |
|             does not reflect its input values.
 | |
|          */
 | |
|         typedef boost::false_type                 is_reflecting_type;
 | |
|         /** \brief  The type to check for register bit length
 | |
| 
 | |
|             This is a Boost integral constant indicating how many
 | |
|             significant bits won't actually be reflected.
 | |
|          */
 | |
|         typedef boost::integral_constant< int, BitLength >      width_c;
 | |
|         /** \brief  The type of (not-)reflected values
 | |
| 
 | |
|             This type is the input and output type for the (possible-)
 | |
|             reflection function, which does nothing here.
 | |
|          */
 | |
|         typedef typename boost::uint_t< BitLength >::fast  value_type;
 | |
| 
 | |
|         /** \brief  Does nothing
 | |
| 
 | |
|             Returns the given value, not reflecting any part of it.
 | |
| 
 | |
|             \param x  The value to not be reflected.
 | |
| 
 | |
|             \return  \a x
 | |
|          */
 | |
|         inline  static  value_type  reflect_q( value_type x )
 | |
|         { return x; }
 | |
|     };
 | |
| 
 | |
|     /** \brief  A mix-in class that reflects (the lower part of) its argument,
 | |
|           generally for types larger than a byte
 | |
| 
 | |
|         This class template makes a function object that returns its argument
 | |
|         after reflecting its lower-order bits.  It's one sub-case for
 | |
|         #possible_reflector.
 | |
| 
 | |
|         \pre  \c CHAR_BIT \< \a BitLength \<= \c std\::numeric_limits\<uintmax_t
 | |
|           \>\::digits
 | |
| 
 | |
|         \tparam BitLength  How many significant bits arguments have.
 | |
|      */
 | |
|     template < int BitLength >
 | |
|     class super_byte_reflector
 | |
|     {
 | |
|     public:
 | |
|         /** \brief  The type to check for specialization
 | |
| 
 | |
|             This is a Boost integral constant indicating that this class
 | |
|             does reflect its input values.
 | |
|          */
 | |
|         typedef boost::true_type                  is_reflecting_type;
 | |
|         /** \brief  The type to check for register bit length
 | |
| 
 | |
|             This is a Boost integral constant indicating how many
 | |
|             significant bits will be reflected.
 | |
|          */
 | |
|         typedef boost::integral_constant< int, BitLength >      width_c;
 | |
|         /** \brief  The type of reflected values
 | |
| 
 | |
|             This is both the input and output type for the reflection function.
 | |
|          */
 | |
|         typedef typename boost::uint_t< BitLength >::fast  value_type;
 | |
| 
 | |
|         /** \brief  Reflect (part of) the given value
 | |
| 
 | |
|             Reverses the order of the given number of bits within a value,
 | |
|             using #reflect_unsigned.
 | |
| 
 | |
|             \param x  The value to be (partially) reflected.
 | |
| 
 | |
|             \return  ( <var>x</var> &
 | |
|               ~(2<sup><var>width_c</var>\::value</sup> - 1) ) | REFLECT(
 | |
|               <var>x</var> & (2<sup><var>width_c</var>\::value</sup> -
 | |
|               1) )
 | |
|          */
 | |
|         inline  static  value_type  reflect_q( value_type x )
 | |
|         { return reflect_unsigned(x, width_c::value); }
 | |
|     };
 | |
| 
 | |
|     /** \brief  A mix-in class that reflects (the lower part of) its argument,
 | |
|           generally for bytes
 | |
| 
 | |
|         This class template makes a function object that returns its argument
 | |
|         after reflecting its lower-order bits.  It's one sub-case for
 | |
|         #possible_reflector.
 | |
| 
 | |
|         \pre  0 \< \a BitLength \<= \c CHAR_BIT
 | |
| 
 | |
|         \tparam BitLength  How many significant bits arguments have.
 | |
|      */
 | |
|     template < int BitLength >
 | |
|     class sub_type_reflector
 | |
|     {
 | |
|     public:
 | |
|         /** \brief  The type to check for specialization
 | |
| 
 | |
|             This is a Boost integral constant indicating that this class
 | |
|             does reflect its input values.
 | |
|          */
 | |
|         typedef boost::true_type              is_reflecting_type;
 | |
|         /** \brief  The type to check for register bit length
 | |
| 
 | |
|             This is a Boost integral constant indicating how many
 | |
|             significant bits will be reflected.
 | |
|          */
 | |
|         typedef boost::integral_constant< int, BitLength >  width_c;
 | |
|         /** \brief  The type of reflected values
 | |
| 
 | |
|             This is both the input and output type for the reflection function.
 | |
|          */
 | |
|         typedef unsigned char                          value_type;
 | |
| 
 | |
|         /** \brief  Reflect (part of) the given value
 | |
| 
 | |
|             Reverses the order of the given number of bits within a value,
 | |
|             using #reflect_sub_byte.
 | |
| 
 | |
|             \param x  The value to be (partially) reflected.
 | |
| 
 | |
|             \return  ( <var>x</var> &
 | |
|               ~(2<sup><var>width_c</var>\::value</sup> - 1) ) | REFLECT(
 | |
|               <var>x</var> & (2<sup><var>width_c</var>\::value</sup> -
 | |
|               1) )
 | |
|          */
 | |
|         inline  static  value_type  reflect_q( value_type x )
 | |
|         { return reflect_sub_byte(x, width_c::value); }
 | |
|     };
 | |
| 
 | |
|     /** \brief  A mix-in class that reflects (the lower part of) its argument
 | |
| 
 | |
|         This class template makes a function object that returns its argument
 | |
|         after reflecting its lower-order bits.  It's one case for
 | |
|         #possible_reflector.
 | |
| 
 | |
|         \pre  0 \< \a BitLength \<= \c std\::numeric_limits\<uintmax_t\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam BitLength  How many significant bits arguments have.
 | |
|      */
 | |
|     template < int BitLength >
 | |
|     class reflector
 | |
|         : public boost::conditional< (BitLength > CHAR_BIT),
 | |
|           super_byte_reflector<BitLength>, sub_type_reflector<BitLength> >::type
 | |
|     { };
 | |
| 
 | |
|     /** This class template adds a member function #reflect_q that will
 | |
|         conditionally reflect its first argument, controlled by a compile-time
 | |
|         parameter.
 | |
| 
 | |
|         \pre  0 \< \a BitLength \<= \c std\::numeric_limits\<uintmax_t\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam BitLength  How many significant bits arguments have.
 | |
|         \tparam DoIt  \c true if #reflect_q will reflect, \c false if it should
 | |
|           return its argument unchanged.
 | |
|         \tparam Id  An extra differentiator if multiple copies of this class
 | |
|           template are mixed-in as base classes.  Defaults to 0 if omitted.
 | |
|      */
 | |
|     template < int BitLength, bool DoIt, int Id >
 | |
|     class possible_reflector
 | |
|         : public boost::conditional< DoIt, reflector<BitLength>,
 | |
|           non_reflector<BitLength> >::type
 | |
|     {
 | |
|     public:
 | |
|         /** \brief  The type to check for ID
 | |
| 
 | |
|             This is a Boost integral constant indicating what ID number this
 | |
|             instantiation used.
 | |
|          */
 | |
|         typedef boost::integral_constant<int, Id>  id_type;
 | |
|     };
 | |
| 
 | |
|     /** \brief  Find the composite remainder update effect from a fixed bit
 | |
|           sequence, for each potential sequence combination.
 | |
| 
 | |
|         For each value between 0 and 2<sup><var>SubOrder</var></sup> - 1,
 | |
|         computes the XOR mask corresponding to the composite effect they update
 | |
|         the incoming remainder, which is the upper part of the dividend that
 | |
|         gets (partially) pushed out of its register by the incoming value's
 | |
|         bits.  The composite value merges the \"partial products\" from each bit
 | |
|         of the value being updated individually.
 | |
| 
 | |
|         \pre  \a Register is a built-in unsigned integer type
 | |
|         \pre  0 \< \a SubOrder \<= \a register_length \<=
 | |
|           std\::numeric_limits\<\a Register\>\::digits
 | |
| 
 | |
|         \tparam SubOrder  The number of low-order significant bits of the trial
 | |
|           new dividends.
 | |
|         \tparam Register  The type used for representing the remainder and
 | |
|           divisor modulo-2 polynomials.  The bit at <code>2<sup>i</sup></code>
 | |
|           is used as the coefficient of <i>x<sup>i</sup></i>.
 | |
| 
 | |
|         \param[in] register_length  The number of significant low-order bits
 | |
|           to be used from \a Register values.  It is the order of the modulo-2
 | |
|           polynomial remainder and one less than the divisor's order.
 | |
|         \param[in] truncated_divisor  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|         \param[in] reflect  If \c false, read from the highest-order marked
 | |
|           bit from a new dividend's bits and go down, as normal.  Otherwise,
 | |
|           proceed from the lowest-order bit and go up.
 | |
| 
 | |
|         \return  An array such that the element at index <var>i</var> is the
 | |
|           composite effect XOR mask for value <var>i</var>.
 | |
| 
 | |
|         \note  This routine performs a modulo-2 polynomial division variant.
 | |
|           The exclusive-or operations are applied in a different order, since
 | |
|           that kind of operation is commutative and associative.  It also
 | |
|           assumes that the zero-valued augment string was applied before this
 | |
|           step, which means that the updated remainder can be directly used as
 | |
|           the final CRC.
 | |
|         \todo  Check that using the unaugmented-CRC division routines give the
 | |
|           same composite mask table as using augmented-CRC routines.
 | |
|      */
 | |
|     template < int SubOrder, typename Register >
 | |
|     boost::array< Register, (UINTMAX_C( 1 ) << SubOrder) >
 | |
|     make_partial_xor_products_table( int register_length, Register
 | |
|      truncated_divisor, bool reflect )
 | |
|     {
 | |
|         boost::array<Register, ( UINTMAX_C(1) << SubOrder )>  result;
 | |
| 
 | |
|         // Loop over every possible dividend value
 | |
|         for ( typename boost::uint_t<SubOrder + 1>::fast  dividend = 0u;
 | |
|          dividend < result.size() ; ++dividend )
 | |
|         {
 | |
|             Register  remainder = 0u;
 | |
| 
 | |
|             crc_modulo_word_update( register_length, remainder, dividend,
 | |
|              truncated_divisor, SubOrder, false );
 | |
|             result[ reflect_optionally(dividend, reflect, SubOrder) ] =
 | |
|              reflect_optionally( remainder, reflect, register_length );
 | |
|         }
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /** \brief  A mix-in class for the table of table-driven CRC algorithms
 | |
| 
 | |
|         Encapsulates the parameters need to make a global (technically,
 | |
|         class-static) table usuable in CRC algorithms, and generates said
 | |
|         table.
 | |
| 
 | |
|         \pre  0 \< \a SubOrder \<= Order \<=
 | |
|           std\::numeric_limits\<uintmax_t\>\::digits
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam SubOrder  The number of low-order significant bits of the trial
 | |
|           new dividends.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|         \tparam Reflect  If \c false, read from the highest-order marked
 | |
|           bit from a new dividend's bits and go down, as normal.  Otherwise,
 | |
|           proceed from the lowest-order bit and go up.
 | |
|      */
 | |
|     template < int Order, int SubOrder, boost::uintmax_t TruncatedPolynomial,
 | |
|      bool Reflect >
 | |
|     class crc_table_t
 | |
|     {
 | |
|     public:
 | |
|         /** \brief  The type to check for register bit length
 | |
| 
 | |
|             This is a Boost integral constant indicating how many
 | |
|             significant bits are in the remainder and (truncated) divisor.
 | |
|          */
 | |
|         typedef boost::integral_constant< int, Order >              width_c;
 | |
|         /** \brief  The type to check for index-unit bit length
 | |
| 
 | |
|             This is a Boost integral constant indicating how many
 | |
|             significant bits are in the trial new dividends.
 | |
|          */
 | |
|         typedef boost::integral_constant< int, SubOrder >      unit_width_c;
 | |
|         /** \brief  The type of registers
 | |
| 
 | |
|             This is the output type for the partial-product map.
 | |
|          */
 | |
|         typedef typename boost::uint_t< Order >::fast          value_type;
 | |
|         /** \brief  The type to check the divisor
 | |
| 
 | |
|             This is a Boost integral constant representing the (truncated)
 | |
|             divisor.
 | |
|          */
 | |
|         typedef boost::integral_constant< value_type, TruncatedPolynomial >
 | |
|           poly_c;
 | |
|         /** \brief  The type to check for reflection
 | |
| 
 | |
|             This is a Boost integral constant representing whether input
 | |
|             units should be read in reverse order.
 | |
|          */
 | |
|         typedef boost::integral_constant< bool, Reflect >           refin_c;
 | |
|         /** \brief  The type to check for map size
 | |
| 
 | |
|             This is a Boost integral constant representing the number of
 | |
|             elements in the partial-product map, based on the unit size.
 | |
|          */
 | |
|         typedef high_bit_mask_c< SubOrder >                  table_size_c;
 | |
|         /** \brief  The type of the unit TO partial-product map
 | |
| 
 | |
|             This is the array type that takes units as the index and said unit's
 | |
|             composite partial-product mask as the element.
 | |
|          */
 | |
|         typedef boost::array<value_type, table_size_c::value>  array_type;
 | |
|         /** \brief  Create a global array for the mapping table
 | |
| 
 | |
|             Creates an instance of a partial-product array with this class's
 | |
|             parameters.
 | |
| 
 | |
|             \return  A reference to a immutable array giving the partial-product
 | |
|               update XOR map for each potential sub-unit value.
 | |
|          */
 | |
|         static  array_type const &  get_table()
 | |
|         {
 | |
|             static  array_type const  table =
 | |
|              make_partial_xor_products_table<unit_width_c::value>(
 | |
|              width_c::value, poly_c::value, refin_c::value );
 | |
| 
 | |
|             return table;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     /** \brief  A mix-in class that handles direct (i.e. non-reflected) byte-fed
 | |
|           table-driven CRC algorithms
 | |
| 
 | |
|         This class template adds member functions #augmented_crc_update and
 | |
|         #crc_update to update remainders from new input bytes.  The bytes aren't
 | |
|         reflected before processing.
 | |
| 
 | |
|         \pre  \c CHAR_BIT \<= \a Order \<= \c std\::numeric_limits\<uintmax_t\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|      */
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial >
 | |
|     class direct_byte_table_driven_crcs
 | |
|         : public crc_table_t<Order, CHAR_BIT, TruncatedPolynomial, false>
 | |
|     {
 | |
|         typedef crc_table_t<Order, CHAR_BIT, TruncatedPolynomial, false>
 | |
|           base_type;
 | |
| 
 | |
|     public:
 | |
|         typedef typename base_type::value_type  value_type;
 | |
|         typedef typename base_type::array_type  array_type;
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             augmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated remainder
 | |
|          */
 | |
|         static  value_type  augmented_crc_update( value_type remainder, unsigned
 | |
|          char const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Locates the merged partial product based on the leading byte
 | |
|                 unsigned char const  index = ( remainder >> (Order - CHAR_BIT) )
 | |
|                  & UCHAR_MAX;
 | |
| 
 | |
|                 // Complete the multi-bit modulo-2 polynomial division
 | |
|                 remainder <<= CHAR_BIT;
 | |
|                 remainder |= *new_dividend_bytes++;
 | |
|                 remainder ^= table.elems[ index ];
 | |
|             }
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             unaugmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated remainder
 | |
|          */
 | |
|         static  value_type  crc_update( value_type remainder, unsigned char
 | |
|          const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Locates the merged partial product based on comparing the
 | |
|                 // leading and incoming bytes
 | |
|                 unsigned char const  index = ( (remainder >> ( Order - CHAR_BIT
 | |
|                  )) & UCHAR_MAX ) ^ *new_dividend_bytes++;
 | |
| 
 | |
|                 // Complete the multi-bit altered modulo-2 polynomial division
 | |
|                 remainder <<= CHAR_BIT;
 | |
|                 remainder ^= table.elems[ index ];
 | |
|             }
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     /** \brief  A mix-in class that handles reflected byte-fed, table-driven CRC
 | |
|           algorithms
 | |
| 
 | |
|         This class template adds member functions #augmented_crc_update and
 | |
|         #crc_update to update remainders from new input bytes.  The bytes are
 | |
|         reflected before processing.
 | |
| 
 | |
|         \pre  \c CHAR_BIT \<= \a Order \<= \c std\::numeric_limits\<uintmax_t\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|      */
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial >
 | |
|     class reflected_byte_table_driven_crcs
 | |
|         : public crc_table_t<Order, CHAR_BIT, TruncatedPolynomial, true>
 | |
|     {
 | |
|         typedef crc_table_t<Order, CHAR_BIT, TruncatedPolynomial, true>
 | |
|           base_type;
 | |
| 
 | |
|     public:
 | |
|         typedef typename base_type::value_type  value_type;
 | |
|         typedef typename base_type::array_type  array_type;
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             reflecting augmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder; since the bytes are
 | |
|               being reflected, this remainder also has to be reflected
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated, reflected remainder
 | |
|          */
 | |
|         static  value_type  augmented_crc_update( value_type remainder, unsigned
 | |
|          char const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Locates the merged partial product based on the leading byte
 | |
|                 // (which is at the low-order end for reflected remainders)
 | |
|                 unsigned char const  index = remainder & UCHAR_MAX;
 | |
| 
 | |
|                 // Complete the multi-bit reflected modulo-2 polynomial division
 | |
|                 remainder >>= CHAR_BIT;
 | |
|                 remainder |= static_cast<value_type>( *new_dividend_bytes++ )
 | |
|                  << ( Order - CHAR_BIT );
 | |
|                 remainder ^= table.elems[ index ];
 | |
|             }
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             reflected unaugmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder; since the bytes are
 | |
|               being reflected, this remainder also has to be reflected
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated, reflected remainder
 | |
|          */
 | |
|         static  value_type  crc_update( value_type remainder, unsigned char
 | |
|          const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Locates the merged partial product based on comparing the
 | |
|                 // leading and incoming bytes
 | |
|                 unsigned char const  index = ( remainder & UCHAR_MAX ) ^
 | |
|                  *new_dividend_bytes++;
 | |
| 
 | |
|                 // Complete the multi-bit reflected altered modulo-2 polynomial
 | |
|                 // division
 | |
|                 remainder >>= CHAR_BIT;
 | |
|                 remainder ^= table.elems[ index ];
 | |
|             }
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     /** \brief  Mix-in class for byte-fed, table-driven CRC algorithms with
 | |
|           parameter values at least a byte in width
 | |
| 
 | |
|         This class template adds member functions #augmented_crc_update and
 | |
|         #crc_update to update remainders from new input bytes.  The bytes may be
 | |
|         reflected before processing, controlled by a compile-time parameter.
 | |
| 
 | |
|         \pre  \c CHAR_BIT \<= \a Order \<= \c std\::numeric_limits\<uintmax_t\>
 | |
|           \::digits
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|         \tparam Reflect  If \c false, read from the highest-order bit from a new
 | |
|           input byte and go down, as normal.  Otherwise, proceed from the
 | |
|           lowest-order bit and go up.
 | |
|      */
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial, bool Reflect >
 | |
|     class byte_table_driven_crcs
 | |
|         : public boost::conditional< Reflect,
 | |
|           reflected_byte_table_driven_crcs<Order, TruncatedPolynomial>,
 | |
|           direct_byte_table_driven_crcs<Order, TruncatedPolynomial> >::type
 | |
|     { };
 | |
| 
 | |
|     /** \brief  A mix-in class that handles direct (i.e. non-reflected) byte-fed
 | |
|           CRC algorithms for sub-byte parameters
 | |
| 
 | |
|         This class template adds member functions #augmented_crc_update and
 | |
|         #crc_update to update remainders from new input bytes.  The bytes aren't
 | |
|         reflected before processing.
 | |
| 
 | |
|         \pre  0 \< \a Order \< \c CHAR_BIT
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|      */
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial >
 | |
|     class direct_sub_byte_crcs
 | |
|         : public crc_table_t<Order, Order, TruncatedPolynomial, false>
 | |
|     {
 | |
|         typedef crc_table_t<Order, Order, TruncatedPolynomial, false>
 | |
|           base_type;
 | |
| 
 | |
|     public:
 | |
|         typedef typename base_type::width_c     width_c;
 | |
|         typedef typename base_type::value_type  value_type;
 | |
|         typedef typename base_type::poly_c       poly_c;
 | |
|         typedef typename base_type::array_type  array_type;
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             augmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated remainder
 | |
| 
 | |
|             \todo  Use this function somewhere so I can test it.
 | |
|          */
 | |
|         static  value_type  augmented_crc_update( value_type remainder, unsigned
 | |
|          char const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             //static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Without a table, process each byte explicitly
 | |
|                 augmented_crc_modulo_word_update( width_c::value, remainder,
 | |
|                  *new_dividend_bytes++, poly_c::value, CHAR_BIT, false );
 | |
|             }
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             unaugmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated remainder
 | |
|          */
 | |
|         static  value_type  crc_update( value_type remainder, unsigned char
 | |
|          const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             //static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Without a table, process each byte explicitly
 | |
|                 crc_modulo_word_update( width_c::value, remainder,
 | |
|                  *new_dividend_bytes++, poly_c::value, CHAR_BIT, false );
 | |
|             }
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     /** \brief  A mix-in class that handles reflected byte-fed, CRC algorithms
 | |
|           for sub-byte parameters
 | |
| 
 | |
|         This class template adds member functions #augmented_crc_update and
 | |
|         #crc_update to update remainders from new input bytes.  The bytes are
 | |
|         reflected before processing.
 | |
| 
 | |
|         \pre  0 \< \a Order \< \c CHAR_BIT
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|      */
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial >
 | |
|     class reflected_sub_byte_crcs
 | |
|         : public crc_table_t<Order, Order, TruncatedPolynomial, true>
 | |
|     {
 | |
|         typedef crc_table_t<Order, Order, TruncatedPolynomial, true>
 | |
|           base_type;
 | |
| 
 | |
|     public:
 | |
|         typedef typename base_type::width_c     width_c;
 | |
|         typedef typename base_type::value_type  value_type;
 | |
|         typedef typename base_type::poly_c       poly_c;
 | |
|         typedef typename base_type::array_type  array_type;
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             reflecting augmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder; since the bytes are
 | |
|               being reflected, this remainder also has to be reflected
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated, reflected remainder
 | |
| 
 | |
|             \todo  Use this function somewhere so I can test it.
 | |
|          */
 | |
|         static  value_type  augmented_crc_update( value_type remainder, unsigned
 | |
|          char const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             //static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             remainder = reflect_sub_byte( remainder, width_c::value );
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Without a table, process each byte explicitly
 | |
|                 augmented_crc_modulo_word_update( width_c::value, remainder,
 | |
|                  *new_dividend_bytes++, poly_c::value, CHAR_BIT, true );
 | |
|             }
 | |
|             remainder = reflect_sub_byte( remainder, width_c::value );
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
| 
 | |
|         /** \brief  Compute the updated remainder after reading some bytes
 | |
| 
 | |
|             The implementation reads from a table to speed-up applying
 | |
|             reflected unaugmented-CRC updates byte-wise.
 | |
| 
 | |
|             \param remainder  The pre-update remainder; since the bytes are
 | |
|               being reflected, this remainder also has to be reflected
 | |
|             \param new_dividend_bytes  The address where the new bytes start
 | |
|             \param new_dividend_byte_count  The number of new bytes to read
 | |
| 
 | |
|             \return  The updated, reflected remainder
 | |
|          */
 | |
|         static  value_type  crc_update( value_type remainder, unsigned char
 | |
|          const *new_dividend_bytes, std::size_t new_dividend_byte_count)
 | |
|         {
 | |
|             //static  array_type const &  table = base_type::get_table();
 | |
| 
 | |
|             remainder = reflect_sub_byte( remainder, width_c::value );
 | |
|             while ( new_dividend_byte_count-- )
 | |
|             {
 | |
|                 // Without a table, process each byte explicitly
 | |
|                 crc_modulo_word_update( width_c::value, remainder,
 | |
|                  *new_dividend_bytes++, poly_c::value, CHAR_BIT, true );
 | |
|             }
 | |
|             remainder = reflect_sub_byte( remainder, width_c::value );
 | |
| 
 | |
|             return remainder;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     /** \brief  Mix-in class for byte-fed, table-driven CRC algorithms with
 | |
|           sub-byte parameters
 | |
| 
 | |
|         This class template adds member functions #augmented_crc_update and
 | |
|         #crc_update to update remainders from new input bytes.  The bytes may be
 | |
|         reflected before processing, controlled by a compile-time parameter.
 | |
| 
 | |
|         \pre  0 \< \a Order \< \c CHAR_BIT
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|         \tparam Reflect  If \c false, read from the highest-order bit from a new
 | |
|           input byte and go down, as normal.  Otherwise, proceed from the
 | |
|           lowest-order bit and go up.
 | |
|      */
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial, bool Reflect >
 | |
|     class sub_byte_crcs
 | |
|         : public boost::conditional< Reflect,
 | |
|           reflected_sub_byte_crcs<Order, TruncatedPolynomial>,
 | |
|           direct_sub_byte_crcs<Order, TruncatedPolynomial> >::type
 | |
|     { };
 | |
| 
 | |
|     /** This class template adds member functions #augmented_crc_update and
 | |
|         #crc_update to update remainders from new input bytes.  The bytes may be
 | |
|         reflected before processing, controlled by a compile-time parameter.
 | |
| 
 | |
|         \pre  0 \< \a Order \<= \c std\::numeric_limits\<uintmax_t\>\::digits
 | |
| 
 | |
|         \tparam Order  The order of the modulo-2 polynomial remainder and one
 | |
|           less than the divisor's order.
 | |
|         \tparam TruncatedPolynomial  The lowest coefficients of the divisor
 | |
|           polynomial.  The highest-order coefficient is omitted and always
 | |
|           assumed to be 1.
 | |
|         \tparam Reflect  If \c false, read from the highest-order bit from a new
 | |
|           input byte and go down, as normal.  Otherwise, proceed from the
 | |
|           lowest-order bit and go up.
 | |
|         \tparam Id  An extra differentiator if multiple copies of this class
 | |
|           template are mixed-in as base classes.  Defaults to 0 if omitted.
 | |
|      */
 | |
|     template < int Order, boost::uintmax_t TruncatedPolynomial, bool Reflect,
 | |
|      int Id >
 | |
|     class crc_driver
 | |
|         : public boost::conditional< (Order < CHAR_BIT), sub_byte_crcs<Order,
 | |
|           TruncatedPolynomial, Reflect>, byte_table_driven_crcs<Order,
 | |
|           TruncatedPolynomial, Reflect> >::type
 | |
|     {
 | |
|     public:
 | |
|         /** \brief  The type to check for ID
 | |
| 
 | |
|             This is a Boost integral constant indicating what ID number this
 | |
|             instantiation used.
 | |
|          */
 | |
|         typedef boost::integral_constant<int, Id>  id_type;
 | |
|     };
 | |
| 
 | |
| 
 | |
| }  // namespace detail
 | |
| //! \endcond
 | |
| 
 | |
| 
 | |
| //  Simple CRC class function definitions  -----------------------------------//
 | |
| 
 | |
| /** Constructs a \c crc_basic object with at least the required parameters to a
 | |
|     particular CRC formula to be processed upon receiving input.
 | |
| 
 | |
|     \param[in] truncated_polynomial  The lowest coefficients of the divisor
 | |
|       polynomial.  The highest-order coefficient is omitted and always assumed
 | |
|       to be 1.  (\e Poly from the RMCA)
 | |
|     \param[in] initial_remainder  The (unaugmented) initial state of the
 | |
|       polynomial remainder.  Defaults to \c 0 if omitted.  (\e Init from the
 | |
|       RMCA)
 | |
|     \param[in] final_xor_value  The (XOR) bit-mask to be applied to the output
 | |
|       remainder, after possible reflection but before returning.  Defaults to
 | |
|       \c 0 (i.e. no bit changes) if omitted.  (\e XorOut from the RMCA)
 | |
|     \param[in] reflect_input  If \c true, input bytes are read lowest-order bit
 | |
|       first, otherwise highest-order bit first.  Defaults to \c false if
 | |
|       omitted.  (\e RefIn from the RMCA)
 | |
|     \param[in] reflect_remainder  If \c true, the output remainder is reflected
 | |
|       before the XOR-mask.  Defaults to \c false if omitted.  (\e RefOut from
 | |
|       the RMCA)
 | |
| 
 | |
|     \post  <code><var>truncated_polynomial</var> ==
 | |
|       this->get_truncated_polynominal()</code>
 | |
|     \post  <code><var>initial_remainder</var> ==
 | |
|       this->get_initial_remainder()</code>
 | |
|     \post  <code><var>final_xor_value</var> ==
 | |
|       this->get_final_xor_value()</code>
 | |
|     \post  <code><var>reflect_input</var> ==
 | |
|       this->get_reflect_input()</code>
 | |
|     \post  <code><var>reflect_remainder</var> ==
 | |
|       this->get_reflect_remainder()</code>
 | |
|     \post  <code><var>initial_remainder</var> ==
 | |
|       this->get_interim_remainder()</code>
 | |
|     \post  <code>(<var>reflect_remainder</var> ?
 | |
|       REFLECT(<var>initial_remainder</var>) : <var>initial_remainder</var>) ^
 | |
|       <var>final_xor_value</var> == this->checksum()</code>
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| crc_basic<Bits>::crc_basic
 | |
| (
 | |
|     value_type  truncated_polynomial,
 | |
|     value_type  initial_remainder,      // = 0
 | |
|     value_type  final_xor_value,        // = 0
 | |
|     bool        reflect_input,          // = false
 | |
|     bool        reflect_remainder       // = false
 | |
| )
 | |
|     : rem_( initial_remainder ), poly_( truncated_polynomial )
 | |
|     , init_( initial_remainder ), final_( final_xor_value )
 | |
|     , rft_in_( reflect_input ), rft_out_( reflect_remainder )
 | |
| {
 | |
| }
 | |
| 
 | |
| /** Returns a representation of the polynomial divisor.  The value of the
 | |
|     2<sup>i</sup> bit is the value of the coefficient of the polynomial's
 | |
|     x<sup>i</sup> term.  The omitted bit for x<sup>(#bit_count)</sup> term is
 | |
|     always 1.
 | |
| 
 | |
|     \return  The bit-packed list of coefficients.  If the bit-length of
 | |
|       #value_type exceeds #bit_count, the values of higher-placed bits should be
 | |
|       ignored (even any for x<sup>(#bit_count)</sup>) since they're unregulated.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| typename crc_basic<Bits>::value_type
 | |
| crc_basic<Bits>::get_truncated_polynominal
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return poly_;
 | |
| }
 | |
| 
 | |
| /** Returns a representation of the polynomial remainder before any input has
 | |
|     been submitted.  The value of the 2<sup>i</sup> bit is the value of the
 | |
|     coefficient of the polynomial's x<sup>i</sup> term.
 | |
| 
 | |
|     \return  The bit-packed list of coefficients.  If the bit-length of
 | |
|       #value_type exceeds #bit_count, the values of higher-placed bits should be
 | |
|       ignored since they're unregulated.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| typename crc_basic<Bits>::value_type
 | |
| crc_basic<Bits>::get_initial_remainder
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return init_;
 | |
| }
 | |
| 
 | |
| /** Returns the mask to be used during creation of a checksum.  The mask is used
 | |
|     for an exclusive-or (XOR) operation applied bit-wise to the interim
 | |
|     remainder representation (after any reflection, if #get_reflect_remainder()
 | |
|     returns \c true).
 | |
| 
 | |
|     \return  The bit-mask.  If the bit-length of #value_type exceeds #bit_count,
 | |
|       the values of higher-placed bits should be ignored since they're
 | |
|       unregulated.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| typename crc_basic<Bits>::value_type
 | |
| crc_basic<Bits>::get_final_xor_value
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return final_;
 | |
| }
 | |
| 
 | |
| /** Returns a whether or not a submitted byte will be \"reflected\" before it is
 | |
|     used to update the interim remainder.  Only the byte-wise operations
 | |
|     #process_byte, #process_block, and #process_bytes are affected.
 | |
| 
 | |
|     \retval true  Input bytes will be read starting from the lowest-order bit.
 | |
|     \retval false  Input bytes will be read starting from the highest-order bit.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| bool
 | |
| crc_basic<Bits>::get_reflect_input
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return rft_in_;
 | |
| }
 | |
| 
 | |
| /** Indicates if the interim remainder will be \"reflected\" before it is passed
 | |
|     to the XOR-mask stage when returning a checksum.
 | |
| 
 | |
|     \retval true  The interim remainder is reflected before further work.
 | |
|     \retval false  The interim remainder is applied to the XOR-mask as-is.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| bool
 | |
| crc_basic<Bits>::get_reflect_remainder
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return rft_out_;
 | |
| }
 | |
| 
 | |
| /** Returns a representation of the polynomial remainder after all the input
 | |
|     submissions since construction or the last #reset call.  The value of the
 | |
|     2<sup>i</sup> bit is the value of the coefficient of the polynomial's
 | |
|     x<sup>i</sup> term.  If CRC processing gets interrupted here, retain the
 | |
|     value returned, and use it to start up the next CRC computer where you left
 | |
|     off (with #reset(value_type) or construction).  The next computer has to
 | |
|     have its other parameters compatible with this computer.
 | |
| 
 | |
|     \return  The bit-packed list of coefficients.  If the bit-length of
 | |
|       #value_type exceeds #bit_count, the values of higher-placed bits should be
 | |
|       ignored since they're unregulated.  No output processing (reflection or
 | |
|       XOR mask) has been applied to the value.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| typename crc_basic<Bits>::value_type
 | |
| crc_basic<Bits>::get_interim_remainder
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return rem_ & detail::low_bits_mask_c<bit_count>::value;
 | |
| }
 | |
| 
 | |
| /** Changes the interim polynomial remainder to \a new_rem, purging any
 | |
|     influence previously submitted input has had.  The value of the
 | |
|     2<sup>i</sup> bit is the value of the coefficient of the polynomial's
 | |
|     x<sup>i</sup> term.
 | |
| 
 | |
|     \param[in] new_rem  The (unaugmented) state of the polynomial remainder
 | |
|       starting from this point, with no output processing applied.
 | |
| 
 | |
|     \post  <code><var>new_rem</var> == this->get_interim_remainder()</code>
 | |
|     \post  <code>((this->get_reflect_remainder() ?
 | |
|       REFLECT(<var>new_rem</var>) : <var>new_rem</var>) ^
 | |
|       this->get_final_xor_value()) == this->checksum()</code>
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| void
 | |
| crc_basic<Bits>::reset
 | |
| (
 | |
|     value_type  new_rem
 | |
| )
 | |
| {
 | |
|     rem_ = new_rem;
 | |
| }
 | |
| 
 | |
| /** Changes the interim polynomial remainder to the initial remainder given
 | |
|     during construction, purging any influence previously submitted input has
 | |
|     had.  The value of the 2<sup>i</sup> bit is the value of the coefficient of
 | |
|     the polynomial's x<sup>i</sup> term.
 | |
| 
 | |
|     \post  <code>this->get_initial_remainder() ==
 | |
|       this->get_interim_remainder()</code>
 | |
|     \post  <code>((this->get_reflect_remainder() ?
 | |
|       REFLECT(this->get_initial_remainder()) :
 | |
|       this->get_initial_remainder()) ^ this->get_final_xor_value())
 | |
|       == this->checksum()</code>
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| void
 | |
| crc_basic<Bits>::reset
 | |
| (
 | |
| )
 | |
| {
 | |
|     this->reset( this->get_initial_remainder() );
 | |
| }
 | |
| 
 | |
| /** Updates the interim remainder with a single altered-CRC-division step.
 | |
| 
 | |
|     \param[in] bit  The new input bit.
 | |
| 
 | |
|     \post  The interim remainder is updated though a modulo-2 polynomial
 | |
|       division, where the division steps are altered for unaugmented CRCs.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| void
 | |
| crc_basic<Bits>::process_bit
 | |
| (
 | |
|     bool  bit
 | |
| )
 | |
| {
 | |
|     detail::crc_modulo_update( bit_count, rem_, bit, poly_ );
 | |
| }
 | |
| 
 | |
| /** Updates the interim remainder with several altered-CRC-division steps.  Each
 | |
|     bit is processed separately, starting from the one at the
 | |
|     2<sup><var>bit_length</var> - 1</sup> place, then proceeding down to the
 | |
|     lowest-placed bit.  Any order imposed by
 | |
|     <code>this->get_reflect_input()</code> is ignored.
 | |
| 
 | |
|     \pre  0 \< \a bit_length \<= \c CHAR_BIT
 | |
| 
 | |
|     \param[in] bits  The byte containing the new input bits.
 | |
|     \param[in] bit_length  The number of bits in the byte to be read.
 | |
| 
 | |
|     \post  The interim remainder is updated though \a bit_length modulo-2
 | |
|       polynomial divisions, where the division steps are altered for unaugmented
 | |
|       CRCs.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| void
 | |
| crc_basic<Bits>::process_bits
 | |
| (
 | |
|     unsigned char  bits,
 | |
|     std::size_t    bit_length
 | |
| )
 | |
| {
 | |
|     // ignore the bits above the ones we want
 | |
|     bits <<= CHAR_BIT - bit_length;
 | |
| 
 | |
|     // compute the CRC for each bit, starting with the upper ones
 | |
|     unsigned char const  high_bit_mask = 1u << ( CHAR_BIT - 1u );
 | |
|     for ( std::size_t i = bit_length ; i > 0u ; --i, bits <<= 1u )
 | |
|     {
 | |
|         process_bit( static_cast<bool>(bits & high_bit_mask) );
 | |
|     }
 | |
| }
 | |
| 
 | |
| /** Updates the interim remainder with a byte's worth of altered-CRC-division
 | |
|     steps.  The bits within the byte are processed from the highest place down
 | |
|     if <code>this->get_reflect_input()</code> is \c false, and lowest place
 | |
|     up otherwise.
 | |
| 
 | |
|     \param[in] byte  The new input byte.
 | |
| 
 | |
|     \post  The interim remainder is updated though \c CHAR_BIT modulo-2
 | |
|       polynomial divisions, where the division steps are altered for unaugmented
 | |
|       CRCs.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| void
 | |
| crc_basic<Bits>::process_byte
 | |
| (
 | |
|     unsigned char  byte
 | |
| )
 | |
| {
 | |
|     process_bits( (rft_in_ ? detail::reflect_byte( byte ) : byte), CHAR_BIT );
 | |
| }
 | |
| 
 | |
| /** Updates the interim remainder with several bytes' worth of
 | |
|     altered-CRC-division steps.  The bits within each byte are processed from
 | |
|     the highest place down if <code>this->get_reflect_input()</code> is
 | |
|     \c false, and lowest place up otherwise.  The bytes themselves are processed
 | |
|     starting from the one pointed by \a bytes_begin until \a bytes_end is
 | |
|     reached through forward iteration, treating the two pointers as if they
 | |
|     point to <code>unsigned char</code> objects.
 | |
| 
 | |
|     \pre  \a bytes_end has to equal \a bytes_begin if the latter is \c NULL or
 | |
|       otherwise doesn't point to a valid buffer.
 | |
|     \pre  \a bytes_end, if not equal to \a bytes_begin, has to point within or
 | |
|       one-byte-past the same buffer \a bytes_begin points into.
 | |
|     \pre  \a bytes_end has to be reachable from \a bytes_begin through a finite
 | |
|       number of forward byte-pointer increments.
 | |
| 
 | |
|     \param[in] bytes_begin  The address where the memory block begins.
 | |
|     \param[in] bytes_end  Points to one-byte past the address of the memory
 | |
|       block's last byte, or \a bytes_begin if no bytes are to be read.
 | |
| 
 | |
|     \post  The interim remainder is updated though <code>CHAR_BIT * (((unsigned
 | |
|       char const *) bytes_end) - ((unsigned char const *) bytes_begin))</code>
 | |
|       modulo-2 polynomial divisions, where the division steps are altered for
 | |
|       unaugmented CRCs.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| void
 | |
| crc_basic<Bits>::process_block
 | |
| (
 | |
|     void const *  bytes_begin,
 | |
|     void const *  bytes_end
 | |
| )
 | |
| {
 | |
|     for ( unsigned char const * p
 | |
|      = static_cast<unsigned char const *>(bytes_begin) ; p < bytes_end ; ++p )
 | |
|     {
 | |
|         process_byte( *p );
 | |
|     }
 | |
| }
 | |
| 
 | |
| /** Updates the interim remainder with several bytes' worth of
 | |
|     altered-CRC-division steps.  The bits within each byte are processed from
 | |
|     the highest place down if <code>this->get_reflect_input()</code> is
 | |
|     \c false, and lowest place up otherwise.  The bytes themselves are processed
 | |
|     starting from the one pointed by \a buffer, forward-iterated (as if the
 | |
|     pointed-to objects were of <code>unsigned char</code>) until \a byte_count
 | |
|     bytes are read.
 | |
| 
 | |
|     \pre  \a byte_count has to equal 0 if \a buffer is \c NULL or otherwise
 | |
|       doesn't point to valid memory.
 | |
|     \pre  If \a buffer points within valid memory, then that block has to have
 | |
|       at least \a byte_count more valid bytes allocated from that point.
 | |
| 
 | |
|     \param[in] buffer  The address where the memory block begins.
 | |
|     \param[in] byte_count  The number of bytes in the memory block.
 | |
| 
 | |
|     \post  The interim remainder is updated though <code>CHAR_BIT *
 | |
|       <var>byte_count</var></code> modulo-2 polynomial divisions, where the
 | |
|       division steps are altered for unaugmented CRCs.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| void
 | |
| crc_basic<Bits>::process_bytes
 | |
| (
 | |
|     void const *  buffer,
 | |
|     std::size_t   byte_count
 | |
| )
 | |
| {
 | |
|     unsigned char const * const  b = static_cast<unsigned char const *>(
 | |
|      buffer );
 | |
| 
 | |
|     process_block( b, b + byte_count );
 | |
| }
 | |
| 
 | |
| /** Computes the checksum of all the submitted bits since construction or the
 | |
|     last call to #reset.  The checksum will be the raw checksum, i.e. the
 | |
|     (interim) remainder after all the modulo-2 polynomial division, plus any
 | |
|     output processing.
 | |
| 
 | |
|     \return  <code>(this->get_reflect_remainder() ?
 | |
|       REFLECT(this->get_interim_remainder()) :
 | |
|       this->get_interim_remainder()) ^ this->get_final_xor_value()</code>
 | |
| 
 | |
|     \note  Since checksums are meant to be compared, any higher-placed bits
 | |
|       (when the bit-length of #value_type exceeds #bit_count) will be set to 0.
 | |
|  */
 | |
| template < std::size_t Bits >
 | |
| inline
 | |
| typename crc_basic<Bits>::value_type
 | |
| crc_basic<Bits>::checksum
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return ( (rft_out_ ? detail::reflect_unsigned( rem_, bit_count ) :
 | |
|      rem_) ^ final_ ) & detail::low_bits_mask_c<bit_count>::value;
 | |
| }
 | |
| 
 | |
| 
 | |
| //  Optimized CRC class function definitions  --------------------------------//
 | |
| 
 | |
| // Macro to compact code
 | |
| #define BOOST_CRC_OPTIMAL_NAME  crc_optimal<Bits, TruncPoly, InitRem, \
 | |
|  FinalXor, ReflectIn, ReflectRem>
 | |
| 
 | |
| /** Constructs a \c crc_optimal object with a particular CRC formula to be
 | |
|     processed upon receiving input.  The initial remainder may be overridden.
 | |
| 
 | |
|     \param[in] init_rem  The (unaugmented) initial state of the polynomial
 | |
|       remainder.  Defaults to #initial_remainder if omitted.
 | |
| 
 | |
|     \post  <code>#truncated_polynominal ==
 | |
|       this->get_truncated_polynominal()</code>
 | |
|     \post  <code>#initial_remainder == this->get_initial_remainder()</code>
 | |
|     \post  <code>#final_xor_value == this->get_final_xor_value()</code>
 | |
|     \post  <code>#reflect_input == this->get_reflect_input()</code>
 | |
|     \post  <code>#reflect_remainder == this->get_reflect_remainder()</code>
 | |
|     \post  <code><var>init_rem</var> == this->get_interim_remainder()</code>
 | |
|     \post  <code>(#reflect_remainder ? REFLECT(<var>init_rem</var>) :
 | |
|       <var>init_rem</var>) ^ #final_xor_value == this->checksum()</code>
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| BOOST_CRC_OPTIMAL_NAME::crc_optimal
 | |
| (
 | |
|     value_type  init_rem  // = initial_remainder
 | |
| )
 | |
|     : rem_( reflect_i_type::reflect_q(init_rem) )
 | |
| {
 | |
| }
 | |
| 
 | |
| //! \copydetails  boost::crc_basic::get_truncated_polynominal
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| typename BOOST_CRC_OPTIMAL_NAME::value_type
 | |
| BOOST_CRC_OPTIMAL_NAME::get_truncated_polynominal
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return truncated_polynominal;
 | |
| }
 | |
| 
 | |
| //! \copydetails  boost::crc_basic::get_initial_remainder
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| typename BOOST_CRC_OPTIMAL_NAME::value_type
 | |
| BOOST_CRC_OPTIMAL_NAME::get_initial_remainder
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return initial_remainder;
 | |
| }
 | |
| 
 | |
| //! \copydetails  boost::crc_basic::get_final_xor_value
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| typename BOOST_CRC_OPTIMAL_NAME::value_type
 | |
| BOOST_CRC_OPTIMAL_NAME::get_final_xor_value
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return final_xor_value;
 | |
| }
 | |
| 
 | |
| //! \copydetails  boost::crc_basic::get_reflect_input
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| bool
 | |
| BOOST_CRC_OPTIMAL_NAME::get_reflect_input
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return reflect_input;
 | |
| }
 | |
| 
 | |
| //! \copydetails  boost::crc_basic::get_reflect_remainder
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| bool
 | |
| BOOST_CRC_OPTIMAL_NAME::get_reflect_remainder
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return reflect_remainder;
 | |
| }
 | |
| 
 | |
| //! \copydetails  boost::crc_basic::get_interim_remainder
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| typename BOOST_CRC_OPTIMAL_NAME::value_type
 | |
| BOOST_CRC_OPTIMAL_NAME::get_interim_remainder
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     // Interim remainder should be _un_-reflected, so we have to undo it.
 | |
|     return reflect_i_type::reflect_q( rem_ ) &
 | |
|      detail::low_bits_mask_c<bit_count>::value;
 | |
| }
 | |
| 
 | |
| /** Changes the interim polynomial remainder to \a new_rem, purging any
 | |
|     influence previously submitted input has had.  The value of the
 | |
|     2<sup>i</sup> bit is the value of the coefficient of the polynomial's
 | |
|     x<sup>i</sup> term.
 | |
| 
 | |
|     \param[in] new_rem  The (unaugmented) state of the polynomial remainder
 | |
|       starting from this point, with no output processing applied.  Defaults to
 | |
|       <code>this->get_initial_remainder()</code> if omitted.
 | |
| 
 | |
|     \post  <code><var>new_rem</var> == this->get_interim_remainder()</code>
 | |
|     \post  <code>((this->get_reflect_remainder() ?
 | |
|       REFLECT(<var>new_rem</var>) : <var>new_rem</var>) ^
 | |
|       this->get_final_xor_value()) == this->checksum()</code>
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| void
 | |
| BOOST_CRC_OPTIMAL_NAME::reset
 | |
| (
 | |
|     value_type  new_rem  // = initial_remainder
 | |
| )
 | |
| {
 | |
|     rem_ = reflect_i_type::reflect_q( new_rem );
 | |
| }
 | |
| 
 | |
| /** \copydetails  boost::crc_basic::process_byte
 | |
| 
 | |
|     \note  Any modulo-2 polynomial divisions may use a table of pre-computed
 | |
|       remainder changes (as XOR masks) to speed computation when reading data
 | |
|       byte-wise.
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| void
 | |
| BOOST_CRC_OPTIMAL_NAME::process_byte
 | |
| (
 | |
|     unsigned char  byte
 | |
| )
 | |
| {
 | |
|     process_bytes( &byte, sizeof(byte) );
 | |
| }
 | |
| 
 | |
| /** \copydetails  boost::crc_basic::process_block
 | |
| 
 | |
|     \note  Any modulo-2 polynomial divisions may use a table of pre-computed
 | |
|       remainder changes (as XOR masks) to speed computation when reading data
 | |
|       byte-wise.
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| void
 | |
| BOOST_CRC_OPTIMAL_NAME::process_block
 | |
| (
 | |
|     void const *  bytes_begin,
 | |
|     void const *  bytes_end
 | |
| )
 | |
| {
 | |
|     process_bytes( bytes_begin, static_cast<unsigned char const *>(bytes_end) -
 | |
|      static_cast<unsigned char const *>(bytes_begin) );
 | |
| }
 | |
| 
 | |
| /** \copydetails  boost::crc_basic::process_bytes
 | |
| 
 | |
|     \note  Any modulo-2 polynomial divisions may use a table of pre-computed
 | |
|       remainder changes (as XOR masks) to speed computation when reading data
 | |
|       byte-wise.
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| void
 | |
| BOOST_CRC_OPTIMAL_NAME::process_bytes
 | |
| (
 | |
|     void const *   buffer,
 | |
|     std::size_t  byte_count
 | |
| )
 | |
| {
 | |
|     rem_ = crc_table_type::crc_update( rem_, static_cast<unsigned char const
 | |
|      *>(buffer), byte_count );
 | |
| }
 | |
| 
 | |
| //! \copydetails  boost::crc_basic::checksum
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| typename BOOST_CRC_OPTIMAL_NAME::value_type
 | |
| BOOST_CRC_OPTIMAL_NAME::checksum
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return ( reflect_o_type::reflect_q(rem_) ^ get_final_xor_value() )
 | |
|      & detail::low_bits_mask_c<bit_count>::value;
 | |
| }
 | |
| 
 | |
| /** Updates the interim remainder with a byte's worth of altered-CRC-division
 | |
|     steps.  The bits within the byte are processed from the highest place down
 | |
|     if <code>this->get_reflect_input()</code> is \c false, and lowest place
 | |
|     up otherwise.  This function is meant to present a function-object interface
 | |
|     to code that wants to process a stream of bytes with
 | |
|     <code>std::for_each</code> or similar range-processing algorithms.  Since
 | |
|     some of these algorithms takes their function object by value, make sure to
 | |
|     copy back the result to this object so the updates can be remembered.
 | |
| 
 | |
|     \param[in] byte  The new input byte.
 | |
| 
 | |
|     \post  The interim remainder is updated though \c CHAR_BIT modulo-2
 | |
|       polynomial divisions, where the division steps are altered for unaugmented
 | |
|       CRCs.
 | |
| 
 | |
|     \note  Any modulo-2 polynomial divisions may use a table of pre-computed
 | |
|       remainder changes (as XOR masks) to speed computation when reading data
 | |
|       byte-wise.
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| void
 | |
| BOOST_CRC_OPTIMAL_NAME::operator ()
 | |
| (
 | |
|     unsigned char  byte
 | |
| )
 | |
| {
 | |
|     process_byte( byte );
 | |
| }
 | |
| 
 | |
| /** Computes the checksum of all the submitted bits since construction or the
 | |
|     last call to #reset.  The checksum will be the raw checksum, i.e. the
 | |
|     (interim) remainder after all the modulo-2 polynomial division, plus any
 | |
|     output processing.  This function is meant to present a function-object
 | |
|     interface to code that wants to receive data like
 | |
|     <code>std::generate_n</code> or similar data-processing algorithms.  Note
 | |
|     that if this object is used as a generator multiple times without an
 | |
|     intervening mutating operation, the same value will always be returned.
 | |
| 
 | |
|     \return  <code>(this->get_reflect_remainder() ?
 | |
|       REFLECT(this->get_interim_remainder()) :
 | |
|       this->get_interim_remainder()) ^ this->get_final_xor_value()</code>
 | |
| 
 | |
|     \note  Since checksums are meant to be compared, any higher-placed bits
 | |
|       (when the bit-length of #value_type exceeds #bit_count) will be set to 0.
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| typename BOOST_CRC_OPTIMAL_NAME::value_type
 | |
| BOOST_CRC_OPTIMAL_NAME::operator ()
 | |
| (
 | |
| ) const
 | |
| {
 | |
|     return checksum();
 | |
| }
 | |
| 
 | |
| 
 | |
| //  CRC computation function definition  -------------------------------------//
 | |
| 
 | |
| /** Computes the polynomial remainder of a CRC run, assuming that \a buffer and
 | |
|     \a byte_count describe a memory block representing the polynomial dividend.
 | |
|     The division steps are altered so the result directly gives a checksum,
 | |
|     without need to augment the memory block with scratch-space bytes.  The
 | |
|     first byte is considered the highest order, going down for subsequent bytes.
 | |
| 
 | |
|     \pre  0 \< \a Bits \<= \c std\::numeric_limits\<uintmax_t\>\::digits
 | |
| 
 | |
|     \tparam Bits  The order of the modulo-2 polynomial divisor.  (\e Width from
 | |
|       the RMCA)
 | |
|     \tparam TruncPoly  The lowest coefficients of the divisor polynomial.  The
 | |
|       highest-order coefficient is omitted and always assumed to be 1.
 | |
|       (\e Poly from the RMCA)
 | |
|     \tparam InitRem  The (unaugmented) initial state of the polynomial
 | |
|       remainder.  (\e Init from the RMCA)
 | |
|     \tparam FinalXor  The (XOR) bit-mask to be applied to the output remainder,
 | |
|       after possible reflection but before returning.  (\e XorOut from the RMCA)
 | |
|     \tparam ReflectIn  If \c True, input bytes are read lowest-order bit first,
 | |
|       otherwise highest-order bit first.  (\e RefIn from the RMCA)
 | |
|     \tparam ReflectRem  If \c True, the output remainder is reflected before the
 | |
|       XOR-mask.  (\e RefOut from the RMCA)
 | |
| 
 | |
|     \param[in] buffer  The address where the memory block begins.
 | |
|     \param[in] byte_count  The number of bytes in the memory block.
 | |
| 
 | |
|     \return  The checksum, which is the last (interim) remainder plus any output
 | |
|       processing.
 | |
| 
 | |
|     \note  Unaugmented-style CRC runs perform modulo-2 polynomial division in
 | |
|       an altered order.  The trailing \a Bits number of zero-valued bits needed
 | |
|       to extracted an (unprocessed) checksum is virtually moved to near the
 | |
|       beginning of the message.  This is OK since the XOR operation is
 | |
|       commutative and associative.  It also means that you can get a checksum
 | |
|       anytime.  Since data is being read byte-wise, a table of pre-computed
 | |
|       remainder changes (as XOR masks) can be used to speed computation.
 | |
| 
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly,
 | |
|            BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor,
 | |
|            bool ReflectIn, bool ReflectRem >
 | |
| inline
 | |
| typename uint_t<Bits>::fast
 | |
| crc
 | |
| (
 | |
|     void const *  buffer,
 | |
|     std::size_t   byte_count
 | |
| )
 | |
| {
 | |
|     BOOST_CRC_OPTIMAL_NAME  computer;
 | |
|     computer.process_bytes( buffer, byte_count );
 | |
|     return computer.checksum();
 | |
| }
 | |
| 
 | |
| 
 | |
| //  Augmented-message CRC computation function definition  -------------------//
 | |
| 
 | |
| /** Computes the polynomial remainder of a CRC run, assuming that \a buffer and
 | |
|     \a byte_count describe a memory block representing the polynomial dividend.
 | |
|     The first byte is considered the highest order, going down for subsequent
 | |
|     bytes.  Within a byte, the highest-order bit is read first (corresponding to
 | |
|     \e RefIn = \c False in the RMCA).  Check the other parts of this function's
 | |
|     documentation to see how a checksum can be gained and/or used.
 | |
| 
 | |
|     \pre  0 \< \a Bits \<= \c std\::numeric_limit\<uintmax_t\>\::digits
 | |
| 
 | |
|     \tparam Bits  The order of the modulo-2 polynomial divisor.  (\e Width from
 | |
|       the RMCA)
 | |
|     \tparam TruncPoly  The lowest coefficients of the divisor polynomial.  The
 | |
|       highest-order coefficient is omitted and always assumed to be 1.
 | |
|       (\e Poly from the RMCA)
 | |
| 
 | |
|     \param[in] buffer  The address where the memory block begins.
 | |
|     \param[in] byte_count  The number of bytes in the memory block.
 | |
|     \param[in] initial_remainder  The initial state of the polynomial
 | |
|       remainder, defaulting to zero if omitted.  If you are reading a memory
 | |
|       block in multiple runs, put the return value of the previous run here.
 | |
|       (Note that initial-remainders given by RMCA parameter lists, as
 | |
|       \e Init, assume that the initial remainder is in its \b unaugmented state,
 | |
|       so you would need to convert the value to make it suitable for this
 | |
|       function.  I currently don't provide a conversion routine.)
 | |
| 
 | |
|     \return  The interim remainder, if no augmentation is used.  A special value
 | |
|       if augmentation is used (see the notes).  No output processing is done on
 | |
|       the value.  (In RMCA terms, \e RefOut is \c False and \e XorOut is \c 0.)
 | |
| 
 | |
|     \note  Augmented-style CRC runs use straight-up modulo-2 polynomial
 | |
|       division.  Since data is being read byte-wise, a table of pre-computed
 | |
|       remainder changes (as XOR masks) can be used to speed computation.
 | |
|     \note  Reading just a memory block will yield an interim remainder, and not
 | |
|       the final checksum.  To get that checksum, allocate \a Bits / \c CHAR_BIT
 | |
|       bytes directly after the block and fill them with zero values, then extend
 | |
|       \a byte_count to include those extra bytes.  A data block is corrupt if
 | |
|       the return value doesn't equal your separately given checksum.
 | |
|     \note  Another way to perform a check is use the zero-byte extension method,
 | |
|       but replace the zero values with your separately-given checksum.  The
 | |
|       checksum must be loaded in big-endian order.  Here corruption, in either
 | |
|       the data block or the given checksum, is confirmed if the return value is
 | |
|       not zero.
 | |
|     \note  The two checksum techniques assume the CRC-run is performed bit-wise,
 | |
|       while this function works byte-wise.  That means that the techniques can
 | |
|       be used only if \c CHAR_BIT divides \a Bits evenly!
 | |
|  */
 | |
| template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly >
 | |
| typename uint_t<Bits>::fast
 | |
| augmented_crc
 | |
| (
 | |
|     void const *                 buffer,
 | |
|     std::size_t                  byte_count,
 | |
|     typename uint_t<Bits>::fast  initial_remainder  // = 0u
 | |
| )
 | |
| {
 | |
|     return detail::low_bits_mask_c<Bits>::value &
 | |
|      detail::byte_table_driven_crcs<Bits, TruncPoly, false>::
 | |
|      augmented_crc_update( initial_remainder, static_cast<unsigned char const
 | |
|      *>(buffer), byte_count );
 | |
| }
 | |
| 
 | |
| 
 | |
| }  // namespace boost
 | |
| 
 | |
| 
 | |
| // Undo header-private macros
 | |
| #undef BOOST_CRC_OPTIMAL_NAME
 | |
| #undef BOOST_CRC_PARM_TYPE
 | |
| 
 | |
| 
 | |
| #endif  // BOOST_CRC_HPP
 | |
| 
 |