file examples/DarkBit_standalone_WIMP.cpp

[No description available] More…

Namespaces

Name
Gambit
TODO: see if we can use this one:
Gambit::DarkBit

Functions

Name
QUICK_FUNCTION(DarkBit , TH_ProcessCatalog , OLD_CAPABILITY , TH_ProcessCatalog_WIMP , TH_ProcessCatalog , () , (mwimp, double) )
std::string voiddump_array_to_file(const std::string & filename, const boost::multi_array< double, 2 > & a, const std::vector< double > & x, const std::vector< double > & y)
voiddumpSpectrum(std::vector< std::string > filenames, double mWIMP, double sv, std::vector< double > brList, double mPhi =-1)
intmain(int argc, char * argv[])

Attributes

Name
WIMP_properties
OLD_CAPABILITY
WIMP_properties_WIMP
WIMPprops
DarkMatter_ID
DarkMatterConj_ID

Defines

Name
addParticle(Name, Mass, spinX2)

Detailed Description

Author:

  • Christoph Weniger
  • Jonathan Cornell
  • Sebastian Wild
  • Torsten Bringmann
  • Sowmiya Balan
  • Patrick Stoecker

Date:

  • 2016 Feb
  • 2016 July
  • 2016 Aug
  • 2022
  • 2023
  • 2023

Example of GAMBIT DarkBit standalone main program.


Authors (add name and date if you modify):


Functions Documentation

function QUICK_FUNCTION

QUICK_FUNCTION(
    DarkBit ,
    TH_ProcessCatalog ,
    OLD_CAPABILITY ,
    TH_ProcessCatalog_WIMP ,
    TH_ProcessCatalog ,
    () ,
    (mwimp, double) 
)

function dump_array_to_file

std::string void dump_array_to_file(
    const std::string & filename,
    const boost::multi_array< double, 2 > & a,
    const std::vector< double > & x,
    const std::vector< double > & y
)

function dumpSpectrum

void dumpSpectrum(
    std::vector< std::string > filenames,
    double mWIMP,
    double sv,
    std::vector< double > brList,
    double mPhi =-1
)

function main

int main(
    int argc,
    char * argv[]
)

Attributes Documentation

variable WIMP_properties

WIMP_properties;

variable OLD_CAPABILITY

OLD_CAPABILITY;

variable WIMP_properties_WIMP

WIMP_properties_WIMP;

variable WIMPprops

WIMPprops;

variable DarkMatter_ID

DarkMatter_ID;

variable DarkMatterConj_ID

DarkMatterConj_ID;

Macros Documentation

define addParticle

#define addParticle(
    Name,
    Mass,
    spinX2
)
        catalog.particleProperties.insert(std::pair<string, TH_ParticleProperty> \
        (Name , TH_ParticleProperty(Mass, spinX2)));

Source code

//   GAMBIT: Global and Modular BSM Inference Tool
//   *********************************************
///  \file
///
///  Example of GAMBIT DarkBit standalone
///  main program.
///
///  *********************************************
///
///  Authors (add name and date if you modify):
///
///  \author Christoph Weniger
///  \date 2016 Feb
///  \author Jonathan Cornell
///  \date 2016 July
///  \author Sebastian Wild
///  \date 2016 Aug
///  \author Torsten Bringmann
///  \date 2022
///  \author Sowmiya Balan
///  \date 2023
///  \author Patrick Stoecker
///  \date 2023
///
///  *********************************************

#include <iostream>
#include <fstream>

#include "gambit/Elements/standalone_module.hpp"
#include "gambit/DarkBit/DarkBit_rollcall.hpp"
#include "gambit/Elements/spectrum_factories.hpp"
#include "gambit/Utils/util_functions.hpp"

#include <boost/multi_array.hpp>

//#define DARKBIT_STANDALONE_WIMP_DEBUG

using namespace DarkBit::Functown;     // Functors wrapping the module's actual module functions
using namespace BackendIniBit::Functown;    // Functors wrapping the backend initialisation functions

QUICK_FUNCTION(DarkBit, TH_ProcessCatalog, OLD_CAPABILITY, TH_ProcessCatalog_WIMP, TH_ProcessCatalog, (), (mwimp, double))
QUICK_FUNCTION(DarkBit, DarkMatter_ID, OLD_CAPABILITY, DarkMatter_ID_WIMP, std::string, ())
QUICK_FUNCTION(DarkBit, DarkMatterConj_ID, OLD_CAPABILITY, DarkMatterConj_ID_WIMP, std::string, ())
QUICK_FUNCTION(DarkBit, DD_couplings, OLD_CAPABILITY, DD_couplings_WIMP, DM_nucleon_couplings, ())
QUICK_FUNCTION(DarkBit, WIMP_properties, OLD_CAPABILITY, WIMP_properties_WIMP, WIMPprops, (), (DarkMatter_ID, std::string), (DarkMatterConj_ID, std::string))


void dump_array_to_file(const std::string & filename, const
    boost::multi_array<double, 2> & a, const std::vector<double> & x, const
    std::vector<double> & y)
{
  std::fstream file;
  file.open(filename, std::ios_base::out);
  file << "0.0 ";
  for (size_t i = 0; i < x.size(); i++)
    file << x[i] << " ";
  file << std::endl;
  for (size_t j = 0; j < y.size(); j++)
  {
    file << y[j] << " ";
    for (size_t i = 0; i < x.size(); i++)
    {
      file << a[i][j] << " ";
    }
    file << std::endl;
  }
  file.close();
}

void dumpSpectrum(std::vector<std::string> filenames, double mWIMP, double sv, std::vector<double> brList, double mPhi = -1)
{
  DarkMatter_ID_WIMP.reset_and_calculate();
  DarkMatterConj_ID_WIMP.reset_and_calculate();
  WIMP_properties_WIMP.setOption<double>("mWIMP", mWIMP);
  WIMP_properties_WIMP.reset_and_calculate();
  mwimp_generic.reset_and_calculate();
  TH_ProcessCatalog_WIMP.setOption<std::vector<double>>("brList", brList);
  TH_ProcessCatalog_WIMP.setOption<double>("sv", sv);
  if (mPhi != -1)
    TH_ProcessCatalog_WIMP.setOption<double>("mPhi", mPhi);
  TH_ProcessCatalog_WIMP.reset_and_calculate();
  DM_process_from_ProcessCatalog.reset_and_calculate();
  RD_fraction_one.reset_and_calculate();
  GA_SimYieldTable_MicrOmegas.reset_and_calculate();
  GA_SimYieldTable_DarkSUSY.reset_and_calculate();
  positron_SimYieldTable_DarkSUSY.reset_and_calculate();
  electron_SimYieldTable_from_positron_SimYieldTable.reset_and_calculate();
  antiproton_SimYieldTable_DarkSUSY.reset_and_calculate();
  antideuteron_SimYieldTable_DarkSUSY.reset_and_calculate();
  Combine_SimYields.reset_and_calculate();
  cascadeMC_FinalStates.reset_and_calculate();
  cascadeMC_InitialStates.reset_and_calculate();
  cascadeMC_DecayTable.reset_and_calculate();
  cascadeMC_LoopManager.reset_and_calculate();
  cascadeMC_gammaSpectra.reset_and_calculate();
  GA_AnnYield_General.reset_and_calculate();
  dump_gammaSpectrum.setOption<std::string>("filename", filenames[0]);
  dump_gammaSpectrum.reset_and_calculate();
  if (filenames.size() == 1) return;
  cascadeMC_positronSpectra.reset_and_calculate();
  positron_AnnYield_General.reset_and_calculate();
  dump_positronSpectrum.setOption<std::string>("filename", filenames[1]);
  dump_positronSpectrum.reset_and_calculate();
  cascadeMC_antiprotonSpectra.reset_and_calculate();
  antiproton_AnnYield_General.reset_and_calculate();
  dump_antiprotonSpectrum.setOption<std::string>("filename", filenames[2]);
  dump_antiprotonSpectrum.reset_and_calculate();
  cascadeMC_antideuteronSpectra.reset_and_calculate();
  antideuteron_AnnYield_General.reset_and_calculate();
  dump_antideuteronSpectrum.setOption<std::string>("filename", filenames[3]);
  dump_antideuteronSpectrum.reset_and_calculate();
}

