// Copyright John Maddock 2016. // Copyright Matt Borland 2024. // Use, modification and distribution are subject to the // Boost Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) #include #include #include #include #include "cuda_managed_ptr.hpp" #include "stopwatch.hpp" // For the CUDA runtime routines (prefixed with "cuda_") #include typedef float float_type; /** * CUDA Kernel Device code * */ __global__ void cuda_test(const float_type *in, float_type *out, int numElements) { using std::cos; int i = blockDim.x * blockIdx.x + threadIdx.x; if (i < numElements) { out[i] = boost::math::sin_pi(in[i]); } } /** * Host main routine */ int main(void) { // Error code to check return values for CUDA calls cudaError_t err = cudaSuccess; // Print the vector length to be used, and compute its size int numElements = 50000; std::cout << "[Vector operation on " << numElements << " elements]" << std::endl; // Allocate the managed input vector A cuda_managed_ptr input_vector(numElements); // Allocate the managed output vector C cuda_managed_ptr output_vector(numElements); // Initialize the input vectors for (int i = 0; i < numElements; ++i) { input_vector[i] = rand()/(float_type)RAND_MAX; } // Launch the Vector Add CUDA Kernel int threadsPerBlock = 1024; int blocksPerGrid =(numElements + threadsPerBlock - 1) / threadsPerBlock; std::cout << "CUDA kernel launch with " << blocksPerGrid << " blocks of " << threadsPerBlock << " threads" << std::endl; watch w; cuda_test<<>>(input_vector.get(), output_vector.get(), numElements); cudaDeviceSynchronize(); std::cout << "CUDA kernal done in: " << w.elapsed() << "s" << std::endl; err = cudaGetLastError(); if (err != cudaSuccess) { std::cerr << "Failed to launch vectorAdd kernel (error code " << cudaGetErrorString(err) << ")!" << std::endl; return EXIT_FAILURE; } // Verify that the result vector is correct std::vector results; results.reserve(numElements); w.reset(); for(int i = 0; i < numElements; ++i) results.push_back(boost::math::sin_pi(input_vector[i])); double t = w.elapsed(); // check the results for(int i = 0; i < numElements; ++i) { if (boost::math::epsilon_difference(output_vector[i], results[i]) > 10) { std::cerr << "Result verification failed at element " << i << "!" << std::endl; return EXIT_FAILURE; } } std::cout << "Test PASSED, normal calculation time: " << t << "s" << std::endl; std::cout << "Done\n"; return 0; }