Skip to content

Commit

Permalink
Add short linac segment example. (#774)
Browse files Browse the repository at this point in the history
* Add short linac segment example.
* Add Hminus particle input to app.
* Update particle type to H anion.
* Update input_linac_segment.in
   Reduce resolution to reduce runtime.
* Update run_linac_segment.py
  Reduce resolution for reduced runtime
* Update analysis script values+tolerance.
* Formatting
  • Loading branch information
cemitch99 authored Jan 4, 2025
1 parent 72eb144 commit e9d4df8
Show file tree
Hide file tree
Showing 7 changed files with 883 additions and 0 deletions.
1 change: 1 addition & 0 deletions docs/source/usage/examples.rst
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,7 @@ Virtual Test Stands
examples/pytorch_surrogate_model/README.rst
examples/apochromatic/README.rst
examples/fodo_tune/README.rst
examples/linac_segment/README.rst


Unit tests
Expand Down
16 changes: 16 additions & 0 deletions examples/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -1040,3 +1040,19 @@ add_impactx_test(rotation-xy.py
examples/rotation/analysis_rotation_xy.py
OFF # no plot script yet
)

# PIP-II linac segment #########################################################
#
# with space charge
add_impactx_test(linac-segment
examples/linac_segment/input_linac_segment.in
ON # ImpactX MPI-parallel
examples/linac_segment/analysis_linac_segment.py
OFF # no plot script yet
)
add_impactx_test(linac-segment.py
examples/linac_segment/run_linac_segment.py
OFF # ImpactX MPI-parallel
examples/linac_segment/analysis_linac_segment.py
OFF # no plot script yet
)
50 changes: 50 additions & 0 deletions examples/linac_segment/README.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
.. _examples-linac-segment:

PIP-II Linac Segment
====================

This example illustrates a small segment of the PIP-II linac, which includes the last RF cavity in the MEBT section (a quarter-wave resonator cavity for longitudinal bunching) and the first
RF cavity in the succeeding cryomodule (a superconducting half-wave resonator for acceleration).

For simplicity, aperture restrictions have been removed, and there are no active kickers. RF cavity and solenoid field maps are represented using the Fourier coefficients of the longitudinal fields on-axis.

The nominal beam current is 4.845 mA (29.83 pC at 162.5 MHz), which assumes ~3% beam loss relative to the 5 mA current entering the MEBT.

In this test, the initial and final values of :math:`\sigma_x`, :math:`\sigma_y`, :math:`\sigma_t`, :math:`\epsilon_x`, :math:`\epsilon_y`, and :math:`\epsilon_t` must agree with nominal values.


Run
---

This example can be run **either** as:

* **Python** script: ``python3 run_linac_segment.py`` or
* ImpactX **executable** using an input file: ``impactx input_linac_segment.in``

For `MPI-parallel <https://www.mpi-forum.org>`__ runs, prefix these lines with ``mpiexec -n 4 ...`` or ``srun -n 4 ...``, depending on the system.

.. tab-set::

.. tab-item:: Python: Script

.. literalinclude:: run_linac_segment.py
:language: python3
:caption: You can copy this file from ``examples/linac_segment/run_linac_segment.py``.

.. tab-item:: Executable: Input File

.. literalinclude:: input_linac_segment.in
:language: ini
:caption: You can copy this file from ``examples/linac_segment/input_linac_segment.in``.


Analyze
-------

We run the following script to analyze correctness:

.. dropdown:: Script ``analysis_linac_segment.py``

.. literalinclude:: analysis_linac_segment.py
:language: python3
:caption: You can copy this file from ``examples/linac_segment/analysis_linac_segment.py``.
98 changes: 98 additions & 0 deletions examples/linac_segment/analysis_linac_segment.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
#!/usr/bin/env python3
#
# Copyright 2022-2025 ImpactX contributors
# Authors: Axel Huebl, Chad Mitchell
# License: BSD-3-Clause-LBNL
#


import numpy as np
import openpmd_api as io
from scipy.stats import moment


def get_moments(beam):
"""Calculate standard deviations of beam position & momenta
and emittance values
Returns
-------
sigx, sigy, sigt, emittance_x, emittance_y, emittance_t
"""
sigx = moment(beam["position_x"], moment=2) ** 0.5 # variance -> std dev.
sigpx = moment(beam["momentum_x"], moment=2) ** 0.5
sigy = moment(beam["position_y"], moment=2) ** 0.5
sigpy = moment(beam["momentum_y"], moment=2) ** 0.5
sigt = moment(beam["position_t"], moment=2) ** 0.5
sigpt = moment(beam["momentum_t"], moment=2) ** 0.5

epstrms = beam.cov(ddof=0)
emittance_x = (sigx**2 * sigpx**2 - epstrms["position_x"]["momentum_x"] ** 2) ** 0.5
emittance_y = (sigy**2 * sigpy**2 - epstrms["position_y"]["momentum_y"] ** 2) ** 0.5
emittance_t = (sigt**2 * sigpt**2 - epstrms["position_t"]["momentum_t"] ** 2) ** 0.5

return (sigx, sigy, sigt, emittance_x, emittance_y, emittance_t)


# initial/final beam
series = io.Series("diags/openPMD/monitor.h5", io.Access.read_only)
last_step = list(series.iterations)[-1]
initial = series.iterations[1].particles["beam"].to_df()
final = series.iterations[last_step].particles["beam"].to_df()

# compare number of particles
num_particles = 10000
assert num_particles == len(initial)
assert num_particles == len(final)

print("Initial Beam:")
sigx, sigy, sigt, emittance_x, emittance_y, emittance_t = get_moments(initial)
print(f" sigx={sigx:e} sigy={sigy:e} sigt={sigt:e}")
print(
f" emittance_x={emittance_x:e} emittance_y={emittance_y:e} emittance_t={emittance_t:e}"
)

atol = 0.0 # ignored
rtol = 3.0 * num_particles**-0.5 # from random sampling of a smooth distribution
print(f" rtol={rtol} (ignored: atol~={atol})")

assert np.allclose(
[sigx, sigy, sigt, emittance_x, emittance_y, emittance_t],
[
2.300582e-03,
2.350613e-03,
9.939595e-02,
3.057531e-06,
2.960797e-06,
5.487870e-06,
],
rtol=rtol,
atol=atol,
)


print("")
print("Final Beam:")
sigx, sigy, sigt, emittance_x, emittance_y, emittance_t = get_moments(final)
print(f" sigx={sigx:e} sigy={sigy:e} sigt={sigt:e}")
print(
f" emittance_x={emittance_x:e} emittance_y={emittance_y:e} emittance_t={emittance_t:e}"
)

atol = 0.0 # ignored
rtol = 3.0 * num_particles**-0.5 # from random sampling of a smooth distribution
print(f" rtol={rtol} (ignored: atol~={atol})")

assert np.allclose(
[sigx, sigy, sigt, emittance_x, emittance_y, emittance_t],
[
1.451059e-03,
1.467217e-03,
2.228492e-02,
3.009007e-06,
3.104729e-06,
5.371249e-06,
],
rtol=rtol,
atol=atol,
)
Loading

0 comments on commit e9d4df8

Please sign in to comment.