// ---- Set up basic internal structures for direct & indirect detection ----

namespace Gambit
{
  namespace DarkBit
  {

    void TH_ProcessCatalog_WIMP(TH_ProcessCatalog& result)
    {
      using namespace Pipes::TH_ProcessCatalog_WIMP;
      using std::vector;
      using std::string;

      // Initialize empty catalog and main annihilation process
      TH_ProcessCatalog catalog;
      TH_Process process_ann("WIMP", "WIMP");
      TH_Process process_dec("phi");
      TH_Process process_dec1("phi1");
      TH_Process process_dec2("phi2");

      ///////////////////////////////////////
      // Import particle masses and couplings
      ///////////////////////////////////////

      #define addParticle(Name, Mass, spinX2)                                        \
        catalog.particleProperties.insert(std::pair<string, TH_ParticleProperty> \
        (Name , TH_ParticleProperty(Mass, spinX2)));

        /// Option mWIMP<double>: WIMP mass in GeV (required)
        double mWIMP = *Dep::mwimp;
        /// Option sv<double>: Cross-section in cm3/s (required)
        double sv = runOptions->getValue<double>("sv");
        double b = 0;  // defined as sv(v) = sv(v=0) + b*(sv=0)*v**2
        /// Option brList<std::vector<double>>: List of branching ratios (required)
        auto brList = runOptions->getValue<std::vector<double>>("brList");
        /// Option mWIMP<double>: WIMP mass in GeV (required)
        double mPhi = runOptions->getValueOrDef<double>(59.0, "mPhi");

        addParticle("gamma", 0.0, 2)
        addParticle("Z0", 91.2,  2)
        addParticle("W+", 80.39, 2)
        addParticle("W-", 80.39, 2)
        addParticle("e+_3", 1.8,  1)
        addParticle("e-_3", 1.8,  1)
        addParticle("e+_1", 0.00051, 1)
        addParticle("e-_1", 0.00051, 1)
        addParticle("b", 4.9,  1)
        addParticle("bbar", 4.9,  1)
        addParticle("d_3", 4.9,  1)
        addParticle("dbar_3", 4.9,  1)
        addParticle("p", m_proton, 1)
        addParticle("pbar", m_proton, 1)
        addParticle("n", m_neutron, 1)
        addParticle("nbar", m_neutron, 1)
        addParticle("D", m_deuteron, 2)
        addParticle("Dbar", m_deuteron, 2)

        addParticle("WIMP", mWIMP,  0)
        addParticle("phi",  mPhi,  0)
        addParticle("phi1", 100.,  0)
        addParticle("phi2", 100.,  0)
      #undef addParticle

      TH_Channel dec_channel(daFunk::vec<string>("gamma", "gamma"), daFunk::cnst(1.));
      process_dec.channelList.push_back(dec_channel);

      TH_Channel dec_channel1(daFunk::vec<string>("e+_3", "e-_3"), daFunk::cnst(1.));
      process_dec1.channelList.push_back(dec_channel1);

      TH_Channel dec_channel2(daFunk::vec<string>("d_3", "dbar_3"), daFunk::cnst(1.));
      process_dec2.channelList.push_back(dec_channel2);

      process_ann.resonances_thresholds.threshold_energy.push_back(2*mWIMP);
      auto p1 = daFunk::vec<string>("d_3", "gamma", "gamma", "e-_3", "W-", "e-_1", "phi");
      auto p2 = daFunk::vec<string>("dbar_3", "Z0", "gamma", "e+_3", "W+", "e+_1", "phi2");
      {
        for ( unsigned int i = 0; i < brList.size()-1; i++ )
        {
          double mtot_final =
            catalog.getParticleProperty(p1[i]).mass +
            catalog.getParticleProperty(p2[i]).mass;
          if ( mWIMP*2 > mtot_final && brList[i]!= 0.)
          {
            // std::cout << p1[i] << " " << p2[i] << " " << brList[i] << std::endl;
            daFunk::Funk kinematicFunction = (daFunk::one("v")+pow(daFunk::var("v"), 2)*b)*sv*brList[i];
            TH_Channel new_channel(
                daFunk::vec<string>(p1[i], p2[i]), kinematicFunction
                );
            process_ann.channelList.push_back(new_channel);
          }
          else
          {
            process_ann.resonances_thresholds.threshold_energy.
              push_back(mtot_final);
          }
        }
      }

      if ( brList[7] > 0. )
      {
        auto E = daFunk::var("E");
        // Note:: The below is an arbitrary form of the differential section for demonstration purposes
        daFunk::Funk kinematicFunction = daFunk::one("v", "E1")/(pow(E-50, 4)+1)*sv*brList[7];
        // Note: In the three body final states, the gamma yield from AnnYield currently is just the contribution
        // from the first particle in the list (here the photon):
        TH_Channel new_channel(daFunk::vec<string>("gamma", "e+_1", "e-_1"), kinematicFunction);
        process_ann.channelList.push_back(new_channel);
      }

      catalog.processList.push_back(process_ann);
      catalog.processList.push_back(process_dec);
      catalog.processList.push_back(process_dec1);
      catalog.processList.push_back(process_dec2);

      catalog.validate();

      result = catalog;
    } // function TH_ProcessCatalog_WIMP

    // Identifier for DM particle
    void DarkMatter_ID_WIMP(std::string& result)
    {
      result = "WIMP";
    }

    // Identifier for conjugate DM particle
    void DarkMatterConj_ID_WIMP(std::string& result)
    {
      result = "WIMP";
    }

    // WIMP properites
    void WIMP_properties_WIMP(WIMPprops &props)
    {
      using namespace Pipes::WIMP_properties_WIMP;
      props.name = *Dep::DarkMatter_ID;
      props.spinx2 = Models::ParticleDB().get_spinx2(props.name);
      props.sc = not Models::ParticleDB().has_antiparticle(props.name);
      props.conjugate = props.sc ? props.name : *Dep::DarkMatterConj_ID;
      props.mass = runOptions->getValue<double>("mWIMP");
    }

    void DD_couplings_WIMP(DM_nucleon_couplings& result)
    {
      using namespace Pipes::DD_couplings_WIMP;
      /// Option gps<double>: gps (default 0)
      result.gps = runOptions->getValueOrDef<double>(0., "gps");
      /// Option gns<double>: gns (default 0)
      result.gns = runOptions->getValueOrDef<double>(0., "gns");
      /// Option gpa<double>: gpa (default 0)
      result.gpa = runOptions->getValueOrDef<double>(0., "gpa");
      /// Option gna<double>: gna (default 0)
      result.gna = runOptions->getValueOrDef<double>(0., "gna");
    }
  }
}

