/Project2-Stream-Compaction

CIS565 Project2-Stream-Compaction

Primary LanguageCMake

CUDA Stream Compaction

University of Pennsylvania, CIS 565: GPU Programming and Architecture, Project 2

  • Ruoyu Fan
  • Tested on: Windows 10 x64, i7-6700K @ 4.00GHz 16GB, GTX 970 4096MB (girlfriend's machine)
    • compiled with Visual Studio 2013 and CUDA 7.5

NOTE: if the program crashes when entering the test for naive sort, try reducing the array sizes in main.cpp. (SIZE and SORT_SIZE.)

preview

Things I have done

  • Implemented CPU scan and compaction, compaction, GPU naive scan, GPU work-efficient scan, GPU work-efficient compaction, GPU radix sort (extra), and compared my scan algorithms with thrust implemention

  • I optimized my work efficient scan, and speed is increased to 270% of my original implementation, please refer to Optimization section.

  • I also wrote an inclusive version of work-efficient scan - because i misunderstood the requirement at first! The difference of the inclusive method is that it creates a buffer that is 1 element larger and swap the last(0) and and second last elements before downsweeping. Although I corrected my implemention to exclusive scan, the inclusive scan can still be called by passing ScanType::inclusive to scan_implenmention method in efficient.cu.

  • Radix sort assumes inputs are between [0, a_given_maximum) . I compared my radix sort with std::sort and thrust's unstable and stable sort.

  • I added a helper class PerformanceTimer in common.h which is used to do performance measurement.

Original Questions

* Roughly optimize the block sizes of each of your implementations for minimal
  run time on your GPU.
  * (You shouldn't compare unoptimized implementations to each other!)

* Compare all of these GPU Scan implementations (Naive, Work-Efficient, and
  Thrust) to the serial CPU version of Scan. Plot a graph of the comparison
  (with array size on the independent axis).
  * You should use CUDA events for timing GPU code. Be sure **not** to include
    any *initial/final* memory operations (`cudaMalloc`, `cudaMemcpy`) in your
    performance measurements, for comparability. Note that CUDA events cannot
    time CPU code.
  * You can use the C++11 `std::chrono` API for timing CPU code. See this
    [Stack Overflow answer](http://stackoverflow.com/a/23000049) for an example.
    Note that `std::chrono` may not provide high-precision timing. If it does
    not, you can either use it to time many iterations, or use another method.
  * To guess at what might be happening inside the Thrust implementation (e.g.
    allocation, memory copy), take a look at the Nsight timeline for its
    execution. Your analysis here doesn't have to be detailed, since you aren't
    even looking at the code for the implementation.
  • Please refer to Performance section.
* Write a brief explanation of the phenomena you see here.
  * Can you find the performance bottlenecks? Is it memory I/O? Computation? Is
    it different for each implementation?
  • I notice that I couldn't get a good measurement for scan and sort of Thrust. I have trouble measuring thrust::exclusive with std::chrono, while I can use std::chrono to measure thrust::scan but the results from CUDA events seems off. I passed a host array to thrust so in my original tests thrust is using a CPU sort algorithm, fixed and updated result in result_radix_max_100.txt and result_radix_max_100000000.txt

  • With max possible value increased (so does MSB), besides the run time of radix sort, that of std::sort also increased significantly, which is unexpected. Does bit length also affect the time for read/write or other operations? or the implementation of std::sort use bit level information?

  • I think the bottleneck for blocksize is the warp size inside GPU.

  • My original work-efficient scan implementation was slower than CPU scan, and then I optimized it by minimizing wasted threads. and it now runs faster than CPU scan, please refer to Optimization section below.

* Paste the output of the test program into a triple-backtick block in your
  README.
  * If you add your own tests (e.g. for radix sort or to test additional corner
    cases), be sure to mention it explicitly.

These questions should help guide you in performance analysis on future
assignments, as well.
  • The tests are done with arrays of lengths 2^26 (67108864) and 2^26-3 (67108861). The array generation uses current time as random seed.

  • I added tests for radix sort, which compares with std::sort as well as Thrust's thrust::sort

Performance

Blocksize

When block size is smaller than 16, my application suffers from performance drop, which is recorded in test_results folder. I decided to just use cudaOccupancyMaxPotentialBlockSize for each device functions, which is almost 1024 on my computer.

chart_blocksize

Block size Naïve Scan(ms) Work-efficient Scan (ms) Reference CPU Scan (ms)
4 755.433 76.5717 134.408
8 379.897 52.1942 134.408
16 212.542 44.3224 134.408
32 133.116 43.4925 134.408
64 114.7 44.7172 134.408
128 113.802 44.7841 134.408
256 114.575 45.3902 134.408
512 114.593 44.3084 134.408
1024 113.867 44.2941 134.408

Array length

All the test are done with block size of 1024. The possible max value for sorting is 100.

Scan : this work-efficient scan implementation is faster than cpu scan on large input but slower than Thrust's

chart_scan

Input array length power of 2 CPU Scan (ms) Naïve Scan (ms) Work-efficient scan (ms) Thrust scan (ms) (CUDA event)
12 0 0.047872 0.110784 0.028192
16 0 0.09216 0.180064 0.166432
20 0 1.29402 0.818528 0.30768
24 23.5625 25.7094 11.3379 2.00646
26 159.925 133.61 44.4511 7.7283

Compaction : this work-efficient compaction implementation is faster than cpu's

chart_compact

Input array length power of 2 CPU compact (ms) CPU scan compact (ms) Work-efficient compact (ms)
12 0 0 0.118368
16 0 0 0.193408
20 0 0 0.966944
24 39.0743 38.6241 13.3034
26 155.406 422.524 53.2659

Sort : radix sort on GPU is faster than std::sort

chart_sort

Input array length power of 2 std::sort (ms) Radix sort (ms)
12 0 0.883328
16 0 1.44288
20 22.1649 7.61686
24 378.025 105.222
26 1481.8 419.345

Data maximum value and radix sort

NOTE: THIS TEST RUNS ON A DIFFERENT MACHINE: Windows 7, Xeon(R) E5-1630 @ 3.70GHz 32GB, GTX 1070 8192MB (Moore 103 SigLab)

All the test are done with block size of 1024; array length is 67108864 33554432.

max value std::sort (ms) Radix sort (ms) thrust::sort (ms)
100 917 143.992 20.2813ms
1000000000 2173 1023.97 20.6702ms

With max possible value increased, besides the run time of radix sort, that of std::sort and thrust sorts also increased.

I peeked at thrust's inner function call and found thrust is using a radix sort algorithm.

Sample Output

* THIS TEST RAN ON A DIFFERENT MACHINE:
    Windows 7, Xeon(R) E5-1630 @ 3.70GHz 32GB, GTX 1070 8192MB (Moore 103 SigLab)

CIS-565 HW2 CUDA Stream Compaction Test (Ruoyu Fan)
    Block size for naive scan: 1024
    Block size for up-sweep: 1024
    Block size for down-sweep: 1024
    Block size for boolean mapping: 1024
    Block size for scattering: 1024
    Block sizes for radix sort: 1024 1024 1024 1024

****************
** SCAN TESTS **
****************
Array size (power of two): 33554432
Array size (non-power of two): 33554429
    [  13  29  47   7   2  28  44  49  35  46   2  49   9 ...  37   0 ]
==== cpu scan, power-of-two ====
   elapsed time: 93ms    (std::chrono Measured)
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752752 821752789 ]
==== cpu scan, non-power-of-two ====
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752661 821752701 ]
   elapsed time: 87ms    (std::chrono Measured)
    passed
==== naive scan, power-of-two ====
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752752 821752789 ]
   elapsed time: 38.1036ms    (CUDA Measured)
    passed
