kernel.hpp 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. //
  2. // Copyright 2005-2007 Adobe Systems Incorporated
  3. // Copyright 2019 Miral Shah <miralshah2211@gmail.com>
  4. // Copyright 2022 Pranam Lashkari <plashkari628@gmail.com>
  5. //
  6. // Distributed under the Boost Software License, Version 1.0
  7. // See accompanying file LICENSE_1_0.txt or copy at
  8. // http://www.boost.org/LICENSE_1_0.txt
  9. //
  10. #ifndef BOOST_GIL_IMAGE_PROCESSING_KERNEL_HPP
  11. #define BOOST_GIL_IMAGE_PROCESSING_KERNEL_HPP
  12. #include <boost/gil/utilities.hpp>
  13. #include <boost/gil/point.hpp>
  14. #include <boost/assert.hpp>
  15. #include <algorithm>
  16. #include <array>
  17. #include <cstddef>
  18. #include <memory>
  19. #include <vector>
  20. #include <cmath>
  21. #include <stdexcept>
  22. namespace boost { namespace gil {
  23. // Definitions of 1D fixed-size and variable-size kernels and related operations
  24. namespace detail {
  25. /// \brief kernel adaptor for one-dimensional cores
  26. /// Core needs to provide size(),begin(),end(),operator[],
  27. /// value_type,iterator,const_iterator,reference,const_reference
  28. template <typename Core>
  29. class kernel_1d_adaptor : public Core
  30. {
  31. public:
  32. kernel_1d_adaptor() = default;
  33. explicit kernel_1d_adaptor(std::size_t center)
  34. : center_(center)
  35. {
  36. BOOST_ASSERT(center_ < this->size());
  37. }
  38. kernel_1d_adaptor(std::size_t size, std::size_t center)
  39. : Core(size) , center_(center)
  40. {
  41. BOOST_ASSERT(this->size() > 0);
  42. BOOST_ASSERT(center_ < this->size()); // also implies `size() > 0`
  43. }
  44. kernel_1d_adaptor(kernel_1d_adaptor const& other)
  45. : Core(other), center_(other.center_)
  46. {
  47. BOOST_ASSERT(this->size() > 0);
  48. BOOST_ASSERT(center_ < this->size()); // also implies `size() > 0`
  49. }
  50. kernel_1d_adaptor& operator=(kernel_1d_adaptor const& other)
  51. {
  52. Core::operator=(other);
  53. center_ = other.center_;
  54. return *this;
  55. }
  56. std::size_t left_size() const
  57. {
  58. BOOST_ASSERT(center_ < this->size());
  59. return center_;
  60. }
  61. std::size_t right_size() const
  62. {
  63. BOOST_ASSERT(center_ < this->size());
  64. return this->size() - center_ - 1;
  65. }
  66. auto center() -> std::size_t&
  67. {
  68. BOOST_ASSERT(center_ < this->size());
  69. return center_;
  70. }
  71. auto center() const -> std::size_t const&
  72. {
  73. BOOST_ASSERT(center_ < this->size());
  74. return center_;
  75. }
  76. private:
  77. std::size_t center_{0};
  78. };
  79. } // namespace detail
  80. /// \brief variable-size kernel
  81. template <typename T, typename Allocator = std::allocator<T> >
  82. class kernel_1d : public detail::kernel_1d_adaptor<std::vector<T, Allocator>>
  83. {
  84. using parent_t = detail::kernel_1d_adaptor<std::vector<T, Allocator>>;
  85. public:
  86. kernel_1d() = default;
  87. kernel_1d(std::size_t size, std::size_t center) : parent_t(size, center) {}
  88. template <typename FwdIterator>
  89. kernel_1d(FwdIterator elements, std::size_t size, std::size_t center)
  90. : parent_t(size, center)
  91. {
  92. detail::copy_n(elements, size, this->begin());
  93. }
  94. kernel_1d(kernel_1d const& other) : parent_t(other) {}
  95. kernel_1d& operator=(kernel_1d const& other) = default;
  96. };
  97. /// \brief static-size kernel
  98. template <typename T,std::size_t Size>
  99. class kernel_1d_fixed : public detail::kernel_1d_adaptor<std::array<T, Size>>
  100. {
  101. using parent_t = detail::kernel_1d_adaptor<std::array<T, Size>>;
  102. public:
  103. static constexpr std::size_t static_size = Size;
  104. static_assert(static_size > 0, "kernel must have size greater than 0");
  105. static_assert(static_size % 2 == 1, "kernel size must be odd to ensure validity at the center");
  106. kernel_1d_fixed() = default;
  107. explicit kernel_1d_fixed(std::size_t center) : parent_t(center) {}
  108. template <typename FwdIterator>
  109. explicit kernel_1d_fixed(FwdIterator elements, std::size_t center)
  110. : parent_t(center)
  111. {
  112. detail::copy_n(elements, Size, this->begin());
  113. }
  114. kernel_1d_fixed(kernel_1d_fixed const& other) : parent_t(other) {}
  115. kernel_1d_fixed& operator=(kernel_1d_fixed const& other) = default;
  116. };
  117. // TODO: This data member is odr-used and definition at namespace scope
  118. // is required by C++11. Redundant and deprecated in C++17.
  119. template <typename T,std::size_t Size>
  120. constexpr std::size_t kernel_1d_fixed<T, Size>::static_size;
  121. /// \brief reverse a kernel
  122. template <typename Kernel>
  123. inline Kernel reverse_kernel(Kernel const& kernel)
  124. {
  125. Kernel result(kernel);
  126. result.center() = kernel.right_size();
  127. std::reverse(result.begin(), result.end());
  128. return result;
  129. }
  130. namespace detail {
  131. template <typename Core>
  132. class kernel_2d_adaptor : public Core
  133. {
  134. public:
  135. kernel_2d_adaptor() = default;
  136. explicit kernel_2d_adaptor(std::size_t center_y, std::size_t center_x)
  137. : center_(center_x, center_y)
  138. {
  139. BOOST_ASSERT(center_.y < this->size() && center_.x < this->size());
  140. }
  141. kernel_2d_adaptor(std::size_t size, std::size_t center_y, std::size_t center_x)
  142. : Core(size * size), square_size(size), center_(center_x, center_y)
  143. {
  144. BOOST_ASSERT(this->size() > 0);
  145. BOOST_ASSERT(center_.y < this->size() && center_.x < this->size()); // implies `size() > 0`
  146. }
  147. kernel_2d_adaptor(kernel_2d_adaptor const& other)
  148. : Core(other), square_size(other.square_size), center_(other.center_.x, other.center_.y)
  149. {
  150. BOOST_ASSERT(this->size() > 0);
  151. BOOST_ASSERT(center_.y < this->size() && center_.x < this->size()); // implies `size() > 0`
  152. }
  153. kernel_2d_adaptor& operator=(kernel_2d_adaptor const& other)
  154. {
  155. Core::operator=(other);
  156. center_.y = other.center_.y;
  157. center_.x = other.center_.x;
  158. square_size = other.square_size;
  159. return *this;
  160. }
  161. std::size_t upper_size() const
  162. {
  163. BOOST_ASSERT(center_.y < this->size());
  164. return center_.y;
  165. }
  166. std::size_t lower_size() const
  167. {
  168. BOOST_ASSERT(center_.y < this->size());
  169. return this->size() - center_.y - 1;
  170. }
  171. std::size_t left_size() const
  172. {
  173. BOOST_ASSERT(center_.x < this->size());
  174. return center_.x;
  175. }
  176. std::size_t right_size() const
  177. {
  178. BOOST_ASSERT(center_.x < this->size());
  179. return this->size() - center_.x - 1;
  180. }
  181. auto center_y() -> std::size_t&
  182. {
  183. BOOST_ASSERT(center_.y < this->size());
  184. return center_.y;
  185. }
  186. auto center_y() const -> std::size_t const&
  187. {
  188. BOOST_ASSERT(center_.y < this->size());
  189. return center_.y;
  190. }
  191. auto center_x() -> std::size_t&
  192. {
  193. BOOST_ASSERT(center_.x < this->size());
  194. return center_.x;
  195. }
  196. auto center_x() const -> std::size_t const&
  197. {
  198. BOOST_ASSERT(center_.x < this->size());
  199. return center_.x;
  200. }
  201. std::size_t size() const
  202. {
  203. return square_size;
  204. }
  205. typename Core::value_type at(std::size_t x, std::size_t y) const
  206. {
  207. if (x >= this->size() || y >= this->size())
  208. {
  209. throw std::out_of_range("Index out of range");
  210. }
  211. return this->begin()[y * this->size() + x];
  212. }
  213. protected:
  214. std::size_t square_size{0};
  215. private:
  216. point<std::size_t> center_{0, 0};
  217. };
  218. /// \brief variable-size kernel
  219. template
  220. <
  221. typename T,
  222. typename Allocator = std::allocator<T>
  223. >
  224. class kernel_2d : public detail::kernel_2d_adaptor<std::vector<T, Allocator>>
  225. {
  226. using parent_t = detail::kernel_2d_adaptor<std::vector<T, Allocator>>;
  227. public:
  228. kernel_2d() = default;
  229. kernel_2d(std::size_t size,std::size_t center_y, std::size_t center_x)
  230. : parent_t(size, center_y, center_x)
  231. {}
  232. template <typename FwdIterator>
  233. kernel_2d(FwdIterator elements, std::size_t size, std::size_t center_y, std::size_t center_x)
  234. : parent_t(static_cast<int>(std::sqrt(size)), center_y, center_x)
  235. {
  236. detail::copy_n(elements, size, this->begin());
  237. }
  238. kernel_2d(kernel_2d const& other) : parent_t(other) {}
  239. kernel_2d& operator=(kernel_2d const& other) = default;
  240. };
  241. /// \brief static-size kernel
  242. template <typename T, std::size_t Size>
  243. class kernel_2d_fixed :
  244. public detail::kernel_2d_adaptor<std::array<T, Size * Size>>
  245. {
  246. using parent_t = detail::kernel_2d_adaptor<std::array<T, Size * Size>>;
  247. public:
  248. static constexpr std::size_t static_size = Size;
  249. static_assert(static_size > 0, "kernel must have size greater than 0");
  250. static_assert(static_size % 2 == 1, "kernel size must be odd to ensure validity at the center");
  251. kernel_2d_fixed()
  252. {
  253. this->square_size = Size;
  254. }
  255. explicit kernel_2d_fixed(std::size_t center_y, std::size_t center_x) :
  256. parent_t(center_y, center_x)
  257. {
  258. this->square_size = Size;
  259. }
  260. template <typename FwdIterator>
  261. explicit kernel_2d_fixed(FwdIterator elements, std::size_t center_y, std::size_t center_x)
  262. : parent_t(center_y, center_x)
  263. {
  264. this->square_size = Size;
  265. detail::copy_n(elements, Size * Size, this->begin());
  266. }
  267. kernel_2d_fixed(kernel_2d_fixed const& other) : parent_t(other) {}
  268. kernel_2d_fixed& operator=(kernel_2d_fixed const& other) = default;
  269. };
  270. // TODO: This data member is odr-used and definition at namespace scope
  271. // is required by C++11. Redundant and deprecated in C++17.
  272. template <typename T, std::size_t Size>
  273. constexpr std::size_t kernel_2d_fixed<T, Size>::static_size;
  274. template <typename Kernel>
  275. inline Kernel reverse_kernel_2d(Kernel const& kernel)
  276. {
  277. Kernel result(kernel);
  278. result.center_x() = kernel.lower_size();
  279. result.center_y() = kernel.right_size();
  280. std::reverse(result.begin(), result.end());
  281. return result;
  282. }
  283. /// \brief reverse a kernel_2d
  284. template<typename T, typename Allocator>
  285. inline kernel_2d<T, Allocator> reverse_kernel(kernel_2d<T, Allocator> const& kernel)
  286. {
  287. return reverse_kernel_2d(kernel);
  288. }
  289. /// \brief reverse a kernel_2d
  290. template<typename T, std::size_t Size>
  291. inline kernel_2d_fixed<T, Size> reverse_kernel(kernel_2d_fixed<T, Size> const& kernel)
  292. {
  293. return reverse_kernel_2d(kernel);
  294. }
  295. } //namespace detail
  296. }} // namespace boost::gil
  297. #endif