/*****************************************************************************
 * Project: RooFit                                                           *
 * Package: RooFitCore                                                       *
 * @(#)root/roofitcore:$Id$
 * Authors:                                                                  *
 *   GR, Gerhard Raven, NIKHEF/VU, Gerhard.Raven@nikhf.nl                    *
 *                                                                           *
 * Copyright (c) 2005, NIKHEF.  All rights reserved.                         *
 *                                                                           *
 * Redistribution and use in source and binary forms,                        *
 * with or without modification, are permitted according to the terms        *
 * listed in LICENSE (http://roofit.sourceforge.net/license.txt)             *
 *****************************************************************************/


//////////////////////////////////////////////////////////////////////////////
// 
// BEGIN_HTML
// RooEffGenContext is a specialized generator context for p.d.fs represented
// by class RooEffProd, which are p.d.fs multiplied with an efficiency function.
// This generator context generates events from such products by first
// generating events from a dedicated generator context of the input p.d.f.
// and applying an extra rejection step based on the efficiency function.
// END_HTML
//

#include <memory>

#include "RooFit.h"
#include "RooEffGenContext.h"
#include "RooAbsPdf.h"
#include "RooRandom.h"

using namespace std;

ClassImp(RooEffGenContext);

//_____________________________________________________________________________
RooEffGenContext::RooEffGenContext(const RooAbsPdf &model, 
                                   const RooAbsPdf& pdf, const RooAbsReal& eff,
                                   const RooArgSet &vars,
                                   const RooDataSet *prototype, const RooArgSet* auxProto,
                                   Bool_t verbose, const RooArgSet* /*forceDirect*/) :
   RooAbsGenContext(model, vars, prototype, auxProto, verbose), _maxEff(0.)
{
   // Constructor of generator context for RooEffProd products

   RooArgSet x(eff,eff.GetName());
   _cloneSet = static_cast<RooArgSet*>(x.snapshot(kTRUE));
   _eff = dynamic_cast<RooAbsReal*>(_cloneSet->find(eff.GetName()));
   _generator = pdf.genContext(vars, prototype, auxProto, verbose);
   _vars = static_cast<RooArgSet*>(vars.snapshot(kTRUE));
}

//_____________________________________________________________________________
RooEffGenContext::~RooEffGenContext()
{
   // Destructor
   delete _generator;
   delete _cloneSet;
   delete _vars;
}

//_____________________________________________________________________________
void RooEffGenContext::initGenerator(const RooArgSet &theEvent)
{
   // One-time initialization of generator.

   _eff->recursiveRedirectServers(theEvent);
   _generator->initGenerator(theEvent);

   // Check if PDF supports maximum finding
   Int_t code = _eff->getMaxVal(*_vars);
   if (!code) {
     _maxEff = 1.;
   } else {
     _maxEff = _eff->maxVal(code);
   }
}

//_____________________________________________________________________________
void RooEffGenContext::generateEvent(RooArgSet &theEvent, Int_t remaining)
{
   // Generate one event. Generate an event from the p.d.f and
   // then perform an accept/reject sampling based on the efficiency
   // function

   while (true) {
      _generator->generateEvent(theEvent, remaining);
      double val = _eff->getVal();
      if (val > _maxEff && !_eff->getMaxVal(*_vars)) {
         coutE(Generation) << ClassName() << "::" << GetName() 
              << ":generateEvent: value of efficiency is larger than assumed maximum of 1."  << std::endl;
         continue;
      }
      if (val > RooRandom::uniform() * _maxEff) {
         break;
      }
   }
}
 RooEffGenContext.cxx:1
 RooEffGenContext.cxx:2
 RooEffGenContext.cxx:3
 RooEffGenContext.cxx:4
 RooEffGenContext.cxx:5
 RooEffGenContext.cxx:6
 RooEffGenContext.cxx:7
 RooEffGenContext.cxx:8
 RooEffGenContext.cxx:9
 RooEffGenContext.cxx:10
 RooEffGenContext.cxx:11
 RooEffGenContext.cxx:12
 RooEffGenContext.cxx:13
 RooEffGenContext.cxx:14
 RooEffGenContext.cxx:15
 RooEffGenContext.cxx:16
 RooEffGenContext.cxx:17
 RooEffGenContext.cxx:18
 RooEffGenContext.cxx:19
 RooEffGenContext.cxx:20
 RooEffGenContext.cxx:21
 RooEffGenContext.cxx:22
 RooEffGenContext.cxx:23
 RooEffGenContext.cxx:24
 RooEffGenContext.cxx:25
 RooEffGenContext.cxx:26
 RooEffGenContext.cxx:27
 RooEffGenContext.cxx:28
 RooEffGenContext.cxx:29
 RooEffGenContext.cxx:30
 RooEffGenContext.cxx:31
 RooEffGenContext.cxx:32
 RooEffGenContext.cxx:33
 RooEffGenContext.cxx:34
 RooEffGenContext.cxx:35
 RooEffGenContext.cxx:36
 RooEffGenContext.cxx:37
 RooEffGenContext.cxx:38
 RooEffGenContext.cxx:39
 RooEffGenContext.cxx:40
 RooEffGenContext.cxx:41
 RooEffGenContext.cxx:42
 RooEffGenContext.cxx:43
 RooEffGenContext.cxx:44
 RooEffGenContext.cxx:45
 RooEffGenContext.cxx:46
 RooEffGenContext.cxx:47
 RooEffGenContext.cxx:48
 RooEffGenContext.cxx:49
 RooEffGenContext.cxx:50
 RooEffGenContext.cxx:51
 RooEffGenContext.cxx:52
 RooEffGenContext.cxx:53
 RooEffGenContext.cxx:54
 RooEffGenContext.cxx:55
 RooEffGenContext.cxx:56
 RooEffGenContext.cxx:57
 RooEffGenContext.cxx:58
 RooEffGenContext.cxx:59
 RooEffGenContext.cxx:60
 RooEffGenContext.cxx:61
 RooEffGenContext.cxx:62
 RooEffGenContext.cxx:63
 RooEffGenContext.cxx:64
 RooEffGenContext.cxx:65
 RooEffGenContext.cxx:66
 RooEffGenContext.cxx:67
 RooEffGenContext.cxx:68
 RooEffGenContext.cxx:69
 RooEffGenContext.cxx:70
 RooEffGenContext.cxx:71
 RooEffGenContext.cxx:72
 RooEffGenContext.cxx:73
 RooEffGenContext.cxx:74
 RooEffGenContext.cxx:75
 RooEffGenContext.cxx:76
 RooEffGenContext.cxx:77
 RooEffGenContext.cxx:78
 RooEffGenContext.cxx:79
 RooEffGenContext.cxx:80
 RooEffGenContext.cxx:81
 RooEffGenContext.cxx:82
 RooEffGenContext.cxx:83
 RooEffGenContext.cxx:84
 RooEffGenContext.cxx:85
 RooEffGenContext.cxx:86
 RooEffGenContext.cxx:87
 RooEffGenContext.cxx:88
 RooEffGenContext.cxx:89
 RooEffGenContext.cxx:90
 RooEffGenContext.cxx:91
 RooEffGenContext.cxx:92
 RooEffGenContext.cxx:93
 RooEffGenContext.cxx:94
 RooEffGenContext.cxx:95
 RooEffGenContext.cxx:96
 RooEffGenContext.cxx:97
 RooEffGenContext.cxx:98
 RooEffGenContext.cxx:99
 RooEffGenContext.cxx:100