forked from QEDjl-project/QEDprocesses.jl
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathphase_spaces.jl
261 lines (224 loc) · 10.3 KB
/
phase_spaces.jl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
#####################
# phase spaces
#
# This file contains a collection of types and functions to handle phase spaces
# for scattering processes.
#
# TODO: ship to interfaces!
#####################
using StaticArrays
import QEDbase.is_particle
import QEDbase.is_anti_particle
import QEDbase.is_fermion
import QEDbase.is_boson
import QEDbase.is_incoming
import QEDbase.is_outgoing
import QEDbase.mass
import QEDbase.charge
import QEDbase:
is_particle,
is_anti_particle,
is_fermion,
is_boson,
is_incoming,
is_outgoing,
mass,
charge,
AbstractFourMomentum,
AbstractParticleType,
AbstractParticle
abstract type AbstractCoordinateSystem end
struct SphericalCoordinateSystem <: AbstractCoordinateSystem end
abstract type AbstractFrameOfReference end
struct CenterOfMomentumFrame <: AbstractFrameOfReference end
struct ElectronRestFrame <: AbstractFrameOfReference end
abstract type AbstractPhasespaceDefinition end
"""
PhasespaceDefinition(coord_sys::AbstractCoordinateSystem, frame::AbstractFrameOfReference)
Convenient type to dispatch on coordiante systems and frames of reference.
"""
struct PhasespaceDefinition{CS<:AbstractCoordinateSystem,F<:AbstractFrameOfReference} <:
AbstractPhasespaceDefinition
coord_sys::CS
frame::F
end
# abstract type for generic phase spaces
#
# Currently, elements can be either four-momenta, or real numbers,
# i.e. coordinates.
AbstractPhasespaceElement = Union{AbstractFourMomentum,Real}
# utility functions
@inline function _check_in_phase_space_dimension(
proc::AbstractProcessDefinition,
::AbstractModelDefinition,
in_phase_space::AbstractVecOrMat{T},
) where {T<:AbstractFourMomentum}
return size(in_phase_space, 1) == number_incoming_particles(proc) || throw(
DimensionMismatch(
"the number of incoming particles <$(number_incoming_particles(proc))> is inconsistent with input size <$(size(in_phase_space,1))>",
),
)
end
@inline function _check_out_phase_space_dimension(
proc::AbstractProcessDefinition,
::AbstractModelDefinition,
out_phase_space::AbstractVecOrMat{T},
) where {T<:AbstractFourMomentum}
return size(out_phase_space, 1) == number_outgoing_particles(proc) || throw(
DimensionMismatch(
"the number of outgoing particles <$(number_outgoing_particles(proc))> is inconsistent with input size <$(size(out_phase_space,1))>",
),
)
end
@inline function _check_in_phase_space_dimension(
proc::AbstractProcessDefinition,
model::AbstractModelDefinition,
in_phase_space::AbstractVecOrMat{T},
) where {T<:Real}
return size(in_phase_space, 1) == in_phase_space_dimension(proc, model) || throw(
DimensionMismatch(
"the dimension of the in-phase-space <$(in_phase_space_dimension(proc,model))> is inconsistent with input size <$(size(in_phase_space,1))>",
),
)
end
@inline function _check_out_phase_space_dimension(
proc::AbstractProcessDefinition,
model::AbstractModelDefinition,
out_phase_space::AbstractVecOrMat{T},
) where {T<:Real}
return size(out_phase_space, 1) == out_phase_space_dimension(proc, model) || throw(
DimensionMismatch(
"the dimension of the out-phase-space <$(out_phase_space_dimension(proc,model))> is inconsistent with input size <$(size(out_phase_space,1))>",
),
)
end
"""
ParticleStateful <: AbstractParticle
Representation of a particle with a state. It has four fields:
- `dir::ParticleDirection`: The direction of the particle, `QEDbase.Incoming()` or `QEDbase.Outgoing()`.
- `species::AbstractParticleType`: The species of the particle, `QEDbase.Electron()`, `QEDbase.Positron()` etc.
- `mom::AbstractFourMomentum`: The momentum of the particle.
- `spin_or_pol::AbstractSpinOrPolarization`: The spin or polarization of the particle, `QEDbase.SpinUp()`, `QEDbase.PolX() etc. Can only use spins with fermions and polarizations with bosons.
Overloads for `QEDbase.is_fermion`, `QEDbase.is_boson`, `QEDbase.is_particle`, `QEDbase.is_anti_particle`, `QEDbase.is_incoming`, `QEDbase.is_outgoing`, `QEDbase.mass`, and `QEDbase.charge` are provided, delegating the call to the correct field and thus implementing the `QEDbase.AbstractParticle` interface.
The legality of the combination of `species` and `spin_or_pol` is checked on construction. If, for example, the construction of an `Electron()` with a polarization is attempted, an [`InvalidInputError`](@ref) is thrown.
"""
struct ParticleStateful{ElType<:AbstractFourMomentum} <: AbstractParticle
dir::ParticleDirection
species::AbstractParticleType
mom::ElType
spin_or_pol::AbstractSpinOrPolarization
function ParticleStateful(
dir::ParticleDirection, species::Species, mom::ElType, spin::Spin=AllSpin()
) where {Species<:FermionLike,ElType<:AbstractFourMomentum,Spin<:AbstractSpin}
# constructor for fermions with spin
return new{ElType}(dir, species, mom, spin)
end
function ParticleStateful(
dir::ParticleDirection, species::Species, mom::ElType, pol::Pol=AllPol()
) where {Species<:BosonLike,ElType<:AbstractFourMomentum,Pol<:AbstractPolarization}
# constructor for bosons with polarization
return new{ElType}(dir, species, mom, pol)
end
end
@inline is_incoming(particle::ParticleStateful) = is_incoming(particle.dir)
@inline is_outgoing(particle::ParticleStateful) = is_outgoing(particle.dir)
@inline is_fermion(particle::ParticleStateful) = is_fermion(particle.species)
@inline is_boson(particle::ParticleStateful) = is_boson(particle.species)
@inline is_particle(particle::ParticleStateful) = is_particle(particle.species)
@inline is_anti_particle(particle::ParticleStateful) = is_anti_particle(particle.species)
@inline mass(particle::ParticleStateful) = mass(particle.species)
@inline charge(particle::ParticleStateful) = charge(particle.species)
@inline _spin(::Species, particle::ParticleStateful) where {Species<:FermionLike} =
particle.spin_or_pol
@inline spin(particle::ParticleStateful) = _spin(particle.species, particle)
@inline _polarization(::Species, particle::ParticleStateful) where {Species<:BosonLike} =
particle.spin_or_pol
@inline polarization(particle::ParticleStateful) = _polarization(particle.species, particle)
"""
PhaseSpacePoint
Representation of a point in the phase space of a process. Contains the process ([`AbstractProcessDefinition`](@ref)), the model ([`AbstractModelDefinition`](@ref)), the phase space definition ([`AbstractPhasespaceDefinition`]), and stateful incoming and outgoing particles ([`ParticleStateful`](@ref)).
The legality of the combination of the given process and the incoming and outgoing particles is checked on construction. If the numbers of particles mismatch, the types of particles mismatch (note that order is important), or incoming particles have an `Outgoing` direction, an error is thrown.
"""
struct PhaseSpacePoint{
PROC<:AbstractProcessDefinition,
MODEL<:AbstractModelDefinition,
PSDEF<:AbstractPhasespaceDefinition,
PhaseSpaceElementType<:AbstractFourMomentum,
N_IN_PARTICLES,
N_OUT_PARTICLES,
}
proc::PROC
model::MODEL
ps_def::PSDEF
in_particles::SVector{N_IN_PARTICLES,ParticleStateful{PhaseSpaceElementType}}
out_particles::SVector{N_OUT_PARTICLES,ParticleStateful{PhaseSpaceElementType}}
function PhaseSpacePoint(
proc::PROC, model::MODEL, ps_def::PSDEF, in_p::IN_P, out_p::OUT_P
) where {
PROC<:AbstractProcessDefinition,
MODEL<:AbstractModelDefinition,
PSDEF<:AbstractPhasespaceDefinition,
PhaseSpaceElementType<:AbstractFourMomentum,
IN_P<:AbstractVector{ParticleStateful{PhaseSpaceElementType}},
OUT_P<:AbstractVector{ParticleStateful{PhaseSpaceElementType}},
}
length(incoming_particles(proc)) == length(in_p) || throw(
InvalidInputError(
"the number of incoming particles given by the process ($(incoming_particles(proc))) mismatches the number of given stateful incoming particles ($(length(in_p)))",
),
)
length(outgoing_particles(proc)) == length(out_p) || throw(
InvalidInputError(
"the number of outgoing particles given by the process ($(outgoing_particles(proc))) mismatches the number of given stateful outgoing particles ($(length(out_p)))",
),
)
for (proc_p, p) in zip(incoming_particles(proc), in_p)
proc_p == p.species || throw(
InvalidInputError(
"process given particle species ($(proc_p)) does not match stateful particle species ($(p.species))",
),
)
is_incoming(p) || throw(
InvalidInputError(
"stateful particle $(p) is given as an incoming particle but is outgoing",
),
)
end
for (proc_p, p) in zip(outgoing_particles(proc), out_p)
proc_p == p.species || throw(
InvalidInputError(
"process given particle species ($(proc_p)) does not match stateful particle species ($(p.species))",
),
)
is_outgoing(p) || throw(
InvalidInputError(
"stateful particle $(p) is given as an outgoing particle but is incoming",
),
)
end
return new{PROC,MODEL,PSDEF,PhaseSpaceElementType,length(in_p),length(out_p)}(
proc, model, ps_def, in_p, out_p
)
end
end
"""
Base.getindex(psp::PhaseSpacePoint, dir::Incoming, n::Int)
Overload for the array indexing operator `[]`. Returns the nth incoming particle in this phase space point.
"""
function Base.getindex(psp::PhaseSpacePoint, ::Incoming, n::Int)
return psp.in_particles[n]
end
"""
Base.getindex(psp::PhaseSpacePoint, dir::Outgoing, n::Int)
Overload for the array indexing operator `[]`. Returns the nth outgoing particle in this phase space point.
"""
function Base.getindex(psp::PhaseSpacePoint, ::Outgoing, n::Int)
return psp.out_particles[n]
end
"""
momentum(psp::PhaseSpacePoint, dir::ParticleDirection, n::Int)
Returns the momentum of the `n`th particle in the given [`PhaseSpacePoint`](@ref) which has direction `dir`. If `n` is outside the valid range for this phase space point, a `BoundsError` is thrown.
"""
function momentum(psp::PhaseSpacePoint, dir::ParticleDirection, n::Int)
return psp[dir, n].mom
end