2021-10-05 21:37:46 +02:00

122 lines
4.0 KiB
C++

//
// Copyright 2019-2020 Mateusz Loskot <mateusz at loskot dot net>
//
// 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
//
#include <boost/gil.hpp>
#include <boost/gil/extension/numeric/convolve.hpp>
#include <boost/core/lightweight_test.hpp>
#include <tuple>
#include <type_traits>
#include "test_fixture.hpp"
#include "core/test_fixture.hpp"
#include "core/image/test_fixture.hpp"
namespace gil = boost::gil;
namespace fixture = boost::gil::test::fixture;
struct test_image_1x1_kernel_1x1_identity
{
template <typename Image>
void operator()(Image const&)
{
using image_t = Image;
auto const img = fixture::create_image<image_t>(1, 1, 7);
image_t img_out(img);
using pixel_t = typename image_t::value_type;
using channel_t = typename gil::channel_type<pixel_t>::type;
auto const kernel = fixture::create_kernel<channel_t>({1});
gil::detail::convolve_1d<pixel_t>(gil::const_view(img_out), kernel, gil::view(img_out));
// 1x1 kernel reduces convolution to multiplication
BOOST_TEST(gil::const_view(img).front() == gil::const_view(img_out).front());
}
static void run()
{
boost::mp11::mp_for_each<fixture::image_types>(test_image_1x1_kernel_1x1_identity{});
}
};
struct test_image_1x1_kernel_3x3_identity
{
template <typename Image>
void operator()(Image const&)
{
using image_t = Image;
auto const img = fixture::create_image<Image>(1, 1, 7);
image_t img_out(img);
using pixel_t = typename image_t::value_type;
using channel_t = typename gil::channel_type<pixel_t>::type;
auto const kernel = fixture::create_kernel<channel_t>({0, 0, 0, 0, 1, 0, 0, 0, 0});
gil::detail::convolve_1d<pixel_t>(gil::const_view(img_out), kernel, gil::view(img_out));
BOOST_TEST(gil::const_view(img).front() == gil::const_view(img_out).front());
}
static void run()
{
boost::mp11::mp_for_each<fixture::image_types>(test_image_1x1_kernel_3x3_identity{});
}
};
struct test_image_3x3_kernel_3x3_identity
{
template <typename Image>
void operator()(Image const&)
{
using image_t = Image;
using pixel_t = typename image_t::value_type;
using channel_t = typename gil::channel_type<pixel_t>::type;
auto const img = fixture::generate_image<image_t>(3, 3, fixture::random_value<channel_t>{});
image_t img_out(img);
auto const kernel = fixture::create_kernel<channel_t>({0, 0, 0, 0, 1, 0, 0, 0, 0});
gil::detail::convolve_1d<pixel_t>(gil::const_view(img_out), kernel, gil::view(img_out));
BOOST_TEST(gil::equal_pixels(gil::const_view(img), gil::const_view(img_out)));
}
static void run()
{
boost::mp11::mp_for_each<fixture::image_types>(test_image_3x3_kernel_3x3_identity{});
}
};
struct test_image_5x5_kernel_3x3_identity
{
template <typename Image>
void operator()(Image const&)
{
using image_t = Image;
using pixel_t = typename image_t::value_type;
using channel_t = typename gil::channel_type<pixel_t>::type;
auto const img = fixture::generate_image<image_t>(5, 5, fixture::random_value<channel_t>{});
image_t img_out(img);
auto const kernel = fixture::create_kernel<channel_t>({0, 0, 0, 0, 1, 0, 0, 0, 0});
gil::detail::convolve_1d<pixel_t>(gil::const_view(img_out), kernel, gil::view(img_out));
// TODO: Test different boundary options
BOOST_TEST(gil::equal_pixels(gil::const_view(img), gil::const_view(img_out)));
}
static void run()
{
boost::mp11::mp_for_each<fixture::image_types>(test_image_5x5_kernel_3x3_identity{});
}
};
int main()
{
test_image_1x1_kernel_1x1_identity::run();
test_image_1x1_kernel_3x3_identity::run();
test_image_3x3_kernel_3x3_identity::run();
test_image_5x5_kernel_3x3_identity::run();
return ::boost::report_errors();
}