200 lines
5.1 KiB
C++
200 lines
5.1 KiB
C++
//
|
|
// Copyright 2007-2008 Christian Henning
|
|
//
|
|
// Distributed under 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_ROW_BUFFER_HELPER_HPP
|
|
#define BOOST_GIL_IO_ROW_BUFFER_HELPER_HPP
|
|
|
|
// TODO: Shall we move toolbox to core?
|
|
#include <boost/gil/extension/toolbox/metafunctions/is_bit_aligned.hpp>
|
|
#include <boost/gil/extension/toolbox/metafunctions/is_homogeneous.hpp>
|
|
#include <boost/gil/extension/toolbox/metafunctions/pixel_bit_size.hpp>
|
|
|
|
#include <boost/gil/detail/mp11.hpp>
|
|
#include <boost/gil/io/typedefs.hpp>
|
|
|
|
#include <cstddef>
|
|
#include <type_traits>
|
|
#include <vector>
|
|
|
|
namespace boost { namespace gil { namespace detail {
|
|
|
|
template< typename Pixel
|
|
, typename DummyT = void
|
|
>
|
|
struct row_buffer_helper
|
|
{
|
|
using element_t = Pixel;
|
|
using buffer_t = std::vector<element_t>;
|
|
using iterator_t = typename buffer_t::iterator;
|
|
|
|
row_buffer_helper( std::size_t width
|
|
, bool
|
|
)
|
|
: _row_buffer( width )
|
|
{}
|
|
|
|
element_t* data() { return &_row_buffer[0]; }
|
|
|
|
iterator_t begin() { return _row_buffer.begin(); }
|
|
iterator_t end() { return _row_buffer.end(); }
|
|
|
|
buffer_t& buffer() { return _row_buffer; }
|
|
|
|
private:
|
|
|
|
buffer_t _row_buffer;
|
|
};
|
|
|
|
template <typename Pixel>
|
|
struct row_buffer_helper
|
|
<
|
|
Pixel,
|
|
typename std::enable_if
|
|
<
|
|
is_bit_aligned<Pixel>::value
|
|
>::type
|
|
>
|
|
{
|
|
using element_t = byte_t;
|
|
using buffer_t = std::vector<element_t>;
|
|
using pixel_type = Pixel;
|
|
using iterator_t = bit_aligned_pixel_iterator<pixel_type>;
|
|
|
|
row_buffer_helper(std::size_t width, bool in_bytes)
|
|
: _c{( width * pixel_bit_size< pixel_type >::value) >> 3}
|
|
, _r{width * pixel_bit_size< pixel_type >::value - (_c << 3)}
|
|
{
|
|
if (in_bytes)
|
|
{
|
|
_row_buffer.resize(width);
|
|
}
|
|
else
|
|
{
|
|
// add one byte if there are remaining bits
|
|
_row_buffer.resize(_c + (_r != 0));
|
|
}
|
|
}
|
|
|
|
element_t* data() { return &_row_buffer[0]; }
|
|
|
|
iterator_t begin() { return iterator_t( &_row_buffer.front(),0 ); }
|
|
iterator_t end() { return _r == 0 ? iterator_t( &_row_buffer.back() + 1, 0 )
|
|
: iterator_t( &_row_buffer.back() , (int) _r );
|
|
}
|
|
|
|
buffer_t& buffer() { return _row_buffer; }
|
|
|
|
private:
|
|
|
|
// For instance 25 pixels of rgb2 type would be:
|
|
// overall 25 pixels * 3 channels * 2 bits/channel = 150 bits
|
|
// c = 18 bytes
|
|
// r = 6 bits
|
|
|
|
std::size_t _c; // number of full bytes
|
|
std::size_t _r; // number of remaining bits
|
|
|
|
buffer_t _row_buffer;
|
|
};
|
|
|
|
template<typename Pixel>
|
|
struct row_buffer_helper
|
|
<
|
|
Pixel,
|
|
typename std::enable_if
|
|
<
|
|
mp11::mp_and
|
|
<
|
|
typename is_bit_aligned<Pixel>::type,
|
|
typename is_homogeneous<Pixel>::type
|
|
>::value
|
|
>
|
|
>
|
|
{
|
|
using element_t = byte_t;
|
|
using buffer_t = std::vector<element_t>;
|
|
using pixel_type = Pixel;
|
|
using iterator_t = bit_aligned_pixel_iterator<pixel_type>;
|
|
|
|
row_buffer_helper( std::size_t width
|
|
, bool in_bytes
|
|
)
|
|
: _c( ( width
|
|
* num_channels< pixel_type >::value
|
|
* channel_type< pixel_type >::type::num_bits
|
|
)
|
|
>> 3
|
|
)
|
|
|
|
, _r( width
|
|
* num_channels< pixel_type >::value
|
|
* channel_type< pixel_type >::type::num_bits
|
|
- ( _c << 3 )
|
|
)
|
|
{
|
|
if( in_bytes )
|
|
{
|
|
_row_buffer.resize( width );
|
|
}
|
|
else
|
|
{
|
|
// add one byte if there are remaining bits
|
|
_row_buffer.resize( _c + ( _r!=0 ));
|
|
}
|
|
}
|
|
|
|
element_t* data() { return &_row_buffer[0]; }
|
|
|
|
iterator_t begin() { return iterator_t( &_row_buffer.front(),0 ); }
|
|
iterator_t end() { return _r == 0 ? iterator_t( &_row_buffer.back() + 1, 0 )
|
|
: iterator_t( &_row_buffer.back() , (int) _r );
|
|
}
|
|
|
|
buffer_t& buffer() { return _row_buffer; }
|
|
|
|
private:
|
|
|
|
// For instance 25 pixels of rgb2 type would be:
|
|
// overall 25 pixels * 3 channels * 2 bits/channel = 150 bits
|
|
// c = 18 bytes
|
|
// r = 6 bits
|
|
|
|
std::size_t _c; // number of full bytes
|
|
std::size_t _r; // number of remaining bits
|
|
|
|
buffer_t _row_buffer;
|
|
};
|
|
|
|
template <typename View, typename D = void>
|
|
struct row_buffer_helper_view : row_buffer_helper<typename View::value_type>
|
|
{
|
|
row_buffer_helper_view(std::size_t width, bool in_bytes)
|
|
: row_buffer_helper<typename View::value_type>(width, in_bytes)
|
|
{}
|
|
};
|
|
|
|
template <typename View>
|
|
struct row_buffer_helper_view
|
|
<
|
|
View,
|
|
typename std::enable_if
|
|
<
|
|
is_bit_aligned<typename View::value_type>::value
|
|
>::type
|
|
> : row_buffer_helper<typename View::reference>
|
|
{
|
|
row_buffer_helper_view(std::size_t width, bool in_bytes)
|
|
: row_buffer_helper<typename View::reference>(width, in_bytes)
|
|
{}
|
|
};
|
|
|
|
} // namespace detail
|
|
} // namespace gil
|
|
} // namespace boost
|
|
|
|
#endif
|