Skip to content

plasma_profiles

PlasmaProfile

Plasma profile class. Initiates the electron density and electron temperature profiles and handles the required physics variables.

Source code in process/models/physics/plasma_profiles.py
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 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
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
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
253
254
255
256
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
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
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
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
class PlasmaProfile:
    """Plasma profile class. Initiates the electron density and electron temperature profiles and
    handles the required physics variables.
    """

    def __init__(self):
        """
        Initialize the PlasmaProfile class.

        Args:
            profile_size (int): The size of the plasma profile.
            outfile (str): The output file path.
            neprofile (NeProfile): An instance of the NeProfile class.
            teprofile (TeProfile): An instance of the TeProfile class.
        """
        # Default profile_size = 501, but it's possible to experiment with this value.
        self.profile_size = 501
        physics_variables.n_plasma_profile_elements = self.profile_size
        self.outfile = constants.NOUT
        self.neprofile = profiles.NeProfile(self.profile_size)
        self.teprofile = profiles.TeProfile(self.profile_size)

    def run(self):
        """Subroutine to execute PlasmaProfile functions.

        This method calls the parameterise_plasma() method to initialize the plasma profiles.
        """
        self.parameterise_plasma()

    def parameterise_plasma(self):
        """This routine initializes the density and temperature
        profile averages and peak values, given the main
        parameters describing these profiles.

        References:
        T&M/PKNIGHT/LOGBOOK24, pp.4-7
        """

        #  Volume-averaged ion temperature
        #  (input value used directly if f_temp_plasma_ion_electron=0.0)

        if physics_variables.f_temp_plasma_ion_electron > 0.0e0:
            physics_variables.temp_plasma_ion_vol_avg_kev = (
                physics_variables.f_temp_plasma_ion_electron
                * physics_variables.temp_plasma_electron_vol_avg_kev
            )

        # Parabolic profile case
        if physics_variables.i_plasma_pedestal == 0:
            self.parabolic_paramterisation()
            self.calculate_profile_factors()
            self.calculate_parabolic_profile_factors()
        # Pedestal profile case
        else:
            self.pedestal_parameterisation()
            self.calculate_profile_factors()

    def parabolic_paramterisation(self):
        """Parameterise plasma profiles in the case where i_plasma_pedestal == 0.

        This routine calculates the parameterization of plasma profiles in the case where i_plasma_pedestal=0.
        It sets the necessary physics variables for the parabolic profile case.
        """
        # Reset pedestal values to agree with original parabolic profiles
        if (
            physics_variables.radius_plasma_pedestal_temp_norm != 1.0
            or physics_variables.radius_plasma_pedestal_density_norm != 1.0
            or physics_variables.temp_plasma_pedestal_kev != 0.0
            or physics_variables.temp_plasma_separatrix_kev != 0.0
            or physics_variables.nd_plasma_pedestal_electron != 0.0
            or physics_variables.nd_plasma_separatrix_electron != 0.0
            or physics_variables.tbeta != 2.0
        ):
            logger.error(
                "Parabolic plasma profiles is used for an L-Mode plasma, "
                "but the physics variables do not describe an L-Mode plasma. "
                "'radius_plasma_pedestal_temp_norm', 'radius_plasma_pedestal_density_norm', 'temp_plasma_pedestal_kev', 'temp_plasma_separatrix_kev', 'nd_plasma_pedestal_electron', 'nd_plasma_separatrix_electron', "
                "and 'tbeta' have all been reset to L-Mode appropriate values"
            )

            physics_variables.radius_plasma_pedestal_temp_norm = 1.0e0
            physics_variables.radius_plasma_pedestal_density_norm = 1.0e0
            physics_variables.temp_plasma_pedestal_kev = 0.0e0
            physics_variables.temp_plasma_separatrix_kev = 0.0e0
            physics_variables.nd_plasma_pedestal_electron = 0.0e0
            physics_variables.nd_plasma_separatrix_electron = 0.0e0
            physics_variables.tbeta = 2.0e0

        # Re-caluclate core and profile values
        self.teprofile.run()
        self.neprofile.run()

        #  Profile factor; ratio of density-weighted to volume-averaged
        #  temperature

        physics_variables.f_temp_plasma_electron_density_vol_avg = (
            (1.0e0 + physics_variables.alphan)
            * (1.0e0 + physics_variables.alphat)
            / (1.0e0 + physics_variables.alphan + physics_variables.alphat)
        )

        # Line averaged electron density (IPDG89)
        # Taken by integrating the parabolic profile over rho in the bounds of 0 and 1 and dividng by the width of the integration bounds

        physics_variables.nd_plasma_electron_line = (
            physics_variables.nd_plasma_electrons_vol_avg
            * (1.0 + physics_variables.alphan)
            * (sp.special.gamma(0.5) / 2.0)
            * sp.special.gamma(physics_variables.alphan + 1.0)
            / sp.special.gamma(physics_variables.alphan + 1.5)
        )

        #  Density-weighted temperatures

        physics_variables.temp_plasma_electron_density_weighted_kev = (
            physics_variables.temp_plasma_electron_vol_avg_kev
            * physics_variables.f_temp_plasma_electron_density_vol_avg
        )
        physics_variables.temp_plasma_ion_density_weighted_kev = (
            physics_variables.temp_plasma_ion_vol_avg_kev
            * physics_variables.f_temp_plasma_electron_density_vol_avg
        )

        #  Central values for temperature (keV) and density (m**-3)

        physics_variables.temp_plasma_electron_on_axis_kev = (
            physics_variables.temp_plasma_electron_vol_avg_kev
            * (1.0 + physics_variables.alphat)
        )
        physics_variables.temp_plasma_ion_on_axis_kev = (
            physics_variables.temp_plasma_ion_vol_avg_kev
            * (1.0 + physics_variables.alphat)
        )

        physics_variables.nd_plasma_electron_on_axis = (
            physics_variables.nd_plasma_electrons_vol_avg
            * (1.0 + physics_variables.alphan)
        )
        physics_variables.nd_plasma_ions_on_axis = (
            physics_variables.nd_plasma_ions_total_vol_avg
            * (1.0 + physics_variables.alphan)
        )

    def pedestal_parameterisation(self):
        """Instance temperature and density profiles then integrate them, setting physics variables temp_plasma_electron_density_weighted_kev and temp_plasma_ion_density_weighted_kev.

        This routine instances temperature and density profiles and integrates them to calculate the values of the physics variables `temp_plasma_electron_density_weighted_kev` and `temp_plasma_ion_density_weighted_kev`.
        """
        #  Run TeProfile and NeProfile class methods:
        #  Re-caluclate core and profile values

        self.teprofile.run()
        self.neprofile.run()

        #  Perform integrations to calculate ratio of density-weighted
        #  to volume-averaged temperature, etc.
        #  Density-weighted temperature = integral(n.T dV) / integral(n dV)
        #  which is approximately equal to the ratio
        #  integral(rho.n(rho).T(rho) drho) / integral(rho.n(rho) drho)

        drho = self.neprofile.profile_dx
        dens = self.neprofile.profile_y
        temp = self.teprofile.profile_y
        rho = self.neprofile.profile_x

        arg1 = rho * dens * temp
        arg2 = rho * dens

        integ1 = sp.integrate.simpson(arg1, x=rho, dx=drho)
        integ2 = sp.integrate.simpson(arg2, x=rho, dx=drho)

        physics_variables.integ1 = integ1
        physics_variables.integ2 = integ2
        #  Density-weighted temperatures
        physics_variables.temp_plasma_electron_density_weighted_kev = integ1 / integ2
        physics_variables.temp_plasma_ion_density_weighted_kev = (
            physics_variables.temp_plasma_ion_vol_avg_kev
            / physics_variables.temp_plasma_electron_vol_avg_kev
            * physics_variables.temp_plasma_electron_density_weighted_kev
        )
        #  Profile factor; ratio of density-weighted to volume-averaged
        #  temperature

        physics_variables.f_temp_plasma_electron_density_vol_avg = (
            physics_variables.temp_plasma_electron_density_weighted_kev
            / physics_variables.temp_plasma_electron_vol_avg_kev
        )

        #  Line-averaged electron density
        #  = integral(n(rho).drho)

        physics_variables.nd_plasma_electron_line = self.neprofile.profile_integ

        #  Scrape-off density / volume averaged density
        #  (Input value is used if i_plasma_pedestal = 0)

        divertor_variables.prn1 = max(
            0.01e0,
            physics_variables.nd_plasma_separatrix_electron
            / physics_variables.nd_plasma_electrons_vol_avg,
        )  # Preventing division by zero later

    def calculate_profile_factors(self):
        """Calculate and set the central pressure (pres_plasma_thermal_on_axis) using the ideal gas law and the pressure profile index (alphap).

        This method calculates the central pressure (pres_plasma_thermal_on_axis) using the ideal gas law and the pressure profile index (alphap).
        It sets the value of the physics variable `pres_plasma_thermal_on_axis`.
        """

        #  Central pressure (Pa), from ideal gas law : p = nkT

        physics_variables.pres_plasma_thermal_on_axis = (
            physics_variables.nd_plasma_electron_on_axis
            * physics_variables.temp_plasma_electron_on_axis_kev
            + physics_variables.nd_plasma_ions_on_axis
            * physics_variables.temp_plasma_ion_on_axis_kev
        ) * constants.KILOELECTRON_VOLT

        # Electron pressure profile (Pa)
        physics_variables.pres_plasma_electron_profile = self.neprofile.profile_y * (
            self.teprofile.profile_y * constants.KILOELECTRON_VOLT
        )

        # Total ion pressure profile (Pa)
        physics_variables.pres_plasma_ion_total_profile = (
            physics_variables.nd_plasma_ions_total_vol_avg
            * (self.neprofile.profile_y / physics_variables.nd_plasma_electrons_vol_avg)
        ) * (
            self.teprofile.profile_y
            * constants.KILOELECTRON_VOLT
            * physics_variables.f_temp_plasma_ion_electron
        )

        # Total pressure profile (Pa)
        physics_variables.pres_plasma_thermal_total_profile = (
            physics_variables.pres_plasma_electron_profile
            + physics_variables.pres_plasma_ion_total_profile
        )

        # Fuel ion pressure profile (Pa)
        physics_variables.pres_plasma_fuel_profile = (
            physics_variables.nd_plasma_fuel_ions_vol_avg
            * (self.neprofile.profile_y / physics_variables.nd_plasma_electrons_vol_avg)
        ) * (
            self.teprofile.profile_y
            * constants.KILOELECTRON_VOLT
            * physics_variables.f_temp_plasma_ion_electron
        )

        #  Pressure profile index (only true for a parabolic profile)
        #  N.B. pres_plasma_thermal_on_axis is NOT equal to <p> * (1 + alphap), but p(rho) = n(rho)*T(rho)
        #  and <p> = <n>.T_n where <...> denotes volume-averages and T_n is the
        #  density-weighted temperature

        physics_variables.alphap = physics_variables.alphan + physics_variables.alphat

        # Calculate the volume averaged plasma thermal pressure from the density-weighted temperatures
        # Density-weighted temperatures are used as <nT> != <n>*<T>
        physics_variables.pres_plasma_thermal_vol_avg = (
            physics_variables.nd_plasma_electrons_vol_avg
            * physics_variables.temp_plasma_electron_density_weighted_kev
            + physics_variables.nd_plasma_ions_total_vol_avg
            * physics_variables.temp_plasma_ion_density_weighted_kev
        ) * constants.KILOELECTRON_VOLT

        # Central plasma current density (A/m^2)
        # Assumes a parabolic profile for the current density
        physics_variables.j_plasma_on_axis = (
            (physics_variables.plasma_current)
            * 2
            / (
                sp.special.beta(0.5, physics_variables.alphaj + 1)
                * physics_variables.a_plasma_poloidal
            )
        )

    @staticmethod
    def calculate_parabolic_profile_factors():
        """Calculate the gradient information for i_plasma_pedestal = 0.

        This function calculates the gradient information for the plasma profiles at the pedestal region
        when the value of i_plasma_pedestal is 0. It is used by the stellarator routines.

        The function uses analytical parametric formulas to calculate the gradient information.
        The maximum normalized radius (rho_max) is obtained by equating the second derivative to zero.
        """
        if physics_variables.i_plasma_pedestal == 0:
            if physics_variables.alphat > 1.0:
                # Rho (normalized radius), where temperature derivative is largest
                rho_te_max = 1.0 / np.sqrt(-1.0 + 2.0 * physics_variables.alphat)
                dtdrho_max = (
                    -(2.0**physics_variables.alphat)
                    * (-1.0 + physics_variables.alphat)
                    ** (-1.0 + physics_variables.alphat)
                    * physics_variables.alphat
                    * (-1.0 + 2.0 * physics_variables.alphat)
                    ** (0.5e0 - physics_variables.alphat)
                    * physics_variables.temp_plasma_electron_on_axis_kev
                )
                te_max = (
                    physics_variables.temp_plasma_electron_on_axis_kev
                    * (1 - rho_te_max**2) ** physics_variables.alphat
                )

            elif physics_variables.alphat <= 1.0 and physics_variables.alphat > 0.0:
                # This makes the profiles very 'boxy'
                # The gradient diverges here at the edge so define some 'wrong' value of 0.9
                # to approximate the gradient
                rho_te_max = 0.9
                dtdrho_max = (
                    -2.0
                    * physics_variables.alphat
                    * rho_te_max
                    * (1 - rho_te_max**2) ** (-1.0 + physics_variables.alphat)
                    * physics_variables.temp_plasma_electron_on_axis_kev
                )
                te_max = (
                    physics_variables.temp_plasma_electron_on_axis_kev
                    * (1 - rho_te_max**2) ** physics_variables.alphat
                )
            else:
                raise ProcessValueError(
                    f"alphat is negative: {physics_variables.alphat}"
                )

            # Same for density
            if physics_variables.alphan > 1.0:
                rho_ne_max = 1.0 / np.sqrt(-1.0 + 2.0 * physics_variables.alphan)
                dndrho_max = (
                    -(2.0**physics_variables.alphan)
                    * (-1.0 + physics_variables.alphan)
                    ** (-1.0 + physics_variables.alphan)
                    * physics_variables.alphan
                    * (-1.0 + 2.0 * physics_variables.alphan)
                    ** (0.5 - physics_variables.alphan)
                    * physics_variables.nd_plasma_electron_on_axis
                )
                ne_max = (
                    physics_variables.nd_plasma_electron_on_axis
                    * (1e0 - rho_ne_max**2) ** physics_variables.alphan
                )
            elif physics_variables.alphan <= 1.0 and physics_variables.alphan > 0.0:
                # This makes the profiles very 'boxy'
                # The gradient diverges here at the edge so define some 'wrong' value of 0.9
                # to approximate the gradient
                rho_ne_max = 0.9
                dndrho_max = (
                    -2.0
                    * physics_variables.alphan
                    * rho_ne_max
                    * (1 - rho_ne_max**2) ** (-1.0 + physics_variables.alphan)
                    * physics_variables.nd_plasma_electron_on_axis
                )
                ne_max = (
                    physics_variables.nd_plasma_electron_on_axis
                    * (1 - rho_ne_max**2) ** physics_variables.alphan
                )
            else:
                raise ProcessValueError(
                    f"alphan is negative: {physics_variables.alphan}"
                )

            # set normalized gradient length
            # te at rho_te_max
            physics_variables.gradient_length_te = (
                -dtdrho_max * physics_variables.rminor * rho_te_max / te_max
            )
            # same for density:
            physics_variables.gradient_length_ne = (
                -dndrho_max * physics_variables.rminor * rho_ne_max / ne_max
            )

