Skip to content

Simulation

Simulation

This class represents the simulation of YASF (Yet Another Scattering Framework). It contains methods for initializing the simulation, computing lookup tables, and calculating mie coefficients.

Initialize the Simulation object.

Parameters:

Name Type Description Default
parameters Parameters

The parameters for the simulation.

required
numerics Numerics

The numerics for the simulation.

required
Source code in yasfpy/simulation.py
36
37
38
39
40
41
42
43
44
45
46
47
48
49
def __init__(self, parameters: Parameters, numerics: Numerics):
    """
    Initialize the Simulation object.

    Args:
        parameters (Parameters): The parameters for the simulation.
        numerics (Numerics): The numerics for the simulation.
    """
    self.parameters = parameters
    self.numerics = numerics

    # self.log = log.infoing_logger(__name__)
    self.log = logging.getLogger(self.__class__.__module__)
    self.__setup()

parameters = parameters instance-attribute

numerics = numerics instance-attribute

log = logging.getLogger(self.__class__.__module__) instance-attribute

legacy_compute_lookup_particle_distances

The largest distance between two particles is divided into segments provided by Numerics.particle_distance_resolution. This array is then used as a lookup for the calculation of the spherical Hankel function.

Notes

This function has been ported from the Matlab Celes framework but is not used by YASF!

Source code in yasfpy/simulation.py
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
def legacy_compute_lookup_particle_distances(self):
    """
    The largest distance between two particles is divided into segments provided by `Numerics.particle_distance_resolution`.
    This array is then used as a lookup for the calculation of the spherical Hankel function.

    Notes
    -----
    This function has been ported from the Matlab Celes framework but is not used by YASF!
    """
    # add two zeros at the beginning to allow interpolation
    # also in the first segment
    step = self.numerics.particle_distance_resolution
    maxdist = (
        self.parameters.particles.max_particle_distance
        + 3 * self.numerics.particle_distance_resolution
    )
    self.lookup_particle_distances = np.concatenate(
        (np.array([0]), np.arange(0, maxdist + np.finfo(float).eps, step))
    )

legacy_compute_h3_table

Computes the spherical hankel function at the points calculated in Simulation.legacy_compute_lookup_particle_distances().

Attributes:

Name Type Description
h3_table ndarray

Lookup table of the spherical hankel function values at self.lookup_particle_distances

Notes

This function has been ported from the Matlab Celes framework but is not used by YASF!

Source code in yasfpy/simulation.py
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
def legacy_compute_h3_table(self):
    """
    Computes the spherical hankel function
    at the points calculated in `Simulation.legacy_compute_lookup_particle_distances()`.

    Attributes:
        h3_table (np.ndarray): Lookup table of the spherical hankel function values at `self.lookup_particle_distances`

    Notes:
        This function has been ported from the Matlab Celes framework but is not used by YASF!
    """
    self.h3_table = np.zeros(
        (
            2 * self.numerics.lmax + 1,
            self.lookup_particle_distances.shape[0],
            self.parameters.medium_refractive_index.shape[0],
        ),
        dtype=complex,
    )
    size_param = np.outer(self.lookup_particle_distances, self.parameters.k_medium)

    for p in range(2 * self.numerics.lmax + 1):
        self.h3_table[p, :, :] = spherical_jn(p, size_param) + 1j * spherical_yn(
            p, size_param
        )

__compute_idx_lookup

Creates a lookup table with the indices used in further calculations. The lookup table is created using compute_idx_lookups function from yasfpy.functions.cpu_numba.

Attributes:

Name Type Description
idx_lookup ndarray

Lookup table of the indices to iterate over large arrays.

Notes

This function utilizes Numba to optimize the computations.

Source code in yasfpy/simulation.py
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
def __compute_idx_lookup(self):
    """
    Creates a lookup table with the indices used in further calculations.
    The lookup table is created using `compute_idx_lookups` function from `yasfpy.functions.cpu_numba`.

    Attributes:
        idx_lookup (np.ndarray): Lookup table of the indices to iterate over large arrays.

    Notes:
        This function utilizes Numba to optimize the computations.
    """
    self.idx_lookup = compute_idx_lookups(
        self.numerics.lmax, self.parameters.particles.number
    )

__compute_lookups

Computes various lookup tables for each particle.