int main(int argc, char* argv[])
{
    std::cout << std::endl;
    std::cout << "Welcome to the DarkBit Generic WIMP standalone program!" << std::endl;
    std::cout << std::endl;
    std::cout << "**************************************************************************************" << std::endl;
    std::cout << "This standalone example demonstrates how to calculate a range of observables and " << std::endl;
    std::cout << "likelihoods for a generic WIMP model defined by the WIMP mass and an annihilation (or " << std::endl;
    std::cout << "scattering) cross section. The model also contains three scalar particles which decay:" << std::endl;
    std::cout << "phi -> gamma gamma    phi_1 -> tau+ tau-  phi_2 -> b bbar" << std::endl;
    std::cout << std::endl;
    std::cout << "Usage: DarkBit_standalone_WIMP mode" << std::endl;
    std::cout << std::endl;
    std::cout << "Mode Options: " << std::endl;
    std::cout << "  0: Outputs spectrum of gamma rays from WIMP annihilation to b bbar (dPhi_dE0.dat)" << std::endl;
    std::cout << "  1: Outputs spectrum of gamma rays from WIMP annihilation to gamma Z_0 (dPhi_dE1.dat)" << std::endl;
    std::cout << "  2: Outputs spectrum of gamma rays from WIMP annihilation to gamma gamma (dPhi_dE2.dat)" << std::endl;
    std::cout << "  3: Outputs spectrum of gamma rays from WIMP annihilation to tau+ tau- (dPhi_dE3.dat)" << std::endl;
    std::cout << "  4: Outputs spectrum of gamma rays from WIMP annihilation to W+ W- (dPhi_dE4.dat)" << std::endl;
    std::cout << "  5: Outputs spectrum of gamma rays from WIMP annihilation to gamma e+ e- " << std::endl;
    std::cout << "      (dPhi_dE5.dat)" << std::endl;
    std::cout << "  6: Outputs tables of gamma-ray likelihoods and the relic density" << std::endl;
    std::cout << "      in <sigma v> / m_WIMP parameter space." << std::endl;
    std::cout << "  7: Outputs tables of direct detection likelihoods in sigma / m_WIMP parameter" << std::endl;
    std::cout << "      space." << std::endl;
    std::cout << "  8: Outputs antiproton likelihoods calculated using flux from DarkRayNet v2 and AMS02 data by the python backend 'pbarlike' " << std::endl;
    std::cout << "  >=10: Outputs spectra of gamma rays, positrons, anti-protons and anti-deuterons from" << std::endl;
    std::cout << "        WIMP annihilation to phi phi_2. The mode value is m_phi while m_phi_2=100 GeV." << std::endl;
    std::cout << "        The output filenames are dPhi_dE_FCMC_(spectrum)_(mode).dat." << std::endl;
    std::cout << " N.B. Here dPhi/dE = sigma v / m_chi^2 * dN/dE" << std::endl;
    std::cout << "**************************************************************************************" << std::endl;
    std::cout << std::endl;

  try
  {
    if (argc==1)
    {
      std::cout << "Please select test mode>=0" << std::endl;
      exit(1);
    }
    int mode = std::stoi((std::string)argv[1]);
    std::cout << "Starting with mode " << mode << std::endl;


    // ---- Initialise logging and exceptions ----

    initialise_standalone_logs("runs/DarkBit_standalone_WIMP/logs/");
    logger()<<"Running DarkBit standalone example"<<LogTags::info<<EOM;
    model_warning().set_fatal(true);
    
    // Initialise settings for printer (required)
    YAML::Node printerNode = get_standalone_printer("cout", "runs/DarkBit_standalone_WIMP/logs/","");
    Printers::PrinterManager printerManager(printerNode, false);
    set_global_printer_manager(&printerManager);


    // ---- Check that required backends are present ----

    if (not Backends::backendInfo().works["DarkSUSY_generic_wimp6.4.0"]) backend_error().raise(LOCAL_INFO, "DarkSUSY_generic_wimp_6.4.0 is missing!");
    if (not Backends::backendInfo().works["gamLike1.0.1"]) backend_error().raise(LOCAL_INFO, "gamLike 1.0.1 is missing!");
    if (not Backends::backendInfo().works["DDCalc2.3.0"]) backend_error().raise(LOCAL_INFO, "DDCalc 2.3.0 is missing!");
    if (not Backends::backendInfo().works["MicrOmegas_MSSM3.6.9.2"]) backend_error().raise(LOCAL_INFO, "MicrOmegas 3.6.9.2 for MSSM is missing!");
    if (not Backends::backendInfo().works["pbarlike1.0"]) backend_error().raise(LOCAL_INFO, "pbarlike1.0 is missing!");

    // ---- Initialize models ----
    // Initialize halo model
    ModelParameters* Halo_primary_parameters = Models::Halo_Einasto::Functown::primary_parameters.getcontentsPtr();
    Halo_primary_parameters->setValue("vrot", 235.); // Local properties
    Halo_primary_parameters->setValue("v0", 235.);
    Halo_primary_parameters->setValue("vesc", 550.);
    Halo_primary_parameters->setValue("rho0", 0.4);
    Halo_primary_parameters->setValue("r_sun", 8.5);

    Halo_primary_parameters->setValue("rs", 20.);  // Global properties
    Halo_primary_parameters->setValue("rhos", 0.08);
    Halo_primary_parameters->setValue("alpha", 0.17);


    // --- Resolve halo dependencies ---
    ExtractLocalMaxwellianHalo.notifyOfModel("Halo_Einasto");
    ExtractLocalMaxwellianHalo.resolveDependency(&Models::Halo_Einasto::Functown::primary_parameters);
    ExtractLocalMaxwellianHalo.reset_and_calculate();

    GalacticHalo_Einasto.notifyOfModel("Halo_Einasto");
    GalacticHalo_Einasto.resolveDependency(&Models::Halo_Einasto::Functown::primary_parameters);
    GalacticHalo_Einasto.reset_and_calculate();

    // ---- Initialize backends ----

    // Assume for direct and indirect detection likelihoods that dark matter
    // density is always the measured one (despite relic density results)
    RD_fraction_one.reset_and_calculate();

    // Calculate DD couplings for DDCalc
    DDCalc_Couplings_WIMP_nucleon.resolveDependency(&DD_couplings_WIMP);
    DDCalc_Couplings_WIMP_nucleon.reset_and_calculate();

    // Set up DDCalc backend initialization
    Backends::DDCalc_2_3_0::Functown::DDCalc_CalcRates_simple.setStatus(FunctorStatus::Active);
    Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment.setStatus(FunctorStatus::Active);
    Backends::DDCalc_2_3_0::Functown::DDCalc_LogLikelihood.setStatus(FunctorStatus::Active);
    DDCalc_2_3_0_init.resolveDependency(&ExtractLocalMaxwellianHalo);
    // Assume for direct and indirect detection likelihoods that dark matter
    // density is always the measured one (despite relic density results)
    DDCalc_2_3_0_init.resolveDependency(&RD_fraction_one);
    DDCalc_2_3_0_init.resolveDependency(&mwimp_generic);
    //DDCalc_2_3_0_init.resolveDependency(&spinwimpx2_generic);
    //DDCalc_2_3_0_init.resolveDependency(&wimp_sc_generic);
    DDCalc_2_3_0_init.resolveDependency(&DDCalc_Couplings_WIMP_nucleon);

    // Initialize gamLike backend
    gamLike_1_0_1_init.reset_and_calculate();

    // Initialize DarkSUSY backend
    DarkSUSY_generic_wimp_6_4_0_init.reset_and_calculate();

    // Initialize MicrOmegas backend
    // The below allows us to initialise MicrOmegas_MSSM without a particular MSSM model.
    MicrOmegas_MSSM_3_6_9_2_init.notifyOfModel("Halo_Einasto");
    MicrOmegas_MSSM_3_6_9_2_init.reset_and_calculate();

    // ---- Gamma-ray and other indirect detection yields ----

    // Initialize tabulated gamma-ray yields
    GA_SimYieldTable_DarkSUSY.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::dsanyield_sim);
    GA_SimYieldTable_MicrOmegas.resolveBackendReq(&Backends::MicrOmegas_MSSM_3_6_9_2::Functown::dNdE);
    GA_SimYieldTable_DarkSUSY.setOption<bool>("allow_yield_extrapolation", true);
    GA_SimYieldTable_MicrOmegas.setOption<bool>("allow_yield_extrapolation", true);

    // Select whether to use DarkSUSY or MicrOmegas for simulated gamma-ray yields
    //auto SimYieldTablePointer = &GA_SimYieldTable_MicrOmegas;
    auto SimYieldTablePointer = &GA_SimYieldTable_DarkSUSY;
    Combine_SimYields.resolveDependency(SimYieldTablePointer);

    // Choose DarkSUSY for e+, e-, pbar and Dbar yields
    positron_SimYieldTable_DarkSUSY.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::dsanyield_sim);
    electron_SimYieldTable_from_positron_SimYieldTable.resolveDependency(&positron_SimYieldTable_DarkSUSY);
    antiproton_SimYieldTable_DarkSUSY.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::dsanyield_sim);
    antideuteron_SimYieldTable_DarkSUSY.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::dsanyield_sim);
    positron_SimYieldTable_DarkSUSY.setOption<bool>("allow_yield_extrapolation", true);
    antiproton_SimYieldTable_DarkSUSY.setOption<bool>("allow_yield_extrapolation", true);
    antideuteron_SimYieldTable_DarkSUSY.setOption<bool>("allow_yield_extrapolation", true);
    Combine_SimYields.resolveDependency(&positron_SimYieldTable_DarkSUSY);
    Combine_SimYields.resolveDependency(&electron_SimYieldTable_from_positron_SimYieldTable);
    Combine_SimYields.resolveDependency(&antiproton_SimYieldTable_DarkSUSY);
    Combine_SimYields.resolveDependency(&antideuteron_SimYieldTable_DarkSUSY);

    // Identify process as annihilation rather than decay
    DM_process_from_ProcessCatalog.resolveDependency(&TH_ProcessCatalog_WIMP);
    DM_process_from_ProcessCatalog.resolveDependency(&DarkMatter_ID_WIMP);

    // Infer for which type of final states particles MC should be performed
    cascadeMC_FinalStates.setOption<std::vector<std::string>>("cMC_finalStates", daFunk::vec((std::string)"gamma"));

    // Set up initial states for cascade MC
    cascadeMC_InitialStates.resolveDependency(&cascadeMC_FinalStates);
    cascadeMC_InitialStates.resolveDependency(&DarkMatter_ID_WIMP);
    cascadeMC_InitialStates.resolveDependency(&DarkMatterConj_ID_WIMP);
    cascadeMC_InitialStates.resolveDependency(&TH_ProcessCatalog_WIMP);
    cascadeMC_InitialStates.resolveDependency(&Combine_SimYields);
    cascadeMC_InitialStates.resolveDependency(&DM_process_from_ProcessCatalog);

    // Collect decay information for cascade MC
    cascadeMC_DecayTable.resolveDependency(&TH_ProcessCatalog_WIMP);
    cascadeMC_DecayTable.resolveDependency(&Combine_SimYields);

    // Set up MC loop manager for cascade MC
    cascadeMC_LoopManager.setOption<int>("cMC_maxEvents", 20000);
    cascadeMC_Histograms.setOption<double>("cMC_endCheckFrequency", 25);
    cascadeMC_Histograms.setOption<double>("cMC_gammaRelError", .05);
    cascadeMC_Histograms.setOption<int>("cMC_numSpecSamples", 25);
    cascadeMC_Histograms.setOption<int>("cMC_NhistBins", 300);
    cascadeMC_LoopManager.resolveDependency(&cascadeMC_InitialStates);
    std::vector<functor*> nested_functions = initVector<functor*>(
        &cascadeMC_GenerateChain, &cascadeMC_Histograms, &cascadeMC_EventCount);
    cascadeMC_LoopManager.setNestedList(nested_functions);

    // Perform MC step for cascade MC
    cascadeMC_GenerateChain.resolveDependency(&cascadeMC_DecayTable);
    cascadeMC_GenerateChain.resolveLoopManager(&cascadeMC_LoopManager);

    // Generate histogram for cascade MC
    cascadeMC_Histograms.resolveDependency(&cascadeMC_GenerateChain);
    cascadeMC_Histograms.resolveDependency(&TH_ProcessCatalog_WIMP);
    cascadeMC_Histograms.resolveDependency(&Combine_SimYields);
    cascadeMC_Histograms.resolveDependency(&cascadeMC_FinalStates);
    cascadeMC_Histograms.resolveLoopManager(&cascadeMC_LoopManager);

    // Check convergence of cascade MC
    cascadeMC_EventCount.resolveLoopManager(&cascadeMC_LoopManager);

    // Infer gamma-ray spectra for recorded MC results
    cascadeMC_gammaSpectra.resolveDependency(&cascadeMC_InitialStates);
    cascadeMC_gammaSpectra.resolveDependency(&cascadeMC_FinalStates);
    cascadeMC_gammaSpectra.resolveDependency(&cascadeMC_Histograms);
    cascadeMC_gammaSpectra.resolveDependency(&cascadeMC_EventCount);

    // Infer positron spectra for recorded MC results
    cascadeMC_positronSpectra.resolveDependency(&cascadeMC_InitialStates);
    cascadeMC_positronSpectra.resolveDependency(&cascadeMC_FinalStates);
    cascadeMC_positronSpectra.resolveDependency(&cascadeMC_Histograms);
    cascadeMC_positronSpectra.resolveDependency(&cascadeMC_EventCount);
    dump_positronSpectrum.resolveDependency(&positron_AnnYield_General);

    // Infer anti-proton spectra for recorded MC results
    cascadeMC_antiprotonSpectra.resolveDependency(&cascadeMC_InitialStates);
    cascadeMC_antiprotonSpectra.resolveDependency(&cascadeMC_FinalStates);
    cascadeMC_antiprotonSpectra.resolveDependency(&cascadeMC_Histograms);
    cascadeMC_antiprotonSpectra.resolveDependency(&cascadeMC_EventCount);
    dump_antiprotonSpectrum.resolveDependency(&antiproton_AnnYield_General);

    // Infer anti-deuteron spectra for recorded MC results
    cascadeMC_antideuteronSpectra.resolveDependency(&cascadeMC_InitialStates);
    cascadeMC_antideuteronSpectra.resolveDependency(&cascadeMC_FinalStates);
    cascadeMC_antideuteronSpectra.resolveDependency(&cascadeMC_Histograms);
    cascadeMC_antideuteronSpectra.resolveDependency(&cascadeMC_EventCount);
    dump_antideuteronSpectrum.resolveDependency(&antideuteron_AnnYield_General);

    // Calculate total gamma-ray yield (cascade MC + tabulated results)
    GA_AnnYield_General.resolveDependency(&TH_ProcessCatalog_WIMP);
    GA_AnnYield_General.resolveDependency(&GA_SimYieldTable_DarkSUSY);
    GA_AnnYield_General.resolveDependency(&DarkMatter_ID_WIMP);
    GA_AnnYield_General.resolveDependency(&DarkMatterConj_ID_WIMP);
    GA_AnnYield_General.resolveDependency(&cascadeMC_gammaSpectra);
    dump_gammaSpectrum.resolveDependency(&GA_AnnYield_General);

    // Calculate total positron yield (cascade MC + tabulated results)
    positron_AnnYield_General.resolveDependency(&TH_ProcessCatalog_WIMP);
    positron_AnnYield_General.resolveDependency(&positron_SimYieldTable_DarkSUSY);
    positron_AnnYield_General.resolveDependency(&DarkMatter_ID_WIMP);
    positron_AnnYield_General.resolveDependency(&DarkMatterConj_ID_WIMP);
    positron_AnnYield_General.resolveDependency(&cascadeMC_positronSpectra);
    dump_positronSpectrum.resolveDependency(&positron_AnnYield_General);

    // Calculate total anti-proton yield (cascade MC + tabulated results)
    antiproton_AnnYield_General.resolveDependency(&TH_ProcessCatalog_WIMP);
    antiproton_AnnYield_General.resolveDependency(&antiproton_SimYieldTable_DarkSUSY);
    antiproton_AnnYield_General.resolveDependency(&DarkMatter_ID_WIMP);
    antiproton_AnnYield_General.resolveDependency(&DarkMatterConj_ID_WIMP);
    antiproton_AnnYield_General.resolveDependency(&cascadeMC_antiprotonSpectra);
    dump_antiprotonSpectrum.resolveDependency(&antiproton_AnnYield_General);

    // Calculate total anti-deuteron yield (cascade MC + tabulated results)
    antideuteron_AnnYield_General.resolveDependency(&TH_ProcessCatalog_WIMP);
    antideuteron_AnnYield_General.resolveDependency(&antideuteron_SimYieldTable_DarkSUSY);
    antideuteron_AnnYield_General.resolveDependency(&DarkMatter_ID_WIMP);
    antideuteron_AnnYield_General.resolveDependency(&DarkMatterConj_ID_WIMP);
    antideuteron_AnnYield_General.resolveDependency(&cascadeMC_antideuteronSpectra);
    dump_antideuteronSpectrum.resolveDependency(&antideuteron_AnnYield_General);

    // Resolve Galactic halo requirements for gamLike
    set_gamLike_GC_halo.resolveDependency(&GalacticHalo_Einasto);
    set_gamLike_GC_halo.resolveBackendReq(&Backends::gamLike_1_0_1::Functown::set_halo_profile);

    // Calculate Fermi LAT dwarf likelihood
    lnL_FermiLATdwarfs_gamLike.resolveDependency(&GA_AnnYield_General);
    lnL_FermiLATdwarfs_gamLike.resolveDependency(&DM_process_from_ProcessCatalog);
    // Assume for direct and indirect detection likelihoods that dark matter
    // density is always the measured one (despite relic density results)
    lnL_FermiLATdwarfs_gamLike.resolveDependency(&RD_fraction_one);
    lnL_FermiLATdwarfs_gamLike.resolveBackendReq(&Backends::gamLike_1_0_1::Functown::lnL);

    lnL_HESSGC_gamLike.resolveDependency(&GA_AnnYield_General);
    lnL_HESSGC_gamLike.resolveDependency(&DM_process_from_ProcessCatalog);
    lnL_HESSGC_gamLike.resolveDependency(&RD_fraction_one);
    lnL_HESSGC_gamLike.resolveBackendReq(&Backends::gamLike_1_0_1::Functown::lnL);

    lnL_CTAGC_gamLike.resolveDependency(&GA_AnnYield_General);
    lnL_CTAGC_gamLike.resolveDependency(&DM_process_from_ProcessCatalog);
    lnL_CTAGC_gamLike.resolveDependency(&RD_fraction_one);
    lnL_CTAGC_gamLike.resolveBackendReq(&Backends::gamLike_1_0_1::Functown::lnL);

    lnL_FermiGC_gamLike.resolveDependency(&GA_AnnYield_General);
    lnL_FermiGC_gamLike.resolveDependency(&DM_process_from_ProcessCatalog);
    lnL_FermiGC_gamLike.resolveDependency(&RD_fraction_one);
    lnL_FermiGC_gamLike.resolveBackendReq(&Backends::gamLike_1_0_1::Functown::lnL);

    // Calculate AMS-02 antiproton log-likelihood ratio
    lnL_pbarAMS02.resolveDependency(&WIMP_properties_WIMP);
    lnL_pbarAMS02.resolveDependency(&TH_ProcessCatalog_WIMP);
    lnL_pbarAMS02.resolveDependency(&ExtractLocalMaxwellianHalo);
    lnL_pbarAMS02.resolveDependency(&RD_fraction_one);
    lnL_pbarAMS02.resolveBackendReq(&Backends::pbarlike_1_0::Functown::c_pbar_logLikes);

    // -- Calculate relic density --
    // *any* of the models listed by "ALLOW_MODELS" in DarkBit_rollcall.hpp will work here
    RD_eff_annrate_from_ProcessCatalog.notifyOfModel("ScalarSingletDM_Z2");
    RD_eff_annrate_from_ProcessCatalog.resolveDependency(&TH_ProcessCatalog_WIMP);
    RD_eff_annrate_from_ProcessCatalog.resolveDependency(&DarkMatter_ID_WIMP);
    RD_eff_annrate_from_ProcessCatalog.resolveDependency(&DarkMatterConj_ID_WIMP);

    RD_spectrum_from_ProcessCatalog.resolveDependency(&TH_ProcessCatalog_WIMP);
    RD_spectrum_from_ProcessCatalog.resolveDependency(&DarkMatter_ID_WIMP);
    RD_spectrum_from_ProcessCatalog.resolveDependency(&DarkMatterConj_ID_WIMP);
    RD_eff_annrate_from_ProcessCatalog.resolveDependency(&DarkMatter_ID_WIMP);
    RD_eff_annrate_from_ProcessCatalog.resolveDependency(&DarkMatterConj_ID_WIMP);

    RD_spectrum_ordered_func.resolveDependency(&RD_spectrum_from_ProcessCatalog);

    RD_oh2_DS6_ini_func.resolveDependency(&RD_spectrum_ordered_func);
    RD_oh2_DS6_ini_func.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::rdpars);
    RD_oh2_DS6_ini_func.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::rdtime);
    RD_oh2_DS6_ini_func.setOption<int>("fast", 1);  // 0: normal; 1: fast; 2: dirty

    RD_oh2_DS_general.resolveDependency(&RD_spectrum_ordered_func);
    RD_oh2_DS_general.resolveDependency(&RD_eff_annrate_from_ProcessCatalog);
    RD_oh2_DS_general.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::dsrdstart);
    RD_oh2_DS_general.resolveBackendReq(&Backends::DarkSUSY_generic_wimp_6_4_0::Functown::dsrdens);
      // Note that this one has to come *last* since it's a conditional dependency
    RD_oh2_DS_general.resolveDependency(&RD_oh2_DS6_ini_func);


    // ---- Calculate direct detection constraints ----

    // Calculate direct detection rates for LZ 2022, PandaX 4T, Xenon 1T and PICO-60
    LZ_2022_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    LZ_2022_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_CalcRates_simple);
    PandaX_4T_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    PandaX_4T_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_CalcRates_simple);
    PICO_60_2019_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    PICO_60_2019_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_CalcRates_simple);
    XENON1T_2018_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    XENON1T_2018_Calc.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_CalcRates_simple);

    // Calculate direct detection likelihood for LZ 2022, PandaX 4T, Xenon 1T and PICO-60
    LZ_2022_GetLogLikelihood.resolveDependency(&LZ_2022_Calc);
    LZ_2022_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    LZ_2022_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_LogLikelihood);
    PandaX_4T_GetLogLikelihood.resolveDependency(&PandaX_4T_Calc);
    PandaX_4T_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    PandaX_4T_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_LogLikelihood);
    XENON1T_2018_GetLogLikelihood.resolveDependency(&XENON1T_2018_Calc);
    XENON1T_2018_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    XENON1T_2018_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_LogLikelihood);
    PICO_60_2019_GetLogLikelihood.resolveDependency(&PICO_60_2019_Calc);
    PICO_60_2019_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    PICO_60_2019_GetLogLikelihood.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_LogLikelihood);

    // Provide bin number in LZ
    LZ_2022_GetBinSignal.resolveDependency(&LZ_2022_Calc);
    LZ_2022_GetBinSignal.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Experiment);
    LZ_2022_GetBinSignal.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_Bins);
    LZ_2022_GetBinSignal.resolveBackendReq(&Backends::DDCalc_2_3_0::Functown::DDCalc_BinSignal);

    // Set generic WIMP mass object
    WIMP_properties_WIMP.resolveDependency(&DarkMatter_ID_WIMP);
    WIMP_properties_WIMP.resolveDependency(&DarkMatterConj_ID_WIMP);
    mwimp_generic.resolveDependency(&WIMP_properties_WIMP);
    TH_ProcessCatalog_WIMP.resolveDependency(&mwimp_generic);
    sigma_SI_p_simple.resolveDependency(&DD_couplings_WIMP);
    sigma_SI_p_simple.resolveDependency(&mwimp_generic);

    // Generate gamma-ray spectra for various final states
    if ( (mode >= 0) and (mode < 6) )
    {
      std::cout << "Producing test spectra." << std::endl;
      double mass = 100.;
      double sv = 3e-26;
      // The array that is being passed to dumpSpectrum give the branching fraction to various final states.
      // They are (as defined in TH_ProcessCatalog_WIMP):
      // 0: b bbar
      // 1: gamma Z_0
      // 2: gamma gamma
      // 3: tau+ tau-
      // 4: W+ W-
      // 5: e+ e-
      // 6: phi phi2
      // 7: gamma e+ e-
      if (mode==5) dumpSpectrum({"dPhi_dE5.dat"}, mass, sv*0.1, daFunk::vec<double>(0., 0., 0., 0., 0., 0., 0., 1.));
      if (mode==0) dumpSpectrum({"dPhi_dE0.dat"}, mass, sv, daFunk::vec<double>(1., 0., 0., 0., 0., 0., 0., 0.));
      if (mode==1) dumpSpectrum({"dPhi_dE1.dat"}, mass, sv, daFunk::vec<double>(0., 1., 0., 0., 0., 0., 0., 0.));
      if (mode==2) dumpSpectrum({"dPhi_dE2.dat"}, mass, sv, daFunk::vec<double>(0., 0., 1., 0., 0., 0., 0., 0.));
      if (mode==3) dumpSpectrum({"dPhi_dE3.dat"}, mass, sv, daFunk::vec<double>(0., 0., 0., 1., 0., 0., 0., 0.));
      if (mode==4) dumpSpectrum({"dPhi_dE4.dat"}, mass, sv, daFunk::vec<double>(0., 0., 0., 0., 1., 0., 0., 0.));
    }

    // CHECK-------------------::-------------;;---------------::-------------------::------------------;;--------------------
    // Generate gamma-ray spectra for various masses
    if (mode >= 10)
    {
      std::cout << "Producing test spectra." << std::endl;
      double mass = 100.;
      double sv = 3e-26;
      std::vector<std::string> filenames =
      {
        "dPhi_dE_FCMC_gamma_" + std::to_string(mode) + ".dat",
        "dPhi_dE_FCMC_positron_" + std::to_string(mode) + ".dat",
        "dPhi_dE_FCMC_antiproton_" + std::to_string(mode) + ".dat",
        "dPhi_dE_FCMC_antideuteron_" + std::to_string(mode) + ".dat"
      };
      dumpSpectrum(filenames, mass, sv, daFunk::vec<double>(0., 0., 0., 0., 0., 0., 1., 0.), mode);
    }

    // Systematic parameter maps annihilation
    if (mode==6)
    {
      std::cout << "Producing gamma ray test maps." << std::endl;
      int mBins = 60;
      int svBins = 60;
      double oh2, lnL;
      std::vector<double> sv_list, m_list;

      GalacticHalo_Einasto.reset_and_calculate();
      set_gamLike_GC_halo.reset_and_calculate();

      boost::multi_array<double, 2>
        lnL_b_array{boost::extents[mBins][svBins]},
        lnL_b_array2{boost::extents[mBins][svBins]},
        lnL_b_array3{boost::extents[mBins][svBins]},
        lnL_b_array4{boost::extents[mBins][svBins]},
        lnL_tau_array{boost::extents[mBins][svBins]};
      boost::multi_array<double, 2> oh2_array{boost::extents[mBins][svBins]};

      sv_list = daFunk::logspace(-28.0, -22.0, svBins);

      std::cout << "Calculating gamma-ray likelihood tables for annihilation to b bbar." << std::endl;
      m_list = daFunk::logspace(log10(5.), 4., mBins);
      for (size_t i = 0; i < m_list.size(); i++)
      {
        for (size_t j = 0; j < sv_list.size(); j++)
        {
          DarkMatter_ID_WIMP.reset_and_calculate();
          DarkMatterConj_ID_WIMP.reset_and_calculate();
          WIMP_properties_WIMP.setOption<double>("mWIMP", m_list[i]);
          WIMP_properties_WIMP.reset_and_calculate();
          mwimp_generic.reset_and_calculate();
          TH_ProcessCatalog_WIMP.setOption<double>("sv", sv_list[j]);


          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Parameters: " << m_list[i] << " " << sv_list[j] << std::endl;
          #endif

          TH_ProcessCatalog_WIMP.setOption<std::vector<double>>("brList", daFunk::vec<double>(1., 0., 0., 0., 0., 0., 0., 0.));
          TH_ProcessCatalog_WIMP.reset_and_calculate();
          DM_process_from_ProcessCatalog.reset_and_calculate();
          RD_fraction_one.reset_and_calculate();
          GA_SimYieldTable_MicrOmegas.reset_and_calculate();
          GA_SimYieldTable_DarkSUSY.reset_and_calculate();
          Combine_SimYields.reset_and_calculate();
          cascadeMC_FinalStates.reset_and_calculate();
          cascadeMC_InitialStates.reset_and_calculate();
          cascadeMC_DecayTable.reset_and_calculate();
          cascadeMC_LoopManager.reset_and_calculate();
          cascadeMC_gammaSpectra.reset_and_calculate();
          GA_AnnYield_General.reset_and_calculate();
          lnL_FermiLATdwarfs_gamLike.setOption<std::string>("version", "pass8");
          lnL_FermiLATdwarfs_gamLike.reset_and_calculate();
          lnL = lnL_FermiLATdwarfs_gamLike(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Fermi dwarf likelihood: " << lnL << std::endl;
          #endif

          lnL_b_array[i][j] = lnL;
          lnL_HESSGC_gamLike.setOption<std::string>("version", "integral_fixedJ");
          lnL_HESSGC_gamLike.reset_and_calculate();
          lnL = lnL_HESSGC_gamLike(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "HESS GC likelihood: " << lnL << std::endl;
          #endif

          lnL_b_array2[i][j] = lnL;
          lnL_CTAGC_gamLike.reset_and_calculate();
          lnL = lnL_CTAGC_gamLike(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "CTA GC likelihood: " << lnL << std::endl;
          #endif

          lnL_b_array3[i][j] = lnL;
          lnL_FermiGC_gamLike.setOption<std::string>("version", "fixedJ");
          lnL_FermiGC_gamLike.reset_and_calculate();
          lnL = lnL_FermiGC_gamLike(0);
          lnL_b_array4[i][j] = lnL;

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Fermi GC likelihood: " << lnL << std::endl;
          #endif
        }
      }

      dump_array_to_file("FermiD_b_table.dat", lnL_b_array, m_list, sv_list);
      dump_array_to_file("HESSGC_b_table.dat", lnL_b_array2, m_list, sv_list);
      dump_array_to_file("CTAGC_b_table.dat", lnL_b_array3, m_list, sv_list);
      dump_array_to_file("FermiGC_b_table.dat", lnL_b_array4, m_list, sv_list);

      std::cout << "Calculating Fermi-LAT dwarf spheroidal likehood table for annihilation to tau+ tau-." << std::endl;
      m_list = daFunk::logspace(log10(1.9), 4., mBins);
      for (size_t i = 0; i < m_list.size(); i++)
      {
        for (size_t j = 0; j < sv_list.size(); j++)
        {
          DarkMatter_ID_WIMP.reset_and_calculate();
          DarkMatterConj_ID_WIMP.reset_and_calculate();
          WIMP_properties_WIMP.setOption<double>("mWIMP", m_list[i]);
          WIMP_properties_WIMP.reset_and_calculate();
          mwimp_generic.reset_and_calculate();
          TH_ProcessCatalog_WIMP.setOption<double>("sv", sv_list[j]);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Parameters: " << m_list[i] << " " << sv_list[j] << std::endl;
          #endif

          TH_ProcessCatalog_WIMP.setOption<std::vector<double>>("brList", daFunk::vec<double>(0., 0., 0., 1., 0., 0., 0., 0.));
          TH_ProcessCatalog_WIMP.reset_and_calculate();
          DM_process_from_ProcessCatalog.reset_and_calculate();
          RD_fraction_one.reset_and_calculate();
          GA_SimYieldTable_MicrOmegas.reset_and_calculate();
          GA_SimYieldTable_DarkSUSY.reset_and_calculate();
          Combine_SimYields.reset_and_calculate();
          cascadeMC_FinalStates.reset_and_calculate();
          cascadeMC_InitialStates.reset_and_calculate();
          cascadeMC_DecayTable.reset_and_calculate();
          cascadeMC_LoopManager.reset_and_calculate();
          cascadeMC_gammaSpectra.reset_and_calculate();
          GA_AnnYield_General.reset_and_calculate();
          lnL_FermiLATdwarfs_gamLike.reset_and_calculate();
          lnL = lnL_FermiLATdwarfs_gamLike(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Fermi LAT likelihood: " << lnL << std::endl;
          #endif

          lnL_tau_array[i][j] = lnL;
        }
      }

      dump_array_to_file("FermiD_tau_table.dat", lnL_tau_array, m_list, sv_list);

      std::cout << "Calculating table of Omega h^2 values." << std::endl;
      m_list = daFunk::logspace(-1.0, 4., mBins);
      for (size_t i = 0; i < m_list.size(); i++)
      {
        for (size_t j = 0; j < sv_list.size(); j++)
        {
          DarkMatter_ID_WIMP.reset_and_calculate();
          DarkMatterConj_ID_WIMP.reset_and_calculate();
          WIMP_properties_WIMP.setOption<double>("mWIMP", m_list[i]);
          WIMP_properties_WIMP.reset_and_calculate();
          mwimp_generic.reset_and_calculate();
          TH_ProcessCatalog_WIMP.setOption<double>("sv", sv_list[j]);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Parameters: " << m_list[i] << " " << sv_list[j] << std::endl;
          #endif

          TH_ProcessCatalog_WIMP.setOption<std::vector<double>>("brList", daFunk::vec<double>(0., 0., 0., 0., 0., 1., 0., 0.));
          TH_ProcessCatalog_WIMP.reset_and_calculate();
          RD_eff_annrate_from_ProcessCatalog.reset_and_calculate();
          RD_spectrum_from_ProcessCatalog.reset_and_calculate();
          RD_spectrum_ordered_func.reset_and_calculate();
          RD_oh2_DS6_ini_func.reset_and_calculate();
          RD_oh2_DS_general.reset_and_calculate();
          oh2 = RD_oh2_DS_general(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Omega h^2 = " << oh2 << std::endl;
          #endif

          oh2_array[i][j] = oh2;
        }
      }

      dump_array_to_file("oh2_table.dat", oh2_array, m_list, sv_list);
    }

    // Systematic parameter maps scattering
    if (mode==7)
    {
      std::cout << "Producing direct detection test maps." << std::endl;
      double lnL1, lnL2, lnL3, lnL4;
      double g, reduced_mass;
      //int mBins = 300;
      //int sBins = 200;
      int mBins = 120;
      int sBins = 80;
      const double mN = (m_proton + m_neutron) / 2;
      std::vector<double> m_list = daFunk::logspace(0.0, 4.0, mBins);
      std::vector<double> s_list;
      boost::multi_array<double, 2> lnL_array1{boost::extents[mBins][sBins]},
          lnL_array2{boost::extents[mBins][sBins]}, lnL_array3{boost::extents[mBins][sBins]},
          lnL_array4{boost::extents[mBins][sBins]};
      TH_ProcessCatalog_WIMP.setOption<double>("sv", 0.);
      TH_ProcessCatalog_WIMP.setOption<std::vector<double>>("brList", daFunk::vec<double>(1., 0., 0., 0., 0., 0., 0., 0.));

      s_list = daFunk::logspace(-47., -40., sBins);
      // Calculate array of sigma_SI and lnL values for LZ, PandaX, XENON1T and PICO-60
      // assuming gps=gns

      std::cout << "Calculating tables of SI likelihoods." << std::endl;
      for (size_t i = 0; i < m_list.size(); i++)
      {
        for (size_t j = 0; j < s_list.size(); j++)
        {
          // Re-initialize DDCalc with LZ/Xenon/PandaX halo parameters
          Halo_primary_parameters->setValue("rho0", 0.3);
          Halo_primary_parameters->setValue("vrot", 232.7); // v_Earth = 245 km/s
          Halo_primary_parameters->setValue("v0", 220.);
          Halo_primary_parameters->setValue("vesc", 544.);
          ExtractLocalMaxwellianHalo.reset_and_calculate();

          DarkMatter_ID_WIMP.reset_and_calculate();
          DarkMatterConj_ID_WIMP.reset_and_calculate();
          WIMP_properties_WIMP.setOption<double>("mWIMP", m_list[i]);
          WIMP_properties_WIMP.reset_and_calculate();
          mwimp_generic.reset_and_calculate();

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Parameters: " << m_list[i] << " " << s_list[j] << std::endl;
          #endif

          reduced_mass = (m_list[i] * mN) / (mN + m_list[i]);
          g = sqrt(s_list[j]*pi/gev2cm2) / (reduced_mass);
          TH_ProcessCatalog_WIMP.reset_and_calculate();

          // Assume for direct and indirect detection likelihoods that dark matter
          // density is always the measured one (despite relic density results)
          RD_fraction_one.reset_and_calculate();
          DD_couplings_WIMP.setOption<double>("gps", g);
          DD_couplings_WIMP.setOption<double>("gns", g);
          DD_couplings_WIMP.setOption<double>("gpa", 0.);
          DD_couplings_WIMP.setOption<double>("gna", 0.);
          DD_couplings_WIMP.reset_and_calculate();
          DDCalc_Couplings_WIMP_nucleon.reset_and_calculate();

          DDCalc_2_3_0_init.reset_and_calculate();
          LZ_2022_Calc.reset_and_calculate();
          LZ_2022_GetLogLikelihood.reset_and_calculate();

          XENON1T_2018_Calc.reset_and_calculate();
          XENON1T_2018_GetLogLikelihood.reset_and_calculate();
          PandaX_4T_Calc.reset_and_calculate();
          PandaX_4T_GetLogLikelihood.reset_and_calculate();

          lnL1 = LZ_2022_GetLogLikelihood(0);
          lnL2 = PandaX_4T_GetLogLikelihood(0);
          lnL3 = XENON1T_2018_GetLogLikelihood(0);

          // Set LocalHalo Model parameters to PICO-60 values
          Halo_primary_parameters->setValue("rho0", 0.3);
          Halo_primary_parameters->setValue("vrot", 220.);
          Halo_primary_parameters->setValue("v0", 220.);
          Halo_primary_parameters->setValue("vesc", 544.);
          ExtractLocalMaxwellianHalo.reset_and_calculate();

          DDCalc_2_3_0_init.reset_and_calculate();
          PICO_60_2019_Calc.reset_and_calculate();
          PICO_60_2019_GetLogLikelihood.reset_and_calculate();
          lnL4 = PICO_60_2019_GetLogLikelihood(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "LZ_2022 SI lnL = " << lnL1 << std::endl;
            std::cout << "PandaX_4T SI lnL = " << lnL2 << std::endl;
            std::cout << "XENON1T_2018 SI lnL = " << lnL3 << std::endl;
            std::cout << "PICO_60_2019 SI lnL = " << lnL4 << std::endl;
          #endif

          DDCalc_2_3_0_init.reset_and_calculate();
          LZ_2022_Calc.reset_and_calculate();
          std::vector<double> events;
          LZ_2022_GetBinSignal.reset_and_calculate();
          events = LZ_2022_GetBinSignal(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            int nbins = events.size();
            std::cout << "Number of LZ bins: " << nbins << std::endl;
            std::cout << "Predicted signal: ";
            for (auto x : events) std::cout << x << " ";
            std::cout << std::endl;
          #endif

          lnL_array1[i][j] = lnL1;
          lnL_array2[i][j] = lnL2;
          lnL_array3[i][j] = lnL3;
          lnL_array4[i][j] = lnL4;
        }
      }

      dump_array_to_file("LZ_2022_SI_table.dat", lnL_array1, m_list, s_list);
      dump_array_to_file("PandaX_4T_SI_table.dat", lnL_array2, m_list, s_list);
      dump_array_to_file("XENON1T_2018_SI_table.dat", lnL_array3, m_list, s_list);
      dump_array_to_file("PICO_60_2019_SI_table.dat", lnL_array4, m_list, s_list);

      s_list = daFunk::logspace(-42., -35., sBins);
      // Calculate array of sigma_SI and lnL values for LZ, PandaX, XENON1T and PICO-60
      // assuming gna=0 (proton-only)

      std::cout << "Calculating tables of SD likelihoods." << std::endl;
      for (size_t i = 0; i < m_list.size(); i++)
      {
        for (size_t j = 0; j < s_list.size(); j++)
        {
          // Re-initialize DDCalc with LZ/Xenon/PandaX halo parameters
          Halo_primary_parameters->setValue("rho0", 0.3);
          Halo_primary_parameters->setValue("vrot", 232.7); // v_Earth = 245 km/s
          Halo_primary_parameters->setValue("v0", 220.);
          Halo_primary_parameters->setValue("vesc", 544.);
          ExtractLocalMaxwellianHalo.reset_and_calculate();

          DarkMatter_ID_WIMP.reset_and_calculate();
          DarkMatterConj_ID_WIMP.reset_and_calculate();
          WIMP_properties_WIMP.setOption<double>("mWIMP", m_list[i]);
          WIMP_properties_WIMP.reset_and_calculate();
          mwimp_generic.reset_and_calculate();
          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "Parameters: " << m_list[i] << " " << s_list[j] << std::endl;
          #endif
          reduced_mass = (m_list[i] * m_proton) / (m_proton + m_list[i]);
          g = sqrt(s_list[j]*pi/(3*gev2cm2)) / (reduced_mass);
          TH_ProcessCatalog_WIMP.reset_and_calculate();

          RD_fraction_one.reset_and_calculate();
          DD_couplings_WIMP.setOption<double>("gps", 0.);
          DD_couplings_WIMP.setOption<double>("gns", 0.);
          DD_couplings_WIMP.setOption<double>("gpa", g);
          DD_couplings_WIMP.setOption<double>("gna", 0.);
          DD_couplings_WIMP.reset_and_calculate();

          DDCalc_2_3_0_init.reset_and_calculate();
          LZ_2022_Calc.reset_and_calculate();
          LZ_2022_GetLogLikelihood.reset_and_calculate();
          XENON1T_2018_Calc.reset_and_calculate();
          XENON1T_2018_GetLogLikelihood.reset_and_calculate();
          PandaX_4T_Calc.reset_and_calculate();
          PandaX_4T_GetLogLikelihood.reset_and_calculate();
          lnL1 = LZ_2022_GetLogLikelihood(0);
          lnL2 = PandaX_4T_GetLogLikelihood(0);
          lnL3 = XENON1T_2018_GetLogLikelihood(0);

          // Set LocalHalo Model parameters to PICO-60 values
          Halo_primary_parameters->setValue("rho0", 0.3);
          Halo_primary_parameters->setValue("vrot", 220.);
          Halo_primary_parameters->setValue("v0", 220.);
          Halo_primary_parameters->setValue("vesc", 544.);
          ExtractLocalMaxwellianHalo.reset_and_calculate();

          DDCalc_2_3_0_init.reset_and_calculate();
          PICO_60_2019_Calc.reset_and_calculate();
          PICO_60_2019_GetLogLikelihood.reset_and_calculate();
          lnL4 = PICO_60_2019_GetLogLikelihood(0);

          #ifdef DARKBIT_STANDALONE_WIMP_DEBUG
            std::cout << "LZ_2022 SD lnL = " << lnL1 << std::endl;
            std::cout << "PandaX_4T SD lnL = " << lnL2 << std::endl;
            std::cout << "XENON1T_2018 SD lnL = " << lnL3 << std::endl;
            std::cout << "PICO_60_2019 SD lnL = " << lnL4 << std::endl;
          #endif

          lnL_array1[i][j] = lnL1;
          lnL_array2[i][j] = lnL2;
          lnL_array3[i][j] = lnL3;
          lnL_array4[i][j] = lnL4;
        }
      }

      dump_array_to_file("LZ_2022_SD_table.dat", lnL_array1, m_list, s_list);
      dump_array_to_file("PandaX_4T_SD_table.dat", lnL_array2, m_list, s_list);
      dump_array_to_file("XENON1T_2018_SD_table.dat", lnL_array3, m_list, s_list);
      dump_array_to_file("PICO_60_2019_SD_table.dat", lnL_array4, m_list, s_list);

      // Reset halo parameters to DarkBit defaults.
      Halo_primary_parameters->setValue("rho0", 0.4);
      Halo_primary_parameters->setValue("vrot", 235.);
      Halo_primary_parameters->setValue("v0", 235.);
      Halo_primary_parameters->setValue("vesc", 550.);
      ExtractLocalMaxwellianHalo.reset_and_calculate();
      GalacticHalo_Einasto.reset_and_calculate();

    }

    // AMS-02 antiproton likelihoods for annihilation into b bbar
    if (mode==8)
    {
      std::cout << "\nCalculating antiproton likelihoods for annihilation to b bbar." << std::endl;
      DarkMatter_ID_WIMP.reset_and_calculate();
      WIMP_properties_WIMP.setOption<double>("mWIMP", 100.0);
      WIMP_properties_WIMP.reset_and_calculate();
      mwimp_generic.reset_and_calculate();
      TH_ProcessCatalog_WIMP.setOption<double>("sv", 3e-26);
      TH_ProcessCatalog_WIMP.setOption<std::vector<double>>("brList", daFunk::vec<double>(1., 0., 0., 0., 0., 0., 0., 0.));
      TH_ProcessCatalog_WIMP.reset_and_calculate();
      pbarlike_1_0_init.setOption<std::string>("PropagationModel","INJ.BRK");
      pbarlike_1_0_init.setOption<bool>("PreventExtrapolation",true);
      pbarlike_1_0_init.setOption<std::vector<std::vector<double>>>("PropagationParameters", std::vector<std::vector<double>>((0.0)));
      std::cout<< "\nParameters: 100 GeV, 3e-26 cm3s-1; Propagation Model: Injection break (INJ.BRK)" << std::endl;
      pbarlike_1_0_init.reset_and_calculate();
      lnL_pbarAMS02.reset_and_calculate();
      map_str_dbl lnL = lnL_pbarAMS02(0);
      std::cout<< "\nLog-likelihood ratio (using uncorrelated errors): " << lnL["uncorrelated"] << std::endl;
      std::cout<< "Log-likelihood ratio (using correlated errors): " << lnL["correlated"] << std::endl;
    }
  }

  catch (std::exception& e)
  {
    std::cout << "DarkBit_standalone_WIMP has exited with fatal exception: " << e.what() << std::endl;
  }

  return 0;
}

Updated on 2024-07-18 at 13:53:34 +0000