algorithm.hpp 40 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016
  1. /*
  2. Copyright 2005-2007 Adobe Systems Incorporated
  3. Use, modification and distribution are subject to the Boost Software License,
  4. Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  5. http://www.boost.org/LICENSE_1_0.txt).
  6. See http://opensource.adobe.com/gil for most recent version including documentation.
  7. */
  8. /*************************************************************************************************/
  9. #ifndef GIL_ALGORITHM_HPP
  10. #define GIL_ALGORITHM_HPP
  11. #include <cassert>
  12. #include <cstddef>
  13. #include <cstring>
  14. #include <algorithm>
  15. #include <iterator>
  16. #include <memory>
  17. #include <typeinfo>
  18. #include "gil_config.hpp"
  19. #include "gil_concept.hpp"
  20. #include "color_base_algorithm.hpp"
  21. #include "image_view.hpp"
  22. #include "image_view_factory.hpp"
  23. #include "bit_aligned_pixel_iterator.hpp"
  24. ////////////////////////////////////////////////////////////////////////////////////////
  25. /// \file
  26. /// \brief Some basic STL-style algorithms when applied to image views
  27. /// \author Lubomir Bourdev and Hailin Jin \n
  28. /// Adobe Systems Incorporated
  29. /// \date 2005-2008 \n Last updated on March 12, 2008
  30. ///
  31. ////////////////////////////////////////////////////////////////////////////////////////
  32. //#ifdef _MSC_VER
  33. //#pragma warning(push)
  34. //#pragma warning(disable : 4244) // conversion from 'gil::image<V,Alloc>::coord_t' to 'int', possible loss of data (visual studio compiler doesn't realize that the two types are the same)
  35. //#endif
  36. namespace boost { namespace gil {
  37. //forward declarations
  38. template <typename ChannelPtr, typename ColorSpace>
  39. struct planar_pixel_iterator;
  40. template <typename Iterator>
  41. class memory_based_step_iterator;
  42. template <typename StepIterator>
  43. class memory_based_2d_locator;
  44. // a tag denoting incompatible arguments
  45. struct error_t {};
  46. /// \defgroup ImageViewSTLAlgorithms STL-like Algorithms
  47. /// \ingroup ImageViewAlgorithm
  48. /// \brief Image view-equivalents of STL algorithms
  49. ///
  50. /// Image views provide 1D iteration of their pixels via \p begin() and \p end() methods,
  51. /// which makes it possible to use STL algorithms with them. However, using nested loops
  52. /// over X and Y is in many cases more efficient. The algorithms in this section resemble
  53. /// STL algorithms, but they abstract away the nested loops and take views (as opposed to ranges) as input.
  54. ///
  55. /// Most algorithms check whether the image views are 1D-traversable. A 1D-traversable image view has no gaps
  56. /// at the end of the rows. In other words, if an x_iterator of that view is advanced past the last pixel in a row
  57. /// it will move to the first pixel of the next row. When image views are 1D-traversable, the algorithms use
  58. /// a single loop and run more efficiently. If one or more of the input views are not 1D-traversable, the algorithms
  59. /// fall-back to an X-loop nested inside a Y-loop.
  60. ///
  61. /// The algorithms typically delegate the work to their corresponding STL algorithms. For example, \p copy_pixels calls
  62. /// \p std::copy either for each row, or, when the images are 1D-traversable, once for all pixels.
  63. ///
  64. /// In addition, overloads are sometimes provided for the STL algorithms. For example, std::copy for planar iterators
  65. /// is overloaded to perform \p std::copy for each of the planes. \p std::copy over bitwise-copiable pixels results in
  66. /// std::copy over unsigned char, which STL typically implements via \p memmove.
  67. ///
  68. /// As a result \p copy_pixels may result in a single call to \p memmove for interleaved 1D-traversable views,
  69. /// or one per each plane of planar 1D-traversable views, or one per each row of interleaved non-1D-traversable images, etc.
  70. /// \defgroup STLOptimizations Performance overloads of STL algorithms
  71. /// \ingroup ImageViewAlgorithm
  72. /// \brief overloads of STL algorithms allowing more efficient implementation when used with GIL constructs
  73. /// \brief A generic binary operation on views
  74. /// \ingroup ImageViewSTLAlgorithms
  75. ///
  76. /// Use this class as a convenience superclass when defining an operation for any image views.
  77. /// Many operations have different behavior when the two views are compatible. This class checks
  78. /// for compatibility and invokes apply_compatible(V1,V2) or apply_incompatible(V1,V2) of the subclass.
  79. /// You must provide apply_compatible(V1,V2) method in your subclass, but apply_incompatible(V1,V2)
  80. /// is not required and the default throws std::bad_cast.
  81. template <typename Derived, typename Result=void>
  82. struct binary_operation_obj {
  83. typedef Result result_type;
  84. template <typename V1, typename V2> GIL_FORCEINLINE
  85. result_type operator()(const std::pair<const V1*,const V2*>& p) const {
  86. return apply(*p.first, *p.second, typename views_are_compatible<V1,V2>::type());
  87. }
  88. template <typename V1, typename V2> GIL_FORCEINLINE
  89. result_type operator()(const V1& v1, const V2& v2) const {
  90. return apply(v1, v2, typename views_are_compatible<V1,V2>::type());
  91. }
  92. result_type operator()(const error_t&) const { throw std::bad_cast(); }
  93. private:
  94. // dispatch from apply overload to a function with distinct name
  95. template <typename V1, typename V2>
  96. GIL_FORCEINLINE result_type apply(const V1& v1, const V2& v2, mpl::false_) const {
  97. return ((const Derived*)this)->apply_incompatible(v1,v2);
  98. }
  99. // dispatch from apply overload to a function with distinct name
  100. template <typename V1, typename V2>
  101. GIL_FORCEINLINE result_type apply(const V1& v1, const V2& v2, mpl::true_) const {
  102. return ((const Derived*)this)->apply_compatible(v1,v2);
  103. }
  104. // function with distinct name - it can be overloaded by subclasses
  105. template <typename V1, typename V2>
  106. GIL_FORCEINLINE result_type apply_incompatible(const V1& v1, const V2& v2) const {
  107. throw std::bad_cast();
  108. }
  109. };
  110. } } // namespace boost::gil
  111. //////////////////////////////////////////////////////////////////////////////////////
  112. ///
  113. /// std::copy and gil::copy_pixels
  114. ///
  115. //////////////////////////////////////////////////////////////////////////////////////
  116. /// \defgroup ImageViewSTLAlgorithmsCopyPixels copy_pixels
  117. /// \ingroup ImageViewSTLAlgorithms
  118. /// \brief std::copy for image views
  119. namespace std {
  120. /// \ingroup STLOptimizations
  121. /// \brief Copy when both src and dst are interleaved and of the same type can be just memmove
  122. template<typename T, typename Cs>
  123. GIL_FORCEINLINE boost::gil::pixel<T,Cs>*
  124. copy(boost::gil::pixel<T,Cs>* first, boost::gil::pixel<T,Cs>* last,
  125. boost::gil::pixel<T,Cs>* dst) {
  126. return (boost::gil::pixel<T,Cs>*)std::copy((unsigned char*)first,(unsigned char*)last, (unsigned char*)dst);
  127. }
  128. /// \ingroup STLOptimizations
  129. /// \brief Copy when both src and dst are interleaved and of the same type can be just memmove
  130. template<typename T, typename Cs>
  131. GIL_FORCEINLINE boost::gil::pixel<T,Cs>*
  132. copy(const boost::gil::pixel<T,Cs>* first, const boost::gil::pixel<T,Cs>* last,
  133. boost::gil::pixel<T,Cs>* dst) {
  134. return (boost::gil::pixel<T,Cs>*)std::copy((unsigned char*)first,(unsigned char*)last, (unsigned char*)dst);
  135. }
  136. } // namespace std
  137. namespace boost { namespace gil {
  138. namespace detail {
  139. template <typename I, typename O> struct copy_fn {
  140. GIL_FORCEINLINE I operator()(I first, I last, O dst) const { return std::copy(first,last,dst); }
  141. };
  142. } // namespace detail
  143. } } // namespace boost::gil
  144. namespace std {
  145. /// \ingroup STLOptimizations
  146. /// \brief Copy when both src and dst are planar pointers is copy for each channel
  147. template<typename Cs, typename IC1, typename IC2> GIL_FORCEINLINE
  148. boost::gil::planar_pixel_iterator<IC2,Cs> copy(boost::gil::planar_pixel_iterator<IC1,Cs> first, boost::gil::planar_pixel_iterator<IC1,Cs> last, boost::gil::planar_pixel_iterator<IC2,Cs> dst) {
  149. boost::gil::gil_function_requires<boost::gil::ChannelsCompatibleConcept<typename std::iterator_traits<IC1>::value_type,typename std::iterator_traits<IC2>::value_type> >();
  150. static_for_each(first,last,dst,boost::gil::detail::copy_fn<IC1,IC2>());
  151. return dst+(last-first);
  152. }
  153. } // namespace std
  154. namespace boost { namespace gil {
  155. namespace detail {
  156. /// Does a copy-n. If the inputs contain image iterators, performs a copy at each row using the row iterators
  157. /// \ingroup CopyPixels
  158. template <typename I, typename O>
  159. struct copier_n {
  160. GIL_FORCEINLINE void operator()(I src, typename std::iterator_traits<I>::difference_type n, O dst) const { std::copy(src,src+n, dst); }
  161. };
  162. /// Source range is delimited by image iterators
  163. template <typename IL, typename O> // IL Models ConstPixelLocatorConcept, O Models PixelIteratorConcept
  164. struct copier_n<iterator_from_2d<IL>,O> {
  165. typedef typename std::iterator_traits<iterator_from_2d<IL> >::difference_type diff_t;
  166. GIL_FORCEINLINE void operator()(iterator_from_2d<IL> src, diff_t n, O dst) const {
  167. gil_function_requires<PixelLocatorConcept<IL> >();
  168. gil_function_requires<MutablePixelIteratorConcept<O> >();
  169. while (n>0) {
  170. typedef typename iterator_from_2d<IL>::difference_type diff_t;
  171. diff_t l=src.width()-src.x_pos();
  172. diff_t numToCopy=(n<l ? n:l);
  173. detail::copy_n(src.x(), numToCopy, dst);
  174. dst+=numToCopy;
  175. src+=numToCopy;
  176. n-=numToCopy;
  177. }
  178. }
  179. };
  180. /// Destination range is delimited by image iterators
  181. template <typename I, typename OL> // I Models ConstPixelIteratorConcept, OL Models PixelLocatorConcept
  182. struct copier_n<I,iterator_from_2d<OL> > {
  183. typedef typename std::iterator_traits<I>::difference_type diff_t;
  184. GIL_FORCEINLINE void operator()(I src, diff_t n, iterator_from_2d<OL> dst) const {
  185. gil_function_requires<PixelIteratorConcept<I> >();
  186. gil_function_requires<MutablePixelLocatorConcept<OL> >();
  187. while (n>0) {
  188. diff_t l=dst.width()-dst.x_pos();
  189. diff_t numToCopy=(n<l ? n:l);
  190. detail::copy_n(src, numToCopy, dst.x());
  191. dst+=numToCopy;
  192. src+=numToCopy;
  193. n-=numToCopy;
  194. }
  195. }
  196. };
  197. /// Both source and destination ranges are delimited by image iterators
  198. template <typename IL, typename OL>
  199. struct copier_n<iterator_from_2d<IL>,iterator_from_2d<OL> > {
  200. typedef typename iterator_from_2d<IL>::difference_type diff_t;
  201. GIL_FORCEINLINE void operator()(iterator_from_2d<IL> src, diff_t n, iterator_from_2d<OL> dst) const {
  202. gil_function_requires<PixelLocatorConcept<IL> >();
  203. gil_function_requires<MutablePixelLocatorConcept<OL> >();
  204. if (src.x_pos()!=dst.x_pos() || src.width()!=dst.width()) {
  205. while(n-->0) {
  206. *dst++=*src++;
  207. }
  208. }
  209. while (n>0) {
  210. diff_t l=dst.width()-dst.x_pos();
  211. diff_t numToCopy=(n<l ? n : l);
  212. detail::copy_n(src.x(), numToCopy, dst.x());
  213. dst+=numToCopy;
  214. src+=numToCopy;
  215. n-=numToCopy;
  216. }
  217. }
  218. };
  219. template <typename SrcIterator, typename DstIterator>
  220. GIL_FORCEINLINE DstIterator copy_with_2d_iterators(SrcIterator first, SrcIterator last, DstIterator dst) {
  221. typedef typename SrcIterator::x_iterator src_x_iterator;
  222. typedef typename DstIterator::x_iterator dst_x_iterator;
  223. typename SrcIterator::difference_type n = last - first;
  224. if (first.is_1d_traversable()) {
  225. if (dst.is_1d_traversable())
  226. copier_n<src_x_iterator,dst_x_iterator>()(first.x(),n, dst.x());
  227. else
  228. copier_n<src_x_iterator,DstIterator >()(first.x(),n, dst);
  229. } else {
  230. if (dst.is_1d_traversable())
  231. copier_n<SrcIterator,dst_x_iterator>()(first,n, dst.x());
  232. else
  233. copier_n<SrcIterator,DstIterator>()(first,n,dst);
  234. }
  235. return dst+n;
  236. }
  237. } // namespace detail
  238. } } // namespace boost::gil
  239. namespace std {
  240. /// \ingroup STLOptimizations
  241. /// \brief std::copy(I1,I1,I2) with I1 and I2 being a iterator_from_2d
  242. template <typename IL, typename OL>
  243. GIL_FORCEINLINE boost::gil::iterator_from_2d<OL> copy1(boost::gil::iterator_from_2d<IL> first, boost::gil::iterator_from_2d<IL> last, boost::gil::iterator_from_2d<OL> dst) {
  244. return boost::gil::detail::copy_with_2d_iterators(first,last,dst);
  245. }
  246. } // namespace std
  247. namespace boost { namespace gil {
  248. /// \ingroup ImageViewSTLAlgorithmsCopyPixels
  249. /// \brief std::copy for image views
  250. template <typename View1, typename View2> GIL_FORCEINLINE
  251. void copy_pixels(const View1& src, const View2& dst) {
  252. assert(src.dimensions()==dst.dimensions());
  253. detail::copy_with_2d_iterators(src.begin(),src.end(),dst.begin());
  254. }
  255. //////////////////////////////////////////////////////////////////////////////////////
  256. ///
  257. /// copy_and_convert_pixels
  258. ///
  259. //////////////////////////////////////////////////////////////////////////////////////
  260. /// \defgroup ImageViewSTLAlgorithmsCopyAndConvertPixels copy_and_convert_pixels
  261. /// \ingroup ImageViewSTLAlgorithms
  262. /// \brief copies src view into dst view, color converting if necessary.
  263. ///
  264. /// Versions taking static and runtime views are provided. Versions taking user-defined color convered are provided.
  265. namespace detail {
  266. template <typename CC>
  267. class copy_and_convert_pixels_fn : public binary_operation_obj<copy_and_convert_pixels_fn<CC> > {
  268. private:
  269. CC _cc;
  270. public:
  271. typedef typename binary_operation_obj<copy_and_convert_pixels_fn<CC> >::result_type result_type;
  272. copy_and_convert_pixels_fn() {}
  273. copy_and_convert_pixels_fn(CC cc_in) : _cc(cc_in) {}
  274. // when the two color spaces are incompatible, a color conversion is performed
  275. template <typename V1, typename V2> GIL_FORCEINLINE
  276. result_type apply_incompatible(const V1& src, const V2& dst) const {
  277. copy_pixels(color_converted_view<typename V2::value_type>(src,_cc),dst);
  278. }
  279. // If the two color spaces are compatible, copy_and_convert is just copy
  280. template <typename V1, typename V2> GIL_FORCEINLINE
  281. result_type apply_compatible(const V1& src, const V2& dst) const {
  282. copy_pixels(src,dst);
  283. }
  284. };
  285. } // namespace detail
  286. /// \ingroup ImageViewSTLAlgorithmsCopyAndConvertPixels
  287. template <typename V1, typename V2,typename CC>
  288. GIL_FORCEINLINE
  289. void copy_and_convert_pixels(const V1& src, const V2& dst,CC cc) {
  290. detail::copy_and_convert_pixels_fn<CC> ccp(cc);
  291. ccp(src,dst);
  292. }
  293. struct default_color_converter;
  294. /// \ingroup ImageViewSTLAlgorithmsCopyAndConvertPixels
  295. template <typename View1, typename View2>
  296. GIL_FORCEINLINE
  297. void copy_and_convert_pixels(const View1& src, const View2& dst) {
  298. detail::copy_and_convert_pixels_fn<default_color_converter> ccp;
  299. ccp(src,dst);
  300. }
  301. } } // namespace boost::gil
  302. //////////////////////////////////////////////////////////////////////////////////////
  303. //
  304. // std::fill and gil::fill_pixels
  305. //
  306. //////////////////////////////////////////////////////////////////////////////////////
  307. /// \defgroup ImageViewSTLAlgorithmsFillPixels fill_pixels
  308. /// \ingroup ImageViewSTLAlgorithms
  309. /// \brief std::fill for image views
  310. namespace std {
  311. /// \ingroup STLOptimizations
  312. /// \brief std::fill(I,I,V) with I being a iterator_from_2d
  313. ///
  314. /// Invoked when one calls std::fill(I,I,V) with I being a iterator_from_2d (which is
  315. /// a 1D iterator over the pixels in an image). For contiguous images (i.e. images that have
  316. /// no alignment gap at the end of each row) it is more efficient to use the underlying
  317. /// pixel iterator that does not check for the end of rows. For non-contiguous images fill
  318. /// resolves to fill of each row using the underlying pixel iterator, which is still faster
  319. template <typename IL, typename V>
  320. void fill(boost::gil::iterator_from_2d<IL> first, boost::gil::iterator_from_2d<IL> last, const V& val) {
  321. boost::gil::gil_function_requires<boost::gil::MutablePixelLocatorConcept<IL> >();
  322. if (first.is_1d_traversable()) {
  323. std::fill(first.x(), last.x(), val);
  324. } else {
  325. // fill row by row
  326. std::ptrdiff_t n=last-first;
  327. while (n>0) {
  328. std::ptrdiff_t numToDo=std::min<const std::ptrdiff_t>(n,(std::ptrdiff_t)(first.width()-first.x_pos()));
  329. fill_n(first.x(), numToDo, val);
  330. first+=numToDo;
  331. n-=numToDo;
  332. }
  333. }
  334. }
  335. } // namespace std
  336. namespace boost { namespace gil {
  337. namespace detail {
  338. /// struct to do std::fill
  339. struct std_fill_t {
  340. template <typename It, typename P>
  341. void operator()(It first, It last, const P& p_in) {
  342. std::fill(first,last,p_in);
  343. }
  344. };
  345. /// std::fill for planar iterators
  346. template <typename It, typename P>
  347. GIL_FORCEINLINE
  348. void fill_aux(It first, It last, const P& p, mpl::true_) {
  349. static_for_each(first,last,p,std_fill_t());
  350. }
  351. /// std::fill for interleaved iterators
  352. template <typename It, typename P>
  353. GIL_FORCEINLINE
  354. void fill_aux(It first, It last, const P& p,mpl::false_) {
  355. std::fill(first,last,p);
  356. }
  357. } // namespace detail
  358. /// \ingroup ImageViewSTLAlgorithmsFillPixels
  359. /// \brief std::fill for image views
  360. template <typename View, typename Value> GIL_FORCEINLINE
  361. void fill_pixels(const View& img_view, const Value& val) {
  362. if (img_view.is_1d_traversable())
  363. detail::fill_aux(img_view.begin().x(), img_view.end().x(),
  364. val,is_planar<View>());
  365. else
  366. for (std::ptrdiff_t y=0; y<img_view.height(); ++y)
  367. detail::fill_aux(img_view.row_begin(y),img_view.row_end(y),
  368. val,is_planar<View>());
  369. }
  370. //////////////////////////////////////////////////////////////////////////////////////
  371. ///
  372. /// destruct_pixels
  373. ///
  374. //////////////////////////////////////////////////////////////////////////////////////
  375. /// \defgroup ImageViewSTLAlgorithmsDestructPixels destruct_pixels
  376. /// \ingroup ImageViewSTLAlgorithms
  377. /// \brief invokes the destructor on every pixel of an image view
  378. namespace detail {
  379. template <typename It> GIL_FORCEINLINE
  380. void destruct_range_impl(It first, It last, mpl::true_) {
  381. typedef typename std::iterator_traits<It>::value_type value_t;
  382. if (boost::has_trivial_destructor<value_t>::value)
  383. return;
  384. while (first!=last) {
  385. first->~value_t();
  386. ++first;
  387. }
  388. }
  389. template <typename It> GIL_FORCEINLINE
  390. void destruct_range_impl(It, It, mpl::false_) {}
  391. template <typename It> GIL_FORCEINLINE
  392. void destruct_range(It first, It last) {
  393. destruct_range_impl(first,last,typename is_pointer<It>::type());
  394. }
  395. struct std_destruct_t {
  396. template <typename It> void operator()(It first, It last) const { destruct_range(first,last); }
  397. };
  398. /// destruct for planar iterators
  399. template <typename It>
  400. GIL_FORCEINLINE
  401. void destruct_aux(It first, It last, mpl::true_) {
  402. static_for_each(first,last,std_destruct_t());
  403. }
  404. /// destruct for interleaved iterators
  405. template <typename It>
  406. GIL_FORCEINLINE
  407. void destruct_aux(It first, It last, mpl::false_) {
  408. destruct_range(first,last);
  409. }
  410. } // namespace detail
  411. /// \ingroup ImageViewSTLAlgorithmsDestructPixels
  412. /// \brief Invokes the in-place destructor on every pixel of the view
  413. template <typename View> GIL_FORCEINLINE
  414. void destruct_pixels(const View& img_view) {
  415. if (img_view.is_1d_traversable())
  416. detail::destruct_aux(img_view.begin().x(), img_view.end().x(),
  417. is_planar<View>());
  418. else
  419. for (std::ptrdiff_t y=0; y<img_view.height(); ++y)
  420. detail::destruct_aux(img_view.row_begin(y),img_view.row_end(y),
  421. is_planar<View>());
  422. }
  423. //////////////////////////////////////////////////////////////////////////////////////
  424. ///
  425. /// uninitialized_fill_pixels
  426. ///
  427. //////////////////////////////////////////////////////////////////////////////////////
  428. /// \defgroup ImageViewSTLAlgorithmsUninitializedFillPixels uninitialized_fill_pixels
  429. /// \ingroup ImageViewSTLAlgorithms
  430. /// \brief std::uninitialized_fill for image views
  431. namespace detail {
  432. /// std::uninitialized_fill for planar iterators
  433. /// If an exception is thrown destructs any in-place copy-constructed objects
  434. template <typename It, typename P>
  435. GIL_FORCEINLINE
  436. void uninitialized_fill_aux(It first, It last,
  437. const P& p, mpl::true_) {
  438. int channel=0;
  439. try {
  440. typedef typename std::iterator_traits<It>::value_type pixel_t;
  441. while (channel < num_channels<pixel_t>::value) {
  442. std::uninitialized_fill(dynamic_at_c(first,channel), dynamic_at_c(last,channel),
  443. dynamic_at_c(p,channel));
  444. ++channel;
  445. }
  446. } catch (...) {
  447. for (int c=0; c<channel; ++c)
  448. destruct_range(dynamic_at_c(first,c), dynamic_at_c(last,c));
  449. throw;
  450. }
  451. }
  452. /// std::uninitialized_fill for interleaved iterators
  453. /// If an exception is thrown destructs any in-place copy-constructed objects
  454. template <typename It, typename P>
  455. GIL_FORCEINLINE
  456. void uninitialized_fill_aux(It first, It last,
  457. const P& p,mpl::false_) {
  458. std::uninitialized_fill(first,last,p);
  459. }
  460. } // namespace detail
  461. /// \ingroup ImageViewSTLAlgorithmsUninitializedFillPixels
  462. /// \brief std::uninitialized_fill for image views.
  463. /// Does not support planar heterogeneous views.
  464. /// If an exception is thrown destructs any in-place copy-constructed pixels
  465. template <typename View, typename Value>
  466. void uninitialized_fill_pixels(const View& img_view, const Value& val) {
  467. if (img_view.is_1d_traversable())
  468. detail::uninitialized_fill_aux(img_view.begin().x(), img_view.end().x(),
  469. val,is_planar<View>());
  470. else {
  471. typename View::y_coord_t y;
  472. try {
  473. for (y=0; y<img_view.height(); ++y)
  474. detail::uninitialized_fill_aux(img_view.row_begin(y),img_view.row_end(y),
  475. val,is_planar<View>());
  476. } catch(...) {
  477. for (typename View::y_coord_t y0=0; y0<y; ++y0)
  478. detail::destruct_aux(img_view.row_begin(y0),img_view.row_end(y0), is_planar<View>());
  479. throw;
  480. }
  481. }
  482. }
  483. //////////////////////////////////////////////////////////////////////////////////////
  484. ///
  485. /// default_construct_pixels
  486. ///
  487. //////////////////////////////////////////////////////////////////////////////////////
  488. /// \defgroup ImageViewSTLAlgorithmsDefaultConstructPixels default_construct_pixels
  489. /// \ingroup ImageViewSTLAlgorithms
  490. /// \brief invokes the default constructor on every pixel of an image view
  491. namespace detail {
  492. template <typename It> GIL_FORCEINLINE
  493. void default_construct_range_impl(It first, It last, mpl::true_) {
  494. typedef typename std::iterator_traits<It>::value_type value_t;
  495. It first1=first;
  496. try {
  497. while (first!=last) {
  498. new (first) value_t();
  499. ++first;
  500. }
  501. } catch (...) {
  502. destruct_range(first1,first);
  503. throw;
  504. }
  505. }
  506. template <typename It> GIL_FORCEINLINE
  507. void default_construct_range_impl(It, It, mpl::false_) {}
  508. template <typename It> GIL_FORCEINLINE
  509. void default_construct_range(It first, It last) { default_construct_range_impl(first, last, typename is_pointer<It>::type()); }
  510. /// uninitialized_default_construct for planar iterators
  511. template <typename It>
  512. GIL_FORCEINLINE
  513. void default_construct_aux(It first, It last, mpl::true_) {
  514. int channel=0;
  515. try {
  516. typedef typename std::iterator_traits<It>::value_type pixel_t;
  517. while (channel < num_channels<pixel_t>::value) {
  518. default_construct_range(dynamic_at_c(first,channel), dynamic_at_c(last,channel));
  519. ++channel;
  520. }
  521. } catch (...) {
  522. for (int c=0; c<channel; ++c)
  523. destruct_range(dynamic_at_c(first,c), dynamic_at_c(last,c));
  524. throw;
  525. }
  526. }
  527. /// uninitialized_default_construct for interleaved iterators
  528. template <typename It>
  529. GIL_FORCEINLINE
  530. void default_construct_aux(It first, It last, mpl::false_) {
  531. default_construct_range(first,last);
  532. }
  533. template <typename View, bool IsPlanar>
  534. struct has_trivial_pixel_constructor : public boost::has_trivial_constructor<typename View::value_type> {};
  535. template <typename View>
  536. struct has_trivial_pixel_constructor<View, true> : public boost::has_trivial_constructor<typename channel_type<View>::type> {};
  537. } // namespace detail
  538. /// \ingroup ImageViewSTLAlgorithmsDefaultConstructPixels
  539. /// \brief Invokes the in-place default constructor on every pixel of the (uninitialized) view.
  540. /// Does not support planar heterogeneous views.
  541. /// If an exception is thrown destructs any in-place default-constructed pixels
  542. template <typename View>
  543. void default_construct_pixels(const View& img_view) {
  544. if (detail::has_trivial_pixel_constructor<View, is_planar<View>::value>::value)
  545. return;
  546. if (img_view.is_1d_traversable())
  547. detail::default_construct_aux(img_view.begin().x(), img_view.end().x(), is_planar<View>());
  548. else {
  549. typename View::y_coord_t y;
  550. try {
  551. for (y=0; y<img_view.height(); ++y)
  552. detail::default_construct_aux(img_view.row_begin(y),img_view.row_end(y), is_planar<View>());
  553. } catch(...) {
  554. for (typename View::y_coord_t y0=0; y0<y; ++y0)
  555. detail::destruct_aux(img_view.row_begin(y0),img_view.row_end(y0), is_planar<View>());
  556. throw;
  557. }
  558. }
  559. }
  560. //////////////////////////////////////////////////////////////////////////////////////
  561. ///
  562. /// uninitialized_copy_pixels
  563. ///
  564. //////////////////////////////////////////////////////////////////////////////////////
  565. /// \defgroup ImageViewSTLAlgorithmsUninitializedCopyPixels uninitialized_copy_pixels
  566. /// \ingroup ImageViewSTLAlgorithms
  567. /// \brief std::uninitialized_copy for image views
  568. namespace detail {
  569. /// std::uninitialized_copy for pairs of planar iterators
  570. template <typename It1, typename It2>
  571. GIL_FORCEINLINE
  572. void uninitialized_copy_aux(It1 first1, It1 last1,
  573. It2 first2, mpl::true_) {
  574. int channel=0;
  575. try {
  576. typedef typename std::iterator_traits<It1>::value_type pixel_t;
  577. while (channel < num_channels<pixel_t>::value) {
  578. std::uninitialized_copy(dynamic_at_c(first1,channel), dynamic_at_c(last1,channel), dynamic_at_c(first2,channel));
  579. ++channel;
  580. }
  581. } catch (...) {
  582. It2 last2=first2;
  583. std::advance(last2, std::distance(first1,last1));
  584. for (int c=0; c<channel; ++c)
  585. destruct_range(dynamic_at_c(first2,c), dynamic_at_c(last2,c));
  586. throw;
  587. }
  588. }
  589. /// std::uninitialized_copy for interleaved or mixed iterators
  590. template <typename It1, typename It2>
  591. GIL_FORCEINLINE
  592. void uninitialized_copy_aux(It1 first1, It1 last1,
  593. It2 first2,mpl::false_) {
  594. std::uninitialized_copy(first1,last1,first2);
  595. }
  596. } // namespace detail
  597. /// \ingroup ImageViewSTLAlgorithmsUninitializedCopyPixels
  598. /// \brief std::uninitialized_copy for image views.
  599. /// Does not support planar heterogeneous views.
  600. /// If an exception is thrown destructs any in-place copy-constructed objects
  601. template <typename View1, typename View2>
  602. void uninitialized_copy_pixels(const View1& view1, const View2& view2) {
  603. typedef mpl::bool_<is_planar<View1>::value && is_planar<View2>::value> is_planar;
  604. assert(view1.dimensions()==view2.dimensions());
  605. if (view1.is_1d_traversable() && view2.is_1d_traversable())
  606. detail::uninitialized_copy_aux(view1.begin().x(), view1.end().x(),
  607. view2.begin().x(),
  608. is_planar());
  609. else {
  610. typename View1::y_coord_t y;
  611. try {
  612. for (y=0; y<view1.height(); ++y)
  613. detail::uninitialized_copy_aux(view1.row_begin(y), view1.row_end(y),
  614. view2.row_begin(y),
  615. is_planar());
  616. } catch(...) {
  617. for (typename View1::y_coord_t y0=0; y0<y; ++y0)
  618. detail::destruct_aux(view2.row_begin(y0),view2.row_end(y0), is_planar());
  619. throw;
  620. }
  621. }
  622. }
  623. //////////////////////////////////////////////////////////////////////////////////////
  624. ///
  625. /// for_each_pixel
  626. ///
  627. //////////////////////////////////////////////////////////////////////////////////////
  628. /// \defgroup ImageViewSTLAlgorithmsForEachPixel for_each_pixel
  629. /// \ingroup ImageViewSTLAlgorithms
  630. /// \brief std::for_each for image views
  631. ///
  632. /// For contiguous images (i.e. images that have no alignment gap at the end of each row) it is
  633. /// more efficient to use the underlying pixel iterator that does not check for the end of rows.
  634. /// For non-contiguous images for_each_pixel resolves to for_each of each row using the underlying
  635. /// pixel iterator, which is still faster
  636. /// \ingroup ImageViewSTLAlgorithmsForEachPixel
  637. template <typename V, typename F>
  638. F for_each_pixel(const V& img, F fun) {
  639. if (img.is_1d_traversable()) {
  640. return std::for_each(img.begin().x(), img.end().x(), fun);
  641. } else {
  642. for (std::ptrdiff_t y=0; y<img.height(); ++y)
  643. fun = std::for_each(img.row_begin(y),img.row_end(y),fun);
  644. return fun;
  645. }
  646. }
  647. /// \defgroup ImageViewSTLAlgorithmsForEachPixelPosition for_each_pixel_position
  648. /// \ingroup ImageViewSTLAlgorithms
  649. /// \brief adobe::for_each_position for image views (passes locators, instead of pixel references, to the function object)
  650. /// \ingroup ImageViewSTLAlgorithmsForEachPixelPosition
  651. template <typename View, typename F>
  652. F for_each_pixel_position(const View& img, F fun) {
  653. typename View::xy_locator loc=img.xy_at(0,0);
  654. for (std::ptrdiff_t y=0; y<img.height(); ++y) {
  655. for (std::ptrdiff_t x=0; x<img.width(); ++x, ++loc.x())
  656. fun(loc);
  657. loc.x()-=img.width(); ++loc.y();
  658. }
  659. return fun;
  660. }
  661. //////////////////////////////////////////////////////////////////////////////////////
  662. ///
  663. /// generate_pixels
  664. ///
  665. //////////////////////////////////////////////////////////////////////////////////////
  666. /// \defgroup ImageViewSTLAlgorithmsGeneratePixels generate_pixels
  667. /// \ingroup ImageViewSTLAlgorithms
  668. /// \brief std::generate for image views
  669. /// \ingroup ImageViewSTLAlgorithmsGeneratePixels
  670. /// \brief std::generate for image views
  671. template <typename View, typename F>
  672. void generate_pixels(const View& v, F fun) {
  673. if (v.is_1d_traversable()) {
  674. std::generate(v.begin().x(), v.end().x(), fun);
  675. } else {
  676. for (std::ptrdiff_t y=0; y<v.height(); ++y)
  677. std::generate(v.row_begin(y),v.row_end(y),fun);
  678. }
  679. }
  680. //////////////////////////////////////////////////////////////////////////////////////
  681. ///
  682. /// std::equal and gil::equal_pixels for GIL constructs
  683. ///
  684. //////////////////////////////////////////////////////////////////////////////////////
  685. /// \defgroup ImageViewSTLAlgorithmsEqualPixels equal_pixels
  686. /// \ingroup ImageViewSTLAlgorithms
  687. /// \brief std::equal for image views
  688. template <typename I1, typename I2> GIL_FORCEINLINE bool equal_n(I1 i1, std::ptrdiff_t n, I2 i2);
  689. namespace detail {
  690. template <typename I1, typename I2>
  691. struct equal_n_fn {
  692. GIL_FORCEINLINE bool operator()(I1 i1, std::ptrdiff_t n, I2 i2) const { return std::equal(i1,i1+n, i2); }
  693. };
  694. /// Equal when both ranges are interleaved and of the same type.
  695. /// GIL pixels are bitwise comparable, so memcmp is used. User-defined pixels that are not bitwise comparable need to provide an overload
  696. template<typename T, typename Cs>
  697. struct equal_n_fn<const pixel<T,Cs>*, const pixel<T,Cs>*> {
  698. GIL_FORCEINLINE bool operator()(const pixel<T,Cs>* i1, std::ptrdiff_t n, const pixel<T,Cs>* i2) const {
  699. return memcmp(i1, i2, n*sizeof(pixel<T,Cs>))==0;
  700. }
  701. };
  702. template<typename T, typename Cs>
  703. struct equal_n_fn<pixel<T,Cs>*, pixel<T,Cs>*> : equal_n_fn<const pixel<T,Cs>*, const pixel<T,Cs>*> {};
  704. /// EqualPixels
  705. /// Equal when both ranges are planar pointers of the same type. memcmp is invoked for each channel plane
  706. /// User-defined channels that are not bitwise comparable need to provide an overload
  707. template<typename IC, typename Cs>
  708. struct equal_n_fn<planar_pixel_iterator<IC,Cs>, planar_pixel_iterator<IC,Cs> > {
  709. GIL_FORCEINLINE bool operator()(const planar_pixel_iterator<IC,Cs> i1, std::ptrdiff_t n, const planar_pixel_iterator<IC,Cs> i2) const {
  710. ptrdiff_t numBytes=n*sizeof(typename std::iterator_traits<IC>::value_type);
  711. for (std::ptrdiff_t i=0; i<mpl::size<Cs>::value; ++i)
  712. if (memcmp(dynamic_at_c(i1,i), dynamic_at_c(i2,i), numBytes)!=0)
  713. return false;
  714. return true;
  715. }
  716. };
  717. /// Source range is delimited by image iterators
  718. template <typename Loc, typename I2> // IL Models ConstPixelLocatorConcept, O Models PixelIteratorConcept
  719. struct equal_n_fn<boost::gil::iterator_from_2d<Loc>,I2> {
  720. GIL_FORCEINLINE bool operator()(boost::gil::iterator_from_2d<Loc> i1, std::ptrdiff_t n, I2 i2) const {
  721. gil_function_requires<boost::gil::PixelLocatorConcept<Loc> >();
  722. gil_function_requires<boost::gil::PixelIteratorConcept<I2> >();
  723. while (n>0) {
  724. std::ptrdiff_t num=std::min<const std::ptrdiff_t>(n, i1.width()-i1.x_pos());
  725. if (!equal_n(i1.x(), num, i2))
  726. return false;
  727. i1+=num;
  728. i2+=num;
  729. n-=num;
  730. }
  731. return true;
  732. }
  733. };
  734. /// Destination range is delimited by image iterators
  735. template <typename I1, typename Loc> // I Models PixelIteratorConcept, OL Models PixelLocatorConcept
  736. struct equal_n_fn<I1,boost::gil::iterator_from_2d<Loc> > {
  737. GIL_FORCEINLINE bool operator()(I1 i1, std::ptrdiff_t n, boost::gil::iterator_from_2d<Loc> i2) const {
  738. gil_function_requires<boost::gil::PixelIteratorConcept<I1> >();
  739. gil_function_requires<boost::gil::PixelLocatorConcept<Loc> >();
  740. while (n>0) {
  741. std::ptrdiff_t num=std::min<const std::ptrdiff_t>(n,i2.width()-i2.x_pos());
  742. if (!equal_n(i1, num, i2.x()))
  743. return false;
  744. i1+=num;
  745. i2+=num;
  746. n-=num;
  747. }
  748. return true;
  749. }
  750. };
  751. /// Both source and destination ranges are delimited by image iterators
  752. template <typename Loc1, typename Loc2>
  753. struct equal_n_fn<boost::gil::iterator_from_2d<Loc1>,boost::gil::iterator_from_2d<Loc2> > {
  754. GIL_FORCEINLINE bool operator()(boost::gil::iterator_from_2d<Loc1> i1, std::ptrdiff_t n, boost::gil::iterator_from_2d<Loc2> i2) const {
  755. gil_function_requires<boost::gil::PixelLocatorConcept<Loc1> >();
  756. gil_function_requires<boost::gil::PixelLocatorConcept<Loc2> >();
  757. if (i1.x_pos()!=i2.x_pos() || i1.width()!=i2.width()) {
  758. while(n-->0) {
  759. if (*i1++!=*i2++) return false;
  760. }
  761. }
  762. while (n>0) {
  763. std::ptrdiff_t num=std::min<const std::ptrdiff_t>(n,i2.width()-i2.x_pos());
  764. if (!equal_n(i1.x(), num, i2.x()))
  765. return false;
  766. i1+=num;
  767. i2+=num;
  768. n-=num;
  769. }
  770. return true;
  771. }
  772. };
  773. } // namespace detail
  774. template <typename I1, typename I2> GIL_FORCEINLINE
  775. bool equal_n(I1 i1, std::ptrdiff_t n, I2 i2) {
  776. return detail::equal_n_fn<I1,I2>()(i1,n,i2);
  777. }
  778. } } // namespace boost::gil
  779. namespace std {
  780. /// \ingroup STLOptimizations
  781. /// \brief std::equal(I1,I1,I2) with I1 and I2 being a iterator_from_2d
  782. ///
  783. /// Invoked when one calls std::equal(I1,I1,I2) with I1 and I2 being a iterator_from_2d (which is
  784. /// a 1D iterator over the pixels in an image). Attempts to demote the source and destination
  785. /// iterators to simpler/faster types if the corresponding range is contiguous.
  786. /// For contiguous images (i.e. images that have
  787. /// no alignment gap at the end of each row) it is more efficient to use the underlying
  788. /// pixel iterator that does not check for the end of rows. If the underlying pixel iterator
  789. /// happens to be a fundamental planar/interleaved pointer, the call may further resolve
  790. /// to memcmp. Otherwise it resolves to copying each row using the underlying pixel iterator
  791. template <typename Loc1, typename Loc2> GIL_FORCEINLINE
  792. bool equal(boost::gil::iterator_from_2d<Loc1> first, boost::gil::iterator_from_2d<Loc1> last, boost::gil::iterator_from_2d<Loc2> first2) {
  793. boost::gil::gil_function_requires<boost::gil::PixelLocatorConcept<Loc1> >();
  794. boost::gil::gil_function_requires<boost::gil::PixelLocatorConcept<Loc2> >();
  795. std::ptrdiff_t n=last-first;
  796. if (first.is_1d_traversable()) {
  797. if (first2.is_1d_traversable())
  798. return boost::gil::detail::equal_n_fn<typename Loc1::x_iterator,typename Loc2::x_iterator>()(first.x(),n, first2.x());
  799. else
  800. return boost::gil::detail::equal_n_fn<typename Loc1::x_iterator,boost::gil::iterator_from_2d<Loc2> >()(first.x(),n, first2);
  801. } else {
  802. if (first2.is_1d_traversable())
  803. return boost::gil::detail::equal_n_fn<boost::gil::iterator_from_2d<Loc1>,typename Loc2::x_iterator>()(first,n, first2.x());
  804. else
  805. return boost::gil::detail::equal_n_fn<boost::gil::iterator_from_2d<Loc1>,boost::gil::iterator_from_2d<Loc2> >()(first,n,first2);
  806. }
  807. }
  808. } // namespace std
  809. namespace boost { namespace gil {
  810. /// \ingroup ImageViewSTLAlgorithmsEqualPixels
  811. /// \brief std::equal for image views
  812. template <typename View1, typename View2> GIL_FORCEINLINE
  813. bool equal_pixels(const View1& v1, const View2& v2) {
  814. assert(v1.dimensions()==v2.dimensions());
  815. return std::equal(v1.begin(),v1.end(),v2.begin()); // std::equal has overloads with GIL iterators for optimal performance
  816. }
  817. //////////////////////////////////////////////////////////////////////////////////////
  818. ///
  819. /// transform_pixels
  820. ///
  821. //////////////////////////////////////////////////////////////////////////////////////
  822. /// \defgroup ImageViewSTLAlgorithmsTransformPixels transform_pixels
  823. /// \ingroup ImageViewSTLAlgorithms
  824. /// \brief std::transform for image views
  825. /// \ingroup ImageViewSTLAlgorithmsTransformPixels
  826. /// \brief std::transform for image views
  827. template <typename View1, typename View2, typename F> GIL_FORCEINLINE
  828. F transform_pixels(const View1& src,const View2& dst, F fun) {
  829. assert(src.dimensions()==dst.dimensions());
  830. for (std::ptrdiff_t y=0; y<src.height(); ++y) {
  831. typename View1::x_iterator srcIt=src.row_begin(y);
  832. typename View2::x_iterator dstIt=dst.row_begin(y);
  833. for (std::ptrdiff_t x=0; x<src.width(); ++x)
  834. dstIt[x]=fun(srcIt[x]);
  835. }
  836. return fun;
  837. }
  838. /// \ingroup ImageViewSTLAlgorithmsTransformPixels
  839. /// \brief transform_pixels with two sources
  840. template <typename View1, typename View2, typename View3, typename F> GIL_FORCEINLINE
  841. F transform_pixels(const View1& src1, const View2& src2,const View3& dst, F fun) {
  842. for (std::ptrdiff_t y=0; y<dst.height(); ++y) {
  843. typename View1::x_iterator srcIt1=src1.row_begin(y);
  844. typename View2::x_iterator srcIt2=src2.row_begin(y);
  845. typename View3::x_iterator dstIt=dst.row_begin(y);
  846. for (std::ptrdiff_t x=0; x<dst.width(); ++x)
  847. dstIt[x]=fun(srcIt1[x],srcIt2[x]);
  848. }
  849. return fun;
  850. }
  851. /// \defgroup ImageViewSTLAlgorithmsTransformPixelPositions transform_pixel_positions
  852. /// \ingroup ImageViewSTLAlgorithms
  853. /// \brief adobe::transform_positions for image views (passes locators, instead of pixel references, to the function object)
  854. /// \ingroup ImageViewSTLAlgorithmsTransformPixelPositions
  855. /// \brief Like transform_pixels but passes to the function object pixel locators instead of pixel references
  856. template <typename View1, typename View2, typename F> GIL_FORCEINLINE
  857. F transform_pixel_positions(const View1& src,const View2& dst, F fun) {
  858. assert(src.dimensions()==dst.dimensions());
  859. typename View1::xy_locator loc=src.xy_at(0,0);
  860. for (std::ptrdiff_t y=0; y<src.height(); ++y) {
  861. typename View2::x_iterator dstIt=dst.row_begin(y);
  862. for (std::ptrdiff_t x=0; x<src.width(); ++x, ++loc.x())
  863. dstIt[x]=fun(loc);
  864. loc.x()-=src.width(); ++loc.y();
  865. }
  866. return fun;
  867. }
  868. /// \ingroup ImageViewSTLAlgorithmsTransformPixelPositions
  869. /// \brief transform_pixel_positions with two sources
  870. template <typename View1, typename View2, typename View3, typename F> GIL_FORCEINLINE
  871. F transform_pixel_positions(const View1& src1,const View2& src2,const View3& dst, F fun) {
  872. assert(src1.dimensions()==dst.dimensions());
  873. assert(src2.dimensions()==dst.dimensions());
  874. typename View1::xy_locator loc1=src1.xy_at(0,0);
  875. typename View2::xy_locator loc2=src2.xy_at(0,0);
  876. for (std::ptrdiff_t y=0; y<src1.height(); ++y) {
  877. typename View3::x_iterator dstIt=dst.row_begin(y);
  878. for (std::ptrdiff_t x=0; x<src1.width(); ++x, ++loc1.x(), ++loc2.x())
  879. dstIt[x]=fun(loc1,loc2);
  880. loc1.x()-=src1.width(); ++loc1.y();
  881. loc2.x()-=src2.width(); ++loc2.y();
  882. }
  883. return fun;
  884. }
  885. } } // namespace boost::gil
  886. //#ifdef _MSC_VER
  887. //#pragma warning(pop)
  888. //#endif
  889. #endif