-
Notifications
You must be signed in to change notification settings - Fork 49
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #399 from hiddenSymmetries/ag/on_axis_iota
compute the rotational transform on a magnetic axis
- Loading branch information
Showing
3 changed files
with
118 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,90 @@ | ||
import numpy as np | ||
from simsopt.geo import CurveRZFourier | ||
from scipy.integrate import solve_ivp | ||
|
||
__all__ = ['compute_on_axis_iota'] | ||
|
||
def compute_on_axis_iota(axis, magnetic_field): | ||
""" | ||
Computes the rotational transform on the magnetic axis of a device using a method based on | ||
equation (13) of Greene, Journal of Mathematical Physics 20, 1183 (1979); doi: 10.1063/1.524170. | ||
This method was shared by Stuart Hudson and Matt Landreman. NOTE: this function does not check that the provided | ||
axis is actually a magnetic axis of the input magnetic_field. | ||
Args: | ||
axis: a CurveRZFourier corresponding to the magnetic axis of the magnetic field. | ||
magnetic_field: the magnetic field in which the axis lies. | ||
Returns: | ||
iota: the rotational transform on the given axis. | ||
""" | ||
assert type(axis) is CurveRZFourier | ||
|
||
def tangent_map(phi, x): | ||
ind = np.array(phi) | ||
out = np.zeros((1, 3)) | ||
axis.gamma_impl(out, ind) | ||
magnetic_field.set_points(out) | ||
|
||
out = out.flatten() | ||
B = magnetic_field.B().flatten() | ||
dB_by_dX = magnetic_field.dB_by_dX().reshape((3, 3)) | ||
B1 = B[0] | ||
B2 = B[1] | ||
B3 = B[2] | ||
|
||
dB1_dx = dB_by_dX[0, 0] | ||
dB1_dy = dB_by_dX[0, 1] | ||
dB1_dz = dB_by_dX[0, 2] | ||
|
||
dB2_dx = dB_by_dX[1, 0] | ||
dB2_dy = dB_by_dX[1, 1] | ||
dB2_dz = dB_by_dX[1, 2] | ||
|
||
dB3_dx = dB_by_dX[2, 0] | ||
dB3_dy = dB_by_dX[2, 1] | ||
dB3_dz = dB_by_dX[2, 2] | ||
|
||
c = np.cos(2*np.pi*phi) | ||
s = np.sin(2*np.pi*phi) | ||
|
||
R = np.sqrt(out[0]**2 + out[1]**2) | ||
dx_dR = c | ||
dy_dR = s | ||
|
||
BR = c*B1 + s*B2 | ||
Bphi =-s*B1 + c*B2 | ||
BZ = B3 | ||
|
||
dB1_dR = dB1_dx * dx_dR + dB1_dy * dy_dR | ||
dB2_dR = dB2_dx * dx_dR + dB2_dy * dy_dR | ||
dB3_dR = dB3_dx * dx_dR + dB3_dy * dy_dR | ||
|
||
dBR_dR = c*dB1_dR + s*dB2_dR | ||
dBphi_dR = -s*dB1_dR + c*dB2_dR | ||
dBZ_dR = dB3_dR | ||
|
||
dBR_dZ = c*dB1_dz + s*dB2_dz | ||
dBphi_dZ = -s*dB1_dz + c*dB2_dz | ||
dBZ_dZ = dB3_dz | ||
|
||
Bphi_R = Bphi/R | ||
d_Bphi_R_dR = (dBphi_dR * R - Bphi)/R**2 | ||
d_Bphi_R_dZ = dBphi_dZ/R | ||
|
||
A11 = (dBR_dR - (BR / Bphi_R) * d_Bphi_R_dR) / Bphi_R | ||
A21 = (dBZ_dR - (BZ / Bphi_R) * d_Bphi_R_dR) / Bphi_R | ||
A12 = (dBR_dZ - (BR / Bphi_R) * d_Bphi_R_dZ) / Bphi_R | ||
A22 = (dBZ_dZ - (BZ / Bphi_R) * d_Bphi_R_dZ) / Bphi_R | ||
A = np.array([[A11,A12],[A21,A22]]) | ||
return 2*np.pi*np.array([A@x[:2], A@x[2:]]).flatten() | ||
|
||
t_span = [0, 1/axis.nfp] | ||
t_eval = t_span | ||
|
||
y0 = np.array([1, 0, 0, 1]) | ||
results = solve_ivp(tangent_map, t_span, y0, t_eval=t_eval, rtol=1e-12, atol=1e-12, method='RK45') | ||
M = results.y[:, -1].reshape((2,2)) | ||
evals, evecs = np.linalg.eig(M) | ||
iota = np.arctan2(np.imag(evals[0]), np.real(evals[0])) * axis.nfp/(2*np.pi) | ||
return iota |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
#!/usr/bin/env python3 | ||
import unittest | ||
import numpy as np | ||
from simsopt.field.biotsavart import BiotSavart | ||
from simsopt.field.coil import coils_via_symmetries | ||
from simsopt.field.magnetic_axis_helpers import compute_on_axis_iota | ||
from simsopt.configs.zoo import get_ncsx_data, get_hsx_data, get_giuliani_data | ||
|
||
|
||
class MagneticAxisHelpers(unittest.TestCase): | ||
|
||
def test_magnetic_axis_iota(self): | ||
""" | ||
Verify that the rotational transform can be computed on axis | ||
""" | ||
for (get_data, target_iota) in zip([get_hsx_data, get_ncsx_data, get_giuliani_data], [1.0418687161633922, 0.39549339846119463, 0.42297724084249616]): | ||
self.subtest_magnetic_axis_iota(get_data, target_iota) | ||
|
||
def subtest_magnetic_axis_iota(self, get_data, target_iota): | ||
curves, currents, ma = get_data() | ||
coils = coils_via_symmetries(curves, currents, ma.nfp, True) | ||
iota = compute_on_axis_iota(ma, BiotSavart(coils)) | ||
np.testing.assert_allclose(iota, target_iota, rtol=1e-10, atol=1e-10) | ||
|
||
if __name__ == "__main__": | ||
unittest.main() |