main203
Back to index.
// main203.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:
//            Juan Rojo
// Keywords:
//            Parton distribution
//            LHAPDF
// This program compares the internal and LHAPDF implementations of
// some NNPDF 2.3 QCD+QED sets, for results and for timing.
// Warning: this example is constructed to work for LHAPDF5,
// but current choice is LHAPDF6, which gives some differences.
#include "Pythia8/Pythia.h"
#include "Pythia8/Plugins.h"
using namespace Pythia8;
//==========================================================================
int main() {
  cout<<"\n NNPDF3.1 QED LO phenomenology \n "<<endl;
  cout<<"\n Check access to NNPDF3.1 NNLO QED sets \n "<<endl;
  // Generator.
  Pythia pythia;
  // Access the PDFs.
  int idBeamIn = 2212;
  string pdfPath = pythia.settings.word("xmlPath") + "../pdfdata";
  Logger logger;
  // Grid of studied points.
  string xpdf[] = {"x*g","x*d","x*u","x*s"};
  double xlha[] = {1e-5, 1e-1};
  double Q2[] = { 2.0, 10000.0 };
  string setName;
  string setName_lha;
  // For timing checks.
  int const nq = 500;
  int const nx = 500;
  int const iqMax = sizeof( xlha )/sizeof( xlha[0] );
  // Loop over two internal PDF sets in Pythia8
  // and compare with their LHAPDF correspondents.
  for (int iFitIn = 3; iFitIn < 5; iFitIn++) {
    // Constructor for LHAPDF.
    if (iFitIn == 3) setName = "NNPDF31_nlo_as_0118_luxqed";
    if (iFitIn == 4) setName = "NNPDF31_nnlo_as_0118_luxqed";
    PDFPtr pdfs_nnpdf_lha =
      make_plugin<PDF>("libpythia8lhapdf6.so", "LHAPDF6");
    if (pdfs_nnpdf_lha == nullptr) return -1;
    pdfs_nnpdf_lha->init(idBeamIn, setName, 0, &logger);
    cout << "\n PDF set = " << setName << " \n" << endl;
    // Constructor for internal PDFs.
    LHAGrid1 pdfs_nnpdf(
      idBeamIn, setName + "_0000.dat", pdfPath, &logger);
    // Check quarks and gluons.
    cout << setprecision(6);
    for (int f = 0; f < 4; f++) {
      for (int iq = 0; iq < iqMax; iq++) {
        cout << "  " << xpdf[f] << ", Q2 = " << Q2[iq] << endl;
        cout << "   x \t     Pythia8\t   LHAPDF\t   diff(%) " << endl;
        for (int ix = 0; ix < 2; ix++) {
          double a = pdfs_nnpdf.xf( f, xlha[ix], Q2[iq]);
          double b = pdfs_nnpdf_lha->xf( f, xlha[ix], Q2[iq]);
          double diff = b != 0 ? 1e2 * abs((a-b)/b) :
            std::numeric_limits<double>::infinity();
          cout << scientific << xlha[ix] << " " << a << " " << b
               << " " << diff << endl;
        }
      }
    }
    // Check photon.
    cout << "\n Now checking the photon PDF \n" << endl;
    for (int iq = 0; iq < iqMax; iq++) {
      cout << "  " << "x*gamma" << ", Q2 = " << Q2[iq] << endl;
      cout << "   x \t     Pythia8\t   LHAPDF\t   diff(%) " << endl;
      for (int ix = 0; ix < 2; ix++) {
        double a = pdfs_nnpdf.xf( 22, xlha[ix], Q2[iq]);
        double b = pdfs_nnpdf_lha->xf( 22, xlha[ix], Q2[iq]);
        double diff = b != 0 ? 1e2 * abs((a-b)/b) :
          std::numeric_limits<double>::infinity();
        cout << scientific << xlha[ix] << " " << a << " " << b
             << " " << diff << endl;
      }
    }
    // Now check the timings for evolution.
    cout << "\n Checking timings " << endl;
    // Internal timing.
    clock_t tBegin = clock();
    for (int f = -4; f < 4; f++) {
      for (int iq = 0; iq < nq; iq++) {
        double qq2 = 2.0 * pow( 1e6 / 2.0, double(iq)/nq);
        for (int ix = 0; ix < nx; ix++) {
          double xx = 1e-6 * pow( 9e-1 / 1e-6, double(ix)/nx);
          pdfs_nnpdf.xf(f,xx,qq2);
        }
      }
    }
    clock_t tEnd = clock();
    double tUsed = double(tEnd - tBegin) / double(CLOCKS_PER_SEC);
    cout << " NNPDF internal timing = " << tUsed << endl;
    // External timing.
    tBegin = clock();
    for (int f = -4; f < 4; f++) {
      for (int iq = 0; iq < nq; iq++) {
        double qq2 = 2.0 * pow(1e6 / 2.0, double(iq)/nq);
        for (int ix = 0; ix < nx; ix++) {
          double xx = 1e-6 * pow( 9e-1 / 1e-6, double(ix)/nx);
          pdfs_nnpdf_lha->xf(f,xx,qq2);
        }
      }
    }
    tEnd = clock();
    tUsed = double(tEnd - tBegin) / double(CLOCKS_PER_SEC);
    cout << " NNPDF LHAPDF   timing = " << tUsed << endl;
  } // End loop over NNPDF internal sets
  // Done.
  cout << "\n Compared LHAPDF and internal Pythia8 results.\n" << endl;
  return 0;
}