/* *----------------------------------------------------------------------* * * * CERN * * * * European Organization for Nuclear Research * * * * * * Program name: HEADTAIL Version 3.0 * * * * Authors and contact: E. BENEDETTO * * G. RUMOLO * * F. ZIMMERMANN * * AB Department * * CERN * * CH-1211 GENEVA 23 * * SWITZERLAND * * * * Tel. [041] (022) 767 1099 * * elena.benedetto@cern.ch * * [041] (022) 767 9054 * * frank.zimmermann@cern.ch * * [041] (022) 767 3264 * * giovanni.rumolo@cern.ch * * * * Copyright CERN, Geneva 2006 - Copyright and any other * * appropriate legal protection of this computer program and * * associated documentation reserved in all countries of the * * world. * * * * Organizations collaborating with CERN may receive this program * * and documentation freely and without charge. * * * * CERN undertakes no obligation for the maintenance of this * * program, nor responsibility for its correctness, and accepts * * no liability whatsoever resulting from its use. * * * * Program and documentation are provided solely for the use of * * the organization to which they are distributed. * * * * This program may not be copied or otherwise distributed * * without permission. This message must be retained on this and * * any other authorized copies. * * * * The material cannot be sold. CERN should be given credit in * * all references. * * * *----------------------------------------------------------------------* this version has been updated in June'04, merging coasting beam option of giovanni, phase advance+kicks in one betatron wavelenght by elena and some syncrotron motion options in case of nkick>1 by frank In this version the possibility of off-centering the beam in t=0 has also been introduced. There are two more parameters to be input, xkick and ykick, which define the kick amplitude. The tune spread introduced by space charge has also been updated: the tune shift of each particle is calculated at each turn taking into account the local and instantaneous sigma in the tune shift Laslett formula. Actually, according to the switch i_rot, it's possible to get the space charge roatation at each turn either around the local centroids and by using the local sigmas (i_rot=0), or around the full bunch centroid and by using the total sigmas (i_rot=1). i_rot is the second last entry of the new input file. The option of the electrons moving in a sinusoidal field is here included - it corresponds to i_dip=2 and takes the input value for the solenoid field (bsol, last line in the input file) The option of the proton space charge has been added - by using the routine ffrank, we make each bunch slice act on each proton in the slice, too. A new switch has been introduced, i_space, which is a value to be input in the input file. The most updated input file is to be found in the directory ~/Elcloud/HEADTAIL/DATA/PIC/TUNESPR and it's nobb0quacin.cfg - sisexp.cfg is in the ../PIC/EXPER directory (input values: 37) !!!! This file has two executables: !!!! xcloud: electron-cloud problems oriented - it has NPR=300000 NEL=100000 NBIN=50 ximped: impedance problems oriented - it has NPR=1000000 NEL=1 NBIN=70 Routine new_fraction is introduced, in which 500 macroelectrons are newly initialized at each interaction slice-cloud. This is done only when a solenoid field is acting. It models the fact that electrons in the solenoid field are actually moving longitudinally. We take these numbers: electron cooler section 3m long, bunch 5m long, electron velocity is 0.145c and proton (or D+) velocity 0.947c. We easily find Deltat_b = 5/0.947c = 17.6ns -> during one full bunch pass the electrons will have advanced by Deltas_e = 0.145c Deltat_b = 0.765m. Therefore 1/4 of the total cooler length. This means that at the end of the interaction, 1/4 of the cloud keeps actually a pregressively "shorter memory" of the previous interaction. If after 50 slices 1/4 of the macroelectrons (25000) will have this partial memory, the effect can be achieved if at each slice we regenerate 25000/50 = 500 electrons. Last improvement has been made with Daniel Schulte, he has basically made the PIC field calculation faster by changing the modules fourtrans with improved versions (fourtrans2 and fourtrans3) which link to the file fourtrans2.c, included in the present program, and to routines from the fftw library which has also been copied in the fftw directory from my home directory (7/12/2001). As the amplitude detuning has been also introduced, there is a new switch i_oct which sets it working or not. It is the last entry in the input file (line 38). If amplitude detuning is turned on, its parametrization is given in the 'init_values' routine of this code and actually reflects the parametrization expected for SPS (from Rogelio's note). 11/04/2002 We have introduced a new option: i_dip=3. The electrons move in the strong combined functions magnetic field, like in PS. With the assumption of strong field, we have simply confined the electron motion on trajectories that follow the field lines. 29/04/2002 The new flag i_incoh has been introduced to study the instability free of the monopole mode as suggested by Evgueni Perevedentsev. The bunch phase space centroid is subtracted turn by turn and slice by slice from the particle phase space coordinates to concentrate only on the e-cloud induced emittance growth The flag elflag defines the electron distribution: 1- Rectangular 2- Elliptical 3- Stripe-like rectangular 4- Parabolic Number of entries of the input file: 40 !!! new entries to be added: Coherent_centroid_motion_(0->off_1->on): Electron_distribution(1->Rect_2->Ellipt_3->Stripe_4->Parab): for most updated input file look into: /afs/cern.ch/user/g/grumolo/Elcloud/HEADTAIL/DATA/PIC/PEREVED/eugeni*.cfg /afs/cern.ch/user/g/grumolo/Elcloud/HEADTAIL/DATA/PIC/PS/COMBIN/cfcurr8ott.cfg 3/06/2002 new flag i_coupl to have possibly linear xy-coupling in the machine. The parameter key_xy is also defined to tune the strength of the coupling. To be input on last line of the input file. Number of entries of the input file: 42 !!! new entries to be added: Linear_coupling_switch(1->on_0->off): 1 Linear_coupling_coefficient_[1/m]: 0.015 17/06/2002 ->new<- to take into account horizontal dispersion: the value is input in the variable dispx (43rd line in the input file !!), and the contribution of dispersion comes in when the interaction between bunch and cloud is calculated. This introduces a x-z correlation first, and obviously it changes the oscillation frequency of the electrons because it changes the horizontal rms-size value. Also the 2-stripes distribution is included for electrons, which takes into account how electrons are actually distributed inside a dipole field. Number of entries of the input file: 45 !!! new entries: Average_dispersion_function_in_the_ring_[m]: 2.0 Position_of_the_stripes_[units_of_sigmax]: 3.0 Width_of_the_stripes_[units_of_sigmax]: 0.5 and the line on the electron distribution becomes: Electron_distrib_(1->Rect_2->Ellip_3->[1_strp]_4->[2_strp]_5->Parab): 08/08/2002 A kick in the longitudinal direction has been introduced, after having changed the bining in such a way that the code can deal with bunch dipole oscillations in z and variations of sigma_z during the evolution (to model for example a debunching beam). New line in the input file is therefore needed: Kick_in_the_longitudinal_direction_[m]: 0.03 If isyn==2 we also give the possibility to the bunch to de-bunch. Synchrotron motion is off and the particles follow their longitudinal straight trajectories according to their initial momenta. 13/08/02 Changing the transverse impedance: the transverse force due to the broad band impedance has been adapted to a case of flat chamber (the two limit cases are now possible: round chamber i_pipe=0 (a=b) or very flat chamber i_pipe=1 (a>>b)) this introduces one more line in the input file to set i_pipe, but also two lines of the input need to be taken out because we have one single set of parameters for the trasverse impedance, which will be used both for horizontal and vertical kicks + the flat chamber correction. OLD: Switch_for_wake_fields: 0 Res_frequency_of_broad_band_resonator_[GHz]: 1.3 Horizontal_quality_factor: 1. Vertical_quality_factor: 1. Shunt_impedance_x_[MOhm/m]: 5. Shunt_impedance_y_[MOhm/m]: 10. NEW: Switch_for_wake_fields: 0 Switch_for_pipe_geometry_(0->round_1->flat): 1 Res_frequency_of_broad_band_resonator_[GHz]: 1.3 Transverse_quality_factor: 1. Transverse_shunt_impedance_[MOhm/m]: 10. 15/08/02 It is possible to store the bunch shape every n_diag turns. This makes sense when we are in debunching mode or in bunched mode with a bunch not well matched to the bucket. One more input parameter: Number_of_turns_between_two_bunch_shape_acquisitions: 20 Longitudinal wake field has been introduced due to a resonator. We want to model the parasitic effect of a cavity on a single bunch. We can neglect the turn-to-turn effect, since the quality factor is such (~150) that the field has completely decayed when the bunch goes again through the same cavity where it was generated (Q=5000 is at the limit if this approximation for a frequency around 200 MHz). Again we need to input three numbers to model this resonator: Res_frequency_of_broad_band_resonator_[GHz]: 1.3 Transverse_quality_factor: 1. Transverse_shunt_impedance_[MOhm/m]: 10. Res_frequency_of_longitudinal_resonator_[MHz]: 200 Longitudinal_quality_factor: 140. Longitudinal_shunt_impedance_[MOhm]: 5. 20/08/02 Fields with conducting boundary conditions on a rectangular box can be calculated instead of the simple periodic condition in x and y. The method consists in unfolding the charge distribution on a double-sized grid on which the image charges are placed. The routine image_unfold has been introduced to accomplish this. One more input needed: 08/05/04 FZ, introduce two new isyn options: isyn=6 for TESLA DR where momentum compaction & rf are localized at one point of the ring isyn=7 for CLIC, NLC, LHC where rf is localized at one point of the ring isyn=8 for LHC Boundary_condition_(0->Periodic_1->Conducting): 1 */ /* Include all facilities */ #include #include //#include #include //#include #include #include #include /* with this package one can define complex variables, if needed; for instance, complex z */ /* Definition of constants */ /* Physical constants.... */ #define PI 3.14159265358979 #define C 2.99792458e8 /* velocity of light [m/s] */ #define E 1.6021892e-19 /* electron charge [Cb] */ #define MU 1.66055e-27 /* proton mass [kg] */ #define ME 9.109534e-31 /* proton-positron mass [kg] */ #define EPS0 8.8542e-12 /* dielectric constant [C^2/Jm] */ #define RP 1.5458775e-18 /* classical proton radius [m] */ #define RE 2.817938e-15 /* classical electron radius [m] */ #define I0 31.356e6 /* ... for the tune shift.. [A] */ #define ELMS 9.109534e-31 /* electron mass [Kg] */ /* Parameters for the computation... */ #define NPR 300000 /* Number of macroparticles (bunch) to be used */ #define NPR1 90000 /* Maximum number of macroparticles in a slice */ #define NEL 100000 /* Number of macroelectrons (cloud) to be used */ #define NBIN 70 /* Number of 'slices' in which each single bunch is sub-divided */ #define NDC 128 /* Number of cells to produce a distribution function */ #define NR_END 1 #define FREE_ARG char* /* type definition */ typedef struct /* GRID */ { int n_cell_x,n_cell_y; int timestep; double delta_x,delta_y; double offset_x,offset_y; double min_x,max_x,min_y,max_y; double cut_x,cut_y,cut_z; double step,n_macro_1,n_macro_2,step1,step2,scal_step[2]; float rho_x_1,rho_y_1,rho_sum_1,rho_x_2,rho_y_2,rho_sum_2; double rho_factor; int n_m_1,n_m_2; double *rho1,*rho2,*phi1,*phi2,*dist,*temp,*potenziale; int integration_method,distribution,interpolation; } GRID; /* Definition of the external variables */ FILE *bunch_pr , *headtail_pr , *prot_phase , *prot_inphase , *prot_ini , *ele_ini , *samp_prot , *samp_prot1 , *samp_prot2 , *samp_prot3 , *samp_prot4 , *samp_prot5 , *samp_prot6 , *prot_distr , *track_pr , *beta_adv , /*ele-May04 */ *el_traj , /* gio-May04 */ *el_traj2 , *el_traj3 , *halo_p , *campo_dat , *campo0_dat , *campo1_dat , *sey_file , *el_diagn , *froz_pot1 , *froz_pot2 , *elwall_ene ; char filename[80] ; long iind[NBIN] , /* iind[i] gives the number of macroparticles in the i-th slice of bunch */ indexp[NBIN][NPR1] , /* in the bunch slicing, indexp[i][j] contains the original bunch index of the particle that has index j in the i-th slice */ hmain , /* down to mmain -> indices used in main for loops */ imain , jmain , kmain , lmain , mmain , contax , /* ele-May04 */ mark[NPR] , kin , /* gio-May04 */ min , elflag , /* flag for the e-distribution: see above */ i_symm_el , /*flag to allow a symmetric distrib of electrons */ i_symm_pr , /* and protons */ i_froz , i_distr_prot , /*if =1, read from a file 'name.pini' the init distrib*/ i_pot_froz , it , /* index used for the main cycle over the total number of steps: turns * kicks */ ind , nturn , /* number of simulated turns -> input parameter */ nkick , /* number of kicks per turn -> input parameter */ nstep , /* total number of steps: nturn * nkick */ isyn , /* flag for the synchrotron motion: input parameter */ ipr_pos , /* flag for proton/positron bunch: input parameter */ imesize , /* scale factor for macroelectron size: input parameter */ ielsizex , /* multiplication factor for the e- cloud x-size: input parameter */ ielsizey , /* multiplication factor for the e- cloud y-size: input parameter */ nsigmaz , /* number of sigma_z to be taken into account to represent the Gaussian bunch (+/- nsigmaz*szt) */ ili , /* flag for the function ffrank: it has to be =! 1 */ iwake_flag , /* flag for the wake field: input parameter */ ishift , /* flag to activate the tune shift option */ sl_ind , /* it is used as slice index when the incoherent space charge tune shift is added */ i_method , /* flag to choose the field calculation method */ i_rot , /* switch for the space charge tune rotation */ flag4print , /* used to print only once the tune shift due to SC */ i_space , /* Flag for the space charge forces */ i_oct , /* Flag for the action dependent tunes */ i_dip , /* flag to switch off the electron horizontal motion in order to simulate their motion in a strong vertical dipole field */ i_incoh , /* flag to investigate on the monopole mode as suggested by Evgueni Perevedentsev */ i_coupl , /* flag to switch on and off linear coupling */ i_sext , /* flag to switch on and off sextupolar kick */ i_loss , /* switch to activate beam losses */ i_pipe , /* flag for pipe shape in effective wake calculation 0 -> circular (a=b) 1 -> flat (a>>b) 3 -> resistive_wall 5 ->collim_geometric_wake 6 ->collim_resistive_wake */ i_bound , /* switch for boundary conditions 0 -> open space 1 -> rectangular box */ i_ecool , /* switch for an electron cooler tuned on the beam 0 -> electron cloud or mis-tuned electron cooler 1 -> electron cooler */ i_phas_adv , /* switch for random phase advance */ i_damp , /* ele-May04; switch for the damper */ n_cool , /* number of sections in which the e-cooler gets subdivided */ n_diag , /* every how many turns the bunch shape is stored */ hh , /* Harmonic number of sinusoidal bucket */ npr_eff , /* number of bunch particles lost turn by turn */ largeemix , largeemiy , /* gio-May04; coasting beam option */ i_coast , /* switch coasting(1)/bunched(0) beam */ i_ang , /* switch for angular dependence of SEY */ nelpc , /* number of macroelectrons per coasting */ ntemp , /* temporary number of macroelectrons at each turn */ npert , /* harmonic number of perturbation on coasting beam */ //i_mauro , /* gio-May04; switch for */ i_beta ; /* gio-May04; switch for different beta functions */ double ycoff; double xpr[NPR] , /* horizontal coordinate for protons or positrons */ ypr[NPR] , /* vertical coordinate for protons or positrons */ xpp[NPR] , /* dx/ds for protons or positrons */ ypp[NPR] , /* dy/ds for protons or positrons */ sz[NPR] , /* longitudinal coordinate for protons or positrons */ dp[NPR] , /* relative momentum spread for protons or positrons */ xe[NEL] , /* horizontal coordinate for electrons */ ye[NEL] , /* vertical coordinate for electrons */ xpe[NEL] , /* dxe/dt for electrons */ ype[NEL] , /* dye/dt for electrons */ qe[NEL] , /* gio-May04; charge of macroelectrons */ //xe0[NEL] , //ye0[NEL] , zs[NBIN] , /* longitudinal position of each bunch slice */ npr[NBIN] , /* number of particles in each slice */ sxs[NBIN] , /* horizontal rms value of each slice */ sys[NBIN] , /* vertical rms value of each slice */ xs[NBIN] , /* horizontal mean offset of each slice */ ys[NBIN] , /* vertical mean offset of each slice */ xps[NBIN] , /* horizontal mean dx/ds of each slice */ yps[NBIN] , /* vertical mean dy/ds of each slice */ //indx[NDC] , /* grid points for the horizontal distribution of electrons */ //hde[NDC] , /* horizontal distribution of electrons after having been kicked */ //indy[NDC] , /* grid points for the vertical distribution of electrons */ //vde[NDC] , /* vertical distribution of electrons after having been kicked */ indxp[NDC] , /* grid points for the bunch long. distribution */ hdp[NDC] , /* Bunch longitudinal distribution */ //indyp[NDC] , /* grid points for the vertical distribution of the bunch particles */ //vdp[NDC] , /* vertical distribution of bunch particles before interacting with the particles */ //indxpp[NDC] , /* grid points for the distribution of horizontal momenta of the bunch particles */ //hpdp[NDC] , /* distribution of horizontal momenta of the bunch particles before interacting with the electrons */ //indypp[NDC] , /* grid points for the distribution of vertical momenta of the bunch particles */ //vpdp[NDC] , /* distribution of vertical momenta of the bunch particles before interacting with the electrons */ vprx[NPR1] , /* temporary vector that contains all the x's of the bunch particles in a slice */ vpry[NPR1] , /* temporary vector that contains all the y's of the bunch particles in a slice */ vprxp[NPR1] , /* temporary vector that contains all the xp's of the bunch particles in a slice */ vpryp[NPR1] , /* temporary vector that contains all the yp's of the bunch particles in a slice */ *betarrayx , /* gio-May04; different beta at different IPs */ *betarrayy , /* now the declaration of the input parameters starts.... it goes up to nus = synchrotron tune */ /* gio-May04; coasting beam */ emax0 , /* standard deviation of the distribution of secondary electrons */ enestep , /* energy step in the bining for electrons hitting the beam pipe */ enedis[500] , /* array for the energy distribution of the electrons on the pipe wall */ qt_imp , /* total number of electrons on the pipe wall */ sz0 , /* [m] initial rms-bunch length */ dp0 , /* initial rms-momentum spread */ sx0 , /* [m] initial rms hor. beam size */ sx0_cd , /* [m] rms hor. beam size dispersion corrected */ sy0 , /* [m] initial rms vert. beam size */ npr0 , /* number of particles in the bunch */ betax0 , /* [m] average hor. beta function around the ring */ betay0 , /* [m] average vert. beta function around the ring */ dispx , /* [m] average dispersion function in the ring */ dispx2 , /* [m] average dispersion function at sextupoles */ dcloud , /* average number of electrons per m^3 in the ring */ circ , /* [m] ring circumference */ gammar , /* relativistic gamma of the bunch particles */ htune , /* horizontal tune of the machine */ vtune , /* vertical tune of the machine */ csix , /* horizontal chromaticity of the machine */ csiy , /* vertical chromaticity of the machine */ theta_coll , /* these three variables are needed for the */ b_coll , /* collimator wake field (resistive wall) */ h_coll , gainx , /* ele-May04; damper */ gainy , noisex , noisey , dampx , dampy , randomnumber , qtot , /* gio-May04; contains the volume density of electrons */ q_secx , /* second order horizontal chromaticity */ q_secy , /* second order vertical chromaticity */ alpha , /* momentum compaction factor of the machine */ nus , /* synchrotron tune */ xkick , /* [m] Amplitude of the horizontal kick in t=0 */ ykick , /* [m] Amplitude of the vertical kick in t=0 */ bsol , /* [T] Strength of the solenoid field */ bdip , /* [T] Strength of the dipole field in a combined functions magnet (like in the PS) */ bgrad , /* [T/m] Gradient of the magnetic field in a combined functions magnet (like in the PS) */ /* gio-May04 */ strpos , /* position of the stripes in units of sigma_x */ strsig , /* width of the stripes in units of sigma_x */ strback , /* fraction of electron to be distributed uniformely in the stripes (ele 11/04)*/ press , /* [nTorr] Pressure of gas in vacuum chamber */ crse , /* [MBarn] Ionization cross section */ pleff , /* Electron yield for proton/ion loss */ plppm , /* lost protons per proton and per meter */ rpipex , /* horizontal semiaxis of the beam pipe: gives a x-size to the e- cloud */ rpipey , /* vertical semiaxis of the beam pipe: gives a y-size to the e- cloud */ gridextx , /* Horizontal extension of the grid for the fields PIC calculation */ gridexty , /* Vertical extension of the grid for the fields PIC calculation */ rmscloudx , rmscloudy , eta , /* slip factor: alpha - 1/gammar^2 */ omega0 , /* bunch revolution frequency */ omegas , /* synchrotron frequency */ /* ele-May04 */ mass , /* these are used if the particles are ions and they*/ charge , /*are defined in the subroutine 'init_values' */ factor1 , /* auxiliary parameter for the force calculation */ factor2 , /* auxiliary parameter for the force calculation */ fscale1 , /* auxiliary parameter for the force calculation */ fscale2 , /* auxiliary parameter for the force calculation */ /* gio-May04 */ fscalesp_ch , /* auxiliary parameter for the space charge force /* calculation for the electrons */ ee_neg , /* auxiliary parameter for the space charge force /* calculation for the electrons */ eel , /* auxiliary parameter for the force calculation */ epr , /* auxiliary parameter for the force calculation */ wakefac , /* factor for the wake field kick */ tstep , /* time step during the interaction bunch-cloud */ dlstep , /* s-distance between two kick points */ tstepp , /* time step between two kick points */ qp0 , /* number of bunch particles in one macroparticle */ qe0 , /* number of electrons in one macroelectron */ p0 , /* average momentum of the bunch particles */ sx2 , /* finite macroelectron horizontal size */ sy2 , /* finite macroelectron vertical size */ key_xy , /* [1/m] strength of linear coupling */ k_2s , /* [1/m^2] strength of sextupole */ cxa , /* cxa -> ssa: transport parameters for the phase */ sxa , /* space coordinates of the bunch particles (linear */ cya , /* transport + chromaticity kick in the transverse */ sya , /* plane, linear synchrotron motion longitudinally) */ gammaadv0 , gammaadv , /* #####################*/ hadv , vadv , *phadv , /*ele-May04 */ csa , ssa , /* gio-May04 */ aux_fix[NEL] , /* auxiliary vector to store the space charge field from the electrons in the solenoidal field */ aux_fiy[NEL] , /* auxiliary vector to store the space charge field from the electrons in the solenoidal field */ dtunxmax , /* maximum space charge horizontal tune shift in the bunch */ dtunymax , /* maximum space charge vertical tune shift in the bunch */ dtunx , dtuny , lambda , /* local variable to store the exponential factor for the tune shift modulation along the bunch */ cent_x , /* cent_*: local variables to store the centroid */ cent_y , /* position and slope of each slice used for the */ cent_xp , /* incoherent space charge tune spread */ cent_yp , xx1 , /* xx1->yyp1: local variables used in the particles */ xxp1 , /* transport around the ring */ yy1 , yyp1 , epsx , /* single particle horizontal emittance variable */ epsy , /* single particle vertical emittance variable */ qpxx , /* Q_x = Q_x0 + eps_x * qpxx + eps_y * qpxy */ qpyy , /* Q_y = Q_y0 + eps_x * qpxy + eps_y * qpyy */ qpxy , bin_frac , /* fraction of bunch particles that are binned */ xave , /* average horizontal offset of the beam */ yave , /* average vertical offset of the beam */ zave , /* average longitudinal offset of the beam */ dpmean , /* average long. momentum offset of the beam */ xpave , /* average beam offset in dx/ds */ ypave , /* average beam offset in dy/ds */ sxsize , /* square of horizontal rms bunch size */ sysize , /* square of vertical rms bunch size */ szsize , /* longitudinal rms bunch size */ sxpsize , sypsize , timep , /* time at which the kick occurs */ xemit , /* horizontal emittance */ yemit , /* vertical emittance */ invarx , /* Courant-Snyder horizontal invariant */ invary , /* Courant-Snyder vertical invariant */ emipx , emipy , corryz , /* y-z correlation */ npr1 , /* number of protons contained in one slice */ xoff , /* slice horizontal off-center */ yoff , /* slice vertical off-center */ sx , /* slice horizontal rms size */ sy , /* slice vertical rms-size */ xta , yta , zstep , /* length of a bunch slice */ zi , /* longitudinal negative coordinate (head of the bunch is zi = 0) for wake field */ x1 , /* macroelectron x position at which the force from each slice is calculated */ ips1 , /* macroelectron y position at which the force from each slice is calculated */ pre , /* coefficient for the y-component of the force */ pim , /* coefficient for the x-component of the force */ kick_xe , /* velocity x-kick from each slice on each macroelectron */ kick_ye , /* velocity y-kick from each slice on each macroelectron */ xof2 , /* macroelectron x-position */ yof2 , /* macroelectron y-position */ x2 , /* x-position of the bunch particle on which the macroelectron acts */ y2 , /* y-position of the bunch particle on which the macroelectron acts */ xof3 , /* initial x-position of the macroelectron - used to clean out fluctuations */ yof3 , /* initial y-position of the macroelectron - used to clean out fluctuations */ xta2 , yta2 , pre2 , /* coefficient for the x-component of the force */ pim2 , /* coefficient for the y-component of the force */ kick_x , /* x-kick on each bunch particle */ kick_y , /* y-kick on each bunch particle */ kick_z , /* z-kick from on each bunch particle */ z0_kick , /* z-kick from each macroelectron on each bunch particle */ kick_xp_sc , /* dx/ds-kick from each bunch slice on each bunch particle */ kick_yp_sc , /* dy/ds-kick from each bunch slice on each bunch particle */ x0 , /* x-coordinate of a bunch particle after having been kicked, before transport to the next kick */ xp0 , /* dx/ds of a bunch particle after having been kicked and before transport to the next kick */ ips0 , /* y-coordinate of a bunch particle after having been kicked, before transport to the next kick */ yp0 , /* dy/ds of a bunch particle after having been kicked and before transport */ szz0 , /* longitudinal coordinate of a bunch particle after having been kicked, before transport to the next kick */ dpp0 , /* momentum spread of a bunch particle after having been kicked, before transport to the next kick */ r11 , /* r11 -> r44, coefficients of the transport matrix for the transverse phase space coordinates between two kicks */ r12 , r21 , r22 , r33 , r34 , r43 , r44 , merit , /* quality factor of the broad band impedance */ freqr , /* resonant frequency of the broad band resonator to be expressed in GHz (x and y assumed equal) */ zetat , /* shunt impedance of the braod band resonator to be expressed in MOhm/m */ meritz , /* quality factor of the longitudinal impedance */ freqrz , /* resonant frequency of the longitudinal resonator to be expressed in MHz */ rshz , /* shunt impedance of the braod band resonator to be expressed in MOhm */ omegar , /* auxiliary parameters for transverse wake */ omegabar , alpha , omegarz , /* auxiliary parameters for longitudinal wake */ omegabarz , alphaz , voltm , /* Maximum voltage in sinusoidal bucket */ wrf , /* Radio-frequency of sinusoidal bucket */ coeff3 , /* Auxiliary parameters for longitudinal motion */ coeff4 , /* in sinusoidal bucket */ coeff5 , coeffspch , /* coefficient for longitudinal space charge in a sinusoidal bucket */ k1 , k2 , k3 , k4 , kp1 , kp2 , kp3 , kp4 , /* gio-May04 */ cff1x , /* coefficents for the transport matrices of the */ cff1y , /* bunch particles between points with different */ cff2x , /* beta functions */ cff2y , xpert , /* x-perturbation amplitude on coasting beam */ ypert , /* y-perturbation amplitude on coasting beam */ yim , /* maximum secondary emission yield */ yemax , /* energy for the max. secondary emission yield */ dl_cool ; /* electron cooler length */ /* For sake of clarity we specify here that the broad band model that we've assumed considers an impedance (omegar = 2*PI*freqr*10^(-9)): Z = omegar/omega * zetat/[1+i merit (omegar/omega - omega/omegar)]. If the impedance is: Z = c/omega * R_s/[1+i merit (omegar/omega - omega/omegar)], then one has to convert R_s into our input parameter by means of: zetat=c/omegar*R_s zetat [Ohm/m] and R_s [Ohm/m^2] In the longitudinal plane: Z|| = rshz/[1+i meritz (omegarz/omega - omega/omegarz)] */ /* Global declarations... function that evaluates the electric mutual force between bunch particles and electrons has to be included */ extern void ffrank (double*, double*, double*, double*, double*, double*, double*, double*, long*); /* begin gio-May04 */ void nrerror(char error_text[]) { fprintf(stderr,"ERROR...\n"); fprintf(stderr,"%s\n",error_text); fprintf(stderr,"...now exiting...\n"); exit(1); } double *dvector(long n1, long nh) /* allocate a double vector with subscript range v[n1..nh] */ { double *v; v=(double *)malloc((size_t) ((nh-n1+1+NR_END)*sizeof(double))); if (!v) nrerror("allocation failure in dvector()"); return v-n1+NR_END; } void free_dvector(double *v, long n1, long nh) /* free a double vector allocated with dvector() */ { free((FREE_ARG) (v+n1-NR_END)); } /* end gio-May04 */ /****************************************************************************** *** Subroutine : read_data *** *** Effect : Define main parameters by reading data from specified *** *** configuration-file *** *** Parameters : none *** *** Gbl var used : *** *** Gbl var effect : *** *** Constants used : none *** *** Subrout. used : none *** ******************************************************************************/ read_data () //{ char cfg_filename[30], dummy_string[64]; // FILE *cfg_file_ptr; /* gio-May04; reads different beta */ { char cfg_filename[20], dummy_string[64], beta_filename[20]; FILE *cfg_file_ptr, *beta_file; long k_cou, condition; //printf ("\n\n Please specify name (without extension) of desired") ; //printf ("\n configuration-file : ") ; //scanf ("%s",filename) ; // strcpy (filename,"ecsdampA"); strcpy (filename,"ecsps_str"); // strcpy (filename,"ec_merge2"); strcpy (cfg_filename,filename); strcat (cfg_filename,".cfg"); cfg_file_ptr = fopen(cfg_filename,"r"); if (cfg_file_ptr == NULL) { printf ("\n\n Configuration-file does not exist.\n") ; // end () ; } exit(2); } fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&ipr_pos); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&dcloud); //printf ("dcloud=%lf\n",dcloud); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&npr0); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&betax0); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&betay0); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&sz0); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&sx0); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&sy0); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&dp0); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&nus); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&alpha); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&circ); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&gammar); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&nkick); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&nturn); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&ielsizex); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&ielsizey); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&nsigmaz); /*ele-May04 */ fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&htune); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&vtune); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&csix); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&csiy); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&isyn); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&imesize); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&iwake_flag); //printf ("iwake_flag=%d\n",iwake_flag); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_pipe); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&freqr); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&merit); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&zetat); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&freqrz); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&meritz); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&rshz); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&ishift); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_method); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_dip); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&xkick); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&ykick); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_space); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_rot); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&bsol); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_oct); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_incoh); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&elflag); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_coupl); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&key_xy); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&dispx); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&strpos); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&strsig); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&z0_kick); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&n_diag); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&voltm); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&hh); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_sext); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&k_2s); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&dispx2); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_loss); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&q_secx); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&q_secy); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_bound); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_phas_adv); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_ecool); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&dl_cool); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&theta_coll); /*these variables are used if i_pipe=5*/ fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&b_coll); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&h_coll); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_damp); /*ele-May04; damper */ fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&gainx); // printf ("gainx=%lf\n",gainx); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&noisex); // printf ("noisex=%lf\n",noisex); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&gainy); // printf ("gainy=%lf\n",gainy); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&noisey); // printf ("noisey=%lf\n",noisey); /* begin gio-May04; coasting beam */ fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_coast); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&nelpc); //printf ("nelpc=%d\n",nelpc); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&press); //printf ("press=%lf\n",press); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&crse); //printf ("crse=%lf\n",crse); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&pleff); //printf ("pleff=%lf\n",pleff); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&plppm); //printf ("plppm=%lf\n",plppm); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_ang); //printf ("i_ang=%d\n",i_ang); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&xpert); //printf ("xpert=%lf\n",xpert); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&ypert); printf ("ypert=%lf\n",ypert); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&npert); printf ("npert=%d\n",npert); //new variable to implement many kicks per turn with one electric field only // fscanf (cfg_file_ptr,"%s",dummy_string); // fscanf (cfg_file_ptr,"%d",&i_mauro); // printf ("i_mauro=%d\n",i_mauro); //maximum secondary emission yield fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&yim); printf ("sey_max=%lf\n",yim); //energy at which the maximum SEY occurs (E_{max}) fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%lf",&yemax); printf ("E_max=%lf\n",yemax); fscanf (cfg_file_ptr,"%s",dummy_string); fscanf (cfg_file_ptr,"%d",&i_beta); printf ("i_beta=%d\n",i_beta); /* end gio-May04 */ /* begin gio-May04 */ /* if(i_ecool==1) { i_dip = 2; printf("\n\n\t --> i_dip set to 2, E-cooler is tuned on the beam\n"); } if(i_dip==2) { elflag = 2; if(i_ecool==0) printf("\n\n\t elflag set to 2, E-cooler and beam are mis-tuned\n"); else printf("\n\n\t elflag set to 2, E-cooler is tuned on the beam\n"); } */ if(i_ecool==1 || i_ecool==2) { i_dip = 2; elflag = 2; printf("\n\n\t --> i_dip set to 2, E-cooler mode\n"); if(i_ecool==2) printf("\n\n\t elflag set to 2, E-cooler and beam are mis-tuned\n"); else printf("\n\n\t elflag set to 2, E-cooler is tuned on the beam\n"); } /* end gio-May04 */ if(i_bound==1) /* gio-May04 */ { elflag = 1; printf("\n\n\t --> elflag set to 1, the beam chamber is rectangular\n"); } /* begin gio-May04; coasting beam */ if(i_coast!=0) { sz0 = circ/4; /* sigma bunch is 1/4 of the chosen circumference */ if (i_coast==1) elflag = 6; else elflag = 7; //printf("elflag = %d", elflag); /* the electrons must be initialized according to residual gas ionization (6) or at the walls from losses (7) */ isyn = 9; /* the coasting (or debunching over all the ring) mode is set */ i_bound = 1; /* conducting boundary conditions are considered and the chamber is assumed to be rectangular */ printf ("\n\n\t\tParameters changed because of coasting beam\n") ; } /* end; gio-May04; coasting beam */ /* begin gio-May04, different beta fuctions */ if(nkick>1 && i_beta==1) { strcpy (beta_filename,filename); strcat (beta_filename,".beta"); beta_file = fopen(beta_filename,"r"); if (beta_file == NULL) { printf ("\n\n File with beta's does not exist.\n") ; exit(2); } betarrayx=dvector(0,nkick-1); betarrayy=dvector(0,nkick-1); printf ( beta_filename); k_cou = 0; do {condition = fscanf (beta_file, "%lg %lg\n", &betarrayx[k_cou],&betarrayy[k_cou]); printf ("%d %f %f \n", k_cou, betarrayx[k_cou], betarrayy[k_cou]); ++k_cou; } while (condition != EOF); if(k_cou!=nkick+1) { printf ("\n\n Number of kicks different from number of assigned beta functions !!\n") ; exit(2); } } //if(i_mauro==1) // { i_method = 2; // i_coast = 0; // } /* end: gio-May04 */ //ele-Mar05: in order to have the average beta0 consistent with the smooth focusing approx, it is enought to set beta0=0 in the input file: if(betax0==0) betax0=circ/(2*PI*htune); if(betay0==0) betay0=circ/(2*PI*vtune); printf ("\n\n\t\tConfiguration-file loaded successfully.\n") ; } /****************************************************************************** *** Subroutine : init_values *** *** Effect : Initialize all parameter values, necessary for *** *** calculation. *** *** Parameters : none *** *** Gbl var used : *** *** Gbl var effect : *** *** Constants used : *** *** Subrout. used : none *** ******************************************************************************/ init_values () { double area,esigma,ltest,elocal; int im; eta = alpha - 1/gammar/gammar; omega0 = C/circ*2*PI; omegas = nus*omega0; tstep = 2.*(double)nsigmaz*sz0/(double)NBIN/C; factor1 = RE*C*sqrt(2*PI)/tstep; fscale1 = factor1/sqrt(2*PI); dlstep = circ/(double)nkick; tstepp = dlstep/C; /* gio-May04 */ fscalesp_ch = RE*C*C/dlstep; for(im=0;im 0.5) { fraction1 = 1.5 - scaled + channel1 ; channel2 = channel1 + 1 ; fraction2 = 1.0 - fraction1 ; } else { fraction1 = scaled - channel1 + 0.5 ; channel2 = channel1 - 1 ; fraction2 = 1.0 - fraction1 ; } if (channel1 >= 0 && channel1 < ncell) distri[channel1] += fraction1 ; if (channel2 >= 0 && channel2 < ncell) distri[channel2] += fraction2 ; } for (j=0 ; j 0.5) { fraction1 = 1.5 - scaled + channel1 ; channel2 = channel1 + 1 ; fraction2 = 1.0 - fraction1 ; } else { fraction1 = scaled - channel1 + 0.5 ; channel2 = channel1 - 1 ; fraction2 = 1.0 - fraction1 ; } if (channel1 >= 0 && channel1 < ncell) distri[channel1] += fraction1 ; if (channel2 >= 0 && channel2 < ncell) distri[channel2] += fraction2 ; } for (j=0 ; j 0.0) { if (tan(r2)*rpipex > rpipey) { if (cos(r2) > 0.0) { ye[i] = rpipey - 1.e-7; xe[i] = (rpipey - 1.e-7)/tan(r2); } else { ye[i] = -(rpipey - 1.e-7); xe[i] = -(rpipey - 1.e-7)/tan(r2); } } else { if (cos(r2) > 0.0) { xe[i] = rpipex - 1.e-7; ye[i] = (rpipex - 1.e-7)*tan(r2); } else { xe[i] = -(rpipex - 1.e-7); ye[i] = -(rpipex - 1.e-7)*tan(r2); } } } if (tan(r2) < 0.0) { if (fabs(tan(r2))*rpipex > rpipey) { if (cos(r2) > 0.0) { ye[i] = -(rpipey - 1.e-7); xe[i] = (rpipey - 1.e-7)/fabs(tan(r2)); } else { ye[i] = (rpipey - 1.e-7); xe[i] = -(rpipey - 1.e-7)/fabs(tan(r2)); } } else { if (cos(r2) > 0.0) { xe[i] = rpipex - 1.e-7; ye[i] = -(rpipex - 1.e-7)*fabs(tan(r2)); } else { xe[i] = -(rpipex - 1.e-7); ye[i] = (rpipex - 1.e-7)*fabs(tan(r2)); } } } if (fabs(ye[i]) >= (rpipey - 2.e-7)) { dn1pl = 0.; dn2pl = -ye[i]/fabs(ye[i]); } if (fabs(xe[i]) >= (rpipex - 2.e-7)) { dn1pl = -xe[i]/fabs(xe[i]); dn2pl = 0.; if (fabs(ye[i]) >= (rpipey - 2.e-7)) { dn1pl = -xe[i]/fabs(xe[i])/sqrt(2.); dn2pl = -ye[i]/fabs(ye[i])/sqrt(2.); } } emax0elpl = 10.0; sigmaelpl = 5.0; enelpl = (sqrt(-2*log(rand ()/(double)RAND_MAX) )*cos(2*PI*rand ()/(double)RAND_MAX))*sigmaelpl + emax0elpl; if (enelpl < 0.0) enelpl = -enelpl; velpl = sqrt(2.0*enelpl*E/ELMS); sinel = sqrt(rand ()/(double)RAND_MAX); cosel = sqrt(1.0 - sinel*sinel); dphiel = 2.0*PI*rand ()/(double)RAND_MAX; xpe[i] = dn1pl*cosel - sinel*dn2pl*cos(dphiel); ype[i] = dn2pl*cosel + sinel*dn1pl*cos(dphiel); xpe[i] *= velpl; ype[i] *= velpl; //fprintf(track_pr,"%13.8e %13.8e %13.8e \n",xe[i],ye[i],qe[i]); //printf("%13.8e %13.8e %13.8e %13.8e %13.8e \n",xe[i],ye[i],xpe[i],ype[i],qe[i]); } //fprintf(track_pr,"\n"); //fprintf(track_pr,"\n"); break; } return 0; } /* end: gio-May04 */ /****************************************************************************** *** Subroutine : initialization *** *** Effect : Initialize particle and velocity distributions. *** *** Parameters : none *** *** Gbl var used : *** *** Gbl var effect : *** *** Constants used : *** *** Subrout. used : none *** ******************************************************************************/ initialization () { double xsum, ysum, xpsum, ypsum, xsum2, ysum2, xeave, yeave, u, v, s, indsz[NDC], lgds[NDC], inddp[NDC], lgmm[NDC]; long i, j, l, part_num; char pdist_name[80] ; FILE *pdist_file; xsum = 0.0; ysum = 0.0; xpsum = 0.0; ypsum = 0.0; xsum2 = 0.0; ysum2 = 0.0; /* - ele Feb'05: if i_symm=1 the initial particle distrib is symmetric (and there is no rediffusion of the electrons), normally has to be set =0 - ele Feb'05: if i_froz=1 the potential in which the protons move is eveluated only at the first interaction point, normally has to be set =0 -> in this case in the 'bining' subroutine (0,sz0) are used, instead of (zave,szt) - ele March'05: if i_distr_prot=1, the initial proton distribution is read from a file 'name.pini', normally has to be set =0 - ele March'05: if i_pot_froz=1, the frozen potential is read from a file 'name.frz1', normally has to be set =0 */ i_symm_el=0; i_symm_pr=0; i_froz=0; i_distr_prot=0; i_pot_froz=0; if (i_symm_el==0) { switch(elflag) { case 1: for (i=0; i= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 22: //ele-Apr'05 gaussian cloud for (i=0; i= 1.0) ; xe[i] = rmscloudx*u*sqrt(-2.0*log(s)/s); ye[i] = rmscloudy*v*sqrt(-2.0*log(s)/s); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 3: for (i=0; i 0.5) //xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = (strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } else //xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = -(strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 44: //uniform stripes strback=0.1; for (i=0; i 0.5) xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); else xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 5: parabolic(); for (i=0; i= 1); xpe[i] = 0.0; ype[i] = 0.0; xe[i+1] = - xe[i]; ye[i+1] = ye[i]; xpe[i+1] = 0.0; ype[i+1] = 0.0; xe[i+2] = xe[i]; ye[i+2] = -ye[i]; xpe[i+2] = 0.0; ype[i+2] = 0.0; xe[i+3] = - xe[i]; ye[i+3] = -ye[i]; xpe[i+3] = 0.0; ype[i+3] = 0.0; xeave+= xe[i]+xe[i+1]+xe[i+2]+xe[i+3]; yeave+= ye[i]+ye[i+1]+ye[i+2]+ye[i+3]; } break; case 22: //ele-Apr'05 gaussian cloud for (i=0; i= 1.0) ; xe[i] = rmscloudx*u*sqrt(-2.0*log(s)/s); ye[i] = rmscloudy*v*sqrt(-2.0*log(s)/s); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xe[i+1] = - xe[i]; ye[i+1] = ye[i]; xpe[i+1] = 0.0; ype[i+1] = 0.0; xe[i+2] = xe[i]; ye[i+2] = -ye[i]; xpe[i+2] = 0.0; ype[i+2] = 0.0; xe[i+3] = - xe[i]; ye[i+3] = -ye[i]; xpe[i+3] = 0.0; ype[i+3] = 0.0; xeave+= xe[i]+xe[i+1]+xe[i+2]+xe[i+3]; yeave+= ye[i]+ye[i+1]+ye[i+2]+ye[i+3]; } break; case 3: for (i=0; i 0.5) // xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = (strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } else //xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = (strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; } break; case 44: //uniform stripes strback=0.1; for (i=0; i 0.5) xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); else xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; } break; case 5: parabolic(); break; } // printf("fine electron distrib \n"); } xeave /= (double)NEL; yeave /= (double)NEL; for (i=0; i= 1.0) ; xpr[i] = sx0 * u * sqrt(-2.0*log(s)/s); xpp[i] = sx0/betax0 * v * sqrt(-2.0*log(s)/s); do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; ypr[i] = sy0 * u * sqrt(-2.0*log(s)/s); ypp[i] = sy0/betay0 * v * sqrt(-2.0*log(s)/s); // if(isyn==0 || isyn==1 || isyn==2) if(isyn==0 || isyn==1 || isyn==2 || isyn==3 || isyn==9) /*gio-May04 what is isyn-3? */ { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; sz[i] = z0_kick + sz0 * u * sqrt(-2.0*log(s)/s); dp[i] = dp0 * v * sqrt(-2.0*log(s)/s); } // if(isyn==3) // sz[i] = sz0 * 4.0 * (rand () / (double)RAND_MAX - 0.5); /*gio-May04 what is isyn=3, 9? */ if(isyn==3 || isyn==9) sz[i] = z0_kick + sz0 * 4.0 * (rand () / (double)RAND_MAX - 0.5); if(isyn==4 || isyn==5 || isyn==6 || isyn ==7 || isyn == 8) { do { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; sz[i] = z0_kick + sz0 * u * sqrt(-2.0*log(s)/s); dp[i] = dp0 * v * sqrt(-2.0*log(s)/s); } while (-0.5*coeff5/coeff3*coeff4*dp[i]*dp[i]-cos(coeff4*sz[i])>1); } if(isyn==44) //ele15/2/05: parabolic bunch + sinusoidal bucket //NB=parabolic semiaxis b=2sz0 ->need to set nsigmaz=2 { do { s=2./sqrt(3.)* sqrt(1.-sqrt(1.-15./16.*(rand()/(double)RAND_MAX))); u=(rand()/(double)RAND_MAX); sz[i] = z0_kick + 2.* sz0 * s * cos(2.0*PI*u) ; dp[i] = 2.*dp0 * s * sin(2.0*PI*u); } while (-0.5*coeff5/coeff3*coeff4*dp[i]*dp[i]-cos(coeff4*sz[i])>1); } xsum += xpr[i]; ysum += ypr[i]; xpsum += xpp[i]; ypsum += ypp[i]; } } else{ for (i=0; i= 1.0) ; xpr[i] = sx0 * u * sqrt(-2.0*log(s)/s); xpp[i] = sx0/betax0 * v * sqrt(-2.0*log(s)/s); do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; ypr[i] = sy0 * u * sqrt(-2.0*log(s)/s); ypp[i] = sy0/betay0 * v * sqrt(-2.0*log(s)/s); xpr[i+1] = -xpr[i]; xpp[i+1] = -xpp[i]; ypr[i+1] = ypr[i]; ypp[i+1] = ypp[i]; xpr[i+2] = xpr[i]; xpp[i+2] = xpp[i]; ypr[i+2] = -ypr[i]; ypp[i+2] = -ypp[i]; xpr[i+3] = -xpr[i]; xpp[i+3] = -xpp[i]; ypr[i+3] = -ypr[i]; ypp[i+3] = -ypp[i]; xsum += xpr[i]+xpr[i+1]+xpr[i+2]+xpr[i+3]; ysum += ypr[i]+ypr[i+1]+ypr[i+2]+ypr[i+3]; xpsum += xpp[i]+xpp[i+1]+xpp[i+2]+xpp[i+3]; ypsum += ypp[i]+ypp[i+1]+ypp[i+2]+ypp[i+3]; // if(isyn==0 || isyn==1 || isyn==2) if(isyn==0 || isyn==1 || isyn==2 || isyn==3 || isyn==9) /*gio-May04 what is isyn-3? */ { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; sz[i] = z0_kick + sz0 * u * sqrt(-2.0*log(s)/s); dp[i] = dp0 * v * sqrt(-2.0*log(s)/s); } // if(isyn==3) // sz[i] = sz0 * 4.0 * (rand () / (double)RAND_MAX - 0.5); /*gio-May04 what is isyn=3, 9? */ if(isyn==3 || isyn==9) sz[i] = z0_kick + sz0 * 4.0 * (rand () / (double)RAND_MAX - 0.5); if(isyn==4 || isyn==5 || isyn==6 || isyn ==7 || isyn == 8) { do { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; sz[i] = z0_kick + sz0 * u * sqrt(-2.0*log(s)/s); dp[i] = dp0 * v * sqrt(-2.0*log(s)/s); } while (-0.5*coeff5/coeff3*coeff4*dp[i]*dp[i]-cos(coeff4*sz[i])>1); } if(isyn==44) //ele15/2/05: parabolic bunch + sinusoidal bucket //NB=parabolic semiaxis b=2sz0 ->need to set nsigmaz=2 { do { s=2./sqrt(3.)* sqrt(1.-sqrt(1.-15./16.*(rand()/(double)RAND_MAX))); u=(rand()/(double)RAND_MAX); sz[i] = z0_kick + 2.* sz0 * s * cos(2.0*PI*u) ; dp[i] = 2.*dp0 * s * sin(2.0*PI*u); } while (-0.5*coeff5/coeff3*coeff4*dp[i]*dp[i]-cos(coeff4*sz[i])>1); } sz[i+1]= sz[i]; dp[i+1]= dp[i]; sz[i+2]= sz[i]; dp[i+2]= dp[i]; sz[i+3]= sz[i]; dp[i+3]= dp[i]; } } xave = xsum/(double)NPR; yave = ysum/(double)NPR; xpave = xpsum/(double)NPR; ypave = ypsum/(double)NPR; contax=0; for (l=0; l(xkick-5e-2*sx0)) && (xpp[l]<1e-1*sx0/betax0 && xpp[l]>-1e-1*sx0/betax0)) {contax+=1; mark[l]=1; // fprintf(track_pr,"%d %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",l,xpr[l],xpp[l],ypr[l],ypp[l],sz[l],dp[l]); } //to be uncommented if i_distr_prot=1) // fprintf(prot_ini,"%13.8e %13.8e %13.8e %13.8e %13.8e %13.8e \n",xpr[l],xpp[l],ypr[l],ypp[l],sz[l],dp[l]); } printf("fine distrib prot \n"); //printf("%d",contax); //fprintf(track_pr,"\n"); // fprintf(track_pr,"\n"); //verteil(sz,NPR,lgds,NDC,indsz); //verteil(dp,NPR,lgmm,NDC,inddp); /* for (l=1; l<=NDC; l++) fprintf(prot_indistr,"%d %13.8e %13.8e\n",l,lgds[l],lgmm[l]); */ /* zstep = 4.0*sz0/(double)NBIN; for (j=0; j= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 22: //ele-Apr'05 gaussian cloud for (i=0; i= 1.0) ; xe[i] = rmscloudx *u*sqrt(-2.0*log(s)/s); ye[i] = rmscloudy *v*sqrt(-2.0*log(s)/s); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 3: for (i=0; i 0.5) //xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = (strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } else //xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = -(strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 44: //uniform stripes strback=0.1; for (i=0; i 0.5) xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); else xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 5: parabolic(); for (i=0; i= 1); xpe[i] = 0.0; ype[i] = 0.0; xe[i+1] = - xe[i]; ye[i+1] = ye[i]; xpe[i+1] = 0.0; ype[i+1] = 0.0; xe[i+2] = xe[i]; ye[i+2] = -ye[i]; xpe[i+2] = 0.0; ype[i+2] = 0.0; xe[i+3] = - xe[i]; ye[i+3] = -ye[i]; xpe[i+3] = 0.0; ype[i+3] = 0.0; xeave+= xe[i]+xe[i+1]+xe[i+2]+xe[i+3]; yeave+= ye[i]+ye[i+1]+ye[i+2]+ye[i+3]; } break; case 22: //ele-Apr'05 gaussian cloud for (i=0; i= 1.0) ; xe[i] = rmscloudx *u*sqrt(-2.0*log(s)/s); ye[i] = rmscloudy *v*sqrt(-2.0*log(s)/s); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xe[i+1] = - xe[i]; ye[i+1] = ye[i]; xpe[i+1] = 0.0; ype[i+1] = 0.0; xe[i+2] = xe[i]; ye[i+2] = -ye[i]; xpe[i+2] = 0.0; ype[i+2] = 0.0; xe[i+3] = - xe[i]; ye[i+3] = -ye[i]; xpe[i+3] = 0.0; ype[i+3] = 0.0; xeave+= xe[i]+xe[i+1]+xe[i+2]+xe[i+3]; yeave+= ye[i]+ye[i+1]+ye[i+2]+ye[i+3]; } break; case 3: for (i=0; i 0.5) //xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = (strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } else //xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); { do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0) ; xe[i] = -(strpos*sx0) +(sx0*strsig)*u*sqrt(-2.0*log(s)/s); } ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; xeave += xe[i]; yeave += ye[i]; } break; case 44: //uniform stripes strback=0.1; for (i=0; i 0.5) xe[i] = strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); else xe[i] = -strpos*sx0+2.*(sx0*strsig)*2.*(rand ()/(double)RAND_MAX - 0.5); ye[i] = rpipey*2.*(rand ()/(double)RAND_MAX - 0.5); } while (xe[i]*xe[i]/rpipex/rpipex + ye[i]*ye[i]/rpipey/rpipey >= 1); xpe[i] = 0.0; ype[i] = 0.0; } break; case 5: parabolic(); break; } } xeave /= (double)NEL; yeave /= (double)NEL; for (i=0; i= 1); xpe[i+jmain] = 0.0; ype[i+jmain] = 0.0; xeave += xe[i+jmain]; yeave += ye[i+jmain]; } xeave /= (double)NEL; yeave /= (double)NEL; for (i=0; i= 0 && index < NBIN) { indexp[index][iind[index]] = i; xbin[index] += xpr[i]; xpbin[index] += xpp[i]; x2bin[index] += xpr[i]*xpr[i]; ybin[index] += ypr[i]; ypbin[index] += ypp[i]; y2bin[index] += ypr[i]*ypr[i]; ++iind[index]; ++ncount; } } for (i=0; iNPR1-1) { fprintf(stderr, "ERROR: too many particles in one slice !!"); exit(2); } if (iind[i] > 0) { xs[i] = xbin[i]/(double)iind[i]; ys[i] = ybin[i]/(double)iind[i]; xps[i] = xpbin[i]/(double)iind[i]; yps[i] = ypbin[i]/(double)iind[i]; npr[i] = (double)iind[i]/(double)NPR*npr0; if (iind[i] > 1) { sxs[i] = sqrt(x2bin[i]/(double)iind[i] - xs[i]*xs[i]); sys[i] = sqrt(y2bin[i]/(double)iind[i] - ys[i]*ys[i]); } else { sxs[i] = sx0_cd; sys[i] = sy0; } } else { xs[i] = 0.0; ys[i] = 0.0; xps[i] = 0.0; yps[i] = 0.0; npr[i] = 0.0; sxs[i] = sx0_cd; sys[i] = sy0; } //printf("%d %lg %d\n",i,npr[i],iind[i]); } /*gio-May04 */ if (it==1 && i_coast!=0) { for (i=0;irpipex) { xtn = *xt/fabs(*xt)*(rpipex - 1.e-7); ytn = *yt; } if (fabs(*yt)>rpipey) { ytn = *yt/fabs(*yt)*(rpipey - 1.e-7); } if (fabs(ytn)>=(rpipey - 2.e-7)) { dn1 = 0.; dn2 = -ytn/fabs(ytn); } if (fabs(xtn)>=(rpipex - 2.e-7)) { dn1 = -xtn/fabs(xtn); dn2 = 0.; if (fabs(ytn)>=(rpipey - 2.e-7)) { dn1 = -xtn/fabs(xtn)/sqrt(2.0); dn2 = -ytn/fabs(ytn)/sqrt(2.0); } } dm1 = -dn2; dm2 = dn1; costheta = fabs(dn1*(*xpt) + dn2*(*ypt))/sqrt((*xpt)*(*xpt) + (*ypt)*(*ypt)); if(costheta<0.2) costheta = 0.2; vel2 = (*xpt)*(*xpt) + (*ypt)*(*ypt); beta2 = vel2/C/C; if(beta2>=1.0) beta2 = 0.999; gammael = 1.0/sqrt(1.0 - beta2); ene = (gammael - 1.0)*ELMS*C*C/E; for (im=1;im<=500;im++) if((int)floor((double)im*enestep/ene)==1) break; if(ene<1.0) im=1; enedis[im-1] += *qt; qt_imp += *qt; //fprintf(elwall_ene,"%lg \n",ene); if (i_ang==1) qtn = syield(ene,costheta)*(*qt); else qtn = syield(ene,1.0)*(*qt); /* Frank's quantum mechanics model for reflectivity to 1 at vanishing energy */ qtelast = (sqrt(ene) - sqrt(ene+150))*(sqrt(ene) - sqrt(ene+150))/((sqrt(ene) + sqrt(ene+150))*(sqrt(ene) + sqrt(ene+150)))*(*qt); qtnew1 = qtelast + qtn; ratio = 1 - qtn/qtnew1; qtn = qtnew1; if (rand ()/(double)RAND_MAX < ratio) { dncomp = dn1*(*xpt) + dn2*(*ypt); dmcomp = dm1*(*xpt) + dm2*(*ypt); xptn = (-dn1*dncomp+dm1*dmcomp); yptn = (-dn2*dncomp+dm2*dmcomp); //printf("%lg %lg here \n",xptn,yptn); } else { //avq += qtn; /* do { im==0; eneemit = rand()/(double)RAND_MAX*emax0; while (1) { if (eneemit<(double)(im+1)*emax0/1000.) { if (0.97*lcount[im]/qtn>=limit[im]) break; lcount[im] = lcount[im] + qtn; } im++; if(im==1001) break; } } while (0.97*lcount[im]/qtn>=limit[im] && im<1001); */ do { u = 2.0 * rand () / (double)RAND_MAX - 1.0 ; v = 2.0 * rand () / (double)RAND_MAX - 1.0 ; s = u*u + v*v ; } while (s >= 1.0 || u < 0) ; eneemit = emax0 * u * sqrt(-2.0*log(s)/s); energyscale = eneemit*E; velo = sqrt(2.0*energyscale/ELMS); xsi = sqrt(rand ()/(double)RAND_MAX); xco = sqrt(1.0 - xsi*xsi); dphi = rand ()/(double)RAND_MAX*2.0*PI; dpco = cos(dphi); dpsi = sin(dphi); xptn = dn1*xco+xsi*dm1*dpco; yptn = dn2*xco+xsi*dm2*dpco; xptn *= velo; yptn *= velo; /* vel2 = xptn*xptn + yptn*yptn; beta2 = vel2/C/C; if(beta2>=1.0) beta2 = 0.999; gammael = 1.0/sqrt(1.0 - beta2); ene = (gammael - 1.0)*ELMS*C*C/E; printf("\n Secondary emission: ene = %lg \n",ene); */ } *xt = xtn; *yt = ytn; *xpt = xptn; *ypt = yptn; *qt = qtn; //printf("\n After seiler %lg %lg %lg %lg %lg \n",*xt,*yt,*xpt,*ypt,*qt); } /****************************************************************************** *** Subroutine : open_files *** *** Effect : Open files for storing calculation data. *** *** Parameters : none *** *** Gbl var used : *** *** Gbl var effect : *** *** Constants used : none *** *** Subrout. used : none *** ******************************************************************************/ open_files () { char bpr_filename[80] , /* bunch properties */ headtail_fname[80] , /* head tail distribution */ prphase_fname[80] , /* proton phase space */ pht_filename[80] , prot_filename[80] , samp_filename[80] , pini_filename[80] , eleini_filename[80] , elec_filename[80] , halo_filename[80] , campo_filename[80] , sey_name[80] , eldia_name[80] , ewall_name[80] , froz1_name[80] , froz2_name[80] , track_filename[80] , /* coordinates of 1000 particles each turn */ betaadv_name[80] ; /*in the case of adv=2 records the phase adv */ int i; double enex,qtn,qtelast,qtnew1; strcpy (track_filename,filename) ; strcat (track_filename,"_trk.dat") ; track_pr = fopen(track_filename,"w"); strcpy (bpr_filename,filename) ; strcat (bpr_filename,"_prt.dat") ; bunch_pr = fopen(bpr_filename,"w"); strcpy (headtail_fname,filename) ; strcat (headtail_fname,"_hdtl.dat") ; headtail_pr = fopen(headtail_fname,"w"); strcpy (prphase_fname,filename) ; strcat (prphase_fname,"_prb.dat") ; prot_phase = fopen(prphase_fname,"w"); strcpy (pht_filename,filename) ; strcat (pht_filename,"_inph.dat") ; prot_inphase = fopen(pht_filename,"w"); strcpy (pini_filename,filename) ; strcat (pini_filename,"_pini.dat") ; prot_ini = fopen(pini_filename,"w"); strcpy (eleini_filename,filename) ; strcat (eleini_filename,"_eini.dat") ; ele_ini = fopen(eleini_filename,"w"); strcpy (samp_filename,filename) ; strcat (samp_filename,"_sample.dat") ; samp_prot = fopen(samp_filename,"w"); strcpy (samp_filename,filename) ; strcat (samp_filename,"_sample1.dat") ; samp_prot1 = fopen(samp_filename,"w"); strcpy (samp_filename,filename) ; strcat (samp_filename,"_sample2.dat") ; samp_prot2 = fopen(samp_filename,"w"); strcpy (samp_filename,filename) ; strcat (samp_filename,"_sample3.dat") ; samp_prot3 = fopen(samp_filename,"w"); strcpy (samp_filename,filename) ; strcat (samp_filename,"_sample4.dat") ; samp_prot4 = fopen(samp_filename,"w"); strcpy (samp_filename,filename) ; strcat (samp_filename,"_sample5.dat") ; samp_prot5 = fopen(samp_filename,"w"); strcpy (samp_filename,filename) ; strcat (samp_filename,"_sample6.dat") ; samp_prot6 = fopen(samp_filename,"w"); strcpy (prot_filename,filename) ; strcat (prot_filename,"_bunchds.dat") ; prot_distr = fopen(prot_filename,"w"); strcpy (elec_filename,filename) ; strcat (elec_filename,"_elec.dat") ; el_traj = fopen(elec_filename,"w"); strcpy (elec_filename,filename) ; strcat (elec_filename,"_elec2.dat") ; el_traj2 = fopen(elec_filename,"w"); strcpy (elec_filename,filename) ; strcat (elec_filename,"_elec3.dat") ; el_traj3 = fopen(elec_filename,"w"); strcpy (halo_filename,filename) ; strcat (halo_filename,"_halo.dat") ; halo_p = fopen(halo_filename,"w"); strcpy (campo_filename,filename) ; strcat (campo_filename,"_campo.dat") ; campo_dat = fopen(campo_filename,"w"); strcpy (campo_filename,filename) ; strcat (campo_filename,"_campo0.dat") ; campo0_dat = fopen(campo_filename,"w"); strcpy (campo_filename,filename) ; strcat (campo_filename,"_campo1.dat") ; campo1_dat = fopen(campo_filename,"w"); strcpy (sey_name,filename) ; strcat (sey_name,"_sey.dat") ; sey_file = fopen(sey_name,"w"); strcpy (eldia_name,filename) ; strcat (eldia_name,"_bup.dat") ; el_diagn = fopen(eldia_name,"w"); strcpy (froz1_name,filename) ; strcat (froz1_name,"_frz1.dat") ; froz_pot1 = fopen(froz1_name,"w"); strcpy (froz2_name,filename) ; strcat (froz2_name,"_frz2.dat") ; froz_pot2 = fopen(froz2_name,"w"); /* strcpy (ewall_name,filename) ; strcat (ewall_name,"_ene.dat") ; elwall_ene = fopen(ewall_name,"w"); */ if (i_phas_adv==2 || i_phas_adv==3) { strcpy (betaadv_name,filename) ; strcat (betaadv_name,"_adv2.dat") ; beta_adv = fopen(betaadv_name,"w"); } if (i_coast!=0) { for (i=0;i<=1000;i++) { enex = (double)i; qtn = syield(enex,1.0); qtelast = (sqrt(enex) - sqrt(enex+150))*(sqrt(enex) - sqrt(enex+150))/((sqrt(enex) + sqrt(enex+150))*(sqrt(enex) + sqrt(enex+150))); qtnew1 = qtelast + qtn; fprintf(sey_file,"%13.8e %13.8e %13.8e %13.8e\n",enex,qtn,qtelast,qtnew1); } } } /****************************************************************************** **** PIC module **** ******************************************************************************/ void* xmalloc (size_t size) { void* tmp; tmp=malloc(size); if (tmp!=NULL) { return tmp; } fprintf(stderr,"ERROR: not enough memory\n"); exit(1); } /* Distributes the beam particles for field calculation */ void particles_distribute(GRID *grid,double x[],double y[],int n, int i_which, double n_macro) { int i,j,i1,i2,n_h,j1,j2,j3,j4; double delta_x_i,delta_y_i,offset_x,offset_y; double h_x,h_y; double *rho,tmp,densita; int n_x,n_y; int distribute_in=0,distribute_out=0; n_x=grid->n_cell_x; n_y=grid->n_cell_y; if (i_which==1) { rho=grid->rho1; } else { rho=grid->rho2; } delta_x_i=1.0/grid->delta_x; delta_y_i=1.0/grid->delta_y; offset_x=grid->offset_x; offset_y=grid->offset_y; for (i=0;idistribution) { case 1: for (i=0;i=0)&&(i1=0)&&(i2=0)&&(i1=0)&&(i2n_cell_x; n_y=grid->n_cell_y; if (i_which==1) { rho=grid->rho1; } else { rho=grid->rho2; } delta_x_i=1.0/grid->delta_x; delta_y_i=1.0/grid->delta_y; offset_x=grid->offset_x; offset_y=grid->offset_y; for (i=0;idistribution) { case 1: for (i=0;i=0)&&(i1=0)&&(i2=0)&&(i1=0)&&(i2= i__2 : i2 <= i__2; i2 += i__3) { if (i2 < i2rev) { i__4 = i2 + ip1 - 2; for (i1 = i2; i1 <= i__4; i1 += 2) { i__5 = ip3; i__6 = ip2; for (i3 = i1; i__6 < 0 ? i3 >= i__5 : i3 <= i__5; i3 += i__6) { i3rev = i2rev + i3 - i2; tempr = data[i3]; tempi = data[i3 + 1]; data[i3] = data[i3rev]; data[i3 + 1] = data[i3rev + 1]; data[i3rev] = tempr; data[i3rev + 1] = tempi; } } } ibit = ip2 / 2; L1: if (ibit >= ip1 && i2rev > ibit) { i2rev -= ibit; ibit /= 2; goto L1; } i2rev += ibit; } ifp1 = ip1; L2: if (ifp1 < ip2) { ifp2 = ifp1 << 1; theta = isign * 6.28318530717959 / (ifp2 / ip1); /* Computing 2nd power */ d__1 = sin(theta * .5); wpr = d__1 * d__1 * -2.; wpi = sin(theta); wr = 1.; wi = 0.; i__3 = ifp1; i__2 = ip1; for (i3 = 1; i__2 < 0 ? i3 >= i__3 : i3 <= i__3; i3 += i__2) { i__4 = i3 + ip1 - 2; for (i1 = i3; i1 <= i__4; i1 += 2) { i__6 = ip3; i__5 = ifp2; for (i2 = i1; i__5 < 0 ? i2 >= i__6 : i2 <= i__6; i2 += i__5) { k1 = i2; k2 = k1 + ifp1; tempr = wr * data[k2] - wi * data[k2 + 1]; tempi = wr * data[k2 + 1] + wi * data[k2]; data[k2] = data[k1] - tempr; data[k2 + 1] = data[k1 + 1] - tempi; data[k1] += tempr; data[k1 + 1] += tempi; } } wtemp = wr; wr = wr * wpr - wi * wpi + wr; wi = wi * wpr + wtemp * wpi + wi; } ifp1 = ifp2; goto L2; } nprev = n * nprev; } return; } /* This routine calculates the potentials with the help of the fast fourier transform */ void fold_fft(double *rho1,double *rho2,double *dist,double *phi1, double *phi2,double temp[],int n_x,int n_y) { FILE *datei; double help; int i1,i2,i,j,j0; int nn[2]; double eps=1e-5; /* return no field */ /* fill array with charge */ for (i=0;i0)&&(anorm0)&&(anormn_cell_x; n_y=grid->n_cell_y; grid->min_x=-((float)n_x-2)/((float)n_x)*cut_x; grid->max_x=((float)n_x-2)/((float)n_x)*cut_x; grid->min_y=-((float)n_y-2)/((float)n_y)*cut_y; grid->max_y=((float)n_y-2)/((float)n_y)*cut_y; //grid->offset_x=0.5*(float)n_x; //grid->offset_y=0.5*(float)n_y; /* gio-May04 corrected the half-a-cell offset in the grid */ grid->offset_x=0.5*(float)n_x+.5; grid->offset_y=0.5*(float)n_y+.5; grid->delta_x=2.0*cut_x/((double)n_x); grid->delta_y=2.0*cut_y/((double)n_y); factor=1.0/(grid->delta_x*grid->delta_y); switch (grid->integration_method) { case 1: case 3: phi0=factor* f_potential_2(0.0,0.0,0.5*grid->delta_x,0.5*grid->delta_y); for (i1=0;i1n_cell_x;i1++) { for (i2=0;i2n_cell_y;i2++) { j=i1*grid->n_cell_y+i2; x0=(double)i1; y0=(double)i2; grid->dist[j]=factor* f_potential_2(x0*grid->delta_x,y0*grid->delta_y, 0.5*grid->delta_x,0.5*grid->delta_y) -phi0; } } break; case 2: phi0=factor* f_potential_2(0.0,0.0,0.5*grid->delta_x,0.5*grid->delta_y); for (i1=0;i1dist[i1]=0.0; } for (i1=0;i1dist[j0]=factor* f_potential_2(x0*grid->delta_x,y0*grid->delta_y, 0.5*grid->delta_x,0.5*grid->delta_y) -phi0; if (i2>=1) { grid->dist[2*((i1+1)*2*n_y-i2)]=grid->dist[j0]; if (i1>=1) { grid->dist[2*(2*n_y*(2*n_x-i1)+i2)]=grid->dist[j0]; grid->dist[2*(2*n_y*(2*n_x-i1)+2*n_y-i2)]=grid->dist[j0]; } } else { if (i1>=1) { grid->dist[2*(2*n_y*(2*n_x-i1)+i2)]=grid->dist[j0]; } } } } nn[0]=2*n_y; nn[1]=2*n_x; i1=2; i2=1; //fourtrans(grid->dist,nn,i2); fourtrans2(grid->dist,nn,i2); break; } } GRID* grid_init_comp (int n_x,int n_y,int integration_method) { GRID *grid; int i1,i2,j; grid=(GRID*)xmalloc(sizeof(GRID)); grid->n_cell_x=n_x; grid->n_cell_y=n_y; grid->integration_method=integration_method; grid->distribution=2; grid->interpolation=2; grid->rho1=(double*)xmalloc(sizeof(double)*n_x*n_y); grid->rho2=(double*)xmalloc(sizeof(double)*n_x*n_y); grid->phi1=(double*)xmalloc(sizeof(double)*n_x*n_y); grid->phi2=(double*)xmalloc(sizeof(double)*n_x*n_y); if (i_froz==1) grid->potenziale=(double*)xmalloc(sizeof(double)*n_x*n_y*NBIN); if (integration_method==2) { grid->dist=(double*)xmalloc(sizeof(double)*n_x*n_y*8); grid->temp=(double*)xmalloc(sizeof(double)*n_x*n_y*8); } else { grid->dist=(double*)xmalloc(sizeof(double)*n_x*n_y); } return grid; } void potential_load(GRID *grid) { char pot_filename[40] ; FILE *pot_file; long i, condit, jslice,j,n_x,n_y; double pottt[114]; n_x=grid->n_cell_x; n_y=grid->n_cell_y; strcpy (pot_filename,filename); strcat (pot_filename,".frz1"); pot_file = fopen(pot_filename,"r"); if (pot_file == NULL) { printf ("\n\n File with potential does not exist.\n") ; exit(2); } printf(pot_filename); i = 0; printf("comincio il loop \n"); do {condit = fscanf (pot_file," %lf\n ",&grid->potenziale[i]); ++i; } while(condit != EOF); if(i!=grid->n_cell_x * grid->n_cell_y *NBIN+1) { printf ("i=%d NBIN=%d n_x=%d n_y=%d n_x*n_y*NBIN=%d \n\n Number of cells different from the file\n",i,NBIN,n_x,n_y, n_x*n_y*NBIN ) ; exit(2); } /* //for the moment the order of the slices is inverted in the print loop(to be checked) but the code works correctly for(jslice=NBIN-1; jslice>0; jslice--) for(j=0;jpotenziale[jslice*n_x*n_y+j*n_x+i]); fclose(froz_pot2); fclose(pot_file); */ printf (" Frozen potential loaded \n"); } /* This routine folds the charge distribution and the greensfunction of a grid to get the potentials. */ void foldfields (double *rho,double *dist,double *phi,int n_x,int n_y) { double s; int i1,i2,i3,i4,j; int field_typ=1; float eps=1e-5; for (i1=0;i1integration_method) { case 1: foldfields(grid->rho1,grid->dist,grid->phi1,grid->n_cell_x,grid->n_cell_y); foldfields(grid->rho2,grid->dist,grid->phi2,grid->n_cell_x,grid->n_cell_y); break; case 2: fold_fft(grid->rho1,grid->rho2,grid->dist,grid->phi1,grid->phi2, grid->temp,grid->n_cell_x,grid->n_cell_y); break; } } void particles_move(GRID *grid,double energy,double ics0[], double yps0[],double icsp[],double ypsp[], int nn,int i_which,double step,double scale) { static float esum=0.0; static int sum_photon=0; static long int n_low=0; int i,i1,i2,l,k; int elconta, rediffcont; double ax,ay, d2x, d2y; double phi1_x,phi2_x,phi3_x,phi1_y,phi2_y,phi3_y,h_x,h_y; int n_x,n_y; register int j; register double h,h_p; double *phi, *phisc; double delta_x_i,delta_y_i,offset_x,offset_y,delta_x,delta_y; double xx,yy,xxp,yyp; double min_x,max_x,min_y,max_y,e_tmp,amp; double k11,k12,k13,k14,k21,k22,k23,k24,k31,k32,k33,k34,k41,k42,k43,k44; double velprx, velpry, bux, buy, bmod; double p; elconta=0; rediffcont=0; min_x=grid->min_x; max_x=grid->max_x; min_y=grid->min_y; max_y=grid->max_y; delta_x=grid->delta_x; delta_y=grid->delta_y; delta_x_i=1.0/grid->delta_x; delta_y_i=1.0/grid->delta_y; offset_x=grid->offset_x; offset_y=grid->offset_y; n_x=grid->n_cell_x; n_y=grid->n_cell_y; if (i_which==1){ //to move the protons phi=grid->phi2; } else{ //to move the electrons phi=grid->phi1; //ele-apr05: includes space charge of electrons WRONG // phisc=grid->phi2; } /* //printf("scrive frz2"); if (i_which==2) for(j=0;jinterpolation){ case 1: for (i=0;i0)&&(i20)){ j=i1*n_y+i2; ax = (phi[j+n_y] -phi[j-n_y])*step /(2.0*delta_x*energy); ay = (phi[j+1] -phi[j-1])*step /(2.0*delta_y*energy); icsp[i] -= ax*scale; ypsp[i] -= ay*scale; } else { fprintf(stderr,"WARNING: particle is not on the grid\n"); } ics0[i] += icsp[i]*step; yps0[i] += ypsp[i]*step; } break; case 2: for (i=0;i=rpipex || fabs(yps0[i])>=rpipey) { rediffcont++; do { if(i_dip==1) icsp[i] = 0.0; else icsp[i] = 1.3e6*rand () / (double)RAND_MAX ; ypsp[i] = 1.3e6*rand () / (double)RAND_MAX ; } while (sqrt(icsp[i]*icsp[i]+ypsp[i]*ypsp[i]) >= 1.3e6) ; if(fabs(ics0[i])>=rpipex) ics0[i] = fabs(ics0[i])/ics0[i]*rpipex; if(fabs(yps0[i])>=rpipey) yps0[i] = fabs(yps0[i])/yps0[i]*rpipey; icsp[i] *= -fabs(ics0[i])/ics0[i]; ypsp[i] *= -fabs(yps0[i])/yps0[i]; } } if(elflag==2 || elflag==22 || elflag==4 || elflag==44) { if (ics0[i]*ics0[i]/rpipex/rpipex+yps0[i]*yps0[i]/rpipey/rpipey>=1.) { rediffcont++; do { if(i_dip==1) icsp[i] = 0.0; else icsp[i] = 1.3e6*rand () / (double)RAND_MAX ; ypsp[i] = 1.3e6*rand () / (double)RAND_MAX ; } while (sqrt(icsp[i]*icsp[i]+ypsp[i]*ypsp[i]) >= 1.3e6) ; ics0[i] = rpipex*cos(atan2(yps0[i],ics0[i])); yps0[i] = rpipey*sin(atan2(yps0[i],ics0[i])); icsp[i] *= -fabs(ics0[i])/ics0[i]; ypsp[i] *= -fabs(yps0[i])/yps0[i]; } } } else //if symm_el=1 if (i_symm_pr==1) { if(elflag==1 || elflag==3) { if (fabs(ics0[i])>=rpipex || fabs(yps0[i])>=rpipey) { rediffcont++; icsp[i] = 0.0; ypsp[i] = 0.0 ; if(fabs(ics0[i])>=rpipex) ics0[i] = fabs(ics0[i])/ics0[i]*rpipex; if(fabs(yps0[i])>=rpipey) yps0[i] = fabs(yps0[i])/yps0[i]*rpipey; icsp[i] *= -fabs(ics0[i])/ics0[i]; ypsp[i] *= -fabs(yps0[i])/yps0[i]; } } if(elflag==2 || elflag==22 || elflag==4 || elflag==44) { if (ics0[i]*ics0[i]/rpipex/rpipex+yps0[i]*yps0[i]/rpipey/rpipey>=1.) { rediffcont++; icsp[i] = 0.0; ypsp[i] = 0.0 ; ics0[i] = rpipex*cos(atan2(yps0[i],ics0[i])); yps0[i] = rpipey*sin(atan2(yps0[i],ics0[i])); icsp[i] *= -fabs(ics0[i])/ics0[i]; ypsp[i] *= -fabs(yps0[i])/yps0[i]; } } } } xx=ics0[i]; yy=yps0[i]; xxp=icsp[i]; yyp=ypsp[i]; if ((xx>min_x)&&(xxmin_y)&&(yymin_x; max_x=grid->max_x; min_y=grid->min_y; max_y=grid->max_y; delta_x=grid->delta_x; delta_y=grid->delta_y; delta_x_i=1.0/grid->delta_x; delta_y_i=1.0/grid->delta_y; offset_x=grid->offset_x; offset_y=grid->offset_y; n_x=grid->n_cell_x; n_y=grid->n_cell_y; if (i_which==1){ //in fact HAS TO BE i_which=1 in this routine... if (it==1 & i_pot_froz==0){ phi=grid->phi2; memcpy(grid->potenziale+(NBIN-1-jmain)*n_x*n_y,phi,sizeof(double)*n_x*n_y); //jmain=NBIN-1->0 /* if (jmain==69 || jmain==30 || jmain==1) { for(j=0;j1 or i_pot_froz=1 phi=grid->phi2; memcpy(phi,grid->potenziale+(NBIN-1-jmain)*n_x*n_y,sizeof(double)*n_x*n_y); /* //if (jmain==69 || jmain==30 || jmain==0) { for(j=0;jpotenziale[jmain*n_x*n_y+j*n_x+i]); // fprintf(froz_pot2,"\n"); // fprintf(froz_pot2,"\n"); //} */ } } else printf("ERROR: subroutine move_froz is used for electrons \n"); switch (grid->interpolation){ case 1: for (i=0;i0)&&(i20)){ j=i1*n_y+i2; ax = (phi[j+n_y] -phi[j-n_y])*step /(2.0*delta_x*energy); ay = (phi[j+1] -phi[j-1])*step /(2.0*delta_y*energy); icsp[i] -= ax*scale; ypsp[i] -= ay*scale; } else { fprintf(stderr,"WARNING: particle is not on the grid\n"); } ics0[i] += icsp[i]*step; yps0[i] += ypsp[i]*step; } break; case 2: for (i=0;imin_x)&&(xxmin_y)&&(yymin_x; max_x=grid->max_x; min_y=grid->min_y; max_y=grid->max_y; delta_x=grid->delta_x; delta_y=grid->delta_y; delta_x_i=1.0/grid->delta_x; delta_y_i=1.0/grid->delta_y; offset_x=grid->offset_x; offset_y=grid->offset_y; n_x=grid->n_cell_x; n_y=grid->n_cell_y; if (i_which==1){ phi=grid->phi2; } else{ phi=grid->phi1; } switch (grid->interpolation){ case 1: for (i=0;i0)&&(i20)){ j=i1*n_y+i2; ax = (phi[j+n_y] -phi[j-n_y])*step /(2.0*delta_x*energy); ay = (phi[j+1] -phi[j-1])*step /(2.0*delta_y*energy); icsp[i] -= ax*scale; ypsp[i] -= ay*scale; } else { fprintf(stderr,"WARNING: particle is not on the grid\n"); } ics0[i] += icsp[i]*step; yps0[i] += ypsp[i]*step; } break; case 2: for (i=0;i=rpipex || fabs(yps0[i])>=rpipey) { do { if(i_dip==1) icsp[i] = 0.0; else icsp[i] = 1.3e6*rand () / (double)RAND_MAX ; ypsp[i] = 1.3e6*rand () / (double)RAND_MAX ; } while (sqrt(icsp[i]*icsp[i]+ypsp[i]*ypsp[i]) >= 1.3e6) ; if(fabs(ics0[i])>=rpipex) ics0[i] = fabs(ics0[i])/ics0[i]*rpipex; if(fabs(yps0[i])>=rpipey) yps0[i] = fabs(yps0[i])/yps0[i]*rpipey; icsp[i] *= -fabs(ics0[i])/ics0[i]; ypsp[i] *= -fabs(yps0[i])/yps0[i]; } } if(elflag==2 || elflag==22 || elflag==4 || elflag==44) { if (ics0[i]*ics0[i]/rpipex/rpipex+yps0[i]*yps0[i]/rpipey/rpipey>=1.) { do { if(i_dip==1) icsp[i] = 0.0; else icsp[i] = 1.3e6*rand () / (double)RAND_MAX ; ypsp[i] = 1.3e6*rand () / (double)RAND_MAX ; } while (sqrt(icsp[i]*icsp[i]+ypsp[i]*ypsp[i]) >= 1.3e6) ; ics0[i] = rpipex*cos(atan2(yps0[i],ics0[i])); yps0[i] = rpipey*sin(atan2(yps0[i],ics0[i])); icsp[i] *= -fabs(ics0[i])/ics0[i]; ypsp[i] *= -fabs(yps0[i])/yps0[i]; } } } //check for electrons if they exceeded boundaries for coasting beams //it applies Seiler to model SEY //if(i_which==2 && i_coast!=0) if(energy<0 && i_coast!=0) { amp1 = fabs(ics0[i])/rpipex; amp2 = fabs(yps0[i])/rpipey; //printf(" %d %d \n",it,jmain); if(i==0 && nturn<10) fprintf(el_traj,"%13.8e %13.8e %13.8e %13.8e %13.8e\n",ics0[i],yps0[i],icsp[i],ypsp[i],qq[i]); if(jmain==NBIN-1) qtot += qq[i]/(4*rpipex*rpipey*circ); if (amp1 > 1.0 || amp2 > 1.0) { //printf("\n %13.8e %13.8e %13.8e %13.8e\n",amp1,amp2,rpipex,rpipey); //printf("%d %13.8e %13.8e %13.8e %13.8e %13.8e\n",i,ics0[i],yps0[i],icsp[i],ypsp[i],qq[i]); seiler (&ics0[i],&yps0[i],&icsp[i],&ypsp[i],&qq[i]); //printf("%d %13.8e %13.8e %13.8e %13.8e %13.8e\n\n\n",i,ics0[i],yps0[i],icsp[i],ypsp[i],qq[i]); } } xx=ics0[i]; yy=yps0[i]; xxp=icsp[i]; yyp=ypsp[i]; //if(energy<0 && i_coast!=0) //fprintf(track_pr,"%13.8e %13.8e %13.8e \n",xx,yy,xxp,yyp,qq[i]); /* if(i==0) { printf("\n %lg %lg %lg \n", xx,min_x,max_x); printf(" %lg %lg %lg \n", yy,min_y,max_y); } */ if ((xx>min_x)&&(xxmin_y)&&(yy0) { icsp[i] -= ax*scale*step; ypsp[i] -= ay*scale*step; } if(i_which==1 && energy<0) { aux_fix[i] = -ax*scale; aux_fiy[i] = -ay*scale; } break; case 3: if(i_which==2) { velprx = icsp[i] - ax*step*scale; velpry = ypsp[i] - ay*step*scale; bux = bgrad*yps0[i]; buy = bdip - bgrad*ics0[i]; bmod = sqrt(bux*bux+buy*buy); bux *= fabs(velprx)/velprx/bmod; buy *= fabs(velpry)/velpry/bmod; icsp[i] = (bux*velprx + buy*velpry)*bux; ypsp[i] = (bux*velprx + buy*velpry)*buy; } if(i_which==1) { icsp[i] -= ax*scale*step; ypsp[i] -= ay*scale*step; } break; } } else{ fprintf(stderr,"WARNING: particle out of grid"); if (i_which==2 || energy<0) { fprintf(stderr,"\t -> electron (%13.8e,%13.8e,%d,%d)\n", xx,yy,i,nn); elconta+=1; } else fprintf(stderr,"\t -> proton (%13.8e,%13.8e,%d,%d)\n", xx,yy,i,nn); } } break; } // fprintf(stderr,"nturn= %d electr= %d \n", it, elconta); } void field_poissonbc(GRID *grid) { double *rho11, *rho22, *work, *rcos, ddx, ddy; int nn, mm, kloc, nloc; ddx = 1.0; ddy = grid->delta_y/grid->delta_x; mm = grid->n_cell_x ; nn = grid->n_cell_y ; rho11=dvector(0,mm*nn-1); rho22=dvector(0,mm*nn-1); work=dvector(0,mm+nn+2); rcos=dvector(0,mm*nn-1); for (kloc=0; klocrho1 + nloc*nn + kloc)); rho22[kloc*mm + nloc] = (4*PI)*(*(grid->rho2 + nloc*nn + kloc)); */ rcos[kloc*nn + nloc]=1.0/(2.0*((cos((double)(kloc)*PI/mm)-1.0)/(ddx*ddx)+(cos((double)(nloc)*PI/nn)-1.0)/(ddy*ddy)))*4.0/(mm*nn); rho11[kloc*nn + nloc] = (4*PI)*(*(grid->rho1 + kloc*nn + nloc)); rho22[kloc*nn + nloc] = (4*PI)*(*(grid->rho2 + kloc*nn + nloc)); } potfft_(rho22,rcos,work,&mm,&nn); potfft_(rho11,rcos,work,&mm,&nn); for (kloc=0; klocphi1 + nloc*nn + kloc) = 1/ddy*rho11[kloc*mm + nloc]; *(grid->phi2 + nloc*nn + kloc) = 1/ddy*rho22[kloc*mm + nloc]; */ *(grid->phi1 + kloc*nn + nloc) = 1/ddy*rho11[kloc*nn + nloc]; *(grid->phi2 + kloc*nn + nloc) = 1/ddy*rho22[kloc*nn + nloc]; }} free_dvector(rho11,0,mm*nn-1); free_dvector(rho22,0,mm*nn-1); // free_dvector(work,0,mm*nn+2); free_dvector(work,0,mm+nn+2); free_dvector(rcos,0,mm*nn-1); } /****************************************************************************** **** Main Program **** ******************************************************************************/ main () { GRID *grid; double amp1, amp2; printf (" Program for studying head-tail instabilities due to\n"); printf (" electron cloud \n\n"); printf ("--> Reading data from configuration file.\n") ; read_data () ; printf ("--> Open files.\n") ; open_files (); printf ("\t\t Complete\n") ; printf ("--> Init values.\n") ; init_values () ; printf ("\t\t Complete\n") ; printf ("--> Generation of e- cloud and bunch distributions.\n"); initialization (); printf ("\t\t Complete\n") ; //if(i_method==2) if(i_method==2 || i_method==3) /* gio-May04 */ { printf ("--> Generation of the grid for the particle-in-cell.\n"); printf (" calculation of the fields \n"); grid=grid_init_comp(128,128,2); if(i_dip==2 && i_coast==0){ gridextx = 3.*rpipex; gridexty = 3.*rpipey; } else{ gridextx = 1.1*rpipex; gridexty = 1.1*rpipey; } if(i_bound==1){ gridextx = 1.05*rpipex; gridexty = 1.05*rpipey; } if(ielsizex==1) gridextx = 3.*rpipex; if(ielsizey==1) gridexty = 3.*rpipey; grid_init_phys(grid,gridextx,gridexty); printf ("\t\t Complete\n") ; if (i_pot_froz==1) potential_load(grid); } fprintf (prot_inphase, "electrons = %d protons = %d bunch slices = %d\n",NEL,NPR,NBIN); fprintf (prot_inphase, "electron sizes -> sx = %g sy = %g\n",sx2,sy2); fprintf (prot_inphase, "beam sizes -> sx = %g sy = %g\n",sx0,sy0); fprintf (prot_inphase, "beta0 values -> betax = %g betay = %g\n",betax0,betay0); //ele-Mar05 fprintf (prot_inphase, "e- cloud sizes -> rpipex = %g rpipey = %g\n",rpipex,rpipey); fprintf (prot_inphase, "maximum tune shifts -> dt_x = %g dt_y = %g\n",dtunxmax,dtunymax); fprintf (prot_inphase, " frozen cloud potential option = %d\n",i_froz); //ele- Feb'05 fprintf (prot_inphase, "potential read from a file = %d\n",i_pot_froz); //ele-Mar05 fprintf (prot_inphase, "proton distribution read from a file = %d\n", i_distr_prot); //ele-Mar05 ili = 0; /* here the loop over the total number of steps (kicks * turns) starts... */ /* gio-May04 i_mauro begin */ // if(i_mauro==0 { for(it=1; it<=nstep; it++) { if(npr_eff==NPR){ printf("no more particles \n"); exit(2); } /* gio-May04 */ if (i_coast!=0) { generate(); //printf ("in generate"); } xave = 0.0; yave = 0.0; zave = 0.0; xpave = 0.0; ypave = 0.0; dpmean = 0.0; sxsize = 0.0; sysize = 0.0; szsize = 0.0; xemit = 0.0; yemit = 0.0; corryz = 0.0; invary = 0.0; invarx = 0.0; // largeemix =0.0; // largeemiy =0.0; for(imain=npr_eff; imain9.*(sx0*sx0/betax0)) { // fprintf(halo_p,"%d %13.8e %13.8e \n",imain, emipx,(sx0*sx0/betax0) ); largeemix+=1; } if(emipy>9.*(sy0*sy0/betay0)) { //fprintf(halo_p,"%d %13.8e %13.8e \n",imain, emipx,(sx0*sx0/betax0) ); largeemiy+=1; } */ } sxsize = sqrt(sxsize/(double)(NPR - npr_eff)); sysize = sqrt(sysize/(double)(NPR - npr_eff)); szsize = sqrt(szsize/(double)(NPR - npr_eff)); sxpsize /= (double)(NPR - npr_eff); sypsize /= (double)(NPR - npr_eff); xemit /= (double)(NPR - npr_eff); yemit /= (double)(NPR - npr_eff); corryz /= (double)(NPR - npr_eff)*szsize; invarx /= (double)(NPR - npr_eff); invary /= (double)(NPR - npr_eff); timep = (double)(it - 1)*tstepp; // printf("particelle dell halo %13.8e \n",(double)largeemix/(NPR-npr_eff)); // fprintf(halo_p,"%d %13.8e %13.8e \n",it,(double)largeemix/(double)(NPR-npr_eff), (double)largeemiy/(double)(NPR-npr_eff)); // fprintf(halo_p,"\n"); // fprintf(halo_p,"\n"); //printf("sx0_cd = %lf (averaging on particles)", sqrt(sxsize)); /* moments of the global transverse distribution have been calculated above and after bining (to get bin_frac updated) will be printed to file */ /* every 20 turns we also store data for all four transverse distributions of the bunch */ /* if (it % 20 == 0) { verteil (x,NPR,hdp,NDC,indxp); verteil (y,NPR,vdp,NDC,indyp); verteil (xp,NPR,hpdp,NDC,indxpp); verteil (yp,NPR,vpdp,NDC,indypp); for (lmain=1; lmain<=NDC; lmain++) fprintf(prot_distr,"%13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",indxp[lmain],hdp[lmain],indyp[lmain],vdp[lmain],indxpp[lmain],hpdp[lmain],indypp[lmain],vpdp[lmain]); } */ /* the electrons distribution is re-initialized before each interaction, but every 20 turns the deformed horizontal and vertical distributions are stored first if (it % 100 == 0) { verteil2 (xe,NEL,hde,NDC,indx); verteil2 (ye,NEL,vde,NDC,indy); for (lmain=1; lmain<=NDC; lmain++) fprintf(ele_distr,"%13.8e %13.8e %13.8e %13.8e\n",indx[lmain],hde[lmain],indy[lmain],vde[lmain]); } */ refresh_el (); //verteil (xe,NEL,hde,NDC,indx); //verteil (ye,NEL,vde,NDC,indy); //for (lmain=1; lmain<=NDC; lmain++) // fprintf(prot_indistr,"%13.8e %13.8e %13.8e %13.8e\n",indx[lmain],hde[lmain],indy[lmain],vde[lmain]); /* the beam must be sliced: each slice will then be made to interact with the electrons concentrated at one section */ if (i_froz==1) //ele-Mar05 { bining (0.,sz0); } else{ // bining (zave,szsize); if (isyn==3 || isyn==9) /* gio-May04 */ bining (0.,sz0); else bining (zave,szsize); /* */ } qtot = 0.; /* gio-May04 */ printf("%d %f \n",it,timep); if (it% nkick==0) fprintf(bunch_pr,"%13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",timep,xave,xpave,yave,zave,dpmean,sxsize,sysize,szsize,sqrt(sxsize*sxsize*sxpsize-xemit*xemit),sqrt(sysize*sysize*sypsize-yemit*yemit),invarx/2/betax0,invary/2/betay0,corryz,(double)(NPR-npr_eff)/(double)NPR,bin_frac); // printf("%13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",timep,xave,yave,sqrt(sxsize),sqrt(sysize),sqrt(sxsize*sxpsize-xemit*xemit),sqrt(sysize*sypsize-yemit*yemit),invarx/2/betax0,invary/2/betay0,corryz); /* dispersion is added before interaction bunch-ecloud */ if(dispx>0.1) for(jmain=npr_eff; jmainNEL) ntemp = NEL; for(jmain=NBIN-1; jmain>=0; jmain--) { sx = sxs[jmain]; sy = sys[jmain]; xoff = xs[jmain]; yoff = ys[jmain]; npr1 = npr[jmain]; /* offsets and rms sizes of the slices are printed to file */ if(it % n_diag == 0 || it == 1) fprintf(headtail_pr, "%13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",zs[jmain], npr1*xoff, npr1*yoff, npr1*sx, npr1*sy, npr1); if(i_dip==2 && i_ecool==2 && i_coast==0) new_fraction (); /* here the loop over the beam-kicked electrons starts... if the soft-Gaussian model is applied */ // if(i_ecool==0) { if(i_ecool!=1) { /*gio-May04*/ switch (i_method) { case 0: break; case 1: for(imain=0; imain0) { for(kmain=0; kmain1 or i_pot_froz=1)) { particles_move_froz(grid,epr,vprx,vpry,vprxp,vpryp,iind[jmain],1,tstep,fscale2,jmain); } } //printf("finitofroz \n"); for(kmain=0; kmain 1.0 || amp2 > 1.0) seiler (&xe[imain],&ye[imain],&xpe[imain],&ype[imain],&qe[imain]); } //if (imain==0) //fprintf(el_traj,"%13.8e %13.8e %13.8e %13.8e %13.8e\n",xe[imain],ye[imain],xpe[imain],ype[imain],qe[imain]); xta = fabs(xe[imain] - xoff); yta = fabs(ye[imain] - yoff); x1 = xe[imain]; ips1 = ye[imain]; ffrank_(&x1,&ips1,&xoff,&yoff,&sx,&sy,&pre,&pim,&ili); kick_xe = - pim * npr1 * factor1 * (xe[imain] - xoff)/xta * tstep; kick_ye = - pre * npr1 * factor1 * (ye[imain] - yoff)/yta * tstep; xpe[imain] += kick_xe; ype[imain] += kick_ye; } for(kmain=0; kmain0 && iwake_flag == 1) { for (kmain=0; kmainjmain; mmain--) { zi = (jmain-mmain)*zstep; switch(i_pipe) { case 0: kick_x += wakefac*npr[mmain]*xs[mmain]*wake_func(zi); kick_y += wakefac*npr[mmain]*ys[mmain]*wake_func(zi); kick_z += wakefac*npr[mmain]*wake_funcz(zi); // printf("field= %13.8e \n",kick_x); break; case 1: kick_x += wakefac*npr[mmain]*0.41/1.23*(xs[mmain]-xpr[ind])*wake_func(zi); kick_y += wakefac*npr[mmain]*0.82/1.23*(ys[mmain]+0.5*ypr[ind])*wake_func(zi); kick_z += wakefac*npr[mmain]*wake_funcz(zi); break; case 2: kick_x += 0.; kick_y += wakefac*npr[mmain]*ys[mmain]*wake_func(zi); kick_z += wakefac*npr[mmain]*wake_funcz(zi); break; case 3: kick_x += wakefac*npr[mmain]*xs[mmain]*wake_reswall(zi); kick_y += wakefac*npr[mmain]*ys[mmain]*wake_reswall(zi); kick_z += wakefac*npr[mmain]*wake_funcz(zi); break; case 6: kick_x += wakefac*npr[mmain]*wake_reswall(zi)*(0.41*xs[mmain]-0.41*xpr[ind]); kick_y += wakefac*npr[mmain]*wake_reswall(zi)*(0.82*ys[mmain]+0.41*ypr[ind]); kick_z = 0.; //prima era +=0 !!!!!!!!! break; } } //if(it==1 && kmain==0 && mmain==NBIN-1) //fprintf(prot_distr,"%13.8e %13.8e\n",zi,wake_func(merit,zetat,zi)); } else //(in the case i_pipe is equal to 5) { ycoff =0.; // b_coll/5. ; kick_x = 0.; kick_y +=-wakefac*1/(4*PI*EPS0)*(npr[jmain]/zstep)*(theta_coll*4/b_coll)*((ys[jmain]+ycoff-ypr[ind])+PI*h_coll/2/b_coll*ypr[ind]); // kick_y=kick_y*1e4; // kick_y=1e-10; // kick_y=1e-9*rand()/(double)RAND_MAX; // kick_y=1e-9*2.*(rand()/(double)RAND_MAX - 0.5); kick_z =0.; } } else //(in the case i_pipe is equal to 4) kick_z = -9.57e-10 + 8.e-11*2.*(rand()/(double)RAND_MAX - 0.5); xpp[ind] += kick_x; ypp[ind] += kick_y; dp[ind] += kick_z; } } // printf("%f %f %f\n",theta_coll,b_coll,h_coll); // printf("%f %f %f\n",kick_x,kick_y,kick_z); // printf("%f %d %f %f\n",wakefac,npr,zstep,PI ); if (i_space == 1) /* { for(kmain=0; kmainjmain /* - i_rot = 1 the rotation is done around the full bunch centroid and using the overall bunch sigmas - i_rot = 0 the rotation is done around the local centroid and by using the local sigmas */ if(ishift==1 && i_rot==1) { //cent_x = xave; //cent_y = yave; //cent_xp = xpave; //cent_yp = ypave; //dtunx = dtunxmax/sxsize/(sxsize+sysize); //dtuny = dtunymax/sysize/(sxsize+sysize); cent_x = 0.; cent_y = 0.; cent_xp = 0.; cent_yp = 0.; dtunx = dtunxmax; dtuny = dtunymax; } /* dispersion is subtracted for final transport of the bunch through the ring */ if(dispx>0.1) for(jmain=npr_eff; jmain1 && i_beta==1) { cff1x = sqrt(betarrayx[it%nkick]/betarrayx[(it-1)%nkick]); cff2x = sqrt(betarrayx[it%nkick]*betarrayx[(it-1)%nkick]); cff1y = sqrt(betarrayy[it%nkick]/betarrayy[(it-1)%nkick]); cff2y = sqrt(betarrayy[it%nkick]*betarrayy[(it-1)%nkick]); //printf("%d %f %f %f %f \n", it, cff1x, cff2x, cff1y, cff2y); } else { cff1x = 1.0; cff2x = betax0; cff1y = 1.0; cff2y = betay0; } /* end gio-May04; beta /* loop for sextupolar kick in a section with non-zero dispersion */ if(i_sext==1) { for(jmain=npr_eff; jmain= 0 && sl_ind < NBIN) { cent_x = xs[sl_ind]; cent_y = ys[sl_ind]; cent_xp = xps[sl_ind]; cent_yp = yps[sl_ind]; dtunx = dtunxmax*sx0_cd*sx0_cd/sxs[sl_ind]/(sxs[sl_ind]+sys[sl_ind])*npr[sl_ind]/npr0*sqrt(2*PI)*sz0/zstep; dtuny = dtunymax*sy0*sy0/sys[sl_ind]/(sxs[sl_ind]+sys[sl_ind])*npr[sl_ind]/npr0*sqrt(2*PI)*sz0/zstep; if(it==1 && flag4print==1 && sl_ind==NBIN/2){ fprintf(prot_inphase,"The tune shift due to space charge at the centre of the bunch (max value) is: \n"); fprintf(prot_inphase,"dtunx = %lf", dtunx); fprintf(prot_inphase,", dtuny = %lf \n", dtuny); flag4print=0; } if(it==1 && flag4print==1 && sl_ind==NBIN-1){ fprintf(prot_inphase,"The tune shift due to space charge at the beginning of the bunch (max value) is: \n"); fprintf(prot_inphase,"dtunx = %lf", dtunx); fprintf(prot_inphase,", dtuny = %lf \n", dtuny); flag4print=0; } if(it==1 && flag4print==1 && sl_ind==1){ fprintf(prot_inphase,"The tune shift due to space charge at the end of the bunch (max value) is: \n"); fprintf(prot_inphase,"dtunx = %lf", dtunx); fprintf(prot_inphase,", dtuny = %lf \n", dtuny); flag4print=0; // ele15/2/05: these value are more 'reliable' than dtunxmax, which is just a constant defined in a certain way (a factor 2 for sure is missing). The only probls here is that if NBIN is an even number, then slices NBIN/2 is not really at the middle and that probab the protons are already been moved for the first turn (but it should not matter...) } //dtunx = dtunxmax*sx0_cd*sx0_cd/sxs[sl_ind]/(sxs[sl_ind]+sys[sl_ind])*lambda; //dtuny = dtunymax*sy0*sy0/sys[sl_ind]/(sxs[sl_ind]+sys[sl_ind])*lambda; } else { if (sl_ind < 0) { cent_x = xs[0]; cent_y = ys[0]; cent_xp = xps[0]; cent_yp = yps[0]; dtunx = dtunxmax*sx0_cd*sx0_cd/sxs[0]/(sxs[0]+sys[0])*npr[0]/npr0*sqrt(2*PI)*sz0/zstep; dtuny = dtunymax*sy0*sy0/sys[0]/(sxs[0]+sys[0])*npr[0]/npr0*sqrt(2*PI)*sz0/zstep; //dtunx = dtunxmax*sx0_cd*sx0_cd/sxs[0]/(sxs[0]+sys[0])*lambda; //dtuny = dtunymax*sy0*sy0/sys[0]/(sxs[0]+sys[0])*lambda; } else { cent_x = xs[NBIN-1]; cent_y = ys[NBIN-1]; cent_xp = xps[NBIN-1]; cent_yp = yps[NBIN-1]; dtunx = dtunxmax*sx0_cd*sx0_cd/sxs[NBIN-1]/(sxs[NBIN-1]+sys[NBIN-1])*npr[NBIN-1]/npr0*sqrt(2*PI)*sz0/zstep; dtuny = dtunymax*sy0*sy0/sys[NBIN-1]/(sxs[NBIN-1]+sys[NBIN-1])*npr[NBIN-1]/npr0*sqrt(2*PI)*sz0/zstep; //dtunx = dtunxmax*sx0_cd*sx0_cd/sxs[NBIN-1]/(sxs[NBIN-1]+sys[NBIN-1])*lambda; //dtuny = dtunymax*sy0*sy0/sys[NBIN-1]/(sxs[NBIN-1]+sys[NBIN-1])*lambda; } } /* begin gio-May04 3 different beta at IPs */ /* xx1 = cos(2*PI*dtunx/(double)nkick)*(x0 - cent_x) - betax0*sin(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_x; xxp1 = 1/betax0*sin(2*PI*dtunx/(double)nkick)*(x0 - cent_x) + cos(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_xp; yy1 = cos(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) - betay0*sin(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_y; yyp1 = 1/betay0*sin(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) + cos(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_yp; */ xx1 = cff1x*cos(2*PI*dtunx/(double)nkick)*(x0 - cent_x) - cff2x*sin(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_x; xxp1 = 1/cff2x*sin(2*PI*dtunx/(double)nkick)*(x0 - cent_x) + 1/cff1x*cos(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_xp; yy1 = cff1y*cos(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) - cff2y*sin(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_y; yyp1 = 1/cff2y*sin(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) + 1/cff1y*cos(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_yp; /* end gio-May04 3 different beta at IPs */ } if(i_rot==1) { /* begin: gio-May04 3 different beta at IPs */ /* xx1 = cos(2*PI*dtunx/(double)nkick)*(x0 - cent_x) - betax0*sin(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_x; xxp1 = 1/betax0*sin(2*PI*dtunx/(double)nkick)*(x0 - cent_x) + cos(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_xp; yy1 = cos(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) - betay0*sin(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_y; yyp1 = 1/betay0*sin(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) + cos(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_yp; */ xx1 = cff1x*cos(2*PI*dtunx/(double)nkick)*(x0 - cent_x) - cff2x*sin(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_x; xxp1 = 1/cff2x*sin(2*PI*dtunx/(double)nkick)*(x0 - cent_x) + 1/cff1x*cos(2*PI*dtunx/(double)nkick)*(xp0 - cent_xp) + cent_xp; yy1 = cff1y*cos(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) - cff2y*sin(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_y; yyp1 = 1/cff2y*sin(2*PI*dtuny/(double)nkick)*(ips0 - cent_y) + 1/cff1y*cos(2*PI*dtuny/(double)nkick)*(yp0 - cent_yp) + cent_yp; /* end: gio-May04 3 different beta at IPs */ } break; case 2: lambda = 2*(rand ()/(double)RAND_MAX - 0.5); xx1 = cos(2*PI*dtunxmax*lambda/(double)nkick)*x0 - betax0*sin(2*PI*dtunxmax*lambda/(double)nkick)*xp0; xxp1 = 1/betax0*sin(2*PI*dtunxmax*lambda/(double)nkick)*x0 + cos(2*PI*dtunxmax*lambda/(double)nkick)*xp0; yy1 = cos(2*PI*dtunymax*lambda/(double)nkick)*ips0 - betay0*sin(2*PI*dtunymax*lambda/(double)nkick)*yp0; yyp1 = 1/betay0*sin(2*PI*dtunymax*lambda/(double)nkick)*ips0 + cos(2*PI*dtunymax*lambda/(double)nkick)*yp0; break; } //these lines about octupoles don't yet take into account the option for a random phase advance!!! if(i_oct==1) { epsx = (x0*x0 + xp0*xp0*betax0*betax0)/betax0; epsy = (ips0*ips0 + yp0*yp0*betay0*betay0)/betay0; // Remember that epsx = 2*actx; epsy = 2*acty cxa = cos(2*PI*(htune + qpxx*epsx + qpxy*epsy)/(double)nkick); sxa = sin(2*PI*(htune + qpxx*epsx + qpxy*epsy)/(double)nkick); cya = cos(2*PI*(vtune + qpxy*epsx + qpyy*epsy)/(double)nkick); sya = sin(2*PI*(vtune + qpxy*epsx + qpyy*epsy)/(double)nkick); } if(i_oct==2) { epsx = x0*x0/sxsize/sxsize; epsy = ips0*ips0/sysize/sysize; // Remember that epsx = 2*actx; epsy = 2*acty cxa = cos(2*PI*(htune + qpxx*epsx)/(double)nkick); sxa = sin(2*PI*(htune + qpxx*epsx)/(double)nkick); cya = cos(2*PI*(vtune + qpyy*epsy)/(double)nkick); sya = sin(2*PI*(vtune + qpyy*epsy)/(double)nkick); } if(i_oct==3) { epsx =(x0*x0 + xp0*xp0*betax0*betax0)/sxsize/sxsize; epsy = (ips0*ips0 + yp0*yp0*betay0*betay0)/sysize/sysize; // Remember that epsx = 2*actx; epsy = 2*acty cxa = cos(2*PI*(htune + qpxx*epsx + qpxy*epsy)/(double)nkick); sxa = sin(2*PI*(htune + qpxx*epsx + qpxy*epsy)/(double)nkick); cya = cos(2*PI*(vtune + qpxy*epsx + qpyy*epsy)/(double)nkick); sya = sin(2*PI*(vtune + qpxy*epsx + qpyy*epsy)/(double)nkick); } switch(i_incoh){ case 0: //set to zero EACH slice centroid if(ishift!=1) { sl_ind = (long)floor((szz0 + 2.*szsize)/(4.*szsize)*NBIN); if (sl_ind >= 0 && sl_ind < NBIN) { cent_x = xs[sl_ind]; cent_y = ys[sl_ind]; cent_xp = xps[sl_ind]; cent_yp = yps[sl_ind]; } else { if (sl_ind < 0) { cent_x = xs[0]; cent_y = ys[0]; cent_xp = xps[0]; cent_yp = yps[0]; } else { cent_x = xs[NBIN-1]; cent_y = ys[NBIN-1]; cent_xp = xps[NBIN-1]; cent_yp = yps[NBIN-1]; } } } xx1 += -cent_x; xxp1 += -cent_xp; yy1 += -cent_y; yyp1 += -cent_yp; break; case 2: //set to zero the bunch centroid xx1 += -xave; xxp1 += -xpave; yy1 += -yave; yyp1 += -ypave; break; case 1: /*ele-May04*/ if (i_damp==1 && (nkick==1 || it%nkick==1)) { xx1 += dampx; yy1 += dampy; } break; } /* next line introduces the possibility to have linear coupling in the machine */ if(i_coupl==1) { xxp1 -= key_xy*yy1; yyp1 -= key_xy*xx1; } /* begin gio-May04 different beta */ /* r11 = cxa*cos(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick) - sxa * sin(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick); r12 = betax0*(sxa*cos(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick) + cxa * sin(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick)); r21 = - r12/betax0/betax0; r22 = r11; */ r11 = cff1x*(cxa*cos(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick) - sxa * sin(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick)); r12 = cff2x*(sxa*cos(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick) + cxa * sin(2*PI*(csix*dpp0+q_secx*dpp0*dpp0/2)/(double)nkick)); r21 = - r12/cff2x/cff2x; r22 = r11/cff1x/cff1x; /* end gio-May04 different beta */ /* horizontal coordinates transformation - linear and with chromatic rotation */ xpr[kmain] = xxp1 * r12 + xx1 * r11; xpp[kmain] = xxp1 * r22 + xx1 * r21; /* begin gio-May04 different beta */ /* r33 = cya*cos(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick) - sya * sin(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick); r34 = betay0*(sya*cos(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick) + cya * sin(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick)); r43 = - r34/betay0/betay0; r44 = r33; */ r33 = cff1y*(cya*cos(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick) - sya * sin(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick)); r34 = cff2y*(sya*cos(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick) + cya * sin(2*PI*(csiy*dpp0+q_secy*dpp0*dpp0/2)/(double)nkick)); r43 = - r34/cff2y/cff2y; r44 = r33/cff1y/cff1y; /* end gio-May04 different beta */ /* vertical coordinates transformation - linear and with chromatic rotation */ ypr[kmain] = yyp1 * r34 + yy1 * r33; ypp[kmain] = yyp1 * r44 + yy1 * r43; /* if one chooses to simulate synchrotron motion too, the longitudinal coordinates are transformed according to a focusing law or debunch or drift in a barrier bucket */ switch(isyn){ case 0: break; case 1: sz[kmain] = csa * szz0 - eta * C/omegas * ssa * dpp0; dp[kmain] = csa * dpp0 + omegas/eta/C * ssa * szz0; break; case 2: sz[kmain] = szz0 - eta*circ*dpp0; break; case 3: sz[kmain] = szz0 - eta*circ*dpp0; if (fabs(sz[kmain]) >= 2*sz0) { sz[kmain] = 4*sz0*fabs(sz[kmain])/sz[kmain] - sz[kmain]; dp[kmain] = -dp[kmain]; } break; case 4: kp1 = coeff3*sin(coeff4*szz0)*tstepp; k1 = coeff5*dpp0*tstepp; kp2 = coeff3*sin(coeff4*(szz0 + k1/2))*tstepp; k2 = coeff5*(dpp0 + kp1/2)*tstepp; kp3 = coeff3*sin(coeff4*(szz0 + k2/2))*tstepp; k3 = coeff5*(dpp0 + kp2/2)*tstepp; kp4 = coeff3*sin(coeff4*(szz0 + k3))*tstepp; k4 = coeff5*(dpp0 + kp3)*tstepp; sz[kmain] = szz0 + k1/6 + k2/3 + k3/3 + k4/6; dp[kmain] = dpp0 + kp1/6 + kp2/3 + kp3/3 + kp4/6; break; case 44: kp1 = coeff3*sin(coeff4*szz0)*tstepp; k1 = coeff5*dpp0*tstepp; kp2 = coeff3*sin(coeff4*(szz0 + k1/2))*tstepp; k2 = coeff5*(dpp0 + kp1/2)*tstepp; kp3 = coeff3*sin(coeff4*(szz0 + k2/2))*tstepp; k3 = coeff5*(dpp0 + kp2/2)*tstepp; kp4 = coeff3*sin(coeff4*(szz0 + k3))*tstepp; k4 = coeff5*(dpp0 + kp3)*tstepp; sz[kmain] = szz0 + k1/6 + k2/3 + k3/3 + k4/6; dp[kmain] = dpp0 + kp1/6 + kp2/3 + kp3/3 + kp4/6; break; case 5: kp1 = (coeff3*sin(coeff4*szz0) + coeffspch*(szz0 - zave)/(szsize*szsize*szsize)*exp(-(szz0 - zave)*(szz0 - zave)/(2*szsize*szsize)))*tstepp; k1 = coeff5*dpp0*tstepp; kp2 = (coeff3*sin(coeff4*(szz0 + k1/2)) + coeffspch*(szz0 - zave)/(szsize*szsize*szsize)*exp(-(szz0 - zave)*(szz0 - zave)/(2*szsize*szsize)))*tstepp; k2 = coeff5*(dpp0 + kp1/2)*tstepp; kp3 = (coeff3*sin(coeff4*(szz0 + k2/2)) + coeffspch*(szz0 - zave)/(szsize*szsize*szsize)*exp(-(szz0 - zave)*(szz0 - zave)/(2*szsize*szsize)))*tstepp; k3 = coeff5*(dpp0 + kp2/2)*tstepp; kp4 = (coeff3*sin(coeff4*(szz0 + k3)) + coeffspch*(szz0 - zave)/(szsize*szsize*szsize)*exp(-(szz0 - zave)*(szz0 - zave)/(2*szsize*szsize)))*tstepp; k4 = coeff5*(dpp0 + kp3)*tstepp; sz[kmain] = szz0 + k1/6 + k2/3 + k3/3 + k4/6; dp[kmain] = dpp0 + kp1/6 + kp2/3 + kp3/3 + kp4/6; break; /* begin frank-May04; new RF options */ case 6: /* for TESLA DR */ if (it%nkick== 1) { kp1 = coeff3*sin(coeff4*szz0)*tstepp*(double)nkick; k1 = coeff5*dpp0*tstepp*(double)nkick; kp2 = coeff3*sin(coeff4*(szz0 + k1/2))*tstepp*(double)nkick; k2 = coeff5*(dpp0 + kp1/2)*tstepp*(double)nkick; kp3 = coeff3*sin(coeff4*(szz0 + k2/2))*tstepp*(double)nkick; k3 = coeff5*(dpp0 + kp2/2)*tstepp*(double)nkick; kp4 = coeff3*sin(coeff4*(szz0 + k3))*tstepp*(double)nkick; k4 = coeff5*(dpp0 + kp3)*tstepp*(double)nkick; sz[kmain] = szz0 + k1 ; dp[kmain] = dpp0 + kp1*sz[kmain]/szz0 ; /* sz[kmain] = szz0 + k1/6 + k2/3 + k3/3 + k4/6; dp[kmain] = dpp0 + kp1/6 + kp2/3 + kp3/3 + kp4/6; */ } break; case 7: /* for CLIC (or NLC?) DR */ if (it%(nkick/2)== 0) { k1 = coeff5*dpp0*tstepp/2.*(double)nkick; k2 = coeff5*(dpp0 + kp1/2)*tstepp/2.*(double)nkick; k3 = coeff5*(dpp0 + kp2/2)*tstepp/2.*(double)nkick; k4 = coeff5*(dpp0 + kp3)*tstepp/2.*(double)nkick; sz[kmain] = szz0 + k1; } if (it%nkick== 0) { kp1 = coeff3*sin(coeff4*szz0)*tstepp*(double)nkick; /* kp2 = coeff3*sin(coeff4*(szz0 + k1/2))*tstepp*(double)nkick; kp3 = coeff3*sin(coeff4*(szz0 + k2/2))*tstepp*(double)nkick; kp4 = coeff3*sin(coeff4*(szz0 + k3))*tstepp*(double)nkick; */ dp[kmain] = dpp0 + kp1*sz[kmain]/szz0 ; /* dp[kmain] = dpp0 + kp1/6 + kp2/3 + kp3/3 + kp4/6; */ } break; case 8: /* for LHC */ k1 = coeff5*dpp0*tstepp; k2 = coeff5*(dpp0 + kp1/2)*tstepp; k3 = coeff5*(dpp0 + kp2/2)*tstepp; k4 = coeff5*(dpp0 + kp3)*tstepp; sz[kmain] = szz0 + k1; if (it%nkick == 0) { kp1 = coeff3*sin(coeff4*szz0)*tstepp*(double)nkick; /* kp2 = coeff3*sin(coeff4*(szz0 + k1/2))*tstepp*(double)nkick; kp3 = coeff3*sin(coeff4*(szz0 + k2/2))*tstepp*(double)nkick; kp4 = coeff3*sin(coeff4*(szz0 + k3))*tstepp*(double)nkick; */ dp[kmain] = dpp0 + kp1*sz[kmain]/szz0 ; /* dp[kmain] = dpp0 + kp1/6 + kp2/3 + kp3/3 + kp4/6; */ } break; /* end frank-May04; new RF options */ case 9: /* gio-May04 */ //sz[kmain] = fmod(szz0 - eta*circ*dpp0,circ); sz[kmain] = szz0 - eta*circ*dpp0; if(sz[kmain]>circ/2) sz[kmain] -= circ; if(sz[kmain]<-circ/2) sz[kmain] += circ; break; } /* if (kmain % 300==0 && it % n_diag==0) fprintf(prot_phase,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",kmain,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); else if (kmain==NPR-1) fprintf(samp_prot,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); */ /* if (it%nkick==1){ if (i_distr_prot==1){ if (kmain==0) fprintf(samp_prot1,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==1) fprintf(samp_prot2,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==2) fprintf(samp_prot3,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==3) fprintf(samp_prot4,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==4) fprintf(samp_prot5,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==5) fprintf(samp_prot6,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); } else{ if (kmain==1) fprintf(samp_prot1,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==1002) fprintf(samp_prot2,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==13006) fprintf(samp_prot3,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==16362) fprintf(samp_prot4,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==112473) fprintf(samp_prot5,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (kmain==258341) fprintf(samp_prot6,"%d \t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\t%13.8e\n",it,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); } } */ //if (mark[kmain]==1) //fprintf(track_pr,"%d %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e %d\n",kmain,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain], it); /* if (it==1) //|| it==24 ||it==36 || it==48 || it==60 || it==72 || it==84 || it==96 || it==120 ) fprintf(prot_ini,"%d %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",kmain,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); if (it==500*nkick) fprintf(track_pr,"%d %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",kmain,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); */ /* if(fabs(xpr[kmain])>=rpipex || fabs(ypr[kmain])>=rpipey || -0.5*coeff5/coeff3*coeff4*dp[kmain]*dp[kmain]-cos(coeff4*sz[kmain])>1) */ if(i_loss==1) { if(fabs(xpr[kmain])>=rpipex || fabs(ypr[kmain])>=rpipey) { //printf("jmain=%d, xpr[jmain]=%lf, xpp[jmain]=%lf\n",kmain,xpr[kmain],xpp[kmain]); // fprintf(track_pr,"%d %d %13.8e %13.8e %13.8e %13.8e %13.8e %13.8e\n",it,kmain,xpr[kmain],xpp[kmain],ypr[kmain],ypp[kmain],sz[kmain],dp[kmain]); xpr[kmain] = xpr[npr_eff]; ypr[kmain] = ypr[npr_eff]; sz[kmain] = sz[npr_eff]; xpp[kmain] = xpp[npr_eff]; ypp[kmain] = ypp[npr_eff]; dp[kmain] = dp[npr_eff]; ++npr_eff; } } } /* if (it %nkick==1) //(it % (n_diag/2)==0) { fprintf(track_pr,"\n"); fprintf(track_pr,"\n"); } */ if (it % n_diag == 0 || it==1) { if(isyn!=3 && isyn!=9) verteil (sz,NPR-npr_eff,hdp,NDC,indxp); else verteil2 (sz,NPR-npr_eff,hdp,NDC,indxp); for (lmain=0; lmain %g \n",(double)(npr_eff)/(double)(NPR)*100); /*for(kmain=0;kmain<499;kmain++) { enedis[kmain] /= qt_imp; fprintf (elwall_ene, "%13.8e %13.8e\n",(double)kmain*enestep,enedis[kmain]); } */ fprintf (prot_inphase, "Fraction of e- on pipe above 1000eV -> %g \n",enedis[499]/qt_imp); fclose(bunch_pr); fclose(headtail_pr); fclose(prot_phase); fclose(prot_inphase); fclose(prot_distr); fclose(prot_ini); fclose(ele_ini); fclose(prot_distr); fclose(samp_prot); fclose(samp_prot1); fclose(samp_prot2); fclose(samp_prot3); fclose(samp_prot4); fclose(samp_prot5); fclose(samp_prot6); fclose(track_pr); fclose(el_traj); fclose(el_traj2); fclose(el_traj3); fclose(halo_p); fclose(campo_dat); fclose(campo0_dat); fclose(campo1_dat); fclose(sey_file); fclose(el_diagn); fclose(froz_pot1); fclose(froz_pot2); //fclose(elwall_ene); if (i_phas_adv==2 || i_phas_adv==3) fclose(beta_adv); }