From b5658fce368306be456b332c3fa228bd9d36b925 Mon Sep 17 00:00:00 2001
From: Alexis Gamelin <alexis.gamelin@synchrotron-soleil.fr>
Date: Tue, 24 Oct 2023 09:54:29 +0200
Subject: [PATCH] Add tuner_diff option to plot_cavitydata

Remove IQ mentions in ProportionalIntegralLoop as not implemented yet.
Add delay < 1 error in ProportionalLoop.
Addition referece for DirectFeedback docstring.
---
 mbtrack2/tracking/monitors/plotting.py | 23 ++++++++++++++++++++++-
 mbtrack2/tracking/rf.py                | 15 ++++++++++-----
 2 files changed, 32 insertions(+), 6 deletions(-)

diff --git a/mbtrack2/tracking/monitors/plotting.py b/mbtrack2/tracking/monitors/plotting.py
index 5f256b9..3077300 100644
--- a/mbtrack2/tracking/monitors/plotting.py
+++ b/mbtrack2/tracking/monitors/plotting.py
@@ -1107,6 +1107,8 @@ def plot_cavitydata(filename, cavity_name, phasor="cavity",
             - "mean" plots the mean phasor voltage and angle versus time for 
             non empty bunches. Needs BeamMonitor data with same save_every as 
             CavityMonitor.
+            - "tuner_diff" plots the tuner difference from optimal tuning 
+            condition (for a given bunch) and the tuning angle versus time.
     bunch_number : int, optional
         Bunch number to select. The default is 0.
     turn : int, optional
@@ -1134,7 +1136,7 @@ def plot_cavitydata(filename, cavity_name, phasor="cavity",
     units = [" voltage [MV]", " voltage [MV]", " voltage [MV]", " current [A]"]
     units_val = [1e-6, 1e-6, 1e-6, 1]
     
-    if plot_type == "bunch" or plot_type == "mean":
+    if (plot_type == "bunch") or (plot_type == "mean"):
     
         if plot_type == "bunch":
             data = [cavity_data["cavity_phasor_record"][bunch_number,:], 
@@ -1267,5 +1269,24 @@ def plot_cavitydata(filename, cavity_name, phasor="cavity",
         ax.set_ylabel("Power [kW]")
         plt.legend()
         
+    if plot_type == "tuner_diff":
+        data0 = np.angle(cavity_data["cavity_phasor_record"][bunch_number,:])
+        data1 = np.array(cavity_data["psi"])
+        data2 = np.array(cavity_data["theta_g"])
+        
+        ylabel1 = "Tuner diff. from optimal [rad]"
+        ylabel2 = r"Tuning angle $\psi$ [rad]"
+        fig, ax = plt.subplots()
+        twin = ax.twinx()
+        p1, = ax.plot(time, data0-data2+data1, color="r",label=ylabel1)
+        p2, = twin.plot(time, data1, color="b", label=ylabel2)
+        ax.set_xlabel("Turn number")
+        ax.set_ylabel(ylabel1)
+        twin.set_ylabel(ylabel2)
+        plots = [p1, p2]
+        ax.legend(handles=plots, loc="best")
+        ax.yaxis.label.set_color("r")
+        twin.yaxis.label.set_color("b")
+        
     file.close()
     return fig
diff --git a/mbtrack2/tracking/rf.py b/mbtrack2/tracking/rf.py
index 36fa232..7f38c0b 100644
--- a/mbtrack2/tracking/rf.py
+++ b/mbtrack2/tracking/rf.py
@@ -993,6 +993,7 @@ class ProportionalLoop():
         Phase gain of the feedback.
     delay : int
         Feedback delay in unit of turns.
+        Must be supperior or equal to 1.
 
     """
     def __init__(self, ring, cav_res, gain_A, gain_P, delay):
@@ -1001,6 +1002,8 @@ class ProportionalLoop():
         self.gain_A = gain_A
         self.gain_P = gain_P
         self.delay = int(delay)
+        if delay < 1:
+            raise ValueError("delay must be >= 1.")
         self.volt_delay = np.ones(self.delay)*self.cav_res.Vc
         self.phase_delay = np.ones(self.delay)*self.cav_res.theta
     
@@ -1108,7 +1111,6 @@ class ProportionalIntegralLoop():
     gain : float list
         Proportional gain (Pgain) and integral gain (Igain) of the feedback 
         system in the form of a list [Pgain, Igain].
-        For IQ feedback, same gain set is used for I and Q.
         In case of normal conducting cavity (QL~1e4), the Pgain of ~1.0 and 
         Igain of ~1e4(5) are usually used.
         In case of super conducting cavity (QL > 1e6), the Pgain of ~100 
@@ -1162,9 +1164,9 @@ class ProportionalIntegralLoop():
        By using ig instead of Vg, the cavity response can be taken account.
        3) ig changes are reflected to Vg after the specifed delay (delay) of the system
 
-    Vc,IQ-->(rot)-->(-)-->(V->I,fac)-->PI-->Ig,IQ {--> Vg,IQ}
-                     |
-                     Ref
+    Vc-->(rot)-->(-)-->(V->I,fac)-->PI-->Ig --> Vg
+                  |
+                 Ref
 
     Examples
     --------
@@ -1299,7 +1301,7 @@ class ProportionalIntegralLoop():
 
 class DirectFeedback(ProportionalIntegralLoop):
     """
-    Direct RF FB (DFB) via generator current (ig) using PI controller [1].
+    Direct RF FB (DFB) via generator current (ig) using PI controller [1,2].
     
     The DirectFeedback inherits from ProportionalIntegralLoop so all 
     mandatory parameters from ProportionalIntegralLoop should be passed as
@@ -1364,6 +1366,9 @@ class DirectFeedback(ProportionalIntegralLoop):
     [1] : Akai, K. (2022). Stability analysis of rf accelerating mode with 
     feedback loops under heavy beam loading in SuperKEKB. PRAB, 25(10), 
     102002.
+    [2] : N. Yamamoto et al. (2023). Stability survey of a double RF system 
+    with RF feedback loops for bunch lengthening in a low-emittance synchrotron 
+    ring. In Proc. IPAC'23. doi:10.18429/JACoW-IPAC2023-WEPL161
 
     """
     def __init__(self, DFB_gain, DFB_phase_shift, DFB_sample_num=None,
-- 
GitLab