You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

129 lines
4.8 KiB

  1. /*
  2. Copyright 2005-2014 Intel Corporation. All Rights Reserved.
  3. This file is part of Threading Building Blocks.
  4. Threading Building Blocks is free software; you can redistribute it
  5. and/or modify it under the terms of the GNU General Public License
  6. version 2 as published by the Free Software Foundation.
  7. Threading Building Blocks is distributed in the hope that it will be
  8. useful, but WITHOUT ANY WARRANTY; without even the implied warranty
  9. of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with Threading Building Blocks; if not, write to the Free Software
  13. Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  14. As a special exception, you may use this file as part of a free software
  15. library without restriction. Specifically, if other files instantiate
  16. templates or use macros or inline functions from this file, or you compile
  17. this file and link it with other files to produce an executable, this
  18. file does not by itself cause the resulting executable to be covered by
  19. the GNU General Public License. This exception does not however
  20. invalidate any other reasons why the executable file might be covered by
  21. the GNU General Public License.
  22. */
  23. #ifndef __TBB_blocked_range_H
  24. #define __TBB_blocked_range_H
  25. #include "tbb_stddef.h"
  26. namespace tbb {
  27. /** \page range_req Requirements on range concept
  28. Class \c R implementing the concept of range must define:
  29. - \code R::R( const R& ); \endcode Copy constructor
  30. - \code R::~R(); \endcode Destructor
  31. - \code bool R::is_divisible() const; \endcode True if range can be partitioned into two subranges
  32. - \code bool R::empty() const; \endcode True if range is empty
  33. - \code R::R( R& r, split ); \endcode Split range \c r into two subranges.
  34. **/
  35. //! A range over which to iterate.
  36. /** @ingroup algorithms */
  37. template<typename Value>
  38. class blocked_range {
  39. public:
  40. //! Type of a value
  41. /** Called a const_iterator for sake of algorithms that need to treat a blocked_range
  42. as an STL container. */
  43. typedef Value const_iterator;
  44. //! Type for size of a range
  45. typedef std::size_t size_type;
  46. //! Construct range with default-constructed values for begin and end.
  47. /** Requires that Value have a default constructor. */
  48. blocked_range() : my_end(), my_begin() {}
  49. //! Construct range over half-open interval [begin,end), with the given grainsize.
  50. blocked_range( Value begin_, Value end_, size_type grainsize_=1 ) :
  51. my_end(end_), my_begin(begin_), my_grainsize(grainsize_)
  52. {
  53. __TBB_ASSERT( my_grainsize>0, "grainsize must be positive" );
  54. }
  55. //! Beginning of range.
  56. const_iterator begin() const {return my_begin;}
  57. //! One past last value in range.
  58. const_iterator end() const {return my_end;}
  59. //! Size of the range
  60. /** Unspecified if end()<begin(). */
  61. size_type size() const {
  62. __TBB_ASSERT( !(end()<begin()), "size() unspecified if end()<begin()" );
  63. return size_type(my_end-my_begin);
  64. }
  65. //! The grain size for this range.
  66. size_type grainsize() const {return my_grainsize;}
  67. //------------------------------------------------------------------------
  68. // Methods that implement Range concept
  69. //------------------------------------------------------------------------
  70. //! True if range is empty.
  71. bool empty() const {return !(my_begin<my_end);}
  72. //! True if range is divisible.
  73. /** Unspecified if end()<begin(). */
  74. bool is_divisible() const {return my_grainsize<size();}
  75. //! Split range.
  76. /** The new Range *this has the second half, the old range r has the first half.
  77. Unspecified if end()<begin() or !is_divisible(). */
  78. blocked_range( blocked_range& r, split ) :
  79. my_end(r.my_end),
  80. my_begin(do_split(r)),
  81. my_grainsize(r.my_grainsize)
  82. {}
  83. private:
  84. /** NOTE: my_end MUST be declared before my_begin, otherwise the forking constructor will break. */
  85. Value my_end;
  86. Value my_begin;
  87. size_type my_grainsize;
  88. //! Auxiliary function used by forking constructor.
  89. /** Using this function lets us not require that Value support assignment or default construction. */
  90. static Value do_split( blocked_range& r ) {
  91. __TBB_ASSERT( r.is_divisible(), "cannot split blocked_range that is not divisible" );
  92. Value middle = r.my_begin + (r.my_end-r.my_begin)/2u;
  93. r.my_end = middle;
  94. return middle;
  95. }
  96. template<typename RowValue, typename ColValue>
  97. friend class blocked_range2d;
  98. template<typename RowValue, typename ColValue, typename PageValue>
  99. friend class blocked_range3d;
  100. };
  101. } // namespace tbb
  102. #endif /* __TBB_blocked_range_H */