Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Tune scan #49

Open
wants to merge 29 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 22 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
daa01d9
tune_scan function description added
Sep 26, 2023
2cd91c1
corrected
Sep 26, 2023
09a77c8
indents
simoneliuzzo Sep 26, 2023
7becaaf
minors for help
simoneliuzzo Sep 26, 2023
1ed4e9f
remove error message. Fix conflict.
Sep 26, 2023
1a5db63
added doc. Removed unexplained inputs of quadrupoles.
Sep 26, 2023
c596445
revert changes
Sep 26, 2023
025d3de
revert changes
Sep 26, 2023
21640de
corrected conflicts
Sep 26, 2023
0d783a9
Merge branch 'master' into tune_scan
linahv Sep 26, 2023
55013c6
fit_tune documentation
Sep 28, 2023
b41ef80
added description of fit_tune
Sep 28, 2023
71410bb
changes in fit_tune
Sep 28, 2023
87f4b27
testing in progress
Sep 29, 2023
db8f62d
quadrupole input now allows N number of families, each with different…
Oct 5, 2023
12ce744
test functions for tune and chroma correction
Oct 5, 2023
1faa6b4
typo
Oct 5, 2023
6c3935d
succesful test
Oct 5, 2023
2d18041
Merge branch 'master' into tune_scan
linahv Oct 5, 2023
2a4eb29
updated documentation
Oct 5, 2023
55c699b
Merge branch 'tune_scan' of https://github.com/lmalina/pySC into tune…
Oct 5, 2023
3eb4a1b
removed the function tune
Oct 5, 2023
1fc73bf
minor corrections
Oct 11, 2023
9e349c5
added the assess line
Oct 11, 2023
f10b0f1
simplified function and updated default values
Oct 11, 2023
9b58b02
assert test function added, test succesful
Oct 11, 2023
14f1f2e
Merge branch 'master' into tune_scan
lmalina Jan 24, 2024
92895a6
Merge branch 'master' into tune_scan
lmalina Jan 24, 2024
962b6ba
Little fixes
lmalina Jan 24, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 31 additions & 20 deletions pySC/correction/tune.py
Original file line number Diff line number Diff line change
Expand Up @@ -129,32 +129,43 @@ def plot_scan(fin_trans, max_turns, first_quads, rel_quad_changes):
return f, [ax1, ax2, ax3]


def fit_tune(SC, q_ords, target_tune=None, xtol=1E-4, ftol=1E-3, fit_integer=True):
# TODO check if experimantally feasible
def fit_tune(SC, q_ords, target_tune=None, init_step_size=np.array([0.001, 0.001]),xtol=1E-4, ftol=1E-3, fit_integer=True):
"""
Applies a tune correction using two quadrupole families.
Note: this is not beam based but assumes the tunes can be measured reasonably well.

Args:
SC: SimulatedCommissioning instance
q_ords: [2xN] array or list [[1 x NQ1],[1 x NQ2], [1 x NQ3], ...] of quadrupole ordinates
target_tune (optional, [1x2] array): Target tunes for correction. Default: tunes of 'SC.IDEALRING'
init_step_size ([1x2] array, optional): Initial step size for the solver. Default: [1,1]
xtol(float, optional): Step tolerance for solver. Default: 1e-4
ftol(float, optional): Merit tolerance for solver. Default: 1e-4
linahv marked this conversation as resolved.
Show resolved Hide resolved
fit_integer(bool, optional): Flag specifying if the integer part should be fitted as well. Default: True.

Returns:
SC: SimulatedCommissioning instance with corrected tunes.
Example:
SC = fit_tune(SC, q_ords=[SCgetOrds(sc.RING, 'QF'), SCgetOrds(sc.RING, 'QD')], target_tune=numpy.array([0.16,0.21]))
"""
if target_tune is None:
target_tune = tune(SC, fit_integer, ideal=True)
LOGGER.debug(f'Fitting tunes from [{tune(SC, fit_integer)}] to [{target_tune}].')
SP0 = np.zeros((len(q_ords), len(q_ords[0]))) # TODO can the lengts vary
target_tune = SC.IDEALRING.get_tune(get_integer=fit_integer)
LOGGER.debug(f'Fitting tunes from [{SC.RING.get_tune(get_integer=fit_integer)}] to [{target_tune}].')
SP0 = []
for n in range(len(q_ords)):
SP0.append(np.zeros_like(q_ords[n])) #working with a list of two arrays
for nFam in range(len(q_ords)):
for n in range(len(q_ords[nFam])):
SP0[nFam][n] = SC.RING[q_ords[nFam][n]].SetPointB[1]
fun = lambda x: _fit_tune_fun(SC, q_ords, x, SP0, target_tune, fit_integer)
sol = fmin(fun, xtol=xtol, ftol=ftol)
SC.set_magnet_setpoints(q_ords, sol + SP0, False, 1, method='abs', dipole_compensation=True)
LOGGER.debug(f' Final tune: [{tune(SC, fit_integer)}]\n Setpoints change: [{sol}]')
sol = fmin(fun, init_step_size, xtol=xtol, ftol=ftol)
LOGGER.debug(f' Final tune: [{SC.RING.get_tune(get_integer=fit_integer)}]\n Setpoints change: [{sol}]')
return SC


