main443

Back to index.

// main443.cc is a part of the PYTHIA event generator.
// Copyright (C) 2025 Torbjorn Sjostrand.
// PYTHIA is licenced under the GNU GPL v2 or later, see COPYING for details.
// Please respect the MCnet Guidelines, see GUIDELINES for details.

// Keywords:
//            Thermal model

// Authors:
//            Torbjörn Sjostrand

// Compare pT spectra and particle composition in thermal vs ordinary
// hadronization models, and also ina "straw man" alternative ("mT2").
// Note that the models are out-of-the-box, without any parameter tuning.

#include "Pythia8/Pythia.h"
using namespace Pythia8;

//==========================================================================

int main() {

  // Number of events. Histograms in line printer mode or as pdf file.
  int nEvent = 2000;
  bool histAsPDF = true;

  // Histogram multiplicities and pT spectra.
  Hist nCh[3], pTpi[3], pTK[3], pTp[3];
  for (int iModel = 0; iModel < 3; ++iModel) {
    nCh[iModel].book("charged multiplicity", 100, 1., 401.);
    pTpi[iModel].book("pT for pi+-", 100, 0., 5.);
    pTK[iModel].book("pT for K+-", 100, 0., 5.);
    pTp[iModel].book("pT for p/pbar", 100, 0., 5.);
  }

  // Tabulate particle composition.
  string nameHad[18] = { "pi+-", "pi0", "K+-", "K0S,L", "eta,eta'", "rho+-",
    "rho0,omega", "K*+-0", "phi0", "p(bar)", "n(bar)", "Lambda(bar)",
    "Sigma(bar)", "Xi(bar)", "Delta(bar)", "Sigma*(bar)", "Xi*(bar)",
    "Omega(bar)"};
  int rates[3][18] = {{0}};

  // Loop over normal and thermal generation.
  for (int iModel = 0; iModel < 3; ++iModel) {

    // Generator. Common process selection. Reduce printout. pi0 stable.
    Pythia pythia;
    pythia.readString("Beams:eCM = 8000.");
    pythia.readString("SoftQCD:nonDiffractive = on");
    pythia.readString("Next:numberShowInfo = 0");
    pythia.readString("Next:numberShowProcess = 0");
    pythia.readString("Next:numberShowEvent = 0");
    pythia.readString("111:mayDecay = off");

    // Model-specific setup. Initialization.
    if (iModel == 1) pythia.readString("StringPT:thermalModel = on");
    if (iModel == 2) pythia.readString("StringPT:mT2suppression = on");

    // If Pythia fails to initialize, exit with error.
    if (!pythia.init()) return 1;

    // Event shorthand.
    Event& event = pythia.event;

    // Begin event loop. Generate event. Skip if error.
    for (int iEvent = 0; iEvent < nEvent; ++iEvent) {
      if (!pythia.next()) continue;

      // Find charged multiplicity and fill histograms.
      int nCharged = 0;
      for (int i = 0; i < event.size(); ++i)
        if (event[i].isFinal() && event[i].isCharged()) ++nCharged;
      nCh[iModel].fill( nCharged );

      // Find pT spectra of some hadron species.
      for (int i = 0; i < event.size(); ++i) if (event[i].isFinal()) {
        int idAbs = event[i].idAbs();
        double pT = event[i].pT();
        if (idAbs ==  211) pTpi[iModel].fill( pT);
        if (idAbs ==  321) pTK[iModel].fill( pT);
        if (idAbs == 2212) pTp[iModel].fill( pT);
      }

      // Tabulate hadron composition, including unstable intermediaries,
      // but excluding charm and bottom.
      for (int i = 0; i < event.size(); ++i) {
        int idAbs = event[i].idAbs();
        if      (idAbs == 211) ++rates[iModel][0];
        else if (idAbs == 111) ++rates[iModel][1];
        else if (idAbs == 321) ++rates[iModel][2];
        else if (idAbs == 130 || idAbs == 310) ++rates[iModel][3];
        else if (idAbs == 221 || idAbs == 331) ++rates[iModel][4];
        else if (idAbs == 213) ++rates[iModel][5];
        else if (idAbs == 113 || idAbs == 223) ++rates[iModel][6];
        else if (idAbs == 313 || idAbs == 323) ++rates[iModel][7];
        else if (idAbs == 333) ++rates[iModel][8];
        else if (idAbs == 2212) ++rates[iModel][9];
        else if (idAbs == 2112) ++rates[iModel][10];
        else if (idAbs == 3122) ++rates[iModel][11];
        else if (idAbs == 3112 || idAbs == 3212 || idAbs == 3222)
          ++rates[iModel][12];
        else if (idAbs == 3312 || idAbs == 3322) ++rates[iModel][13];
        else if (idAbs == 1114 || idAbs == 2114 || idAbs == 2214
          || idAbs == 2224) ++rates[iModel][14];
        else if (idAbs == 3114 || idAbs == 3214 || idAbs == 3224)
          ++rates[iModel][15];
        else if (idAbs == 3314 || idAbs == 3324) ++rates[iModel][16];
        else if (idAbs == 3334) ++rates[iModel][17];
     }

    // End of event loop. Statistics. Normalize histograms.
    }
    pythia.stat();
    nCh[iModel]  *= 0.5 / double(nEvent);
    pTpi[iModel] *= 20. / double(nEvent);
    pTK[iModel]  *= 20. / double(nEvent);
    pTp[iModel]  *= 20. / double(nEvent);

  // End of model loop.
  }

  // Print historams.
  if (!histAsPDF) {
    for (int iModel = 0; iModel < 3; ++iModel) cout << nCh[iModel];
    for (int iModel = 0; iModel < 3; ++iModel) cout << pTpi[iModel];
    for (int iModel = 0; iModel < 3; ++iModel) cout << pTK[iModel];
    for (int iModel = 0; iModel < 3; ++iModel) cout << pTp[iModel];

  // Alternatively plot histograms.
  } else {
    HistPlot hpl("plot443");
    hpl.frame("fig443", "charged multiplicity",
      "$n_{\\mathrm{charged}}$", "Probability", 8.0, 5.4);
    hpl.add( nCh[0], "-,black",  "default");
    hpl.add( nCh[1], "-,red",    "thermal");
    hpl.add( nCh[2], "-,blue",   "$m_{\\perp}^2$-suppressed");
    hpl.plot();
    hpl.frame("", "$\\pi^{\\pm}$ transverse momentum spectrum",
      "$p_{\\perp}$", "$\\mathrm{d}n/\\mathrm{d}p_{\\perp}$", 8.0, 5.4);
    hpl.add( pTpi[0], "-,black",  "default");
    hpl.add( pTpi[1], "-,red",    "thermal");
    hpl.add( pTpi[2], "-,blue",   "$m_{\\perp}^2$-suppressed");
    hpl.plot();
    hpl.frame("", "K$^{\\pm}$ transverse momentum spectrum",
      "$p_{\\perp}$", "$\\mathrm{d}n/\\mathrm{d}p_{\\perp}$", 8.0, 5.4);
    hpl.add( pTK[0], "-,black",  "default");
    hpl.add( pTK[1], "-,red",    "thermal");
    hpl.add( pTK[2], "-,blue",   "$m_{\\perp}^2$-suppressed");
    hpl.plot();
    hpl.frame("", "p,$\\overline{\\mathrm{p}}$ transverse momentum spectrum",
      "$p_{\\perp}$", "$\\mathrm{d}n/\\mathrm{d}p_{\\perp}$", 8.0, 5.4);
    hpl.add( pTp[0], "-,black",  "default");
    hpl.add( pTp[1], "-,red",    "thermal");
    hpl.add( pTp[2], "-,blue",   "$m_{\\perp}^2$-suppressed");
    hpl.plot();
  }

  // Print table.
  double norm = 1. / double(nEvent);
  cout << "\n\n Particle composition per event, including unstable"
       << "\n    Particle     default     thermal     mT2-suppressed" << endl
       << fixed << setprecision(3);
  for (int i = 0; i < 18; ++i) cout << setw(12) << nameHad[i]
       << setw(12) << norm * rates[0][i] << setw(12) << norm * rates[1][i]
       << setw(12) << norm * rates[2][i] << endl;

  // Done.
  return 0;
}