main262
Back to index.
// main262.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:
// Analysis
// Tuning
// Uncertainty bands
// Gnuplot
// Illustrate how to set up automatic uncertainty band calculations.
#include "Pythia8/Pythia.h"
using namespace Pythia8;
//==========================================================================
int main() {
// Initialize Pythia.
Pythia pythia;
pythia.readFile("main262.cmnd");
// If Pythia fails to initialize, exit with error.
if (!pythia.init()) return 1;
// Define multiple histograms, one for each variation.
int nWeights = pythia.info.nWeights();
vector<double> sumOfWeights;
vector<Hist> pTtop, nCh;
vector<string> names;
vector<string> weightStrings = pythia.settings.wvec("UncertaintyBands:List");
// Loop through weights to initialize the histograms.
for (int iWeight=0; iWeight < nWeights; ++iWeight) {
names.push_back( (iWeight==0)
? "baseline" : pythia.info.getGroupName(iWeight));
pTtop.push_back ( Hist("top transverse momentum", 100, 0., 200.));
nCh.push_back ( Hist("charged particle multiplicity", 100, -1., 399.));
sumOfWeights.push_back(0.);
}
// Event generation loop.
int nEvent = pythia.mode("Main:numberOfEvents");
for (int iEvent=0; iEvent < nEvent; ++iEvent) {
// Generate next event. Break out of event loop if at end of an LHE file.
if ( !pythia.next() ) {
if ( pythia.info.atEndOfFile() ) break;
else continue;
}
// Find last top in event record and count number of charged particles.
int iTop = 0;
int nChg = 0;
for (int i=0; i < pythia.event.size(); ++i) {
if (pythia.event[i].id() == 6) iTop = i;
if (pythia.event[i].isFinal() && pythia.event[i].isCharged()) ++nChg;
}
// Get top pT.
double pT = pythia.event[iTop].pT();
// Fill histograms with variation weights.
for (int iWeight = 0; iWeight < nWeights; ++iWeight) {
// Get weight
double w = pythia.info.getGroupWeight(iWeight);
// Add the weight of the current event to the wsum of weights.
sumOfWeights[iWeight] += w;
// Fill histograms.
pTtop[iWeight].fill(pT, w);
nCh[iWeight].fill(nChg, w);
}
}
// Print cross section information.
pythia.stat();
// Normalize histograms and print data tables. Also, construct a simple
// gnuplot command to plot the histograms.
stringstream gnuplotCommand;
gnuplotCommand << "gnuplot -e \"plot";
for (int iWeight=0; iWeight < nWeights; ++iWeight) {
if (sumOfWeights[iWeight] != 0) {
cout << "Normalize histogram for weight " << names[iWeight] << " to "
<< scientific << setprecision(8) << sumOfWeights[iWeight] << endl;
pTtop[iWeight] *= pythia.info.sigmaGen() / sumOfWeights[iWeight]
/ 2.; // ... and divide by bin width.
nCh[iWeight] *= pythia.info.sigmaGen() / sumOfWeights[iWeight]
/ 4.; // ... and divide by bin width.
}
// Print data tables.
ofstream write;
write.open( ("pTtop-" + names[iWeight] + ".dat").c_str());
pTtop[iWeight].table(write);
write.close();
write.open( ("nCh-" + names[iWeight] + ".dat").c_str());
nCh[iWeight].table(write);
write.close();
// Construct gnuplot command.
gnuplotCommand << " \'pTtop-" << names[iWeight]
<< ".dat\' using 1:2 w steps title \'" << names[iWeight] << "\',";
}
gnuplotCommand.seekp(-1, std::ios_base::end);
gnuplotCommand << "; pause -1\"";
// Print suggested gnuplot command.
cout << "You can plot the variation by e.g. using the command:" << endl;
cout << gnuplotCommand.str() << endl << endl;
// Done.
return 0;
}