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> 20 #include <type_traits> 22 #ifdef BOOST_GIL_DOXYGEN_ONLY 23 #define BOOST_GIL_CONFIG_HAS_UNALIGNED_ACCESS 33 #ifdef BOOST_GIL_CONFIG_HAS_UNALIGNED_ACCESS 34 #if defined(sun) || defined(__sun) || \ // SunOS 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__) 48 BOOST_PRAGMA_MESSAGE(
"CAUTION: Unaligned access tolerated on little-endian may cause undefined behaviour")
50 #error Unaligned access disabled for unknown platforms and architectures 52 #endif // defined(BOOST_GIL_CONFIG_HAS_UNALIGNED_ACCESS) 54 namespace boost {
namespace gil {
73 template <
typename T,
bool IsClass>
74 struct channel_traits_impl;
78 struct 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(); }
92 struct 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)(); }
105 template <
typename T>
106 struct 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;
133 template <
typename T>
134 struct channel_traits : detail::channel_traits_impl<T, std::is_class<T>::value> {};
137 template <
typename T>
138 struct channel_traits<T&> : channel_traits<T> {};
141 template <
typename T>
142 struct 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;
176 template <
typename BaseChannelValue,
typename MinVal,
typename MaxVal>
177 struct 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(
const scoped_channel_value& c) : value_(c.value_) {}
193 scoped_channel_value(BaseChannelValue val) : value_(val) {}
195 scoped_channel_value& operator++() { ++value_;
return *
this; }
196 scoped_channel_value& operator--() { --value_;
return *
this; }
198 scoped_channel_value operator++(
int) { scoped_channel_value tmp=*
this; this->operator++();
return tmp; }
199 scoped_channel_value operator--(
int) { scoped_channel_value tmp=*
this; this->operator--();
return tmp; }
201 template <
typename Scalar2> scoped_channel_value& operator+=(Scalar2 v) { value_+=v;
return *
this; }
202 template <
typename Scalar2> scoped_channel_value& operator-=(Scalar2 v) { value_-=v;
return *
this; }
203 template <
typename Scalar2> scoped_channel_value& operator*=(Scalar2 v) { value_*=v;
return *
this; }
204 template <
typename Scalar2> scoped_channel_value& operator/=(Scalar2 v) { value_/=v;
return *
this; }
206 scoped_channel_value& operator=(BaseChannelValue v) { value_=v;
return *
this; }
207 operator BaseChannelValue()
const {
return value_; }
209 BaseChannelValue value_{};
212 template <
typename T>
213 struct float_point_zero
215 static constexpr T apply() {
return 0.0f; }
218 template <
typename T>
219 struct float_point_one
221 static constexpr T apply() {
return 1.0f; }
235 template <
int NumBits>
236 struct min_fast_uint :
241 typename std::conditional
245 typename std::conditional
255 template <int NumBits>
257 : std::conditional<NumBits < 32, std::uint32_t, std::uint64_t>
260 template <int NumBits>
262 : std::conditional<NumBits <= 32, std::uint32_t, std::uint64_t>
283 template <int NumBits>
284 class packed_channel_value
287 using integer_t = typename detail::min_fast_uint<NumBits>::type;
289 using value_type = packed_channel_value<NumBits>;
290 using reference = value_type&;
291 using const_reference = value_type const&;
292 using pointer = value_type*;
293 using const_pointer = value_type const*;
294 static constexpr bool is_mutable = true;
296 static value_type min_value() { return 0; }
297 static value_type max_value() { return low_bits_mask_t< NumBits >::sig_bits; }
299 packed_channel_value() = default;
300 packed_channel_value(integer_t v)
302 value_ = static_cast<integer_t>(v & low_bits_mask_t<NumBits>::sig_bits_fast);
305 template <typename Scalar>
306 packed_channel_value(Scalar v)
308 value_ = packed_channel_value(static_cast<integer_t>(v));
311 static unsigned int num_bits() { return NumBits; }
313 operator integer_t() const { return value_; }
321 template <std::size_t K>
322 struct static_copy_bytes
324 void operator()(unsigned char const* from, unsigned char* to) const
327 static_copy_bytes<K - 1>()(++from, ++to);
332 struct static_copy_bytes<0>
334 void operator()(unsigned char const*, unsigned char*) const {}
337 template <typename Derived, typename BitField, int NumBits, bool IsMutable>
338 class packed_channel_reference_base
341 using data_ptr_t = typename std::conditional<IsMutable, void*, void const*>::type;
343 data_ptr_t _data_ptr;
345 using value_type = packed_channel_value<NumBits>;
346 using reference = const Derived;
347 using pointer = value_type *;
348 using const_pointer = const value_type *;
349 static constexpr int num_bits = NumBits;
350 static constexpr bool is_mutable = IsMutable;
352 static value_type min_value() { return channel_traits<value_type>::min_value(); }
353 static value_type max_value() { return channel_traits<value_type>::max_value(); }
355 using bitfield_t = BitField;
356 using integer_t = typename value_type::integer_t;
358 packed_channel_reference_base(data_ptr_t data_ptr) : _data_ptr(data_ptr) {}
359 packed_channel_reference_base(const packed_channel_reference_base& ref) : _data_ptr(ref._data_ptr) {}
360 const Derived& operator=(integer_t v) const { set(v); return derived(); }
362 const Derived& operator++() const { set(get()+1); return derived(); }
363 const Derived& operator--() const { set(get()-1); return derived(); }
365 Derived operator++(int) const { Derived tmp=derived(); this->operator++(); return tmp; }
366 Derived operator--(int) const { Derived tmp=derived(); this->operator--(); return tmp; }
368 template <typename Scalar2> const Derived& operator+=(Scalar2 v) const { set( static_cast<integer_t>( get() + v )); return derived(); }
369 template <typename Scalar2> const Derived& operator-=(Scalar2 v) const { set( static_cast<integer_t>( get() - v )); return derived(); }
370 template <typename Scalar2> const Derived& operator*=(Scalar2 v) const { set( static_cast<integer_t>( get() * v )); return derived(); }
371 template <typename Scalar2> const Derived& operator/=(Scalar2 v) const { set( static_cast<integer_t>( get() / v )); return derived(); }
373 operator integer_t() const { return get(); }
374 data_ptr_t operator &() const {return _data_ptr;}
377 using num_value_t = typename detail::num_value_fn<NumBits>::type;
378 using max_value_t = typename detail::max_value_fn<NumBits>::type;
380 static const num_value_t num_values = static_cast< num_value_t >( 1 ) << NumBits ;
381 static const max_value_t max_val = static_cast< max_value_t >( num_values - 1 );
383 #if defined(BOOST_GIL_CONFIG_HAS_UNALIGNED_ACCESS)
384 const bitfield_t& get_data() const { return *static_cast<const bitfield_t*>(_data_ptr); }
385 void set_data(const bitfield_t& val) const { *static_cast< bitfield_t*>(_data_ptr) = val; }
387 bitfield_t get_data() const {
389 static_copy_bytes<sizeof(bitfield_t) >()(gil_reinterpret_cast_c<const unsigned char*>(_data_ptr),gil_reinterpret_cast<unsigned char*>(&ret));
392 void set_data(const bitfield_t& val) const {
393 static_copy_bytes<sizeof(bitfield_t) >()(gil_reinterpret_cast_c<const unsigned char*>(&val),gil_reinterpret_cast<unsigned char*>(_data_ptr));
398 void set(integer_t value) const {
399 this->derived().set_unsafe(((value % num_values) + num_values) % num_values);
401 integer_t get() const { return derived().get(); }
402 const Derived& derived() const { return static_cast<const Derived&>(*this); }
423 template <typename BitField, int FirstBit, int NumBits, bool IsMutable>
424 class packed_channel_reference;
429 template <typename BitField, int NumBits, bool IsMutable>
430 class packed_dynamic_channel_reference;
434 template <typename BitField, int FirstBit, int NumBits>
435 class packed_channel_reference<BitField, FirstBit, NumBits, false>
436 : public detail::packed_channel_reference_base
438 packed_channel_reference<BitField, FirstBit, NumBits, false>,
444 using parent_t = detail::packed_channel_reference_base
446 packed_channel_reference<BitField, FirstBit, NumBits, false>,
452 friend class packed_channel_reference<BitField, FirstBit, NumBits, true>;
454 static const BitField channel_mask = static_cast<BitField>(parent_t::max_val) << FirstBit;
456 void operator=(packed_channel_reference const&);
458 using const_reference = packed_channel_reference<BitField,FirstBit,NumBits,false> const;
459 using mutable_reference = packed_channel_reference<BitField,FirstBit,NumBits,true> const;
460 using integer_t = typename parent_t::integer_t;
462 explicit packed_channel_reference(const void* data_ptr) : parent_t(data_ptr) {}
463 packed_channel_reference(const packed_channel_reference& ref) : parent_t(ref._data_ptr) {}
464 packed_channel_reference(const mutable_reference& ref) : parent_t(ref._data_ptr) {}
466 unsigned first_bit() const { return FirstBit; }
468 integer_t get() const { return integer_t((this->get_data()&channel_mask) >> FirstBit); }
473 template <typename BitField, int FirstBit, int NumBits>
474 class packed_channel_reference<BitField,FirstBit,NumBits,true>
475 : public detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,true>,BitField,NumBits,true>
477 using parent_t = detail::packed_channel_reference_base<packed_channel_reference<BitField,FirstBit,NumBits,true>,BitField,NumBits,true>;
478 friend class packed_channel_reference<BitField,FirstBit,NumBits,false>;
480 static const BitField channel_mask = static_cast< BitField >( parent_t::max_val ) << FirstBit;
483 using const_reference = packed_channel_reference<BitField,FirstBit,NumBits,false> const;
484 using mutable_reference = packed_channel_reference<BitField,FirstBit,NumBits,true> const;
485 using integer_t = typename parent_t::integer_t;
487 explicit packed_channel_reference(void* data_ptr) : parent_t(data_ptr) {}
488 packed_channel_reference(const packed_channel_reference& ref) : parent_t(ref._data_ptr) {}
490 packed_channel_reference const& operator=(integer_t value) const
492 BOOST_ASSERT(value <= parent_t::max_val);
497 const packed_channel_reference& operator=(const mutable_reference& ref) const { set_from_reference(ref.get_data()); return *this; }
498 const packed_channel_reference& operator=(const const_reference& ref) const { set_from_reference(ref.get_data()); return *this; }
500 template <bool Mutable1>
501 const packed_channel_reference& operator=(const packed_dynamic_channel_reference<BitField,NumBits,Mutable1>& ref) const { set_unsafe(ref.get()); return *this; }
503 unsigned first_bit() const { return FirstBit; }
505 integer_t get() const { return integer_t((this->get_data()&channel_mask) >> FirstBit); }
506 void set_unsafe(integer_t value) const { this->set_data((this->get_data() & ~channel_mask) | (( static_cast< BitField >( value )<<FirstBit))); }
508 void set_from_reference(const BitField& other_bits) const { this->set_data((this->get_data() & ~channel_mask) | (other_bits & channel_mask)); }
522 template <typename BF, int FB, int NB, bool M, typename R>
524 void swap(boost::gil::packed_channel_reference<BF, FB, NB, M> const x, R& y)
526 boost::gil::swap_proxy
528 typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type
535 template <typename BF, int FB, int NB, bool M>
538 typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type& x,
539 boost::gil::packed_channel_reference<BF, FB, NB, M> const y)
541 boost::gil::swap_proxy
543 typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type
549 template <typename BF, int FB, int NB, bool M> inline
551 boost::gil::packed_channel_reference<BF, FB, NB, M> const x,
552 boost::gil::packed_channel_reference<BF, FB, NB, M> const y)
554 boost::gil::swap_proxy
556 typename boost::gil::packed_channel_reference<BF, FB, NB, M>::value_type
562 namespace boost { namespace gil {
582 template <typename BitField, int NumBits>
583 class packed_dynamic_channel_reference<BitField,NumBits,false>
584 : public detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,false>,BitField,NumBits,false>
586 using parent_t = detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,false>,BitField,NumBits,false>;
587 friend class packed_dynamic_channel_reference<BitField,NumBits,true>;
591 void operator=(const packed_dynamic_channel_reference&);
593 using const_reference = packed_dynamic_channel_reference<BitField,NumBits,false> const;
594 using mutable_reference = packed_dynamic_channel_reference<BitField,NumBits,true> const;
595 using integer_t = typename parent_t::integer_t;
597 packed_dynamic_channel_reference(const void* data_ptr, unsigned first_bit) : parent_t(data_ptr), _first_bit(first_bit) {}
598 packed_dynamic_channel_reference(const const_reference& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {}
599 packed_dynamic_channel_reference(const mutable_reference& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {}
601 unsigned first_bit() const { return _first_bit; }
603 integer_t get() const {
604 const BitField channel_mask = static_cast< integer_t >( parent_t::max_val ) <<_first_bit;
605 return static_cast< integer_t >(( this->get_data()&channel_mask ) >> _first_bit );
612 template <typename BitField, int NumBits>
613 class packed_dynamic_channel_reference<BitField,NumBits,true>
614 :
public detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,true>,BitField,NumBits,true>
616 using parent_t = detail::packed_channel_reference_base<packed_dynamic_channel_reference<BitField,NumBits,true>,BitField,NumBits,
true>;
617 friend class packed_dynamic_channel_reference<BitField,NumBits,false>;
624 using integer_t =
typename parent_t::integer_t;
626 packed_dynamic_channel_reference(
void* data_ptr,
unsigned first_bit) : parent_t(data_ptr), _first_bit(first_bit) {}
627 packed_dynamic_channel_reference(
const packed_dynamic_channel_reference& ref) : parent_t(ref._data_ptr), _first_bit(ref._first_bit) {}
629 packed_dynamic_channel_reference
const& operator=(integer_t value)
const 631 BOOST_ASSERT(value <= parent_t::max_val);
636 const packed_dynamic_channel_reference& operator=(
const mutable_reference& ref)
const { set_unsafe(ref.get());
return *
this; }
637 const packed_dynamic_channel_reference& operator=(
const const_reference& ref)
const { set_unsafe(ref.get());
return *
this; }
639 template <
typename BitField1,
int FirstBit1,
bool Mutable1>
640 const packed_dynamic_channel_reference& operator=(
const packed_channel_reference<BitField1, FirstBit1, NumBits, Mutable1>& ref)
const 641 { set_unsafe(ref.get());
return *
this; }
643 unsigned first_bit()
const {
return _first_bit; }
645 integer_t
get()
const {
646 const BitField channel_mask =
static_cast< integer_t
>( parent_t::max_val ) << _first_bit;
647 return static_cast< integer_t
>(( this->get_data()&channel_mask ) >> _first_bit );
650 void set_unsafe(integer_t value)
const {
651 const BitField channel_mask =
static_cast< integer_t
>( parent_t::max_val ) << _first_bit;
652 this->set_data((this->get_data() & ~channel_mask) | value<<_first_bit);
667 template <
typename BF,
int NB,
bool M,
typename R>
inline 668 void swap(
const boost::gil::packed_dynamic_channel_reference<BF,NB,M> x, R& y) {
669 boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y);
675 template <
typename BF,
int NB,
bool M>
inline 676 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) {
677 boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y);
682 template <
typename BF,
int NB,
bool M>
inline 683 void swap(
const boost::gil::packed_dynamic_channel_reference<BF,NB,M> x,
const boost::gil::packed_dynamic_channel_reference<BF,NB,M> y) {
684 boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y);
689 namespace boost {
namespace gil {
690 template <
typename T>
691 struct base_channel_type_impl {
using type = T; };
694 struct base_channel_type_impl<packed_channel_value<N> >
695 {
using type =
typename packed_channel_value<N>::integer_t; };
697 template <
typename B,
int F,
int N,
bool M>
698 struct base_channel_type_impl<packed_channel_reference<B, F, N, M> >
700 using type =
typename packed_channel_reference<B,F,N,M>::integer_t;
703 template <
typename B,
int N,
bool M>
704 struct base_channel_type_impl<packed_dynamic_channel_reference<B, N, M> >
706 using type =
typename packed_dynamic_channel_reference<B,N,M>::integer_t;
709 template <
typename ChannelValue,
typename MinV,
typename MaxV>
710 struct base_channel_type_impl<scoped_channel_value<ChannelValue, MinV, MaxV> >
711 {
using type = ChannelValue; };
713 template <
typename T>
714 struct base_channel_type : base_channel_type_impl<typename std::remove_cv<T>::type> {};
void swap(const boost::gil::packed_dynamic_channel_reference< BF, NB, M > x, const boost::gil::packed_dynamic_channel_reference< BF, NB, M > y)
swap for packed_dynamic_channel_reference
Definition: channel.hpp:683
Definition: algorithm.hpp:30
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.
Definition: channel.hpp:583
Definition: algorithm.hpp:133
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.
Definition: channel.hpp:613