Mercurial > hg > vamp-build-and-test
diff DEPENDENCIES/generic/include/boost/gil/channel.hpp @ 16:2665513ce2d3
Add boost headers
author | Chris Cannam |
---|---|
date | Tue, 05 Aug 2014 11:11:38 +0100 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/DEPENDENCIES/generic/include/boost/gil/channel.hpp Tue Aug 05 11:11:38 2014 +0100 @@ -0,0 +1,672 @@ +/* + Copyright 2005-2007 Adobe Systems Incorporated + + 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). + + See http://stlab.adobe.com/gil for most recent version including documentation. +*/ + +/*************************************************************************************************/ + +#ifndef GIL_CHANNEL_HPP +#define GIL_CHANNEL_HPP + +//////////////////////////////////////////////////////////////////////////////////////// +/// \file +/// \brief Channel utilities +/// \author Lubomir Bourdev and Hailin Jin \n +/// Adobe Systems Incorporated +/// \date 2005-2007 \n Last updated on May 6, 2007 +/// +/// Definitions of standard GIL channel models +/// +//////////////////////////////////////////////////////////////////////////////////////// + +#include <limits> +#include <cassert> +#include <boost/cstdint.hpp> +#include "gil_config.hpp" +#include "utilities.hpp" + +namespace boost { namespace gil { + + +/////////////////////////////////////////// +//// channel_traits +//// +//// \ingroup ChannelModel +//// \class channel_traits +//// \brief defines properties of channels, such as their range and associated types +//// +//// The channel traits must be defined for every model of ChannelConcept +//// Default traits are provided. For built-in types the default traits use +//// built-in pointer and reference and the channel range is the physical +//// range of the type. For classes, the default traits forward the associated types +//// and range to the class. +//// +/////////////////////////////////////////// + +namespace detail { + template <typename T, bool is_class> struct channel_traits_impl; + + // channel traits for custom class + template <typename T> + struct channel_traits_impl<T, true> { + typedef typename T::value_type value_type; + typedef typename T::reference reference; + typedef typename T::pointer pointer; + typedef typename T::const_reference const_reference; + typedef typename T::const_pointer const_pointer; + BOOST_STATIC_CONSTANT(bool, is_mutable=T::is_mutable); + static value_type min_value() { return T::min_value(); } + static value_type max_value() { return T::max_value(); } + }; + + // channel traits implementation for built-in integral or floating point channel type + template <typename T> + struct channel_traits_impl<T, false> { + typedef T value_type; + typedef T& reference; + typedef T* pointer; + typedef const T& const_reference; + typedef T const* const_pointer; + BOOST_STATIC_CONSTANT(bool, is_mutable=true); + static value_type min_value() { return (std::numeric_limits<T>::min)(); } + static value_type max_value() { return (std::numeric_limits<T>::max)(); } + }; + + // channel traits implementation for constant built-in scalar or floating point type + template <typename T> + struct channel_traits_impl<const T, false> : public channel_traits_impl<T, false> { + typedef const T& reference; + typedef const T* pointer; + BOOST_STATIC_CONSTANT(bool, is_mutable=false); + }; +} + +/** +\ingroup ChannelModel +\brief Traits for channels. Contains the following members: +\code +template <typename Channel> +struct channel_traits { + typedef ... value_type; + typedef ... reference; + typedef ... pointer; + typedef ... const_reference; + typedef ... const_pointer; + + static const bool is_mutable; + static value_type min_value(); + static value_type max_value(); +}; +\endcode +*/ +template <typename T> +struct channel_traits : public detail::channel_traits_impl<T, is_class<T>::value> {}; + +// Channel traits for C++ reference type - remove the reference +template <typename T> struct channel_traits< T&> : public channel_traits<T> {}; + +// Channel traits for constant C++ reference type +template <typename T> struct channel_traits<const T&> : public channel_traits<T> { + typedef typename channel_traits<T>::const_reference reference; + typedef typename channel_traits<T>::const_pointer pointer; + BOOST_STATIC_CONSTANT(bool, is_mutable=false); +}; + +/////////////////////////////////////////// +//// +//// scoped_channel_value +//// +/////////////////////////////////////////// + +/** +\defgroup ScopedChannelValue scoped_channel_value +\ingroup ChannelModel +\brief A channel adaptor that modifies the range of the source channel. Models: ChannelValueConcept + +Example: +\code +// Create a double channel with range [-0.5 .. 0.5] +struct double_minus_half { static double apply() { return -0.5; } }; +struct double_plus_half { static double apply() { return 0.5; } }; +typedef scoped_channel_value<double, double_minus_half, double_plus_half> bits64custom_t; + +// channel_convert its maximum should map to the maximum +bits64custom_t x = channel_traits<bits64custom_t>::max_value(); +assert(x == 0.5); +bits16 y = channel_convert<bits16>(x); +assert(y == 65535); +\endcode +*/ + +/// \ingroup ScopedChannelValue +/// \brief A channel adaptor that modifies the range of the source channel. Models: ChannelValueConcept +template <typename BaseChannelValue, // base channel (models ChannelValueConcept) + typename MinVal, typename MaxVal> // classes with a static apply() function returning the minimum/maximum channel values +struct scoped_channel_value { + typedef scoped_channel_value value_type; + typedef value_type& reference; + typedef value_type* pointer; + typedef const value_type& const_reference; + typedef const value_type* const_pointer; + BOOST_STATIC_CONSTANT(bool, is_mutable=channel_traits<BaseChannelValue>::is_mutable); + + typedef BaseChannelValue base_channel_t; + + static value_type min_value() { return MinVal::apply(); } + static value_type max_value() { return MaxVal::apply(); } + + scoped_channel_value() {} + scoped_channel_value(const scoped_channel_value& c) : _value(c._value) {} + scoped_channel_value(BaseChannelValue val) : _value(val) {} + + scoped_channel_value& operator++() { ++_value; return *this; } + scoped_channel_value& operator--() { --_value; return *this; } + + scoped_channel_value operator++(int) { scoped_channel_value tmp=*this; this->operator++(); return tmp; } + scoped_channel_value operator--(int) { scoped_channel_value tmp=*this; this->operator--(); return tmp; } + + template <typename Scalar2> scoped_channel_value& operator+=(Scalar2 v) { _value+=v; return *this; } + template <typename Scalar2> scoped_channel_value& operator-=(Scalar2 v) { _value-=v; return *this; } + template <typename Scalar2> scoped_channel_value& operator*=(Scalar2 v) { _value*=v; return *this; } + template <typename Scalar2> scoped_channel_value& operator/=(Scalar2 v) { _value/=v; return *this; } + + scoped_channel_value& operator=(BaseChannelValue v) { _value=v; return *this; } + operator BaseChannelValue() const { return _value; } +private: + BaseChannelValue _value; +}; + +struct float_zero { static float apply() { return 0.0f; } }; +struct float_one { static float apply() { return 1.0f; } }; + + +/////////////////////////////////////////// +//// +//// Support for sub-byte channels. These are integral channels whose value is contained in a range of bits inside an integral type +//// +/////////////////////////////////////////// + +// It is necessary for packed channels to have their own value type. They cannot simply use an integral large enough to store the data. Here is why: +// - Any operation that requires returning the result by value will otherwise return the built-in integral type, which will have incorrect range +// That means that after getting the value of the channel we cannot properly do channel_convert, channel_invert, etc. +// - Two channels are declared compatible if they have the same value type. That means that a packed channel is incorrectly declared compatible with an integral type +namespace detail { + // returns the smallest fast unsigned integral type that has at least NumBits bits + template <int NumBits> + struct min_fast_uint : public mpl::if_c< (NumBits<=8), + uint_least8_t, + typename mpl::if_c< (NumBits<=16), + uint_least16_t, + typename mpl::if_c< (NumBits<=32), + uint_least32_t, + uintmax_t + >::type + >::type + > {}; + + template <int NumBits> + struct num_value_fn : public mpl::if_c< ( NumBits < 32 ) + , uint32_t + , uint64_t + > {}; + + template <int NumBits> + struct max_value_fn : public mpl::if_c< ( NumBits <= 32 ) + , uint32_t + , uint64_t + > {}; +} + +/** +\defgroup PackedChannelValueModel packed_channel_value +\ingroup ChannelModel +\brief Represents the value of an unsigned integral channel operating over a bit range. Models: ChannelValueConcept +Example: +\code +// A 4-bit unsigned integral channel. +typedef packed_channel_value<4> bits4; + +assert(channel_traits<bits4>::min_value()==0); +assert(channel_traits<bits4>::max_value()==15); +assert(sizeof(bits4)==1); +BOOST_STATIC_ASSERT((boost::is_integral<bits4>::value)); +\endcode +*/ + +/// \ingroup PackedChannelValueModel +/// \brief The value of a subbyte channel. Models: ChannelValueConcept +template <int NumBits> +class packed_channel_value { + + typedef typename detail::num_value_fn< NumBits >::type num_value_t; + static const num_value_t num_values = static_cast< num_value_t >( 1 ) << NumBits ; + +public: + typedef typename detail::min_fast_uint<NumBits>::type integer_t; + + + typedef packed_channel_value value_type; + typedef value_type& reference; + typedef const value_type& const_reference; + typedef value_type* pointer; + typedef const value_type* const_pointer; + + static value_type min_value() { return value_type(0); } + static value_type max_value() { return value_type(num_values-1); } + BOOST_STATIC_CONSTANT(bool, is_mutable=true); + + packed_channel_value() {} + packed_channel_value(integer_t v) { _value = static_cast< integer_t >( v % num_values ); } + packed_channel_value(const packed_channel_value& v) : _value(v._value) {} + template <typename Scalar> packed_channel_value(Scalar v) { _value = static_cast< integer_t >( v ) % num_values; } + + static unsigned int num_bits() { return NumBits; } + + + operator integer_t() const { return _value; } +private: + integer_t _value; +}; + +namespace detail { + +template <std::size_t K> +struct static_copy_bytes { + void operator()(const unsigned char* from, unsigned char* to) const { + *to = *from; + static_copy_bytes<K-1>()(++from,++to); + } +}; + +template <> +struct static_copy_bytes<0> { + void operator()(const unsigned char* , unsigned char*) const {} +}; + +template <typename Derived, typename BitField, int NumBits, bool Mutable> +class packed_channel_reference_base { +protected: + typedef typename mpl::if_c<Mutable,void*,const void*>::type data_ptr_t; +public: + data_ptr_t _data_ptr; // void* pointer to the first byte of the bit range + + typedef packed_channel_value<NumBits> value_type; + typedef const Derived reference; + typedef value_type* pointer; + typedef const value_type* const_pointer; + BOOST_STATIC_CONSTANT(int, num_bits=NumBits); + BOOST_STATIC_CONSTANT(bool, is_mutable=Mutable); + + static value_type min_value() { return channel_traits<value_type>::min_value(); } + static value_type max_value() { return channel_traits<value_type>::max_value(); } + + typedef BitField bitfield_t; + typedef typename value_type::integer_t integer_t; + + packed_channel_reference_base(data_ptr_t data_ptr) : _data_ptr(data_ptr) {} + packed_channel_reference_base(const packed_channel_reference_base& ref) : _data_ptr(ref._data_ptr) {} + const Derived& operator=(integer_t v) const { set(v); return derived(); } + + const Derived& operator++() const { set(get()+1); return derived(); } + const Derived& operator--() const { set(get()-1); return derived(); } + + Derived operator++(int) const { Derived tmp=derived(); this->operator++(); return tmp; } + Derived operator--(int) const { Derived tmp=derived(); this->operator--(); return tmp; } + + template <typename Scalar2> const Derived& operator+=(Scalar2 v) const { set(get()+v); return derived(); } + template <typename Scalar2> const Derived& operator-=(Scalar2 v) const { set(get()-v); return derived(); } + template <typename Scalar2> const Derived& operator*=(Scalar2 v) const { set(get()*v); return derived(); } + template <typename Scalar2> const Derived& operator/=(Scalar2 v) const { set(get()/v); return derived(); } + + operator integer_t() const { return get(); } + data_ptr_t operator &() const {return _data_ptr;} +protected: + + typedef typename detail::num_value_fn< NumBits >::type num_value_t; + typedef typename detail::max_value_fn< NumBits >::type max_value_t; + + static const num_value_t num_values = static_cast< num_value_t >( 1 ) << NumBits ; + static const max_value_t max_val = static_cast< max_value_t >( num_values - 1 ); + +#ifdef GIL_NONWORD_POINTER_ALIGNMENT_SUPPORTED + const bitfield_t& get_data() const { return *static_cast<const bitfield_t*>(_data_ptr); } + void set_data(const bitfield_t& val) const { *static_cast< bitfield_t*>(_data_ptr) = val; } +#else + bitfield_t get_data() const { + bitfield_t ret; + static_copy_bytes<sizeof(bitfield_t) >()(gil_reinterpret_cast_c<const unsigned char*>(_data_ptr),gil_reinterpret_cast<unsigned char*>(&ret)); + return ret; + } + void set_data(const bitfield_t& val) const { + static_copy_bytes<sizeof(bitfield_t) >()(gil_reinterpret_cast_c<const unsigned char*>(&val),gil_reinterpret_cast<unsigned char*>(_data_ptr)); + } +#endif + +private: + void set(integer_t value) const { // can this be done faster?? + const integer_t num_values = max_val+1; + this->derived().set_unsafe(((value % num_values) + num_values) % num_values); + } + integer_t get() const { return derived().get(); } + const Derived& derived() const { return static_cast<const Derived&>(*this); } +}; +} // namespace detail + +/** +\defgroup PackedChannelReferenceModel packed_channel_reference +\ingroup ChannelModel +\brief Represents a reference proxy to a channel operating over a bit range whose offset is fixed at compile time. Models ChannelConcept +Example: +\code +// Reference to a 2-bit channel starting at bit 1 (i.e. the second bit) +typedef const packed_channel_reference<uint16_t,1,2,true> bits2_1_ref_t; + +uint16_t data=0; +bits2_1_ref_t channel_ref(&data); +channel_ref = channel_traits<bits2_1_ref_t>::max_value(); // == 3 +assert(data == 6); // == 3<<1 == 6 +\endcode +*/ + +template <typename BitField, // A type that holds the bits of the pixel from which the channel is referenced. Typically an integral type, like boost::uint16_t + int FirstBit, int NumBits,// Defines the sequence of bits in the data value that contain the channel + bool Mutable> // true if the reference is mutable +class packed_channel_reference; + +template <typename BitField, // A type that holds the bits of the pixel from which the channel is referenced. Typically an integral type, like boost::uint16_t + int NumBits, // Defines the sequence of bits in the data value that contain the channel + bool Mutable> // true if the reference is mutable +class packed_dynamic_channel_reference; + +/// \ingroup PackedChannelReferenceModel +/// \brief A constant subbyte channel reference whose bit offset is fixed at compile time. Models ChannelConcept +template <typename BitField, int FirstBit, int NumBits> +class packed_channel_reference<BitField,FirstBit,NumBits,false> + : public detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,false>,BitField,NumBits,false> { + typedef detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,false>,BitField,NumBits,false> parent_t; + friend class packed_channel_reference<BitField,FirstBit,NumBits,true>; + + static const BitField channel_mask = static_cast< BitField >( parent_t::max_val ) << FirstBit; + + void operator=(const packed_channel_reference&); +public: + typedef const packed_channel_reference<BitField,FirstBit,NumBits,false> const_reference; + typedef const packed_channel_reference<BitField,FirstBit,NumBits,true> mutable_reference; + typedef typename parent_t::integer_t integer_t; + + explicit packed_channel_reference(const void* data_ptr) : parent_t(data_ptr) {} + packed_channel_reference(const packed_channel_reference& ref) : parent_t(ref._data_ptr) {} + packed_channel_reference(const mutable_reference& ref) : parent_t(ref._data_ptr) {} + + unsigned first_bit() const { return FirstBit; } + + integer_t get() const { return integer_t((this->get_data()&channel_mask) >> FirstBit); } +}; + +/// \ingroup PackedChannelReferenceModel +/// \brief A mutable subbyte channel reference whose bit offset is fixed at compile time. Models ChannelConcept +template <typename BitField, int FirstBit, int NumBits> +class packed_channel_reference<BitField,FirstBit,NumBits,true> + : public detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,true>,BitField,NumBits,true> { + typedef detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,true>,BitField,NumBits,true> parent_t; + friend class packed_channel_reference<BitField,FirstBit,NumBits,false>; + + static const BitField channel_mask = static_cast< BitField >( parent_t::max_val ) << FirstBit; + +public: + typedef const packed_channel_reference<BitField,FirstBit,NumBits,false> const_reference; + typedef const packed_channel_reference<BitField,FirstBit,NumBits,true> mutable_reference; + typedef typename parent_t::integer_t integer_t; + + explicit packed_channel_reference(void* data_ptr) : parent_t(data_ptr) {} + packed_channel_reference(const packed_channel_reference& ref) : parent_t(ref._data_ptr) {} + + const packed_channel_reference& operator=(integer_t value) const { assert(value<=parent_t::max_val); set_unsafe(value); return *this; } + const packed_channel_reference& operator=(const mutable_reference& ref) const { set_from_reference(ref.get_data()); return *this; } + const packed_channel_reference& operator=(const const_reference& ref) const { set_from_reference(ref.get_data()); return *this; } + + template <bool Mutable1> + const packed_channel_reference& operator=(const packed_dynamic_channel_reference<BitField,NumBits,Mutable1>& ref) const { set_unsafe(ref.get()); return *this; } + + unsigned first_bit() const { return FirstBit; } + + integer_t get() const { return integer_t((this->get_data()&channel_mask) >> FirstBit); } + void set_unsafe(integer_t value) const { this->set_data((this->get_data() & ~channel_mask) | (( static_cast< BitField >( value )<<FirstBit))); } +private: + void set_from_reference(const BitField& other_bits) const { this->set_data((this->get_data() & ~channel_mask) | (other_bits & channel_mask)); } +}; + +} } // namespace boost::gil + +namespace std { +// We are forced to define swap inside std namespace because on some platforms (Visual Studio 8) STL calls swap qualified. +// swap with 'left bias': +// - swap between proxy and anything +// - swap between value type and proxy +// - swap between proxy and proxy + +/// \ingroup PackedChannelReferenceModel +/// \brief swap for packed_channel_reference +template <typename BF, int FB, int NB, bool M, typename R> inline +void swap(const boost::gil::packed_channel_reference<BF,FB,NB,M> x, R& y) { + boost::gil::swap_proxy<typename boost::gil::packed_channel_reference<BF,FB,NB,M>::value_type>(x,y); +} + + +/// \ingroup PackedChannelReferenceModel +/// \brief swap for packed_channel_reference +template <typename BF, int FB, int NB, bool M> inline +void swap(typename boost::gil::packed_channel_reference<BF,FB,NB,M>::value_type& x, const boost::gil::packed_channel_reference<BF,FB,NB,M> y) { + boost::gil::swap_proxy<typename boost::gil::packed_channel_reference<BF,FB,NB,M>::value_type>(x,y); +} + + +/// \ingroup PackedChannelReferenceModel +/// \brief swap for packed_channel_reference +template <typename BF, int FB, int NB, bool M> inline +void swap(const boost::gil::packed_channel_reference<BF,FB,NB,M> x, const boost::gil::packed_channel_reference<BF,FB,NB,M> y) { + boost::gil::swap_proxy<typename boost::gil::packed_channel_reference<BF,FB,NB,M>::value_type>(x,y); +} +} // namespace std + +namespace boost { namespace gil { + +/** +\defgroup PackedChannelDynamicReferenceModel packed_dynamic_channel_reference +\ingroup ChannelModel +\brief Represents a reference proxy to a channel operating over a bit range whose offset is specified at run time. Models ChannelConcept + +Example: +\code +// Reference to a 2-bit channel whose offset is specified at construction time +typedef const packed_dynamic_channel_reference<uint8_t,2,true> bits2_dynamic_ref_t; + +uint16_t data=0; +bits2_dynamic_ref_t channel_ref(&data,1); +channel_ref = channel_traits<bits2_dynamic_ref_t>::max_value(); // == 3 +assert(data == 6); // == (3<<1) == 6 +\endcode +*/ + +/// \brief Models a constant subbyte channel reference whose bit offset is a runtime parameter. Models ChannelConcept +/// Same as packed_channel_reference, except that the offset is a runtime parameter +/// \ingroup PackedChannelDynamicReferenceModel +template <typename BitField, int NumBits> +class packed_dynamic_channel_reference<BitField,NumBits,false> + : public detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,false>,BitField,NumBits,false> { + typedef detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,false>,BitField,NumBits,false> parent_t; + friend class packed_dynamic_channel_reference<BitField,NumBits,true>; + + unsigned _first_bit; // 0..7 + + void operator=(const packed_dynamic_channel_reference&); +public: + typedef const packed_dynamic_channel_reference<BitField,NumBits,false> const_reference; + typedef const packed_dynamic_channel_reference<BitField,NumBits,true> mutable_reference; + typedef typename parent_t::integer_t integer_t; + + packed_dynamic_channel_reference(const void* data_ptr, unsigned first_bit) : parent_t(data_ptr), _first_bit(first_bit) {} + packed_dynamic_channel_reference(const const_reference& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {} + packed_dynamic_channel_reference(const mutable_reference& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {} + + unsigned first_bit() const { return _first_bit; } + + integer_t get() const { + const BitField channel_mask = static_cast< integer_t >( parent_t::max_val ) <<_first_bit; + return static_cast< integer_t >(( this->get_data()&channel_mask ) >> _first_bit ); + } +}; + +/// \brief Models a mutable subbyte channel reference whose bit offset is a runtime parameter. Models ChannelConcept +/// Same as packed_channel_reference, except that the offset is a runtime parameter +/// \ingroup PackedChannelDynamicReferenceModel +template <typename BitField, int NumBits> +class packed_dynamic_channel_reference<BitField,NumBits,true> + : public detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,true>,BitField,NumBits,true> { + typedef detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,true>,BitField,NumBits,true> parent_t; + friend class packed_dynamic_channel_reference<BitField,NumBits,false>; + + unsigned _first_bit; + +public: + typedef const packed_dynamic_channel_reference<BitField,NumBits,false> const_reference; + typedef const packed_dynamic_channel_reference<BitField,NumBits,true> mutable_reference; + typedef typename parent_t::integer_t integer_t; + + packed_dynamic_channel_reference(void* data_ptr, unsigned first_bit) : parent_t(data_ptr), _first_bit(first_bit) {} + packed_dynamic_channel_reference(const packed_dynamic_channel_reference& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {} + + const packed_dynamic_channel_reference& operator=(integer_t value) const { assert(value<=parent_t::max_val); set_unsafe(value); return *this; } + const packed_dynamic_channel_reference& operator=(const mutable_reference& ref) const { set_unsafe(ref.get()); return *this; } + const packed_dynamic_channel_reference& operator=(const const_reference& ref) const { set_unsafe(ref.get()); return *this; } + + template <typename BitField1, int FirstBit1, bool Mutable1> + const packed_dynamic_channel_reference& operator=(const packed_channel_reference<BitField1, FirstBit1, NumBits, Mutable1>& ref) const + { set_unsafe(ref.get()); return *this; } + + unsigned first_bit() const { return _first_bit; } + + integer_t get() const { + const BitField channel_mask = static_cast< integer_t >( parent_t::max_val ) << _first_bit; + return static_cast< integer_t >(( this->get_data()&channel_mask ) >> _first_bit ); + } + + void set_unsafe(integer_t value) const { + const BitField channel_mask = static_cast< integer_t >( parent_t::max_val ) << _first_bit; + this->set_data((this->get_data() & ~channel_mask) | value<<_first_bit); + } +}; +} } // namespace boost::gil + +namespace std { +// We are forced to define swap inside std namespace because on some platforms (Visual Studio 8) STL calls swap qualified. +// swap with 'left bias': +// - swap between proxy and anything +// - swap between value type and proxy +// - swap between proxy and proxy + + +/// \ingroup PackedChannelDynamicReferenceModel +/// \brief swap for packed_dynamic_channel_reference +template <typename BF, int NB, bool M, typename R> inline +void swap(const boost::gil::packed_dynamic_channel_reference<BF,NB,M> x, R& y) { + boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y); +} + + +/// \ingroup PackedChannelDynamicReferenceModel +/// \brief swap for packed_dynamic_channel_reference +template <typename BF, int NB, bool M> inline +void swap(typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type& x, const boost::gil::packed_dynamic_channel_reference<BF,NB,M> y) { + boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y); +} + + +/// \ingroup PackedChannelDynamicReferenceModel +/// \brief swap for packed_dynamic_channel_reference +template <typename BF, int NB, bool M> inline +void swap(const boost::gil::packed_dynamic_channel_reference<BF,NB,M> x, const boost::gil::packed_dynamic_channel_reference<BF,NB,M> y) { + boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y); +} +} // namespace std + +namespace boost { namespace gil { +/////////////////////////////////////////// +//// +//// Built-in channel models +//// +/////////////////////////////////////////// + +/// \defgroup bits8 bits8 +/// \ingroup ChannelModel +/// \brief 8-bit unsigned integral channel type (typedef from uint8_t). Models ChannelValueConcept + +/// \ingroup bits8 +typedef uint8_t bits8; + +/// \defgroup bits16 bits16 +/// \ingroup ChannelModel +/// \brief 16-bit unsigned integral channel type (typedef from uint16_t). Models ChannelValueConcept + +/// \ingroup bits16 +typedef uint16_t bits16; + +/// \defgroup bits32 bits32 +/// \ingroup ChannelModel +/// \brief 32-bit unsigned integral channel type (typedef from uint32_t). Models ChannelValueConcept + +/// \ingroup bits32 +typedef uint32_t bits32; + +/// \defgroup bits8s bits8s +/// \ingroup ChannelModel +/// \brief 8-bit signed integral channel type (typedef from int8_t). Models ChannelValueConcept + +/// \ingroup bits8s +typedef int8_t bits8s; + +/// \defgroup bits16s bits16s +/// \ingroup ChannelModel +/// \brief 16-bit signed integral channel type (typedef from int16_t). Models ChannelValueConcept + +/// \ingroup bits16s +typedef int16_t bits16s; + +/// \defgroup bits32s bits32s +/// \ingroup ChannelModel +/// \brief 32-bit signed integral channel type (typedef from int32_t). Models ChannelValueConcept + +/// \ingroup bits32s +typedef int32_t bits32s; + +/// \defgroup bits32f bits32f +/// \ingroup ChannelModel +/// \brief 32-bit floating point channel type with range [0.0f ... 1.0f]. Models ChannelValueConcept + +/// \ingroup bits32f +typedef scoped_channel_value<float,float_zero,float_one> bits32f; + +} } // namespace boost::gil + +namespace boost { + +template <int NumBits> +struct is_integral<gil::packed_channel_value<NumBits> > : public mpl::true_ {}; + +template <typename BitField, int FirstBit, int NumBits, bool IsMutable> +struct is_integral<gil::packed_channel_reference<BitField,FirstBit,NumBits,IsMutable> > : public mpl::true_ {}; + +template <typename BitField, int NumBits, bool IsMutable> +struct is_integral<gil::packed_dynamic_channel_reference<BitField,NumBits,IsMutable> > : public mpl::true_ {}; + +template <typename BaseChannelValue, typename MinVal, typename MaxVal> +struct is_integral<gil::scoped_channel_value<BaseChannelValue,MinVal,MaxVal> > : public is_integral<BaseChannelValue> {}; + +} + +#endif