early-access version 3088

This commit is contained in:
pineappleEA
2022-11-05 15:35:56 +01:00
parent 4e4fc25ce3
commit b601909c6d
35519 changed files with 5996896 additions and 860 deletions

View File

@@ -0,0 +1,8 @@
#~ Copyright Rene Rivera 2008, Daryle Walker 2011
#~ Distributed under the Boost Software License, Version 1.0.
#~ (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
import testing ;
run crc_test.cpp ;
run crc_test2.cpp ;

View File

@@ -0,0 +1,756 @@
// Boost CRC test program file ---------------------------------------------//
// Copyright 2001, 2003, 2004 Daryle Walker. Use, modification, and
// distribution are subject to the Boost Software License, Version 1.0. (See
// 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.
// Revision History
// 28 Aug 2004 Added CRC tests for polynominals shorter than 8 bits
// (Daryle Walker, by patch from Bert Klaps)
// 23 Aug 2003 Adjust to updated Test framework (Daryle Walker)
// 14 May 2001 Initial version (Daryle Walker)
#include <boost/config.hpp> // for BOOST_MSVC, etc.
#include <boost/crc.hpp> // for boost::crc_basic, etc.
#include <boost/cstdint.hpp> // for boost::uint16_t, etc.
#include <boost/cstdlib.hpp> // for boost::exit_success
#include <boost/integer.hpp> // for boost::uint_t
#include <boost/random/linear_congruential.hpp> // for boost::minstd_rand
#include <boost/core/lightweight_test.hpp>
#include <boost/timer.hpp> // for boost::timer
#include <algorithm> // for std::for_each, std::generate_n, std::count
#include <climits> // for CHAR_BIT
#include <cstddef> // for std::size_t
#include <iostream> // for std::cout (std::ostream and std::endl indirectly)
#if CHAR_BIT != 8
#error The expected results assume an eight-bit byte.
#endif
#if !(defined(BOOST_NO_DEPENDENT_TYPES_IN_TEMPLATE_VALUE_PARAMETERS) || (defined(BOOST_MSVC) && (BOOST_MSVC <= 1300)))
#define CRC_PARM_TYPE typename boost::uint_t<Bits>::fast
#else
#define CRC_PARM_TYPE unsigned long
#endif
#if !defined(BOOST_MSVC) && !defined(__GNUC__)
#define PRIVATE_DECLARE_BOOST( TypeName ) using boost:: TypeName
#else
#define PRIVATE_DECLARE_BOOST( TypeName ) typedef boost:: TypeName TypeName
#endif
// Types
template < std::size_t Bits, CRC_PARM_TYPE TrPo, CRC_PARM_TYPE InRe,
CRC_PARM_TYPE FiXo, bool ReIn, bool ReRe >
class crc_tester
{
public:
// All the following were separate function templates, but they have
// been moved to class-static member functions of a class template
// because MS VC++ 6 can't handle function templates that can't
// deduce all their template arguments from their function arguments.
typedef typename boost::uint_t<Bits>::fast value_type;
static void master_test( char const *test_name, value_type expected );
private:
typedef boost::crc_optimal<Bits, TrPo, InRe, FiXo, ReIn, ReRe>
optimal_crc_type;
typedef boost::crc_basic<Bits> basic_crc_type;
static void compute_test( value_type expected );
static void interrupt_test( value_type expected );
static void error_test();
}; // crc_tester
// Global data
unsigned char const std_data[] = { 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37,
0x38, 0x39 };
std::size_t const std_data_len = sizeof( std_data ) / sizeof( std_data[0] );
boost::uint16_t const std_crc_ccitt_result = 0x29B1;
boost::uint16_t const std_crc_16_result = 0xBB3D;
boost::uint32_t const std_crc_32_result = 0xCBF43926;
// Function prototypes
void timing_test();
boost::uint32_t basic_crc32( void const *buffer, std::size_t byte_count );
boost::uint32_t optimal_crc32( void const *buffer, std::size_t byte_count );
boost::uint32_t quick_crc32( void const *buffer, std::size_t byte_count );
boost::uint32_t quick_reflect( boost::uint32_t value, std::size_t bits );
double time_trial( char const *name,
boost::uint32_t (*crc_func)(void const *, std::size_t),
boost::uint32_t expected, void const *data, std::size_t length );
void augmented_tests();
boost::uint32_t native_to_big( boost::uint32_t x );
boost::uint32_t big_to_native( boost::uint32_t x );
void small_crc_test1();
void small_crc_test2();
// Macro to compact code
#define PRIVATE_TESTER_NAME crc_tester<Bits, TrPo, InRe, FiXo, ReIn, ReRe>
// Run a test on slow and fast CRC computers and function
template < std::size_t Bits, CRC_PARM_TYPE TrPo, CRC_PARM_TYPE InRe,
CRC_PARM_TYPE FiXo, bool ReIn, bool ReRe >
void
PRIVATE_TESTER_NAME::compute_test
(
typename PRIVATE_TESTER_NAME::value_type expected
)
{
std::cout << "\tDoing computation tests." << std::endl;
optimal_crc_type fast_crc;
basic_crc_type slow_crc( TrPo, InRe, FiXo, ReIn, ReRe );
value_type const func_result = boost::crc<Bits, TrPo, InRe, FiXo, ReIn,
ReRe>( std_data, std_data_len );
fast_crc.process_bytes( std_data, std_data_len );
slow_crc.process_bytes( std_data, std_data_len );
BOOST_TEST_EQ( fast_crc.checksum(), expected );
BOOST_TEST_EQ( slow_crc.checksum(), expected );
BOOST_TEST_EQ( func_result, expected );
}
// Run a test in two runs, and check all the inspectors
template < std::size_t Bits, CRC_PARM_TYPE TrPo, CRC_PARM_TYPE InRe,
CRC_PARM_TYPE FiXo, bool ReIn, bool ReRe >
void
PRIVATE_TESTER_NAME::interrupt_test
(
typename PRIVATE_TESTER_NAME::value_type expected
)
{
std::cout << "\tDoing interrupt tests." << std::endl;
// Process the first half of the data (also test accessors)
optimal_crc_type fast_crc1;
basic_crc_type slow_crc1( fast_crc1.get_truncated_polynominal(),
fast_crc1.get_initial_remainder(), fast_crc1.get_final_xor_value(),
fast_crc1.get_reflect_input(), fast_crc1.get_reflect_remainder() );
BOOST_TEST_EQ( fast_crc1.get_interim_remainder(),
slow_crc1.get_initial_remainder() );
std::size_t const mid_way = std_data_len / 2;
unsigned char const * const std_data_end = std_data + std_data_len;
fast_crc1.process_bytes( std_data, mid_way );
slow_crc1.process_bytes( std_data, mid_way );
BOOST_TEST_EQ( fast_crc1.checksum(), slow_crc1.checksum() );
// Process the second half of the data (also test accessors)
boost::crc_optimal<optimal_crc_type::bit_count,
optimal_crc_type::truncated_polynominal, optimal_crc_type::initial_remainder,
optimal_crc_type::final_xor_value, optimal_crc_type::reflect_input,
optimal_crc_type::reflect_remainder>
fast_crc2( fast_crc1.get_interim_remainder() );
boost::crc_basic<basic_crc_type::bit_count> slow_crc2(
slow_crc1.get_truncated_polynominal(), slow_crc1.get_interim_remainder(),
slow_crc1.get_final_xor_value(), slow_crc1.get_reflect_input(),
slow_crc1.get_reflect_remainder() );
fast_crc2.process_block( std_data + mid_way, std_data_end );
slow_crc2.process_block( std_data + mid_way, std_data_end );
BOOST_TEST_EQ( fast_crc2.checksum(), slow_crc2.checksum() );
BOOST_TEST_EQ( fast_crc2.checksum(), expected );
BOOST_TEST_EQ( slow_crc2.checksum(), expected );
}
// Run a test to see if a single-bit error is detected
template < std::size_t Bits, CRC_PARM_TYPE TrPo, CRC_PARM_TYPE InRe,
CRC_PARM_TYPE FiXo, bool ReIn, bool ReRe >
void
PRIVATE_TESTER_NAME::error_test
(
)
{
PRIVATE_DECLARE_BOOST( uint32_t );
// A single-bit error is ensured to be detected if the polynominal
// has at least two bits set. The highest bit is what is removed
// to give the truncated polynominal, and it is always set. This
// means that the truncated polynominal needs at least one of its
// bits set, which implies that it cannot be zero.
if ( !(TrPo & boost::detail::low_bits_mask_c<Bits>::value) )
{
BOOST_ERROR( "truncated CRC polymonial is zero" );
}
std::cout << "\tDoing error tests." << std::endl;
// Create a random block of data
uint32_t ran_data[ 256 ];
std::size_t const ran_length = sizeof(ran_data) / sizeof(ran_data[0]);
std::generate_n( ran_data, ran_length, boost::minstd_rand() );
// Create computers and compute the checksum of the data
optimal_crc_type fast_tester;
basic_crc_type slow_tester( TrPo, InRe, FiXo, ReIn, ReRe );
fast_tester.process_bytes( ran_data, sizeof(ran_data) );
slow_tester.process_bytes( ran_data, sizeof(ran_data) );
uint32_t const fast_checksum = fast_tester.checksum();
uint32_t const slow_checksum = slow_tester.checksum();
BOOST_TEST_EQ( fast_checksum, slow_checksum );
// Do the checksum again (and test resetting ability)
fast_tester.reset();
slow_tester.reset( InRe );
fast_tester.process_bytes( ran_data, sizeof(ran_data) );
slow_tester.process_bytes( ran_data, sizeof(ran_data) );
BOOST_TEST_EQ( fast_tester.checksum(), slow_tester.checksum() );
BOOST_TEST_EQ( fast_tester.checksum(), fast_checksum );
BOOST_TEST_EQ( slow_tester.checksum(), slow_checksum );
// Produce a single-bit error
ran_data[ ran_data[0] % ran_length ] ^= ( 1 << (ran_data[1] % 32) );
// Compute the checksum of the errorenous data
// (and continue testing resetting ability)
fast_tester.reset( InRe );
slow_tester.reset();
fast_tester.process_bytes( ran_data, sizeof(ran_data) );
slow_tester.process_bytes( ran_data, sizeof(ran_data) );
BOOST_TEST_EQ( fast_tester.checksum(), slow_tester.checksum() );
BOOST_TEST_NE( fast_tester.checksum(), fast_checksum );
BOOST_TEST_NE( slow_tester.checksum(), slow_checksum );
}
// Run the other CRC object tests
template < std::size_t Bits, CRC_PARM_TYPE TrPo, CRC_PARM_TYPE InRe,
CRC_PARM_TYPE FiXo, bool ReIn, bool ReRe >
void
PRIVATE_TESTER_NAME::master_test
(
char const * test_name,
typename PRIVATE_TESTER_NAME::value_type expected
)
{
std::cout << "Doing test suite for " << test_name << '.' << std::endl;
compute_test( expected );
interrupt_test( expected );
error_test();
}
// Undo limited macros
#undef PRIVATE_TESTER_NAME
// A CRC-32 computer based on crc_basic, for timing
boost::uint32_t
basic_crc32
(
void const * buffer,
std::size_t byte_count
)
{
static boost::crc_basic<32> computer( 0x04C11DB7, 0xFFFFFFFF, 0xFFFFFFFF,
true, true );
computer.reset();
computer.process_bytes( buffer, byte_count );
return computer.checksum();
}
// A CRC-32 computer based on crc_optimal, for timing
inline
boost::uint32_t
optimal_crc32
(
void const * buffer,
std::size_t byte_count
)
{
static boost::crc_32_type computer;
computer.reset();
computer.process_bytes( buffer, byte_count );
return computer.checksum();
}
// Reflect the lower "bits" bits of a "value"
boost::uint32_t
quick_reflect
(
boost::uint32_t value,
std::size_t bits
)
{
boost::uint32_t reflection = 0;
for ( std::size_t i = 0 ; i < bits ; ++i )
{
if ( value & (1u << i) )
{
reflection |= 1 << ( bits - 1 - i );
}
}
return reflection;
}
// A customized CRC-32 computer, for timing
boost::uint32_t
quick_crc32
(
void const * buffer,
std::size_t byte_count
)
{
PRIVATE_DECLARE_BOOST( uint32_t );
typedef unsigned char byte_type;
// Compute the CRC table (first run only)
static bool did_init = false;
static uint32_t crc_table[ 1ul << CHAR_BIT ];
if ( !did_init )
{
uint32_t const value_high_bit = static_cast<uint32_t>(1) << 31u;
byte_type dividend = 0;
do
{
uint32_t remainder = 0;
for ( byte_type mask = 1u << (CHAR_BIT - 1u) ; mask ; mask >>= 1 )
{
if ( dividend & mask )
{
remainder ^= value_high_bit;
}
if ( remainder & value_high_bit )
{
remainder <<= 1;
remainder ^= 0x04C11DB7u;
}
else
{
remainder <<= 1;
}
}
crc_table[ quick_reflect(dividend, CHAR_BIT) ]
= quick_reflect( remainder, 32 );
}
while ( ++dividend );
did_init = true;
}
// Compute the CRC of the data
uint32_t rem = 0xFFFFFFFF;
byte_type const * const b_begin = static_cast<byte_type const *>( buffer );
byte_type const * const b_end = b_begin + byte_count;
for ( byte_type const *p = b_begin ; p < b_end ; ++p )
{
byte_type const byte_index = *p ^ rem;
rem >>= CHAR_BIT;
rem ^= crc_table[ byte_index ];
}
return ~rem;
}
// Run an individual timing trial
double
time_trial
(
char const * name,
boost::uint32_t (*crc_func)(void const *, std::size_t),
boost::uint32_t expected,
void const * data,
std::size_t length
)
{
PRIVATE_DECLARE_BOOST( uint32_t );
using std::cout;
// Limits of a trial
static uint32_t const max_count = 1L << 16; // ~square-root of max
static double const max_time = 3.14159; // easy as pi(e)
// Mark the trial
cout << '\t' << name << " CRC-32: ";
// Trial loop
uint32_t trial_count = 0, wrong_count = 0;
double elapsed_time = 0.0;
boost::timer t;
do
{
uint32_t const scratch = (*crc_func)( data, length );
if ( scratch != expected )
{
++wrong_count;
}
elapsed_time = t.elapsed();
++trial_count;
} while ( (trial_count < max_count) && (elapsed_time < max_time) );
if ( wrong_count )
{
BOOST_ERROR( "at least one time trial didn't match expected" );
}
// Report results
double const rate = trial_count / elapsed_time;
cout << trial_count << " runs, " << elapsed_time << " s, " << rate
<< " run/s" << std::endl;
return rate;
}
// Time runs of Boost CRCs vs. a customized CRC function
void
timing_test
(
)
{
PRIVATE_DECLARE_BOOST( uint32_t );
using std::cout;
using std::endl;
cout << "Doing timing tests." << endl;
// Create a random block of data
boost::int32_t ran_data[ 256 ];
std::size_t const ran_length = sizeof(ran_data) / sizeof(ran_data[0]);
std::generate_n( ran_data, ran_length, boost::minstd_rand() );
// Use the first runs as a check. This gives a chance for first-
// time static initialization to not interfere in the timings.
uint32_t const basic_result = basic_crc32( ran_data, sizeof(ran_data) );
uint32_t const optimal_result = optimal_crc32( ran_data, sizeof(ran_data) );
uint32_t const quick_result = quick_crc32( ran_data, sizeof(ran_data) );
BOOST_TEST_EQ( basic_result, optimal_result );
BOOST_TEST_EQ( optimal_result, quick_result );
BOOST_TEST_EQ( quick_result, basic_result );
// Run trials
double const basic_rate = time_trial( "Boost-Basic", basic_crc32,
basic_result, ran_data, sizeof(ran_data) );
double const optimal_rate = time_trial( "Boost-Optimal", optimal_crc32,
optimal_result, ran_data, sizeof(ran_data) );
double const quick_rate = time_trial( "Reference", quick_crc32,
quick_result, ran_data, sizeof(ran_data) );
// Report results
cout << "\tThe optimal Boost version has " << optimal_rate / quick_rate *
100.0 << "% the speed of the reference version.\n";
cout << "\tThe basic Boost version has " << basic_rate / quick_rate * 100.0
<< "% the speed of the reference version.\n";
cout << "\tThe basic Boost version has " << basic_rate / optimal_rate *
100.0 << "% the speed of the optimal Boost version."
<< endl;
}
// Reformat an integer to the big-endian storage format
boost::uint32_t
native_to_big
(
boost::uint32_t x
)
{
boost::uint32_t temp;
unsigned char * tp = reinterpret_cast<unsigned char *>( &temp );
for ( std::size_t i = sizeof(x) ; i > 0 ; --i )
{
tp[ i - 1 ] = static_cast<unsigned char>( x );
x >>= CHAR_BIT;
}
return temp;
}
// Restore an integer from the big-endian storage format
boost::uint32_t
big_to_native
(
boost::uint32_t x
)
{
boost::uint32_t temp = 0;
unsigned char * xp = reinterpret_cast<unsigned char *>( &x );
for ( std::size_t i = 0 ; i < sizeof(x) ; ++i )
{
temp <<= CHAR_BIT;
temp |= xp[ i ];
}
return temp;
}
// Run tests on using CRCs on augmented messages
void
augmented_tests
(
)
{
#define PRIVATE_ACRC_FUNC boost::augmented_crc<32, 0x04C11DB7>
using std::size_t;
PRIVATE_DECLARE_BOOST( uint32_t );
std::cout << "Doing CRC-augmented message tests." << std::endl;
// Create a random block of data, with space for a CRC.
uint32_t ran_data[ 257 ];
size_t const ran_length = sizeof(ran_data) / sizeof(ran_data[0]);
size_t const data_length = ran_length - 1;
std::generate_n( ran_data, data_length, boost::minstd_rand() );
// When creating a CRC for an augmented message, use
// zeros in the appended CRC spot for the first run.
uint32_t & ran_crc = ran_data[ data_length ];
ran_crc = 0;
// Compute the CRC with augmented-CRC computing function
typedef boost::uint_t<32>::fast return_type;
ran_crc = PRIVATE_ACRC_FUNC( ran_data, sizeof(ran_data) );
// With the appended CRC set, running the checksum again should get zero.
// NOTE: CRC algorithm assumes numbers are in big-endian format
ran_crc = native_to_big( ran_crc );
uint32_t ran_crc_check = PRIVATE_ACRC_FUNC( ran_data, sizeof(ran_data) );
BOOST_TEST_EQ( 0, ran_crc_check );
// Compare that result with other CRC computing functions
// and classes, which don't accept augmented messages.
typedef boost::crc_optimal<32, 0x04C11DB7> fast_crc_type;
typedef boost::crc_basic<32> slow_crc_type;
fast_crc_type fast_tester;
slow_crc_type slow_tester( 0x04C11DB7 );
size_t const data_size = data_length * sizeof(ran_data[0]);
uint32_t const func_tester = boost::crc<32, 0x04C11DB7, 0, 0, false,
false>( ran_data, data_size );
fast_tester.process_bytes( ran_data, data_size );
slow_tester.process_bytes( ran_data, data_size );
BOOST_TEST_EQ( fast_tester.checksum(), slow_tester.checksum() );
ran_crc = big_to_native( ran_crc );
BOOST_TEST_EQ( fast_tester.checksum(), ran_crc );
BOOST_TEST_EQ( func_tester, ran_crc );
// Do a single-bit error test
ran_crc = native_to_big( ran_crc );
ran_data[ ran_data[0] % ran_length ] ^= ( 1 << (ran_data[1] % 32) );
ran_crc_check = PRIVATE_ACRC_FUNC( ran_data, sizeof(ran_data) );
BOOST_TEST_NE( 0, ran_crc_check );
// Run a version of these tests with a nonzero initial remainder.
uint32_t const init_rem = ran_data[ ran_data[2] % ran_length ];
ran_crc = 0;
ran_crc = PRIVATE_ACRC_FUNC( ran_data, sizeof(ran_data), init_rem );
// Have some fun by processing data in two steps.
size_t const mid_index = ran_length / 2;
ran_crc = native_to_big( ran_crc );
ran_crc_check = PRIVATE_ACRC_FUNC( ran_data, mid_index
* sizeof(ran_data[0]), init_rem );
ran_crc_check = PRIVATE_ACRC_FUNC( &ran_data[mid_index], sizeof(ran_data)
- mid_index * sizeof(ran_data[0]), ran_crc_check );
BOOST_TEST_EQ( 0, ran_crc_check );
// This substep translates an augmented-CRC initial
// remainder to an unaugmented-CRC initial remainder.
uint32_t const zero = 0;
uint32_t const new_init_rem = PRIVATE_ACRC_FUNC( &zero, sizeof(zero),
init_rem );
slow_crc_type slow_tester2( 0x04C11DB7, new_init_rem );
slow_tester2.process_bytes( ran_data, data_size );
ran_crc = big_to_native( ran_crc );
BOOST_TEST_EQ( slow_tester2.checksum(), ran_crc );
// Redo single-bit error test
ran_data[ ran_data[3] % ran_length ] ^= ( 1 << (ran_data[4] % 32) );
ran_crc_check = PRIVATE_ACRC_FUNC( ran_data, sizeof(ran_data), init_rem );
BOOST_TEST_NE( 0, ran_crc_check );
#undef PRIVATE_ACRC_FUNC
}
// Run tests on CRCs below a byte in size (here, 3 bits)
void
small_crc_test1
(
)
{
std::cout << "Doing short-CRC (3-bit augmented) message tests."
<< std::endl;
// The CRC standard is a SDH/SONET Low Order LCAS control word with CRC-3
// taken from ITU-T G.707 (12/03) XIII.2.
// Four samples, each four bytes; should all have a CRC of zero
unsigned char const samples[4][4]
= {
{ 0x3A, 0xC4, 0x08, 0x06 },
{ 0x42, 0xC5, 0x0A, 0x41 },
{ 0x4A, 0xC5, 0x08, 0x22 },
{ 0x52, 0xC4, 0x08, 0x05 }
};
// Basic computer
boost::crc_basic<3> tester1( 0x03 );
tester1.process_bytes( samples[0], 4 );
BOOST_TEST_EQ( tester1.checksum(), 0 );
tester1.reset();
tester1.process_bytes( samples[1], 4 );
BOOST_TEST_EQ( tester1.checksum(), 0 );
tester1.reset();
tester1.process_bytes( samples[2], 4 );
BOOST_TEST_EQ( tester1.checksum(), 0 );
tester1.reset();
tester1.process_bytes( samples[3], 4 );
BOOST_TEST_EQ( tester1.checksum(), 0 );
// Optimal computer
#define PRIVATE_CRC_FUNC boost::crc<3, 0x03, 0, 0, false, false>
#define PRIVATE_ACRC_FUNC boost::augmented_crc<3, 0x03>
BOOST_TEST_EQ( 0, PRIVATE_CRC_FUNC(samples[0], 4) );
BOOST_TEST_EQ( 0, PRIVATE_CRC_FUNC(samples[1], 4) );
BOOST_TEST_EQ( 0, PRIVATE_CRC_FUNC(samples[2], 4) );
BOOST_TEST_EQ( 0, PRIVATE_CRC_FUNC(samples[3], 4) );
// maybe the fix to CRC functions needs to be applied to augmented CRCs?
#undef PRIVATE_ACRC_FUNC
#undef PRIVATE_CRC_FUNC
}
// Run tests on CRCs below a byte in size (here, 7 bits)
void
small_crc_test2
(
)
{
std::cout << "Doing short-CRC (7-bit augmented) message tests."
<< std::endl;
// The CRC standard is a SDH/SONET J0/J1/J2/N1/N2/TR TTI (trace message)
// with CRC-7, o.a. ITU-T G.707 Annex B, G.832 Annex A.
// Two samples, each sixteen bytes
// Sample 1 is '\x80' + ASCII("123456789ABCDEF")
// Sample 2 is '\x80' + ASCII("TTI UNAVAILABLE")
unsigned char const samples[2][16]
= {
{ 0x80, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x41,
0x42, 0x43, 0x44, 0x45, 0x46 },
{ 0x80, 0x54, 0x54, 0x49, 0x20, 0x55, 0x4E, 0x41, 0x56, 0x41, 0x49,
0x4C, 0x41, 0x42, 0x4C, 0x45 }
};
unsigned const results[2] = { 0x62, 0x23 };
// Basic computer
boost::crc_basic<7> tester1( 0x09 );
tester1.process_bytes( samples[0], 16 );
BOOST_TEST_EQ( tester1.checksum(), results[0] );
tester1.reset();
tester1.process_bytes( samples[1], 16 );
BOOST_TEST_EQ( tester1.checksum(), results[1] );
// Optimal computer
#define PRIVATE_CRC_FUNC boost::crc<7, 0x09, 0, 0, false, false>
#define PRIVATE_ACRC_FUNC boost::augmented_crc<7, 0x09>
BOOST_TEST_EQ( results[0], PRIVATE_CRC_FUNC(samples[0], 16) );
BOOST_TEST_EQ( results[1], PRIVATE_CRC_FUNC(samples[1], 16) );
// maybe the fix to CRC functions needs to be applied to augmented CRCs?
#undef PRIVATE_ACRC_FUNC
#undef PRIVATE_CRC_FUNC
}
#ifndef BOOST_MSVC
// Explicit template instantiations
// (needed to fix a link error in Metrowerks CodeWarrior Pro 5.3)
template class crc_tester<16, 0x1021, 0xFFFF, 0, false, false>;
template class crc_tester<16, 0x8005, 0, 0, true, true>;
template class crc_tester<32, 0x04C11DB7, 0xFFFFFFFF, 0xFFFFFFFF, true, true>;
#endif
// Main testing function
int main()
{
using std::cout;
using std::endl;
// Run simulations on some CRC types
typedef crc_tester<16, 0x1021, 0xFFFF, 0, false, false> crc_ccitt_tester;
typedef crc_tester<16, 0x8005, 0, 0, true, true> crc_16_tester;
typedef crc_tester<32, 0x04C11DB7, 0xFFFFFFFF, 0xFFFFFFFF, true, true>
crc_32_tester;
crc_ccitt_tester::master_test( "CRC-CCITT", std_crc_ccitt_result );
crc_16_tester::master_test( "CRC-16", std_crc_16_result );
crc_32_tester::master_test( "CRC-32", std_crc_32_result );
// Run a timing comparison test
timing_test();
// Test using augmented messages
augmented_tests();
// Test with CRC types smaller than a byte
small_crc_test1();
small_crc_test2();
// Try a CRC based on the (x + 1) polynominal, which is a factor in
// many real-life polynominals and doesn't fit evenly in a byte.
cout << "Doing one-bit polynominal CRC test." << endl;
boost::crc_basic<1> crc_1( 1 );
crc_1.process_bytes( std_data, std_data_len );
BOOST_TEST_EQ( crc_1.checksum(), 1 );
// Test the function object interface
cout << "Doing functional object interface test." << endl;
boost::crc_optimal<16, 0x8005, 0, 0, true, true> crc_16;
crc_16 = std::for_each( std_data, std_data + std_data_len, crc_16 );
BOOST_TEST_EQ( crc_16(), std_crc_16_result );
return boost::report_errors();
}

