2012-05-16 15:03:49 +10:00
|
|
|
/*
|
2015-04-13 18:05:28 +10:00
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
2012-05-16 15:03:49 +10:00
|
|
|
*
|
2015-04-13 18:05:28 +10:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2015-02-04 15:59:06 +11:00
|
|
|
*
|
2015-04-13 18:05:28 +10:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
2012-05-16 15:03:49 +10:00
|
|
|
*
|
2015-02-04 15:59:06 +11:00
|
|
|
* Copyright 2011-2015 Danny Robson <danny@nerdcruft.net>
|
2012-05-16 15:03:49 +10:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "image.hpp"
|
|
|
|
|
2015-06-03 23:24:26 +10:00
|
|
|
#include "debug.hpp"
|
2012-05-16 15:03:49 +10:00
|
|
|
|
2015-05-18 14:09:34 +10:00
|
|
|
using util::image::buffer;
|
|
|
|
|
|
|
|
|
2015-02-05 20:34:38 +11:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
struct box {
|
|
|
|
static constexpr float support = 0.5f;
|
|
|
|
|
|
|
|
static constexpr float weight [[gnu::pure]] (float x)
|
|
|
|
{
|
|
|
|
return (x >= -0.5f && x <= 0.5f) ? 1.f : 0.f;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// XXX: Known to cause hard black ringing
|
|
|
|
template <size_t S>
|
|
|
|
struct lanczos {
|
|
|
|
static constexpr float support = float {S};
|
|
|
|
|
|
|
|
static float weight [[gnu::pure]] (float x)
|
|
|
|
{
|
|
|
|
if (x < 0.f)
|
|
|
|
x = -x;
|
|
|
|
|
|
|
|
if (x <= S)
|
|
|
|
return sincn (x) * sincn (x / S);
|
|
|
|
|
|
|
|
return 0.f;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template struct lanczos< 3>;
|
|
|
|
template struct lanczos< 4>;
|
|
|
|
template struct lanczos< 6>;
|
|
|
|
template struct lanczos<12>;
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
struct bspline {
|
|
|
|
static constexpr float support = 2.f;
|
|
|
|
|
|
|
|
static float weight [[gnu::pure]] (float x)
|
|
|
|
{
|
|
|
|
if (x < 0.f)
|
|
|
|
x = -x;
|
|
|
|
|
|
|
|
if (x < 1.f)
|
|
|
|
return .5f * x * x * x - x * x + (2.f / 3.f);
|
|
|
|
|
|
|
|
if (x < 2.f) {
|
|
|
|
x = 2.f - x;
|
|
|
|
return (1.f / 6.f) * x * x * x;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0.f;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// XXX: Known to cause hard black ringing
|
|
|
|
struct mitchell {
|
|
|
|
static constexpr float support = 2.f;
|
|
|
|
|
|
|
|
static float weight [[gnu::pure]] (float x)
|
|
|
|
{
|
|
|
|
return eval (x, 1.f / 3.f, 1.f / 3.f);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static float eval (float x, float B, float C)
|
|
|
|
{
|
|
|
|
if (x < 0.f)
|
|
|
|
x = -x;
|
|
|
|
|
|
|
|
if (x < 1.f) {
|
|
|
|
x = ( 12.f - 9.f * B - 6.f * C) * x * x * x
|
|
|
|
+ (-18.f + 12.f * B + 6.f * C) * x * x
|
|
|
|
+ 6.f - 2.f * B;
|
|
|
|
|
|
|
|
return x / 6.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (x < 2.f) {
|
|
|
|
x = ( -1.f * B - 6.f * C) * x * x * x
|
|
|
|
+ ( 6.f * B + 30.f * C) * x * x
|
|
|
|
+ (-12.f * B - 48.f * C) * x
|
|
|
|
+ 8.f * B + 24.f * C;
|
|
|
|
|
|
|
|
return x / 6.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0.f;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-05-26 16:25:21 +10:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
util::image::buffer<T>::buffer (util::extent2u _size):
|
|
|
|
buffer<T> (_size.w, _size.h)
|
|
|
|
{ ; }
|
|
|
|
|
|
|
|
|
2015-02-04 15:59:06 +11:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
util::image::buffer<T>::buffer (size_t _w, size_t _h, size_t _s):
|
|
|
|
w (_w),
|
|
|
|
h (_h),
|
|
|
|
s (_s),
|
2015-07-23 18:18:12 +10:00
|
|
|
m_data (std::make_unique<T[]> (_h * _s))
|
2015-02-04 15:59:06 +11:00
|
|
|
{
|
|
|
|
CHECK_NEQ (w * h, 0);
|
|
|
|
CHECK_GE (s, w);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
util::image::buffer<T>::buffer (size_t _w, size_t _h):
|
|
|
|
buffer<T> (_w, _h, _w)
|
|
|
|
{ ; }
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
util::image::buffer<T>::buffer (size_t _w,
|
|
|
|
size_t _h,
|
|
|
|
size_t _s,
|
|
|
|
std::unique_ptr<T[]> &&_data):
|
|
|
|
w (_w),
|
|
|
|
h (_h),
|
|
|
|
s (_s),
|
2015-05-18 14:09:34 +10:00
|
|
|
m_data (std::move (_data))
|
2015-02-04 15:59:06 +11:00
|
|
|
{
|
|
|
|
CHECK_NEQ (w * h, 0);
|
|
|
|
CHECK_GE (s, w);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
template <typename U>
|
|
|
|
util::image::buffer<U>
|
|
|
|
util::image::buffer<T>::alloc (void) const
|
|
|
|
{
|
|
|
|
return buffer<U> (w, h, s);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T, typename U>
|
|
|
|
static U
|
|
|
|
rescale (T v)
|
|
|
|
{
|
|
|
|
return v * sizeof (U) / sizeof (T);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
template <typename U>
|
|
|
|
util::image::buffer<U>
|
|
|
|
util::image::buffer<T>::clone (void) const
|
|
|
|
{
|
|
|
|
auto out = alloc<U> ();
|
|
|
|
|
2015-05-18 14:09:34 +10:00
|
|
|
std::transform (begin (), end (), out.begin (), renormalise<T,U>);
|
2015-02-04 15:59:06 +11:00
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
2015-02-05 20:34:38 +11:00
|
|
|
// HACK: This code hasn't really been tested with multiple filters (it is
|
|
|
|
// known to produce dark ringing artefacts with sinc) or robustly with
|
|
|
|
// fractional factors. But... it works more or less.
|
2015-02-04 15:59:06 +11:00
|
|
|
template <typename T>
|
|
|
|
util::image::buffer<T>
|
2015-02-05 20:34:38 +11:00
|
|
|
util::image::buffer<T>::downsample (float factor) const
|
2015-02-04 15:59:06 +11:00
|
|
|
{
|
2015-02-05 20:34:38 +11:00
|
|
|
CHECK_GE (factor, 0);
|
|
|
|
|
|
|
|
using filter = bspline;
|
|
|
|
|
|
|
|
const buffer &src = *this;
|
|
|
|
buffer dst (static_cast<size_t> (w / factor),
|
|
|
|
static_cast<size_t> (h / factor));
|
|
|
|
|
|
|
|
const float scale = float (dst.w) / src.w;
|
|
|
|
const float half_width = filter::support / scale;
|
|
|
|
const int half_pixels = int (std::ceil (half_width));
|
|
|
|
|
|
|
|
// Calculate the weighted sum of the src pixels for each dst pixel
|
|
|
|
for (size_t d_y = 0; d_y < dst.h; ++d_y)
|
|
|
|
for (size_t d_x = 0; d_x < dst.w; ++d_x) {
|
|
|
|
// Initialise value and weight sums
|
|
|
|
float v = 0.f;
|
|
|
|
float m = 0.f;
|
|
|
|
|
|
|
|
// Find the centre of the src pixel
|
|
|
|
float o_y = (d_y + 0.5f) / scale;
|
|
|
|
float o_x = (d_x + 0.5f) / scale;
|
|
|
|
|
|
|
|
// Do a full summation across the window. This isn't using
|
|
|
|
// seperable filtering because:
|
|
|
|
// a) we need high precision with fractional factors
|
|
|
|
// b) it's much easier to implement as is
|
|
|
|
//
|
|
|
|
// TODO: seperable filters
|
|
|
|
for (int s_y = -half_pixels; s_y <= half_pixels; ++s_y)
|
|
|
|
for (int s_x = -half_pixels; s_x <= half_pixels; ++s_x) {
|
|
|
|
float m_x = filter::weight (s_x * scale);
|
|
|
|
float m_y = filter::weight (s_y * scale);
|
|
|
|
float m_ = m_x * m_y;
|
|
|
|
|
|
|
|
// Simple clamp to border for edges
|
|
|
|
int x = int (limit (o_x + s_x - 0.5f, 0, src.w));
|
|
|
|
int y = int (limit (o_y + s_y - 0.5f, 0, src.h));
|
|
|
|
|
|
|
|
// Collection the contribution
|
2015-05-18 14:09:34 +10:00
|
|
|
v += m_ * src.m_data[y * src.s + x];
|
2015-02-05 20:34:38 +11:00
|
|
|
m += m_;
|
|
|
|
}
|
|
|
|
|
|
|
|
CHECK_NEZ (m);
|
|
|
|
|
2015-05-18 14:09:34 +10:00
|
|
|
dst.m_data[d_y * dst.s + d_x] = uint8_t (v / m);
|
2015-02-05 20:34:38 +11:00
|
|
|
}
|
2015-02-04 15:59:06 +11:00
|
|
|
|
2015-02-05 20:34:38 +11:00
|
|
|
return dst;
|
|
|
|
}
|
2015-02-04 15:59:06 +11:00
|
|
|
|
|
|
|
|
2015-06-03 23:17:37 +10:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
template <typename T>
|
|
|
|
T
|
|
|
|
buffer<T>::operator[] (point<2,size_t> p) const
|
|
|
|
{
|
2015-06-15 17:47:49 +10:00
|
|
|
CHECK_LT (p.x, w);
|
|
|
|
CHECK_LT (p.y, h);
|
|
|
|
|
2015-07-23 21:18:15 +10:00
|
|
|
return begin ()[offset (p)];
|
2015-06-03 23:17:37 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
T&
|
|
|
|
buffer<T>::operator[] (point<2,size_t> p)
|
|
|
|
{
|
2015-06-15 17:47:49 +10:00
|
|
|
CHECK_LT (p.x, w);
|
|
|
|
CHECK_LT (p.y, h);
|
|
|
|
|
2015-07-23 21:18:15 +10:00
|
|
|
return begin ()[offset (p)];
|
2015-06-03 23:17:37 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-02-05 20:34:38 +11:00
|
|
|
//-----------------------------------------------------------------------------
|
2015-05-18 14:09:34 +10:00
|
|
|
template <typename T>
|
2015-06-03 23:17:37 +10:00
|
|
|
T
|
|
|
|
buffer<T>::operator[] (size_t idx) const
|
|
|
|
{
|
2015-06-15 17:47:49 +10:00
|
|
|
CHECK_LT (idx, h * s);
|
|
|
|
|
2015-06-03 23:17:37 +10:00
|
|
|
return begin ()[idx];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
T&
|
|
|
|
buffer<T>::operator[] (size_t idx)
|
|
|
|
{
|
2015-06-15 17:47:49 +10:00
|
|
|
CHECK_LT (idx, h * s);
|
|
|
|
|
2015-06-03 23:17:37 +10:00
|
|
|
return begin ()[idx];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
template <typename T>
|
2015-07-24 01:36:42 +10:00
|
|
|
inline T*
|
2015-05-18 14:09:34 +10:00
|
|
|
buffer<T>::data (void)
|
|
|
|
{
|
|
|
|
return begin ();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
2015-07-24 01:36:42 +10:00
|
|
|
inline T*
|
2015-05-18 14:09:34 +10:00
|
|
|
buffer<T>::begin (void)
|
|
|
|
{
|
|
|
|
return m_data.get ();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
2015-07-24 01:36:42 +10:00
|
|
|
inline T*
|
2015-05-18 14:09:34 +10:00
|
|
|
buffer<T>::end (void)
|
|
|
|
{
|
|
|
|
return begin () + h * s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
2015-07-24 01:36:42 +10:00
|
|
|
inline const T*
|
2015-05-18 14:09:34 +10:00
|
|
|
buffer<T>::data (void) const
|
|
|
|
{
|
|
|
|
return begin ();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
2015-07-24 01:36:42 +10:00
|
|
|
inline const T*
|
2015-05-18 14:09:34 +10:00
|
|
|
buffer<T>::begin (void) const
|
|
|
|
{
|
|
|
|
return m_data.get ();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
2015-07-24 01:36:42 +10:00
|
|
|
inline const T*
|
2015-05-18 14:09:34 +10:00
|
|
|
buffer<T>::end (void) const
|
|
|
|
{
|
|
|
|
return begin () + h * s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-02-04 15:59:06 +11:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template struct util::image::buffer<uint8_t>;
|
2015-07-23 21:18:54 +10:00
|
|
|
template struct util::image::buffer<uint16_t>;
|
2015-08-18 00:00:35 +10:00
|
|
|
template struct util::image::buffer<uint32_t>;
|
|
|
|
template struct util::image::buffer< int32_t>;
|
2015-05-18 14:11:08 +10:00
|
|
|
template struct util::image::buffer<float>;
|
2015-05-18 22:01:43 +10:00
|
|
|
template struct util::image::buffer<double>;
|
2015-05-18 14:11:08 +10:00
|
|
|
|
2015-02-04 15:59:06 +11:00
|
|
|
template util::image::buffer<uint8_t> util::image::buffer<uint8_t>::alloc (void) const;
|
2015-02-04 18:39:08 +11:00
|
|
|
template util::image::buffer<uint8_t> util::image::buffer<uint8_t>::clone (void) const;
|
2015-05-18 14:11:08 +10:00
|
|
|
template util::image::buffer<uint8_t> util::image::buffer<float>::clone (void) const;
|
2015-05-18 22:01:43 +10:00
|
|
|
template util::image::buffer<uint8_t> util::image::buffer<double>::clone (void) const;
|
2015-06-04 14:34:38 +10:00
|
|
|
|
2015-08-18 00:00:35 +10:00
|
|
|
template util::image::buffer<float> util::image::buffer<float>::alloc (void) const;
|
|
|
|
template util::image::buffer<uint32_t> util::image::buffer<float>::alloc (void) const;
|
|
|
|
template util::image::buffer< int32_t> util::image::buffer<float>::alloc (void) const;
|
|
|
|
|
2015-06-04 14:34:38 +10:00
|
|
|
template util::image::buffer<float> util::image::buffer<float>::clone (void) const;
|
2015-08-18 00:00:35 +10:00
|
|
|
|
|
|
|
template util::image::buffer<uint32_t> util::image::buffer<uint32_t>::alloc (void) const;
|