#include "TUnfoldBinning.h"
#include <TVectorD.h>
#include <TAxis.h>
#include <TString.h>
#include <iostream>
#include <fstream>
#include <TMath.h>
#include <TF1.h>
#include <TH1D.h>
#include <TH2D.h>
#include <TH3D.h>
#include <TList.h>
#include <TIterator.h>
using namespace std;
ClassImp(TUnfoldBinning)
void TUnfoldBinning::Initialize(Int_t nBins)
{
parentNode=0;
childNode=0;
nextNode=0;
prevNode=0;
fAxisList=new TObjArray();
fAxisLabelList=new TObjArray();
fAxisList->SetOwner();
fAxisLabelList->SetOwner();
fHasUnderflow=0;
fHasOverflow=0;
fDistributionSize=nBins;
fBinFactorFunction=0;
fBinFactorConstant=1.0;
}
Int_t TUnfoldBinning::UpdateFirstLastBin(Bool_t startWithRootNode)
{
if(startWithRootNode) {
return GetRootNode()->UpdateFirstLastBin(kFALSE);
}
if(GetPrevNode()) {
fFirstBin=GetPrevNode()->GetEndBin();
} else if(GetParentNode()) {
fFirstBin=GetParentNode()->GetStartBin()+
GetParentNode()->GetDistributionNumberOfBins();
} else {
fFirstBin=1;
if((!GetChildNode())&&(GetDistributionDimension()==1)&&
(fHasUnderflow==1)) {
fFirstBin=0;
}
}
fLastBin=fFirstBin+fDistributionSize;
for(TUnfoldBinning *node=childNode;node;node=node->nextNode) {
fLastBin=node->UpdateFirstLastBin(kFALSE);
}
return fLastBin;
}
TUnfoldBinning::TUnfoldBinning
(const char *name,Int_t nBins,const char *binNames)
: TNamed(name ? name : "",name ? name : "")
{
Initialize(nBins);
if(binNames) {
TString nameString(binNames);
delete fAxisLabelList;
fAxisLabelList=nameString.Tokenize(";");
}
UpdateFirstLastBin();
}
TUnfoldBinning::TUnfoldBinning
(const TAxis &axis,Int_t includeUnderflow,Int_t includeOverflow)
: TNamed(axis.GetName(),axis.GetTitle())
{
Initialize(0);
AddAxis(axis,includeUnderflow,includeOverflow);
UpdateFirstLastBin();
}
TUnfoldBinning::~TUnfoldBinning(void)
{
while(childNode) delete childNode;
if(GetParentNode() && (GetParentNode()->GetChildNode()==this)) {
parentNode->childNode=nextNode;
}
if(GetPrevNode()) prevNode->nextNode=nextNode;
if(GetNextNode()) nextNode->prevNode=prevNode;
delete fAxisList;
delete fAxisLabelList;
}
TUnfoldBinning *TUnfoldBinning::AddBinning
(const char *name,Int_t nBins,const char *binNames)
{
return AddBinning(new TUnfoldBinning(name,nBins,binNames));
}
TUnfoldBinning *TUnfoldBinning::AddBinning(TUnfoldBinning *binning)
{
TUnfoldBinning *r=0;
if(binning->GetParentNode()) {
Error("binning \"%s\" already has parent \"%s\", can not be added to %s",
(char *)binning->GetName(),
(char *)binning->GetParentNode()->GetName(),
(char *)GetName());
} else if(binning->GetPrevNode()) {
Error("binning \"%s\" has previous node \"%s\", can not be added to %s",
(char *)binning->GetName(),
(char *)binning->GetPrevNode()->GetName(),
(char *)GetName());
} else if(binning->GetNextNode()) {
Error("binning \"%s\" has next node \"%s\", can not be added to %s",
(char *)binning->GetName(),
(char *)binning->GetNextNode()->GetName(),
(char *)GetName());
} else {
r=binning;
binning->parentNode=this;
if(childNode) {
TUnfoldBinning *child=childNode;
while(child->nextNode) {
child=child->nextNode;
}
child->nextNode=r;
r->prevNode=child;
} else {
childNode=r;
}
UpdateFirstLastBin();
r=binning;
}
return r;
}
Bool_t TUnfoldBinning::AddAxis
(const char *name,Int_t nBin,Double_t xMin,Double_t xMax,
Bool_t hasUnderflow,Bool_t hasOverflow)
{
Bool_t r=kFALSE;
if(nBin<=0) {
Fatal("AddAxis","number of bins %d is not positive",
nBin);
} else if((!TMath::Finite(xMin))||(!TMath::Finite(xMax))||
(xMin>=xMax)) {
Fatal("AddAxis","xmin=%f required to be smaller than xmax=%f",
xMin,xMax);
} else {
Double_t *binBorders=new Double_t[nBin+1];
Double_t x=xMin;
Double_t dx=(xMax-xMin)/nBin;
for(Int_t i=0;i<=nBin;i++) {
binBorders[i]=x+i*dx;
}
r=AddAxis(name,nBin,binBorders,hasUnderflow,hasOverflow);
delete [] binBorders;
}
return r;
}
Bool_t TUnfoldBinning::AddAxis
(const TAxis &axis,Bool_t hasUnderflow,Bool_t hasOverflow)
{
Int_t nBin=axis.GetNbins();
Double_t *binBorders=new Double_t[nBin+1];
for(Int_t i=0;i<nBin;i++) {
binBorders[i]=axis.GetBinLowEdge(i+1);
}
binBorders[nBin]=axis.GetBinUpEdge(nBin);
Bool_t r=AddAxis(axis.GetTitle(),nBin,binBorders,hasUnderflow,hasOverflow);
delete [] binBorders;
return r;
}
Bool_t TUnfoldBinning::AddAxis
(const char *name,Int_t nBin,const Double_t *binBorders,
Bool_t hasUnderflow,Bool_t hasOverflow)
{
Bool_t r=kFALSE;
if(HasUnconnectedBins()) {
Fatal("AddAxis","node already has %d bins without axis",
GetDistributionNumberOfBins());
} else if(nBin<=0) {
Fatal("AddAxis","number of bins %d is not positive",
nBin);
} else {
TVectorD *bins=new TVectorD(nBin+1);
r=kTRUE;
for(Int_t i=0;i<=nBin;i++) {
(*bins)(i)=binBorders[i];
if(!TMath::Finite((*bins)(i))) {
Fatal("AddAxis","bin border %d is not finite",i);
r=kFALSE;
} else if((i>0)&&((*bins)(i)<=(*bins)(i-1))) {
Fatal("AddAxis","bins not in order x[%d]=%f <= %f=x[%d]",
i,(*bins)(i),(*bins)(i-1),i-1);
r=kFALSE;
}
}
if(r) {
Int_t axis=fAxisList->GetEntriesFast();
Int_t bitMask=1<<axis;
Int_t nBinUO=nBin;
if(hasUnderflow) {
fHasUnderflow |= bitMask;
nBinUO++;
} else {
fHasUnderflow &= ~bitMask;
}
if(hasOverflow) {
fHasOverflow |= bitMask;
nBinUO++;
} else {
fHasOverflow &= ~bitMask;
}
fAxisList->AddLast(bins);
fAxisLabelList->AddLast(new TObjString(name));
if(!fDistributionSize) fDistributionSize=1;
fDistributionSize *= nBinUO;
UpdateFirstLastBin();
}
}
return r;
}
void TUnfoldBinning::PrintStream(ostream &out,Int_t indent) const
{
for(Int_t i=0;i<indent;i++) out<<" ";
out<<"TUnfoldBinning \""<<GetName()<<"\" has ";
Int_t nBin=GetEndBin()-GetStartBin();
if(nBin==1) {
out<<"1 bin";
} else {
out<<nBin<<" bins";
}
out<<" ["
<<GetStartBin()<<","<<GetEndBin()<<"] nTH1x="
<<GetTH1xNumberOfBins()
<<"\n";
if(GetDistributionNumberOfBins()) {
for(Int_t i=0;i<indent;i++) out<<" ";
out<<" distribution: "<<GetDistributionNumberOfBins()<<" bins\n";
if(fAxisList->GetEntriesFast()) {
for(Int_t axis=0;axis<GetDistributionDimension();axis++) {
for(Int_t i=0;i<indent;i++) out<<" ";
out<<" \""
<<GetDistributionAxisLabel(axis)
<<"\" nbin="<<GetDistributionBinning(axis)->GetNrows()-1;
if(fHasUnderflow & (1<<axis)) out<<" plus underflow";
if(fHasOverflow & (1<<axis)) out<<" plus overflow";
out<<"\n";
}
} else {
for(Int_t i=0;i<indent;i++) out<<" ";
out<<" no axis\n";
for(Int_t i=0;i<indent;i++) out<<" ";
out<<" names: ";
for(Int_t ibin=0;(ibin<GetDistributionNumberOfBins())&&
(ibin<fAxisLabelList->GetEntriesFast());ibin++) {
if(ibin) out<<";";
if(GetDistributionAxisLabel(ibin)) {
out<<GetDistributionAxisLabel(ibin);
}
}
out<<"\n";
}
}
TUnfoldBinning const *child=GetChildNode();
if(child) {
while(child) {
child->PrintStream(out,indent+1);
child=child->GetNextNode();
}
}
}
TUnfoldBinning const *TUnfoldBinning::FindNode(char const *name) const
{
TUnfoldBinning const *r=0;
if((!name)||(!TString(GetName()).CompareTo(name))) {
r=this;
}
for(TUnfoldBinning const *child=GetChildNode();
(!r) && child;child=child->GetNextNode()) {
r=child->FindNode(name);
}
return r;
}
TUnfoldBinning *TUnfoldBinning::GetRootNode(void)
{
TUnfoldBinning *node=this;
while(node->GetParentNode()) node=node->parentNode;
return node;
}
TUnfoldBinning const *TUnfoldBinning::GetRootNode(void) const
{
TUnfoldBinning const *node=this;
while(node->GetParentNode()) node=node->GetParentNode();
return node;
}
TString TUnfoldBinning::BuildHistogramTitle
(const char *histogramName,const char *histogramTitle,Int_t const *axisList)
const
{
TString r;
if(histogramTitle) {
r=histogramTitle;
} else {
r=histogramName;
Int_t iEnd;
for(iEnd=2;iEnd>0;iEnd--) {
if(axisList[iEnd]>=0) break;
}
for(Int_t i=0;i<=iEnd;i++) {
r += ";";
if(axisList[i]<0) {
r += GetName();
} else {
r += GetNonemptyNode()->GetDistributionAxisLabel(axisList[i]);
}
}
}
return r;
}
TString TUnfoldBinning::BuildHistogramTitle2D
(const char *histogramName,const char *histogramTitle,
Int_t xAxis,const TUnfoldBinning *yAxisBinning,Int_t yAxis) const
{
TString r;
if(histogramTitle) {
r=histogramTitle;
} else {
r=histogramName;
r += ";";
if(xAxis==-1) {
r += GetName();
} else if(xAxis>=0) {
r += GetNonemptyNode()->GetDistributionAxisLabel(xAxis);
}
r+= ";";
if(yAxis==-1) {
r += yAxisBinning->GetName();
} else if(yAxis>=0) {
r += yAxisBinning->GetNonemptyNode()->GetDistributionAxisLabel(yAxis);
}
}
return r;
}
Int_t TUnfoldBinning::GetTH1xNumberOfBins
(Bool_t originalAxisBinning,const char *axisSteering) const
{
Int_t axisBins[3],axisList[3];
GetTHxxBinning(originalAxisBinning ? 1 : 0,axisBins,axisList,
axisSteering);
return axisBins[0];
}
TH1 *TUnfoldBinning::CreateHistogram
(const char *histogramName,Bool_t originalAxisBinning,Int_t **binMap,
const char *histogramTitle,const char *axisSteering) const
{
Int_t nBin[3],axisList[3];
Int_t nDim=GetTHxxBinning(originalAxisBinning ? 3 : 0,nBin,axisList,
axisSteering);
const TUnfoldBinning *neNode=GetNonemptyNode();
TString title=BuildHistogramTitle(histogramName,histogramTitle,axisList);
TH1 *r=0;
if(nDim>0) {
const TVectorD *axisBinsX=
neNode->GetDistributionBinning(axisList[0]);
if(nDim>1) {
const TVectorD *axisBinsY=
neNode->GetDistributionBinning(axisList[1]);
if(nDim>2) {
const TVectorD *axisBinsZ=
neNode->GetDistributionBinning(axisList[2]);
r=new TH3D(histogramName,title,
nBin[0],axisBinsX->GetMatrixArray(),
nBin[1],axisBinsY->GetMatrixArray(),
nBin[2],axisBinsZ->GetMatrixArray());
} else {
r=new TH2D(histogramName,title,
nBin[0],axisBinsX->GetMatrixArray(),
nBin[1],axisBinsY->GetMatrixArray());
}
} else {
r=new TH1D(histogramName,title,nBin[0],axisBinsX->GetMatrixArray());
}
} else {
if(originalAxisBinning) {
Warning("CreateHistogram",
"Original binning can not be represented as THxx");
}
r=new TH1D(histogramName,title,nBin[0],0.5,nBin[0]+0.5);
nDim=0;
}
if(binMap) {
*binMap=CreateBinMap(r,nDim,axisList,axisSteering);
}
return r;
}
TH2D *TUnfoldBinning::CreateErrorMatrixHistogram
(const char *histogramName,Bool_t originalAxisBinning,Int_t **binMap,
const char *histogramTitle,const char *axisSteering) const
{
Int_t nBin[3],axisList[3];
Int_t nDim=GetTHxxBinning(originalAxisBinning ? 1 : 0,nBin,axisList,
axisSteering);
TString title=BuildHistogramTitle(histogramName,histogramTitle,axisList);
TH2D *r=0;
if(nDim==1) {
const TVectorD *axisBinsX=(TVectorD const *)
GetNonemptyNode()->fAxisList->At(axisList[0]);
r=new TH2D(histogramName,title,nBin[0],axisBinsX->GetMatrixArray(),
nBin[0],axisBinsX->GetMatrixArray());
} else {
if(originalAxisBinning) {
Info("CreateErrorMatrixHistogram",
"Original binning can not be represented on one axis");
}
r=new TH2D(histogramName,title,nBin[0],0.5,nBin[0]+0.5,
nBin[0],0.5,nBin[0]+0.5);
nDim=0;
}
if(binMap) {
*binMap=CreateBinMap(r,nDim,axisList,axisSteering);
}
return r;
}
TH2D *TUnfoldBinning::CreateHistogramOfMigrations
(TUnfoldBinning const *xAxis,TUnfoldBinning const *yAxis,
char const *histogramName,Bool_t originalXAxisBinning,
Bool_t originalYAxisBinning,char const *histogramTitle)
{
Int_t nBinX[3],axisListX[3];
Int_t nDimX=
xAxis->GetTHxxBinning(originalXAxisBinning ? 1 : 0,nBinX,axisListX,0);
Int_t nBinY[3],axisListY[3];
Int_t nDimY=
yAxis->GetTHxxBinning(originalYAxisBinning ? 1 : 0,nBinY,axisListY,0);
TString title=xAxis->BuildHistogramTitle2D
(histogramName,histogramTitle,axisListX[0],yAxis,axisListY[0]);
if(nDimX==1) {
const TVectorD *axisBinsX=(TVectorD const *)
xAxis->fAxisList->At(axisListX[0]);
if(nDimY==1) {
const TVectorD *axisBinsY=(TVectorD const *)
yAxis->fAxisList->At(axisListY[0]);
return new TH2D(histogramName,title,
nBinX[0],axisBinsX->GetMatrixArray(),
nBinY[0],axisBinsY->GetMatrixArray());
} else {
return new TH2D(histogramName,title,
nBinX[0],axisBinsX->GetMatrixArray(),
nBinY[0],0.5,0.5+nBinY[0]);
}
} else {
if(nDimY==1) {
const TVectorD *axisBinsY=(TVectorD const *)
yAxis->fAxisList->At(axisListY[0]);
return new TH2D(histogramName,title,
nBinX[0],0.5,0.5+nBinX[0],
nBinY[0],axisBinsY->GetMatrixArray());
} else {
return new TH2D(histogramName,title,
nBinX[0],0.5,0.5+nBinX[0],
nBinY[0],0.5,0.5+nBinY[0]);
}
}
}
Int_t TUnfoldBinning::GetTHxxBinning
(Int_t maxDim,Int_t *axisBins,Int_t *axisList,
const char *axisSteering) const
{
for(Int_t i=0;i<3;i++) {
axisBins[i]=0;
axisList[i]=-1;
}
const TUnfoldBinning *theNode=GetNonemptyNode();
if(theNode) {
return theNode->GetTHxxBinningSingleNode
(maxDim,axisBins,axisList,axisSteering);
} else {
axisBins[0]=GetTHxxBinsRecursive(axisSteering);
return 0;
}
}
const TUnfoldBinning *TUnfoldBinning::GetNonemptyNode(void) const
{
const TUnfoldBinning *r=GetDistributionNumberOfBins()>0 ? this : 0;
for(TUnfoldBinning const *child=GetChildNode();child;
child=child->GetNextNode()) {
const TUnfoldBinning *c=child->GetNonemptyNode();
if(!r) {
r=c;
} else {
if(c) {
r=0;
break;
}
}
}
return r;
}
Int_t TUnfoldBinning::GetTHxxBinningSingleNode
(Int_t maxDim,Int_t *axisBins,Int_t *axisList,const char *axisSteering) const
{
Int_t isOptionGiven[3];
DecodeAxisSteering(axisSteering,"CUO",isOptionGiven);
Int_t numDimension=GetDistributionDimension();
Int_t r=0;
for(Int_t i=0;i<numDimension;i++) {
if(isOptionGiven[0] & (1<<i)) continue;
r++;
}
if((r>0)&&(r<=maxDim)) {
r=0;
for(Int_t i=0;i<numDimension;i++) {
if(isOptionGiven[0] & (1<<i)) continue;
axisList[r]=i;
axisBins[r]=GetDistributionBinning(i)->GetNrows()-1;
r++;
}
} else {
if(HasUnconnectedBins() || (GetDistributionNumberOfBins()<=0)) {
axisBins[0] = GetDistributionNumberOfBins();
} else {
Int_t nBin=1;
for(Int_t i=0;i<numDimension;i++) {
Int_t mask=(1<<i);
if(isOptionGiven[0] & mask) continue;
Int_t nBinI=GetDistributionBinning(i)->GetNrows()-1;
if((fHasUnderflow & mask)&& !(isOptionGiven[1] & mask)) nBinI++;
if((fHasOverflow & mask)&& !(isOptionGiven[2] & mask)) nBinI++;
nBin *= nBinI;
}
axisBins[0] = nBin;
}
r=0;
}
return r;
}
Int_t TUnfoldBinning::GetTHxxBinsRecursive(const char *axisSteering) const
{
Int_t r=0;
for(TUnfoldBinning const *child=GetChildNode();child;
child=child->GetNextNode()) {
r +=child->GetTHxxBinsRecursive(axisSteering);
}
Int_t axisBins[3],axisList[3];
GetTHxxBinningSingleNode(0,axisBins,axisList,axisSteering);
r += axisBins[0];
return r;
}
Int_t *TUnfoldBinning::CreateBinMap
(const TH1 *hist,Int_t nDim,const Int_t *axisList,const char *axisSteering)
const
{
Int_t nMax=GetRootNode()->GetEndBin()+1;
Int_t *r=new Int_t[nMax];
for(Int_t i=0;i<nMax;i++) {
r[i]=-1;
}
Int_t startBin=GetRootNode()->GetStartBin();
if(nDim>0) {
const TUnfoldBinning *nonemptyNode=GetNonemptyNode();
if(nonemptyNode) {
FillBinMapSingleNode(hist,startBin,nDim,axisList,axisSteering,r);
} else {
Fatal("CreateBinMap","called with nDim=%d but GetNonemptyNode()=0",
nDim);
}
} else {
FillBinMapRecursive(startBin,axisSteering,r);
}
return r;
}
Int_t TUnfoldBinning::FillBinMapRecursive
(Int_t startBin,const char *axisSteering,Int_t *binMap) const
{
Int_t nbin=0;
nbin = FillBinMapSingleNode(0,startBin,0,0,axisSteering,binMap);
for(TUnfoldBinning const *child=GetChildNode();child;
child=child->GetNextNode()) {
nbin += child->FillBinMapRecursive(startBin+nbin,axisSteering,binMap);
}
return nbin;
}
Int_t TUnfoldBinning::FillBinMapSingleNode
(const TH1 *hist,Int_t startBin,Int_t nDim,const Int_t *axisList,
const char *axisSteering,Int_t *binMap) const
{
Int_t isOptionGiven[3];
DecodeAxisSteering(axisSteering,"CUO",isOptionGiven);
Int_t axisBins[MAXDIM];
Int_t dimension=GetDistributionDimension();
Int_t axisNbin[MAXDIM];
for(Int_t i=0;i<dimension;i++) {
const TVectorD *binning=GetDistributionBinning(i);
axisNbin[i]=binning->GetNrows()-1;
};
for(Int_t i=0;i<GetDistributionNumberOfBins();i++) {
Int_t globalBin=GetStartBin()+i;
const TUnfoldBinning *dest=ToAxisBins(globalBin,axisBins);
if(dest!=this) {
if(!dest) {
Fatal("FillBinMapSingleNode",
"bin %d outside binning scheme",
globalBin);
} else {
Fatal("FillBinMapSingleNode",
"bin %d located in %s %d-%d rather than %s %d=%d",
i,(const char *)dest->GetName(),
dest->GetStartBin(),dest->GetEndBin(),
(const char *)GetName(),GetStartBin(),GetEndBin());
}
}
Bool_t skip=kFALSE;
for(Int_t axis=0;axis<dimension;axis++) {
Int_t mask=(1<<axis);
if(((axisBins[axis]<0)&&(isOptionGiven[1] & mask))||
((axisBins[axis]>=axisNbin[axis])&&(isOptionGiven[2] & mask)))
skip=kTRUE;
}
if(skip) continue;
if(nDim>0) {
if(nDim==hist->GetDimension()) {
Int_t ibin[3];
ibin[0]=ibin[1]=ibin[2]=0;
for(Int_t hdim=0;hdim<nDim;hdim++) {
Int_t axis=axisList[hdim];
ibin[hdim]=axisBins[axis]+1;
}
binMap[globalBin]=hist->GetBin(ibin[0],ibin[1],ibin[2]);
} else if(nDim==1) {
for(Int_t ii=0;ii<hist->GetDimension();ii++) {
if(axisList[ii]>=0) {
binMap[globalBin]=axisBins[axisList[ii]]+1;
break;
}
}
} else {
Fatal("FillBinMapSingleNode","unexpected bin mapping %d %d",nDim,
hist->GetDimension());
}
} else {
if(dimension>0) {
Int_t r=0;
for(Int_t axis=dimension-1;axis>=0;axis--) {
Int_t mask=(1<<axis);
if(isOptionGiven[0] & mask) {
continue;
}
Int_t iBin=axisBins[axis];
Int_t nMax=axisNbin[axis];
if((fHasUnderflow & ~isOptionGiven[1]) & mask) {
nMax +=1;
iBin +=1;
}
if((fHasOverflow & ~isOptionGiven[2]) & mask) {
nMax += 1;
}
r = r*nMax +iBin;
}
binMap[globalBin] = startBin + r;
} else {
binMap[globalBin] = startBin + axisBins[0];
}
}
}
Int_t nbin;
if(dimension>0) {
nbin=1;
for(Int_t axis=dimension-1;axis>=0;axis--) {
Int_t mask=(1<<axis);
if(isOptionGiven[0] & mask) {
continue;
}
Int_t nMax=axisNbin[axis];
if((fHasUnderflow & ~isOptionGiven[1]) & mask) {
nMax +=1;
}
if((fHasOverflow & ~isOptionGiven[2]) & mask) {
nMax += 1;
}
nbin = nbin*nMax;
}
} else {
nbin=GetDistributionNumberOfBins();
}
return nbin;
}
TH1 *TUnfoldBinning::ExtractHistogram
(const char *histogramName,const TH1 *globalBins,
const TH2 *globalBinsEmatrix,Bool_t originalAxisBinning,
const char *axisSteering) const
{
Int_t *binMap=0;
TH1 *r=CreateHistogram(histogramName,originalAxisBinning,&binMap,0,
axisSteering);
TUnfoldBinning const *root=GetRootNode();
Int_t nMax=0;
for(Int_t iSrc=root->GetStartBin();iSrc<root->GetEndBin();iSrc++) {
if(binMap[iSrc]>nMax) nMax=binMap[iSrc];
}
TVectorD eSquared(nMax+1);
for(Int_t iSrc=root->GetStartBin();iSrc<root->GetEndBin();iSrc++) {
Int_t iDest=binMap[iSrc];
if(iDest>=0) {
Double_t c=r->GetBinContent(iDest);
r->SetBinContent(iDest,c+globalBins->GetBinContent(iSrc));
if(!globalBinsEmatrix) {
eSquared(iDest) += TMath::Power(globalBins->GetBinError(iSrc),2.);
} else {
for(Int_t jSrc=root->GetStartBin();jSrc<root->GetEndBin();jSrc++) {
if(binMap[jSrc]==iDest) {
eSquared(iDest) +=
TMath::Power(globalBins->GetBinError(jSrc),2.);
}
}
}
}
}
for(Int_t i=0;i<nMax;i++) {
Double_t e2=eSquared(i);
if(e2>0.0) {
r->SetBinError(i,TMath::Sqrt(e2));
}
}
return r;
}
Int_t TUnfoldBinning::GetGlobalBinNumber(Double_t x) const
{
if(GetDistributionDimension()!=1) {
Fatal("GetBinNumber",
"called with 1 argument for %d dimensional distribution",
GetDistributionDimension());
}
return GetGlobalBinNumber(&x);
}
Int_t TUnfoldBinning::GetGlobalBinNumber(Double_t x,Double_t y) const
{
if(GetDistributionDimension()!=2) {
Fatal("GetBinNumber",
"called with 2 arguments for %d dimensional distribution",
GetDistributionDimension());
}
Double_t xx[2];
xx[0]=x;
xx[1]=y;
return GetGlobalBinNumber(xx);
}
Int_t TUnfoldBinning::GetGlobalBinNumber
(Double_t x,Double_t y,Double_t z) const
{
if(GetDistributionDimension()!=3) {
Fatal("GetBinNumber",
"called with 3 arguments for %d dimensional distribution",
GetDistributionDimension());
}
Double_t xx[3];
xx[0]=x;
xx[1]=y;
xx[2]=z;
return GetGlobalBinNumber(xx);
}
Int_t TUnfoldBinning::GetGlobalBinNumber
(Double_t x0,Double_t x1,Double_t x2,Double_t x3) const
{
if(GetDistributionDimension()!=4) {
Fatal("GetBinNumber",
"called with 4 arguments for %d dimensional distribution",
GetDistributionDimension());
}
Double_t xx[4];
xx[0]=x0;
xx[1]=x1;
xx[2]=x2;
xx[3]=x3;
return GetGlobalBinNumber(xx);
}
Int_t TUnfoldBinning::GetGlobalBinNumber(const Double_t *x) const
{
if(!GetDistributionDimension()) {
Fatal("GetBinNumber",
"no axes are defined for node %s",
(char const *)GetName());
}
Int_t iAxisBins[MAXDIM];
for(Int_t dim=0;dim<GetDistributionDimension();dim++) {
TVectorD const *bins=(TVectorD const *) fAxisList->At(dim);
Int_t i0=0;
Int_t i1=bins->GetNrows()-1;
Int_t iBin= 0;
if(x[dim]<(*bins)[i0]) {
iBin += i0-1;
} else if(x[dim]>=(*bins)[i1]) {
iBin += i1;
} else {
while(i1-i0>1) {
Int_t i2=(i0+i1)/2;
if(x[dim]<(*bins)[i2]) {
i1=i2;
} else {
i0=i2;
}
}
iBin += i0;
}
iAxisBins[dim]=iBin;
}
Int_t r=ToGlobalBin(iAxisBins);
if(r<0) r=0;
return r;
}
TString TUnfoldBinning::GetBinName(Int_t iBin) const
{
Int_t axisBins[MAXDIM];
TString r=TString::Format("#%d",iBin);
TUnfoldBinning const *distribution=ToAxisBins(iBin,axisBins);
if(distribution) {
r +=" (";
r += distribution->GetName();
Int_t dimension=distribution->GetDistributionDimension();
if(dimension>0) {
TString axisString;
for(Int_t axis=0;axis<dimension;axis++) {
TString thisAxisString=
distribution->GetDistributionAxisLabel(axis);
TVectorD const *bins=distribution->GetDistributionBinning(axis);
Int_t i=axisBins[axis];
if(i<0) thisAxisString += "[ufl]";
else if(i>=bins->GetNrows()-1) thisAxisString += "[ofl]";
else {
thisAxisString +=
TString::Format("[%.3g,%.3g]",(*bins)[i],(*bins)[i+1]);
}
axisString = ":"+thisAxisString+axisString;
}
r += axisString;
} else {
Int_t i=axisBins[0];
if((i>=0)&&(i<distribution->fAxisLabelList->GetEntriesFast())) {
r += distribution->GetDistributionAxisLabel(i);
} else {
r += TString::Format(" %d",i);
}
}
r +=")";
}
return r;
}
Double_t TUnfoldBinning::GetBinSize(Int_t iBin) const
{
Int_t axisBins[MAXDIM];
TUnfoldBinning const *distribution=ToAxisBins(iBin,axisBins);
Double_t r=0.0;
if(distribution) {
if(distribution->GetDistributionDimension()>0) r=1.0;
for(Int_t axis=0;axis<distribution->GetDistributionDimension();axis++) {
TVectorD const *bins=distribution->GetDistributionBinning(axis);
Int_t pos=axisBins[axis];
if(pos<0) {
r *= distribution->GetDistributionUnderflowBinWidth(axis);
} else if(pos>=bins->GetNrows()-1) {
r *= distribution->GetDistributionOverflowBinWidth(axis);
} else {
r *= (*bins)(pos+1)-(*bins)(pos);
}
if(r<=0.) break;
}
}
return r;
}
Double_t TUnfoldBinning::GetBinFactor(Int_t iBin) const
{
Int_t axisBins[MAXDIM];
TUnfoldBinning const *distribution=ToAxisBins(iBin,axisBins);
Double_t r=distribution->fBinFactorConstant;
if((r!=0.0) && distribution->fBinFactorFunction) {
Double_t x[MAXDIM];
Int_t dimension=distribution->GetDistributionDimension();
if(dimension>0) {
for(Int_t axis=0;axis<dimension;axis++) {
x[axis]=distribution->GetDistributionBinCenter
(axis,axisBins[axis]);
}
r *= distribution->fBinFactorFunction->EvalPar
(x,distribution->fBinFactorFunction->GetParameters());
} else {
x[0]=axisBins[0];
r *= distribution->fBinFactorFunction->Eval(x[0]);
}
}
return r;
}
void TUnfoldBinning::GetBinNeighbours
(Int_t bin,Int_t axis,Int_t *prev,Double_t *distPrev,
Int_t *next,Double_t *distNext) const
{
Int_t axisBins[MAXDIM];
TUnfoldBinning const *distribution=ToAxisBins(bin,axisBins);
Int_t dimension=distribution->GetDistributionDimension();
*prev=-1;
*next=-1;
*distPrev=0.;
*distNext=0.;
if((axis>=0)&&(axis<dimension)) {
Int_t centerBin= axisBins[axis];
axisBins[axis] =centerBin-1;
*prev=ToGlobalBin(axisBins);
if(*prev>=0) {
*distPrev=distribution->GetDistributionBinCenter(axis,axisBins[axis])-
distribution->GetDistributionBinCenter(axis,centerBin);
}
axisBins[axis] =centerBin+1;
*next=ToGlobalBin(axisBins);
if(*next>=0) {
*distNext=distribution->GetDistributionBinCenter(axis,axisBins[axis])-
distribution->GetDistributionBinCenter(axis,centerBin);
}
}
}
void TUnfoldBinning::GetBinUnderflowOverflowStatus
(Int_t iBin,Int_t *uStatus,Int_t *oStatus) const
{
Int_t axisBins[MAXDIM];
TUnfoldBinning const *distribution=ToAxisBins(iBin,axisBins);
Int_t dimension=distribution->GetDistributionDimension();
*uStatus=0;
*oStatus=0;
for(Int_t axis=0;axis<dimension;axis++) {
TVectorD const *bins=distribution->GetDistributionBinning(axis);
Int_t nBin=bins->GetNrows()-1;
if(axisBins[axis]<0) *uStatus |= (1<<axis);
if(axisBins[axis]>=nBin) *oStatus |= (1<<axis);
}
}
const TUnfoldBinning *TUnfoldBinning::GetBinLocation
(Int_t binTHxx,const char *axisSteering,Int_t axisBins[MAXDIM]) const
{
Int_t offset=binTHxx-GetStartBin();
return GetBinLocationRecursive(offset,axisSteering,axisBins);
}
const TUnfoldBinning *TUnfoldBinning::GetBinLocationRecursive
(Int_t &offset,const char *axisSteering,Int_t axisBins[MAXDIM]) const
{
Int_t isOptionGiven[3];
DecodeAxisSteering(axisSteering,"CUO",isOptionGiven);
const TUnfoldBinning *r=0;
if(offset>=0) {
if(GetDistributionDimension()>0) {
Int_t nBinsTotal=1;
Int_t i=offset;
for(Int_t axis=0;axis<GetDistributionDimension();axis++) {
Int_t mask=1<<axis;
if(isOptionGiven[0] & mask) {
axisBins[axis]=-2;
if((isOptionGiven[1] & mask)&&
(fHasUnderflow & mask)) axisBins[axis] -= 1;
if((isOptionGiven[2] & mask)&&
(fHasOverflow & mask)) axisBins[axis] -= 2;
} else {
Int_t nBin=GetDistributionBinning(axis)->GetNrows()-1;
axisBins[axis]=0;
if((fHasUnderflow & mask) && !(isOptionGiven[1] & mask)) {
nBin++;
axisBins[axis]=-1;
}
if((fHasOverflow & mask) && !(isOptionGiven[2] & mask)) {
nBin++;
}
axisBins[axis] += i % nBin;
i /= nBin;
nBinsTotal *= nBin;
}
}
offset -= nBinsTotal;
if(offset<0) {
r=this;
}
} else {
axisBins[0]=offset;
offset -= GetDistributionNumberOfBins();
if(offset<0) r=this;
}
}
if(!r) {
for(TUnfoldBinning const *child=GetChildNode();child;
child=child->GetNextNode()) {
r=child->GetBinLocationRecursive(offset,axisSteering,axisBins);
if(r) break;
}
}
return r;
}
Bool_t TUnfoldBinning::HasUnconnectedBins(void) const
{
return (!GetDistributionDimension())&&(GetDistributionNumberOfBins()>0);
}
Double_t TUnfoldBinning::GetDistributionAverageBinSize
(Int_t axis,Bool_t includeUnderflow,Bool_t includeOverflow) const
{
Double_t r=0.0;
if((axis>=0)&&(axis<GetDistributionDimension())) {
TVectorD const *bins=GetDistributionBinning(axis);
Double_t d=(*bins)[bins->GetNrows()-1]-(*bins)[0];
Double_t nBins=bins->GetNrows()-1;
if(includeUnderflow && (fHasUnderflow & (1<<axis))) {
Double_t w=GetDistributionUnderflowBinWidth(axis);
if(w>0) {
nBins++;
d += w;
}
}
if(includeOverflow && (fHasOverflow & (1<<axis))) {
Double_t w=GetDistributionOverflowBinWidth(axis);
if(w>0.0) {
nBins++;
d += w;
}
}
if(nBins>0) {
r=d/nBins;
}
} else {
Error("GetDistributionAverageBinSize","axis %d does not exist",axis);
}
return r;
}
Double_t TUnfoldBinning::GetDistributionUnderflowBinWidth(Int_t axis) const
{
TVectorD const *bins=GetDistributionBinning(axis);
return (*bins)[1]-(*bins)[0];
}
Double_t TUnfoldBinning::GetDistributionOverflowBinWidth(Int_t axis) const
{
TVectorD const *bins=GetDistributionBinning(axis);
return (*bins)[bins->GetNrows()-1]-(*bins)[bins->GetNrows()-2];
}
Double_t TUnfoldBinning::GetDistributionBinCenter
(Int_t axis,Int_t bin) const
{
TVectorD const *bins=GetDistributionBinning(axis);
Double_t r=0.0;
if(bin<0) {
r=(*bins)[0]-0.5*GetDistributionUnderflowBinWidth(axis);
} else if(bin>=bins->GetNrows()-1) {
r=(*bins)[bins->GetNrows()-1]+0.5*GetDistributionOverflowBinWidth(axis);
} else {
r=0.5*((*bins)[bin+1]+(*bins)[bin]);
}
return r;
}
Int_t TUnfoldBinning::ToGlobalBin(Int_t const *axisBins) const
{
Int_t dimension=GetDistributionDimension();
Int_t r=0;
if(dimension>0) {
for(Int_t axis=dimension-1;axis>=0;axis--) {
Int_t nMax=GetDistributionBinning(axis)->GetNrows()-1;
Int_t i=axisBins[axis];
if(fHasUnderflow & (1<<axis)) {
nMax +=1;
i +=1;
}
if(fHasOverflow & (1<<axis)) nMax +=1;
if((i>=0)&&(i<nMax)) {
r = r*nMax +i;
} else {
r=-1;
break;
}
}
if(r>=0) {
r += GetStartBin();
}
} else {
if((axisBins[0]>=0)&&(axisBins[0]<GetDistributionNumberOfBins()))
r=GetStartBin()+axisBins[0];
}
return r;
}
TUnfoldBinning const *TUnfoldBinning::ToAxisBins
(Int_t globalBin,Int_t *axisBins) const
{
TUnfoldBinning const *r=0;
if((globalBin>=GetStartBin())&&(globalBin<GetEndBin())) {
TUnfoldBinning const *node;
for(node=GetChildNode();node && !r; node=node->GetNextNode()) {
r=node->ToAxisBins(globalBin,axisBins);
}
if(!r) {
r=this;
Int_t i=globalBin-GetStartBin();
Int_t dimension=GetDistributionDimension();
if(dimension>0) {
for(int axis=0;axis<dimension;axis++) {
Int_t nMax=GetDistributionBinning(axis)->GetNrows()-1;
axisBins[axis]=0;
if(fHasUnderflow & (1<<axis)) {
axisBins[axis] =-1;
nMax += 1;
}
if(fHasOverflow & (1<<axis)) nMax +=1;
axisBins[axis] += i % nMax;
i /= nMax;
}
} else {
axisBins[0]=i;
}
}
}
return r;
}
void TUnfoldBinning::DecodeAxisSteering
(const char *axisSteering,const char *options,Int_t *isOptionGiven) const
{
Int_t nOpt=TString(options).Length();
for(Int_t i=0;i<nOpt;i++) isOptionGiven[i]=0;
if(axisSteering) {
TObjArray *patterns=TString(axisSteering).Tokenize(";");
Int_t nPattern=patterns->GetEntries();
Int_t nAxis=fAxisLabelList->GetEntries();
for(Int_t i=0;i<nPattern;i++) {
TString const &pattern=((TObjString * const)patterns->At(i))
->GetString();
Int_t bracketBegin=pattern.Last('[');
Int_t len=pattern.Length();
if((bracketBegin>0)&&(pattern[len-1]==']')) {
TString axisId=pattern(0,bracketBegin);
Int_t mask=0;
if((axisId[0]=='*')&&(axisId.Length()==1)) {
mask=(1<<nAxis)-1;
} else {
for(Int_t j=0;j<nAxis;j++) {
if(!axisId.CompareTo(GetDistributionAxisLabel(j))) {
mask|= (1<<j);
}
}
}
for(Int_t o=0;o<nOpt;o++) {
if(pattern.Last(options[o])>bracketBegin) {
isOptionGiven[o] |= mask;
}
}
} else {
Error("DecodeAxisSteering",
"steering \"%s\" does not end with [options]",
(const char *)pattern);
}
}
}
}