profile_size = 501 instance-attribute

outfile = constants.NOUT instance-attribute

neprofile = profiles.NeProfile(self.profile_size) instance-attribute

teprofile = profiles.TeProfile(self.profile_size) instance-attribute

run()

Subroutine to execute PlasmaProfile functions.

This method calls the parameterise_plasma() method to initialize the plasma profiles.

Source code in process/models/physics/plasma_profiles.py
36
37
38
39
40
41
def run(self):
    """Subroutine to execute PlasmaProfile functions.

    This method calls the parameterise_plasma() method to initialize the plasma profiles.
    """
    self.parameterise_plasma()

parameterise_plasma()

This routine initializes the density and temperature profile averages and peak values, given the main parameters describing these profiles.

References: T&M/PKNIGHT/LOGBOOK24, pp.4-7

Source code in process/models/physics/plasma_profiles.py
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
def parameterise_plasma(self):
    """This routine initializes the density and temperature
    profile averages and peak values, given the main
    parameters describing these profiles.

    References:
    T&M/PKNIGHT/LOGBOOK24, pp.4-7
    """

    #  Volume-averaged ion temperature
    #  (input value used directly if f_temp_plasma_ion_electron=0.0)

    if physics_variables.f_temp_plasma_ion_electron > 0.0e0:
        physics_variables.temp_plasma_ion_vol_avg_kev = (
            physics_variables.f_temp_plasma_ion_electron
            * physics_variables.temp_plasma_electron_vol_avg_kev
        )

    # Parabolic profile case
    if physics_variables.i_plasma_pedestal == 0:
        self.parabolic_paramterisation()
        self.calculate_profile_factors()
        self.calculate_parabolic_profile_factors()
    # Pedestal profile case
    else:
        self.pedestal_parameterisation()
        self.calculate_profile_factors()

