sl@0: // Boost CRC library crc.hpp header file -----------------------------------// sl@0: sl@0: // Copyright 2001, 2004 Daryle Walker. Use, modification, and distribution are sl@0: // subject to the Boost Software License, Version 1.0. (See accompanying file sl@0: // LICENSE_1_0.txt or a copy at .) sl@0: sl@0: // See for the library's home page. sl@0: sl@0: #ifndef BOOST_CRC_HPP sl@0: #define BOOST_CRC_HPP sl@0: sl@0: #include // for BOOST_STATIC_CONSTANT, etc. sl@0: #include // for boost::uint_t sl@0: sl@0: #include // for CHAR_BIT, etc. sl@0: #include // for std::size_t sl@0: sl@0: #include // for std::numeric_limits sl@0: sl@0: sl@0: // The type of CRC parameters that can go in a template should be related sl@0: // on the CRC's bit count. This macro expresses that type in a compact sl@0: // form, but also allows an alternate type for compilers that don't support sl@0: // dependent types (in template value-parameters). sl@0: #if !(defined(BOOST_NO_DEPENDENT_TYPES_IN_TEMPLATE_VALUE_PARAMETERS) || (defined(BOOST_MSVC) && (BOOST_MSVC <= 1300))) sl@0: #define BOOST_CRC_PARM_TYPE typename ::boost::uint_t::fast sl@0: #else sl@0: #define BOOST_CRC_PARM_TYPE unsigned long sl@0: #endif sl@0: sl@0: // Some compilers [MS VC++ 6] cannot correctly set up several versions of a sl@0: // function template unless every template argument can be unambiguously sl@0: // deduced from the function arguments. (The bug is hidden if only one version sl@0: // is needed.) Since all of the CRC function templates have this problem, the sl@0: // workaround is to make up a dummy function argument that encodes the template sl@0: // arguments. Calls to such template functions need all their template sl@0: // arguments explicitly specified. At least one compiler that needs this sl@0: // workaround also needs the default value for the dummy argument to be sl@0: // specified in the definition. sl@0: #if defined(__GNUC__) || !defined(BOOST_NO_EXPLICIT_FUNCTION_TEMPLATE_ARGUMENTS) sl@0: #define BOOST_CRC_DUMMY_PARM_TYPE sl@0: #define BOOST_CRC_DUMMY_INIT sl@0: #define BOOST_ACRC_DUMMY_PARM_TYPE sl@0: #define BOOST_ACRC_DUMMY_INIT sl@0: #else sl@0: namespace boost { namespace detail { sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: struct dummy_crc_argument { }; sl@0: } } sl@0: #define BOOST_CRC_DUMMY_PARM_TYPE , detail::dummy_crc_argument *p_ sl@0: #define BOOST_CRC_DUMMY_INIT BOOST_CRC_DUMMY_PARM_TYPE = 0 sl@0: #define BOOST_ACRC_DUMMY_PARM_TYPE , detail::dummy_crc_argument *p_ sl@0: #define BOOST_ACRC_DUMMY_INIT BOOST_ACRC_DUMMY_PARM_TYPE = 0 sl@0: #endif sl@0: sl@0: sl@0: namespace boost sl@0: { sl@0: sl@0: sl@0: // Forward declarations ----------------------------------------------------// sl@0: sl@0: template < std::size_t Bits > sl@0: class crc_basic; sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly = 0u, sl@0: BOOST_CRC_PARM_TYPE InitRem = 0u, sl@0: BOOST_CRC_PARM_TYPE FinalXor = 0u, bool ReflectIn = false, sl@0: bool ReflectRem = false > sl@0: class crc_optimal; sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: typename uint_t::fast crc( void const *buffer, sl@0: std::size_t byte_count sl@0: BOOST_CRC_DUMMY_PARM_TYPE ); sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly > sl@0: typename uint_t::fast augmented_crc( void const *buffer, sl@0: std::size_t byte_count, typename uint_t::fast initial_remainder sl@0: BOOST_ACRC_DUMMY_PARM_TYPE ); sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly > sl@0: typename uint_t::fast augmented_crc( void const *buffer, sl@0: std::size_t byte_count sl@0: BOOST_ACRC_DUMMY_PARM_TYPE ); sl@0: sl@0: typedef crc_optimal<16, 0x8005, 0, 0, true, true> crc_16_type; sl@0: typedef crc_optimal<16, 0x1021, 0xFFFF, 0, false, false> crc_ccitt_type; sl@0: typedef crc_optimal<16, 0x8408, 0, 0, true, true> crc_xmodem_type; sl@0: sl@0: typedef crc_optimal<32, 0x04C11DB7, 0xFFFFFFFF, 0xFFFFFFFF, true, true> sl@0: crc_32_type; sl@0: sl@0: sl@0: // Forward declarations for implementation detail stuff --------------------// sl@0: // (Just for the stuff that will be needed for the next two sections) sl@0: sl@0: namespace detail sl@0: { sl@0: template < std::size_t Bits > sl@0: struct mask_uint_t; sl@0: sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits >; sl@0: sl@0: #if USHRT_MAX > UCHAR_MAX sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits >; sl@0: #endif sl@0: sl@0: #if UINT_MAX > USHRT_MAX sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits >; sl@0: #endif sl@0: sl@0: #if ULONG_MAX > UINT_MAX sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits >; sl@0: #endif sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, bool Reflect > sl@0: struct crc_table_t; sl@0: sl@0: template < std::size_t Bits, bool DoReflect > sl@0: class crc_helper; sl@0: sl@0: #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION sl@0: template < std::size_t Bits > sl@0: class crc_helper< Bits, false >; sl@0: #endif sl@0: sl@0: } // namespace detail sl@0: sl@0: sl@0: // Simple cyclic redundancy code (CRC) class declaration -------------------// sl@0: sl@0: template < std::size_t Bits > sl@0: class crc_basic sl@0: { sl@0: // Implementation type sl@0: typedef detail::mask_uint_t masking_type; sl@0: sl@0: public: sl@0: // Type sl@0: typedef typename masking_type::least value_type; sl@0: sl@0: // Constant for the template parameter sl@0: BOOST_STATIC_CONSTANT( std::size_t, bit_count = Bits ); sl@0: sl@0: // Constructor sl@0: explicit crc_basic( value_type truncated_polynominal, sl@0: value_type initial_remainder = 0, value_type final_xor_value = 0, sl@0: bool reflect_input = false, bool reflect_remainder = false ); sl@0: sl@0: // Internal Operations sl@0: value_type get_truncated_polynominal() const; sl@0: value_type get_initial_remainder() const; sl@0: value_type get_final_xor_value() const; sl@0: bool get_reflect_input() const; sl@0: bool get_reflect_remainder() const; sl@0: sl@0: value_type get_interim_remainder() const; sl@0: void reset( value_type new_rem ); sl@0: void reset(); sl@0: sl@0: // External Operations sl@0: void process_bit( bool bit ); sl@0: void process_bits( unsigned char bits, std::size_t bit_count ); sl@0: void process_byte( unsigned char byte ); sl@0: void process_block( void const *bytes_begin, void const *bytes_end ); sl@0: void process_bytes( void const *buffer, std::size_t byte_count ); sl@0: sl@0: value_type checksum() const; sl@0: sl@0: private: sl@0: // Member data sl@0: value_type rem_; sl@0: value_type poly_, init_, final_; // non-const to allow assignability sl@0: bool rft_in_, rft_out_; // non-const to allow assignability sl@0: sl@0: }; // boost::crc_basic sl@0: sl@0: sl@0: // Optimized cyclic redundancy code (CRC) class declaration ----------------// sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: class crc_optimal sl@0: { sl@0: // Implementation type sl@0: typedef detail::mask_uint_t masking_type; sl@0: sl@0: public: sl@0: // Type sl@0: typedef typename masking_type::fast value_type; sl@0: sl@0: // Constants for the template parameters sl@0: BOOST_STATIC_CONSTANT( std::size_t, bit_count = Bits ); sl@0: BOOST_STATIC_CONSTANT( value_type, truncated_polynominal = TruncPoly ); sl@0: BOOST_STATIC_CONSTANT( value_type, initial_remainder = InitRem ); sl@0: BOOST_STATIC_CONSTANT( value_type, final_xor_value = FinalXor ); sl@0: BOOST_STATIC_CONSTANT( bool, reflect_input = ReflectIn ); sl@0: BOOST_STATIC_CONSTANT( bool, reflect_remainder = ReflectRem ); sl@0: sl@0: // Constructor sl@0: explicit crc_optimal( value_type init_rem = InitRem ); sl@0: sl@0: // Internal Operations sl@0: value_type get_truncated_polynominal() const; sl@0: value_type get_initial_remainder() const; sl@0: value_type get_final_xor_value() const; sl@0: bool get_reflect_input() const; sl@0: bool get_reflect_remainder() const; sl@0: sl@0: value_type get_interim_remainder() const; sl@0: void reset( value_type new_rem = InitRem ); sl@0: sl@0: // External Operations sl@0: void process_byte( unsigned char byte ); sl@0: void process_block( void const *bytes_begin, void const *bytes_end ); sl@0: void process_bytes( void const *buffer, std::size_t byte_count ); sl@0: sl@0: value_type checksum() const; sl@0: sl@0: // Operators sl@0: void operator ()( unsigned char byte ); sl@0: value_type operator ()() const; sl@0: sl@0: private: sl@0: // The implementation of output reflection depends on both reflect states. sl@0: BOOST_STATIC_CONSTANT( bool, reflect_output = (ReflectRem != ReflectIn) ); sl@0: sl@0: #ifndef __BORLANDC__ sl@0: #define BOOST_CRC_REF_OUT_VAL reflect_output sl@0: #else sl@0: typedef crc_optimal self_type; sl@0: #define BOOST_CRC_REF_OUT_VAL (self_type::reflect_output) sl@0: #endif sl@0: sl@0: // More implementation types sl@0: typedef detail::crc_table_t crc_table_type; sl@0: typedef detail::crc_helper helper_type; sl@0: typedef detail::crc_helper reflect_out_type; sl@0: sl@0: #undef BOOST_CRC_REF_OUT_VAL sl@0: sl@0: // Member data sl@0: value_type rem_; sl@0: sl@0: }; // boost::crc_optimal sl@0: sl@0: sl@0: // Implementation detail stuff ---------------------------------------------// sl@0: sl@0: namespace detail sl@0: { sl@0: // Forward declarations for more implementation details sl@0: template < std::size_t Bits > sl@0: struct high_uint_t; sl@0: sl@0: template < std::size_t Bits > sl@0: struct reflector; sl@0: sl@0: sl@0: // Traits class for mask; given the bit number sl@0: // (1-based), get the mask for that bit by itself. sl@0: template < std::size_t Bits > sl@0: struct high_uint_t sl@0: : boost::uint_t< Bits > sl@0: { sl@0: typedef boost::uint_t base_type; sl@0: typedef typename base_type::least least; sl@0: typedef typename base_type::fast fast; sl@0: sl@0: #if defined(__EDG_VERSION__) && __EDG_VERSION__ <= 243 sl@0: static const least high_bit = 1ul << ( Bits - 1u ); sl@0: static const fast high_bit_fast = 1ul << ( Bits - 1u ); sl@0: #else sl@0: BOOST_STATIC_CONSTANT( least, high_bit = (least( 1u ) << ( Bits sl@0: - 1u )) ); sl@0: BOOST_STATIC_CONSTANT( fast, high_bit_fast = (fast( 1u ) << ( Bits sl@0: - 1u )) ); sl@0: #endif sl@0: sl@0: }; // boost::detail::high_uint_t sl@0: sl@0: sl@0: // Reflection routine class wrapper sl@0: // (since MS VC++ 6 couldn't handle the unwrapped version) sl@0: template < std::size_t Bits > sl@0: struct reflector sl@0: { sl@0: typedef typename boost::uint_t::fast value_type; sl@0: sl@0: static value_type reflect( value_type x ); sl@0: sl@0: }; // boost::detail::reflector sl@0: sl@0: // Function that reflects its argument sl@0: template < std::size_t Bits > sl@0: typename reflector::value_type sl@0: reflector::reflect sl@0: ( sl@0: typename reflector::value_type x sl@0: ) sl@0: { sl@0: value_type reflection = 0; sl@0: value_type const one = 1; sl@0: sl@0: for ( std::size_t i = 0 ; i < Bits ; ++i, x >>= 1 ) sl@0: { sl@0: if ( x & one ) sl@0: { sl@0: reflection |= ( one << (Bits - 1u - i) ); sl@0: } sl@0: } sl@0: sl@0: return reflection; sl@0: } sl@0: sl@0: sl@0: // Traits class for masks; given the bit number (1-based), sl@0: // get the mask for that bit and its lower bits. sl@0: template < std::size_t Bits > sl@0: struct mask_uint_t sl@0: : high_uint_t< Bits > sl@0: { sl@0: typedef high_uint_t base_type; sl@0: typedef typename base_type::least least; sl@0: typedef typename base_type::fast fast; sl@0: sl@0: #ifndef __BORLANDC__ sl@0: using base_type::high_bit; sl@0: using base_type::high_bit_fast; sl@0: #else sl@0: BOOST_STATIC_CONSTANT( least, high_bit = base_type::high_bit ); sl@0: BOOST_STATIC_CONSTANT( fast, high_bit_fast = base_type::high_bit_fast ); sl@0: #endif sl@0: sl@0: #if defined(__EDG_VERSION__) && __EDG_VERSION__ <= 243 sl@0: static const least sig_bits = (~( ~( 0ul ) << Bits )) ; sl@0: #else sl@0: BOOST_STATIC_CONSTANT( least, sig_bits = (~( ~(least( 0u )) << Bits )) ); sl@0: #endif sl@0: #if defined(__GNUC__) && __GNUC__ == 4 && __GNUC_MINOR__ == 0 && __GNUC_PATCHLEVEL__ == 2 sl@0: // Work around a weird bug that ICEs the compiler in build_c_cast sl@0: BOOST_STATIC_CONSTANT( fast, sig_bits_fast = static_cast(sig_bits) ); sl@0: #else sl@0: BOOST_STATIC_CONSTANT( fast, sig_bits_fast = fast(sig_bits) ); sl@0: #endif sl@0: }; // boost::detail::mask_uint_t sl@0: sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits > sl@0: : high_uint_t< std::numeric_limits::digits > sl@0: { sl@0: typedef high_uint_t::digits> sl@0: base_type; sl@0: typedef base_type::least least; sl@0: typedef base_type::fast fast; sl@0: sl@0: #ifndef __BORLANDC__ sl@0: using base_type::high_bit; sl@0: using base_type::high_bit_fast; sl@0: #else sl@0: BOOST_STATIC_CONSTANT( least, high_bit = base_type::high_bit ); sl@0: BOOST_STATIC_CONSTANT( fast, high_bit_fast = base_type::high_bit_fast ); sl@0: #endif sl@0: sl@0: BOOST_STATIC_CONSTANT( least, sig_bits = (~( least(0u) )) ); sl@0: BOOST_STATIC_CONSTANT( fast, sig_bits_fast = fast(sig_bits) ); sl@0: sl@0: }; // boost::detail::mask_uint_t sl@0: sl@0: #if USHRT_MAX > UCHAR_MAX sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits > sl@0: : high_uint_t< std::numeric_limits::digits > sl@0: { sl@0: typedef high_uint_t::digits> sl@0: base_type; sl@0: typedef base_type::least least; sl@0: typedef base_type::fast fast; sl@0: sl@0: #ifndef __BORLANDC__ sl@0: using base_type::high_bit; sl@0: using base_type::high_bit_fast; sl@0: #else sl@0: BOOST_STATIC_CONSTANT( least, high_bit = base_type::high_bit ); sl@0: BOOST_STATIC_CONSTANT( fast, high_bit_fast = base_type::high_bit_fast ); sl@0: #endif sl@0: sl@0: BOOST_STATIC_CONSTANT( least, sig_bits = (~( least(0u) )) ); sl@0: BOOST_STATIC_CONSTANT( fast, sig_bits_fast = fast(sig_bits) ); sl@0: sl@0: }; // boost::detail::mask_uint_t sl@0: #endif sl@0: sl@0: #if UINT_MAX > USHRT_MAX sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits > sl@0: : high_uint_t< std::numeric_limits::digits > sl@0: { sl@0: typedef high_uint_t::digits> sl@0: base_type; sl@0: typedef base_type::least least; sl@0: typedef base_type::fast fast; sl@0: sl@0: #ifndef __BORLANDC__ sl@0: using base_type::high_bit; sl@0: using base_type::high_bit_fast; sl@0: #else sl@0: BOOST_STATIC_CONSTANT( least, high_bit = base_type::high_bit ); sl@0: BOOST_STATIC_CONSTANT( fast, high_bit_fast = base_type::high_bit_fast ); sl@0: #endif sl@0: sl@0: BOOST_STATIC_CONSTANT( least, sig_bits = (~( least(0u) )) ); sl@0: BOOST_STATIC_CONSTANT( fast, sig_bits_fast = fast(sig_bits) ); sl@0: sl@0: }; // boost::detail::mask_uint_t sl@0: #endif sl@0: sl@0: #if ULONG_MAX > UINT_MAX sl@0: template < > sl@0: struct mask_uint_t< std::numeric_limits::digits > sl@0: : high_uint_t< std::numeric_limits::digits > sl@0: { sl@0: typedef high_uint_t::digits> sl@0: base_type; sl@0: typedef base_type::least least; sl@0: typedef base_type::fast fast; sl@0: sl@0: #ifndef __BORLANDC__ sl@0: using base_type::high_bit; sl@0: using base_type::high_bit_fast; sl@0: #else sl@0: BOOST_STATIC_CONSTANT( least, high_bit = base_type::high_bit ); sl@0: BOOST_STATIC_CONSTANT( fast, high_bit_fast = base_type::high_bit_fast ); sl@0: #endif sl@0: sl@0: BOOST_STATIC_CONSTANT( least, sig_bits = (~( least(0u) )) ); sl@0: BOOST_STATIC_CONSTANT( fast, sig_bits_fast = fast(sig_bits) ); sl@0: sl@0: }; // boost::detail::mask_uint_t sl@0: #endif sl@0: sl@0: sl@0: // CRC table generator sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, bool Reflect > sl@0: struct crc_table_t sl@0: { sl@0: BOOST_STATIC_CONSTANT( std::size_t, byte_combos = (1ul << CHAR_BIT) ); sl@0: sl@0: typedef mask_uint_t masking_type; sl@0: typedef typename masking_type::fast value_type; sl@0: #if defined(__BORLANDC__) && defined(_M_IX86) && (__BORLANDC__ == 0x560) sl@0: // for some reason Borland's command line compiler (version 0x560) sl@0: // chokes over this unless we do the calculation for it: sl@0: typedef value_type table_type[ 0x100 ]; sl@0: #else sl@0: typedef value_type table_type[ byte_combos ]; sl@0: #endif sl@0: sl@0: static void init_table(); sl@0: sl@0: static table_type table_; sl@0: sl@0: }; // boost::detail::crc_table_t sl@0: sl@0: // CRC table generator static data member definition sl@0: // (Some compilers [Borland C++] require the initializer to be present.) sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, bool Reflect > sl@0: typename crc_table_t::table_type sl@0: crc_table_t::table_ sl@0: = { 0 }; sl@0: sl@0: // Populate CRC lookup table sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, bool Reflect > sl@0: void sl@0: crc_table_t::init_table sl@0: ( sl@0: ) sl@0: { sl@0: // compute table only on the first run sl@0: static bool did_init = false; sl@0: if ( did_init ) return; sl@0: sl@0: // factor-out constants to avoid recalculation sl@0: value_type const fast_hi_bit = masking_type::high_bit_fast; sl@0: unsigned char const byte_hi_bit = 1u << (CHAR_BIT - 1u); sl@0: sl@0: // loop over every possible dividend value sl@0: unsigned char dividend = 0; sl@0: do sl@0: { sl@0: value_type remainder = 0; sl@0: sl@0: // go through all the dividend's bits sl@0: for ( unsigned char mask = byte_hi_bit ; mask ; mask >>= 1 ) sl@0: { sl@0: // check if divisor fits sl@0: if ( dividend & mask ) sl@0: { sl@0: remainder ^= fast_hi_bit; sl@0: } sl@0: sl@0: // do polynominal division sl@0: if ( remainder & fast_hi_bit ) sl@0: { sl@0: remainder <<= 1; sl@0: remainder ^= TruncPoly; sl@0: } sl@0: else sl@0: { sl@0: remainder <<= 1; sl@0: } sl@0: } sl@0: sl@0: table_[ crc_helper::reflect(dividend) ] sl@0: = crc_helper::reflect( remainder ); sl@0: } sl@0: while ( ++dividend ); sl@0: sl@0: did_init = true; sl@0: } sl@0: sl@0: #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION sl@0: // Align the msb of the remainder to a byte sl@0: template < std::size_t Bits, bool RightShift > sl@0: class remainder sl@0: { sl@0: public: sl@0: typedef typename uint_t::fast value_type; sl@0: sl@0: static unsigned char align_msb( value_type rem ) sl@0: { return rem >> (Bits - CHAR_BIT); } sl@0: }; sl@0: sl@0: // Specialization for the case that the remainder has less sl@0: // bits than a byte: align the remainder msb to the byte msb sl@0: template < std::size_t Bits > sl@0: class remainder< Bits, false > sl@0: { sl@0: public: sl@0: typedef typename uint_t::fast value_type; sl@0: sl@0: static unsigned char align_msb( value_type rem ) sl@0: { return rem << (CHAR_BIT - Bits); } sl@0: }; sl@0: #endif sl@0: sl@0: // CRC helper routines sl@0: template < std::size_t Bits, bool DoReflect > sl@0: class crc_helper sl@0: { sl@0: public: sl@0: // Type sl@0: typedef typename uint_t::fast value_type; sl@0: sl@0: #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION sl@0: // Possibly reflect a remainder sl@0: static value_type reflect( value_type x ) sl@0: { return detail::reflector::reflect( x ); } sl@0: sl@0: // Compare a byte to the remainder's highest byte sl@0: static unsigned char index( value_type rem, unsigned char x ) sl@0: { return x ^ rem; } sl@0: sl@0: // Shift out the remainder's highest byte sl@0: static value_type shift( value_type rem ) sl@0: { return rem >> CHAR_BIT; } sl@0: #else sl@0: // Possibly reflect a remainder sl@0: static value_type reflect( value_type x ) sl@0: { return DoReflect ? detail::reflector::reflect( x ) : x; } sl@0: sl@0: // Compare a byte to the remainder's highest byte sl@0: static unsigned char index( value_type rem, unsigned char x ) sl@0: { return x ^ ( DoReflect ? rem : sl@0: ((Bits>CHAR_BIT)?( rem >> (Bits - CHAR_BIT) ) : sl@0: ( rem << (CHAR_BIT - Bits) ))); } sl@0: sl@0: // Shift out the remainder's highest byte sl@0: static value_type shift( value_type rem ) sl@0: { return DoReflect ? rem >> CHAR_BIT : rem << CHAR_BIT; } sl@0: #endif sl@0: sl@0: }; // boost::detail::crc_helper sl@0: sl@0: #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION sl@0: template < std::size_t Bits > sl@0: class crc_helper sl@0: { sl@0: public: sl@0: // Type sl@0: typedef typename uint_t::fast value_type; sl@0: sl@0: // Possibly reflect a remainder sl@0: static value_type reflect( value_type x ) sl@0: { return x; } sl@0: sl@0: // Compare a byte to the remainder's highest byte sl@0: static unsigned char index( value_type rem, unsigned char x ) sl@0: { return x ^ remainderCHAR_BIT)>::align_msb( rem ); } sl@0: sl@0: // Shift out the remainder's highest byte sl@0: static value_type shift( value_type rem ) sl@0: { return rem << CHAR_BIT; } sl@0: sl@0: }; // boost::detail::crc_helper sl@0: #endif sl@0: sl@0: sl@0: } // namespace detail sl@0: sl@0: sl@0: // Simple CRC class function definitions -----------------------------------// sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: crc_basic::crc_basic sl@0: ( sl@0: typename crc_basic::value_type truncated_polynominal, sl@0: typename crc_basic::value_type initial_remainder, // = 0 sl@0: typename crc_basic::value_type final_xor_value, // = 0 sl@0: bool reflect_input, // = false sl@0: bool reflect_remainder // = false sl@0: ) sl@0: : rem_( initial_remainder ), poly_( truncated_polynominal ) sl@0: , init_( initial_remainder ), final_( final_xor_value ) sl@0: , rft_in_( reflect_input ), rft_out_( reflect_remainder ) sl@0: { sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: typename crc_basic::value_type sl@0: crc_basic::get_truncated_polynominal sl@0: ( sl@0: ) const sl@0: { sl@0: return poly_; sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: typename crc_basic::value_type sl@0: crc_basic::get_initial_remainder sl@0: ( sl@0: ) const sl@0: { sl@0: return init_; sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: typename crc_basic::value_type sl@0: crc_basic::get_final_xor_value sl@0: ( sl@0: ) const sl@0: { sl@0: return final_; sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: bool sl@0: crc_basic::get_reflect_input sl@0: ( sl@0: ) const sl@0: { sl@0: return rft_in_; sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: bool sl@0: crc_basic::get_reflect_remainder sl@0: ( sl@0: ) const sl@0: { sl@0: return rft_out_; sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: typename crc_basic::value_type sl@0: crc_basic::get_interim_remainder sl@0: ( sl@0: ) const sl@0: { sl@0: return rem_ & masking_type::sig_bits; sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: void sl@0: crc_basic::reset sl@0: ( sl@0: typename crc_basic::value_type new_rem sl@0: ) sl@0: { sl@0: rem_ = new_rem; sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: void sl@0: crc_basic::reset sl@0: ( sl@0: ) sl@0: { sl@0: this->reset( this->get_initial_remainder() ); sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: void sl@0: crc_basic::process_bit sl@0: ( sl@0: bool bit sl@0: ) sl@0: { sl@0: value_type const high_bit_mask = masking_type::high_bit; sl@0: sl@0: // compare the new bit with the remainder's highest sl@0: rem_ ^= ( bit ? high_bit_mask : 0u ); sl@0: sl@0: // a full polynominal division step is done when the highest bit is one sl@0: bool const do_poly_div = static_cast( rem_ & high_bit_mask ); sl@0: sl@0: // shift out the highest bit sl@0: rem_ <<= 1; sl@0: sl@0: // carry out the division, if needed sl@0: if ( do_poly_div ) sl@0: { sl@0: rem_ ^= poly_; sl@0: } sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: void sl@0: crc_basic::process_bits sl@0: ( sl@0: unsigned char bits, sl@0: std::size_t bit_count sl@0: ) sl@0: { sl@0: // ignore the bits above the ones we want sl@0: bits <<= CHAR_BIT - bit_count; sl@0: sl@0: // compute the CRC for each bit, starting with the upper ones sl@0: unsigned char const high_bit_mask = 1u << ( CHAR_BIT - 1u ); sl@0: for ( std::size_t i = bit_count ; i > 0u ; --i, bits <<= 1u ) sl@0: { sl@0: process_bit( static_cast(bits & high_bit_mask) ); sl@0: } sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: void sl@0: crc_basic::process_byte sl@0: ( sl@0: unsigned char byte sl@0: ) sl@0: { sl@0: process_bits( (rft_in_ ? detail::reflector::reflect(byte) sl@0: : byte), CHAR_BIT ); sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: void sl@0: crc_basic::process_block sl@0: ( sl@0: void const * bytes_begin, sl@0: void const * bytes_end sl@0: ) sl@0: { sl@0: for ( unsigned char const * p sl@0: = static_cast(bytes_begin) ; p < bytes_end ; ++p ) sl@0: { sl@0: process_byte( *p ); sl@0: } sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: void sl@0: crc_basic::process_bytes sl@0: ( sl@0: void const * buffer, sl@0: std::size_t byte_count sl@0: ) sl@0: { sl@0: unsigned char const * const b = static_cast( sl@0: buffer ); sl@0: sl@0: process_block( b, b + byte_count ); sl@0: } sl@0: sl@0: template < std::size_t Bits > sl@0: inline sl@0: typename crc_basic::value_type sl@0: crc_basic::checksum sl@0: ( sl@0: ) const sl@0: { sl@0: return ( (rft_out_ ? detail::reflector::reflect( rem_ ) : rem_) sl@0: ^ final_ ) & masking_type::sig_bits; sl@0: } sl@0: sl@0: sl@0: // Optimized CRC class function definitions --------------------------------// sl@0: sl@0: // Macro to compact code sl@0: #define BOOST_CRC_OPTIMAL_NAME crc_optimal sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: BOOST_CRC_OPTIMAL_NAME::crc_optimal sl@0: ( sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type init_rem // = InitRem sl@0: ) sl@0: : rem_( helper_type::reflect(init_rem) ) sl@0: { sl@0: crc_table_type::init_table(); sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type sl@0: BOOST_CRC_OPTIMAL_NAME::get_truncated_polynominal sl@0: ( sl@0: ) const sl@0: { sl@0: return TruncPoly; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type sl@0: BOOST_CRC_OPTIMAL_NAME::get_initial_remainder sl@0: ( sl@0: ) const sl@0: { sl@0: return InitRem; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type sl@0: BOOST_CRC_OPTIMAL_NAME::get_final_xor_value sl@0: ( sl@0: ) const sl@0: { sl@0: return FinalXor; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: bool sl@0: BOOST_CRC_OPTIMAL_NAME::get_reflect_input sl@0: ( sl@0: ) const sl@0: { sl@0: return ReflectIn; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: bool sl@0: BOOST_CRC_OPTIMAL_NAME::get_reflect_remainder sl@0: ( sl@0: ) const sl@0: { sl@0: return ReflectRem; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type sl@0: BOOST_CRC_OPTIMAL_NAME::get_interim_remainder sl@0: ( sl@0: ) const sl@0: { sl@0: // Interim remainder should be _un_-reflected, so we have to undo it. sl@0: return helper_type::reflect( rem_ ) & masking_type::sig_bits_fast; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: void sl@0: BOOST_CRC_OPTIMAL_NAME::reset sl@0: ( sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type new_rem // = InitRem sl@0: ) sl@0: { sl@0: rem_ = helper_type::reflect( new_rem ); sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: void sl@0: BOOST_CRC_OPTIMAL_NAME::process_byte sl@0: ( sl@0: unsigned char byte sl@0: ) sl@0: { sl@0: process_bytes( &byte, sizeof(byte) ); sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: void sl@0: BOOST_CRC_OPTIMAL_NAME::process_block sl@0: ( sl@0: void const * bytes_begin, sl@0: void const * bytes_end sl@0: ) sl@0: { sl@0: // Recompute the CRC for each byte passed sl@0: for ( unsigned char const * p sl@0: = static_cast(bytes_begin) ; p < bytes_end ; ++p ) sl@0: { sl@0: // Compare the new byte with the remainder's higher bits to sl@0: // get the new bits, shift out the remainder's current higher sl@0: // bits, and update the remainder with the polynominal division sl@0: // of the new bits. sl@0: unsigned char const byte_index = helper_type::index( rem_, *p ); sl@0: rem_ = helper_type::shift( rem_ ); sl@0: rem_ ^= crc_table_type::table_[ byte_index ]; sl@0: } sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: void sl@0: BOOST_CRC_OPTIMAL_NAME::process_bytes sl@0: ( sl@0: void const * buffer, sl@0: std::size_t byte_count sl@0: ) sl@0: { sl@0: unsigned char const * const b = static_cast( sl@0: buffer ); sl@0: process_block( b, b + byte_count ); sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type sl@0: BOOST_CRC_OPTIMAL_NAME::checksum sl@0: ( sl@0: ) const sl@0: { sl@0: return ( reflect_out_type::reflect(rem_) ^ get_final_xor_value() ) sl@0: & masking_type::sig_bits_fast; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: void sl@0: BOOST_CRC_OPTIMAL_NAME::operator () sl@0: ( sl@0: unsigned char byte sl@0: ) sl@0: { sl@0: process_byte( byte ); sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: typename BOOST_CRC_OPTIMAL_NAME::value_type sl@0: BOOST_CRC_OPTIMAL_NAME::operator () sl@0: ( sl@0: ) const sl@0: { sl@0: return checksum(); sl@0: } sl@0: sl@0: sl@0: // CRC computation function definition -------------------------------------// sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly, sl@0: BOOST_CRC_PARM_TYPE InitRem, BOOST_CRC_PARM_TYPE FinalXor, sl@0: bool ReflectIn, bool ReflectRem > sl@0: inline sl@0: typename uint_t::fast sl@0: crc sl@0: ( sl@0: void const * buffer, sl@0: std::size_t byte_count sl@0: BOOST_CRC_DUMMY_INIT sl@0: ) sl@0: { sl@0: BOOST_CRC_OPTIMAL_NAME computer; sl@0: computer.process_bytes( buffer, byte_count ); sl@0: return computer.checksum(); sl@0: } sl@0: sl@0: sl@0: // Augmented-message CRC computation function definitions ------------------// sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly > sl@0: typename uint_t::fast sl@0: augmented_crc sl@0: ( sl@0: void const * buffer, sl@0: std::size_t byte_count, sl@0: typename uint_t::fast initial_remainder sl@0: BOOST_ACRC_DUMMY_INIT sl@0: ) sl@0: { sl@0: typedef unsigned char byte_type; sl@0: typedef detail::mask_uint_t masking_type; sl@0: typedef detail::crc_table_t crc_table_type; sl@0: sl@0: typename masking_type::fast rem = initial_remainder; sl@0: byte_type const * const b = static_cast( buffer ); sl@0: byte_type const * const e = b + byte_count; sl@0: sl@0: crc_table_type::init_table(); sl@0: for ( byte_type const * p = b ; p < e ; ++p ) sl@0: { sl@0: // Use the current top byte as the table index to the next sl@0: // "partial product." Shift out that top byte, shifting in sl@0: // the next augmented-message byte. Complete the division. sl@0: byte_type const byte_index = rem >> ( Bits - CHAR_BIT ); sl@0: rem <<= CHAR_BIT; sl@0: rem |= *p; sl@0: rem ^= crc_table_type::table_[ byte_index ]; sl@0: } sl@0: sl@0: return rem & masking_type::sig_bits_fast; sl@0: } sl@0: sl@0: template < std::size_t Bits, BOOST_CRC_PARM_TYPE TruncPoly > sl@0: inline sl@0: typename uint_t::fast sl@0: augmented_crc sl@0: ( sl@0: void const * buffer, sl@0: std::size_t byte_count sl@0: BOOST_ACRC_DUMMY_INIT sl@0: ) sl@0: { sl@0: // The last function argument has its type specified so the other version of sl@0: // augmented_crc will be called. If the cast wasn't in place, and the sl@0: // BOOST_ACRC_DUMMY_INIT added a third argument (for a workaround), the "0" sl@0: // would match as that third argument, leading to infinite recursion. sl@0: return augmented_crc( buffer, byte_count, sl@0: static_cast::fast>(0) ); sl@0: } sl@0: sl@0: sl@0: } // namespace boost sl@0: sl@0: sl@0: // Undo header-private macros sl@0: #undef BOOST_CRC_OPTIMAL_NAME sl@0: #undef BOOST_ACRC_DUMMY_INIT sl@0: #undef BOOST_ACRC_DUMMY_PARM_TYPE sl@0: #undef BOOST_CRC_DUMMY_INIT sl@0: #undef BOOST_CRC_DUMMY_PARM_TYPE sl@0: #undef BOOST_CRC_PARM_TYPE sl@0: sl@0: sl@0: #endif // BOOST_CRC_HPP sl@0: