00001
00002
00003
00004 #include "GaudiKernel/ToolFactory.h"
00005
00006 #include "GaudiKernel/IDataProviderSvc.h"
00007
00008
00009 #include "Event/State.h"
00010 #include "Event/TrackParameters.h"
00011
00012
00013 #include "TrackTools/TrueStateCreator.h"
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024 static const ToolFactory<TrueStateCreator> s_factory;
00025 const IToolFactory& TrueStateCreatorFactory = s_factory;
00026
00027
00028
00029
00030 TrueStateCreator::TrueStateCreator( const std::string& type,
00031 const std::string& name,
00032 const IInterface* parent )
00033 : GaudiTool( type, name, parent )
00034 , m_velo(0)
00035 , m_p2VeloHitAsct(0)
00036 , m_p2ITHitAsct(0)
00037 , m_p2OTHitAsct(0)
00038 , m_extrapolator(0)
00039 , m_p2VeloHitAsctName("")
00040 , m_p2ITHitAsctName("")
00041 , m_p2OTHitAsctName("")
00042 , m_eX2(2.e-5*mm2)
00043 , m_eY2(2.e-5*mm2)
00044 , m_eTx2(1.e-7)
00045 , m_eTy2(1.e-7)
00046 , m_eP(0.005)
00047 {
00048
00049 declareInterface<IStateCreator>(this);
00050
00051
00052 declareProperty( "MCP2VeloMCHitAscName",
00053 m_p2VeloHitAsctName = "MCP2VeloMCHitAsc" );
00054 declareProperty( "MCP2ITMCHitAscName",
00055 m_p2ITHitAsctName = "MCP2ITMCHitAsc" );
00056 declareProperty( "MCP2OTMCHitAscName",
00057 m_p2OTHitAsctName = "MCP2OTMCHitAsc" );
00058 declareProperty( "extrapolatorName",
00059 m_extrapolatorName = "TrFirstCleverExtrapolator" );
00060 declareProperty( "eX2", m_eX2 );
00061 declareProperty( "eY2", m_eY2 );
00062 declareProperty( "eTx2", m_eTx2 );
00063 declareProperty( "eTy2", m_eTy2 );
00064 declareProperty( "eP", m_eP );
00065 }
00066
00067
00068
00069
00070 TrueStateCreator::~TrueStateCreator() {};
00071
00072
00073
00074
00075 StatusCode TrueStateCreator::initialize()
00076 {
00078 StatusCode sc;
00079
00080 std::string ascType = "Associator<MCParticle,MCHit>";
00081 std::string ascVeloType = "Associator<MCParticle,MCVeloHit>";
00082
00083
00084 sc = toolSvc() -> retrieveTool( ascVeloType,
00085 m_p2VeloHitAsctName, m_p2VeloHitAsct );
00086 if ( sc.isFailure() ) {
00087 error() << "Unable to retrieve the Velo MCHit Associator "
00088 << m_p2VeloHitAsctName << endreq;
00089 return sc;
00090 }
00091
00092
00093 sc = toolSvc() -> retrieveTool( ascType, m_p2OTHitAsctName, m_p2OTHitAsct );
00094 if ( sc.isFailure() ) {
00095 error() << "Unable to retrieve the OT MCHit Associator "
00096 << m_p2OTHitAsctName << endreq;
00097 return sc;
00098 }
00099
00100
00101 sc = toolSvc()->retrieveTool(ascType, m_p2ITHitAsctName, m_p2ITHitAsct);
00102 if ( sc.isFailure() ) {
00103 error() << "Unable to retrieve the IT MCHit Associator "
00104 << m_p2ITHitAsctName << endreq;
00105 return sc;
00106 }
00107
00108
00109 m_extrapolator = tool<ITrackExtrapolator>( m_extrapolatorName );
00110
00111 return StatusCode::SUCCESS;
00112 }
00113
00114
00115
00116
00117
00118 StatusCode TrueStateCreator::createState( const MCParticle* mcPart,
00119 double zRec,
00120 State*& state ) const {
00121
00122 HepSymMatrix stateCov = HepSymMatrix(5, 1);
00123 State* pState = new State();
00124 pState -> setZ( zRec );
00125 pState -> setCovariance( stateCov );
00126 state = pState;
00127
00128
00129 if ( mcPart == 0 ) return StatusCode::FAILURE;
00130
00131
00132 if ( !m_p2VeloHitAsct || !m_p2OTHitAsct || !m_p2ITHitAsct )
00133 return Error( "MCParticle 2 MCHit Associator missing." );
00134
00135 MCHit* closestHit = 0;
00136 MCHit* secondClosestHit = 0;
00137 double closestZ = 10000;
00138 double secondClosestZ = 10000;
00139
00140
00141 MCVeloHitAsct::ToRange mcVeloHitsRange = m_p2VeloHitAsct->rangeFrom(mcPart);
00142 MCVeloHitAsct::ToIterator vt;
00143 for ( vt = mcVeloHitsRange.begin(); vt != mcVeloHitsRange.end(); ++vt) {
00144
00145 MCVeloHit* mcVeloHit = vt->to();
00146 if ( !mcVeloHit )
00147 return Error( "Failed retrieving Velo MCHit" );
00148
00149
00150
00151 HepPoint3D midPointTmp = ( mcVeloHit->entry() + mcVeloHit->exit() ) / 2.0;
00152 Hep3Vector midPoint = ( midPointTmp.x(), midPointTmp.y(), midPointTmp.z() );
00153
00154
00155
00156 if ( fabs(midPoint.z() - zRec) < closestZ ) {
00157 secondClosestHit = closestHit;
00158 secondClosestZ = closestZ;
00159 closestHit = mcVeloHit;
00160 closestZ = fabs(midPoint.z()- zRec);
00161 }
00162 else if ( fabs(midPoint.z()- zRec) < secondClosestZ ) {
00163 secondClosestHit = mcVeloHit;
00164 secondClosestZ = fabs(midPoint.z()- zRec);
00165 }
00166 }
00167
00168
00169 MCHitAsct::ToRange mcOTHitsRange = m_p2OTHitAsct->rangeFrom(mcPart);
00170 MCHitAsct::ToIterator ot;
00171 for ( ot = mcOTHitsRange.begin(); ot != mcOTHitsRange.end(); ++ot) {
00172
00173 MCHit* mcHit = ot->to();
00174 if ( !mcHit )
00175 return Error( "Failed retrieving OT MCHit" );
00176
00177
00178
00179 HepPoint3D midPointTmp = ( mcHit->entry() + mcHit->exit() ) / 2.0;
00180 Hep3Vector midPoint = ( midPointTmp.x(), midPointTmp.y(), midPointTmp.z() );
00181
00182
00183
00184 if ( fabs(midPoint.z() - zRec) < closestZ ) {
00185 secondClosestHit = closestHit;
00186 secondClosestZ = closestZ;
00187 closestHit = mcHit;
00188 closestZ = fabs(midPoint.z()- zRec);
00189 }
00190 else if ( fabs(midPoint.z()- zRec) < secondClosestZ ) {
00191 secondClosestHit = mcHit;
00192 secondClosestZ = fabs(midPoint.z()- zRec);
00193 }
00194 }
00195
00196
00197 MCHitAsct::ToRange mcITHitsRange = m_p2ITHitAsct->rangeFrom(mcPart);
00198 MCHitAsct::ToIterator it;
00199 for ( it = mcITHitsRange.begin(); it != mcITHitsRange.end(); ++it) {
00200
00201 MCHit* mcHit = it->to();
00202 if ( !mcHit )
00203 return Error( "Failed retrieving IT MCHit" );
00204
00205
00206
00207 HepPoint3D midPointTmp = ( mcHit->entry() + mcHit->exit() ) / 2.0;
00208 Hep3Vector midPoint = ( midPointTmp.x(), midPointTmp.y(), midPointTmp.z() );
00209
00210
00211
00212 if ( fabs(midPoint.z()- zRec) < closestZ ) {
00213 secondClosestHit = closestHit;
00214 secondClosestZ = closestZ;
00215 closestHit = mcHit;
00216 closestZ = fabs(midPoint.z()- zRec);
00217 }
00218 else if ( fabs(midPoint.z()- zRec) < secondClosestZ ) {
00219 secondClosestHit = mcHit;
00220 secondClosestZ = fabs(midPoint.z()- zRec);
00221 }
00222
00223 }
00224
00225 if ( !closestHit || !secondClosestHit ) {
00226 warning() << "No two closest MCHits found!!" << endreq;
00227 return StatusCode::FAILURE;
00228 }
00229
00230
00231 Hep3Vector beginPoint = closestHit->entry();
00232 if ( beginPoint.z() > closestHit->exit().z() )
00233 beginPoint = closestHit->exit();
00234 if ( beginPoint.z() > secondClosestHit->entry().z() )
00235 beginPoint = secondClosestHit->entry();
00236 if ( beginPoint.z() > secondClosestHit->exit().z() )
00237 beginPoint = secondClosestHit->exit();
00238
00239
00240 Hep3Vector endPoint = secondClosestHit->exit();
00241 if ( endPoint.z() < secondClosestHit->entry().z() )
00242 endPoint = secondClosestHit->entry();
00243 if ( endPoint.z() < closestHit->entry().z() )
00244 endPoint = closestHit->entry();
00245 if ( endPoint.z() < closestHit->exit().z() )
00246 endPoint = closestHit->exit();
00247
00248 double dz = (beginPoint.z() - endPoint.z());
00249 if ( dz == 0. ) {
00250 warning() << "Delta z between two hits equals zero." << endreq;
00251 return StatusCode::FAILURE;
00252 }
00253 double slopeX = (beginPoint.x() - endPoint.x()) / dz;
00254 double slopeY = (beginPoint.y() - endPoint.y()) / dz;
00255 double x = (beginPoint.x());
00256 double y = (beginPoint.y());
00257 double z = (beginPoint.z());
00258
00259
00260 pState->setState( x, y, z, slopeX, slopeY, this->qOverP(mcPart) );
00261
00262
00263 HepSymMatrix cov = HepSymMatrix(5,0);
00264 cov.fast(1,1) = m_eX2;
00265 cov.fast(2,2) = m_eY2;
00266 cov.fast(3,3) = m_eTx2;
00267 cov.fast(4,4) = m_eTy2;
00268 cov.fast(5,5) = pow(m_eP * pState->qOverP(), 2.);
00269 pState -> setCovariance( cov );
00270
00271
00272 ParticleID partID = 211;
00273 StatusCode sc = m_extrapolator -> propagate( *pState, zRec, partID );
00274 if ( sc.isFailure() ) {
00275 debug() << "Extrapolation of True State from z = "
00276 << z << " to z = " << zRec << " failed!" << endreq;
00277 }
00278
00279 return sc;
00280 }
00281
00282
00283
00284
00285
00286 StatusCode TrueStateCreator::createStateVertex( const MCParticle* mcParticle,
00287 State*& state ) const
00288 {
00290 HepSymMatrix stateCov = HepSymMatrix(5, 1);
00291 State* trueState = new State();
00292 trueState -> setCovariance( stateCov );
00293 state = trueState;
00294
00295
00296 if ( mcParticle == 0 ) return StatusCode::FAILURE;
00297
00298
00299 const MCVertex* mcVertex = mcParticle -> originVertex();
00300 const HepPoint3D mcPos = mcVertex -> position();
00301 const HepLorentzVector mc4Mom = mcParticle -> momentum();
00302
00303
00304 double trueQdivP = this -> qOverP( mcParticle );
00305
00306
00307 double trueTx = 99999999;
00308 double trueTy = 99999999;
00309 double px = mc4Mom.px();
00310 double py = mc4Mom.py();
00311 double pz = mc4Mom.pz();
00312 if ( fabs(pz) > TrackParameters::lowTolerance ) {
00313 trueTx = px/pz;
00314 trueTy = py/pz;
00315 }
00316 else {
00317 warning() << "No momentum z component." << endreq;
00318 return StatusCode::FAILURE;
00319 }
00320
00321
00322 trueState->setState( mcPos.x(),
00323 mcPos.y(),
00324 mcPos.z(),
00325 trueTx,
00326 trueTy,
00327 trueQdivP );
00328
00329
00330 HepSymMatrix cov = HepSymMatrix(5,0);
00331 cov.fast(1,1) = m_eX2;
00332 cov.fast(2,2) = m_eY2;
00333 cov.fast(3,3) = m_eTx2;
00334 cov.fast(4,4) = m_eTy2;
00335 cov.fast(5,5) = pow(m_eP * trueState->qOverP(), 2.);
00336 trueState -> setCovariance( cov );
00337
00338 return StatusCode::SUCCESS;
00339 }
00340
00341
00342
00343
00344 double TrueStateCreator::qOverP( const MCParticle* mcPart ) const
00345 {
00347 double momentum = mcPart -> momentum().vect().mag();
00348 double charge = (mcPart -> particleID().threeCharge())/3.;
00349 if ( momentum > TrackParameters::lowTolerance) {
00350 return charge/momentum;
00351 }
00352 else {
00353 return 0.0;
00354 }
00355 }
00356
00357