==== naive scan, non-power-of-two ====
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752661 821752701 ]
   elapsed time: 38.112ms    (CUDA Measured)
    passed
==== work-efficient scan, power-of-two ====
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752752 821752789 ]
   elapsed time: 15.0276ms    (CUDA Measured)
    passed
==== work-efficient scan, non-power-of-two ====
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752661 821752701 ]
   elapsed time: 15.0576ms    (CUDA Measured)
    passed
==== thrust scan, power-of-two ====
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752752 821752789 ]
   elapsed time: 8.90237ms    (CUDA Measured)
    passed
==== thrust scan, non-power-of-two ====
    [   0  13  42  89  96  98 126 170 219 254 300 302 351 ... 821752661 821752701 ]
   elapsed time: 2.74368ms    (CUDA Measured)
    passed

*****************************
** STREAM COMPACTION TESTS **
*****************************
Array size (power of two): 33554432
Array size (non-power of two): 33554429
    [   1   1   3   1   1   3   0   2   3   2   0   1   0 ...   3   0 ]
==== cpu compact without scan, power-of-two ====
    [   1   1   3   1   1   3   2   3   2   1   3   2   2 ...   2   3 ]
   elapsed time: 83ms    (std::chrono Measured)
    passed
==== cpu compact without scan, non-power-of-two ====
    [   1   1   3   1   1   3   2   3   2   1   3   2   2 ...   3   2 ]
   elapsed time: 84ms    (std::chrono Measured)
    passed
