8#ifndef BOOST_GIL_CHANNEL_HPP 
    9#define BOOST_GIL_CHANNEL_HPP 
   11#include <boost/gil/utilities.hpp> 
   13#include <boost/assert.hpp> 
   14#include <boost/config.hpp> 
   15#include <boost/config/pragma_message.hpp> 
   16#include <boost/integer/integer_mask.hpp> 
   22#ifdef BOOST_GIL_DOXYGEN_ONLY 
   30#define BOOST_GIL_CONFIG_HAS_UNALIGNED_ACCESS 
   33#ifdef BOOST_GIL_CONFIG_HAS_UNALIGNED_ACCESS 
   34#if defined(sun) || defined(__sun) || \              
   35    defined(__osf__) || defined(__osf) || \         
 
   36    defined(_hpux) || defined(hpux) || \            
 
   37    defined(__arm__) || defined(__ARM_ARCH) || \    
 
   39#error Unaligned access strictly disabled for some UNIX platforms or ARM architecture 
   40#elif defined(__i386__) || defined(__x86_64__) || defined(__vax__) 
   48BOOST_PRAGMA_MESSAGE(
"CAUTION: Unaligned access tolerated on little-endian may cause undefined behaviour")
 
   50#error Unaligned access disabled for unknown platforms and architectures 
   54namespace boost { 
namespace gil {
 
   73template <
typename T, 
bool IsClass>
 
   74struct channel_traits_impl;
 
   78struct channel_traits_impl<T, true>
 
   80    using value_type = 
typename T::value_type;
 
   81    using reference = 
typename T::reference;
 
   82    using pointer = 
typename T::pointer;
 
   83    using const_reference = 
typename T::const_reference;
 
   84    using const_pointer = 
typename T::const_pointer;
 
   85    static constexpr bool is_mutable = T::is_mutable;
 
   86    static value_type min_value() { 
return T::min_value(); }
 
   87    static value_type max_value() { 
return T::max_value(); }
 
   92struct channel_traits_impl<T, false>
 
   97    using const_reference = T 
const&;
 
   98    using const_pointer = T 
const*;
 
   99    static constexpr bool is_mutable = 
true;
 
  100    static value_type min_value() { 
return (std::numeric_limits<T>::min)(); }
 
  101    static value_type max_value() { 
return (std::numeric_limits<T>::max)(); }
 
  106struct channel_traits_impl<T const, false> : channel_traits_impl<T, false>
 
  108    using reference = T 
const&;
 
  109    using pointer = T 
const*;
 
  110    static constexpr bool is_mutable = 
false;
 
  134struct channel_traits : detail::channel_traits_impl<T, std::is_class<T>::value> {};
 
  138struct channel_traits<T&> : channel_traits<T> {};
 
  142struct channel_traits<T const&> : channel_traits<T>
 
  144    using reference = 
typename channel_traits<T>::const_reference;
 
  145    using pointer = 
typename channel_traits<T>::const_pointer;
 
  146    static constexpr bool is_mutable = 
false;
 
  176template <
typename BaseChannelValue, 
typename MinVal, 
typename MaxVal>
 
  177struct scoped_channel_value
 
  179    using value_type = scoped_channel_value<BaseChannelValue, MinVal, MaxVal>;
 
  180    using reference = value_type&;
 
  181    using pointer = value_type*;
 
  182    using const_reference = value_type 
const&;
 
  183    using const_pointer = value_type 
const*;
 
  184    static constexpr bool is_mutable = channel_traits<BaseChannelValue>::is_mutable;
 
  186    using base_channel_t = BaseChannelValue;
 
  188    static value_type min_value() { 
return MinVal::apply(); }
 
  189    static value_type max_value() { 
return MaxVal::apply(); }
 
  191    scoped_channel_value() = 
default;
 
  192    scoped_channel_value(scoped_channel_value 
const& other) : value_(other.value_) {}
 
  193    scoped_channel_value& operator=(scoped_channel_value 
const& other) = 
default;
 
  194    scoped_channel_value(BaseChannelValue value) : value_(value) {}
 
  195    scoped_channel_value& operator=(BaseChannelValue value)
 
  201    auto operator++() -> scoped_channel_value& { ++value_; 
return *
this; }
 
  202    auto operator--() -> scoped_channel_value& { --value_; 
return *
this; }
 
  204    auto operator++(
int) -> scoped_channel_value
 
  206        scoped_channel_value tmp=*
this;
 
  207        this->operator++(); 
return tmp;
 
  210    auto operator--(
int) -> scoped_channel_value
 
  212        scoped_channel_value tmp=*
this;
 
  213        this->operator--(); 
return tmp;
 
  216    template <
typename Scalar2>
 
  217    auto operator+=(Scalar2 v) -> scoped_channel_value& { value_+=v; 
return *
this; }
 
  219    template <
typename Scalar2>
 
  220    auto operator-=(Scalar2 v) -> scoped_channel_value& { value_-=v; 
return *
this; }
 
  222    template <
typename Scalar2>
 
  223    auto operator*=(Scalar2 v) -> scoped_channel_value& { value_*=v; 
return *
this; }
 
  225    template <
typename Scalar2>
 
  226    auto operator/=(Scalar2 v) -> scoped_channel_value& { value_/=v; 
return *
this; }
 
  228    operator BaseChannelValue()
 const { 
return value_; }
 
  230    BaseChannelValue value_{};
 
  234struct float_point_zero
 
  236    static constexpr T apply() { 
return 0.0f; }
 
  240struct float_point_one
 
  242    static constexpr T apply() { 
return 1.0f; }
 
  256template <
int NumBits>
 
  257struct min_fast_uint :
 
  262        typename std::conditional
 
  266            typename std::conditional
 
  276template <int NumBits>
 
  278    : std::conditional<NumBits < 32, std::uint32_t, std::uint64_t>
 
  281template <int NumBits>
 
  283    : std::conditional<NumBits <= 32, std::uint32_t, std::uint64_t>
 
  304template <int NumBits>
 
  305class packed_channel_value
 
  308    using integer_t = typename detail::min_fast_uint<NumBits>::type;
 
  310    using value_type = packed_channel_value<NumBits>;
 
  311    using reference = value_type&;
 
  312    using const_reference = value_type const&;
 
  313    using pointer = value_type*;
 
  314    using const_pointer = value_type const*;
 
  315    static constexpr bool is_mutable = true;
 
  317    static value_type min_value() { return 0; }
 
  318    static value_type max_value() { return low_bits_mask_t< NumBits >::sig_bits; }
 
  320    packed_channel_value() = default;
 
  321    packed_channel_value(integer_t v)
 
  323        value_ = static_cast<integer_t>(v & low_bits_mask_t<NumBits>::sig_bits_fast);
 
  326    template <typename Scalar>
 
  327    packed_channel_value(Scalar v)
 
  329        value_ = packed_channel_value(static_cast<integer_t>(v));
 
  332    static auto num_bits() -> unsigned int { return NumBits; }
 
  334    operator integer_t() const { return value_; }
 
  342template <std::size_t K>
 
  343struct static_copy_bytes
 
  345    void operator()(unsigned char const* from, unsigned char* to) const
 
  348        static_copy_bytes<K - 1>()(++from, ++to);
 
  353struct static_copy_bytes<0>
 
  355    void operator()(unsigned char const*, unsigned char*) const {}
 
  358template <typename Derived, typename BitField, int NumBits, bool IsMutable>
 
  359class packed_channel_reference_base
 
  362    using data_ptr_t = typename std::conditional<IsMutable, void*, void const*>::type;
 
  364    data_ptr_t _data_ptr;   
 
  366    using value_type = packed_channel_value<NumBits>;
 
  367    using reference = Derived const;
 
  368    using pointer = value_type*;
 
  369    using const_pointer = value_type const*;
 
  370    static constexpr int num_bits = NumBits;
 
  371    static constexpr bool is_mutable = IsMutable;
 
  373    static auto min_value() -> value_type { return channel_traits<value_type>::min_value(); }
 
  374    static auto max_value() -> value_type { return channel_traits<value_type>::max_value(); }
 
  376    using bitfield_t = BitField;
 
  377    using integer_t = typename value_type::integer_t;
 
  379    packed_channel_reference_base(data_ptr_t data_ptr) : _data_ptr(data_ptr) {}
 
  380    packed_channel_reference_base(packed_channel_reference_base const& ref) : _data_ptr(ref._data_ptr) {}
 
  382    auto operator=(integer_t v) const -> Derived const& { set(v); return derived(); }
 
  384    auto operator++() const -> Derived const& { set(get()+1); 
return derived(); }
 
  385    auto operator--() const -> Derived const& { set(get()-1); 
return derived(); }
 
  387    auto operator++(
int) 
const -> Derived
 
  389        Derived tmp=derived();
 
  390        this->operator++(); 
return tmp;
 
  393    auto operator--(
int) 
const -> Derived
 
  395        Derived tmp=derived();
 
  400    template <
typename Scalar2>
 
  401    auto operator+=(Scalar2 v) 
const -> Derived 
const&
 
  403        set( 
static_cast<integer_t
>(  get() + v ));
 
  407    template <
typename Scalar2>
 
  408    auto operator-=(Scalar2 v) 
const -> Derived 
const&
 
  410        set( 
static_cast<integer_t
>(  get() - v )); 
return derived();
 
  413    template <
typename Scalar2>
 
  414    auto operator*=(Scalar2 v) 
const -> Derived 
const&
 
  416        set( 
static_cast<integer_t
>(  get() * v ));
 
  420    template <
typename Scalar2>
 
  421    auto operator/=(Scalar2 v) 
const -> Derived 
const&
 
  423        set( 
static_cast<integer_t
>(  get() / v ));
 
  427    operator integer_t()
 const { 
return get(); }
 
  428    auto operator&() const -> data_ptr_t {
return _data_ptr;}
 
  432    using num_value_t = 
typename detail::num_value_fn<NumBits>::type;
 
  433    using max_value_t = 
typename detail::max_value_fn<NumBits>::type;
 
  435    static const num_value_t num_values = 
static_cast< num_value_t 
>( 1 ) << NumBits ;
 
  436    static const max_value_t max_val    = 
static_cast< max_value_t 
>( num_values - 1 );
 
  438#if defined(BOOST_GIL_CONFIG_HAS_UNALIGNED_ACCESS) 
  439    const bitfield_t& get_data()
                      const { 
return *
static_cast<const bitfield_t*
>(_data_ptr); }
 
  440    void              set_data(
const bitfield_t& val)
 const {        *
static_cast<      bitfield_t*
>(_data_ptr) = val; }
 
  442    auto get_data() const -> bitfield_t
 
  445        static_copy_bytes<sizeof(bitfield_t) >()(gil_reinterpret_cast_c<const unsigned char*>(_data_ptr),gil_reinterpret_cast<unsigned char*>(&ret));
 
  449    void set_data(bitfield_t 
const& val)
 const 
  451        static_copy_bytes<sizeof(bitfield_t) >()(gil_reinterpret_cast_c<const unsigned char*>(&val),gil_reinterpret_cast<unsigned char*>(_data_ptr));
 
  456    void set(integer_t value)
 const {     
 
  457        this->derived().set_unsafe(((value % num_values) + num_values) % num_values);
 
  459    auto get() const -> integer_t { 
return derived().get(); }
 
  460    auto derived() const -> Derived const& { 
return static_cast<const Derived&
>(*this); }
 
  481template <
typename BitField, 
int FirstBit, 
int NumBits, 
bool IsMutable>
 
  482class packed_channel_reference;
 
  487template <
typename BitField, 
int NumBits, 
bool IsMutable>
 
  488class packed_dynamic_channel_reference;
 
  492template <
typename BitField, 
int FirstBit, 
int NumBits>
 
  493class packed_channel_reference<BitField, FirstBit, NumBits, false>
 
  494    : 
public detail::packed_channel_reference_base
 
  496            packed_channel_reference<BitField, FirstBit, NumBits, false>,
 
  502    using parent_t = detail::packed_channel_reference_base
 
  504            packed_channel_reference<BitField, FirstBit, NumBits, false>,
 
  510    friend class packed_channel_reference<BitField, FirstBit, NumBits, true>;
 
  512    static const BitField channel_mask = 
static_cast<BitField
>(parent_t::max_val) << FirstBit;
 
  514    void operator=(packed_channel_reference 
const&);
 
  516    using const_reference = packed_channel_reference<BitField,FirstBit,NumBits,false> 
const;
 
  517    using mutable_reference = packed_channel_reference<BitField,FirstBit,NumBits,true> 
const;
 
  518    using integer_t = 
typename parent_t::integer_t;
 
  520    explicit packed_channel_reference(
const void* data_ptr) : parent_t(data_ptr) {}
 
  521    packed_channel_reference(
const packed_channel_reference& ref) : parent_t(ref._data_ptr) {}
 
  522    packed_channel_reference(
const mutable_reference& ref) : parent_t(ref._data_ptr) {}
 
  524    auto first_bit() const -> 
unsigned int { 
return FirstBit; }
 
  526    auto get() const -> integer_t { 
return integer_t((this->get_data()&channel_mask) >> FirstBit); }
 
  531template <
typename BitField, 
int FirstBit, 
int NumBits>
 
  532class packed_channel_reference<BitField,FirstBit,NumBits,true>
 
  533   : 
public detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,true>,BitField,NumBits,true>
 
  535    using parent_t = detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,true>,BitField,NumBits,
true>;
 
  536    friend class packed_channel_reference<BitField,FirstBit,NumBits,false>;
 
  538    static const BitField channel_mask = 
static_cast< BitField 
>( parent_t::max_val ) << FirstBit;
 
  541    using const_reference = packed_channel_reference<BitField,FirstBit,NumBits,false> 
const;
 
  542    using mutable_reference = packed_channel_reference<BitField,FirstBit,NumBits,true> 
const;
 
  543    using integer_t = 
typename parent_t::integer_t;
 
  545    explicit packed_channel_reference(
void* data_ptr) : parent_t(data_ptr) {}
 
  546    packed_channel_reference(
const packed_channel_reference& ref) : parent_t(ref._data_ptr) {}
 
  548    packed_channel_reference 
const& operator=(integer_t value)
 const 
  550        BOOST_ASSERT(value <= parent_t::max_val);
 
  555    auto operator=(mutable_reference 
const& ref) 
const -> packed_channel_reference 
const& { set_from_reference(ref.get_data()); 
return *
this; }
 
  556    auto operator=(const_reference 
const& ref) 
const -> packed_channel_reference 
const& { set_from_reference(ref.get_data()); 
return *
this; }
 
  558    template <
bool Mutable1>
 
  559    auto operator=(packed_dynamic_channel_reference<BitField,NumBits,Mutable1> 
const& ref) 
const -> packed_channel_reference 
const& { set_unsafe(ref.get()); 
return *
this; }
 
  561    auto first_bit() const -> 
unsigned int { 
return FirstBit; }
 
  563    auto get() const -> integer_t { 
return integer_t((this->get_data()&channel_mask) >> FirstBit); }
 
  564    void set_unsafe(integer_t value)
 const { this->set_data((this->get_data() & ~channel_mask) | (( 
static_cast< BitField 
>( value )<<FirstBit))); }
 
  567    void set_from_reference(
const BitField& other_bits)
 const { this->set_data((this->get_data() & ~channel_mask) | (other_bits & channel_mask)); }
 
  581template <
typename BF, 
int FB, 
int NB, 
bool M, 
typename R>
 
  583void swap(boost::gil::packed_channel_reference<BF, FB, NB, M> 
const x, R& y)
 
  585    boost::gil::swap_proxy
 
  587        typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type
 
  594template <
typename BF, 
int FB, 
int NB, 
bool M>
 
  597    typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type& x,
 
  598    boost::gil::packed_channel_reference<BF, FB, NB, M> 
const y)
 
  600    boost::gil::swap_proxy
 
  602        typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type
 
  608template <
typename BF, 
int FB, 
int NB, 
bool M> 
inline 
  610    boost::gil::packed_channel_reference<BF, FB, NB, M> 
const x,
 
  611    boost::gil::packed_channel_reference<BF, FB, NB, M> 
const y)
 
  613    boost::gil::swap_proxy
 
  615        typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type
 
  621namespace boost { 
namespace gil {
 
  641template <
typename BitField, 
int NumBits>
 
  642class packed_dynamic_channel_reference<BitField,NumBits,false>
 
  643   : 
public detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,false>,BitField,NumBits,false>
 
  645    using parent_t = detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,false>,BitField,NumBits,
false>;
 
  646    friend class packed_dynamic_channel_reference<BitField,NumBits,true>;
 
  650    void operator=(
const packed_dynamic_channel_reference&);
 
  652    using const_reference = packed_dynamic_channel_reference<BitField,NumBits,false> 
const;
 
  653    using mutable_reference = packed_dynamic_channel_reference<BitField,NumBits,true> 
const;
 
  654    using integer_t = 
typename parent_t::integer_t;
 
  656    packed_dynamic_channel_reference(
void const* data_ptr, 
unsigned first_bit) : parent_t(data_ptr), _first_bit(first_bit) {}
 
  657    packed_dynamic_channel_reference(const_reference 
const&   ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {}
 
  658    packed_dynamic_channel_reference(mutable_reference 
const& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {}
 
  660    auto first_bit() 
const -> 
unsigned int { 
return _first_bit; }
 
  662    auto get() 
const -> integer_t
 
  664        const BitField channel_mask = 
static_cast< integer_t 
>( parent_t::max_val ) <<_first_bit;
 
  665        return static_cast< integer_t 
>(( this->get_data()&channel_mask ) >> _first_bit );
 
  672template <
typename BitField, 
int NumBits>
 
  673class packed_dynamic_channel_reference<BitField,NumBits,true>
 
  674   : 
public detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,true>,BitField,NumBits,true>
 
  676    using parent_t = detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,true>,BitField,NumBits,
true>;
 
  677    friend class packed_dynamic_channel_reference<BitField,NumBits,false>;
 
  682    using const_reference = packed_dynamic_channel_reference<BitField,NumBits,false> 
const;
 
  683    using mutable_reference = packed_dynamic_channel_reference<BitField,NumBits,true> 
const;
 
  684    using integer_t = 
typename parent_t::integer_t;
 
  686    packed_dynamic_channel_reference(
void* data_ptr, 
unsigned first_bit) : parent_t(data_ptr), _first_bit(first_bit) {}
 
  687    packed_dynamic_channel_reference(packed_dynamic_channel_reference 
const& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {}
 
  689    auto operator=(integer_t value) 
const -> packed_dynamic_channel_reference 
const&
 
  691        BOOST_ASSERT(value <= parent_t::max_val);
 
  696    auto operator=(mutable_reference 
const& ref) 
const -> packed_dynamic_channel_reference 
const& {  set_unsafe(ref.get()); 
return *
this; }
 
  697    auto operator=(const_reference 
const& ref) 
const -> packed_dynamic_channel_reference 
const& {  set_unsafe(ref.get()); 
return *
this; }
 
  699    template <
typename BitField1, 
int FirstBit1, 
bool Mutable1>
 
  700    auto operator=(packed_channel_reference<BitField1, FirstBit1, NumBits, Mutable1> 
const& ref) 
const -> packed_dynamic_channel_reference 
const&
 
  702        set_unsafe(ref.get());
 
  706    auto first_bit() 
const -> 
unsigned int { 
return _first_bit; }
 
  708    auto get() 
const -> integer_t
 
  710        BitField 
const channel_mask = 
static_cast< integer_t 
>( parent_t::max_val ) << _first_bit;
 
  711        return static_cast< integer_t 
>(( this->get_data()&channel_mask ) >> _first_bit );
 
  714    void set_unsafe(integer_t value)
 const {
 
  715        const BitField channel_mask = 
static_cast< integer_t 
>( parent_t::max_val ) << _first_bit;
 
  716        this->set_data((this->get_data() & ~channel_mask) | value<<_first_bit);
 
  731template <
typename BF, 
int NB, 
bool M, 
typename R> 
inline 
  732void swap(
const boost::gil::packed_dynamic_channel_reference<BF,NB,M> x, R& y) {
 
  733    boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y);
 
  739template <
typename BF, 
int NB, 
bool M> 
inline 
  740void 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) {
 
  741    boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y);
 
  746template <
typename BF, 
int NB, 
bool M> 
inline 
  747void swap(
const boost::gil::packed_dynamic_channel_reference<BF,NB,M> x, 
const boost::gil::packed_dynamic_channel_reference<BF,NB,M> y) {
 
  748    boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y);
 
  753namespace boost { 
namespace gil {
 
  755struct base_channel_type_impl { 
using type = T; };
 
  758struct base_channel_type_impl<packed_channel_value<N> >
 
  759{ 
using type = 
typename packed_channel_value<N>::integer_t; };
 
  761template <
typename B, 
int F, 
int N, 
bool M>
 
  762struct base_channel_type_impl<packed_channel_reference<B, F, N, M> >
 
  764    using type = 
typename packed_channel_reference<B,F,N,M>::integer_t;
 
  767template <
typename B, 
int N, 
bool M>
 
  768struct base_channel_type_impl<packed_dynamic_channel_reference<B, N, M> >
 
  770    using type = 
typename packed_dynamic_channel_reference<B,N,M>::integer_t;
 
  773template <
typename ChannelValue, 
typename MinV, 
typename MaxV>
 
  774struct base_channel_type_impl<scoped_channel_value<ChannelValue, MinV, MaxV> >
 
  775{ 
using type = ChannelValue; };
 
  778struct base_channel_type : base_channel_type_impl<typename std::remove_cv<T>::type> {};
 
defined(BOOST_NO_CXX17_HDR_MEMORY_RESOURCE)
Definition: algorithm.hpp:36