Thermal-FIST 1.5
Package for hadron resonance gas model applications
Loading...
Searching...
No Matches
ThermalModelPCEAnnihilation.cpp
Go to the documentation of this file.
2
3#include <iostream>
4#include <stdexcept>
5
6using namespace std;
7
8namespace thermalfist {
9
11 bool FreezeLonglived,
12 double LonglivedResoWidthCut) :
13 ThermalModelPCE(THMbase, FreezeLonglived, LonglivedResoWidthCut)
14 {
15 m_PionAnnihilationNumber = 5.;
16 auto stability_flags = RecalculateStabilityFlags();
17
18 SetStabilityFlags(stability_flags);
19 }
20
22 {
24
25 // Indices of conserved hadron numbers
26 // In case of baryon-antibaryon annihilation, the index corresponds to the baryon
27 m_StableNormal.clear();
28 for (int i = 0; i < m_StabilityFlags.size(); ++i) {
29 if (m_StabilityFlags[i] == 1) {
30 long long tpdgid = ThermalModel()->TPS()->Particle(i).PdgId();
31 int idanti = ThermalModel()->TPS()->PdgToId(-tpdgid);
32 if (tpdgid == 211 || tpdgid == 111 || tpdgid == -211
33 || (idanti != -1 && m_StabilityFlags[idanti] == 2))
34 continue;
35
36 m_StableNormal.push_back(i);
37 }
38 }
39
40 // Indices of annihilating baryons and antibaryons
41 m_StableAnnihilate.clear();
42 m_StableAnnihilateAnti.clear();
43 for (int i = 0; i < m_StabilityFlags.size(); ++i) {
44 if (m_StabilityFlags[i] == 2 && ThermalModel()->TPS()->Particle(i).PdgId() > 0) {
45 m_StableAnnihilate.push_back(i);
46 long long tpdgid = ThermalModel()->TPS()->Particle(i).PdgId();
47 int idanti = ThermalModel()->TPS()->PdgToId(-tpdgid);
48 if (idanti == -1) {
49 throw std::runtime_error("ThermalModelPCEAnnihilation: No antibaryon with pdg code " + std::to_string(-tpdgid) + " exists in the list!");
50 }
51 m_StableAnnihilateAnti.push_back(idanti);
52 }
53 }
54
55 // Indices of the pions
56 m_Pions.clear();
57 for (int i = 0; i < m_StabilityFlags.size(); ++i) {
58 long long tpdgid = ThermalModel()->TPS()->Particle(i).PdgId();
59 if (tpdgid == 211 || tpdgid == -211 || tpdgid == 111) {
60 m_Pions.push_back(i);
61 }
62 }
63
64 }
65
67 {
69 throw std::runtime_error("ThermalModelPCE::CalculatePCE: Tried to make a PCE calculation without setting the chemical freze-out! Call ThermalModelPCE::SetChemicalFreezeout() first.");
70 }
71
74 }
75
76 double T = param;
77 if (mode == PCEMode::AtFixedVolume) {
78 // Initial guess for the new temperature
79 T = m_ParametersCurrent.T * pow(m_ParametersCurrent.V / param, 1. / 3.);
80 m_ParametersCurrent.V = param;
81 }
82 else {
83 // Initial guess for the new volume
85 }
86
87
88 BroydenEquationsPCEAnnihilation eqs(this, mode);
89 Broyden broydn(&eqs);
90
91 // PCE parameters: volume/temperature + chemical potentials
92 std::vector<double> PCEParams(m_StableNormal.size() + m_StableAnnihilate.size() + m_Pions.size(), 0.);
93 int stab_index = 0;
94
95 // First go through all the stable species that do not take part in annihilations
96 // This excluded the chemical potentials of annihilating antibaryons since they are not independent
97 for (int ind = 0; ind < m_StableNormal.size(); ++ind) {
98 int i = m_StableNormal[ind];
99
100 PCEParams[stab_index] = m_ChemCurrent[i] * T / m_ParametersCurrent.T + ThermalModel()->TPS()->Particle(i).Mass() * (1. - T / m_ParametersCurrent.T);
101 stab_index++;
102 }
103
104 // Then the chemical potentials of annihilating baryons
105 for (int ind = 0; ind < m_StableAnnihilate.size(); ++ind) {
106 int i = m_StableAnnihilate[ind];
107
108 PCEParams[stab_index] = m_ChemCurrent[i] * T / m_ParametersCurrent.T + ThermalModel()->TPS()->Particle(i).Mass() * (1. - T / m_ParametersCurrent.T);
109 stab_index++;
110 }
111
112 // Finally, the three pions
113 for (int ind = 0; ind < m_Pions.size(); ++ind) {
114 int i = m_Pions[ind];
115
116 PCEParams[stab_index] = m_ChemCurrent[i] * T / m_ParametersCurrent.T + ThermalModel()->TPS()->Particle(i).Mass() * (1. - T / m_ParametersCurrent.T);
117 stab_index++;
118 }
119
121
122 // The final parameter is the volume/temperature
123 //PCEParams.push_back(m_ParametersCurrent.V);
124 if (mode == PCEMode::AtFixedTemperature)
125 PCEParams.push_back(m_ParametersCurrent.V);
126 else
127 PCEParams.push_back(m_ParametersCurrent.T);
128
130 //broydn.UseNewton(true);
131 //PCEParams = broydn.Solve(PCEParams);
132 PCEParams = broydn.Solve(PCEParams, &crit);
133
134 m_ChemCurrent = m_model->ChemicalPotentials();
135 if (mode == 0)
136 m_ParametersCurrent.V = PCEParams[PCEParams.size() - 1];
137 else
138 m_ParametersCurrent.T = PCEParams[PCEParams.size() - 1];
139
140 m_model->CalculateFeeddown();
141
142 m_IsCalculated = true;
143 }
144
146 {
147 for (int i = 0; i < m_StableMapTo.size(); ++i)
148 if (m_StableMapTo[i] == globalid)
149 return i;
150 return -1;
151 }
152
153 std::vector<double> ThermalModelPCEAnnihilation::StableChemsFromBroydenInput(const std::vector<double>& x)
154 {
155 std::vector<double> ret(m_StableComponentsNumber);
156
157 int stab_index = 0;
158
159 // First go through all the stable species that do not take part in annihilations
160 for (int ind = 0; ind < m_StableNormal.size(); ++ind) {
161 int iglob = m_StableNormal[ind];
162
163 int istab = StableHadronIndexByGlobalId(iglob);
164
165 ret[istab] = x[stab_index];
166
167 stab_index++;
168 }
169
170 // Then the chemical potentials of annihilating baryons
171 for (int ind = 0; ind < m_StableAnnihilate.size(); ++ind) {
172 int iglob = m_StableAnnihilate[ind];
173
174 int istab = StableHadronIndexByGlobalId(iglob);
175
176 ret[istab] = x[stab_index];
177
178 stab_index++;
179 }
180
181 // The three pions
182 for (int ind = 0; ind < m_Pions.size(); ++ind) {
183 int iglob = m_Pions[ind];
184
185 int istab = StableHadronIndexByGlobalId(iglob);
186
187 ret[istab] = x[stab_index];
188
189 stab_index++;
190 }
191
192 // Finally, the chemical potentials of the annihilating antibaryons
193 for (int ind = 0; ind < m_StableAnnihilate.size(); ++ind) {
194 int iglob = m_StableAnnihilate[ind];
195 int iglobanti = m_StableAnnihilateAnti[ind];
196
197 int istab = StableHadronIndexByGlobalId(iglob);
198 int istabanti = StableHadronIndexByGlobalId(iglobanti);
199
200 int istabpip = StableHadronIndexByGlobalId(ThermalModel()->TPS()->PdgToId( 211));
201 int istabpiz = StableHadronIndexByGlobalId(ThermalModel()->TPS()->PdgToId( 111));
202 int istabpim = StableHadronIndexByGlobalId(ThermalModel()->TPS()->PdgToId(-211));
203
204 // mu_N + mu_{\bar{N}} = <npi> * (mu_pi+ + mu_pi- + mu_pi0) / 3
205 ret[istabanti] = -ret[istab] + m_PionAnnihilationNumber / 3. * (ret[istabpip] + ret[istabpiz] + ret[istabpim]);
206
207 stab_index++;
208 }
209
210 return ret;
211 }
212
213 std::vector<int> ThermalModelPCEAnnihilation::RecalculateStabilityFlags(const vector<long long int> &annihilationpdgs) {
215 for(auto& tpdg: annihilationpdgs) {
216 // Baryon
217 int id = ThermalModel()->TPS()->PdgToId(tpdg);
218 if (id != -1)
219 stability_flags[id] = 2;
220
221 // Antibaryon
222 id = ThermalModel()->TPS()->PdgToId(-tpdg);
223 if (id != -1)
224 stability_flags[id] = 2;
225 }
226 return stability_flags;
227 }
228
231 std::vector<double> ThermalModelPCEAnnihilation::BroydenEquationsPCEAnnihilation::Equations(const std::vector<double>& x)
232 {
233 std::vector<double> ret(x.size(), 0.);
234
236 std::vector<double> ChemStable = m_THM->StableChemsFromBroydenInput(x);
237
239 std::vector<double> Chem(m_THM->ThermalModel()->Densities().size(), 0.);
240 for (int i = 0; i < m_THM->m_EffectiveCharges.size(); ++i) {
241 for (int is = 0; is < m_THM->m_EffectiveCharges[i].size(); ++is) {
242 Chem[i] += m_THM->m_EffectiveCharges[i][is] * ChemStable[is];
243 }
244 }
245
246 ThermalModelBase *model = m_THM->ThermalModel();
247 model->SetChemicalPotentials(Chem);
248
249
250 if (m_Mode == PCEMode::AtFixedTemperature) {
251 const double& Vtmp = x[x.size() - 1];
252 m_THM->m_ParametersCurrent.V = Vtmp;
253 }
254 else {
255 const double& Ttmp = x[x.size() - 1];
256 m_THM->m_ParametersCurrent.T = Ttmp;
257 }
258 double V = m_THM->m_ParametersCurrent.V;
259 model->SetParameters(m_THM->m_ParametersCurrent);
260
262 model->CalculateDensities();
263
264 // Calculate the conserved quantities
265 int stab_index = 0;
266
267 // Total yields of stable species that do not take part in annihilations
268 for (int ind = 0; ind < m_THM->m_StableNormal.size(); ++ind) {
269 int iglob = m_THM->m_StableNormal[ind];
270
271 int istab = m_THM->StableHadronIndexByGlobalId(iglob);
272
273 double totdens = 0.;
274 for (int i = 0; i < m_THM->m_EffectiveCharges.size(); ++i) {
275 totdens += m_THM->m_EffectiveCharges[i][istab] * model->Densities()[i];
276 }
277
278 ret[stab_index] = (totdens * V) / (m_THM->m_StableDensitiesInit[istab] * m_THM->m_ParametersInit.V) - 1.;
279
280 stab_index++;
281 }
282
283 // Net numbers of annihilating baryons
284 for (int ind = 0; ind < m_THM->m_StableAnnihilate.size(); ++ind) {
285 int iglob = m_THM->m_StableAnnihilate[ind];
286 int iglobanti = m_THM->m_StableAnnihilateAnti[ind];
287
288 int istab = m_THM->StableHadronIndexByGlobalId(iglob);
289 int istabanti = m_THM->StableHadronIndexByGlobalId(iglobanti);
290
291 double totdens = 0.;
292 for (int i = 0; i < m_THM->m_EffectiveCharges.size(); ++i) {
293 totdens += m_THM->m_EffectiveCharges[i][istab] * model->Densities()[i];
294 totdens -= m_THM->m_EffectiveCharges[i][istabanti] * model->Densities()[i];
295 }
296
297 double netInit = (m_THM->m_StableDensitiesInit[istab] - m_THM->m_StableDensitiesInit[istabanti]) * m_THM->m_ParametersInit.V;
298 if (abs(netInit) > 1.e-12) {
299 ret[stab_index] = (totdens * V) / netInit - 1.;
300 }
301 else {
302 double sumInit = (m_THM->m_StableDensitiesInit[istab] + m_THM->m_StableDensitiesInit[istabanti]) * m_THM->m_ParametersInit.V;
303 ret[stab_index] = totdens * V / sumInit;
304 }
305
306 stab_index++;
307 }
308
309 // Net-pion numbers
310 int istabpip = m_THM->StableHadronIndexByGlobalId(m_THM->ThermalModel()->TPS()->PdgToId(211));
311 int istabpim = m_THM->StableHadronIndexByGlobalId(m_THM->ThermalModel()->TPS()->PdgToId(-211));
312 int istabpiz = m_THM->StableHadronIndexByGlobalId(m_THM->ThermalModel()->TPS()->PdgToId(111));
313 {
314 // pi+ - pi-
315 double netpicurrent = 0.;
316 for (int i = 0; i < m_THM->m_EffectiveCharges.size(); ++i) {
317 netpicurrent += m_THM->m_EffectiveCharges[i][istabpip] * model->Densities()[i];
318 netpicurrent -= m_THM->m_EffectiveCharges[i][istabpim] * model->Densities()[i];
319 }
320 netpicurrent *= V;
321
322 double netpiinit = (m_THM->m_StableDensitiesInit[istabpip] - m_THM->m_StableDensitiesInit[istabpim]) * m_THM->m_ParametersInit.V;
323
324 if (abs(netpiinit) > 1.e-12) {
325 ret[stab_index] = netpicurrent / netpiinit - 1.;
326 }
327 else {
328 double sumpiinit = (m_THM->m_StableDensitiesInit[istabpip] + m_THM->m_StableDensitiesInit[istabpim]) * m_THM->m_ParametersInit.V;
329 ret[stab_index] = netpicurrent / sumpiinit;
330 }
331
332 stab_index++;
333
334 // pi+ - pi0
335 double pippi0current = 0.;
336 for (int i = 0; i < m_THM->m_EffectiveCharges.size(); ++i) {
337 pippi0current += m_THM->m_EffectiveCharges[i][istabpip] * model->Densities()[i];
338 pippi0current -= m_THM->m_EffectiveCharges[i][istabpiz] * model->Densities()[i];
339 }
340 pippi0current *= V;
341
342 double pippi0init = (m_THM->m_StableDensitiesInit[istabpip] - m_THM->m_StableDensitiesInit[istabpiz]) * m_THM->m_ParametersInit.V;
343
344 if (abs(pippi0init) > 1.e-12) {
345 ret[stab_index] = pippi0current / pippi0init - 1.;
346 }
347 else {
348 double sumpippi0init = (m_THM->m_StableDensitiesInit[istabpip] + m_THM->m_StableDensitiesInit[istabpiz]) * m_THM->m_ParametersInit.V;
349 ret[stab_index] = pippi0current / sumpippi0init;
350 }
351
352 stab_index++;
353 }
354
355 // Conservation of (N + \bar{N}) / 2 + (pi+ + pi- + pi0)/<npi>
356 {
357 double tret = 0.;
358 double tretinit = 0.;
359
360 // Pions
361 for (int i = 0; i < m_THM->m_EffectiveCharges.size(); ++i) {
362 tret += m_THM->m_EffectiveCharges[i][istabpip] * model->Densities()[i] * 2. / m_THM->m_PionAnnihilationNumber;
363 tret += m_THM->m_EffectiveCharges[i][istabpiz] * model->Densities()[i] * 2. / m_THM->m_PionAnnihilationNumber;
364 tret += m_THM->m_EffectiveCharges[i][istabpim] * model->Densities()[i] * 2. / m_THM->m_PionAnnihilationNumber;
365 }
366
367 // Initial pions
368 tretinit += (m_THM->m_StableDensitiesInit[istabpip] + m_THM->m_StableDensitiesInit[istabpiz] +
369 m_THM->m_StableDensitiesInit[istabpim]) * 2. / m_THM->m_PionAnnihilationNumber;
370
371 // Baryon + antibaryon
372 for (int ind = 0; ind < m_THM->m_StableAnnihilate.size(); ++ind) {
373 int iglob = m_THM->m_StableAnnihilate[ind];
374 int iglobanti = m_THM->m_StableAnnihilateAnti[ind];
375
376 int istab = m_THM->StableHadronIndexByGlobalId(iglob);
377 int istabanti = m_THM->StableHadronIndexByGlobalId(iglobanti);
378
379
380 double totdens = 0.;
381 for (int i = 0; i < m_THM->m_EffectiveCharges.size(); ++i) {
382 tret += m_THM->m_EffectiveCharges[i][istab] * model->Densities()[i];
383 tret += m_THM->m_EffectiveCharges[i][istabanti] * model->Densities()[i];
384 }
385
386 tretinit += m_THM->m_StableDensitiesInit[istab] + m_THM->m_StableDensitiesInit[istabanti];
387 }
388
389 ret[stab_index] = (tret * V) / (tretinit * m_THM->m_ParametersInit.V) - 1.;
390
391 stab_index++;
392 }
393
394 // Entropy
395 ret[ret.size() - 1] = model->CalculateEntropyDensity() * V - m_THM->m_EntropyDensityInit * m_THM->m_ParametersInit.V;
396 ret[ret.size() - 1] /= m_THM->m_EntropyDensityInit * m_THM->m_ParametersInit.V;
397
398 return ret;
399 }
400
401} // namespace thermalfist
Sub-class where it is determined whether the required accuracy is achieved in the Broyden's method.
Definition Broyden.h:144
Class implementing the Broyden method to solve a system of non-linear equations.
Definition Broyden.h:132
virtual std::vector< double > Solve(const std::vector< double > &x0, BroydenSolutionCriterium *solcrit=NULL, int max_iterations=MAX_ITERS)
Definition Broyden.cpp:83
Abstract base class for an HRG model implementation.
int StableHadronIndexByGlobalId(int globalid)
Returns the PCE-based index of the stable hadron based on its global (particle list) index.
std::vector< int > RecalculateStabilityFlags(const std::vector< long long > &annihilationpdgs={2212, 2112})
ThermalModelPCEAnnihilation(ThermalModelBase *THMbase, bool FreezeLonglived=false, double LonglivedResoWidthCut=0.015)
Construct a new ThermalModelPCEAnnihilation object.
std::vector< double > StableChemsFromBroydenInput(const std::vector< double > &x)
Chemical potentials of all PCE-based hadrons from the solution to PCE equations.
virtual void SetStabilityFlags(const std::vector< int > &StabilityFlags)
Manually set the PCE stability flags for all species.
virtual void CalculatePCE(double param, PCEMode mode=AtFixedTemperature)
Solves the equations of partial chemical equilibrium at a fixed temperature or a fixed volume.
ThermalModelPCE(ThermalModelBase *THMbase, bool FreezeLonglived=false, double LonglivedResoWidthCut=0.015)
Construct a new ThermalModelPCE object.
double LonglivedResonanceWidthCut() const
void SetChemicalFreezeout(const ThermalModelParameters &params, const std::vector< double > &ChemInit=std::vector< double >(0))
Sets the chemical freeze-out conditions to be used as an initial condition for PCE calculations.
ThermalModelParameters m_ParametersCurrent
The current PCE thermal paratmeres and chemical potentials.
ThermalModelParameters m_ParametersInit
Parameters at the chemical freeze-out.
bool m_ChemicalFreezeoutSet
Whether the chemical freeze-out "initial" condition has been set.
virtual void SetStabilityFlags(const std::vector< int > &StabilityFlags)
Manually set the PCE stability flags for all species.
std::vector< int > m_StableMapTo
std::vector< double > m_ChemCurrent
bool m_IsCalculated
Whether PCE has been calculated.
std::vector< int > m_StabilityFlags
std::vector< double > m_ChemInit
static std::vector< int > ComputePCEStabilityFlags(const ThermalParticleSystem *TPS, bool SahaEquationForNuclei=true, bool FreezeLongLived=false, double WidthCut=0.015)
Computes the PCE stability flags based on the provided particle list and a number of parameters.
ThermalModelBase * ThermalModel() const
Pointer to the HRG model used in calculations.
PCEMode
Whether partial chemical equilibrium should be calculated at a fixed value of the temperature or a fi...
@ AtFixedVolume
Partial chemical equilibrium at fixed value of the volume.
@ AtFixedTemperature
Partial chemical equilibrium at fixed value of the temperature.
const std::vector< int > & StabilityFlags() const
std::vector< std::vector< double > > m_EffectiveCharges
bool m_StabilityFlagsSet
PCE configuration, list of stable species etc.
The main namespace where all classes and functions of the Thermal-FIST library reside.
Definition CosmicEoS.h:9