11#include "GaudiKernel/MsgStream.h"
12#include "GaudiKernel/Bootstrap.h"
13#include "GaudiKernel/PropertyMgr.h"
14#include "GaudiKernel/IJobOptionsSvc.h"
15#include "GaudiKernel/ISvcLocator.h"
16#include "GaudiKernel/IDataProviderSvc.h"
18#include "BesTofDigitizerEcV4.hh"
19#include "BesTofDigi.hh"
20#include "BesTofHit.hh"
21#include "G4DigiManager.hh"
22#include "Randomize.hh"
34 PropertyMgr m_propMgr;
35 m_propMgr.declareProperty(
"FileName", m_fileName =
"mrpc.root");
36 m_propMgr.declareProperty(
"RootFlag", m_rootFlag =
false);
37 m_propMgr.declareProperty(
"E", m_V = 7000);
38 m_propMgr.declareProperty(
"Threshold", m_threshold = 5.5e+08);
40 m_propMgr.declareProperty(
"nstep", m_nstep = -1);
41 m_propMgr.declareProperty(
"E_weight", m_E_weight = -1);
42 m_propMgr.declareProperty(
"saturationFlag", m_saturationFlag =
true);
43 m_propMgr.declareProperty(
"tdcRes_const", tdcRes_const = -1);
44 m_propMgr.declareProperty(
"adcRes_const", adcRes_const = -1);
45 m_propMgr.declareProperty(
"calTdcRes_charge_flag", m_calTdcRes_charge_flag=0);
46 m_propMgr.declareProperty(
"charge2Time_flag", m_charge2Time_flag=0);
47 m_propMgr.declareProperty(
"calAdcRes_charge_flag", m_calAdcRes_charge_flag=0);
48 cout<<
"Property:"<<endl
49 <<
" FileName= "<<m_fileName
51 <<
" Threshold= "<<m_threshold
53 <<
" E_weight= "<<m_E_weight
54 <<
" saturationFlag= "<<m_saturationFlag
55 <<
" tdcRes_const= "<<tdcRes_const
56 <<
" adcRes_const= "<<adcRes_const
57 <<
" calTdcRes_charge_flag= "<<m_calTdcRes_charge_flag
58 <<
" charge2Time_flag= "<<m_charge2Time_flag
59 <<
" calAdcRes_charge_flag= "<<m_calAdcRes_charge_flag
63 IJobOptionsSvc* jobSvc;
64 Gaudi::svcLocator()->service(
"JobOptionsSvc", jobSvc);
65 jobSvc->setMyProperties(
"BesTofDigitizerEcV4", &m_propMgr);
71 m_file =
new TFile(m_fileName.c_str(),
"RECREATE");
72 m_tree =
new TTree(
"mrpc",
"mrpc");
74 m_tree->Branch(
"event", &m_event,
"event/D");
75 m_tree->Branch(
"partId", &m_partId,
"partId/D");
76 m_tree->Branch(
"module", &m_module,
"module/D");
77 m_tree->Branch(
"time_leading_sphi", &m_time_leading_sphi,
"time_leading_sphi/D");
78 m_tree->Branch(
"time_leading_xphi", &m_time_leading_xphi,
"time_leading_xphi/D");
79 m_tree->Branch(
"time_trailing_sphi", &m_time_trailing_sphi,
"time_trailing_sphi/D");
80 m_tree->Branch(
"time_trailing_xphi", &m_time_trailing_xphi,
"time_trailing_xphi/D");
81 m_tree->Branch(
"tdcRes", &m_tdcRes,
"tdcRes/D");
82 m_tree->Branch(
"tdcRes_charge", &m_tdcRes_charge,
"tdcRes_charge/D");
83 m_tree->Branch(
"adc", &m_adc,
"adc/D");
84 m_tree->Branch(
"adcRes", &m_adcRes,
"adcRes/D");
85 m_tree->Branch(
"adcRes_charge", &m_adcRes_charge,
"adcRes_charge/D");
86 m_tree->Branch(
"strip", &m_strip,
"strip/D");
87 m_tree->Branch(
"trkIndex", &m_trkIndex,
"trkIndex/D");
88 m_tree->Branch(
"tStart", &m_tStart,
"tStart/D");
89 m_tree->Branch(
"tPropagate_sphi", &m_tPropagate_sphi,
"tPropagate_sphi/D");
90 m_tree->Branch(
"tPropagate_xphi", &m_tPropagate_xphi,
"tPropagate_xphi/D");
91 m_tree->Branch(
"tThreshold", &m_tThreshold,
"tThreshold/D");
92 m_tree->Branch(
"charge", &m_charge,
"charge/D");
93 m_tree->Branch(
"nhit", &m_nhit,
"nhit/I");
94 m_tree->Branch(
"ions_hit", m_ions_hit,
"ions_hit[nhit]/D");
95 m_tree->Branch(
"trkIndex_hit", m_trkIndex_hit,
"trkIndex_hit[nhit]/D");
96 m_tree->Branch(
"pdgCode_hit", m_pdgCode_hit,
"pdgCode_hit[nhit]/D");
97 m_tree->Branch(
"gap_hit", m_gap_hit,
"gap_hit[nhit]/D");
98 m_tree->Branch(
"underStrip_hit", m_underStrip_hit,
"underStrip_hit[nhit]/D");
99 m_tree->Branch(
"locx_hit", m_locx_hit,
"locx_hit[nhit]/D");
100 m_tree->Branch(
"locy_hit", m_locy_hit,
"locy_hit[nhit]/D");
101 m_tree->Branch(
"locz_hit", m_locz_hit,
"locz_hit[nhit]/D");
102 m_tree->Branch(
"x_hit", m_x_hit,
"x_hit[nhit]/D");
103 m_tree->Branch(
"y_hit", m_y_hit,
"y_hit[nhit]/D");
104 m_tree->Branch(
"z_hit", m_z_hit,
"z_hit[nhit]/D");
105 m_tree->Branch(
"px_hit", m_px_hit,
"px_hit[nhit]/D");
106 m_tree->Branch(
"py_hit", m_py_hit,
"py_hit[nhit]/D");
107 m_tree->Branch(
"pz_hit", m_pz_hit,
"pz_hit[nhit]/D");
124 m_param.
setPar(m_nstep, m_E_weight);
130 if(tdcRes_const<0) tdcRes_const = 38;
134 if(adcRes_const<0) adcRes_const = 27;
137 time_leading_sphi = -999;
138 time_leading_xphi = -999;
139 time_trailing_sphi = -999;
140 time_trailing_xphi = -999;
146 G4DigiManager* digiManager = G4DigiManager::GetDMpointer();
147 G4int THCID = digiManager->GetHitsCollectionID(
"BesTofHitsCollection");
156 int nstrip = m_param.
nstrip;
158 for(
int i=0; i<nstrip; i++)
160 stripStruct[i].
m_param = m_param;
161 stripStruct[i].
setPar(m_V, m_threshold, m_saturationFlag);
182 hitStruct.
x = hit->
GetPos().x()/mm;
183 hitStruct.
y = hit->
GetPos().y()/mm;
184 hitStruct.
z = hit->
GetPos().z()/mm;
199 for(
int i=0; i<nstrip; i++)
201 if(stripStruct[i].hitStructCol.size()==0)
continue;
202 stripStruct[i].
strip = i;
207 if(stripStruct[i].tThreshold<=0)
continue;
212 double tdcRes_charge;
213 if(m_calTdcRes_charge_flag==0)
217 else if(m_calTdcRes_charge_flag==1)
221 else if(m_calTdcRes_charge_flag==2)
226 tdcRes = sqrt(tdcRes_charge*tdcRes_charge+tdcRes_const*tdcRes_const)/1000;
228 tdc_sphi = G4RandGauss::shoot(tdc_sphi, tdcRes);
229 tdc_xphi = G4RandGauss::shoot(tdc_xphi, tdcRes);
231 if(m_charge2Time_flag==0)
235 else if(m_charge2Time_flag==1)
240 double adcRes_charge;
241 if(m_calAdcRes_charge_flag==0)
245 else if(m_calAdcRes_charge_flag==1)
249 else if(m_calAdcRes_charge_flag==2)
254 adcRes = sqrt(adcRes_charge*adcRes_charge+adcRes_const*adcRes_const)/1000;
255 adc = G4RandGauss::shoot(adc, adcRes);
258 time_leading_sphi = tdc_sphi;
259 time_leading_xphi = tdc_xphi;
260 time_trailing_sphi = tdc_sphi+adc;
261 time_trailing_xphi = tdc_xphi+adc;
269 digi->
SetStrip(stripStruct[i].strip);
270 int mo = (partId-3)*36+module;
271 int st = stripStruct[i].
strip;
310 m_time_leading_sphi = time_leading_sphi;
311 m_time_leading_xphi = time_leading_xphi;
312 m_time_trailing_sphi = time_trailing_sphi;
313 m_time_trailing_xphi = time_trailing_xphi;
315 m_tdcRes_charge = tdcRes_charge;
318 m_adcRes_charge = adcRes_charge;
320 m_strip = stripStruct[i].
strip;
321 m_trkIndex = stripStruct[i].
trkIndex;
322 m_tStart = stripStruct[i].
tStart;
326 m_charge = stripStruct[i].
charge;
330 for(
int j=0; j<m_nhit; j++)
333 m_trkIndex_hit[j] = stripStruct[i].
hitStructCol[j].trkIndex;
334 m_pdgCode_hit[j] = stripStruct[i].
hitStructCol[j].pdgCode;
336 m_underStrip_hit[j] = stripStruct[i].
hitStructCol[j].underStrip;
356 int nstrip = m_param.
nstrip;
358 double length = locZ+stripWidth*nstrip/2-0.5;
363 else if(length<stripWidth*nstrip)
365 for(
int i=0; i<nstrip; i++)
367 if(length>i*stripWidth && length<(i+1)*stripWidth)
379 if(strip>nstrip-1) strip=nstrip-1;
389 int nstrip = m_param.
nstrip;
391 double length = locZ+stripWidth*nstrip/2-0.5;
392 if(length<stripWidth*nstrip)
394 for(
int i=0; i<nstrip; i++)
396 if(length>i*stripWidth && length<(i+1)*stripWidth)
399 if(length>i*stripWidth+m_param.
strip_gap/2 && length<(i+1)*stripWidth-m_param.
strip_gap/2
427 if(strip<0 || strip>m_param.
nstrip-1)
429 cout<<
"!! BesTofDigitizerEcV4::HitStruct::calTPropagate Wrong Strip !!!"<<endl;
434 double length_sphi = m_param.
strip_x[strip]/2-locx;
435 tPropagate_sphi =
abs(length_sphi)/v_propagate;
437 double length_xphi = m_param.
strip_x[strip]/2+locx;
438 tPropagate_xphi =
abs(length_xphi)/v_propagate;
447 if(gap>=0 && gap<m_param.
ngap/2) length = m_param.
gapWidth/2+locy;
448 else if(gap<m_param.
ngap) length = m_param.
gapWidth/2-locy;
451 cout<<
"BesTofDigitizerEcV4::StripStruct::calAvaLength Wrong gap calculation !!!"<<endl;
461 for(
unsigned int i=0; i<hitStructCol.size(); i++)
463 hitStructCol[i].ava_pos.clear();
464 hitStructCol[i].ava_num.clear();
466 hitStructCol[i].ava_pos[0] = hitStructCol[i].calAvaLength();
467 hitStructCol[i].ava_num[0] = hitStructCol[i].ions;
469 for(
int j=1; j<m_param.
nstep; j++)
471 hitStructCol[i].ava_pos[j] = hitStructCol[i].ava_pos[j-1] + m_param.
stepWidth;
472 if(saturationFlag==
true && hitStructCol[i].ava_num[j-1]>1.5e+07)
474 hitStructCol[i].ava_num[j] = hitStructCol[i].ava_num[j-1];
478 hitStructCol[i].ava_num[j] = calNextN(hitStructCol[i].ava_num[j-1]);
480 if(hitStructCol[i].ava_pos[j]>m_param.
gapWidth)
break;
485 bool over_threshold =
false;
487 for(
int i=0; i<m_param.
nstep; i++)
489 for(
unsigned int j=0; j<hitStructCol.size(); j++)
491 if(i<hitStructCol[j].ava_pos.size() && hitStructCol[j].ava_pos[i]<m_param.
gapWidth)
493 sum += hitStructCol[j].ava_num[i];
498 if(over_threshold==
false)
502 over_threshold =
true;
503 tThreshold = (m_param.
gapWidth)/(m_param.
nstep)/v_drift*(i+1);
518 for(
int i=0; i<
num; i++)
520 rdm = G4UniformRand();
521 nextN += multiply(rdm);
528 double sigma = calSigma();
529 double mean =
num*nbar;
530 double resolution = G4RandGauss::shoot(0,(sqrt(
num)*sigma));
531 long int nextN = mean+resolution;
539 double k = eta/
alpha;
540 double rdm_border = k*(nbar-1)/(nbar-k);
547 long int number = 1.+1./log((nbar-1.)/(nbar-k))*log((nbar-k)*(rdm-1)/(k-1)/nbar);
555 double k = eta/
alpha;
556 double sigma = sqrt((1+k)/(1-k)*nbar*(nbar-1));
565 time = 100.764*
exp(-charge_fC*0.0413966+0.377154)+ 13.814;
569 time = 12.8562+0.000507299*charge_fC;
578 time=-120.808/log(charge_fC*30.1306)+26.6024;
588 time = 72.6005*
exp(-charge_fC*0.0302626 + 1.49059) + 40.8627;
592 time = 32.6233+0.00404149*charge_fC;
603 result = 67.6737*
exp(-charge_fC/50.9995-0.27755)+9.06223;
607 result = 176.322-2.98345*charge_fC;
609 if(result<0) result=0;
616 time=-4.50565/log(charge_fC*0.0812208)+16.6653;
623 double time = 64.3326*
exp(-charge_fC/25.4638 + 0.944184)+19.4846;
652 v_propagate = 0.5*0.299792458e+3;
653 tPropagate_sphi = -999.0;
654 tPropagate_xphi = -999.0;
668 tPropagate_sphi = -999.0;
669 tPropagate_xphi = -999.0;
675 alpha = 144800./1000;
680 hitStructCol.clear();
685 threshold = threshold_n;
686 E = E_V/1000*2/6/(m_param.
gapWidth/10);
691 saturationFlag = saturationFlag_n;
696 if(nstep_n>0) nstep = nstep_n;
697 if(E_weight_n>0) E_weight = E_weight_n;
706 std::stringstream ss;
707 for(
int i=0; i<nstrip; i++)
710 ss<<
"strip_x["<<i<<
"]";
711 strip_x[i] = tofPara->Get(ss.str().c_str());
713 strip_z = tofPara->Get(
"strip_z");
714 strip_gap = tofPara->Get(
"strip_gap");
719 stepWidth = gapWidth/nstep;
720 E_weight = 1./(6.*0.22+(5.*0.4+2*0.55)/3.7);
721 eCharge = 1.60217733e-7;
722 tofPara->Get_deadChannel(deadChannel);
784 TSpline3* sp_alpha =
new TSpline3(
"sp_alpha", e,
alpha, 22);
785 double alphaVal = sp_alpha->Eval(E);
846 TSpline3* sp_eta =
new TSpline3(
"sp_eta", e, eta, 22);
847 double etaVal = sp_eta->Eval(E);
908 TSpline3* sp_v =
new TSpline3(
"sp_v", e,
v, 22);
909 double vVal = sp_v->Eval(E);
915 cout<<
"Fixed parameters: "<<endl;
916 for(
int i=0; i<nstrip; i++)
918 cout<<
" strip_x["<<i<<
"]= "<<strip_x[i];
920 for(
int i=0; i<nmodule; i++)
922 for(
int j=0; j<nstrip; j++)
924 cout<<
" deadChannel["<<i<<
"]["<<j<<
"]= "<<deadChannel[i][j];
928 cout<<
" strip_z= "<<strip_z
929 <<
" strip_gap= "<<strip_gap
931 <<
" gapWidth= "<<gapWidth
933 <<
" stepWidth= "<<stepWidth
934 <<
" E_weight= "<<E_weight
935 <<
" eCharge= "<<eCharge
941 cout<<
"Hit information: "<<endl;
942 cout<<
" trkIndex= "<<trkIndex
943 <<
" pdgCode= "<<pdgCode
947 <<
" glbTime= "<<glbTime
957 <<
" v_propagate= "<<v_propagate
958 <<
" tPropagate_sphi= "<<tPropagate_sphi
959 <<
" tPropagate_xphi= "<<tPropagate_xphi
965 cout<<
"Strip information: "<<endl;
966 cout<<
" strip= "<<strip
967 <<
" trkIndex= "<<trkIndex
968 <<
" tStart= "<<tStart
969 <<
" tPropagate_sphi= "<<tPropagate_sphi
970 <<
" tPropagate_xphi= "<<tPropagate_xphi
971 <<
" tThreshold "<<tThreshold
972 <<
" charge= "<<charge
976 <<
" threshold= "<<threshold
977 <<
" v_drift= "<<v_drift
EvtComplex exp(const EvtComplex &c)
G4TDigiCollection< BesTofDigi > BesTofDigitsCollection
G4THitsCollection< BesTofHit > BesTofHitsCollection
**********Class see also m_nmax DOUBLE PRECISION m_amel DOUBLE PRECISION m_x2 DOUBLE PRECISION m_alfinv DOUBLE PRECISION m_Xenph INTEGER m_KeyWtm INTEGER m_idyfs DOUBLE PRECISION m_zini DOUBLE PRECISION m_q2 DOUBLE PRECISION m_Wt_KF DOUBLE PRECISION m_WtCut INTEGER m_KFfin *COMMON c_KarLud $ !Input CMS energy[GeV] $ !CMS energy after beam spread beam strahlung[GeV] $ !Beam energy spread[GeV] $ !z boost due to beam spread $ !electron beam mass *ff pair spectrum $ !minimum v
void SetForwT1(G4double t1)
void SetPartId(G4int partId)
void SetTrackIndex(G4int index)
void SetForwT2(G4double t2)
void SetBackT1(G4double t1)
void SetModule(G4int module)
void SetBackT2(G4double t2)
void SetStrip(G4int strip)
double calTdcRes_charge1(double charge_fC)
double calAdcRes_charge(double charge_fC)
double calAdcRes_charge1(double charge_fC)
bool underStrip(double locX, double locZ)
int calStrip(double locZ)
virtual void Digitize(ScintSingle *, BesTofDigitsCollection *)
double charge2Time1(double charge_fC)
double charge2Time(double charge_fC)
double calTdcRes_charge(double charge_fC)
BesTofHitsCollection * m_THC
BesTofDigitsCollection * m_besTofDigitsCollection
static BesTofGeoParameter * GetInstance()
G4ThreeVector GetLocPos()
G4ThreeVector GetMomentum()
vector< G4int > * GetHitIndexes_mrpc()
void setPar(int nstep, double E_weight)
vector< HitStruct > hitStructCol
double getAlpha(double E)
long int calNextN(int num)
long int multiply(double rdm)
void setPar(double V, double threshold, bool saturationFlag=true)