forked from h2gglobe/h2gglobe
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDiPhoEfficiencySmearer.cc
164 lines (135 loc) · 6.43 KB
/
DiPhoEfficiencySmearer.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
#include "DiPhoEfficiencySmearer.h"
#include "PhotonReducedInfo.h"
#include "TRandom3.h"
#include <assert.h>
DiPhoEfficiencySmearer::DiPhoEfficiencySmearer(const diPhoEfficiencySmearingParameters& par) : myParameters_(par)
{
rgen_ = new TRandom3(0);
name_="DiPhoEfficiencySmearer_"+ par.categoryType + "_" + par.parameterSetName;
passFailWeights_=false;
doVtxEff_ =false;
doMvaIdEff_ =false;
}
DiPhoEfficiencySmearer::~DiPhoEfficiencySmearer()
{
delete rgen_;
}
bool DiPhoEfficiencySmearer::smearDiPhoton( TLorentzVector & p4, TVector3 & selVtx, float & weight, const int & category,
const int & genMassPoint, const TVector3 & trueVtx, float & idMVA1, float & idMVA2, float syst_shift) const
{
if (doMvaIdEff_){
float frac_idvary = 0.025;
idMVA1 += syst_shift*frac_idvary;
idMVA2 += syst_shift*frac_idvary;
} else {
std::string cat=Form("cat%d", category);
if (cat == "")
{
std::cout << effName_ <<" No category has been found associated with this diphoton. Giving Up" << std::endl;
return false;
}
/////////////////////// changing weigh of photon according to efficiencies ///////////////////////////////////////////
assert( ! smearing_eff_graph_.empty() );
if( doVtxEff_ ) {
if( (selVtx - trueVtx).Mag() < 1. ) {
cat += "_pass";
}
else {
cat += "_fail";
syst_shift *=-1; // shift for pass and fail need to be in opposite directions
}
}
weight = getWeight( p4.Pt(), cat, syst_shift );
weight = weight<0 ? 0 : weight;
}
return true;
}
bool DiPhoEfficiencySmearer::init()
{
// if map is not empty, yuo're initilized and happy..
if( !smearing_eff_graph_.empty() ){
std:cout << "initialization of DI-photon efficiency smearer " << effName_ << " already done; proceed with usage. " << std::endl;
return true;
}
if( doVtxEff_ ) { passFailWeights_ = true; }
else { passFailWeights_ = false;}
//otherwise, get smearing functions from file and set up map
std::cout << "\n>>>initializing one efficiency for DI-photon re-weighting: " << effName_ << std::endl;
// do basic sanity checks first
if( effName_.empty()){
std::cout << effName_ << " - you're initializing reweighting for efficiency but effName_ is empty" << std::endl; assert(false); }
if( myParameters_.efficiency_file.empty()){
std::cout << effName_ << "- you're initializing reweighting for efficiency: " << effName_ << " but input file with TGraphErrors is not specified; doing nothing. " << std::endl; assert(false); }
theDiPhoEfficiencyFile_ = TFile::Open(myParameters_.efficiency_file.c_str());
// initialize formulas for the di-photon categories;
for( int ii=0; ii<myParameters_.n_categories; ++ii ) {
std::string cat = Form("cat%d", ii);
//std::cout << "GF for efficiency: " << effName_ << " passFailWeights_: " << passFailWeights_ << " doVtxEff_: " << doVtxEff_ << std::endl;
if( passFailWeights_ ) {
// std::cout << "GF cloning: " << ( (effName_+"_"+cat+"_pass").c_str() ) << std::endl;
smearing_eff_graph_[cat+"_pass"]=(TGraphAsymmErrors*) theDiPhoEfficiencyFile_->Get((effName_+"_"+cat+"_pass").c_str())->Clone();
smearing_eff_graph_[cat+"_fail"]=(TGraphAsymmErrors*) theDiPhoEfficiencyFile_->Get((effName_+"_"+cat+"_fail").c_str())->Clone();
std::cerr << "DiPhoEfficiencySmearerc " << cat+"_pass" << std::endl;
} else {
// std::cout << "GF cloning: " << ( (effName_+"_"+cat).c_str() ) << std::endl;
smearing_eff_graph_[cat]=(TGraphAsymmErrors*) theDiPhoEfficiencyFile_->Get((effName_+"_"+cat).c_str())->Clone();
}
}
theDiPhoEfficiencyFile_->Close();
return true;
}
double DiPhoEfficiencySmearer::getWeight(double pt, std::string theCategory, float syst_shift) const
{
static int nwarnings = 10;
std::map<std::string,TGraphAsymmErrors*>::const_iterator theIter = smearing_eff_graph_.find(theCategory);
if( theIter != smearing_eff_graph_.end() ) {
// determine the pair of bins between which you interpolate
int numPoints = ( theIter->second )->GetN();
double x, y, xPrevious;
int myBin = -1;
xPrevious =-1e9;
for (int bin=0; bin<numPoints; bin++ ){
( theIter->second )->GetPoint(bin, x, y);
assert( xPrevious < x ) ; // points in TGraphAsymmErrors must be in increasing order
xPrevious=x;
if(pt > x) {
myBin = bin; }
else break;
}
int binLow, binHigh; bool atBoundary(false);
if (myBin == -1) {binHigh = 0; binLow=0; atBoundary=true;}
else if (myBin == (numPoints-1)) {binHigh = numPoints-1; binLow=numPoints-1; atBoundary=true;}
else {binLow=myBin; binHigh=myBin+1;}
// get hold of efficiency ratio and error at either points
// low-high refer to the points ; up-down refers to the errors
double xLow, yLow; double xHigh, yHigh;
( theIter->second )->GetPoint(binLow, xLow, yLow);
( theIter->second )->GetPoint(binHigh, xHigh, yHigh);
double errLowYup = ( theIter->second )->GetErrorYhigh(binLow);
double errLowYdown = ( theIter->second )->GetErrorYlow(binLow);
double errHighYup = ( theIter->second )->GetErrorYhigh(binHigh);
double errHighYdown = ( theIter->second )->GetErrorYlow(binHigh);
double theErrorLow, theErrorHigh;
if(syst_shift>0) {theErrorLow = errLowYup; theErrorHigh = errHighYup;}
else {theErrorLow = errLowYdown; theErrorHigh = errHighYdown;}
double theWeight, theError;
// if you're NOT at the boundaris of TGraphAsymmErrors, linearly interpolate values and errors
if(!atBoundary) {
theWeight = yLow + (yHigh-yLow) / (xHigh-xLow) * (pt-xLow);
theError = theErrorLow + (theErrorHigh-theErrorLow) / (xHigh-xLow) * (pt-xLow);
} else {
// if instead you ARE at the boundaris of TGraphAsymmErrors, collapse on first or last of its points
if(myBin == (numPoints-1)) {
theWeight = yHigh; theError = theErrorHigh; }
else if (myBin == -1) {
theWeight = yLow; theError = theErrorLow; }
else { std::cout << effName_ << " ** you claim to be at boundaries of TGraphAsymmErrors but your not! This is a problem " << std::endl;}
}
float ret = theWeight + theError*syst_shift;
return ret;
}
else {
std::cout << effName_ << "- category asked: " << theCategory << " was not found - which is a problem. Returning weight 1. " << std::endl;
return 1.;
}
}