This notebook was created by Sergey Tomin (sergey.tomin@desy.de) and Martin Dohlus (DESY). May 2018.
8. Laser Heater
The OCELOT Charged Particle Beam Dynamics (CPBD) module includes various physics processes like CSR, SpaceCharge, Wake3D, BeamTransform. There are other "physics processes", which do not have anything in common with physics but use the same interface (or parent class "PhysProc") e.g. SaveBeam. Using this interface, one can implement own "physics process". In this tutorial we will show how to do it on the example of the laser heater.
We will start with the theory of the laser heater and then will show how to implement the heating process in the code.
Please note, we chose ϕ0=0 in exponential to simplify the notation, the phase accounts
for the fact that the light wave will in general be phase-shifted against the sinusoidal
trajectory of an arbitrary electron because the bunch is far longer than the light
wavelength. This phase will be reintroduced in the last step.
Remembering from section [Electron motion in undulator](#Electron motion) z(t)=vzt−8γ2kuK2sin(2ωut).
If we ensure fulfillment of the resonance condition only for the first harmonic as we did in the "Option 1" (in that case n=0) the only one term of the sum will survive the only one term of the sum will survive. Indeed, for the n=0 argument of the exponential will be constant but for other harmonic numbers it will oscillate and give in average 0. So our final expression:
Finally, remembering derivation of the electric field amplitude E0=πw24Z0P(0), we can write expression for amplitude of energy modulation on axis (x=y=0)
where ϕ0=λph2πzi is phase shift of the light wave against the sinusoidal trajectory of an arbitrary electron (with longitudinal coordinate zi) what we chose 0 at the at the beginning of this journey.
Transverse particle distribution:
P(x,y)=2πσxσy1e−2σx2(x−x)2−2σy2(y−y)2
RMS energy spread:
Erms=RdE=∫2πdϕdxdy×P(x,y)(ΔE(x,y,ϕ))2
where R - shape factor (particle-photon matching), for instance for round beam no offset R=2+2(σp/σI)21
Practical formulas and the laser heater parameters
To simulate the laser heater process we need to calculate dE and then using the expression for the [energy modulation](#practical formula) we can calculate the energy spread increasing. In that case, the shape factor will be automatically taking into account.
Suppose we want to have energy spread Erms=RdE=12.5keV. So we can estimate the laser beam power which is needed for that. We just need to invert the expression that we got before with minor modifications:
dE[eV]=2βEbeam[eV]w(mc2)2LuKP0P(0)
The laser power is:
P(0)=(2(mc2)2ErmsEβRLuKw)2P0
And finally, using expression for [position and phase dependent energy modulation](#practical formula), we can write the energy kick in the OCELOT coordinates for i-particle passing dz of the laser heater undulator with the particle coordinates x,y - transverse coordinates, {τ=cdt;δ=p0cΔE} - longitudinal coordinates:
import numpy as np from scipy.special import jv from ocelot import* defK2field(K, lu =0.04): field = K*m_e_eV*2.*pi/(lu*speed_of_light) return field # RMS energy modulation Erms =4e6# eV # electron beam energy Eb =2.4e9# eV #Eb = 17.5e9 # eV # wavelength l_ph =800e-9 mc2 =0.511e6 gamma = Eb/mc2 print("gamma = ", gamma) # undulator period length lperiod =0.2# m print("lperiod = ", lperiod) K =2*gamma * np.sqrt(l_ph/lperiod) print("K estimated = ", K) print("B estimated = ", K2field(K, lu=lperiod)) ku =2*np.pi/lperiod # beam size sigma_x =50e-6# m # Undulator length # Undulator parameter #Ku = 18.8 Ku = K # Impedance of free space Z0 =376.73# Ohm # shape factor - ideal matching R =0.5 P0 =8.7e9# GW Y = Ku**2/(4+2*Ku**2) # amplitude of beam oscilation in modulator Ax = Ku/(gamma * ku) print("amplitude of oscilation", Ax*1e6," um") # the waist radius w =2*sigma_x +2* Ax print("waist = ", w *1e6," um") # Rayleigh length Zr = np.pi*w**2/l_ph Lu =2* Zr# m print("Zr = ", Zr," m") # Laser power to obtain Erms energy spread with shape factor R P =(Erms*Eb*w/(2*mc2**2*R*Lu*Ku*(-jv(1,Y)+ jv(0, Y))))**2*P0 print("Laser power =", P*1e-9,"GW")
initializing ocelot... gamma =4696.673189823875 lperiod =0.2 K estimated =18.7866927592955 B estimated =1.0060033803412982 amplitude of oscilation 127.32395447351627 um waist =354.64790894703253 um Zr =0.49391781711690497 m Laser power =8.810302238315355 GW
import numpy as np from scipy.special import jv # RMS energy spread Erms =20e3# eV # electron beam energy Eb =130e6# eV mc2 =0.511e6 # undulator period length lperiod =0.074# m # beam size sigma_I =300e-6# m # Undulator length Lu =0.666# m # Undulator parameter Ku =1.294 # Impedance of free space Z0 =376.73# Ohm # shape factor - ideal matching R =0.5 P0 =8.7e9# GW Y = Ku**2/(4+2*Ku**2) # the waist radius w =2*sigma_I # Laser power to obtain Erms energy spread with shape factor R P =(Erms*Eb*w/(2*mc2**2*R*Lu*Ku*(-jv(1,Y)+ jv(0, Y))))**2*P0 print("Laser power =", P,"W")
To create class LaserHeater we use parent class PhysProc that includes two standard methods: prepare(self, MagneticLattice) and apply(self, ParticleArray, dz).
To implement the energy kick from the laser heater we just need redefine method apply(self, ParticleArray, dz)
But first, we import main OCELOT modules and functions and class PhysProc will be imported as well.
# the output of plotting commands is displayed inline within frontends, # directly below the code cell that produced it %matplotlib inline # this python library provides generic shallow (copy) # and deep copy (deepcopy) operations from copy import deepcopy # import from Ocelot graphical modules from ocelot.gui.accelerator import* # import injector lattice from injector_lattice import* # import from Ocelot main modules and functions # including parent class PhysProc from ocelot import* classLaserHeater(PhysProc): def__init__(self, step=1): PhysProc.__init__(self, step=step) # amplitude of energy modulation on axis self.dE =12500e-9# GeV self.Ku =1.294# undulator parameter self.Lu =0.8# [m] - undulator length self.lperiod =0.074# [m] - undulator period length self.sigma_l =300e-6# [m] self.sigma_x = self.sigma_l self.sigma_y = self.sigma_l self.x_mean =0 self.y_mean =0 defapply(self, p_array, dz): gamma = p_array.E/m_e_GeV lbda_ph = self.lperiod/(2*gamma**2)*(1+self.Ku**2/2) k_ph =2*np.pi/lbda_ph pc = np.sqrt(p_array.E**2- m_e_GeV**2) A = self.dE/(pc)*dz/self.Lu dx = p_array.x()[:]- self.x_mean dy = p_array.y()[:]- self.y_mean p_array.p()[:]+= A*np.cos(k_ph*p_array.tau()[:])*np.exp( -0.25*dx**2/self.sigma_x**2 -0.25*dy**2/self.sigma_y**2)