laura is hosted by Hepforge, IPPP Durham
Laura++  3.6.0
A maximum likelihood fitting package for performing Dalitz-plot analysis.
LauEffModel.cc
Go to the documentation of this file.
1 
2 /*
3 Copyright 2004 University of Warwick
4 
5 Licensed under the Apache License, Version 2.0 (the "License");
6 you may not use this file except in compliance with the License.
7 You may obtain a copy of the License at
8 
9  http://www.apache.org/licenses/LICENSE-2.0
10 
11 Unless required by applicable law or agreed to in writing, software
12 distributed under the License is distributed on an "AS IS" BASIS,
13 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 See the License for the specific language governing permissions and
15 limitations under the License.
16 */
17 
18 /*
19 Laura++ package authors:
20 John Back
21 Paul Harrison
22 Thomas Latham
23 */
24 
29 #include "LauEffModel.hh"
30 
31 #include "Lau2DHistDP.hh"
32 #include "Lau2DSplineDP.hh"
33 #include "LauDaughters.hh"
34 #include "LauKinematics.hh"
35 #include "LauVetoes.hh"
36 
37 #include "TSystem.h"
38 
39 #include <cstdlib>
40 #include <iostream>
41 
42 LauEffModel::LauEffModel( const LauDaughters* daughters, const LauVetoes* vetoes ) :
43  daughters_( daughters ),
44  vetoes_( vetoes ),
45  effHisto_( 0 ),
46  fluctuateEffHisto_( kFALSE ),
47  lowBinWarningIssued_( kFALSE ),
48  highBinWarningIssued_( kFALSE )
49 {
50  if ( daughters_ == 0 ) {
51  std::cerr << "ERROR in LauEffModel Constructor : invalid pointer to daughters object supplied."
52  << std::endl;
53  gSystem->Exit( EXIT_FAILURE );
54  }
55 }
56 
58 {
59  std::vector<Lau2DAbsDP*>::iterator it = effHisto_.begin();
60  std::vector<Lau2DAbsDP*>::iterator end = effHisto_.end();
61  for ( ; it != end; ++it ) {
62  delete *it;
63  *it = 0;
64  }
65  effHisto_.clear();
66 }
67 
68 void LauEffModel::setEffHisto( const TH2* effHisto,
69  Bool_t useInterpolation,
70  Bool_t fluctuateBins,
71  Double_t avEff,
72  Double_t absError,
73  Bool_t useUpperHalfOnly,
74  Bool_t squareDP )
75 {
76  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
77  // with x = m_13^2, y = m_23^2.
78  Bool_t upperHalf( kFALSE );
79  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
80  upperHalf = kTRUE;
81  }
82  std::cout << "INFO in LauEffModel::setEffHisto : Efficiency histogram has upperHalf = "
83  << static_cast<Int_t>( upperHalf ) << std::endl;
84 
85  std::vector<Lau2DAbsDP*>::iterator it = effHisto_.begin();
86  std::vector<Lau2DAbsDP*>::iterator end = effHisto_.end();
87  for ( ; it != end; ++it ) {
88  delete *it;
89  *it = 0;
90  }
91  effHisto_.clear();
92 
93  // Copy the histogram.
94  effHisto_.push_back( new Lau2DHistDP( effHisto,
95  daughters_,
96  useInterpolation,
97  fluctuateBins,
98  avEff,
99  absError,
100  upperHalf,
101  squareDP ) );
102  fluctuateEffHisto_ = fluctuateBins;
103 
104  if ( avEff > 0.0 && absError > 0.0 ) {
105  fluctuateEffHisto_ = kTRUE;
106  }
107 }
108 
109 void LauEffModel::setEffHisto( const TH2* effHisto,
110  const TH2* errorHi,
111  const TH2* errorLo,
112  Bool_t useInterpolation,
113  Bool_t fluctuateBins,
114  Double_t avEff,
115  Double_t absError,
116  Bool_t useUpperHalfOnly,
117  Bool_t squareDP )
118 {
119  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
120  // with x = m_13^2, y = m_23^2.
121  Bool_t upperHalf( kFALSE );
122  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
123  upperHalf = kTRUE;
124  }
125  std::cout << "INFO in LauEffModel::setEffHisto : Efficiency histogram has upperHalf = "
126  << static_cast<Int_t>( upperHalf ) << std::endl;
127 
128  std::vector<Lau2DAbsDP*>::iterator it = effHisto_.begin();
129  std::vector<Lau2DAbsDP*>::iterator end = effHisto_.end();
130  for ( ; it != end; ++it ) {
131  delete *it;
132  *it = 0;
133  }
134  effHisto_.clear();
135 
136  // Copy the histogram.
137  effHisto_.push_back( new Lau2DHistDP( effHisto,
138  errorHi,
139  errorLo,
140  daughters_,
141  useInterpolation,
142  fluctuateBins,
143  avEff,
144  absError,
145  upperHalf,
146  squareDP ) );
147  fluctuateEffHisto_ = fluctuateBins;
148 
149  if ( avEff > 0.0 && absError > 0.0 ) {
150  fluctuateEffHisto_ = kTRUE;
151  }
152 }
153 
154 void LauEffModel::setEffSpline( const TH2* effHisto,
155  Bool_t fluctuateBins,
156  Double_t avEff,
157  Double_t absError,
158  Bool_t useUpperHalfOnly,
159  Bool_t squareDP )
160 {
161  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
162  // with x = m_13^2, y = m_23^2.
163  Bool_t upperHalf( kFALSE );
164  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
165  upperHalf = kTRUE;
166  }
167  std::cout << "INFO in LauEffModel::setEffSpline : Efficiency histogram has upperHalf = "
168  << static_cast<Int_t>( upperHalf ) << std::endl;
169 
170  std::vector<Lau2DAbsDP*>::iterator it = effHisto_.begin();
171  std::vector<Lau2DAbsDP*>::iterator end = effHisto_.end();
172  for ( ; it != end; ++it ) {
173  delete *it;
174  *it = 0;
175  }
176  effHisto_.clear();
177 
178  // Copy the histogram.
179  effHisto_.push_back(
180  new Lau2DSplineDP( effHisto, daughters_, fluctuateBins, avEff, absError, upperHalf, squareDP ) );
181  fluctuateEffHisto_ = fluctuateBins;
182 
183  if ( avEff > 0.0 && absError > 0.0 ) {
184  fluctuateEffHisto_ = kTRUE;
185  }
186 }
187 
188 void LauEffModel::setEffSpline( const TH2* effHisto,
189  const TH2* errorHi,
190  const TH2* errorLo,
191  Bool_t fluctuateBins,
192  Double_t avEff,
193  Double_t absError,
194  Bool_t useUpperHalfOnly,
195  Bool_t squareDP )
196 {
197  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
198  // with x = m_13^2, y = m_23^2.
199  Bool_t upperHalf( kFALSE );
200  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
201  upperHalf = kTRUE;
202  }
203  std::cout << "INFO in LauEffModel::setEffSpline : Efficiency histogram has upperHalf = "
204  << static_cast<Int_t>( upperHalf ) << std::endl;
205 
206  std::vector<Lau2DAbsDP*>::iterator it = effHisto_.begin();
207  std::vector<Lau2DAbsDP*>::iterator end = effHisto_.end();
208  for ( ; it != end; ++it ) {
209  delete *it;
210  *it = 0;
211  }
212  effHisto_.clear();
213 
214  // Copy the histogram.
215  effHisto_.push_back( new Lau2DSplineDP( effHisto,
216  errorHi,
217  errorLo,
218  daughters_,
219  fluctuateBins,
220  avEff,
221  absError,
222  upperHalf,
223  squareDP ) );
224  fluctuateEffHisto_ = fluctuateBins;
225 
226  if ( avEff > 0.0 && absError > 0.0 ) {
227  fluctuateEffHisto_ = kTRUE;
228  }
229 }
230 
231 void LauEffModel::addEffHisto( const TH2* effHisto,
232  Bool_t useInterpolation,
233  Double_t avEff,
234  Double_t absError,
235  Bool_t useUpperHalfOnly,
236  Bool_t squareDP )
237 {
238  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
239  // with x = m_13^2, y = m_23^2.
240  Bool_t upperHalf( kFALSE );
241  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
242  upperHalf = kTRUE;
243  }
244  std::cout << "INFO in LauEffModel::addEffHisto : Efficiency histogram has upperHalf = "
245  << static_cast<Int_t>( upperHalf ) << std::endl;
246 
247  // Copy the histogram.
248  effHisto_.push_back( new Lau2DHistDP( effHisto,
249  daughters_,
250  useInterpolation,
252  avEff,
253  absError,
254  upperHalf,
255  squareDP ) );
256 }
257 
258 void LauEffModel::addEffHisto( const TH2* effHisto,
259  const TH2* errorHi,
260  const TH2* errorLo,
261  Bool_t useInterpolation,
262  Double_t avEff,
263  Double_t absError,
264  Bool_t useUpperHalfOnly,
265  Bool_t squareDP )
266 {
267  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
268  // with x = m_13^2, y = m_23^2.
269  Bool_t upperHalf( kFALSE );
270  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
271  upperHalf = kTRUE;
272  }
273  std::cout << "INFO in LauEffModel::addEffHisto : Efficiency histogram has upperHalf = "
274  << static_cast<Int_t>( upperHalf ) << std::endl;
275 
276  // Copy the histogram.
277  effHisto_.push_back( new Lau2DHistDP( effHisto,
278  errorHi,
279  errorLo,
280  daughters_,
281  useInterpolation,
283  avEff,
284  absError,
285  upperHalf,
286  squareDP ) );
287 }
288 
289 void LauEffModel::addEffSpline( const TH2* effHisto,
290  Double_t avEff,
291  Double_t absError,
292  Bool_t useUpperHalfOnly,
293  Bool_t squareDP )
294 {
295  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
296  // with x = m_13^2, y = m_23^2.
297  Bool_t upperHalf( kFALSE );
298  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
299  upperHalf = kTRUE;
300  }
301  std::cout << "INFO in LauEffModel::addEffSpline : Efficiency histogram has upperHalf = "
302  << static_cast<Int_t>( upperHalf ) << std::endl;
303 
304  // Copy the histogram.
305  effHisto_.push_back( new Lau2DSplineDP( effHisto,
306  daughters_,
308  avEff,
309  absError,
310  upperHalf,
311  squareDP ) );
312 }
313 
314 void LauEffModel::addEffSpline( const TH2* effHisto,
315  const TH2* errorHi,
316  const TH2* errorLo,
317  Double_t avEff,
318  Double_t absError,
319  Bool_t useUpperHalfOnly,
320  Bool_t squareDP )
321 {
322  // Set the efficiency across the Dalitz plot using a predetermined 2D histogram
323  // with x = m_13^2, y = m_23^2.
324  Bool_t upperHalf( kFALSE );
325  if ( daughters_->gotSymmetricalDP() && useUpperHalfOnly == kTRUE ) {
326  upperHalf = kTRUE;
327  }
328  std::cout << "INFO in LauEffModel::addEffSpline : Efficiency histogram has upperHalf = "
329  << static_cast<Int_t>( upperHalf ) << std::endl;
330 
331  // Copy the histogram.
332  effHisto_.push_back( new Lau2DSplineDP( effHisto,
333  errorHi,
334  errorLo,
335  daughters_,
337  avEff,
338  absError,
339  upperHalf,
340  squareDP ) );
341 }
342 
343 Double_t LauEffModel::getEffHistValue( const LauKinematics* kinematics ) const
344 {
345  // Get the efficiency from the 2D histo.
346  Double_t eff( 1.0 );
347 
348  Double_t xVal( 0.0 );
349  Double_t yVal( 0.0 );
350 
351  std::vector<Lau2DAbsDP*>::const_iterator it = effHisto_.begin();
352  const std::vector<Lau2DAbsDP*>::const_iterator end = effHisto_.end();
353  for ( ; it != end; ++it ) {
354  if ( ( *it )->usingSquareDP() ) {
355  xVal = kinematics->getmPrime();
356  yVal = kinematics->getThetaPrime();
357  } else {
358  xVal = kinematics->getm13Sq();
359  yVal = kinematics->getm23Sq();
360  }
361  eff *= ( *it )->interpolateXY( xVal, yVal );
362  }
363 
364  return eff;
365 }
366 
367 Double_t LauEffModel::calcEfficiency( const LauKinematics* kinematics ) const
368 {
369  // Routine to calculate the efficiency for the given event/point in
370  // the Dalitz plot. This routine uses the 2D histogram set by the
371  // setEffHisto() function.
372  Double_t eff( 1.0 );
373 
374  // Check for vetoes
375  Bool_t vetoOK( kTRUE );
376  if ( vetoes_ != 0 ) {
377  vetoOK = vetoes_->passVeto( kinematics );
378  }
379 
380  if ( vetoOK == kFALSE ) {
381  // We failed the veto.
382  eff = 0.0;
383  } else {
384  // We are using a 2D histogram representation of the efficiency across the Dalitz plot.
385  eff = this->getEffHistValue( kinematics );
386  }
387 
388  // Check that the efficiency is in the allowed range (0-1)
389  // If we're using a spline then out-of-range efficiencies can be caused by adjacent bins that all contain a value of either zero or one.
390  // The spline requires the efficiency, its first derivatives and the mixed second derivative to be continuous and to match the input histogram
391  // at the bin centres. Derivatives are calculated using a finite difference approximation taking the difference between the neighbouring bins.
392  // If two bins are zero but the third is not then the second bin will have a positive first derivative causing the spline to dip below zero
393  // between the two zero bins to remain smooth. The analogous case with adjacent maximised bins will cause peaks above one. Such dips are
394  // unavoidable but are correctly removed here.
395  if ( eff < 0.0 ) {
396  if ( ! lowBinWarningIssued_ ) {
397  std::cerr << "WARNING in LauEffModel::calcEfficiency : Efficiency " << eff
398  << " is less than 0 - setting to 0. You may want to check your histogram!"
399  << std::endl
400  << " : If you are using a spline then this could be caused by adjacent empty bins. Further warnings will be suppressed."
401  << std::endl;
402  lowBinWarningIssued_ = kTRUE;
403  }
404  eff = 0.0;
405  } else if ( eff > 1.0 ) {
406  if ( ! highBinWarningIssued_ ) {
407  std::cerr << "WARNING in LauEffModel::calcEfficiency : Efficiency " << eff
408  << " is greater than 1 - setting to 1. You may want to check your histogram!"
409  << std::endl
410  << " : If you are using a spline then this could be caused by adjacent full bins. Further warnings will be suppressed."
411  << std::endl;
412  highBinWarningIssued_ = kTRUE;
413  }
414  eff = 1.0;
415  }
416 
417  return eff;
418 }
419 
420 Bool_t LauEffModel::passVeto( const LauKinematics* kinematics ) const
421 {
422  Bool_t pass = kTRUE;
423  if ( vetoes_ != 0 ) {
424  pass = vetoes_->passVeto( kinematics );
425  }
426  return pass;
427 }
File containing declaration of LauEffModel class.
void addEffSpline(const TH2 *effHisto, Double_t avEff=-1.0, Double_t absError=-1.0, Bool_t useUpperHalfOnly=kFALSE, Bool_t squareDP=kFALSE)
Add a multiplicative efficiency variation across the phase space using a spline based on a predetermi...
Definition: LauEffModel.cc:289
virtual ~LauEffModel()
Destructor.
Definition: LauEffModel.cc:57
std::vector< Lau2DAbsDP * > effHisto_
The efficiency histogram objects.
Definition: LauEffModel.hh:292
Bool_t fluctuateEffHisto_
Fluctuate histogram within the error.
Definition: LauEffModel.hh:295
Double_t getThetaPrime() const
Get theta' value.
Double_t getmPrime() const
Get m' value.
Double_t calcEfficiency(const LauKinematics *kinematics) const
Determine the efficiency for a given point in the Dalitz plot.
Definition: LauEffModel.cc:367
LauEffModel(const LauDaughters *daughters, const LauVetoes *vetoes)
Constructor.
Definition: LauEffModel.cc:42
Bool_t passVeto(const LauKinematics *kinematics) const
Check whether the specified Dalitz plot point passes the vetoes.
Definition: LauVetoes.cc:124
void setEffHisto(const TH2 *effHisto, Bool_t useInterpolation=kTRUE, Bool_t fluctuateBins=kFALSE, Double_t avEff=-1.0, Double_t absError=-1.0, Bool_t useUpperHalfOnly=kFALSE, Bool_t squareDP=kFALSE)
Set the efficiency variation across the phase space using a predetermined 2D histogram.
Definition: LauEffModel.cc:68
const LauVetoes * vetoes_
The vetoes object.
Definition: LauEffModel.hh:289
File containing declaration of Lau2DHistDP class.
Bool_t lowBinWarningIssued_
Flag to track whether a warning has been issued for bin values less than zero.
Definition: LauEffModel.hh:298
void addEffHisto(const TH2 *effHisto, Bool_t useInterpolation=kTRUE, Double_t avEff=-1.0, Double_t absError=-1.0, Bool_t useUpperHalfOnly=kFALSE, Bool_t squareDP=kFALSE)
Add a multiplicative efficiency variation across the phase space using a predetermined 2D histogram.
Definition: LauEffModel.cc:231
File containing declaration of LauDaughters class.
Class for defining variations across a 2D DP using a spline.
void setEffSpline(const TH2 *effHisto, Bool_t fluctuateBins=kFALSE, Double_t avEff=-1.0, Double_t absError=-1.0, Bool_t useUpperHalfOnly=kFALSE, Bool_t squareDP=kFALSE)
Set the efficiency variation across the phase space using a spline based on a predetermined 2D histog...
Definition: LauEffModel.cc:154
Class for defining a 2D DP histogram.
Definition: Lau2DHistDP.hh:48
Bool_t highBinWarningIssued_
Flag to track whether a warning has been issued for bin values greater than one.
Definition: LauEffModel.hh:301
Class for defining vetoes within the Dalitz plot.
Definition: LauVetoes.hh:49
File containing declaration of Lau2DSplineDP class.
Class for calculating 3-body kinematic quantities.
Bool_t passVeto(const LauKinematics *kinematics) const
Determine whether the given DP position is outside the vetoes.
Definition: LauEffModel.cc:420
File containing declaration of LauVetoes class.
Class that defines the particular 3-body decay under study.
Definition: LauDaughters.hh:47
Double_t getm23Sq() const
Get the m23 invariant mass square.
Double_t getm13Sq() const
Get the m13 invariant mass square.
Bool_t gotSymmetricalDP() const
Is Dalitz plot symmetric, i.e. 2 identical particles.
Definition: LauDaughters.hh:84
Double_t getEffHistValue(const LauKinematics *kinematics) const
Get the efficiency from a two-dimensional histogram.
Definition: LauEffModel.cc:343
const LauDaughters * daughters_
The daughters object.
Definition: LauEffModel.hh:286
File containing declaration of LauKinematics class.