parabolic_paramterisation()

Parameterise plasma profiles in the case where i_plasma_pedestal == 0.

This routine calculates the parameterization of plasma profiles in the case where i_plasma_pedestal=0. It sets the necessary physics variables for the parabolic profile case.

Source code in process/models/physics/plasma_profiles.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
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
def parabolic_paramterisation(self):
    """Parameterise plasma profiles in the case where i_plasma_pedestal == 0.

    This routine calculates the parameterization of plasma profiles in the case where i_plasma_pedestal=0.
    It sets the necessary physics variables for the parabolic profile case.
    """
    # Reset pedestal values to agree with original parabolic profiles
    if (
        physics_variables.radius_plasma_pedestal_temp_norm != 1.0
        or physics_variables.radius_plasma_pedestal_density_norm != 1.0
        or physics_variables.temp_plasma_pedestal_kev != 0.0
        or physics_variables.temp_plasma_separatrix_kev != 0.0
        or physics_variables.nd_plasma_pedestal_electron != 0.0
        or physics_variables.nd_plasma_separatrix_electron != 0.0
        or physics_variables.tbeta != 2.0
    ):
        logger.error(
            "Parabolic plasma profiles is used for an L-Mode plasma, "
            "but the physics variables do not describe an L-Mode plasma. "
            "'radius_plasma_pedestal_temp_norm', 'radius_plasma_pedestal_density_norm', 'temp_plasma_pedestal_kev', 'temp_plasma_separatrix_kev', 'nd_plasma_pedestal_electron', 'nd_plasma_separatrix_electron', "
            "and 'tbeta' have all been reset to L-Mode appropriate values"
        )

        physics_variables.radius_plasma_pedestal_temp_norm = 1.0e0
        physics_variables.radius_plasma_pedestal_density_norm = 1.0e0
        physics_variables.temp_plasma_pedestal_kev = 0.0e0
        physics_variables.temp_plasma_separatrix_kev = 0.0e0
        physics_variables.nd_plasma_pedestal_electron = 0.0e0
        physics_variables.nd_plasma_separatrix_electron = 0.0e0
        physics_variables.tbeta = 2.0e0

    # Re-caluclate core and profile values
    self.teprofile.run()
    self.neprofile.run()

    #  Profile factor; ratio of density-weighted to volume-averaged
    #  temperature

    physics_variables.f_temp_plasma_electron_density_vol_avg = (
        (1.0e0 + physics_variables.alphan)
        * (1.0e0 + physics_variables.alphat)
        / (1.0e0 + physics_variables.alphan + physics_variables.alphat)
    )

    # Line averaged electron density (IPDG89)
    # Taken by integrating the parabolic profile over rho in the bounds of 0 and 1 and dividng by the width of the integration bounds

    physics_variables.nd_plasma_electron_line = (
        physics_variables.nd_plasma_electrons_vol_avg
        * (1.0 + physics_variables.alphan)
        * (sp.special.gamma(0.5) / 2.0)
        * sp.special.gamma(physics_variables.alphan + 1.0)
        / sp.special.gamma(physics_variables.alphan + 1.5)
    )

    #  Density-weighted temperatures

    physics_variables.temp_plasma_electron_density_weighted_kev = (
        physics_variables.temp_plasma_electron_vol_avg_kev
        * physics_variables.f_temp_plasma_electron_density_vol_avg
    )
    physics_variables.temp_plasma_ion_density_weighted_kev = (
        physics_variables.temp_plasma_ion_vol_avg_kev
        * physics_variables.f_temp_plasma_electron_density_vol_avg
    )

    #  Central values for temperature (keV) and density (m**-3)

    physics_variables.temp_plasma_electron_on_axis_kev = (
        physics_variables.temp_plasma_electron_vol_avg_kev
        * (1.0 + physics_variables.alphat)
    )
    physics_variables.temp_plasma_ion_on_axis_kev = (
        physics_variables.temp_plasma_ion_vol_avg_kev
        * (1.0 + physics_variables.alphat)
    )

    physics_variables.nd_plasma_electron_on_axis = (
        physics_variables.nd_plasma_electrons_vol_avg
        * (1.0 + physics_variables.alphan)
    )
    physics_variables.nd_plasma_ions_on_axis = (
        physics_variables.nd_plasma_ions_total_vol_avg
        * (1.0 + physics_variables.alphan)
    )

