Skip to content

Misc

jmult_max

Calculate the maximum value of jmult.

Parameters: num_part (int): The number of particles. lmax (int): The maximum value of l.

Returns:

Type Description
int

The maximum value of jmult.

Source code in yasfpy/functions/misc.py
10
11
12
13
14
15
16
17
18
19
20
21
def jmult_max(num_part, lmax):
    """
    Calculate the maximum value of jmult.

    Parameters:
    num_part (int): The number of particles.
    lmax (int): The maximum value of l.

    Returns:
        (int): The maximum value of jmult.
    """
    return num_part * 2 * lmax * (lmax + 2)

multi2single_index

Converts the multi-index (j_s, tau, l, m) to a single index.

Parameters:

Name Type Description Default
j_s int

Particle index.

required
tau int

Polarization value (1 or 2).

required
l int

The value of l (between 0 and lmax).

required
m int

The value of m (between -l and l).

required
lmax int

Cutoff value for the field expansion.

required

Returns:

Type Description
int

The single index corresponding to the multi-index (j_s, tau, l, m).

Source code in yasfpy/functions/misc.py
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
def multi2single_index(j_s, tau, l, m, lmax):
    """
    Converts the multi-index (j_s, tau, l, m) to a single index.

    Args:
        j_s (int): Particle index.
        tau (int): Polarization value (1 or 2).
        l (int): The value of l (between 0 and lmax).
        m (int): The value of m (between -l and l).
        lmax (int): Cutoff value for the field expansion.

    Returns:
        (int): The single index corresponding to the multi-index (j_s, tau, l, m).
    """
    return (
        j_s * 2 * lmax * (lmax + 2)
        + (tau - 1) * lmax * (lmax + 2)
        + (l - 1) * (l + 1)
        + m
        + l
    )

single_index2multi

Convert a single index to multi-indices (j_s, tau, l, m) for spherical harmonics.

Parameters:

Name Type Description Default
idx int

The single index.

required
lmax int

Cutoff value for the field expansion.

required

Returns:

Name Type Description
j_s int

Particle index.

tau int

Polarization value (1 or 2).

l float

The value of l (between 0 and lmax).

m int

The value of m (between -l and l).

Source code in yasfpy/functions/misc.py
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
@nb.njit(nogil=True, fastmath=True)
def single_index2multi(idx, lmax):
    """
    Convert a single index to multi-indices (j_s, tau, l, m) for spherical harmonics.

    Args:
        idx (int): The single index.
        lmax (int): Cutoff value for the field expansion.

    Returns:
        j_s (int): Particle index.
        tau (int): Polarization value (1 or 2).
        l (float): The value of l (between 0 and lmax).
        m (int): The value of m (between -l and l).
    """
    j_s = idx // (2 * lmax * (lmax + 2))
    idx_new = idx % (2 * lmax * (lmax + 2))
    tau = idx_new // (lmax * (lmax + 2)) + 1
    idx_new = idx_new % (lmax * (lmax + 2))
    l = np.floor(np.sqrt(idx_new + 1))
    m = idx_new - (l * l + l - 1)
    return j_s, tau, l, m

transformation_coefficients

Calculate the transformation coefficients for spherical harmonics.

Parameters:

Name Type Description Default
pilm ndarray

Array of spherical harmonics.

required
taulm ndarray

Array of spherical harmonics.

required
tau int

Polarization state.

required
l int

Degree of the spherical harmonics.

required
m int

Order of the spherical harmonics.

required
pol int

Polarization state.

required
dagger bool

Whether to apply the dagger operation. Defaults to False.

False

Returns:

Type Description
float

The transformation coefficient.

Source code in yasfpy/functions/misc.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
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
def transformation_coefficients(pilm, taulm, tau, l, m, pol, dagger: bool = False):
    """
    Calculate the transformation coefficients for spherical harmonics.

    Args:
        pilm (ndarray): Array of spherical harmonics.
        taulm (ndarray): Array of spherical harmonics.
        tau (int): Polarization state.
        l (int): Degree of the spherical harmonics.
        m (int): Order of the spherical harmonics.
        pol (int): Polarization state.
        dagger (bool, optional): Whether to apply the dagger operation. Defaults to False.

    Returns:
        (float): The transformation coefficient.

    """
    ifac = 1j
    if dagger:
        ifac *= -1

    # Polarized light
    if np.any(np.equal(pol, [1, 2])):
        if tau == pol:
            spher_fun = taulm[l, np.abs(m)]
        else:
            spher_fun = m * pilm[l, np.abs(m)]

        return (
            -1
            / np.power(ifac, l + 1)
            / np.sqrt(2 * l * (l + 1))
            * (ifac * (pol == 1) + (pol == 2))
            * spher_fun
        )

    # Unpolarized light
    return (
        -1
        / np.power(ifac, l + 1)
        / np.sqrt(2 * l * (l + 1))
        * (ifac + 1)
        * (taulm[l, np.abs(m)] + m * pilm[l, np.abs(m)])
        / 2
    )

