| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248 |
- /*
- Copyright 2007-2008 Christian Henning, Andreas Pokorny, Lubomir Bourdev
- Use, modification and distribution are subject to 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).
- */
- /*************************************************************************************************/
- #ifndef BOOST_GIL_IO_BIT_OPERATIONS_HPP
- #define BOOST_GIL_IO_BIT_OPERATIONS_HPP
- ////////////////////////////////////////////////////////////////////////////////////////
- /// \file
- /// \brief
- /// \author Christian Henning \n
- ///
- /// \date 2008 \n
- ///
- ////////////////////////////////////////////////////////////////////////////////////////
- #include <boost/bind.hpp>
- #include <array>
- namespace boost { namespace gil { namespace detail {
- // 1110 1100 -> 0011 0111
- template< typename Buffer
- , typename IsBitAligned
- >
- struct mirror_bits
- {
- mirror_bits( bool ) {}
- void operator() ( Buffer& ) {}
- void operator() ( byte_t* , std::size_t )
- {
- }
- };
- // The functor will generate a lookup table since the
- // mirror operation is quite costly.
- template< typename Buffer >
- struct mirror_bits< Buffer
- , mpl::true_
- >
- {
- mirror_bits( bool apply_operation = true )
- : _apply_operation( apply_operation )
- {
- if( _apply_operation == true )
- {
- byte_t i = 0;
- do
- {
- _lookup[i] = mirror( i );
- }
- while( i++ != 255 );
- }
- }
- void operator() ( Buffer& buf )
- {
- if( _apply_operation == true )
- {
- for_each( buf.begin()
- , buf.end()
- , boost::bind( &mirror_bits< Buffer
- , mpl::true_
- >::lookup
- , *this
- , ::_1
- )
- );
- }
- }
- void operator() ( byte_t* dst, std::size_t size )
- {
- for( std::size_t i = 0; i < size; ++i )
- {
- lookup(*dst);
- ++dst;
- }
- }
- private:
- void lookup( byte_t& c )
- {
- c = _lookup[ c ];
- }
- static byte_t mirror( byte_t c )
- {
- byte_t result = 0;
- for( int i = 0; i < 8; ++i )
- {
- result = result << 1;
- result |= ( c & 1 );
- c = c >> 1;
- }
- return result;
- }
-
- private:
- bool _apply_operation;
- std::array< byte_t, 256 > _lookup;
- };
- // 0011 1111 -> 1100 0000
- template< typename Buffer
- , typename IsBitAligned
- >
- struct negate_bits
- {
- void operator() ( Buffer& ) {}
- };
- template< typename Buffer >
- struct negate_bits< Buffer, mpl::true_ >
- {
- void operator() ( Buffer& buf )
- {
- for_each( buf.begin()
- , buf.end()
- , negate_bits< Buffer, mpl::true_ >::negate
- );
- }
- void operator() ( byte_t* dst, std::size_t size )
- {
- for( std::size_t i = 0; i < size; ++i )
- {
- negate(*dst);
- ++dst;
- }
- }
- private:
- static void negate( byte_t& b )
- {
- b = ~b;
- }
- };
- // 11101100 -> 11001110
- template< typename Buffer
- , typename IsBitAligned
- >
- struct swap_half_bytes
- {
- void operator() ( Buffer& ) {}
- };
- template< typename Buffer >
- struct swap_half_bytes< Buffer
- , mpl::true_
- >
- {
- void operator() ( Buffer& buf )
- {
- for_each( buf.begin()
- , buf.end()
- , swap_half_bytes< Buffer, mpl::true_ >::swap
- );
- }
- void operator() ( byte_t* dst, std::size_t size )
- {
- for( std::size_t i = 0; i < size; ++i )
- {
- swap(*dst);
- ++dst;
- }
- }
- private:
- static void swap( byte_t& c )
- {
- c = (( c << 4 ) & 0xF0 ) | (( c >> 4 ) & 0x0F );
- }
- };
- template< typename Buffer >
- struct do_nothing
- {
- do_nothing() {}
- void operator() ( Buffer& ) {}
- };
- /// Count consecutive zeros on the right
- template< typename T >
- inline
- unsigned int trailing_zeros( T x )
- throw()
- {
- unsigned int n = 0;
- x = ~x & (x - 1);
- while( x )
- {
- n = n + 1;
- x = x >> 1;
- }
- return n;
- }
- /// Counts ones in a bit-set
- template< typename T >
- inline
- unsigned int count_ones( T x )
- throw()
- {
- unsigned int n = 0;
- while( x )
- {
- // clear the least significant bit set
- x &= x - 1;
- ++n;
- }
- return n;
- }
- } // namespace detail
- } // namespace gil
- } // namespace boost
- #endif
|