ASCOT5
Loading...
Searching...
No Matches
dist_rho6D.c
Go to the documentation of this file.
1
5#include <stdio.h>
6#include <stdlib.h>
7#include <math.h>
8#include "../ascot5.h"
9#include "../consts.h"
10#include "../physlib.h"
11#include "../gctransform.h"
12#include "dist_rho6D.h"
13
17size_t dist_rho6D_index(int i_rho, int i_theta, int i_phi, int i_pr, int i_pphi,
18 int i_pz, int i_time, int i_q, size_t step_7,
19 size_t step_6, size_t step_5, size_t step_4,
20 size_t step_3, size_t step_2, size_t step_1) {
21 return (size_t)(i_rho) * step_7
22 + (size_t)(i_theta) * step_6
23 + (size_t)(i_phi) * step_5
24 + (size_t)(i_pr) * step_4
25 + (size_t)(i_pphi) * step_3
26 + (size_t)(i_pz) * step_2
27 + (size_t)(i_time) * step_1
28 + (size_t)(i_q);
29}
30
35
36 size_t n_q = (size_t)(data->n_q);
37 size_t n_time = (size_t)(data->n_time);
38 size_t n_pz = (size_t)(data->n_pz);
39 size_t n_pphi = (size_t)(data->n_pphi);
40 size_t n_pr = (size_t)(data->n_pr);
41 size_t n_phi = (size_t)(data->n_phi);
42 size_t n_theta = (size_t)(data->n_theta);
43 data->step_7 = n_q * n_time * n_pz * n_pphi * n_pr * n_phi * n_theta;
44 data->step_6 = n_q * n_time * n_pz * n_pphi * n_pr * n_phi;
45 data->step_5 = n_q * n_time * n_pz * n_pphi * n_pr;
46 data->step_4 = n_q * n_time * n_pz * n_pphi;
47 data->step_3 = n_q * n_time * n_pz;
48 data->step_2 = n_q * n_time;
49 data->step_1 = n_q;
50
51 data->histogram = calloc(data->step_7 * (size_t)data->n_rho, sizeof(real));
52 return data->histogram == NULL;
53}
54
59 free(data->histogram);
60}
61
68 GPU_MAP_TO_DEVICE(
69 data->histogram[0:data->n_rho*data->n_theta*data->n_phi*data->n_pr*data->n_pphi*data->n_pz*data->n_time*data->n_q]
70 )
71}
72
79 GPU_UPDATE_FROM_DEVICE(
80 data->histogram[0:data->n_rho*data->n_theta*data->n_phi*data->n_pr*data->n_pphi*data->n_pz*data->n_time*data->n_q]
81 )
82}
83
96 particle_simd_fo* p_i) {
97
98#ifdef GPU
99 size_t index;
100 real weight;
101#else
102 size_t index[NSIMD];
103 real weight[NSIMD];
104#endif
105
106 GPU_PARALLEL_LOOP_ALL_LEVELS
107 for(int i = 0; i < p_f->n_mrk; i++) {
108 if(p_f->running[i]) {
109
110 int i_rho = floor((p_f->rho[i] - dist->min_rho)
111 / ((dist->max_rho - dist->min_rho)/dist->n_rho));
112
113 real phi = fmod(p_f->phi[i], 2*CONST_PI);
114 if(phi < 0) {
115 phi = phi + 2*CONST_PI;
116 }
117 int i_phi = floor((phi - dist->min_phi)
118 / ((dist->max_phi - dist->min_phi)/dist->n_phi));
119
120 real theta = fmod(p_f->theta[i], 2*CONST_PI);
121 if(theta < 0) {
122 theta += 2*CONST_PI;
123 }
124 int i_theta = floor((theta - dist->min_theta)
125 / ((dist->max_theta - dist->min_theta)
126 / dist->n_theta));
127
128 int i_pr = floor((p_f->p_r[i] - dist->min_pr)
129 / ((dist->max_pr - dist->min_pr) / dist->n_pr));
130
131 int i_pphi = floor((p_f->p_phi[i] - dist->min_pphi)
132 / ((dist->max_pphi - dist->min_pphi)
133 / dist->n_pphi));
134
135 int i_pz = floor((p_f->p_z[i] - dist->min_pz)
136 / ((dist->max_pz - dist->min_pz) / dist->n_pz));
137
138 int i_time = floor((p_f->time[i] - dist->min_time)
139 / ((dist->max_time - dist->min_time) / dist->n_time));
140
141 int i_q = floor((p_f->charge[i]/CONST_E - dist->min_q)
142 / ((dist->max_q - dist->min_q) / dist->n_q));
143
144 if(i_rho >= 0 && i_rho <= dist->n_rho - 1 &&
145 i_theta >=0 && i_theta <= dist->n_theta -1 &&
146 i_phi >=0 && i_phi <= dist->n_phi - 1 &&
147 i_pr >= 0 && i_pr <= dist->n_pr - 1 &&
148 i_pphi >= 0 && i_pphi <= dist->n_pphi - 1 &&
149 i_pz >= 0 && i_pz <= dist->n_pz - 1 &&
150 i_time >= 0 && i_time <= dist->n_time - 1 &&
151 i_q >= 0 && i_q <= dist->n_q - 1 ) {
152#ifdef GPU
153 index = dist_rho6D_index(
154 i_rho, i_theta, i_phi, i_pr, i_pphi, i_pz,
155 i_time, i_q, dist->step_7, dist->step_6, dist->step_5,
156 dist->step_4, dist->step_3, dist->step_2, dist->step_1);
157 weight = p_f->weight[i] * (p_f->time[i] - p_i->time[i]);
158 GPU_ATOMIC
159 dist->histogram[index] += weight;
160#else
161 index[i] = dist_rho6D_index(
162 i_rho, i_theta, i_phi, i_pr, i_pphi, i_pz,
163 i_time, i_q, dist->step_7, dist->step_6, dist->step_5,
164 dist->step_4, dist->step_3, dist->step_2, dist->step_1);
165 weight[i] = p_f->weight[i] * (p_f->time[i] - p_i->time[i]);
166#endif
167 }
168 }
169 }
170#ifndef GPU
171 for(int i = 0; i < p_f->n_mrk; i++) {
172 if(p_f->running[i] && index[i] >= 0 &&
173 index[i] < dist->step_7 * dist->n_rho) {
174 GPU_ATOMIC
175 dist->histogram[index[i]] += weight[i];
176 }
177 }
178#endif
179}
180
193 particle_simd_gc* p_i) {
194 real phi[NSIMD];
195 real theta[NSIMD];
196
197 int i_rho[NSIMD];
198 int i_theta[NSIMD];
199 int i_phi[NSIMD];
200 int i_pr[NSIMD];
201 int i_pphi[NSIMD];
202 int i_pz[NSIMD];
203 int i_time[NSIMD];
204 int i_q[NSIMD];
205
206 int ok[NSIMD];
207 real weight[NSIMD];
208
209 #pragma omp simd
210 for(int i = 0; i < NSIMD; i++) {
211 if(p_f->running[i]) {
212
213 real pr, pphi, pz;
214 real B_dB[12] = {p_f->B_r[i],
215 p_f->B_r_dr[i],
216 p_f->B_r_dphi[i],
217 p_f->B_r_dz[i],
218 p_f->B_phi[i],
219 p_f->B_phi_dr[i],
220 p_f->B_phi_dphi[i],
221 p_f->B_phi_dz[i],
222 p_f->B_z[i],
223 p_f->B_z_dr[i],
224 p_f->B_z_dphi[i],
225 p_f->B_z_dz[i]};
226 gctransform_pparmuzeta2prpphipz(p_f->mass[i], p_f->charge[i], B_dB,
227 p_f->phi[i], p_f->ppar[i],
228 p_f->mu[i], p_f->zeta[i],
229 &pr, &pphi, &pz);
230
231 i_rho[i] = floor((p_f->rho[i] - dist->min_rho)
232 / ((dist->max_rho - dist->min_rho)/dist->n_rho));
233
234 phi[i] = fmod(p_f->phi[i], 2*CONST_PI);
235 if(phi[i] < 0) {
236 phi[i] = phi[i] + 2*CONST_PI;
237 }
238 i_phi[i] = floor((phi[i] - dist->min_phi)
239 / ((dist->max_phi - dist->min_phi)/dist->n_phi));
240
241 theta[i] = fmod(p_f->theta[i], 2*CONST_PI);
242 if(theta[i] < 0) {
243 theta[i] = theta[i] + 2*CONST_PI;
244 }
245 i_theta[i] = floor((theta[i] - dist->min_theta)
246 / ((dist->max_theta - dist->min_theta)
247 / dist->n_theta));
248
249 i_pr[i] = floor((pr - dist->min_pr)
250 / ((dist->max_pr - dist->min_pr) / dist->n_pr));
251
252 i_pphi[i] = floor((pphi - dist->min_pphi)
253 / ((dist->max_pphi - dist->min_pphi)
254 / dist->n_pphi));
255
256 i_pz[i] = floor((pz - dist->min_pz)
257 / ((dist->max_pz - dist->min_pz) / dist->n_pz));
258
259 i_time[i] = floor((p_f->time[i] - dist->min_time)
260 / ((dist->max_time - dist->min_time) / dist->n_time));
261
262 i_q[i] = floor((p_f->charge[i]/CONST_E - dist->min_q)
263 / ((dist->max_q - dist->min_q) / dist->n_q));
264
265 if(i_rho[i] >= 0 && i_rho[i] <= dist->n_rho - 1 &&
266 i_theta[i] >= 0 && i_theta[i] <= dist->n_theta -1 &&
267 i_phi[i] >= 0 && i_phi[i] <= dist->n_phi - 1 &&
268 i_pr[i] >= 0 && i_pr[i] <= dist->n_pr - 1 &&
269 i_pphi[i] >= 0 && i_pphi[i] <= dist->n_pphi - 1 &&
270 i_pz[i] >= 0 && i_pz[i] <= dist->n_pz - 1 &&
271 i_time[i] >= 0 && i_time[i] <= dist->n_time - 1 &&
272 i_q[i] >= 0 && i_q[i] <= dist->n_q - 1 ) {
273 ok[i] = 1;
274 weight[i] = p_f->weight[i] * (p_f->time[i] - p_i->time[i]);
275 }
276 else {
277 ok[i] = 0;
278 }
279 }
280 }
281
282 for(int i = 0; i < NSIMD; i++) {
283 if(p_f->running[i] && ok[i]) {
284 size_t index = dist_rho6D_index(
285 i_rho[i], i_theta[i], i_phi[i], i_pr[i], i_pphi[i], i_pz[i],
286 i_time[i], i_q[i], dist->step_7, dist->step_6, dist->step_5,
287 dist->step_4, dist->step_3, dist->step_2, dist->step_1);
288 #pragma omp atomic
289 dist->histogram[index] += weight[i];
290 }
291 }
292}
Main header file for ASCOT5.
double real
Definition ascot5.h:85
#define NSIMD
Number of particles simulated simultaneously in a particle group operations.
Definition ascot5.h:91
Header file containing physical and mathematical constants.
#define CONST_PI
pi
Definition consts.h:11
#define CONST_E
Elementary charge [C].
Definition consts.h:35
void dist_rho6D_onload(dist_rho6D_data *data)
Onload data back to the host.
Definition dist_rho6D.c:78
int dist_rho6D_init(dist_rho6D_data *data)
Initializes distribution data.
Definition dist_rho6D.c:34
void dist_rho6D_update_gc(dist_rho6D_data *dist, particle_simd_gc *p_f, particle_simd_gc *p_i)
Update the histogram from guiding-center particles.
Definition dist_rho6D.c:192
size_t dist_rho6D_index(int i_rho, int i_theta, int i_phi, int i_pr, int i_pphi, int i_pz, int i_time, int i_q, size_t step_7, size_t step_6, size_t step_5, size_t step_4, size_t step_3, size_t step_2, size_t step_1)
Internal function calculating the index in the histogram array.
Definition dist_rho6D.c:17
void dist_rho6D_free(dist_rho6D_data *data)
Free allocated resources.
Definition dist_rho6D.c:58
void dist_rho6D_update_fo(dist_rho6D_data *dist, particle_simd_fo *p_f, particle_simd_fo *p_i)
Update the histogram from full-orbit particles.
Definition dist_rho6D.c:95
void dist_rho6D_offload(dist_rho6D_data *data)
Offload data to the accelerator.
Definition dist_rho6D.c:67
Header file for dist_rho6D.c.
void gctransform_pparmuzeta2prpphipz(real mass, real charge, real *B_dB, real phi, real ppar, real mu, real zeta, real *pr, real *pphi, real *pz)
Transform particle ppar, mu, and zeta to momentum vector.
Header file for gctransform.c.
real fmod(real x, real y)
Compute the modulus of two real numbers.
Definition math.c:22
Header file for math.c.
Methods to evaluate elementary physical quantities.
Histogram parameters on target.
Definition dist_rho6D.h:15
Struct representing NSIMD particle markers.
Definition particle.h:210
integer * running
Definition particle.h:252
Struct representing NSIMD guiding center markers.
Definition particle.h:275