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
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__)
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
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(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_{};
233 template <
typename T>
234 struct float_point_zero
236 static constexpr T apply() {
return 0.0f; }
239 template <
typename T>
240 struct float_point_one
242 static constexpr T apply() {
return 1.0f; }
256 template <
int NumBits>
257 struct min_fast_uint :
262 typename std::conditional
266 typename std::conditional
276 template <int NumBits>
278 : std::conditional<NumBits < 32, std::uint32_t, std::uint64_t>
281 template <int NumBits>
283 : std::conditional<NumBits <= 32, std::uint32_t, std::uint64_t>
304 template <int NumBits>
305 class 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_; }
342 template <std::size_t K>
343 struct static_copy_bytes
345 void operator()(unsigned char const* from, unsigned char* to) const
348 static_copy_bytes<K - 1>()(++from, ++to);
353 struct static_copy_bytes<0>
355 void operator()(unsigned char const*, unsigned char*) const {}
358 template <typename Derived, typename BitField, int NumBits, bool IsMutable>
359 class 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); }
481 template <
typename BitField,
int FirstBit,
int NumBits,
bool IsMutable>
482 class packed_channel_reference;
487 template <
typename BitField,
int NumBits,
bool IsMutable>
488 class packed_dynamic_channel_reference;
492 template <
typename BitField,
int FirstBit,
int NumBits>
493 class 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); }
531 template <
typename BitField,
int FirstBit,
int NumBits>
532 class 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)); }
581 template <
typename BF,
int FB,
int NB,
bool M,
typename R>
583 void 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
594 template <
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
608 template <
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
621 namespace boost {
namespace gil {
641 template <
typename BitField,
int NumBits>
642 class 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&);
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 );
672 template <
typename BitField,
int NumBits>
673 class 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>;
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);
731 template <
typename BF,
int NB,
bool M,
typename R>
inline
732 void 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);
739 template <
typename BF,
int NB,
bool M>
inline
740 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) {
741 boost::gil::swap_proxy<typename boost::gil::packed_dynamic_channel_reference<BF,NB,M>::value_type>(x,y);
746 template <
typename BF,
int NB,
bool M>
inline
747 void 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);
753 namespace boost {
namespace gil {
754 template <
typename T>
755 struct base_channel_type_impl {
using type = T; };
758 struct base_channel_type_impl<packed_channel_value<N> >
759 {
using type =
typename packed_channel_value<N>::integer_t; };
761 template <
typename B,
int F,
int N,
bool M>
762 struct base_channel_type_impl<packed_channel_reference<B, F, N, M> >
764 using type =
typename packed_channel_reference<B,F,N,M>::integer_t;
767 template <
typename B,
int N,
bool M>
768 struct base_channel_type_impl<packed_dynamic_channel_reference<B, N, M> >
770 using type =
typename packed_dynamic_channel_reference<B,N,M>::integer_t;
773 template <
typename ChannelValue,
typename MinV,
typename MaxV>
774 struct base_channel_type_impl<scoped_channel_value<ChannelValue, MinV, MaxV> >
775 {
using type = ChannelValue; };
777 template <
typename T>
778 struct base_channel_type : base_channel_type_impl<typename std::remove_cv<T>::type> {};
Models a constant subbyte channel reference whose bit offset is a runtime parameter....
Definition: channel.hpp:644
Models a mutable subbyte channel reference whose bit offset is a runtime parameter....
Definition: channel.hpp:675
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:747
defined(BOOST_NO_CXX17_HDR_MEMORY_RESOURCE)
Definition: algorithm.hpp:36