pedestal_parameterisation()

Instance temperature and density profiles then integrate them, setting physics variables temp_plasma_electron_density_weighted_kev and temp_plasma_ion_density_weighted_kev.

This routine instances temperature and density profiles and integrates them to calculate the values of the physics variables temp_plasma_electron_density_weighted_kev and temp_plasma_ion_density_weighted_kev.

Source code in process/models/physics/plasma_profiles.py
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
def pedestal_parameterisation(self):
    """Instance temperature and density profiles then integrate them, setting physics variables temp_plasma_electron_density_weighted_kev and temp_plasma_ion_density_weighted_kev.

    This routine instances temperature and density profiles and integrates them to calculate the values of the physics variables `temp_plasma_electron_density_weighted_kev` and `temp_plasma_ion_density_weighted_kev`.
    """
    #  Run TeProfile and NeProfile class methods:
    #  Re-caluclate core and profile values

    self.teprofile.run()
    self.neprofile.run()

    #  Perform integrations to calculate ratio of density-weighted
    #  to volume-averaged temperature, etc.
    #  Density-weighted temperature = integral(n.T dV) / integral(n dV)
    #  which is approximately equal to the ratio
    #  integral(rho.n(rho).T(rho) drho) / integral(rho.n(rho) drho)

    drho = self.neprofile.profile_dx
    dens = self.neprofile.profile_y
    temp = self.teprofile.profile_y
    rho = self.neprofile.profile_x

    arg1 = rho * dens * temp
    arg2 = rho * dens

    integ1 = sp.integrate.simpson(arg1, x=rho, dx=drho)
    integ2 = sp.integrate.simpson(arg2, x=rho, dx=drho)

    physics_variables.integ1 = integ1
    physics_variables.integ2 = integ2
    #  Density-weighted temperatures
    physics_variables.temp_plasma_electron_density_weighted_kev = integ1 / integ2
    physics_variables.temp_plasma_ion_density_weighted_kev = (
        physics_variables.temp_plasma_ion_vol_avg_kev
        / physics_variables.temp_plasma_electron_vol_avg_kev
        * physics_variables.temp_plasma_electron_density_weighted_kev
    )
    #  Profile factor; ratio of density-weighted to volume-averaged
    #  temperature

    physics_variables.f_temp_plasma_electron_density_vol_avg = (
        physics_variables.temp_plasma_electron_density_weighted_kev
        / physics_variables.temp_plasma_electron_vol_avg_kev
    )

    #  Line-averaged electron density
    #  = integral(n(rho).drho)

    physics_variables.nd_plasma_electron_line = self.neprofile.profile_integ

    #  Scrape-off density / volume averaged density
    #  (Input value is used if i_plasma_pedestal = 0)

    divertor_variables.prn1 = max(
        0.01e0,
        physics_variables.nd_plasma_separatrix_electron
        / physics_variables.nd_plasma_electrons_vol_avg,
    )  # Preventing division by zero later

