Skip to content

Commit

Permalink
Fixed Chromaticity fit (#43)
Browse files Browse the repository at this point in the history
---------

Co-authored-by: Lina Hoummi <[email protected]>
Co-authored-by: lmalina <[email protected]>
  • Loading branch information
3 people authored Jan 24, 2024
1 parent e003dff commit c382cbc
Show file tree
Hide file tree
Showing 3 changed files with 75 additions and 15 deletions.
3 changes: 3 additions & 0 deletions doc/modules/correction.rst
Original file line number Diff line number Diff line change
Expand Up @@ -12,3 +12,6 @@ Correction algorithms
.. automodule:: pySC.correction.tune
:members:

.. automodule:: pySC.correction.chroma
:members:

51 changes: 36 additions & 15 deletions pySC/correction/chroma.py
Original file line number Diff line number Diff line change
@@ -1,36 +1,57 @@
"""
Chromaticity
-------------
This module contains functions to fit the chromaticity of 'SC.RING'.
"""

import numpy as np
from scipy.optimize import fmin

from pySC.utils.at_wrapper import atlinopt
from pySC.utils import logging_tools

LOGGER = logging_tools.get_logger(__name__)


def fit_chroma(SC, s_ords, target_chroma=None, init_step_size=np.array([2, 2]), xtol=1E-4, ftol=1E-3,
tune_knobs_ords=None, tune_knobs_delta_k=None):
def fit_chroma(SC, s_ords, target_chroma=None, init_step_size=np.array([2, 2]), xtol=1E-4, ftol=1E-3):
"""
Applies a chromaticity correction using two sextupole families.
Args:
SC: SimulatedCommissioning instance
s_ords: [2xN] array or list [[1 x NS1],[1 x NS2], [1 x NS3], ...] of sextupole ordinates
target_chroma ([1x2] array, optional): Target chromaticity for correction. Default: chromaticity of 'SC.IDEALRING'
init_step_size ([1x2] array, optional): Initial step size for the solver. Default: [2,2]
xtol(float, optional): Step tolerance for solver. Default: 1e-4
ftol(float, optional): Merit tolerance for solver. Default: 1e-3
Returns:
SC: SimulatedCommissioning instance with corrected chromaticity.
Example:
SC = fit_chroma(SC, s_ords=[SCgetOrds(sc.RING, 'SF'), SCgetOrds(sc.RING, 'SD')], target_chroma=numpy.array([1,1]))
"""
if target_chroma is None:
_, _, target_chroma = atlinopt(SC.IDEALRING, 0, [])
target_chroma = SC.IDEALRING.get_chrom()[0:2]
if np.sum(np.isnan(target_chroma)):
LOGGER.error('Target chromaticity must not contain NaN. Aborting.')
return SC
if tune_knobs_ords is not None and tune_knobs_delta_k is not None:
for nFam in range(len(tune_knobs_ords)):
SC.set_magnet_setpoints(tune_knobs_ords[nFam], tune_knobs_delta_k[nFam], False, 1,
method='add') # TODO quads here?
LOGGER.debug(f'Fitting chromaticities from {atlinopt(SC.RING, 0, [])[2]} to {target_chroma}.') # first two elements
SP0 = np.zeros((len(s_ords), len(s_ords[0]))) # TODO can the lengts vary

LOGGER.debug(f'Fitting chromaticities from {SC.RING.get_chrom()} to {target_chroma}.') # first two elements
SP0 = []
for n in range(len(s_ords)):
SP0.append(np.zeros_like(s_ords[n]))
for nFam in range(len(s_ords)):
for n in range(len(s_ords[nFam])):
SP0[nFam][n] = SC.RING[s_ords[nFam][n]].SetPointB[2]
fun = lambda x: _fit_chroma_fun(SC, s_ords, x, SP0, target_chroma)
sol = fmin(fun, init_step_size, xtol=xtol, ftol=ftol)
SC.set_magnet_setpoints(s_ords, sol + SP0, False, 1, method='abs', dipole_compensation=True)
LOGGER.debug(f' Final chromaticity: {atlinopt(SC.RING, 0, [])[2]}\n Setpoints change: {sol}.') # first two elements
# TODO needs to set the solution to SC
LOGGER.debug(f' Final chromaticity: {SC.RING.get_chrom()}\n Setpoints change: {sol}.') # first two elements
return SC


def _fit_chroma_fun(SC, q_ords, setpoints, init_setpoints, target):
SC.set_magnet_setpoints(q_ords, setpoints + init_setpoints, False, 2, method='abs', dipole_compensation=True)
_, _, nu = atlinopt(SC.RING, 0, [])
def _fit_chroma_fun(SC, s_ords, setpoints, init_setpoints, target):
for n in range(len(s_ords)):
SC.set_magnet_setpoints(s_ords[n], setpoints[n] + init_setpoints[n], False, 2, method='abs', dipole_compensation=True)
nu = SC.RING.get_chrom()[0:2]
return np.sqrt(np.mean((nu - target) ** 2))
36 changes: 36 additions & 0 deletions tests/test_chroma.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import at
import numpy as np
from numpy.testing import assert_allclose
import pytest
from pathlib import Path
from pySC.core.simulated_commissioning import SimulatedCommissioning
from pySC.utils import logging_tools, sc_tools
from pySC.correction.chroma import fit_chroma

LOGGER = logging_tools.get_logger(__name__)
INPUTS = Path(__file__).parent / "inputs"


def test_chroma_hmba(at_ring):
np.random.seed(12345678)
sc = SimulatedCommissioning(at_ring)
sc.register_bpms(sc_tools.ords_from_regex(sc.RING, 'BPM'), Roll=0.0)
sc.register_magnets(sc_tools.ords_from_regex(sc.RING, 'SF|SD'), CalErrorB=np.array([0, 0, 0.05])) # [1/m]
sc.register_cavities(sc_tools.ords_from_regex(sc.RING, 'RFC'))
sc.apply_errors()
s_ords = [sc_tools.ords_from_regex(sc.RING, '^SF'), sc_tools.ords_from_regex(sc.RING, '^SD')]
target = np.array([2.0,2.0])
sc = fit_chroma(sc, s_ords, target_chroma=target)
assert_allclose(sc.RING.get_chrom()[0:2], target, atol=2e-6, rtol=1e-6)


@pytest.fixture
def at_ring():
ring = at.load_mat(f'{INPUTS}/hmba.mat')
ring.enable_6d()
at.set_cavity_phase(ring)
at.set_rf_frequency(ring)

ring.tapering(niter=3, quadrupole=True, sextupole=True)
ring = at.Lattice(ring)
return ring

0 comments on commit c382cbc

Please sign in to comment.