def _fit_tune_fun(SC, q_ords, setpoints, init_setpoints, target, fit_integer):
SC.set_magnet_setpoints(q_ords, setpoints + init_setpoints, False, 1, method='abs', dipole_compensation=True)
nu = tune(SC, fit_integer)
return np.sqrt(np.mean((nu - target) ** 2))


def tune(SC, fit_integer: bool = False, ideal: bool = False):
ring = SC.IDEALRING if ideal else SC.RING
if fit_integer:
ld, _, _ = atlinopt(ring, 0, range(len(ring) + 1))
return ld[-1].mu / 2 / np.pi
_, nu, _ = atlinopt(ring, 0)
return nu
for nFam in range(len(q_ords)):
SC.set_magnet_setpoints(q_ords[nFam], setpoints[nFam] + init_setpoints[nFam], False, 1, method='abs', dipole_compensation=True)
nu = SC.RING.get_tune(get_integer=fit_integer)
nu = nu[0:2]
return np.sqrt(np.mean((nu - target) ** 2))
29 changes: 29 additions & 0 deletions tests/test_tune_chroma.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import copy
import at
import pytest
import numpy as np
from numpy.testing import assert_equal, assert_allclose
from tests.test_at_wrapper import at_lattice
from pySC.correction.tune import tune_scan, fit_tune
from pySC.correction.chroma import fit_chroma
from pySC.core.simulated_commissioning import SimulatedCommissioning
from pySC.utils.sc_tools import SCgetOrds


def test_fit_tune(at_lattice):
sc = SimulatedCommissioning(at_lattice)
sc.register_cavities(SCgetOrds(sc.RING, 'RFCav'),
FrequencyOffset=5E3,
VoltageOffset=5E3,
TimeLagOffset=0.5)
sc.register_magnets(SCgetOrds(sc.RING, 'QF'), HCM=1E-3,
CalErrorB=np.array([5E-2, 1E-3]),
MagnetOffset=200E-6 * np.array([1, 1, 0]),
MagnetRoll=200E-6 * np.array([1, 0, 0]))
sc.register_magnets(SCgetOrds(sc.RING, 'QD'), VCM=1E-3,
CalErrorA=np.array([5E-2, 0]),
CalErrorB=np.array([0, 1E-3]),
MagnetOffset=200E-6 * np.array([1, 1, 0]),
MagnetRoll=200E-6 * np.array([1, 0, 0]))
sc = fit_tune(sc, q_ords=[SCgetOrds(sc.RING, 'QF'), SCgetOrds(sc.RING, 'QD')], target_tune=np.array([0.16, 0.21]))
return sc
linahv marked this conversation as resolved.
Show resolved Hide resolved