calculate_profile_factors()

Calculate and set the central pressure (pres_plasma_thermal_on_axis) using the ideal gas law and the pressure profile index (alphap).

This method calculates the central pressure (pres_plasma_thermal_on_axis) using the ideal gas law and the pressure profile index (alphap). It sets the value of the physics variable pres_plasma_thermal_on_axis.

Source code in process/models/physics/plasma_profiles.py
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
256
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
283
284
285
286
287
288
def calculate_profile_factors(self):
    """Calculate and set the central pressure (pres_plasma_thermal_on_axis) using the ideal gas law and the pressure profile index (alphap).

    This method calculates the central pressure (pres_plasma_thermal_on_axis) using the ideal gas law and the pressure profile index (alphap).
    It sets the value of the physics variable `pres_plasma_thermal_on_axis`.
    """

    #  Central pressure (Pa), from ideal gas law : p = nkT

    physics_variables.pres_plasma_thermal_on_axis = (
        physics_variables.nd_plasma_electron_on_axis
        * physics_variables.temp_plasma_electron_on_axis_kev
        + physics_variables.nd_plasma_ions_on_axis
        * physics_variables.temp_plasma_ion_on_axis_kev
    ) * constants.KILOELECTRON_VOLT

    # Electron pressure profile (Pa)
    physics_variables.pres_plasma_electron_profile = self.neprofile.profile_y * (
        self.teprofile.profile_y * constants.KILOELECTRON_VOLT
    )

    # Total ion pressure profile (Pa)
    physics_variables.pres_plasma_ion_total_profile = (
        physics_variables.nd_plasma_ions_total_vol_avg
        * (self.neprofile.profile_y / physics_variables.nd_plasma_electrons_vol_avg)
    ) * (
        self.teprofile.profile_y
        * constants.KILOELECTRON_VOLT
        * physics_variables.f_temp_plasma_ion_electron
    )

    # Total pressure profile (Pa)
    physics_variables.pres_plasma_thermal_total_profile = (
        physics_variables.pres_plasma_electron_profile
        + physics_variables.pres_plasma_ion_total_profile
    )

    # Fuel ion pressure profile (Pa)
    physics_variables.pres_plasma_fuel_profile = (
        physics_variables.nd_plasma_fuel_ions_vol_avg
        * (self.neprofile.profile_y / physics_variables.nd_plasma_electrons_vol_avg)
    ) * (
        self.teprofile.profile_y
        * constants.KILOELECTRON_VOLT
        * physics_variables.f_temp_plasma_ion_electron
    )

    #  Pressure profile index (only true for a parabolic profile)
    #  N.B. pres_plasma_thermal_on_axis is NOT equal to <p> * (1 + alphap), but p(rho) = n(rho)*T(rho)
    #  and <p> = <n>.T_n where <...> denotes volume-averages and T_n is the
    #  density-weighted temperature

    physics_variables.alphap = physics_variables.alphan + physics_variables.alphat

    # Calculate the volume averaged plasma thermal pressure from the density-weighted temperatures
    # Density-weighted temperatures are used as <nT> != <n>*<T>
    physics_variables.pres_plasma_thermal_vol_avg = (
        physics_variables.nd_plasma_electrons_vol_avg
        * physics_variables.temp_plasma_electron_density_weighted_kev
        + physics_variables.nd_plasma_ions_total_vol_avg
        * physics_variables.temp_plasma_ion_density_weighted_kev
    ) * constants.KILOELECTRON_VOLT

    # Central plasma current density (A/m^2)
    # Assumes a parabolic profile for the current density
    physics_variables.j_plasma_on_axis = (
        (physics_variables.plasma_current)
        * 2
        / (
            sp.special.beta(0.5, physics_variables.alphaj + 1)
            * physics_variables.a_plasma_poloidal
        )
    )