Attributes:

Name Type Description
sph_j ndarray

Spherical Bessel function lookup table calculated for pair-wise particle distances.

sph_h ndarray

Spherical Hankel function lookup table calculated for pair-wise particle distances.

plm ndarray

Associated Legendre polynomial lookup table calculated for the cosine value of the pairwise particle inclination angles.

e_j_dm_phi ndarray

Exponential function lookup table calculated for the pairwise particle azimuthal angles.

Notes

This function uses numba (https://numba.pydata.org/) under the hood to speed up the computations.

Source code in yasfpy/simulation.py
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
def __compute_lookups(self):
    """
    Computes various lookup tables for each particle.

    Attributes:
        sph_j (np.ndarray): Spherical Bessel function lookup table calculated for pair-wise particle distances.
        sph_h (np.ndarray): Spherical Hankel function lookup table calculated for pair-wise particle distances.
        plm (np.ndarray): Associated Legendre polynomial lookup table calculated for the cosine value of the pairwise particle inclination angles.
        e_j_dm_phi (np.ndarray): Exponential function lookup table calculated for the pairwise particle azimuthal angles.

    Notes:
        This function uses numba (https://numba.pydata.org/) under the hood to speed up the computations.
    """
    lookup_computation_time_start = time()
    # TODO: new, could be error prone and is not tested yet!
    self.sph_j, self.sph_h, self.e_j_dm_phi, self.plm = mutual_lookup(
        self.numerics.lmax,
        self.parameters.particles.position,
        self.parameters.particles.position,
        self.parameters.k_medium,
    )[:4]

    # lmax = self.numerics.lmax
    # particle_number = self.parameters.particles.number

    # dists = squareform(pdist(self.parameters.particles.position))
    # ct = np.divide(
    #   np.subtract.outer(
    #     self.parameters.particles.position[:, 2], self.parameters.particles.position[:, 2]),
    #   dists,
    #   out = np.zeros((particle_number, particle_number)),
    #   where = dists != 0)
    # phi = np.arctan2(
    #   np.subtract.outer(
    #     self.parameters.particles.position[:, 1], self.parameters.particles.position[:, 1]),
    #   np.subtract.outer(self.parameters.particles.position[:, 0], self.parameters.particles.position[:, 0]))

    # size_param = np.outer(dists.ravel(), self.parameters.k_medium).reshape(
    #   [particle_number, particle_number, self.parameters.k_medium.shape[0]])

    # self.sph_h = np.zeros((2 * lmax + 1, particle_number, particle_number, self.parameters.k_medium.shape[0]), dtype=complex)
    # self.sph_j = np.zeros_like(self.sph_h)
    # self.e_j_dm_phi = np.zeros((4 * lmax + 1, particle_number, particle_number), dtype=complex)
    # self.plm = np.zeros(((lmax + 1) * (2 * lmax + 1),
    #           particle_number, particle_number))

    # for p in range(2 * lmax + 1):
    #   self.sph_h[p, :, :, :] = np.sqrt(
    #     np.divide(
    #       np.pi / 2,
    #       size_param,
    #       out=np.zeros_like(size_param),
    #       where=size_param != 0)
    #   ) * hankel1(p + 1/2, size_param)
    #   self.sph_j[p, :, :, :] = spherical_jn(p, size_param)
    #   self.e_j_dm_phi[p, :, :] = np.exp(1j * (p - 2 * lmax) * phi)
    #   self.e_j_dm_phi[p + 2 * lmax, :, :] = np.exp(1j * p * phi)
    #   for absdm in range(p + 1):
    #     cml = np.sqrt((2 * p + 1) / 2 /
    #             np.prod(np.arange(p - absdm + 1, p + absdm + 1)))
    #     self.plm[p * (p + 1) // 2 + absdm, :, :] = cml * \
    #       np.power(-1.0, absdm) * lpmv(absdm, p, ct)

    # self.sph_h = np.nan_to_num(
    #   self.sph_h, nan=0) + np.isnan(self.sph_h) * 1

    lookup_computation_time_stop = time()
    self.log.info(
        "Computing lookup tables took %f s"
        % (lookup_computation_time_stop - lookup_computation_time_start)
    )

__setup

An internal setup function called upon object creation. The following functions are called:

Source code in yasfpy/simulation.py
184
185
186
187
188
189
190
191
192
193
def __setup(self):
    """
    An internal setup function called upon object creation.
    The following functions are called:

    - [__compute_idx_lookups][simulation.Simulation.__compute_idx_lookup]
    - [__compute_lookups][simulation.Simulation.__compute_lookups]
    """
    self.__compute_idx_lookup()
    self.__compute_lookups()

compute_mie_coefficients

Computes the mie coefficients for the unique pair of particle radius and the refractive index of the particle.

Attributes:

Name Type Description
mie_coefficients ndarray

Mie coefficients table

See Also

t_entry : T-Matrix entry function

Notes

Due to the four nested loops (particles, tau, l, and m), it could be rewritten using numba to speed the process up.

Source code in yasfpy/simulation.py
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
def compute_mie_coefficients(self):
    """
    Computes the mie coefficients for the unique pair
    of particle radius and the refractive index of the particle.

    Attributes:
        mie_coefficients (np.ndarray): Mie coefficients table

    See Also:
        [t_entry][functions.t_entry.t_entry] : T-Matrix entry function

    Notes:
        Due to the four nested loops (particles, tau, l, and m),
        it could be rewritten using `numba` to speed the process up.
    """
    self.mie_coefficients = np.zeros(
        (
            self.parameters.particles.num_unique_pairs,
            self.numerics.nmax,
            self.parameters.wavelength.shape[0],
        ),
        dtype=complex,
    )

    self.scatter_to_internal = np.zeros_like(self.mie_coefficients)

    for u_i in range(self.parameters.particles.num_unique_pairs):
        for tau in range(1, 3):
            for l in range(1, self.numerics.lmax + 1):
                for m in range(-l, l + 1):
                    jmult = multi2single_index(0, tau, l, m, self.numerics.lmax)
                    self.mie_coefficients[u_i, jmult, :] = t_entry(
                        tau=tau,
                        l=l,
                        k_medium=self.parameters.k_medium,
                        k_sphere=self.parameters.omega
                        * self.parameters.particles.unique_radius_index_pairs[
                            u_i, 1:
                        ],
                        radius=np.real(
                            self.parameters.particles.unique_radius_index_pairs[
                                u_i, 0
                            ]
                        ),
                    )

                    self.scatter_to_internal[u_i, jmult, :] = t_entry(
                        tau=tau,
                        l=l,
                        k_medium=self.parameters.k_medium,
                        k_sphere=self.parameters.omega
                        * self.parameters.particles.unique_radius_index_pairs[
                            u_i, 1:
                        ],
                        radius=np.real(
                            self.parameters.particles.unique_radius_index_pairs[
                                u_i, 0
                            ]
                        ),
                        field_type="ratio",
                    )

compute_initial_field_coefficients

Computes initial field coefficients \(a_{\\tau ,l,m}\) and \(b_{\\tau ,l,m}\). Depending on the beam_width, one of two functions is called:

Attributes:

Name Type Description
initial_field_coefficients ndarray

Initial field coefficients

Source code in yasfpy/simulation.py
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
def compute_initial_field_coefficients(self):
    r"""
    Computes initial field coefficients $a_{\\tau ,l,m}$ and $b_{\\tau ,l,m}$.
    Depending on the `beam_width`, one of two functions is called:

    - [__compute_initial_field_coefficients_wavebundle_normal_incidence][simulation.Simulation.__compute_initial_field_coefficients_wavebundle_normal_incidence], $\\text{beam width} \\in (0, \\infty)$
    - [__compute_initial_field_coefficients_planewave][simulation.Simulation.__compute_initial_field_coefficients_planewave], $\\text{beam width} = 0$ or $\\text{beam width} = \\infty$

    Attributes:
        initial_field_coefficients (np.ndarray): Initial field coefficients
    """
    self.log.info("compute initial field coefficients ...")

    if np.isfinite(self.parameters.initial_field.beam_width) and (
        self.parameters.initial_field.beam_width > 0
    ):
        self.log.info("\t Gaussian beam ...")
        if self.parameters.initial_field.normal_incidence:
            self.__compute_initial_field_coefficients_wavebundle_normal_incidence()
        else:
            self.log.error("\t this case is not implemented")
    else:
        self.log.info("\t plane wave ...")
        self.__compute_initial_field_coefficients_planewave()

    self.log.info("done")

compute_right_hand_side

Computes the right hand side \(T \\cdot a_I\) of the equation \(M \\cdot b = T \\cdot a_I\).

Attributes

right_hand_side : np.ndarray Right hand side of the equation \(M \\cdot b = T \\cdot a_I\)

Notes

For more information regarding the equation, please refer to the paper by Celes (https://arxiv.org/abs/1706.02145).

Source code in yasfpy/simulation.py
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
def compute_right_hand_side(self):
    r"""
    Computes the right hand side $T \\cdot a_I$ of the equation $M \\cdot b = T \\cdot a_I$.

    Attributes
    ----------
    right_hand_side : np.ndarray
        Right hand side of the equation $M \\cdot b = T \\cdot a_I$

    Notes
    -----
    For more information regarding the equation, please refer to the paper by Celes (https://arxiv.org/abs/1706.02145).
    """
    self.right_hand_side = (
        self.mie_coefficients[self.parameters.particles.single_unique_array_idx, :]
        * self.initial_field_coefficients
    )

__compute_initial_field_coefficients_planewave

The function computes the initial field coefficients for a plane wave based on given parameters and spherical coordinates.

Source code in yasfpy/simulation.py
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
def __compute_initial_field_coefficients_planewave(self):
    """The function computes the initial field coefficients for a plane wave based on given parameters
    and spherical coordinates.

    """
    lmax = self.numerics.lmax
    E0 = self.parameters.initial_field.amplitude
    k = self.parameters.k_medium

    beta = self.parameters.initial_field.polar_angle
    cb = np.cos(beta)
    sb = np.sin(beta)
    alpha = self.parameters.initial_field.azimuthal_angle

    # pi and tau symbols for transformation matrix B_dagger
    pilm, taulm = spherical_functions_trigon(lmax, beta)

    # cylindrical coordinates for relative particle positions
    relative_particle_positions = (
        self.parameters.particles.position
        - self.parameters.initial_field.focal_point
    )
    kvec = np.outer(np.array((sb * np.cos(alpha), sb * np.sin(alpha), cb)), k)
    eikr = np.exp(1j * np.matmul(relative_particle_positions, kvec))

    # clean up some memory?
    del (k, beta, cb, sb, kvec, relative_particle_positions)

    self.initial_field_coefficients = np.zeros(
        (
            self.parameters.particles.number,
            self.numerics.nmax,
            self.parameters.k_medium.size,
        ),
        dtype=complex,
    )
    for m in range(-lmax, lmax + 1):
        for tau in range(1, 3):
            for l in range(np.max([1, np.abs(m)]), lmax + 1):
                n = multi2single_index(0, tau, l, m, lmax)
                self.initial_field_coefficients[:, n, :] = (
                    4
                    * E0
                    * np.exp(-1j * m * alpha)
                    * eikr
                    * transformation_coefficients(
                        pilm,
                        taulm,
                        tau,
                        l,
                        m,
                        self.parameters.initial_field.pol,
                        dagger=True,
                    )
                )

__compute_initial_field_coefficients_wavebundle_normal_incidence

The function initializes the field coefficients for a wave bundle incident at normal incidence.

TODO

Implement this function using the celes function initial_field_coefficients_wavebundle_normal_incidence.m

Source code in yasfpy/simulation.py
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
def __compute_initial_field_coefficients_wavebundle_normal_incidence(self):
    """The function initializes the field coefficients for a wave bundle incident at normal incidence.

    TODO:
        Implement this function using the celes function [initial_field_coefficients_wavebundle_normal_incidence.m](https://github.com/disordered-photonics/celes/blob/master/src/initial/initial_field_coefficients_wavebundle_normal_incidence.m)
    """
    self.initial_field_coefficients = (
        np.zeros(
            (
                self.parameters.particles.number,
                self.numerics.nmax,
                self.parameters.k_medium.size,
            ),
            dtype=complex,
        )
        * np.nan
    )

coupling_matrix_multiply

Computes the coupling matrix wx based on the input parameters.

Parameters:

Name Type Description Default
x ndarray

An input array of shape (n,) or (n, m), where n is the number of particles and m is the number of features for each particle. This array represents the input data for which the coupling matrix needs to be computed.

required
idx int

An optional integer that specifies the index of a specific spherical harmonic mode. If idx is provided, the computation will only be performed for that specific mode. If idx is not provided or set to None, the computation will be performed for all spherical harmonic modes.

None

Returns:

Name Type Description
wx ndarray

An array of shape (n, m, p), where n is the number of particles, m is the number of features for each particle, and p is the number of wavelengths. It represents the coupling matrix wx.

Source code in yasfpy/simulation.py
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
def coupling_matrix_multiply(self, x: np.ndarray, idx: int = None):
    """Computes the coupling matrix `wx` based on the input parameters.

    Args:
        x (np.ndarray): An input array of shape (n,) or (n, m), where n is the number of particles and m is the number
            of features for each particle. This array represents the input data for which the coupling
            matrix needs to be computed.
        idx (int): An optional integer that specifies the index of a specific spherical harmonic mode. If `idx` is provided,
            the computation will only be performed for that specific mode. If `idx` is not provided or set to `None`,
            the computation will be performed for all spherical harmonic modes.

    Returns:
        wx (np.ndarray): An array of shape (n, m, p), where n is the number of particles, m is the number of features for each
            particle, and p is the number of wavelengths. It represents the coupling matrix `wx`.
    """
    self.log.debug("prepare particle coupling ... ")
    preparation_time = time()

    lmax = self.numerics.lmax
    particle_number = self.parameters.particles.number
    jmax = particle_number * 2 * lmax * (lmax + 2)
    wavelengths_size = self.parameters.k_medium.shape[0]
    translation_table = self.numerics.translation_ab5
    associated_legendre_lookup = self.plm
    spherical_hankel_lookup = self.sph_h
    e_j_dm_phi_loopup = self.e_j_dm_phi

    idx_lookup = self.idx_lookup

    if idx is not None:
        spherical_hankel_lookup = spherical_hankel_lookup[:, :, :, idx]
        spherical_hankel_lookup = np.copy(
            spherical_hankel_lookup[:, :, :, np.newaxis]
        )
        wavelengths_size = 1

    self.log.debug("\t Starting Wx computation")
    if self.numerics.gpu:
        wx_real = np.zeros(x.shape + (wavelengths_size,), dtype=float)
        wx_imag = np.zeros_like(wx_real)

        idx_device = cuda.to_device(idx_lookup)
        x_device = cuda.to_device(x)
        wx_real_device = cuda.to_device(wx_real)
        wx_imag_device = cuda.to_device(wx_imag)
        translation_device = cuda.to_device(translation_table)
        associated_legendre_device = cuda.to_device(associated_legendre_lookup)
        spherical_hankel_device = cuda.to_device(spherical_hankel_lookup)
        e_j_dm_phi_device = cuda.to_device(e_j_dm_phi_loopup)

        threads_per_block = (16, 16, 2)
        blocks_per_grid_x = ceil(jmax / threads_per_block[0])
        blocks_per_grid_y = ceil(jmax / threads_per_block[1])
        blocks_per_grid_z = ceil(wavelengths_size / threads_per_block[2])
        blocks_per_grid = (blocks_per_grid_x, blocks_per_grid_y, blocks_per_grid_z)

        coupling_matrix_time = time()
        particle_interaction_gpu[blocks_per_grid, threads_per_block](
            lmax,
            particle_number,
            idx_device,
            x_device,
            wx_real_device,
            wx_imag_device,
            translation_device,
            associated_legendre_device,
            spherical_hankel_device,
            e_j_dm_phi_device,
        )
        wx_real = wx_real_device.copy_to_host()
        wx_imag = wx_imag_device.copy_to_host()
        wx = wx_real + 1j * wx_imag
        # particle_interaction.parallel_diagnostics(level=4)
        time_end = time()
        self.log.debug(
            "\t Time taken for preparation: %f"
            % (coupling_matrix_time - preparation_time)
        )
        self.log.debug(
            "\t Time taken for coupling matrix: %f"
            % (time_end - coupling_matrix_time)
        )
    else:
        # from numba_progress import ProgressBar
        # num_iterations = jmax * jmax * wavelengths
        # progress = ProgressBar(total=num_iterations)
        # progress = None
        wx = particle_interaction(
            lmax,
            particle_number,
            idx_lookup,
            x,
            translation_table,
            associated_legendre_lookup,
            spherical_hankel_lookup,
            e_j_dm_phi_loopup,
        )
        time_end = time()
        self.log.debug(
            "\t Time taken for coupling matrix: %f" % (time_end - preparation_time)
        )

    if idx is not None:
        wx = np.squeeze(wx)

    return wx

master_matrix_multiply

Applies a T-matrix to a given value and returns the result.

Parameters:

Name Type Description Default
value ndarray

The input value for the matrix multiplication operation.

required
idx int

The index of the matrix to be multiplied.

required

Returns:

Name Type Description
mx ndarray

The result of the matrix multiplication operation.

Source code in yasfpy/simulation.py
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
def master_matrix_multiply(self, value: np.ndarray, idx: int):
    """Applies a T-matrix to a given value and returns the result.

    Args:
        value (np.ndarray): The input value for the matrix multiplication operation.
        idx (int): The index of the matrix to be multiplied.

    Returns:
        mx (np.ndarray): The result of the matrix multiplication operation.

    """
    wx = self.coupling_matrix_multiply(value, idx)

    self.log.debug("apply T-matrix ...")
    t_matrix_start = time()

    twx = (
        self.mie_coefficients[
            self.parameters.particles.single_unique_array_idx, :, idx
        ].ravel(order="C")
        * wx
    )
    mx = value - twx

    t_matrix_stop = time()
    self.log.debug(f"\t done in {t_matrix_stop - t_matrix_start} seconds.")

    return mx

compute_scattered_field_coefficients

The function computes the scattered field coefficients using a linear operator and a solver.

Parameters:

Name Type Description Default
guess ndarray

Optional. The initial guess for the solution of the linear system. If no guess is provided, the right_hand_side variable is used as the initial guess.

None
Source code in yasfpy/simulation.py
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
def compute_scattered_field_coefficients(self, guess: np.ndarray = None):
    """The function computes the scattered field coefficients using a linear operator and a solver.

    Args:
        guess (np.ndarray): Optional. The initial guess for the solution of the linear system. If no guess is provided,
            the `right_hand_side` variable is used as the initial guess.

    """
    self.log.info("compute scattered field coefficients ...")
    jmax = self.parameters.particles.number * self.numerics.nmax
    self.scattered_field_coefficients = np.zeros_like(
        self.initial_field_coefficients
    )
    self.scattered_field_err_codes = np.zeros(self.parameters.wavelengths_number)
    if guess is None:
        guess = self.right_hand_side
    # TODO: Look into performing this loop in parallel
    for w in range(self.parameters.wavelengths_number):
        A = LinearOperator(
            shape=(jmax, jmax), matvec=lambda x: self.master_matrix_multiply(x, w)
        )
        b = self.right_hand_side[:, :, w].ravel()
        x0 = guess[:, :, w].ravel()
        self.log.info(
            "Solver run %d/%d" % (w + 1, self.parameters.wavelengths_number)
        )
        x, err_code = self.numerics.solver.run(A, b, x0)
        self.scattered_field_coefficients[:, :, w] = x.reshape(
            self.right_hand_side.shape[:2]
        )
        self.scattered_field_err_codes[w] = err_code

compute_fields

The function compute_fields calculates the field at given sampling points using either CPU or GPU computation.

Parameters:

Name Type Description Default
sampling_points ndarray

The numpy array that represents the coordinates of the sampling points. It should have a shape of (n, 3), where n is the number of sampling points and each row represents the (x, y, z) coordinates of a point.

required
Source code in yasfpy/simulation.py
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
def compute_fields(self, sampling_points: np.ndarray):
    """The function `compute_fields` calculates the field at given sampling points using either CPU or
    GPU computation.

    Args:
        sampling_points (np.ndarray): The numpy array that represents the coordinates of the sampling points.
            It should have a shape of `(n, 3)`, where `n` is the number of sampling points and each row
            represents the `(x, y, z)` coordinates of a point.
    """
    if sampling_points.shape[0] < 1:
        self.log.error("Number of sampling points must be bigger than zero!")
        return
    if sampling_points.shape[1] != 3:
        self.log.error("The points have to have three coordinates (x,y,z)!")
        return

    # scatter_to_internal_table = np.sum((self.parameters.particles.position[:, np.newaxis, :] - sampling_points[np.newaxis, :, :])**2, axis = 2)
    # scatter_to_internal_table = scatter_to_internal_table < self.parameters.particles.r[:, np.newaxis]**2

    self.log.info("Computing mutual lookup")
    lookup_computation_time_start = time()
    (
        _,
        sph_h,
        e_j_dm_phi,
        p_lm,
        e_r,
        e_theta,
        e_phi,
        cosine_theta,
        sine_theta,
        size_parameter,
        sph_h_derivative,
    ) = mutual_lookup(
        self.numerics.lmax,
        self.parameters.particles.position,
        sampling_points,
        self.parameters.k_medium,
        derivatives=True,
        parallel=False,
    )
    lookup_computation_time_stop = time()
    self.log.info(
        "Computing lookup tables took %f s",
        lookup_computation_time_stop - lookup_computation_time_start,
    )
    pi_lm, tau_lm = spherical_functions_trigon(
        self.numerics.lmax, cosine_theta, sine_theta
    )

    self.log.info("Computing field...")
    field_time_start = time()
    self.sampling_points = sampling_points
    if not self.numerics.gpu:
        self.log.info("\t...using GPU")
        field_real = np.zeros(
            (self.parameters.k_medium.size, sampling_points.shape[0], 3),
            dtype=float,
        )
        field_imag = np.zeros_like(field_real)

        idx_device = cuda.to_device(self.idx_lookup)
        size_parameter_device = cuda.to_device(np.ascontiguousarray(size_parameter))
        sph_h_device = cuda.to_device(np.ascontiguousarray(sph_h))
        sph_h_derivative_device = cuda.to_device(
            np.ascontiguousarray(sph_h_derivative)
        )
        e_j_dm_phi_device = cuda.to_device(np.ascontiguousarray(e_j_dm_phi))
        p_lm_device = cuda.to_device(np.ascontiguousarray(p_lm))
        pi_lm_device = cuda.to_device(np.ascontiguousarray(pi_lm))
        tau_lm_device = cuda.to_device(np.ascontiguousarray(tau_lm))
        e_r_device = cuda.to_device(np.ascontiguousarray(e_r))
        e_theta_device = cuda.to_device(np.ascontiguousarray(e_theta))
        e_phi_device = cuda.to_device(np.ascontiguousarray(e_phi))
        sfc_device = cuda.to_device(
            np.ascontiguousarray(self.scattered_field_coefficients)
        )

        field_real_device = cuda.to_device(field_real)
        field_imag_device = cuda.to_device(field_imag)

        threads_per_block = (16, 16, 2)
        blocks_per_grid = (
            sampling_points.shape[0],
            sph_h.shape[1] * 2 * self.numerics.lmax * (self.numerics.lmax + 2),
            self.parameters.k_medium.size,
        )
        # blocks_per_grid = tuple(
        #     [
        #         ceil(blocks_per_grid[i] / threads_per_block[i])
        #         for i in range(len(threads_per_block))
        #     ]
        # )
        blocks_per_grid = tuple(
            ceil(blocks_per_grid[i] / threads_per_block[i])
            for i in range(len(threads_per_block))
        )

        compute_field_gpu[blocks_per_grid, threads_per_block](
            self.numerics.lmax,
            idx_device,
            size_parameter_device,
            sph_h_device,
            sph_h_derivative_device,
            e_j_dm_phi_device,
            p_lm_device,
            pi_lm_device,
            tau_lm_device,
            e_r_device,
            e_theta_device,
            e_phi_device,
            sfc_device,
            field_real_device,
            field_imag_device,
        )

        field_real = field_real_device.copy_to_host()
        field_imag = field_imag_device.copy_to_host()
        self.scattered_field = field_real + 1j * field_imag

    else:
        self.log.info("\t...using CPU")
        self.scattered_field = compute_field(
            self.numerics.lmax,
            self.idx_lookup,
            size_parameter,
            sph_h,
            sph_h_derivative,
            e_j_dm_phi,
            p_lm,
            pi_lm,
            tau_lm,
            e_r,
            e_theta,
            e_phi,
            scattered_field_coefficients=self.scattered_field_coefficients,
        )

    field_time_stop = time()
    self.log.info(
        f"\t Time taken for field calculation: {field_time_stop - field_time_start}"
    )

Comments