main263
Back to index.
// main263.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.
// Authors:
// Christian Bierlich
// Stephen Mrenna
// Philip Ilten
// Keywords:
// Hadronization
// Reweighting
// Tuning
// Parallelism
// Matplotlib
// Demonstrates how to reweight an event for different kinematic or flavor
// hadronization parameters using in-situ reweighting.
// Pythia includes.
#include "Pythia8/Pythia.h"
#include "Pythia8/PythiaParallel.h"
using namespace Pythia8;
//==========================================================================
int main() {
// Choose to reweight kinematic (0) or flavor (1) hadronization
// parameters. Both can be reweighted simultaneously, but the
// separation is kept here for illustrative purposes.
int type = 0;
// Number of events to generate per run.
int nEvent = 1e6;
// Define the new set of kinematic parameters that we wish to reweight to.
double aLund = 0.6; // StringZ:aLund, default 0.68
double bLund = 0.9; // StringZ:bLund, default 0.98
double rFactC = 1.3; // StringZ:rFactC, default 1.32
double rFactB = 0.9; // StringZ:rFactB, default 0.855
double ptSigma = 0.3; // StringPT:sigma, default 0.335
// Define the new set of flavor parameters that we wish to reweight to.
double rho = 0.2; // StringFlav:ProbStoUD, default 0.217
double xi = 0.1; // StringFlav:ProbQQtoQ, default 0.081
double x = 0.9; // StringFlav:ProbSQtoQQ, default 0.915
double y = 0.04; // StringFlav:ProbQQ1toQQ0, default 0.0275
// Create and configure Pythia.
PythiaParallel pythia;
pythia.readString("Beams:idA = 11");
pythia.readString("Beams:idB = -11");
pythia.readString("Beams:eCM = 91.189");
pythia.readString("PDF:lepton = off");
pythia.readString("WeakSingleBoson:ffbar2gmZ = on");
pythia.readString("23:onMode = off");
pythia.readString("23:onIfAny = 1 2 3 4 5");
// Configure the in-situ kinematic or flavor reweighting.
if (type == 0) {
pythia.readString("VariationFrag:List = {kineVar frag:aLund="
+ to_string(aLund) + " frag:bLund=" + to_string(bLund) + " frag:rFactC="
+ to_string(rFactC) + " frag:rFactB=" + to_string(rFactB)
+ " frag:ptSigma=" + to_string(ptSigma) + "}");
} else if (type == 1) {
pythia.readString("VariationFrag:List = {flavVar frag:rho="
+ to_string(rho) + " frag:xi=" + to_string(xi) + " frag:x="
+ to_string(x) + " frag:y=" + to_string(y) + "}");
pythia.readString("StringFlav:popcornRate = 0");
pythia.readString("HadronLevel:Decay = off");
}
// If Pythia fails to initialize, exit with error.
if (!pythia.init()) return 1;
// Define the plot title.
string title = "default: (";
if (type == 0)
title +=
toString(pythia.settings.parm("StringZ:aLund")) + ", " +
toString(pythia.settings.parm("StringZ:bLund")) + ", " +
toString(pythia.settings.parm("StringZ:rFactC")) + ", " +
toString(pythia.settings.parm("StringZ:rFactB")) + ", " +
toString(pythia.settings.parm("StringPT:sigma")) + "), " +
"variation: (" + toString(aLund) + ", " + toString(bLund) + ", " +
toString(rFactC) + ", " + toString(rFactB) + ", " +
toString(ptSigma) + ") ";
else if (type == 1)
title +=
toString(pythia.settings.parm("StringFlav:ProbStoUD")) + ", " +
toString(pythia.settings.parm("StringFlav:ProbQQtoQ")) + ", " +
toString(pythia.settings.parm("StringFlav:ProbSQtoQQ")) + ", " +
toString(pythia.settings.parm("StringFlav:ProbQQ1toQQ0")) + "), " +
"variation: (" + toString(rho) + ", " + toString(xi) + ", " +
toString(x) + ", " + toString(y) + ") ";
// Identified final state hadrons to include in the flavor histograms.
vector<int> hadrons = {
211, 221, 331, 213, 223, 321, 311, 333, 2212, 2112, 2214, 2224, 3222,
3212, 3122, 3322, 3334};
// Define multiplicity histograms. For kinematics, we look at
// charged multiplicity while for flavor we look at multiplicity per
// species.
// default: the default parameters in Pythia
// insitu: in-situ reweighted
// rerun: a run with the varied parameters
vector<string> names = {"default", "insitu", "rerun"};
map<string, Hist> hists;
for (string &name : names) {
if (type == 0)
hists[name] = Hist(name, 25, 2, 51);
else if (type == 1)
hists[name] = Hist(name, hadrons.size(), 0, hadrons.size());
}
// Track the weights.
map<string, double> wgts, sumWgts, sumWgt2s;
for (string &name : names)
wgts[name] = sumWgts[name] = sumWgt2s[name] = 0;
names.pop_back();
// Run events.
// This defines a lambda function that acts as a callback.
// This function is called for each event generated.
// The argument is a pointer to the instance that generated the event.
// This is neccesary to use PythiaParallel (multi-core).
pythia.run( nEvent, [&](Pythia* pythiaPtr) {
// For the default parameters, the weight is just 1.
wgts["default"] = 1;
// The weight given by the in-situ reweighting.
wgts["insitu"] = pythiaPtr->info.getGroupWeight(0);
// Keep track of the weights.
for (string &name : names) {
sumWgts[name] += wgts[name];
sumWgt2s[name] += pow2(wgts[name]);
}
// Fill the histograms.
int mult = 0;
for (const Particle &prt : pythiaPtr->event) {
if (!prt.isFinal()) continue;
if (type == 0) {
if (prt.isCharged()) ++mult;
} else if (type == 1) {
int pid = prt.idAbs();
int idx = -1;
for (int iHad = 0; iHad < (int)hadrons.size(); ++iHad)
if (pid == hadrons[iHad]) {idx = iHad; break;}
if (idx < 0) continue;
for (string &name : names) hists[name].fill(idx, wgts[name]);
}
}
if (type == 0)
for (string &name : names) hists[name].fill(mult, wgts[name]);
});
pythia.stat();
// Rerun Pythia with the varied parameters.
if (type == 0) {
pythia.settings.parm("StringZ:aLund", aLund);
pythia.settings.parm("StringZ:bLund", bLund);
pythia.settings.parm("StringZ:rFactC", rFactC);
pythia.settings.parm("StringZ:rFactB", rFactB);
pythia.settings.parm("StringPT:sigma", ptSigma);
} else if (type == 1) {
pythia.settings.parm("StringFlav:ProbStoUD", rho);
pythia.settings.parm("StringFlav:ProbQQtoQ", xi);
pythia.settings.parm("StringFlav:ProbSQtoQQ", x);
pythia.settings.parm("StringFlav:ProbQQ1toQQ0", y);
}
pythia.settings.wvec("VariationFrag:List", {});
// If Pythia fails to initialize, exit with error.
if (!pythia.init()) return 1;
// Run events.
// This defines a lambda function that acts as a callback.
// This function is called for each event generated.
// The argument is a pointer to the instance that generated the event.
// This is neccesary to use PythiaParallel (multi-core).
pythia.run( nEvent, [&](Pythia* pythiaPtr) {
sumWgts["rerun"] += 1;
sumWgt2s["rerun"] += 1;
int mult = 0;
for (const Particle &prt : pythiaPtr->event) {
if (!prt.isFinal()) continue;
if (type == 0) {
if (prt.isCharged()) ++mult;
} else if (type == 1) {
int pid = prt.idAbs();
int idx = -1;
for (int iHad = 0; iHad < (int)hadrons.size(); ++iHad)
if (pid == hadrons[iHad]) {idx = iHad; break;}
if (idx >= 0) hists["rerun"].fill(idx, 1.);
}
}
if (type == 0) hists["rerun"].fill(mult, 1);
});
pythia.stat();
// Normalize the histograms.
for (auto &hist : hists) hist.second /= sumWgts[hist.first];
// Print the histogram ratios.
string xlabel;
if (type == 0) {
xlabel = "multiplicity";
} else if (type == 1) {
for (int iHad = 0; iHad < (int)hadrons.size(); ++iHad) {
string name = pythia.particleData.name(hadrons[iHad]);
cout << left << setw(3) << iHad << ": " << name << "\n";
xlabel += " " + name + "(" + to_string(iHad) + ")";
}
}
for (auto &hist : hists)
cout << "\n" << hist.first << hist.second/hists["default"];
// Print the reweighting stats.
// The 1 - mu should be statistically consistent with zero if the
// reweighting has proper coverage.
// The n_eff gives the statistical power of the reweighted sample.
for (string &name : names) {
double w(sumWgts[name]), w2(sumWgt2s[name]), n(sumWgts["default"]);
cout << name << "\n"
<< "\t1 - mu = " << scientific << setprecision(3) << abs(1. - w/n)
<< " +- "<< abs(1. - sqrt((w2/n - pow2(w/n))*n/(n - 1)))/sqrt(n)
<< "\n\tn_eff = " << scientific << setprecision(3) << w*w/(n*w2)
<< "\n";
}
// Create the Python plot and return.
HistPlot hpl("plot263");
hpl.frame("fig263", title, xlabel, "n(variation)/n(default)");
for (string &name : names)
hpl.add(hists[name]/hists["default"], "e", name);
hpl.add(hists["rerun"]/hists["default"], "e", "rerun");
hpl.plot();
return 0;
}