calculate_parabolic_profile_factors() staticmethod

Calculate the gradient information for i_plasma_pedestal = 0.

This function calculates the gradient information for the plasma profiles at the pedestal region when the value of i_plasma_pedestal is 0. It is used by the stellarator routines.

The function uses analytical parametric formulas to calculate the gradient information. The maximum normalized radius (rho_max) is obtained by equating the second derivative to zero.

Source code in process/models/physics/plasma_profiles.py
290
291
292
293
294
295
296
297
298
299
300
301
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
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
@staticmethod
def calculate_parabolic_profile_factors():
    """Calculate the gradient information for i_plasma_pedestal = 0.

    This function calculates the gradient information for the plasma profiles at the pedestal region
    when the value of i_plasma_pedestal is 0. It is used by the stellarator routines.

    The function uses analytical parametric formulas to calculate the gradient information.
    The maximum normalized radius (rho_max) is obtained by equating the second derivative to zero.
    """
    if physics_variables.i_plasma_pedestal == 0:
        if physics_variables.alphat > 1.0:
            # Rho (normalized radius), where temperature derivative is largest
            rho_te_max = 1.0 / np.sqrt(-1.0 + 2.0 * physics_variables.alphat)
            dtdrho_max = (
                -(2.0**physics_variables.alphat)
                * (-1.0 + physics_variables.alphat)
                ** (-1.0 + physics_variables.alphat)
                * physics_variables.alphat
                * (-1.0 + 2.0 * physics_variables.alphat)
                ** (0.5e0 - physics_variables.alphat)
                * physics_variables.temp_plasma_electron_on_axis_kev
            )
            te_max = (
                physics_variables.temp_plasma_electron_on_axis_kev
                * (1 - rho_te_max**2) ** physics_variables.alphat
            )

        elif physics_variables.alphat <= 1.0 and physics_variables.alphat > 0.0:
            # This makes the profiles very 'boxy'
            # The gradient diverges here at the edge so define some 'wrong' value of 0.9
            # to approximate the gradient
            rho_te_max = 0.9
            dtdrho_max = (
                -2.0
                * physics_variables.alphat
                * rho_te_max
                * (1 - rho_te_max**2) ** (-1.0 + physics_variables.alphat)
                * physics_variables.temp_plasma_electron_on_axis_kev
            )
            te_max = (
                physics_variables.temp_plasma_electron_on_axis_kev
                * (1 - rho_te_max**2) ** physics_variables.alphat
            )
        else:
            raise ProcessValueError(
                f"alphat is negative: {physics_variables.alphat}"
            )

        # Same for density
        if physics_variables.alphan > 1.0:
            rho_ne_max = 1.0 / np.sqrt(-1.0 + 2.0 * physics_variables.alphan)
            dndrho_max = (
                -(2.0**physics_variables.alphan)
                * (-1.0 + physics_variables.alphan)
                ** (-1.0 + physics_variables.alphan)
                * physics_variables.alphan
                * (-1.0 + 2.0 * physics_variables.alphan)
                ** (0.5 - physics_variables.alphan)
                * physics_variables.nd_plasma_electron_on_axis
            )
            ne_max = (
                physics_variables.nd_plasma_electron_on_axis
                * (1e0 - rho_ne_max**2) ** physics_variables.alphan
            )
        elif physics_variables.alphan <= 1.0 and physics_variables.alphan > 0.0:
            # This makes the profiles very 'boxy'
            # The gradient diverges here at the edge so define some 'wrong' value of 0.9
            # to approximate the gradient
            rho_ne_max = 0.9
            dndrho_max = (
                -2.0
                * physics_variables.alphan
                * rho_ne_max
                * (1 - rho_ne_max**2) ** (-1.0 + physics_variables.alphan)
                * physics_variables.nd_plasma_electron_on_axis
            )
            ne_max = (
                physics_variables.nd_plasma_electron_on_axis
                * (1 - rho_ne_max**2) ** physics_variables.alphan
            )
        else:
            raise ProcessValueError(
                f"alphan is negative: {physics_variables.alphan}"
            )

        # set normalized gradient length
        # te at rho_te_max
        physics_variables.gradient_length_te = (
            -dtdrho_max * physics_variables.rminor * rho_te_max / te_max
        )
        # same for density:
        physics_variables.gradient_length_ne = (
            -dndrho_max * physics_variables.rminor * rho_ne_max / ne_max
        )