| /****************************************************************************** |
| * Copyright (c) 2011, Duane Merrill. All rights reserved. |
| * Copyright (c) 2011-2018, NVIDIA CORPORATION. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are met: |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * * Neither the name of the NVIDIA CORPORATION nor the |
| * names of its contributors may be used to endorse or promote products |
| * derived from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND |
| * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL NVIDIA CORPORATION BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
| * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| * |
| ******************************************************************************/ |
| |
| /** |
| * \file |
| * The cub::BlockDiscontinuity class provides [<em>collective</em>](index.html#sec0) methods for flagging discontinuities within an ordered set of items partitioned across a CUDA thread block. |
| */ |
| |
| #pragma once |
| |
| #include "../config.cuh" |
| #include "../util_type.cuh" |
| #include "../util_ptx.cuh" |
| |
| /// Optional outer namespace(s) |
| CUB_NS_PREFIX |
| |
| /// CUB namespace |
| namespace cub { |
| |
| /** |
| * \brief The BlockDiscontinuity class provides [<em>collective</em>](index.html#sec0) methods for flagging discontinuities within an ordered set of items partitioned across a CUDA thread block.  |
| * \ingroup BlockModule |
| * |
| * \tparam T The data type to be flagged. |
| * \tparam BLOCK_DIM_X The thread block length in threads along the X dimension |
| * \tparam BLOCK_DIM_Y <b>[optional]</b> The thread block length in threads along the Y dimension (default: 1) |
| * \tparam BLOCK_DIM_Z <b>[optional]</b> The thread block length in threads along the Z dimension (default: 1) |
| * \tparam PTX_ARCH <b>[optional]</b> \ptxversion |
| * |
| * \par Overview |
| * - A set of "head flags" (or "tail flags") is often used to indicate corresponding items |
| * that differ from their predecessors (or successors). For example, head flags are convenient |
| * for demarcating disjoint data segments as part of a segmented scan or reduction. |
| * - \blocked |
| * |
| * \par Performance Considerations |
| * - \granularity |
| * |
| * \par A Simple Example |
| * \blockcollective{BlockDiscontinuity} |
| * \par |
| * The code snippet below illustrates the head flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Collectively compute head flags for discontinuities in the segment |
| * int head_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagHeads(head_flags, thread_data, cub::Inequality()); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], [3,4,4,4], ... }</tt>. |
| * The corresponding output \p head_flags in those threads will be |
| * <tt>{ [1,0,1,0], [0,0,0,0], [1,1,0,0], [0,1,0,0], ... }</tt>. |
| * |
| * \par Performance Considerations |
| * - Incurs zero bank conflicts for most types |
| * |
| */ |
| template < |
| typename T, |
| int BLOCK_DIM_X, |
| int BLOCK_DIM_Y = 1, |
| int BLOCK_DIM_Z = 1, |
| int PTX_ARCH = CUB_PTX_ARCH> |
| class BlockDiscontinuity |
| { |
| private: |
|
|
| /****************************************************************************** |
| * Constants and type definitions |
| ******************************************************************************/ |
| |
| /// Constants |
| enum |
| { |
| /// The thread block size in threads |
| BLOCK_THREADS = BLOCK_DIM_X * BLOCK_DIM_Y * BLOCK_DIM_Z, |
| }; |
| |
|
|
| /// Shared memory storage layout type (last element from each thread's input) |
| struct _TempStorage |
| { |
| T first_items[BLOCK_THREADS]; |
| T last_items[BLOCK_THREADS]; |
| }; |
| |
|
|
| /****************************************************************************** |
| * Utility methods |
| ******************************************************************************/ |
| |
| /// Internal storage allocator |
| __device__ __forceinline__ _TempStorage& PrivateStorage() |
| { |
| __shared__ _TempStorage private_storage; |
| return private_storage; |
| } |
| |
|
|
| /// Specialization for when FlagOp has third index param |
| template <typename FlagOp, bool HAS_PARAM = BinaryOpHasIdxParam<T, FlagOp>::HAS_PARAM> |
| struct ApplyOp |
| { |
| // Apply flag operator |
| static __device__ __forceinline__ bool FlagT(FlagOp flag_op, const T &a, const T &b, int idx) |
| { |
| return flag_op(a, b, idx); |
| } |
| }; |
| |
| /// Specialization for when FlagOp does not have a third index param |
| template <typename FlagOp> |
| struct ApplyOp<FlagOp, false> |
| { |
| // Apply flag operator |
| static __device__ __forceinline__ bool FlagT(FlagOp flag_op, const T &a, const T &b, int /*idx*/) |
| { |
| return flag_op(a, b); |
| } |
| }; |
| |
| /// Templated unrolling of item comparison (inductive case) |
| template <int ITERATION, int MAX_ITERATIONS> |
| struct Iterate |
| { |
| // Head flags |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| static __device__ __forceinline__ void FlagHeads( |
| int linear_tid, |
| FlagT (&flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| T (&preds)[ITEMS_PER_THREAD], ///< [out] Calling thread's predecessor items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| preds[ITERATION] = input[ITERATION - 1]; |
| |
| flags[ITERATION] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| preds[ITERATION], |
| input[ITERATION], |
| (linear_tid * ITEMS_PER_THREAD) + ITERATION); |
| |
| Iterate<ITERATION + 1, MAX_ITERATIONS>::FlagHeads(linear_tid, flags, input, preds, flag_op); |
| } |
| |
| // Tail flags |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| static __device__ __forceinline__ void FlagTails( |
| int linear_tid, |
| FlagT (&flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| flags[ITERATION] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| input[ITERATION], |
| input[ITERATION + 1], |
| (linear_tid * ITEMS_PER_THREAD) + ITERATION + 1); |
| |
| Iterate<ITERATION + 1, MAX_ITERATIONS>::FlagTails(linear_tid, flags, input, flag_op); |
| } |
| |
| }; |
| |
| /// Templated unrolling of item comparison (termination case) |
| template <int MAX_ITERATIONS> |
| struct Iterate<MAX_ITERATIONS, MAX_ITERATIONS> |
| { |
| // Head flags |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| static __device__ __forceinline__ void FlagHeads( |
| int /*linear_tid*/, |
| FlagT (&/*flags*/)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&/*input*/)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| T (&/*preds*/)[ITEMS_PER_THREAD], ///< [out] Calling thread's predecessor items |
| FlagOp /*flag_op*/) ///< [in] Binary boolean flag predicate |
| {} |
| |
| // Tail flags |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| static __device__ __forceinline__ void FlagTails( |
| int /*linear_tid*/, |
| FlagT (&/*flags*/)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&/*input*/)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp /*flag_op*/) ///< [in] Binary boolean flag predicate |
| {} |
| }; |
| |
|
|
| /****************************************************************************** |
| * Thread fields |
| ******************************************************************************/ |
| |
| /// Shared storage reference |
| _TempStorage &temp_storage; |
| |
| /// Linear thread-id |
| unsigned int linear_tid; |
| |
|
|
| public: |
|
|
| /// \smemstorage{BlockDiscontinuity} |
| struct TempStorage : Uninitialized<_TempStorage> {}; |
| |
|
|
| /******************************************************************//** |
| * \name Collective constructors |
| *********************************************************************/ |
| //@{ |
| |
| /** |
| * \brief Collective constructor using a private static allocation of shared memory as temporary storage. |
| */ |
| __device__ __forceinline__ BlockDiscontinuity() |
| : |
| temp_storage(PrivateStorage()), |
| linear_tid(RowMajorTid(BLOCK_DIM_X, BLOCK_DIM_Y, BLOCK_DIM_Z)) |
| {} |
| |
|
|
| /** |
| * \brief Collective constructor using the specified memory allocation as temporary storage. |
| */ |
| __device__ __forceinline__ BlockDiscontinuity( |
| TempStorage &temp_storage) ///< [in] Reference to memory allocation having layout type TempStorage |
| : |
| temp_storage(temp_storage.Alias()), |
| linear_tid(RowMajorTid(BLOCK_DIM_X, BLOCK_DIM_Y, BLOCK_DIM_Z)) |
| {} |
| |
|
|
| //@} end member group |
| /******************************************************************//** |
| * \name Head flag operations |
| *********************************************************************/ |
| //@{ |
| |
|
|
| #ifndef DOXYGEN_SHOULD_SKIP_THIS // Do not document |
| |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeads( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| T (&preds)[ITEMS_PER_THREAD], ///< [out] Calling thread's predecessor items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| // Share last item |
| temp_storage.last_items[linear_tid] = input[ITEMS_PER_THREAD - 1]; |
| |
| CTA_SYNC(); |
| |
| if (linear_tid == 0) |
| { |
| // Set flag for first thread-item (preds[0] is undefined) |
| head_flags[0] = 1; |
| } |
| else |
| { |
| preds[0] = temp_storage.last_items[linear_tid - 1]; |
| head_flags[0] = ApplyOp<FlagOp>::FlagT(flag_op, preds[0], input[0], linear_tid * ITEMS_PER_THREAD); |
| } |
| |
| // Set head_flags for remaining items |
| Iterate<1, ITEMS_PER_THREAD>::FlagHeads(linear_tid, head_flags, input, preds, flag_op); |
| } |
| |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeads( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| T (&preds)[ITEMS_PER_THREAD], ///< [out] Calling thread's predecessor items |
| FlagOp flag_op, ///< [in] Binary boolean flag predicate |
| T tile_predecessor_item) ///< [in] <b>[<em>thread</em><sub>0</sub> only]</b> Item with which to compare the first tile item (<tt>input<sub>0</sub></tt> from <em>thread</em><sub>0</sub>). |
| { |
| // Share last item |
| temp_storage.last_items[linear_tid] = input[ITEMS_PER_THREAD - 1]; |
| |
| CTA_SYNC(); |
| |
| // Set flag for first thread-item |
| preds[0] = (linear_tid == 0) ? |
| tile_predecessor_item : // First thread |
| temp_storage.last_items[linear_tid - 1]; |
| |
| head_flags[0] = ApplyOp<FlagOp>::FlagT(flag_op, preds[0], input[0], linear_tid * ITEMS_PER_THREAD); |
| |
| // Set head_flags for remaining items |
| Iterate<1, ITEMS_PER_THREAD>::FlagHeads(linear_tid, head_flags, input, preds, flag_op); |
| } |
| |
| #endif // DOXYGEN_SHOULD_SKIP_THIS |
| |
| |
| /** |
| * \brief Sets head flags indicating discontinuities between items partitioned across the thread block, for which the first item has no reference and is always flagged. |
| * |
| * \par |
| * - The flag <tt>head_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(</tt><em>previous-item</em><tt>, input<sub><em>i</em></sub>)</tt> |
| * returns \p true (where <em>previous-item</em> is either the preceding item |
| * in the same thread or the last item in the previous thread). |
| * - For <em>thread</em><sub>0</sub>, item <tt>input<sub>0</sub></tt> is always flagged. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the head-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Collectively compute head flags for discontinuities in the segment |
| * int head_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagHeads(head_flags, thread_data, cub::Inequality()); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], [3,4,4,4], ... }</tt>. |
| * The corresponding output \p head_flags in those threads will be |
| * <tt>{ [1,0,1,0], [0,0,0,0], [1,1,0,0], [0,1,0,0], ... }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeads( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| T preds[ITEMS_PER_THREAD]; |
| FlagHeads(head_flags, input, preds, flag_op); |
| } |
| |
|
|
| /** |
| * \brief Sets head flags indicating discontinuities between items partitioned across the thread block. |
| * |
| * \par |
| * - The flag <tt>head_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(</tt><em>previous-item</em><tt>, input<sub><em>i</em></sub>)</tt> |
| * returns \p true (where <em>previous-item</em> is either the preceding item |
| * in the same thread or the last item in the previous thread). |
| * - For <em>thread</em><sub>0</sub>, item <tt>input<sub>0</sub></tt> is compared |
| * against \p tile_predecessor_item. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the head-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Have thread0 obtain the predecessor item for the entire tile |
| * int tile_predecessor_item; |
| * if (threadIdx.x == 0) tile_predecessor_item == ... |
| * |
| * // Collectively compute head flags for discontinuities in the segment |
| * int head_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagHeads( |
| * head_flags, thread_data, cub::Inequality(), tile_predecessor_item); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], [3,4,4,4], ... }</tt>, |
| * and that \p tile_predecessor_item is \p 0. The corresponding output \p head_flags in those threads will be |
| * <tt>{ [0,0,1,0], [0,0,0,0], [1,1,0,0], [0,1,0,0], ... }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeads( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op, ///< [in] Binary boolean flag predicate |
| T tile_predecessor_item) ///< [in] <b>[<em>thread</em><sub>0</sub> only]</b> Item with which to compare the first tile item (<tt>input<sub>0</sub></tt> from <em>thread</em><sub>0</sub>). |
| { |
| T preds[ITEMS_PER_THREAD]; |
| FlagHeads(head_flags, input, preds, flag_op, tile_predecessor_item); |
| } |
| |
|
|
|
|
| //@} end member group |
| /******************************************************************//** |
| * \name Tail flag operations |
| *********************************************************************/ |
| //@{ |
| |
|
|
| /** |
| * \brief Sets tail flags indicating discontinuities between items partitioned across the thread block, for which the last item has no reference and is always flagged. |
| * |
| * \par |
| * - The flag <tt>tail_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(input<sub><em>i</em></sub>, </tt><em>next-item</em><tt>)</tt> |
| * returns \p true (where <em>next-item</em> is either the next item |
| * in the same thread or the first item in the next thread). |
| * - For <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>, item |
| * <tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> is always flagged. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the tail-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Collectively compute tail flags for discontinuities in the segment |
| * int tail_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagTails(tail_flags, thread_data, cub::Inequality()); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], ..., [124,125,125,125] }</tt>. |
| * The corresponding output \p tail_flags in those threads will be |
| * <tt>{ [0,1,0,0], [0,0,0,1], [1,0,0,...], ..., [1,0,0,1] }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagTails( |
| FlagT (&tail_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity tail_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| // Share first item |
| temp_storage.first_items[linear_tid] = input[0]; |
| |
| CTA_SYNC(); |
| |
| // Set flag for last thread-item |
| tail_flags[ITEMS_PER_THREAD - 1] = (linear_tid == BLOCK_THREADS - 1) ? |
| 1 : // Last thread |
| ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| input[ITEMS_PER_THREAD - 1], |
| temp_storage.first_items[linear_tid + 1], |
| (linear_tid * ITEMS_PER_THREAD) + ITEMS_PER_THREAD); |
| |
| // Set tail_flags for remaining items |
| Iterate<0, ITEMS_PER_THREAD - 1>::FlagTails(linear_tid, tail_flags, input, flag_op); |
| } |
| |
|
|
| /** |
| * \brief Sets tail flags indicating discontinuities between items partitioned across the thread block. |
| * |
| * \par |
| * - The flag <tt>tail_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(input<sub><em>i</em></sub>, </tt><em>next-item</em><tt>)</tt> |
| * returns \p true (where <em>next-item</em> is either the next item |
| * in the same thread or the first item in the next thread). |
| * - For <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>, item |
| * <tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> is compared |
| * against \p tile_successor_item. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the tail-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Have thread127 obtain the successor item for the entire tile |
| * int tile_successor_item; |
| * if (threadIdx.x == 127) tile_successor_item == ... |
| * |
| * // Collectively compute tail flags for discontinuities in the segment |
| * int tail_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagTails( |
| * tail_flags, thread_data, cub::Inequality(), tile_successor_item); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], ..., [124,125,125,125] }</tt> |
| * and that \p tile_successor_item is \p 125. The corresponding output \p tail_flags in those threads will be |
| * <tt>{ [0,1,0,0], [0,0,0,1], [1,0,0,...], ..., [1,0,0,0] }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagTails( |
| FlagT (&tail_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity tail_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op, ///< [in] Binary boolean flag predicate |
| T tile_successor_item) ///< [in] <b>[<em>thread</em><sub><tt>BLOCK_THREADS</tt>-1</sub> only]</b> Item with which to compare the last tile item (<tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> from <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>). |
| { |
| // Share first item |
| temp_storage.first_items[linear_tid] = input[0]; |
| |
| CTA_SYNC(); |
| |
| // Set flag for last thread-item |
| T successor_item = (linear_tid == BLOCK_THREADS - 1) ? |
| tile_successor_item : // Last thread |
| temp_storage.first_items[linear_tid + 1]; |
| |
| tail_flags[ITEMS_PER_THREAD - 1] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| input[ITEMS_PER_THREAD - 1], |
| successor_item, |
| (linear_tid * ITEMS_PER_THREAD) + ITEMS_PER_THREAD); |
| |
| // Set tail_flags for remaining items |
| Iterate<0, ITEMS_PER_THREAD - 1>::FlagTails(linear_tid, tail_flags, input, flag_op); |
| } |
| |
|
|
| //@} end member group |
| /******************************************************************//** |
| * \name Head & tail flag operations |
| *********************************************************************/ |
| //@{ |
| |
|
|
| /** |
| * \brief Sets both head and tail flags indicating discontinuities between items partitioned across the thread block. |
| * |
| * \par |
| * - The flag <tt>head_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(</tt><em>previous-item</em><tt>, input<sub><em>i</em></sub>)</tt> |
| * returns \p true (where <em>previous-item</em> is either the preceding item |
| * in the same thread or the last item in the previous thread). |
| * - For <em>thread</em><sub>0</sub>, item <tt>input<sub>0</sub></tt> is always flagged. |
| * - The flag <tt>tail_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(input<sub><em>i</em></sub>, </tt><em>next-item</em><tt>)</tt> |
| * returns \p true (where <em>next-item</em> is either the next item |
| * in the same thread or the first item in the next thread). |
| * - For <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>, item |
| * <tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> is always flagged. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the head- and tail-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Collectively compute head and flags for discontinuities in the segment |
| * int head_flags[4]; |
| * int tail_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagTails( |
| * head_flags, tail_flags, thread_data, cub::Inequality()); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], ..., [124,125,125,125] }</tt> |
| * and that the tile_successor_item is \p 125. The corresponding output \p head_flags |
| * in those threads will be <tt>{ [1,0,1,0], [0,0,0,0], [1,1,0,0], [0,1,0,0], ... }</tt>. |
| * and the corresponding output \p tail_flags in those threads will be |
| * <tt>{ [0,1,0,0], [0,0,0,1], [1,0,0,...], ..., [1,0,0,1] }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeadsAndTails( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| FlagT (&tail_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity tail_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| // Share first and last items |
| temp_storage.first_items[linear_tid] = input[0]; |
| temp_storage.last_items[linear_tid] = input[ITEMS_PER_THREAD - 1]; |
| |
| CTA_SYNC(); |
| |
| T preds[ITEMS_PER_THREAD]; |
| |
| // Set flag for first thread-item |
| preds[0] = temp_storage.last_items[linear_tid - 1]; |
| if (linear_tid == 0) |
| { |
| head_flags[0] = 1; |
| } |
| else |
| { |
| head_flags[0] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| preds[0], |
| input[0], |
| linear_tid * ITEMS_PER_THREAD); |
| } |
| |
|
|
| // Set flag for last thread-item |
| tail_flags[ITEMS_PER_THREAD - 1] = (linear_tid == BLOCK_THREADS - 1) ? |
| 1 : // Last thread |
| ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| input[ITEMS_PER_THREAD - 1], |
| temp_storage.first_items[linear_tid + 1], |
| (linear_tid * ITEMS_PER_THREAD) + ITEMS_PER_THREAD); |
| |
| // Set head_flags for remaining items |
| Iterate<1, ITEMS_PER_THREAD>::FlagHeads(linear_tid, head_flags, input, preds, flag_op); |
| |
| // Set tail_flags for remaining items |
| Iterate<0, ITEMS_PER_THREAD - 1>::FlagTails(linear_tid, tail_flags, input, flag_op); |
| } |
| |
|
|
| /** |
| * \brief Sets both head and tail flags indicating discontinuities between items partitioned across the thread block. |
| * |
| * \par |
| * - The flag <tt>head_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(</tt><em>previous-item</em><tt>, input<sub><em>i</em></sub>)</tt> |
| * returns \p true (where <em>previous-item</em> is either the preceding item |
| * in the same thread or the last item in the previous thread). |
| * - For <em>thread</em><sub>0</sub>, item <tt>input<sub>0</sub></tt> is always flagged. |
| * - The flag <tt>tail_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(input<sub><em>i</em></sub>, </tt><em>next-item</em><tt>)</tt> |
| * returns \p true (where <em>next-item</em> is either the next item |
| * in the same thread or the first item in the next thread). |
| * - For <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>, item |
| * <tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> is compared |
| * against \p tile_predecessor_item. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the head- and tail-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Have thread127 obtain the successor item for the entire tile |
| * int tile_successor_item; |
| * if (threadIdx.x == 127) tile_successor_item == ... |
| * |
| * // Collectively compute head and flags for discontinuities in the segment |
| * int head_flags[4]; |
| * int tail_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagTails( |
| * head_flags, tail_flags, tile_successor_item, thread_data, cub::Inequality()); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], ..., [124,125,125,125] }</tt> |
| * and that the tile_successor_item is \p 125. The corresponding output \p head_flags |
| * in those threads will be <tt>{ [1,0,1,0], [0,0,0,0], [1,1,0,0], [0,1,0,0], ... }</tt>. |
| * and the corresponding output \p tail_flags in those threads will be |
| * <tt>{ [0,1,0,0], [0,0,0,1], [1,0,0,...], ..., [1,0,0,0] }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeadsAndTails( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| FlagT (&tail_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity tail_flags |
| T tile_successor_item, ///< [in] <b>[<em>thread</em><sub><tt>BLOCK_THREADS</tt>-1</sub> only]</b> Item with which to compare the last tile item (<tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> from <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>). |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| // Share first and last items |
| temp_storage.first_items[linear_tid] = input[0]; |
| temp_storage.last_items[linear_tid] = input[ITEMS_PER_THREAD - 1]; |
| |
| CTA_SYNC(); |
| |
| T preds[ITEMS_PER_THREAD]; |
| |
| // Set flag for first thread-item |
| if (linear_tid == 0) |
| { |
| head_flags[0] = 1; |
| } |
| else |
| { |
| preds[0] = temp_storage.last_items[linear_tid - 1]; |
| head_flags[0] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| preds[0], |
| input[0], |
| linear_tid * ITEMS_PER_THREAD); |
| } |
| |
| // Set flag for last thread-item |
| T successor_item = (linear_tid == BLOCK_THREADS - 1) ? |
| tile_successor_item : // Last thread |
| temp_storage.first_items[linear_tid + 1]; |
| |
| tail_flags[ITEMS_PER_THREAD - 1] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| input[ITEMS_PER_THREAD - 1], |
| successor_item, |
| (linear_tid * ITEMS_PER_THREAD) + ITEMS_PER_THREAD); |
| |
| // Set head_flags for remaining items |
| Iterate<1, ITEMS_PER_THREAD>::FlagHeads(linear_tid, head_flags, input, preds, flag_op); |
| |
| // Set tail_flags for remaining items |
| Iterate<0, ITEMS_PER_THREAD - 1>::FlagTails(linear_tid, tail_flags, input, flag_op); |
| } |
| |
|
|
| /** |
| * \brief Sets both head and tail flags indicating discontinuities between items partitioned across the thread block. |
| * |
| * \par |
| * - The flag <tt>head_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(</tt><em>previous-item</em><tt>, input<sub><em>i</em></sub>)</tt> |
| * returns \p true (where <em>previous-item</em> is either the preceding item |
| * in the same thread or the last item in the previous thread). |
| * - For <em>thread</em><sub>0</sub>, item <tt>input<sub>0</sub></tt> is compared |
| * against \p tile_predecessor_item. |
| * - The flag <tt>tail_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(input<sub><em>i</em></sub>, </tt><em>next-item</em><tt>)</tt> |
| * returns \p true (where <em>next-item</em> is either the next item |
| * in the same thread or the first item in the next thread). |
| * - For <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>, item |
| * <tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> is always flagged. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the head- and tail-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Have thread0 obtain the predecessor item for the entire tile |
| * int tile_predecessor_item; |
| * if (threadIdx.x == 0) tile_predecessor_item == ... |
| * |
| * // Have thread127 obtain the successor item for the entire tile |
| * int tile_successor_item; |
| * if (threadIdx.x == 127) tile_successor_item == ... |
| * |
| * // Collectively compute head and flags for discontinuities in the segment |
| * int head_flags[4]; |
| * int tail_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagTails( |
| * head_flags, tile_predecessor_item, tail_flags, tile_successor_item, |
| * thread_data, cub::Inequality()); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], ..., [124,125,125,125] }</tt>, |
| * that the \p tile_predecessor_item is \p 0, and that the |
| * \p tile_successor_item is \p 125. The corresponding output \p head_flags |
| * in those threads will be <tt>{ [0,0,1,0], [0,0,0,0], [1,1,0,0], [0,1,0,0], ... }</tt>. |
| * and the corresponding output \p tail_flags in those threads will be |
| * <tt>{ [0,1,0,0], [0,0,0,1], [1,0,0,...], ..., [1,0,0,1] }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeadsAndTails( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T tile_predecessor_item, ///< [in] <b>[<em>thread</em><sub>0</sub> only]</b> Item with which to compare the first tile item (<tt>input<sub>0</sub></tt> from <em>thread</em><sub>0</sub>). |
| FlagT (&tail_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity tail_flags |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| // Share first and last items |
| temp_storage.first_items[linear_tid] = input[0]; |
| temp_storage.last_items[linear_tid] = input[ITEMS_PER_THREAD - 1]; |
| |
| CTA_SYNC(); |
| |
| T preds[ITEMS_PER_THREAD]; |
| |
| // Set flag for first thread-item |
| preds[0] = (linear_tid == 0) ? |
| tile_predecessor_item : // First thread |
| temp_storage.last_items[linear_tid - 1]; |
| |
| head_flags[0] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| preds[0], |
| input[0], |
| linear_tid * ITEMS_PER_THREAD); |
| |
| // Set flag for last thread-item |
| tail_flags[ITEMS_PER_THREAD - 1] = (linear_tid == BLOCK_THREADS - 1) ? |
| 1 : // Last thread |
| ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| input[ITEMS_PER_THREAD - 1], |
| temp_storage.first_items[linear_tid + 1], |
| (linear_tid * ITEMS_PER_THREAD) + ITEMS_PER_THREAD); |
| |
| // Set head_flags for remaining items |
| Iterate<1, ITEMS_PER_THREAD>::FlagHeads(linear_tid, head_flags, input, preds, flag_op); |
| |
| // Set tail_flags for remaining items |
| Iterate<0, ITEMS_PER_THREAD - 1>::FlagTails(linear_tid, tail_flags, input, flag_op); |
| } |
| |
|
|
| /** |
| * \brief Sets both head and tail flags indicating discontinuities between items partitioned across the thread block. |
| * |
| * \par |
| * - The flag <tt>head_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(</tt><em>previous-item</em><tt>, input<sub><em>i</em></sub>)</tt> |
| * returns \p true (where <em>previous-item</em> is either the preceding item |
| * in the same thread or the last item in the previous thread). |
| * - For <em>thread</em><sub>0</sub>, item <tt>input<sub>0</sub></tt> is compared |
| * against \p tile_predecessor_item. |
| * - The flag <tt>tail_flags<sub><em>i</em></sub></tt> is set for item |
| * <tt>input<sub><em>i</em></sub></tt> when |
| * <tt>flag_op(input<sub><em>i</em></sub>, </tt><em>next-item</em><tt>)</tt> |
| * returns \p true (where <em>next-item</em> is either the next item |
| * in the same thread or the first item in the next thread). |
| * - For <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>, item |
| * <tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> is compared |
| * against \p tile_successor_item. |
| * - \blocked |
| * - \granularity |
| * - \smemreuse |
| * |
| * \par Snippet |
| * The code snippet below illustrates the head- and tail-flagging of 512 integer items that |
| * are partitioned in a [<em>blocked arrangement</em>](index.html#sec5sec3) across 128 threads |
| * where each thread owns 4 consecutive items. |
| * \par |
| * \code |
| * #include <cub/cub.cuh> // or equivalently <cub/block/block_discontinuity.cuh> |
| * |
| * __global__ void ExampleKernel(...) |
| * { |
| * // Specialize BlockDiscontinuity for a 1D block of 128 threads on type int |
| * typedef cub::BlockDiscontinuity<int, 128> BlockDiscontinuity; |
| * |
| * // Allocate shared memory for BlockDiscontinuity |
| * __shared__ typename BlockDiscontinuity::TempStorage temp_storage; |
| * |
| * // Obtain a segment of consecutive items that are blocked across threads |
| * int thread_data[4]; |
| * ... |
| * |
| * // Have thread0 obtain the predecessor item for the entire tile |
| * int tile_predecessor_item; |
| * if (threadIdx.x == 0) tile_predecessor_item == ... |
| * |
| * // Have thread127 obtain the successor item for the entire tile |
| * int tile_successor_item; |
| * if (threadIdx.x == 127) tile_successor_item == ... |
| * |
| * // Collectively compute head and flags for discontinuities in the segment |
| * int head_flags[4]; |
| * int tail_flags[4]; |
| * BlockDiscontinuity(temp_storage).FlagTails( |
| * head_flags, tile_predecessor_item, tail_flags, tile_successor_item, |
| * thread_data, cub::Inequality()); |
| * |
| * \endcode |
| * \par |
| * Suppose the set of input \p thread_data across the block of threads is |
| * <tt>{ [0,0,1,1], [1,1,1,1], [2,3,3,3], ..., [124,125,125,125] }</tt>, |
| * that the \p tile_predecessor_item is \p 0, and that the |
| * \p tile_successor_item is \p 125. The corresponding output \p head_flags |
| * in those threads will be <tt>{ [0,0,1,0], [0,0,0,0], [1,1,0,0], [0,1,0,0], ... }</tt>. |
| * and the corresponding output \p tail_flags in those threads will be |
| * <tt>{ [0,1,0,0], [0,0,0,1], [1,0,0,...], ..., [1,0,0,0] }</tt>. |
| * |
| * \tparam ITEMS_PER_THREAD <b>[inferred]</b> The number of consecutive items partitioned onto each thread. |
| * \tparam FlagT <b>[inferred]</b> The flag type (must be an integer type) |
| * \tparam FlagOp <b>[inferred]</b> Binary predicate functor type having member <tt>T operator()(const T &a, const T &b)</tt> or member <tt>T operator()(const T &a, const T &b, unsigned int b_index)</tt>, and returning \p true if a discontinuity exists between \p a and \p b, otherwise \p false. \p b_index is the rank of b in the aggregate tile of data. |
| */ |
| template < |
| int ITEMS_PER_THREAD, |
| typename FlagT, |
| typename FlagOp> |
| __device__ __forceinline__ void FlagHeadsAndTails( |
| FlagT (&head_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity head_flags |
| T tile_predecessor_item, ///< [in] <b>[<em>thread</em><sub>0</sub> only]</b> Item with which to compare the first tile item (<tt>input<sub>0</sub></tt> from <em>thread</em><sub>0</sub>). |
| FlagT (&tail_flags)[ITEMS_PER_THREAD], ///< [out] Calling thread's discontinuity tail_flags |
| T tile_successor_item, ///< [in] <b>[<em>thread</em><sub><tt>BLOCK_THREADS</tt>-1</sub> only]</b> Item with which to compare the last tile item (<tt>input</tt><sub><em>ITEMS_PER_THREAD</em>-1</sub> from <em>thread</em><sub><em>BLOCK_THREADS</em>-1</sub>). |
| T (&input)[ITEMS_PER_THREAD], ///< [in] Calling thread's input items |
| FlagOp flag_op) ///< [in] Binary boolean flag predicate |
| { |
| // Share first and last items |
| temp_storage.first_items[linear_tid] = input[0]; |
| temp_storage.last_items[linear_tid] = input[ITEMS_PER_THREAD - 1]; |
| |
| CTA_SYNC(); |
| |
| T preds[ITEMS_PER_THREAD]; |
| |
| // Set flag for first thread-item |
| preds[0] = (linear_tid == 0) ? |
| tile_predecessor_item : // First thread |
| temp_storage.last_items[linear_tid - 1]; |
| |
| head_flags[0] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| preds[0], |
| input[0], |
| linear_tid * ITEMS_PER_THREAD); |
| |
| // Set flag for last thread-item |
| T successor_item = (linear_tid == BLOCK_THREADS - 1) ? |
| tile_successor_item : // Last thread |
| temp_storage.first_items[linear_tid + 1]; |
| |
| tail_flags[ITEMS_PER_THREAD - 1] = ApplyOp<FlagOp>::FlagT( |
| flag_op, |
| input[ITEMS_PER_THREAD - 1], |
| successor_item, |
| (linear_tid * ITEMS_PER_THREAD) + ITEMS_PER_THREAD); |
| |
| // Set head_flags for remaining items |
| Iterate<1, ITEMS_PER_THREAD>::FlagHeads(linear_tid, head_flags, input, preds, flag_op); |
| |
| // Set tail_flags for remaining items |
| Iterate<0, ITEMS_PER_THREAD - 1>::FlagTails(linear_tid, tail_flags, input, flag_op); |
| } |
| |
|
|
|
|
|
|
| //@} end member group |
| |
| }; |
|
|
|
|
| } // CUB namespace |
| CUB_NS_POSTFIX // Optional outer namespace(s) |
|
|