View File

@@ -0,0 +1,637 @@
// Boost CRC unit test program file ----------------------------------------//
// Copyright 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.
#include <boost/core/lightweight_test.hpp>
#include <boost/crc.hpp> // for boost::crc_basic,crc_optimal,augmented_crc,crc
#include <boost/cstdint.hpp> // for boost::uint16_t, uint32_t, uintmax_t
#include <boost/predef/other/endian.h>
#include <boost/integer.hpp> // for boost::uint_t
#include <boost/typeof/typeof.hpp> // for BOOST_AUTO
#include <boost/random/linear_congruential.hpp> // for boost::minstd_rand
#include <algorithm> // for std::generate_n, for_each
#include <climits> // for CHAR_BIT
#include <cstddef> // for std::size_t
// Sanity check
#if CHAR_BIT != 8
#error The expected results assume octet-sized bytes.
#endif
// Control tests at compile-time
#ifndef CONTROL_SUB_BYTE_MISMATCHED_REFLECTION_TEST
#define CONTROL_SUB_BYTE_MISMATCHED_REFLECTION_TEST 1
#endif
// Common definitions -------------------------------------------------------//
namespace {
// Many CRC configurations use the string "123456789" in ASCII as test data.
unsigned char const std_data[] = { 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37,
0x38, 0x39 };
std::size_t const std_data_len = sizeof( std_data ) / sizeof( std_data[0] );
// Checksums of the standard test data for common configurations
boost::uint16_t const std_crc_ccitt_false_result = 0x29B1u;
boost::uint16_t const std_crc_ccitt_true_result = 0x2189u;
boost::uint16_t const std_crc_xmodem_result = 0x31C3u;
boost::uint16_t const std_crc_16_result = 0xBB3Du;
boost::uint32_t const std_crc_32_result = 0xCBF43926ul;
// Conversion functions between native- and big-endian representations
#if BOOST_ENDIAN_BIG_BYTE
boost::uint32_t native_to_big( boost::uint32_t x ) { return x; }
boost::uint32_t big_to_native( boost::uint32_t x ) { return x; }
#else
union endian_convert
{
boost::uint32_t w;
unsigned char p[ 4 ];
};
boost::uint32_t native_to_big( boost::uint32_t x )
{
endian_convert e;
e.p[ 0 ] = x >> 24;
e.p[ 1 ] = x >> 16;
e.p[ 2 ] = x >> 8;
e.p[ 3 ] = x;
return e.w;
}
boost::uint32_t big_to_native( boost::uint32_t x )
{
endian_convert e;
e.w = x;
x = e.p[ 0 ];
x <<= 8;
x |= e.p[ 1 ];
x <<= 8;
x |= e.p[ 2 ];
x <<= 8;
x |= e.p[ 3 ];
return x;
}
#endif
// Define CRC parameters inside traits classes. Probably will use this in a
// future version of the CRC libray!
template < std::size_t Bits >
class my_crc_rt_traits
{
public:
typedef boost::integral_constant<std::size_t, Bits> register_length_c;
typedef typename boost::uint_t<Bits>::fast register_type;
typedef boost::crc_basic<Bits> computer_type;
register_type divisor_polynominal;
register_type initial_remainder;
bool reflect_input_byte;
bool reflect_output_remainder;
register_type final_xor_mask;
computer_type make_crc_basic() const
{ return computer_type(divisor_polynominal, initial_remainder,
final_xor_mask, reflect_input_byte, reflect_output_remainder); }
};
template < std::size_t Bits, boost::uintmax_t DivisorPolynominal,
boost::uintmax_t InitialRemainder, bool ReflectInputBytes,
bool ReflectOutputRemainder, boost::uintmax_t FinalXorMask >
class my_crc_ct_traits
{
public:
typedef my_crc_rt_traits<Bits> rt_adaptor_type;
typedef typename rt_adaptor_type::register_type register_type;
typedef boost::crc_optimal<Bits, DivisorPolynominal, InitialRemainder,
FinalXorMask, ReflectInputBytes, ReflectOutputRemainder> computer_type;
typedef boost::integral_constant<std::size_t, Bits> register_length_c;
typedef boost::integral_constant<register_type, DivisorPolynominal>
divisor_polynominal_c;
typedef boost::integral_constant<register_type, InitialRemainder>
initial_remainder_c;
typedef boost::integral_constant<bool, ReflectInputBytes> reflect_input_byte_c;
typedef boost::integral_constant<bool, ReflectOutputRemainder>
reflect_output_remainder_c;
typedef boost::integral_constant<register_type, FinalXorMask>
final_xor_mask_c;
operator rt_adaptor_type() const
{
rt_adaptor_type const result = { divisor_polynominal_c::value,
initial_remainder_c::value, reflect_input_byte_c::value,
reflect_output_remainder_c::value, final_xor_mask_c::value };
return result;
}
static computer_type make_crc_optimal()
{ return boost::crc_optimal<register_length_c::value,
divisor_polynominal_c::value, initial_remainder_c::value,
final_xor_mask_c::value, reflect_input_byte_c::value,
reflect_output_remainder_c::value>(); }
};
template < std::size_t Bits, boost::uintmax_t DivisorPolynominal,
boost::uintmax_t InitialRemainder, bool ReflectInputBytes,
bool ReflectOutputRemainder, boost::uintmax_t FinalXorMask,
boost::uintmax_t StandardTestDataResult >
class my_crc_test_traits
{
public:
typedef my_crc_ct_traits<Bits, DivisorPolynominal, InitialRemainder,
ReflectInputBytes, ReflectOutputRemainder, FinalXorMask> ct_traits_type;
typedef my_crc_rt_traits<Bits> rt_traits_type;
typedef typename rt_traits_type::register_type register_type;
typedef boost::integral_constant<std::size_t, Bits> register_length_c;
typedef boost::integral_constant<register_type, DivisorPolynominal>
divisor_polynominal_c;
typedef boost::integral_constant<register_type, InitialRemainder>
initial_remainder_c;
typedef boost::integral_constant<bool, ReflectInputBytes> reflect_input_byte_c;
typedef boost::integral_constant<bool, ReflectOutputRemainder>
reflect_output_remainder_c;
typedef boost::integral_constant<register_type, FinalXorMask>
final_xor_mask_c;
typedef boost::integral_constant<register_type, StandardTestDataResult>
standard_test_data_CRC_c;
typedef typename ct_traits_type::computer_type computer_ct_type;
typedef typename rt_traits_type::computer_type computer_rt_type;
static computer_ct_type make_crc_optimal()
{ return ct_traits_type::make_crc_optimal(); }
static computer_rt_type make_crc_basic()
{ return ct_traits_type().operator rt_traits_type().make_crc_basic(); }
};
// Now make some example CRC profiles
typedef my_crc_test_traits<16u, 0x8005u, 0u, true, true, 0u, std_crc_16_result>
my_crc_16_traits;
typedef my_crc_test_traits<16u, 0x1021u, 0xFFFFu, false, false, 0u,
std_crc_ccitt_false_result> my_crc_ccitt_false_traits;
typedef my_crc_test_traits<16u, 0x1021u, 0u, true, true, 0u,
std_crc_ccitt_true_result> my_crc_ccitt_true_traits;
typedef my_crc_test_traits<16u, 0x1021u, 0u, false, false, 0u,
std_crc_xmodem_result> my_crc_xmodem_traits;
typedef my_crc_test_traits<32u, 0x04C11DB7ul, 0xFFFFFFFFul, true, true,
0xFFFFFFFFul, std_crc_32_result> my_crc_32_traits;
template<class Test>
void run_crc_test_policies()
{
Test()(my_crc_16_traits());
Test()(my_crc_ccitt_false_traits());
Test()(my_crc_ccitt_true_traits());
Test()(my_crc_xmodem_traits());
Test()(my_crc_32_traits());
}
// Need to test when ReflectInputBytes and ReflectOutputRemainder differ
// (Grabbed from table at <http://regregex.bbcmicro.net/crc-catalogue.htm>.)
typedef my_crc_test_traits<6u, 0x19u, 0u, true, false, 0u, 0x19u>
my_crc_6_darc_traits;
typedef my_crc_test_traits<12u, 0x80Fu, 0u, false, true, 0u, 0xDAFu>
my_crc_12_3gpp_traits;
template<class Test>
void run_crc_extended_test_policies()
{
Test()(my_crc_16_traits());
Test()(my_crc_ccitt_false_traits());
Test()(my_crc_ccitt_true_traits());
Test()(my_crc_xmodem_traits());
Test()(my_crc_32_traits());
#if CONTROL_SUB_BYTE_MISMATCHED_REFLECTION_TEST
Test()(my_crc_6_darc_traits());
#endif
Test()(my_crc_12_3gpp_traits());
}
// Bit mask constants
template < std::size_t BitIndex >
struct high_bit_mask_c
: boost::detail::high_bit_mask_c<BitIndex>
{};
template < std::size_t BitCount >
struct low_bits_mask_c
: boost::detail::low_bits_mask_c<BitCount>
{};
} // anonymous namespace
// Unit tests ---------------------------------------------------------------//
struct computation_comparison_test {
template<class CRCPolicy>
void operator()(CRCPolicy)
{
BOOST_AUTO( crc_f, CRCPolicy::make_crc_optimal() );
BOOST_AUTO( crc_s, CRCPolicy::make_crc_basic() );
typename CRCPolicy::register_type const func_result
= boost::crc<CRCPolicy::register_length_c::value,
CRCPolicy::divisor_polynominal_c::value,
CRCPolicy::initial_remainder_c::value,
CRCPolicy::final_xor_mask_c::value,
CRCPolicy::reflect_input_byte_c::value,
CRCPolicy::reflect_output_remainder_c::value>( std_data, std_data_len );
crc_f.process_bytes( std_data, std_data_len );
crc_s.process_bytes( std_data, std_data_len );
BOOST_TEST_EQ( crc_f.checksum(),
CRCPolicy::standard_test_data_CRC_c::value );
BOOST_TEST_EQ( crc_s.checksum(),
CRCPolicy::standard_test_data_CRC_c::value );
BOOST_TEST_EQ( CRCPolicy::standard_test_data_CRC_c::value,
func_result );
}
};
struct accessor_and_split_run_test {
template<class CRCPolicy>
void operator()(CRCPolicy)
{
typedef typename CRCPolicy::computer_ct_type optimal_crc_type;
typedef typename CRCPolicy::computer_rt_type basic_crc_type;
// Test accessors
optimal_crc_type faster_crc1;
basic_crc_type slower_crc1( faster_crc1.get_truncated_polynominal(),
faster_crc1.get_initial_remainder(), faster_crc1.get_final_xor_value(),
faster_crc1.get_reflect_input(), faster_crc1.get_reflect_remainder() );
BOOST_TEST_EQ( faster_crc1.get_interim_remainder(),
slower_crc1.get_initial_remainder() );
// Process the first half of the standard data
std::size_t const mid_way = std_data_len / 2u;
faster_crc1.process_bytes( std_data, mid_way );
slower_crc1.process_bytes( std_data, mid_way );
BOOST_TEST_EQ( faster_crc1.checksum(), slower_crc1.checksum() );
// Process the second half of the standard data, testing more accessors
unsigned char const * const std_data_end = std_data + std_data_len;
boost::crc_optimal<optimal_crc_type::bit_count,
optimal_crc_type::truncated_polynominal,
optimal_crc_type::initial_remainder, optimal_crc_type::final_xor_value,
optimal_crc_type::reflect_input, optimal_crc_type::reflect_remainder>
faster_crc2( faster_crc1.get_interim_remainder() );
boost::crc_basic<basic_crc_type::bit_count> slower_crc2(
slower_crc1.get_truncated_polynominal(),
slower_crc1.get_interim_remainder(), slower_crc1.get_final_xor_value(),
slower_crc1.get_reflect_input(), slower_crc1.get_reflect_remainder() );
faster_crc2.process_block( std_data + mid_way, std_data_end );
slower_crc2.process_block( std_data + mid_way, std_data_end );
BOOST_TEST_EQ( slower_crc2.checksum(), faster_crc2.checksum() );
BOOST_TEST_EQ( faster_crc2.checksum(),
CRCPolicy::standard_test_data_CRC_c::value );
BOOST_TEST_EQ( CRCPolicy::standard_test_data_CRC_c::value,
slower_crc2.checksum() );
}
};
struct reset_and_single_bit_error_test {
template<class CRCPolicy>
void operator()(CRCPolicy)
{
// A single-bit error in a CRC can be guaranteed to be detected if the
// modulo-2 polynomial divisor has at least two non-zero coefficients. The
// implicit highest coefficient is always one, so that leaves an explicit
// coefficient, i.e. at least one of the polynomial's bits is set.
BOOST_TEST( CRCPolicy::divisor_polynominal_c::value &
low_bits_mask_c<CRCPolicy::register_length_c::value>::value );
// Create a random block of data
boost::uint32_t ran_data[ 256 ];
std::size_t const ran_length = sizeof(ran_data) / sizeof(ran_data[0]);
std::generate_n( ran_data, ran_length, boost::minstd_rand() );
// Create computers and compute the checksum of the data
BOOST_AUTO( optimal_tester, CRCPolicy::make_crc_optimal() );
BOOST_AUTO( basic_tester, CRCPolicy::make_crc_basic() );
optimal_tester.process_bytes( ran_data, sizeof(ran_data) );
basic_tester.process_bytes( ran_data, sizeof(ran_data) );
BOOST_AUTO( const optimal_checksum, optimal_tester.checksum() );
BOOST_AUTO( const basic_checksum, basic_tester.checksum() );
BOOST_TEST_EQ( optimal_checksum, basic_checksum );
// Do the checksum again, while testing the capability to reset the current
// remainder (to either a default or a given value)
optimal_tester.reset();
basic_tester.reset( CRCPolicy::initial_remainder_c::value );
optimal_tester.process_bytes( ran_data, sizeof(ran_data) );
basic_tester.process_bytes( ran_data, sizeof(ran_data) );
BOOST_TEST_EQ( optimal_tester.checksum(), basic_tester.checksum() );
BOOST_TEST_EQ( optimal_tester.checksum(), optimal_checksum );
BOOST_TEST_EQ( basic_tester.checksum(), basic_checksum );
// Introduce a single-bit error
ran_data[ ran_data[0] % ran_length ] ^= ( 1u << (ran_data[ 1 ] % 32u) );
// Compute the checksum of the errorenous data, while continuing to test
// the remainder-resetting methods
optimal_tester.reset( CRCPolicy::initial_remainder_c::value );
basic_tester.reset();
optimal_tester.process_bytes( ran_data, sizeof(ran_data) );
basic_tester.process_bytes( ran_data, sizeof(ran_data) );
BOOST_TEST_EQ( basic_tester.checksum(), optimal_tester.checksum() );
BOOST_TEST_NE( optimal_checksum, optimal_tester.checksum() );
BOOST_TEST_NE( basic_checksum, basic_tester.checksum() );
}
};
void augmented_crc_test()
{
using std::size_t;
using boost::uint32_t;
using boost::augmented_crc;
// Common CRC parameters, all others are zero/false
static size_t const bits = 32u;
static uint32_t const poly = 0x04C11DB7ul;
// Create a random block of data, with space at the end for a CRC
static size_t const data_length = 256u;
static size_t const run_length = data_length + 1u;
uint32_t run_data[ run_length ];
uint32_t & run_crc = run_data[ data_length ];
size_t const data_size = sizeof( run_data ) - sizeof( run_crc );
std::generate_n( run_data, data_length, boost::minstd_rand() );
run_crc = 0u;
// The augmented-CRC routine needs to push an appropriate number of zero
// bits (the register size) through before the checksum can be extracted.
// The other CRC methods, which are un-augmented, don't need to do this.
uint32_t const checksum = boost::crc<bits, poly, 0u, 0u, false, false>(
run_data, data_size );
BOOST_TEST_EQ( (augmented_crc<bits, poly>)(run_data, sizeof( run_data
)), checksum );
// Now appending a message's CRC to the message should lead to a zero-value
// checksum. Note that the CRC must be read from the largest byte on down,
// i.e. big-endian!
run_crc = native_to_big( checksum );
BOOST_TEST_EQ( (augmented_crc<bits, poly>)(run_data, sizeof( run_data
)), 0u );
// Check again with the non-augmented methods
boost::crc_basic<bits> crc_b( poly );
crc_b.process_bytes( run_data, data_size );
BOOST_TEST_EQ( crc_b.checksum(), checksum );
// Introduce a single-bit error, now the checksum shouldn't match!
uint32_t const affected_word_index = run_data[ 0 ] % data_length;
uint32_t const affected_bit_index = run_data[ 1 ] % 32u;
uint32_t const affecting_mask = 1ul << affected_bit_index;
run_data[ affected_word_index ] ^= affecting_mask;
crc_b.reset();
crc_b.process_bytes( run_data, data_size );
BOOST_TEST_NE( crc_b.checksum(), checksum );
BOOST_TEST_NE( (augmented_crc<bits, poly>)(run_data, sizeof( run_data )),
0u );
run_crc = 0u;
BOOST_TEST_NE( (augmented_crc<bits, poly>)(run_data, sizeof( run_data )),
checksum );
// Now introduce the single error in the checksum instead
run_data[ affected_word_index ] ^= affecting_mask;
run_crc = native_to_big( checksum ) ^ affecting_mask;
BOOST_TEST_NE( (augmented_crc<bits, poly>)(run_data, sizeof( run_data )),
0u );
// Repeat these tests with a non-zero initial remainder. Before we can
// check the results against a non-augmented CRC computer, realize that they
// interpret the inital remainder differently. However, the two standards
// can convert between each other.
// (checksum2 initial value is as a scratch pad. So are the address and new
// value of run_crc, but it's also useful for the next sub-step.)
// (TODO: getting the equivalent unaugmented-CRC initial-remainder given an
// augmented-CRC initial-remainder is done by putting said augmented-CRC
// initial-remainder through the augmented-CRC computation with a
// zero-value message. I don't know how to go the other way, yet.)
run_crc = 0u;
uint32_t checksum2 = run_data[ run_data[2] % data_length ];
uint32_t const initial_residue = checksum2 + !checksum2; // ensure nonzero
uint32_t const initial_residue_unaugmented = augmented_crc<bits, poly>(
&run_crc, sizeof(run_crc), initial_residue );
BOOST_TEST_NE( initial_residue, 0u );
crc_b.reset( initial_residue_unaugmented );
crc_b.process_bytes( run_data, data_size );
checksum2 = crc_b.checksum();
BOOST_TEST_EQ( run_crc, 0u );
BOOST_TEST_EQ( (augmented_crc<bits, poly>)(run_data, sizeof( run_data ),
initial_residue), checksum2 );
run_crc = native_to_big( checksum2 );
BOOST_TEST_EQ( (augmented_crc<bits, poly>)(run_data, sizeof( run_data ),
initial_residue), 0u );
// Use the inital remainder argument to split a CRC-computing run
size_t const split_index = data_length / 2u;
uint32_t const intermediate = augmented_crc<bits, poly>( run_data,
sizeof(run_crc) * split_index, initial_residue );
BOOST_TEST_EQ( (augmented_crc<bits, poly>)(&run_data[ split_index ],
sizeof( run_data ) - sizeof( run_crc ) * split_index, intermediate), 0u );
run_crc = 0u;
BOOST_TEST_EQ( (augmented_crc<bits, poly>)(&run_data[ split_index ],
sizeof( run_data ) - sizeof( run_crc ) * split_index, intermediate),
checksum2 );
// Repeat the single-bit error test, with a non-zero initial-remainder
run_data[ run_data[3] % data_length ] ^= ( 1ul << (run_data[4] % 32u) );
run_crc = native_to_big( checksum2 );
BOOST_TEST_NE( (augmented_crc<bits, poly>)(run_data, sizeof( run_data ),
initial_residue), 0u );
}
// Optimal computer, via the single-run function
unsigned crc_f1( const void * buffer, std::size_t byte_count )
{
return boost::crc<3u, 0x03u, 0u, 0u, false, false>( buffer, byte_count );
}
void sub_nybble_polynominal_test()
{
// The CRC standard is a SDH/SONET Low Order LCAS control word with CRC-3
// taken from ITU-T G.707 (12/03) XIII.2.
// Four samples, each four bytes; should all have a CRC of zero
unsigned char const samples[4][4]
= {
{ 0x3Au, 0xC4u, 0x08u, 0x06u },
{ 0x42u, 0xC5u, 0x0Au, 0x41u },
{ 0x4Au, 0xC5u, 0x08u, 0x22u },
{ 0x52u, 0xC4u, 0x08u, 0x05u }
};
// Basic computer
boost::crc_basic<3u> crc_1( 0x03u );
crc_1.process_bytes( samples[0], 4u );
BOOST_TEST_EQ( crc_1.checksum(), 0u );
crc_1.reset();
crc_1.process_bytes( samples[1], 4u );
BOOST_TEST_EQ( crc_1.checksum(), 0u );
crc_1.reset();
crc_1.process_bytes( samples[2], 4u );
BOOST_TEST_EQ( crc_1.checksum(), 0u );
crc_1.reset();
crc_1.process_bytes( samples[3], 4u );
BOOST_TEST_EQ( crc_1.checksum(), 0u );
BOOST_TEST_EQ( crc_f1(samples[ 0 ], 4u), 0u );
BOOST_TEST_EQ( crc_f1(samples[ 1 ], 4u), 0u );
BOOST_TEST_EQ( crc_f1(samples[ 2 ], 4u), 0u );
BOOST_TEST_EQ( crc_f1(samples[ 3 ], 4u), 0u );
// TODO: do similar tests with boost::augmented_crc<3, 0x03>
// (Now I think that this can't be done right now, since that function reads
// byte-wise, so the register size needs to be a multiple of CHAR_BIT.)
}
// Optimal computer, via the single-run function
unsigned crc_f2( const void * buffer, std::size_t byte_count )
{
return boost::crc<7u, 0x09u, 0u, 0u, false, false>( buffer, byte_count );
}
void sub_octet_polynominal_test()
{
// The CRC standard is a SDH/SONET J0/J1/J2/N1/N2/TR TTI (trace message)
// with CRC-7, o.a. ITU-T G.707 Annex B, G.832 Annex A.
// Two samples, each sixteen bytes
// Sample 1 is '\x80' + ASCII("123456789ABCDEF")
// Sample 2 is '\x80' + ASCII("TTI UNAVAILABLE")
unsigned char const samples[2][16]
= {
{ 0x80u, 0x31u, 0x32u, 0x33u, 0x34u, 0x35u, 0x36u, 0x37u, 0x38u,
0x39u, 0x41u, 0x42u, 0x43u, 0x44u, 0x45u, 0x46u },
{ 0x80u, 0x54u, 0x54u, 0x49u, 0x20u, 0x55u, 0x4Eu, 0x41u, 0x56u,
0x41u, 0x49u, 0x4Cu, 0x41u, 0x42u, 0x4Cu, 0x45u }
};
unsigned const results[2] = { 0x62u, 0x23u };
// Basic computer
boost::crc_basic<7u> crc_1( 0x09u );
crc_1.process_bytes( samples[0], 16u );
BOOST_TEST_EQ( crc_1.checksum(), results[0] );
crc_1.reset();
crc_1.process_bytes( samples[1], 16u );
BOOST_TEST_EQ( crc_1.checksum(), results[1] );
BOOST_TEST_EQ( crc_f2(samples[ 0 ], 16u), results[0] );
BOOST_TEST_EQ( crc_f2(samples[ 1 ], 16u), results[1] );
// TODO: do similar tests with boost::augmented_crc<7, 0x09>
// (Now I think that this can't be done right now, since that function reads
// byte-wise, so the register size needs to be a multiple of CHAR_BIT.)
}
void one_bit_polynominal_test()
{
// Try a CRC based on the (x + 1) polynominal, which is a factor in
// many real-life polynominals and doesn't fit evenly in a byte.
boost::crc_basic<1u> crc_1( 1u );
crc_1.process_bytes( std_data, std_data_len );
BOOST_TEST_EQ( crc_1.checksum(), 1u );
// Do it again, but using crc_optimal. The real purpose of this is to test
// crc_optimal::process_byte, which doesn't get exercised anywhere else in
// this file (directly or indirectly).
boost::crc_optimal<1u, 1u, 0u, 0u, false, false> crc_2;
for ( std::size_t i = 0u ; i < std_data_len ; ++i )
crc_2.process_byte( std_data[i] );
BOOST_TEST_EQ( crc_2.checksum(), 1u );
}
struct function_object_test {
template<class CRCPolicy>
void operator()(CRCPolicy)
{
typename CRCPolicy::computer_ct_type crc_c;
crc_c = std::for_each( std_data, std_data + std_data_len, crc_c );
BOOST_TEST_EQ( crc_c(), CRCPolicy::standard_test_data_CRC_c::value );
}
};
// Ticket #2492: crc_optimal with reversed CRC16
// <https://svn.boost.org/trac/boost/ticket/2492>
void issue_2492_test()
{
// I'm trusting that the original bug reporter got his/her calculations
// correct.
boost::uint16_t const expected_result = 0xF990u;
boost::crc_optimal<16, 0x100Bu, 0xFFFFu, 0x0000, true, false> boost_crc_1;
boost::crc_basic<16> boost_crc_2( 0x100Bu, 0xFFFFu, 0x0000, true, false );
// This should be right...
boost_crc_1.process_byte( 0u );
BOOST_TEST_EQ( boost_crc_1.checksum(), expected_result );
// ...but the reporter said this didn't reflect, giving 0x099F as the
// (wrong) result. However, I get the right answer!
boost_crc_2.process_byte( 0u );
BOOST_TEST_EQ( boost_crc_2.checksum(), expected_result );
}
int main()
{
run_crc_extended_test_policies<computation_comparison_test>();
run_crc_test_policies<accessor_and_split_run_test>();
run_crc_test_policies<reset_and_single_bit_error_test>();
augmented_crc_test();
sub_nybble_polynominal_test();
sub_octet_polynominal_test();
one_bit_polynominal_test();
run_crc_test_policies<function_object_test>();
issue_2492_test();
return boost::report_errors();
}