==== cpu compact with scan ====
    [   1   1   3   1   1   3   2   3   2   1   3   2   2 ...   2   3 ]
   elapsed time: 237ms    (std::chrono Measured)
    passed
==== work-efficient compact, power-of-two ====
    [   1   1   3   1   1   3   2   3   2   1   3   2   2 ...   2   3 ]
   elapsed time: 18.2364ms    (CUDA Measured)
    passed
==== work-efficient compact, non-power-of-two ====
    [   1   1   3   1   1   3   2   3   2   1   3   2   2 ...   3   2 ]
   elapsed time: 18.2344ms    (CUDA Measured)
    passed

*****************************
** RADIX SORT TESTS **
*****************************
Array size (power of two): 33554432
Array size (non-power of two): 33554429
Max value: 1000000000
    [ 21520 17257 9407 8648 31232 11282 11169 22994 15890 9350 22656 25538 29919 ... 23658   0 ]
==== std::sort, power-of-two ====
    [   0   0   0   0   0   0   0   0   0   0   0   0   0 ... 32767 32767 ]
   elapsed time: 2173ms    (std::chrono Measured)
==== thrust::sort (which calls Thrust's radix sort), power-of-two ====
    [   0   0   0   0   0   0   0   0   0   0   0   0   0 ... 32767 32767 ]
   elapsed time: 20.6702ms    (CUDA Measured)
    passed
==== radix sort, power-of-two ====
    [   0   0   0   0   0   0   0   0   0   0   0   0   0 ... 32767 32767 ]
   elapsed time: 628.72ms    (CUDA Measured)
    passed
==== std::sort, non power-of-two ====
    [   0   0   0   0   0   0   0   0   0   0   0   0   0 ... 32767 32767 ]
   elapsed time: 2153ms    (std::chrono Measured)
==== radix sort, non power-of-two ====
    [   0   0   0   0   0   0   0   0   0   0   0   0   0 ... 32767 32767 ]
   elapsed time: 617.616ms    (CUDA Measured)
    passed

Optimization

Run less threads for work-efficient scan

For work-efficient scan, my original implementation was using the same of amount of threads for every up sweep and down sweeps. Then I optimized it by using only necessary amount of threads for each iteration.

The performance for scanning an array of length 67108861 using work-efficient approach boosted from ~120.5ms to ~44.4ms, which is 270% speed of my original approach. You can see the data in the files under test_results/ folder

chart_scan_optimization

Original implementation (in which (index + 1) % (add_distance * 2) == 0 is false at many threads so these threads were wasted):

// running unnecessary threads
__global__ void kernScanUpSweepPass(int N, int add_distance, int* buffer)
{
    auto index = threadIdx.x + blockIdx.x * blockDim.x;
    if (index >= N) { return; }

    if ((index + 1) % (add_distance * 2) == 0)
    {
        buffer[index] = buffer[index] + buffer[index - add_distance];
    }
}

__global__ void kernScanDownSweepPass(int N, int distance, int* buffer)
{
    auto index = threadIdx.x + blockIdx.x * blockDim.x;
    if (index >= N) { return; }

    if ((index + 1) % (distance * 2) == 0)
    {
        auto temp = buffer[index - distance];
        buffer[index - distance] = buffer[index];
        buffer[index] = temp + buffer[index];
    }
}

New implementation:

// optimized: only launch necessary amount of threads in host code
__global__ void kernScanUpSweepPass(int max_thread_index, int add_distance, int* buffer)
{
    auto tindex = threadIdx.x + blockIdx.x * blockDim.x;

    if (tindex >= max_thread_index) { return; }

    // I encountered overflow problem with index < N here so I changed to tindex < max_thread_index
    size_t index = (add_distance * 2) * (1 + tindex) - 1;

    buffer[index] = buffer[index] + buffer[index - add_distance];
}

// optimized: only launch necessary amount of threads in host code
__global__ void kernScanDownSweepPass(int max_thread_index, int distance, int* buffer)
{
    auto tindex = threadIdx.x + blockIdx.x * blockDim.x;

    if (tindex >= max_thread_index) { return; }

    size_t index = (distance * 2) * (1 + tindex) - 1;  

    auto temp = buffer[index - distance];
    buffer[index - distance] = buffer[index];
    buffer[index] = temp + buffer[index];
}

And I calculated the number of threads needed as well as the maximum thread index for every up-sweep and down-sweep pass.

Originally I was still using length of buffer as first parameter, but when I was calculating indices for a thread by using the condition of (distance * 2) * (1 + tindex) - 1 > N. There can come some weird result because of the multiplication result is out of bound (even for size_t - it took me 2 hours to debug that). So lessons learned, and I'll use more n > b/a instead of a*n > b as condition in the future.

Helper class for performance measurement

I create a RAII PerformanceTimer class for performance measurement. Which is like:

/**
* This class is used for timing the performance
* Uncopyable and unmovable
*/
class PerformanceTimer
{
public:
    PerformanceTimer()
    {
        cudaEventCreate(&event_start);
        cudaEventCreate(&event_end);
    }

    ~PerformanceTimer()
    {
        cudaEventDestroy(event_start);
        cudaEventDestroy(event_end);
    }

    void startCpuTimer()
    {
        if (cpu_timer_started) { throw std::runtime_error("CPU timer already started"); }
        cpu_timer_started = true;

        time_start_cpu = std::chrono::high_resolution_clock::now();
    }

    void endCpuTimer()
    {
        time_end_cpu = std::chrono::high_resolution_clock::now();

        if (!cpu_timer_started) { throw std::runtime_error("CPU timer not started"); }

        std::chrono::duration<double, std::milli> duro = time_end_cpu - time_start_cpu;
        prev_elapsed_time_cpu_milliseconds =
            static_cast<decltype(prev_elapsed_time_cpu_milliseconds)>(duro.count());

        cpu_timer_started = false;
    }

    void startGpuTimer()
    {
        if (gpu_timer_started) { throw std::runtime_error("GPU timer already started"); }
        gpu_timer_started = true;

        cudaEventRecord(event_start);
    }

    void endGpuTimer()
    {
        cudaEventRecord(event_end);
        cudaEventSynchronize(event_end);

        if (!gpu_timer_started) { throw std::runtime_error("GPU timer not started"); }

        cudaEventElapsedTime(&prev_elapsed_time_gpu_milliseconds, event_start, event_end);
        gpu_timer_started = false;
    }

    float getCpuElapsedTimeForPreviousOperation() noexcept
    {
        return prev_elapsed_time_cpu_milliseconds;
    }

    float getGpuElapsedTimeForPreviousOperation() noexcept
    {
        return prev_elapsed_time_gpu_milliseconds;
    }

    // remove copy and move functions
    PerformanceTimer(const PerformanceTimer&) = delete;
    PerformanceTimer(PerformanceTimer&&) = delete;
    PerformanceTimer& operator=(const PerformanceTimer&) = delete;
    PerformanceTimer& operator=(PerformanceTimer&&) = delete;

private:
    cudaEvent_t event_start = nullptr;
    cudaEvent_t event_end = nullptr;

    using time_point_t = std::chrono::high_resolution_clock::time_point;
    time_point_t time_start_cpu;
    time_point_t time_end_cpu;

    bool cpu_timer_started = false;
    bool gpu_timer_started = false;

    float prev_elapsed_time_cpu_milliseconds = 0.f;
    float prev_elapsed_time_gpu_milliseconds = 0.f;
};

And inside a module I have:

using StreamCompaction::Common::PerformanceTimer;
PerformanceTimer& timer()
{
    // not thread-safe
    static PerformanceTimer timer;
    return timer;
}

Therefore, I can use

void someFunc()
{
    allocateYourBuffers()

    timer().startGpuTimer();

    doYourGpuScan();

    timer().endGpuTimer();

    endYourJob();
}

and

timer().getGpuElapsedTimeForPreviousOperation();

to get the measured elapsed time for the operation.