source: tags/Mars-V0.10.1/mhcalib/MHCalibrationChargeBlindPix.cc

Last change on this file was 8019, checked in by tbretz, 18 years ago
*** empty log message ***
File size: 29.7 KB
Line 
1/* ======================================================================== *\
2!
3! *
4! * This file is part of MARS, the MAGIC Analysis and Reconstruction
5! * Software. It is distributed to you in the hope that it can be a useful
6! * and timesaving tool in analysing Data of imaging Cerenkov telescopes.
7! * It is distributed WITHOUT ANY WARRANTY.
8! *
9! * Permission to use, copy, modify and distribute this software and its
10! * documentation for any purpose is hereby granted without fee,
11! * provided that the above copyright notice appear in all copies and
12! * that both that copyright notice and this permission notice appear
13! * in supporting documentation. It is provided "as is" without express
14! * or implied warranty.
15! *
16!
17!
18! Author(s): Markus Gaug 02/2004 <mailto:markus@ifae.es>
19!
20! Copyright: MAGIC Software Development, 2000-2004
21!
22!
23\* ======================================================================== */
24
25//////////////////////////////////////////////////////////////////////////////
26//
27// MHCalibrationChargeBlindPix
28//
29// Histogram class for the charge calibration of the Blind Pixel.
30// Stores and fits the charges and stores the averaged assumed pedestal and
31// single-phe FADC slice entries. Charges are taken from MExtractedSignalBlindPix.
32// Performs the Single Photo-electron fit to extract the Poisson mean and its errors.
33//
34// Different fits can be chosen with the function SetFitFunc().
35//
36// The fit result is accepted under the condition that:
37// 1) the Probability is greater than fProbLimit (default 0.001 == 99.7%)
38// 2) at least fNumSinglePheLimit events are found in the single Photo-electron peak
39//
40// The single FADC slice entries are averaged and stored in fASinglePheFADCSlices, if
41// their sum exceeds fSinglePheCut, otherwise in fAPedestalFADCSlices.
42//
43// Used numbers are the following:
44//
45// Electronic conversion factor:
46// Assume, we have N_e electrons at the anode,
47// thus a charge of N_e*e (e = electron charge) Coulomb.
48//
49// This charge is AC coupled and runs into a R_pre = 50 Ohm resistency.
50// The corresponding current is amplified by a gain factor G_pre = 400
51// (the precision of this value still has to be checked !!!) and again AC coupled to
52// the output.
53// The corresponding signal goes through the whole transmission and
54// amplification chain and is digitized in the FADCs.
55// The conversion Signal Area to FADC counts (Conv_trans) has been measured
56// by David and Oscar to be approx. 3.9 pVs^-1
57//
58// Thus: Conversion FADC counts to Number of Electrons at Anode:
59// FADC counts = (1/Conv_tran) * G_pre * R_pre * e * N_e = 8 * 10^-4 N_e.
60//
61// Also: FADC counts = 8*10^-4 * GAIN * N_phe
62//
63// In the blind pixel, there is an additional pre-amplifier with an amplification of
64// about 10. Therefore, we have for the blind pixel:
65//
66//
67// FADC counts (Blind Pixel) = 8*10^-3 * GAIN * N_phe
68//
69//////////////////////////////////////////////////////////////////////////////
70#include "MHCalibrationChargeBlindPix.h"
71
72#include <TStyle.h>
73#include <TCanvas.h>
74#include <TPaveText.h>
75#include <TPaveStats.h>
76#include <TGraph.h>
77
78#include <TVector.h>
79#include <TF1.h>
80#include <TH1.h>
81#include <TH2D.h>
82#include <TRandom.h>
83
84#include "MLog.h"
85#include "MLogManip.h"
86
87#include "MParList.h"
88
89#include "MRawEvtPixelIter.h"
90
91ClassImp(MHCalibrationChargeBlindPix);
92
93using namespace std;
94
95const Double_t MHCalibrationChargeBlindPix::gkElectronicAmp = 0.008;
96const Double_t MHCalibrationChargeBlindPix::gkElectronicAmpErr = 0.002;
97const Float_t MHCalibrationChargeBlindPix::gkSignalInitializer = -9999.;
98const Float_t MHCalibrationChargeBlindPix::fgNumSinglePheLimit = 50.;
99// --------------------------------------------------------------------------
100//
101// Default Constructor.
102//
103// Sets:
104// - the default number for fNumSinglePheLimit (fgNumSinglePheLimit)
105// - the default number of bins after stripping (30)
106//
107// Initializes:
108// - all pointers to NULL
109// - fASinglePheFADCSlices(0);
110// - fAPedestalFADCSlices(0);
111//
112// Calls:
113// - Clear()
114//
115MHCalibrationChargeBlindPix::MHCalibrationChargeBlindPix(const char *name, const char *title)
116 : fSinglePheFit(NULL),
117 fFitLegend(NULL),
118 fHSinglePheFADCSlices(NULL), fHPedestalFADCSlices(NULL)
119{
120
121 fName = name ? name : "MHCalibrationChargeBlindPix";
122 fTitle = title ? title : "Statistics of the FADC sums of Blind Pixel calibration events";
123
124 fASinglePheFADCSlices.ResizeTo(1);
125 fAPedestalFADCSlices.ResizeTo(1);
126
127 SetNumSinglePheLimit();
128 SetProbLimit(0.001);
129 SetBinsAfterStripping(0);
130
131 Clear();
132}
133
134// --------------------------------------------------------------------------
135//
136// Default Destructor.
137//
138// Deletes (if Pointer is not NULL):
139//
140// - fFitLegend
141// - fHSinglePheFADCSlices
142// - fHPedestalFADCSlices
143// - fSinglePheFit
144//
145MHCalibrationChargeBlindPix::~MHCalibrationChargeBlindPix()
146{
147
148 //
149 // The next two lines are important for the case that
150 // the class has been stored to a file and is read again.
151 // In this case, the next two lines prevent a segm. violation
152 // in the destructor
153 //
154 gROOT->GetListOfFunctions()->Remove(fSinglePheFit);
155
156 if (fSinglePheFit)
157 delete fSinglePheFit;
158
159 if (fFitLegend)
160 delete fFitLegend;
161
162 if (fHSinglePheFADCSlices)
163 delete fHSinglePheFADCSlices;
164
165 if (fHPedestalFADCSlices)
166 delete fHPedestalFADCSlices;
167
168}
169
170// --------------------------------------------------------------------------
171//
172// Sets:
173// - all variables to 0., except the fit result variables to gkSignalInitializer
174// - all flags to kFALSE
175// - all pointers to NULL
176// - the default fit function (kEPoisson5)
177//
178// Deletes:
179// - all pointers unequal NULL
180//
181// Calls:
182// - MHCalibrationChargePix::Clear()
183//
184void MHCalibrationChargeBlindPix::Clear(Option_t *o)
185{
186
187 fLambda = gkSignalInitializer;
188 fMu0 = gkSignalInitializer;
189 fMu1 = gkSignalInitializer;
190 fSigma0 = gkSignalInitializer;
191 fSigma1 = gkSignalInitializer;
192 fLambdaErr = gkSignalInitializer;
193 fMu0Err = gkSignalInitializer;
194 fMu1Err = gkSignalInitializer;
195 fSigma0Err = gkSignalInitializer;
196 fSigma1Err = gkSignalInitializer;
197
198 fLambdaCheck = gkSignalInitializer;
199 fLambdaCheckErr = gkSignalInitializer;
200
201 fFitFunc = kEPoisson5;
202
203 fNumSinglePhes = 0;
204 fNumPedestals = 0;
205
206 fChisquare = 0.;
207 fNDF = 0 ;
208 fProb = 0.;
209
210 SetSinglePheFitOK ( kFALSE );
211 SetPedestalFitOK ( kFALSE );
212
213 if (fFitLegend)
214 {
215 delete fFitLegend;
216 fFitLegend = NULL;
217 }
218
219 if (fSinglePheFit)
220 {
221 delete fSinglePheFit;
222 fSinglePheFit = NULL;
223 }
224
225 if (fHSinglePheFADCSlices)
226 {
227 delete fHSinglePheFADCSlices;
228 fHSinglePheFADCSlices = NULL;
229 }
230
231 if (fHPedestalFADCSlices)
232 {
233 delete fHPedestalFADCSlices;
234 fHPedestalFADCSlices = NULL;
235 }
236
237 MHCalibrationPix::Clear();
238 return;
239}
240
241/*
242// --------------------------------------------------------------------------
243//
244// Our own clone function is necessary since root 3.01/06 or Mars 0.4
245// I don't know the reason.
246//
247// Creates new MHCalibrationCam
248//
249TObject *MHCalibrationChargeBlindPix::Clone(const char *) const
250{
251
252 MHCalibrationChargeBlindPix *pix = new MHCalibrationChargeBlindPix();
253 this->Copy(*pix);
254
255 this->fHGausHist.Copy(pix->fHGausHist);
256 this->fSinglePheFit->Copy(*(pix->fSinglePheFit));
257 this->fHSinglePheFADCSlices->Copy(*(pix->fHSinglePheFADCSlices));
258 this->fHPedestalFADCSlices->Copy(*(pix->fHPedestalFADCSlices));
259
260
261 return pix;
262}
263*/
264
265// --------------------------------------------------------------------------
266//
267// Set bit kSinglePheFitOK from outside
268//
269void MHCalibrationChargeBlindPix::SetSinglePheFitOK (const Bool_t b )
270{
271 b ? SETBIT(fFlags,kSinglePheFitOK) : CLRBIT(fFlags,kSinglePheFitOK);
272}
273
274// --------------------------------------------------------------------------
275//
276// Set bit kPedestalFitOK from outside
277//
278void MHCalibrationChargeBlindPix::SetPedestalFitOK(const Bool_t b)
279{
280 b ? SETBIT(fFlags,kPedestalFitOK) : CLRBIT(fFlags,kPedestalFitOK);
281}
282
283// --------------------------------------------------------------------------
284//
285// Ask for status of bit kSinglePheFitOK
286//
287const Bool_t MHCalibrationChargeBlindPix::IsSinglePheFitOK() const
288{
289 return TESTBIT(fFlags,kSinglePheFitOK);
290}
291
292// --------------------------------------------------------------------------
293//
294// Ask for status of bit kPedestalFitOK
295//
296const Bool_t MHCalibrationChargeBlindPix::IsPedestalFitOK() const
297{
298 return TESTBIT(fFlags,kPedestalFitOK);
299}
300
301// --------------------------------------------------------------------------
302//
303// Normalizes fASinglePheFADCSlices and fAPedestalFADCSlices
304//
305void MHCalibrationChargeBlindPix::FinalizeSinglePheSpectrum()
306{
307
308 if (fNumSinglePhes > 1)
309 for (Int_t i=0;i<fASinglePheFADCSlices.GetNrows();i++)
310 fASinglePheFADCSlices[i] = fASinglePheFADCSlices[i]/fNumSinglePhes;
311 if (fNumPedestals > 1)
312 for (Int_t i=0;i<fAPedestalFADCSlices.GetNrows();i++)
313 fAPedestalFADCSlices[i] = fAPedestalFADCSlices[i]/fNumPedestals;
314}
315
316// --------------------------------------------------------------------------
317//
318// Checks again for the size and fills fASinglePheFADCSlices with the FADC slice entries
319//
320void MHCalibrationChargeBlindPix::FillSinglePheFADCSlices(const MRawEvtPixelIter &iter)
321{
322
323 const Int_t n = iter.GetNumHiGainSamples() + iter.GetNumLoGainSamples();
324
325 if (fASinglePheFADCSlices.GetNrows() < n)
326 fASinglePheFADCSlices.ResizeTo(n);
327
328 Int_t i=0;
329
330 Byte_t *start = iter.GetHiGainSamples();
331 Byte_t *end = start + iter.GetNumHiGainSamples();
332
333 for (Byte_t *ptr = start; ptr < end; ptr++, i++)
334 fASinglePheFADCSlices(i) = fASinglePheFADCSlices(i) + (Float_t)*ptr;
335
336 start = iter.GetLoGainSamples();
337 end = start + iter.GetNumLoGainSamples();
338
339 for (Byte_t *ptr = start; ptr < end; ptr++, i++)
340 fASinglePheFADCSlices(i) = fASinglePheFADCSlices(i) + (Float_t)*ptr;
341
342 fNumSinglePhes++;
343}
344
345// --------------------------------------------------------------------------
346//
347// Checks again for the size and fills fAPedestalFADCSlices with the FADC slice entries
348//
349void MHCalibrationChargeBlindPix::FillPedestalFADCSlices(const MRawEvtPixelIter &iter)
350{
351
352 const Int_t n = iter.GetNumHiGainSamples() + iter.GetNumLoGainSamples();
353
354 if (fAPedestalFADCSlices.GetNrows() < n)
355 fAPedestalFADCSlices.ResizeTo(n);
356
357 Int_t i = 0;
358 Byte_t *start = iter.GetHiGainSamples();
359 Byte_t *end = start + iter.GetNumHiGainSamples();
360
361 for (Byte_t *ptr = start; ptr < end; ptr++, i++)
362 fAPedestalFADCSlices(i) = fAPedestalFADCSlices(i)+ (Float_t)*ptr;
363
364 start = iter.GetLoGainSamples();
365 end = start + iter.GetNumLoGainSamples();
366
367 for (Byte_t *ptr = start; ptr < end; ptr++, i++)
368 fAPedestalFADCSlices(i) = fAPedestalFADCSlices(i)+ (Float_t)*ptr;
369
370 fNumPedestals++;
371}
372
373
374// --------------------------------------------------------------------------
375//
376// Task to simulate single phe spectrum with the given parameters
377//
378Bool_t MHCalibrationChargeBlindPix::SimulateSinglePhe(Double_t lambda, Double_t mu0, Double_t mu1, Double_t sigma0, Double_t sigma1)
379{
380
381 gRandom->SetSeed();
382
383 if (fHGausHist.GetIntegral() != 0)
384 {
385 *fLog << err << "Histogram " << fHGausHist.GetTitle() << " is already filled. " << endl;
386 *fLog << err << "Create new class MHCalibrationBlindPixel for simulation! " << endl;
387 return kFALSE;
388 }
389
390 if (!InitFit())
391 return kFALSE;
392
393 for (Int_t i=0;i<10000; i++)
394 fHGausHist.Fill(fSinglePheFit->GetRandom());
395
396 return kTRUE;
397}
398
399// --------------------------------------------------------------------------
400//
401// - Get the ranges from the stripped histogram
402// - choose reasonable start values for the fit
403// - initialize the fit function depending on fFitFunc
404// - initialize parameter names and limits depending on fFitFunc
405//
406Bool_t MHCalibrationChargeBlindPix::InitFit()
407{
408
409 //
410 // Get the fitting ranges
411 //
412 Axis_t rmin = fHGausHist.GetBinCenter(fHGausHist.GetXaxis()->GetFirst());
413 Axis_t rmax = fHGausHist.GetBinCenter(fHGausHist.GetXaxis()->GetLast());
414
415 if (rmin < 0.)
416 rmin = 0.;
417
418 //
419 // First guesses for the fit (should be as close to reality as possible,
420 // otherwise the fit goes gaga because of high number of dimensions ...
421 //
422 const Stat_t entries = fHGausHist.Integral("width");
423 const Double_t lambda_guess = 0.5;
424 //const Double_t maximum_bin = fHGausHist.GetBinCenter(fHGausHist.GetMaximumBin());
425 const Double_t norm = entries/TMath::Sqrt(TMath::TwoPi());
426
427 //
428 // Initialize the fit function
429 //
430 switch (fFitFunc)
431 {
432 case kEPoisson4:
433 fSinglePheFit = new TF1("SinglePheFit",&fPoissonKto4,rmin,rmax,6);
434 rmin += 6.5;
435 break;
436 case kEPoisson5:
437 fSinglePheFit = new TF1("SinglePheFit",&fPoissonKto5,rmin,rmax,6);
438 rmin = 0.;
439 break;
440 case kEPoisson6:
441 fSinglePheFit = new TF1("SinglePheFit",&fPoissonKto6,rmin,rmax,6);
442 break;
443 case kEPolya:
444 fSinglePheFit = new TF1("SinglePheFit",&fPolya,rmin,rmax,8);
445 break;
446 case kEMichele:
447 fSinglePheFit = new TF1("SinglePheFit",&fFitFuncMichele,rmin,rmax,9);
448 break;
449 default:
450 *fLog << warn << "WARNING: Could not find Fit Function for Blind Pixel " << endl;
451 return kFALSE;
452 break;
453 }
454
455 if (!fSinglePheFit)
456 {
457 *fLog << warn << dbginf << "WARNING: Could not create fit function for Single Phe fit" << endl;
458 return kFALSE;
459 }
460
461 //
462 // For the fits, we have to take special care since ROOT
463 // has stored the function pointer in a global list which
464 // lead to removing the object twice. We have to take out
465 // the following functions of the global list of functions
466 // as well:
467 //
468 gROOT->GetListOfFunctions()->Remove(fSinglePheFit);
469
470 const Double_t mu_0_guess = 13.5;
471 const Double_t si_0_guess = 2.5;
472 const Double_t mu_1_guess = 30.;
473 const Double_t si_1_guess = si_0_guess + si_0_guess;
474 // Michele
475 const Double_t lambda_1cat_guess = 1.00;
476 const Double_t lambda_1dyn_guess = lambda_1cat_guess/10.;
477 const Double_t mu_1cat_guess = 50.;
478 const Double_t mu_1dyn_guess = 17.;
479 const Double_t si_1cat_guess = si_0_guess + si_0_guess;
480 const Double_t si_1dyn_guess = si_0_guess + si_0_guess/2.;
481 // Polya
482 const Double_t excessPoisson_guess = 0.5;
483 const Double_t delta1_guess = 8.;
484 const Double_t delta2_guess = 5.;
485 const Double_t electronicAmp_guess = gkElectronicAmp;
486 const Double_t electronicAmp_limit = gkElectronicAmpErr;
487
488 //
489 // Initialize boundaries and start parameters
490 //
491 switch (fFitFunc)
492 {
493
494 case kEPoisson4:
495 fSinglePheFit->SetParNames( "#lambda", "#mu_{0}", "#mu_{1}", "#sigma_{0}", "#sigma_{1}","Area");
496 fSinglePheFit->SetParameters(lambda_guess,mu_0_guess,mu_1_guess,si_0_guess,si_1_guess,norm);
497 fSinglePheFit->SetParLimits(0,0.,2.);
498 fSinglePheFit->SetParLimits(1,10.,17.);
499 fSinglePheFit->SetParLimits(2,17.,50.);
500 fSinglePheFit->SetParLimits(3,1.,5.);
501 fSinglePheFit->SetParLimits(4,5.,30.);
502 fSinglePheFit->SetParLimits(5,norm-(0.5*norm),norm+(0.7*norm));
503 break;
504 case kEPoisson5:
505 case kEPoisson6:
506 fSinglePheFit->SetParNames("#lambda","#mu_{0}","#mu_{1}","#sigma_{0}","#sigma_{1}","Area");
507 fSinglePheFit->SetParameters(lambda_guess,mu_0_guess,800.,si_0_guess,500.,norm);
508 fSinglePheFit->SetParLimits(0,0.,2.);
509 fSinglePheFit->SetParLimits(1,0.,100.);
510 fSinglePheFit->SetParLimits(2,300.,1500.);
511 fSinglePheFit->SetParLimits(3,30.,250.);
512 fSinglePheFit->SetParLimits(4,100.,1000.);
513 fSinglePheFit->SetParLimits(5,norm/1.5,norm*1.5);
514 break;
515
516 case kEPolya:
517 fSinglePheFit->SetParameters(lambda_guess, excessPoisson_guess,
518 delta1_guess,delta2_guess,
519 electronicAmp_guess,
520 10.,
521 norm,
522 0.);
523 fSinglePheFit->SetParNames("#lambda","b_{tot}",
524 "#delta_{1}","#delta_{2}",
525 "amp_{e}","#sigma_{0}",
526 "Area", "#mu_{0}");
527 fSinglePheFit->SetParLimits(0,0.,1.);
528 fSinglePheFit->SetParLimits(1,0.,1.);
529 fSinglePheFit->SetParLimits(2,6.,12.);
530 fSinglePheFit->SetParLimits(3,3.,8.);
531 fSinglePheFit->SetParLimits(4,electronicAmp_guess-electronicAmp_limit,
532 electronicAmp_guess+electronicAmp_limit);
533 fSinglePheFit->SetParLimits(5,0.,40.);
534 fSinglePheFit->SetParLimits(6,norm-0.1,norm+0.1);
535 fSinglePheFit->SetParLimits(7,-10.,10.);
536 break;
537 case kEMichele:
538 fSinglePheFit->SetParNames("#lambda_{cat}","#lambda_{dyn}",
539 "#mu_{0}","#mu_{1cat}","#mu_{1dyn}",
540 "#sigma_{0}","#sigma_{1cat}","#sigma_{1dyn}",
541 "Area");
542 fSinglePheFit->SetParameters(lambda_1cat_guess, lambda_1dyn_guess,
543 mu_0_guess, mu_1cat_guess,mu_1dyn_guess,
544 si_0_guess, si_1cat_guess,si_1dyn_guess,
545 norm);
546 fSinglePheFit->SetParLimits(0,0.01,2.0);
547 fSinglePheFit->SetParLimits(1,0.,0.5);
548 fSinglePheFit->SetParLimits(2,10.,16.);
549 fSinglePheFit->SetParLimits(3,25.,50.);
550 fSinglePheFit->SetParLimits(4,16.,18.5);
551 fSinglePheFit->SetParLimits(5,1.,5.);
552 fSinglePheFit->SetParLimits(6,10.,50.);
553 fSinglePheFit->SetParLimits(7,5.,10.);
554 fSinglePheFit->SetParLimits(8,norm/2.,norm*2.5);
555 break;
556
557 default:
558 *fLog << warn << "WARNING: Could not find Fit Function for Blind Pixel " << endl;
559 return kFALSE;
560 break;
561 }
562
563 fSinglePheFit->SetRange(rmin,rmax);
564
565 return kTRUE;
566}
567
568// --------------------------------------------------------------------------
569//
570// - Retrieve the parameters depending on fFitFunc
571// - Retrieve probability, Chisquare and NDF
572//
573void MHCalibrationChargeBlindPix::ExitFit()
574{
575
576
577 //
578 // Finalize
579 //
580 switch (fFitFunc)
581 {
582
583 case kEPoisson4:
584 case kEPoisson5:
585 case kEPoisson6:
586 case kEPoisson7:
587 fLambda = fSinglePheFit->GetParameter(0);
588 fMu0 = fSinglePheFit->GetParameter(1);
589 fMu1 = fSinglePheFit->GetParameter(2);
590 fSigma0 = fSinglePheFit->GetParameter(3);
591 fSigma1 = fSinglePheFit->GetParameter(4);
592
593 fLambdaErr = fSinglePheFit->GetParError(0);
594 fMu0Err = fSinglePheFit->GetParError(1);
595 fMu1Err = fSinglePheFit->GetParError(2);
596 fSigma0Err = fSinglePheFit->GetParError(3);
597 fSigma1Err = fSinglePheFit->GetParError(4);
598 break;
599 case kEPolya:
600 fLambda = fSinglePheFit->GetParameter(0);
601 fMu0 = fSinglePheFit->GetParameter(7);
602 fMu1 = 0.;
603 fSigma0 = fSinglePheFit->GetParameter(5);
604 fSigma1 = 0.;
605
606 fLambdaErr = fSinglePheFit->GetParError(0);
607 fMu0Err = fSinglePheFit->GetParError(7);
608 fMu1Err = 0.;
609 fSigma0Err = fSinglePheFit->GetParError(5);
610 fSigma1Err = 0.;
611 case kEMichele:
612 fLambda = fSinglePheFit->GetParameter(0);
613 fMu0 = fSinglePheFit->GetParameter(2);
614 fMu1 = fSinglePheFit->GetParameter(3);
615 fSigma0 = fSinglePheFit->GetParameter(5);
616 fSigma1 = fSinglePheFit->GetParameter(6);
617
618 fLambdaErr = fSinglePheFit->GetParError(0);
619 fMu0Err = fSinglePheFit->GetParError(2);
620 fMu1Err = fSinglePheFit->GetParError(3);
621 fSigma0Err = fSinglePheFit->GetParError(5);
622 fSigma1Err = fSinglePheFit->GetParError(6);
623 break;
624 default:
625 break;
626 }
627
628 fProb = fSinglePheFit->GetProb();
629 fChisquare = fSinglePheFit->GetChisquare();
630 fNDF = fSinglePheFit->GetNDF();
631
632 *fLog << all << "Results of the Blind Pixel Fit: " << endl;
633 *fLog << all << "Chisquare: " << fChisquare << endl;
634 *fLog << all << "DoF: " << fNDF << endl;
635 *fLog << all << "Probability: " << fProb << endl;
636
637}
638
639// --------------------------------------------------------------------------
640//
641// - Executes InitFit()
642// - Fits the fHGausHist with fSinglePheFit
643// - Executes ExitFit()
644//
645// The fit result is accepted under condition:
646// 1) The results are not nan's
647// 2) The NDF is not smaller than fNDFLimit (5)
648// 3) The Probability is greater than fProbLimit (default 0.001 == 99.9%)
649// 4) at least fNumSinglePheLimit events are in the single Photo-electron peak
650//
651Bool_t MHCalibrationChargeBlindPix::FitSinglePhe(Option_t *opt)
652{
653
654 if (!InitFit())
655 return kFALSE;
656
657 fHGausHist.Fit(fSinglePheFit,opt);
658
659 ExitFit();
660
661 //
662 // The fit result is accepted under condition:
663 // 1) The results are not nan's
664 // 2) The NDF is not smaller than fNDFLimit (5)
665 // 3) The Probability is greater than fProbLimit (default 0.001 == 99.9%)
666 // 4) at least fNumSinglePheLimit events are in the single Photo-electron peak
667 //
668 // !Finitite means either infinite or not-a-number
669 if ( !TMath::Finite(fLambda)
670 || !TMath::Finite(fLambdaErr)
671 || !TMath::Finite(fProb)
672 || !TMath::Finite(fMu0)
673 || !TMath::Finite(fMu0Err)
674 || !TMath::Finite(fMu1)
675 || !TMath::Finite(fMu1Err)
676 || !TMath::Finite(fSigma0)
677 || !TMath::Finite(fSigma0Err)
678 || !TMath::Finite(fSigma1)
679 || !TMath::Finite(fSigma1Err)
680 || fNDF < GetNDFLimit()
681 || fProb < GetProbLimit() )
682 return kFALSE;
683
684 const Stat_t entries = fHGausHist.Integral("width");
685 const Float_t numSinglePhe = TMath::Exp(-1.0*fLambda)*fLambda*entries;
686
687 if (numSinglePhe < fNumSinglePheLimit)
688 {
689 *fLog << warn << "WARNING - Statistics is too low: Only " << numSinglePhe
690 << " in the Single Photo-Electron peak " << endl;
691 return kFALSE;
692 }
693 else
694 *fLog << all << numSinglePhe << " in Single Photo-Electron peak " << endl;
695
696 SetSinglePheFitOK();
697 return kTRUE;
698}
699
700// --------------------------------------------------------------------------
701//
702// - Retrieves limits for the fit
703// - Fits the fHGausHist with Gauss
704// - Retrieves the results to fLambdaCheck and fLambdaCheckErr
705// - Sets a flag IsPedestalFitOK()
706//
707void MHCalibrationChargeBlindPix::FitPedestal (Option_t *opt)
708{
709
710 // Perform the cross-check fitting only the pedestal:
711 const Axis_t rmin = 0.;
712 // const Axis_t rmax = fHGausHist.GetBinCenter(fHGausHist.GetMaximumBin());
713 const Axis_t rmax = fSinglePheCut;
714
715 FitGaus(opt, rmin, rmax);
716
717 const Stat_t entries = fHGausHist.Integral("width");
718 const Double_t pedarea = fFGausFit->Integral(0.,fSinglePheCut);
719
720 fLambdaCheck = TMath::Log(entries/pedarea);
721 // estimate the error by the error of the obtained area from the Gauss-function:
722 fLambdaCheckErr = fFGausFit->GetParError(0)/fFGausFit->GetParameter(0);
723
724 SetPedestalFitOK(IsGausFitOK());
725 return;
726}
727
728
729// -------------------------------------------------------------------------
730//
731// Draw a legend with the fit results
732//
733void MHCalibrationChargeBlindPix::DrawLegend(Option_t *opt)
734{
735
736 TString option(opt);
737
738 if (!fFitLegend)
739 {
740 fFitLegend = new TPaveText(0.05,0.05,0.95,0.95);
741 fFitLegend->SetLabel(Form("%s%s", "Results of the single PhE Fit (",
742 (fFitFunc == kEPoisson4) ? "Poisson(k=4))" :
743 (fFitFunc == kEPoisson5) ? "Poisson(k=5))" :
744 (fFitFunc == kEPoisson6) ? "Poisson(k=6))" :
745 (fFitFunc == kEPolya ) ? "Polya(k=4))" :
746 (fFitFunc == kEMichele ) ? "Michele)"
747 : " none )" ));
748 fFitLegend->SetTextSize(0.05);
749 }
750 else
751 fFitLegend->Clear();
752
753 const TString line1 =
754 Form("Mean: #lambda = %2.2f #pm %2.2f",fLambda,fLambdaErr);
755 TText *t1 = fFitLegend->AddText(line1.Data());
756 t1->SetBit(kCanDelete);
757
758 const TString line6 =
759 Form("Mean #lambda_{check} = %2.2f #pm %2.2f",fLambdaCheck,fLambdaCheckErr);
760 TText *t2 = fFitLegend->AddText(line6.Data());
761 t2->SetBit(kCanDelete);
762
763 if (option.Contains("datacheck"))
764 {
765 if (fLambda + 3.*fLambdaErr < fLambdaCheck - 3.*fLambdaCheckErr
766 ||
767 fLambda - 3.*fLambdaErr > fLambdaCheck + 3.*fLambdaCheckErr )
768 {
769 TText *t = fFitLegend->AddText("#lambda and #lambda_{check} more than 3#sigma apart!");
770 t->SetBit(kCanDelete);
771 }
772 }
773 else
774 {
775
776 const TString line2 =
777 Form("Pedestal: #mu_{0} = %2.2f #pm %2.2f",fMu0,fMu0Err);
778 TText *t3 = fFitLegend->AddText(line2.Data());
779 t3->SetBit(kCanDelete);
780
781 const TString line3 =
782 Form("Width Pedestal: #sigma_{0} = %2.2f #pm %2.2f",fSigma0,fSigma0Err);
783 TText *t4 = fFitLegend->AddText(line3.Data());
784 t4->SetBit(kCanDelete);
785
786 const TString line4 =
787 Form("1^{st} Phe-peak: #mu_{1} = %2.2f #pm %2.2f",fMu1,fMu1Err);
788 TText *t5 = fFitLegend->AddText(line4.Data());
789 t5->SetBit(kCanDelete);
790
791 const TString line5 =
792 Form("Width 1^{st} Phe-peak: #sigma_{1} = %2.2f #pm %2.2f",fSigma1,fSigma1Err);
793 TText *t6 = fFitLegend->AddText(line5.Data());
794 t6->SetBit(kCanDelete);
795 }
796
797 const TString line7 =
798 Form("#chi^{2} / N_{dof}: %4.2f / %3i",fChisquare,fNDF);
799 TText *t7 = fFitLegend->AddText(line7.Data());
800 t7->SetBit(kCanDelete);
801
802 const TString line8 =
803 Form("Probability: %6.4f ",fProb);
804 TText *t8 = fFitLegend->AddText(line8.Data());
805 t8->SetBit(kCanDelete);
806
807 if (IsSinglePheFitOK())
808 {
809 TText *t = fFitLegend->AddText("Result of the Fit: OK");
810 t->SetBit(kCanDelete);
811 }
812 else
813 {
814 TText *t = fFitLegend->AddText("Result of the Fit: NOT OK");
815 t->SetBit(kCanDelete);
816 }
817
818 fFitLegend->SetFillColor(IsSinglePheFitOK() ? 80 : 2);
819 fFitLegend->Draw();
820
821 return;
822}
823
824
825// -------------------------------------------------------------------------
826//
827// Draw the histogram
828//
829// The following options can be chosen:
830//
831// "": displays the fHGausHist, the fits, the legend and fASinglePheFADCSlices and fAPedestalFADCSlices
832// "all": executes additionally MHGausEvents::Draw(), with option "fourierevents"
833// "datacheck" display the fHGausHist, the fits and the legend
834//
835void MHCalibrationChargeBlindPix::Draw(Option_t *opt)
836{
837
838 TString option(opt);
839 option.ToLower();
840
841 Int_t win = 1;
842
843 TVirtualPad *oldpad = gPad ? gPad : MH::MakeDefCanvas(this,900, 600);
844 TVirtualPad *pad = NULL;
845
846 if (option.Contains("all"))
847 {
848 option.ReplaceAll("all","");
849 oldpad->Divide(2,1);
850 win = 2;
851 oldpad->cd(1);
852 TVirtualPad *newpad = gPad;
853 pad = newpad;
854 pad->Divide(2,2);
855 pad->cd(1);
856 }
857 else if (option.Contains("datacheck"))
858 {
859 pad = oldpad;
860 pad->Divide(1,2);
861 pad->cd(1);
862 fHGausHist.SetStats(0);
863 }
864 else
865 {
866 pad = oldpad;
867 pad->Divide(2,2);
868 pad->cd(1);
869 }
870
871 if (!IsEmpty() && !IsOnlyOverflow() && !IsOnlyUnderflow())
872 gPad->SetLogy();
873
874 gPad->SetTicks();
875
876 fHGausHist.Draw();
877 if (fFGausFit )
878 {
879 fFGausFit->SetLineColor(kBlue);
880 fFGausFit->Draw("same");
881 if (!option.Contains("datacheck"))
882 {
883 TLine *line = new TLine(fSinglePheCut, 0., fSinglePheCut, 10.);
884 line->SetBit(kCanDelete);
885 line->SetLineColor(kBlue);
886 line->SetLineWidth(3);
887 line->DrawLine(fSinglePheCut, 0., fSinglePheCut, 2.);
888 }
889 }
890
891 if (fSinglePheFit)
892 {
893 fSinglePheFit->SetFillStyle(0);
894 fSinglePheFit->SetLineWidth(3);
895 fSinglePheFit->SetLineColor(IsSinglePheFitOK() ? kGreen : kRed);
896 fSinglePheFit->Draw("same");
897 }
898
899 pad->cd(2);
900 DrawLegend(option.Data());
901
902 if (option.Contains("datacheck"))
903 return;
904
905 pad->cd(3);
906
907 if (fASinglePheFADCSlices.GetNrows()!=1)
908 {
909 if (fHSinglePheFADCSlices)
910 delete fHSinglePheFADCSlices;
911
912 fHSinglePheFADCSlices = new TH1F(fASinglePheFADCSlices);
913 fHSinglePheFADCSlices->SetName("SinglePheFADCSlices");
914 fHSinglePheFADCSlices->SetTitle(Form("%s%4.1f","Assumed Single Phe FADC Slices, Sum > ",fSinglePheCut));
915 fHSinglePheFADCSlices->SetXTitle("FADC slice number");
916 fHSinglePheFADCSlices->SetYTitle("FADC counts");
917 const Int_t nbins = fHSinglePheFADCSlices->GetNbinsX();
918 TH2D *nulls = new TH2D("Nulls",fHSinglePheFADCSlices->GetTitle(),nbins,0.,
919 fHSinglePheFADCSlices->GetXaxis()->GetBinCenter(nbins),
920 100,0.,50.);
921 nulls->SetDirectory(NULL);
922 nulls->SetBit(kCanDelete);
923 nulls->GetXaxis()->SetTitle(fHSinglePheFADCSlices->GetXaxis()->GetTitle());
924 nulls->GetYaxis()->SetTitle(fHSinglePheFADCSlices->GetYaxis()->GetTitle());
925 nulls->GetXaxis()->CenterTitle();
926 nulls->GetYaxis()->CenterTitle();
927 nulls->SetStats(0);
928 nulls->Draw();
929 fHSinglePheFADCSlices->Draw("same");
930 }
931
932 pad->cd(4);
933 if (fAPedestalFADCSlices.GetNrows()!=1)
934 {
935
936 if (fHPedestalFADCSlices)
937 delete fHPedestalFADCSlices;
938
939 fHPedestalFADCSlices = new TH1F(fAPedestalFADCSlices);
940 fHPedestalFADCSlices->SetName("PedestalFADCSlices");
941 fHPedestalFADCSlices->SetTitle(Form("%s%4.1f","Pedestal FADC Slices, Sum < ",fSinglePheCut));
942 fHPedestalFADCSlices->SetXTitle("FADC slice number");
943 fHPedestalFADCSlices->SetYTitle("FADC counts");
944 const Int_t nbins = fHPedestalFADCSlices->GetNbinsX();
945 TH2D *nullp = new TH2D("Nullp",fHPedestalFADCSlices->GetTitle(),nbins,0.,
946 fHPedestalFADCSlices->GetXaxis()->GetBinCenter(nbins),
947 100,0.,50.);
948 nullp->SetDirectory(NULL);
949 nullp->SetBit(kCanDelete);
950 nullp->GetXaxis()->SetTitle(fHPedestalFADCSlices->GetXaxis()->GetTitle());
951 nullp->GetYaxis()->SetTitle(fHPedestalFADCSlices->GetYaxis()->GetTitle());
952 nullp->GetXaxis()->CenterTitle();
953 nullp->GetYaxis()->CenterTitle();
954 nullp->SetStats(0);
955 nullp->Draw();
956 fHPedestalFADCSlices->Draw("same");
957 }
958
959 if (win < 2)
960 return;
961
962 oldpad->cd(2);
963 MHCalibrationPix::Draw("fourierevents");
964}
965
966
967
968
969
970
971
972
973
974
975
Note: See TracBrowser for help on using the repository browser.