mutual_lookup

Calculate mutual lookup tables for scattering calculations.

Parameters:

Name Type Description Default
lmax int

The maximum degree of the spherical harmonics expansion.

required
positions_1 ndarray

The positions of the first set of particles.

required
positions_2 ndarray

The positions of the second set of particles.

required
refractive_index ndarray

The refractive indices of the particles.

required
derivatives bool

Whether to calculate the derivatives of the lookup tables. Defaults to False.

False
parallel bool

Whether to use parallel computation. Defaults to False.

False

Returns:

Name Type Description
spherical_bessel ndarray

The spherical Bessel functions.

spherical_hankel ndarray

The spherical Hankel functions.

e_j_dm_phi ndarray

The exponential term in the scattering calculation.

p_lm ndarray

The normalized Legendre polynomials.

e_r ndarray

The unit vectors in the radial direction.

e_theta ndarray

The unit vectors in the polar direction.

e_phi ndarray

The unit vectors in the azimuthal direction.

cosine_theta ndarray

The cosine of the polar angle.

sine_theta ndarray

The sine of the polar angle.

size_parameter ndarray

The size parameter of the particles.

spherical_hankel_derivative ndarray

The derivative of the spherical Hankel functions.

Source code in yasfpy/functions/misc.py
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
183
184
185
186
187
188
189
190
191
192
193
194
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
def mutual_lookup(
    lmax: int,
    positions_1: np.ndarray,
    positions_2: np.ndarray,
    refractive_index: np.ndarray,
    derivatives: bool = False,
    parallel: bool = False,
):
    """
    Calculate mutual lookup tables for scattering calculations.

    Args:
        lmax (int): The maximum degree of the spherical harmonics expansion.
        positions_1 (np.ndarray): The positions of the first set of particles.
        positions_2 (np.ndarray): The positions of the second set of particles.
        refractive_index (np.ndarray): The refractive indices of the particles.
        derivatives (bool, optional): Whether to calculate the derivatives of the lookup tables. Defaults to False.
        parallel (bool, optional): Whether to use parallel computation. Defaults to False.

    Returns:
        spherical_bessel (np.ndarray): The spherical Bessel functions.
        spherical_hankel (np.ndarray): The spherical Hankel functions.
        e_j_dm_phi (np.ndarray): The exponential term in the scattering calculation.
        p_lm (np.ndarray): The normalized Legendre polynomials.
        e_r (np.ndarray): The unit vectors in the radial direction.
        e_theta (np.ndarray): The unit vectors in the polar direction.
        e_phi (np.ndarray): The unit vectors in the azimuthal direction.
        cosine_theta (np.ndarray): The cosine of the polar angle.
        sine_theta (np.ndarray): The sine of the polar angle.
        size_parameter (np.ndarray): The size parameter of the particles.
        spherical_hankel_derivative (np.ndarray): The derivative of the spherical Hankel functions.
    """
    differences = positions_1[:, np.newaxis, :] - positions_2[np.newaxis, :, :]
    distances = np.sqrt(np.sum(differences**2, axis=2))
    distances = distances[:, :, np.newaxis]
    e_r = np.divide(
        differences, distances, out=np.zeros_like(differences), where=distances != 0
    )
    cosine_theta = e_r[:, :, 2]
    # cosine_theta = np.divide(
    #   differences[:, :, 2],
    #   distances,
    #   out = np.zeros_like(distances),
    #   where = distances != 0
    # )
    # correct possible rounding errors
    cosine_theta[cosine_theta < -1] = -1
    cosine_theta[cosine_theta > 1] = 1
    sine_theta = np.sqrt(1 - cosine_theta**2)
    phi = np.arctan2(differences[:, :, 1], differences[:, :, 0])
    e_theta = np.stack(
        [cosine_theta * np.cos(phi), cosine_theta * np.sin(phi), -sine_theta], axis=-1
    )
    e_phi = np.stack([-np.sin(phi), np.cos(phi), np.zeros_like(phi)], axis=-1)

    size_parameter = distances * np.array(refractive_index)[np.newaxis, np.newaxis, :]

    if parallel:
        from yasfpy.functions.cpu_numba import compute_lookup_tables

        spherical_bessel, spherical_hankel, e_j_dm_phi, p_lm = compute_lookup_tables(
            lmax, size_parameter, phi, cosine_theta
        )
    else:
        p_range = np.arange(2 * lmax + 1)
        p_range = p_range[:, np.newaxis, np.newaxis, np.newaxis]
        size_parameter_extended = size_parameter[np.newaxis, :, :, :]
        spherical_hankel = np.sqrt(
            np.divide(
                np.pi / 2,
                size_parameter_extended,
                out=np.zeros_like(size_parameter_extended),
                where=size_parameter_extended != 0,
            )
        ) * hankel1(p_range + 1 / 2, size_parameter_extended)
        spherical_bessel = spherical_jn(p_range, size_parameter_extended)

        if derivatives:
            spherical_hankel_lower = np.sqrt(
                np.divide(
                    np.pi / 2,
                    size_parameter_extended,
                    out=np.zeros_like(size_parameter_extended),
                    where=size_parameter_extended != 0,
                )
            ) * hankel1(-1 / 2, size_parameter_extended)
            spherical_hankel_lower = np.vstack(
                (spherical_hankel_lower, spherical_hankel[:-1, :, :, :])
            )
            spherical_hankel_derivative = (
                size_parameter_extended * spherical_hankel_lower
                - p_range * spherical_hankel
            )

            # p_range = np.arange(2 * lmax + 2) - 1
            # p_range = p_range[:, np.newaxis, np.newaxis, np.newaxis]
            # spherical_hankel = np.sqrt(np.divide(np.pi / 2, size_parameter_extended, out = np.zeros_like(size_parameter_extended), where = size_parameter_extended != 0)) * hankel1(p_range + 1/2, size_parameter_extended)
            # spherical_hankel_derivative = size_parameter_extended * spherical_hankel[:-1, :, :, :] - p_range[1:, :, :, :] * spherical_hankel[1:, :, :, :]

            p_lm = legendre_normalized_trigon(lmax, cosine_theta, sine_theta)
        else:
            spherical_hankel_derivative = None

            p_lm = np.zeros(
                (lmax + 1) * (2 * lmax + 1) * np.prod(size_parameter.shape[:2])
            ).reshape(((lmax + 1) * (2 * lmax + 1),) + size_parameter.shape[:2])
            for p in range(2 * lmax + 1):
                for absdm in range(p + 1):
                    cml = np.sqrt(
                        (2 * p + 1)
                        / 2
                        * np.prod(1 / np.arange(p - absdm + 1, p + absdm + 1))
                    )
                    p_lm[p * (p + 1) // 2 + absdm, :, :] = (
                        cml * np.power(-1.0, absdm) * lpmv(absdm, p, cosine_theta)
                    )

        phi = phi[np.newaxis, :, :]
        p_range = np.arange(-2 * lmax, 2 * lmax + 1)
        p_range = p_range[:, np.newaxis, np.newaxis]
        e_j_dm_phi = np.exp(1j * p_range * phi)

    return (
        spherical_bessel,
        spherical_hankel,
        e_j_dm_phi,
        p_lm,
        e_r,
        e_theta,
        e_phi,
        cosine_theta,
        sine_theta,
        size_parameter,
        spherical_hankel_derivative,
    )

generate_refractive_index_table

The function generate_refractive_index_table takes a list of URLs, retrieves data from each URL using the material_handler function, and returns a list of the retrieved data.

Parameters:

Name Type Description Default
urls list

A list of URLs representing different materials.

required

Returns:

Name Type Description
data list

A list of data. Each element in the list corresponds to a URL in the input list, and the data is obtained by calling the material_handler function on each URL.

Source code in yasfpy/functions/misc.py
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
def generate_refractive_index_table(urls: list) -> list:
    """The function `generate_refractive_index_table` takes a list of URLs, retrieves data from each
    URL using the `material_handler` function, and returns a list of the retrieved data.

    Args:
        urls (list): A list of URLs representing different materials.

    Returns:
        data (list): A list of data. Each element in the list corresponds to a URL in the input list,
            and the data is obtained by calling the `material_handler` function on each URL.

    """
    # data = [None] * len(urls)
    # for k, url in enumerate(urls):
    # data[k] = material_handler(url)
    data = []
    for url in urls:
        handle = Handler(url=url)
        data.append(dict(ref_idx=handle.nk))

    return data

Comments