#include "THaElossCorrection.h"
#include "THaSpectrometer.h"
#include "THaTrack.h"
#include "THaTrackInfo.h"
#include "THaVertexModule.h"
#include "TMath.h"
#include "TVector3.h"
#include "VarDef.h"
#include "VarType.h"
#include <iostream>
using namespace std;
THaElossCorrection::THaElossCorrection( const char* name,
const char* description,
const char* input_tracks,
Double_t particle_mass,
Int_t hadron_charge ) :
THaPhysicsModule(name,description), fZ(hadron_charge),
fZmed(0.0), fAmed(0.0), fDensity(0.0), fPathlength(0.0),
fZref(0.0), fScale(0.0),
fTestMode(kFALSE), fExtPathMode(kFALSE), fInputName(input_tracks),
fVertexModule(NULL)
{
SetMass(particle_mass);
Clear();
}
THaElossCorrection::~THaElossCorrection()
{
DefineVariables( kDelete );
}
void THaElossCorrection::Clear( Option_t* opt )
{
THaPhysicsModule::Clear(opt);
if( !fTestMode )
fEloss = kBig;
}
THaAnalysisObject::EStatus THaElossCorrection::Init( const TDatime& run_time )
{
if( fExtPathMode ) {
fVertexModule = dynamic_cast<THaVertexModule*>
( FindModule( fVertexName.Data(), "THaVertexModule" ));
if( !fVertexModule )
return fStatus;
}
THaPhysicsModule::Init( run_time );
return fStatus;
}
Int_t THaElossCorrection::DefineVariables( EMode mode )
{
if( mode == kDefine && fIsSetup ) return kOK;
fIsSetup = ( mode == kDefine );
const RVarDef var[] = {
{ "eloss", "Calculated energy loss correction (GeV)", "fEloss" },
{ "pathl", "Pathlength thru medium for this event", "fPathlength" },
{ 0 }
};
DefineVarsFromList( var, mode );
return 0;
}
Int_t THaElossCorrection::ReadRunDatabase( const TDatime& date )
{
Int_t err = THaPhysicsModule::ReadRunDatabase( date );
if( err ) return err;
FILE* f = OpenRunDBFile( date );
if( !f ) return kFileError;
TagDef tags[] = {
{ "M", &fM, 1 },
{ "Z", &fZ, 0, 0, kInt },
{ "Z_med", &fZmed, 3 },
{ "A_med", &fAmed, 4 },
{ "density", &fDensity, 5 },
{ "pathlength", &fPathlength, 6 },
{ 0 }
};
if( fExtPathMode )
tags[5].name = 0;
TagDef* item = tags;
while( item->name ) {
if( *((double*)item->var) != 0.0 )
item->var = 0;
item++;
}
err = LoadDB( f, date, tags, fPrefix );
fclose(f);
if( err ) {
const char* errmsg[] = { "M (particle mass [GeV/c^2])", "Z (particle Z)",
"Z_med (Z of medium)", "A_med (A of medium)",
"density (of medium [g/cm^3])",
"pathlength (through medium [m])" };
const char* here = Here("ReadRunDatabase");
if( err>0 && err<=6 )
Error( here, "Required database entry \"%s\" missing "
"in the run database. Module not initialized", errmsg[err-1] );
else
Error( here, "Error reading run database. Module not initialized" );
return kInitError;
}
return kOK;
}
void THaElossCorrection::SetInputModule( const char* name )
{
if( !IsInit())
fInputName = name;
else
PrintInitError("SetInputModule");
}
void THaElossCorrection::SetMass( Double_t m )
{
if( !IsInit() ) {
fM = m;
fElectronMode = (TMath::Abs(fM) < 1e-3);
} else
PrintInitError("SetMass");
}
void THaElossCorrection::SetTestMode( Bool_t enable, Double_t eloss_value )
{
if( !IsInit() ) {
fTestMode = enable;
if( enable )
fEloss = TMath::Max(eloss_value,0.0);
} else
PrintInitError("SetTestMode");
}
void THaElossCorrection::SetMedium( Double_t Z, Double_t A,
Double_t density )
{
if( !IsInit() ) {
fZmed = Z;
fAmed = A;
fDensity = density;
} else
PrintInitError("SetMedium");
}
void THaElossCorrection::SetPathlength( Double_t pathlength )
{
if( !IsInit() ) {
fPathlength = pathlength;
fExtPathMode = kFALSE;
} else
PrintInitError("SetPathlength");
}
void THaElossCorrection::SetPathlength( const char* vertex_module,
Double_t z_ref, Double_t scale )
{
if( !IsInit() ) {
fZref = z_ref;
fScale = scale;
fVertexName = vertex_module;
fExtPathMode = kTRUE;
} else
PrintInitError("SetPathlength");
}
Double_t THaElossCorrection::ElossElectron( Double_t beta, Double_t z_med,
Double_t a_med, Double_t d_med,
Double_t pathlength )
{
Double_t BETA2,BETA3,PLAS,EKIN,TAU,FTAU;
Double_t BETH,DENS,ESTP,EXEN,GAMMA;
static const Double_t COEF = 0.1535374581;
static const Double_t EMAS = 0.5109989020;
static const Double_t log2 = TMath::Log(2.0);
if( beta <= 0.0 || beta >= 1.0 || z_med == 0.0 ||
a_med == 0.0 || pathlength == 0.0 )
return 0.0;
pathlength *= 1e2;
BETA2 = beta * beta;
BETA3 = 1.0 - BETA2;
GAMMA = 1.0/TMath::Sqrt(BETA3);
EXEN = ExEnerg(z_med,d_med);
if( EXEN == 0.0 )
return 0.0;
EKIN = EMAS * ( GAMMA - 1.0 );
TAU = EKIN / EMAS;
FTAU = 1.0+(TAU*TAU/8.0)-((2.0*TAU+1.0)*log2);
BETH = 2.0 * TMath::Log(1e6*EKIN/EXEN) + BETA3 * FTAU;
BETH = BETH + TMath::Log(1.0+(0.5*TAU));
PLAS = 28.8084 * TMath::Sqrt(d_med*z_med/a_med);
DENS = 2.0*(TMath::Log(PLAS)+TMath::Log(beta*GAMMA)-TMath::Log(EXEN))-1.0;
if(DENS < 0.0) DENS = 0.0;
ESTP = COEF * z_med * ( BETH - DENS ) / a_med / BETA2;
Double_t eloss = ESTP * d_med * pathlength * 1e-3;
return eloss;
}
Double_t THaElossCorrection:: ElossHadron( Int_t Z_hadron, Double_t beta,
Double_t z_med, Double_t a_med,
Double_t d_med,
Double_t pathlength )
{
Double_t BETA2,GAMA2,GAMA,PLAS,ETA,ETA2,ETA4,ETA6;
Double_t BETH,DENS,SHE0,SHE1,SHEL,HSTP;
Double_t X,X0,X1,M,EXEN,C,A;
static const Double_t COEF = 0.3070749162;
static const Double_t EMAS = 0.5109989020;
static const Double_t log100 = TMath::Log(1e2);
if( Z_hadron == 0 || beta <= 0.0 || beta >= 1.0 || z_med == 0.0 ||
a_med == 0.0 || pathlength == 0.0 )
return 0.0;
pathlength *= 1e2;
BETA2 = beta * beta;
GAMA2 = 1.0 / (1.0 - BETA2);
GAMA = TMath::Sqrt( GAMA2 );
ETA = beta * GAMA ;
EXEN = ExEnerg(z_med,d_med);
if( EXEN == 0.0 )
return 0.0;
BETH = TMath::Log(2e6*EMAS*BETA2*GAMA2/EXEN) - BETA2;
PLAS = 28.8084 * TMath::Sqrt(d_med*z_med/a_med);
C = 2.0*TMath::Log(PLAS) - 2.0*TMath::Log(EXEN) - 1.0;
X = TMath::Log10(ETA);
HaDensi(z_med,d_med,X0,X1,M);
if((X0+X1+M) == 0.0)
return 0.0;
A = -1.0 * ( C + log100*X0 ) / TMath::Power(X1-X0,M);
if(X<X0)
DENS = 0.0;
else if(X<X1)
DENS = log100*X + C + A*TMath::Power(X1-X,M);
else
DENS = log100*X + C;
DENS *= 0.5;
ETA2 = 1.0 / (ETA*ETA) ;
ETA4 = ETA2 * ETA2 ;
ETA6 = ETA4 * ETA2 ;
SHE0 = 4.2237e-1*ETA2 + 3.040e-2*ETA4 - 3.80e-4*ETA6;
SHE1 = 3.8580e0 *ETA2 - 1.668e-1*ETA4 + 1.58e-3*ETA6;
SHEL = 1e-6 * EXEN * EXEN * (SHE0 + 1e-3*SHE1*EXEN) / z_med;
HSTP = COEF * z_med * Double_t(Z_hadron*Z_hadron) / a_med;
HSTP = HSTP * ( BETH - DENS - SHEL ) / BETA2;
Double_t eloss = HSTP * d_med * pathlength * 1e-3;
return eloss;
}
Double_t THaElossCorrection::ExEnerg( Double_t z_med, Double_t d_med )
{
Double_t EXEN;
if( TMath::Abs(z_med-1.0)<0.5) {
if(d_med<0.01)
EXEN = 19.2;
else if(d_med<0.1)
EXEN = 21.8;
else
EXEN = 21.8;
}
else if( TMath::Abs(z_med-2.0)<0.1 )
EXEN = 41.8;
else if( TMath::Abs(z_med-3.37)<0.1 )
EXEN = 64.7;
else if( TMath::Abs(z_med-5.03)<0.1 )
EXEN = 79.6;
else if( TMath::Abs(z_med-6.0)<0.1 )
EXEN = 78.0;
else if( TMath::Abs(z_med-7.22)<0.1 )
EXEN = 85.7;
else if( TMath::Abs(z_med-13.0)<0.1 )
EXEN = 166.0;
else if( TMath::Abs(z_med-29.0)<0.1 )
EXEN = 322.0;
else if( TMath::Abs(z_med-22.0)<0.1 )
EXEN = 233.0;
else {
cout << endl;
cout << "Warning... Unknown selected material !!" << endl;
cout << "z_med = " << z_med << " d_med = " << d_med << endl;
cout << "Implement THaElossCorrection::ExEnerg routine for proper use."
<< endl << endl;
EXEN = 0.0;
}
return EXEN;
}
void THaElossCorrection::HaDensi( Double_t z_med, Double_t d_med,
Double_t& X0, Double_t& X1, Double_t& M )
{
if( TMath::Abs( z_med-1.0 ) < 0.1 ) {
if( d_med < 0.01 ) {
X0 = 1.8639;
X1 = 3.2718;
M = 5.7273;
}
else if( d_med < 0.1 ) {
X0 = 0.4759;
X1 = 1.9215;
M = 5.6249;
}
else if( d_med >= 0.1 ) {
X0 = 0.4759;
X1 = 1.9215;
M = 5.6249;
}
}
else if( TMath::Abs( z_med-2.0 ) < 0.1 ) {
X0 = 2.2017;
X1 = 3.6122;
M = 5.8347;
}
else if( TMath::Abs( z_med-3.37 ) < 0.1 ) {
X0 = 0.1464;
X1 = 2.4855;
M = 3.2393;
}
else if( TMath::Abs( z_med-5.03 ) < 0.1 ) {
X0 = 0.1509;
X1 = 2.5631;
M = 3.1921;
}
else if( TMath::Abs( z_med-6.0 ) < 0.1 ) {
if( d_med < 1.750) {
X0 = 0.0480;
X1 = 2.5387;
M = 2.9532;
}
else if( d_med < 2.050) {
X0 = -0.0351;
X1 = 2.4860;
M = 3.0036;
}
else if( d_med < 2.270) {
X0 = -0.0178;
X1 = 2.3415;
M = 2.8697;
}
}
else if( TMath::Abs( z_med-7.22 ) < 0.1 ) {
X0 = 1.7418;
X1 = 4.2759;
M = 3.3994;
}
else if( TMath::Abs( z_med-13.0 ) < 0.1 ) {
X0 = 0.1708;
X1 = 3.0127;
M = 3.6345;
}
else if( TMath::Abs( z_med-22.0 ) < 0.1 ) {
X0 = 0.0957;
X1 = 3.0386;
M = 3.0302;
}
else {
cout << endl;
cout << "Warning... Inconsistent material density... " << endl;
cout << "z_med = " << z_med << " d_med = " << d_med << endl;
cout << "Implement THaElossCorrection::HaDensi routine for proper use."
<< endl << endl;
X0 = 0.;
X1 = 0.;
M = 0.;
}
return;
}
ClassImp(THaElossCorrection)
Last change: Sat Nov 7 21:26:45 2009
Last generated: 2009-11-07 21:26
This page has been automatically generated. If you have any comments or suggestions about the page layout send a mail to ROOT support, or contact the developers with any questions or problems regarding ROOT.