PSCF v1.2
CudaRandom.cu
1#include "CudaRandom.h"
2#include <util/global.h>
3#include <sys/time.h>
4#include <string>
5
6namespace Pscf {
7
8 using namespace Util;
9
10 /*
11 * Constructor.
12 */
14 : gen_(),
15 seed_(0),
16 isInitialized_(false)
17 {
18 // Create pseudo-random number generator on gpu
19 curandStatus_t status;
20 status = curandCreateGenerator(&gen_, CURAND_RNG_PSEUDO_DEFAULT);
21 errorCheck(status);
22 }
23
24 /*
25 * Destructor.
26 */
29
30 /*
31 * Sets of random seed, and initializes random number generator.
32 *
33 * \param seed value for random seed (private member variable seed)
34 */
35 void CudaRandom::setSeed(unsigned long long seed)
36 {
37 if (seed == 0) {
38 timeval time;
39 gettimeofday(&time, NULL);
40 seed_ = time.tv_sec + 1123*time.tv_usec;
41 } else {
42 seed_ = seed;
43 }
44 curandStatus_t status;
45 status = curandSetPseudoRandomGeneratorSeed(gen_, seed_);
46 errorCheck(status);
47
48 isInitialized_ = true;
49 }
50
51 /*
52 * Populate array on device with random floats in (0, 1], uniform dist.
53 */
55 {
56 UTIL_CHECK(data.capacity() > 0);
57 if (!isInitialized_) {
58 setSeed(0);
59 }
60
61 curandStatus_t status = curandGenerateUniform(gen_, data.cArray(),
62 data.capacity());
63 errorCheck(status);
64 }
65
66 /*
67 * Populate array on device with random doubles in (0, 1], uniform dist.
68 */
70 {
71 UTIL_CHECK(data.capacity() > 0);
72 if (!isInitialized_) {
73 setSeed(0);
74 }
75
76 curandStatus_t status = curandGenerateUniformDouble(gen_, data.cArray(),
77 data.capacity());
78 errorCheck(status);
79 }
80
81 /*
82 * Populate array with normal-distributed random floats.
83 */
84 void CudaRandom::normal(DeviceArray<float>& data, float stddev, float mean)
85 {
86 UTIL_CHECK(data.capacity() > 0);
87 if (!isInitialized_) {
88 setSeed(0);
89 }
90
91 int n = data.capacity();
92 if (n % 2 == 1) {
93 UTIL_THROW("normal() requires array size to be an even number.");
94 }
95
96 curandStatus_t status = curandGenerateNormal(gen_, data.cArray(),
97 n, mean, stddev);
98 errorCheck(status);
99 }
100
101 /*
102 * Populate array with normal-distributed random doubles.
103 */
105 double stddev, double mean)
106 {
107 UTIL_CHECK(data.capacity() > 0);
108 if (!isInitialized_) {
109 setSeed(0);
110 }
111
112 int n = data.capacity();
113 if (n % 2 == 1) {
114 UTIL_THROW("normal() requires array size to be an even number.");
115 }
116
117 curandStatus_t status = curandGenerateNormalDouble(gen_, data.cArray(),
118 n, mean, stddev);
119 errorCheck(status);
120 }
121
122 /*
123 * Check generator error status. If not success, print info and throw error.
124 */
125 void CudaRandom::errorCheck(curandStatus_t const & error)
126 {
127 if (error == CURAND_STATUS_SUCCESS) {
128 return;
129 } else {
130 std::string errString;
131 switch (error)
132 {
133 default:
134 errString = "UNKNOWN";
135 break;
136 case CURAND_STATUS_VERSION_MISMATCH:
137 errString = "CURAND_STATUS_VERSION_MISMATCH";
138 break;
139 case CURAND_STATUS_NOT_INITIALIZED:
140 errString = "CURAND_STATUS_NOT_INITIALIZED";
141 break;
142 case CURAND_STATUS_ALLOCATION_FAILED:
143 errString = "CURAND_STATUS_ALLOCATION_FAILED";
144 break;
145 case CURAND_STATUS_TYPE_ERROR:
146 errString = "CURAND_STATUS_TYPE_ERROR";
147 break;
148 case CURAND_STATUS_OUT_OF_RANGE:
149 errString = "CURAND_STATUS_OUT_OF_RANGE";
150 break;
151 case CURAND_STATUS_LENGTH_NOT_MULTIPLE:
152 errString = "CURAND_STATUS_LENGTH_NOT_MULTIPLE";
153 break;
154 case CURAND_STATUS_DOUBLE_PRECISION_REQUIRED:
155 errString = "CURAND_STATUS_DOUBLE_PRECISION_REQUIRED";
156 break;
157 case CURAND_STATUS_LAUNCH_FAILURE:
158 errString = "CURAND_STATUS_LAUNCH_FAILURE";
159 break;
160 case CURAND_STATUS_PREEXISTING_FAILURE:
161 errString = "CURAND_STATUS_PREEXISTING_FAILURE";
162 break;
163 case CURAND_STATUS_INITIALIZATION_FAILED:
164 errString = "CURAND_STATUS_INITIALIZATION_FAILED";
165 break;
166 case CURAND_STATUS_INTERNAL_ERROR:
167 errString = "CURAND_STATUS_INTERNAL_ERROR";
168 break;
169 case CURAND_STATUS_ARCH_MISMATCH:
170 errString = "CURAND_STATUS_ARCH_MISMATCH";
171 break;
172 }
173
174 Log::file() << "CudaRandom error: " << errString << std::endl;
175 UTIL_THROW("CudaRandom number generation failed.");
176 }
177 }
178
179}
void normal(DeviceArray< float > &data, float stddev, float mean=0.0)
Populate array on device with normal-distributed random floats.
Definition CudaRandom.cu:84
CudaRandom()
Constructor.
Definition CudaRandom.cu:13
long seed()
Returns value of random seed (private member variable seed_).
Definition CudaRandom.h:116
void uniform(DeviceArray< float > &data)
Populate array on device with random floats in (0, 1], uniform dist.
Definition CudaRandom.cu:54
virtual ~CudaRandom()
Destructor.
Definition CudaRandom.cu:27
void setSeed(unsigned long long seed)
Set value of random seed (private member variable seed_).
Definition CudaRandom.cu:35
Dynamic array on the GPU device with aligned data.
Definition rpg/System.h:32
int capacity() const
Return allocated capacity.
Data * cArray()
Return pointer to underlying C array.
static std::ostream & file()
Get log ostream by reference.
Definition Log.cpp:57
File containing preprocessor macros for error handling.
#define UTIL_CHECK(condition)
Assertion macro suitable for serial or parallel production code.
Definition global.h:68
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition global.h:51
PSCF package top-level namespace.
Definition param_pc.dox:1
Utility classes for scientific computation.