00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011 #include "RecoMuon/MuonSeedGenerator/src/RPCSeedPattern.h"
00012 #include <RecoMuon/TrackingTools/interface/MuonPatternRecoDumper.h>
00013 #include <MagneticField/Engine/interface/MagneticField.h>
00014 #include <MagneticField/Records/interface/IdealMagneticFieldRecord.h>
00015 #include <TrackingTools/TrajectoryState/interface/TrajectoryStateTransform.h>
00016 #include <TrackingTools/DetLayers/interface/DetLayer.h>
00017 #include <DataFormats/TrajectoryState/interface/PTrajectoryStateOnDet.h>
00018 #include <DataFormats/Common/interface/OwnVector.h>
00019 #include <DataFormats/MuonDetId/interface/RPCDetId.h>
00020 #include <FWCore/Framework/interface/ESHandle.h>
00021 #include <FWCore/MessageLogger/interface/MessageLogger.h>
00022 #include <Geometry/CommonDetUnit/interface/GeomDet.h>
00023 #include <Geometry/RPCGeometry/interface/RPCChamber.h>
00024 #include <Geometry/RPCGeometry/interface/RPCGeometry.h>
00025 #include <Geometry/Records/interface/MuonGeometryRecord.h>
00026
00027 #include "gsl/gsl_statistics.h"
00028 #include "TH1F.h"
00029 #include "math.h"
00030
00031 using namespace std;
00032 using namespace edm;
00033
00034
00035 RPCSeedPattern::RPCSeedPattern() {
00036
00037 isPatternChecked = false;
00038 isConfigured = false;
00039 MagnecticFieldThreshold = 0.5;
00040 }
00041
00042 RPCSeedPattern::~RPCSeedPattern() {
00043
00044 }
00045
00046 void RPCSeedPattern::configure(const edm::ParameterSet& iConfig) {
00047
00048 MaxRSD = iConfig.getParameter<double>("MaxRSD");
00049 deltaRThreshold = iConfig.getParameter<double>("deltaRThreshold");
00050 AlgorithmType = iConfig.getParameter<unsigned int>("AlgorithmType");
00051 autoAlgorithmChoose = iConfig.getParameter<bool>("autoAlgorithmChoose");
00052 ZError = iConfig.getParameter<double>("ZError");
00053 MinDeltaPhi = iConfig.getParameter<double>("MinDeltaPhi");
00054 MagnecticFieldThreshold = iConfig.getParameter<double>("MagnecticFieldThreshold");
00055 stepLength = iConfig.getParameter<double>("stepLength");
00056 sampleCount = iConfig.getParameter<unsigned int>("sampleCount");
00057 isConfigured = true;
00058 }
00059
00060 RPCSeedPattern::weightedTrajectorySeed RPCSeedPattern::seed(const edm::EventSetup& eSetup, int& isGoodSeed) {
00061
00062 if(isConfigured == false)
00063 {
00064 cout << "Configuration not set yet" << endl;
00065 return createFakeSeed(isGoodSeed, eSetup);
00066 }
00067
00068
00069 unsigned int NumberofHitsinSeed = nrhit();
00070 if(NumberofHitsinSeed < 3)
00071 return createFakeSeed(isGoodSeed, eSetup);
00072
00073 if(NumberofHitsinSeed == 3)
00074 ThreePointsAlgorithm();
00075
00076 if(NumberofHitsinSeed > 3)
00077 {
00078 if(autoAlgorithmChoose == false)
00079 {
00080 cout << "computePtWithmorerecHits" << endl;
00081 if(AlgorithmType == 0)
00082 ThreePointsAlgorithm();
00083 if(AlgorithmType == 1)
00084 MiddlePointsAlgorithm();
00085 if(AlgorithmType == 2)
00086 SegmentAlgorithm();
00087 if(AlgorithmType == 3)
00088 {
00089 if(checkSegment())
00090 SegmentAlgorithmSpecial(eSetup);
00091 else
00092 {
00093 cout << "Not enough recHits for Special Segment Algorithm" << endl;
00094 return createFakeSeed(isGoodSeed, eSetup);
00095 }
00096 }
00097 }
00098 else
00099 {
00100 if(checkSegment())
00101 {
00102 AlgorithmType = 3;
00103 SegmentAlgorithmSpecial(eSetup);
00104 }
00105 else
00106 {
00107 AlgorithmType = 2;
00108 SegmentAlgorithm();
00109 }
00110 }
00111 }
00112
00113
00114 if(isPatternChecked == false){
00115 if(AlgorithmType != 3){
00116 checkSimplePattern(eSetup);
00117 } else {
00118 checkSegmentAlgorithmSpecial(eSetup);
00119 }
00120 }
00121
00122 return createSeed(isGoodSeed, eSetup);
00123 }
00124
00125 void RPCSeedPattern::ThreePointsAlgorithm()
00126 {
00127 cout << "computePtWith3recHits" << endl;
00128 unsigned int NumberofHitsinSeed = nrhit();
00129
00130 if(NumberofHitsinSeed < 3)
00131 {
00132 isPatternChecked = true;
00133 isGoodPattern = -1;
00134 return;
00135 }
00136
00137 unsigned int NumberofPart = NumberofHitsinSeed * (NumberofHitsinSeed - 1) * (NumberofHitsinSeed - 2) / (3 * 2);;
00138 double *pt = new double[NumberofPart];
00139 double *pt_err = new double[NumberofPart];
00140
00141 ConstMuonRecHitPointer precHit[3];
00142 unsigned int n = 0;
00143 unsigned int NumberofStraight = 0;
00144 for(unsigned int i = 0; i < (NumberofHitsinSeed - 2); i++)
00145 for(unsigned int j = (i + 1); j < (NumberofHitsinSeed - 1); j++)
00146 for(unsigned int k = (j + 1); k < NumberofHitsinSeed; k++)
00147 {
00148 precHit[0] = theRecHits[i];
00149 precHit[1] = theRecHits[j];
00150 precHit[2] = theRecHits[k];
00151 bool checkStraight = checkStraightwithThreerecHits(precHit, MinDeltaPhi);
00152 if(!checkStraight)
00153 {
00154 GlobalVector Center_temp = computePtwithThreerecHits(pt[n], pt_err[n], precHit);
00155
00156 Center += Center_temp;
00157 }
00158 else
00159 {
00160
00161 NumberofStraight++;
00162 pt[n] = upper_limit_pt;
00163 pt_err[n] = 0;
00164 }
00165 n++;
00166 }
00167
00168 if(NumberofStraight == NumberofPart)
00169 {
00170 isStraight = true;
00171 meanRadius = -1;
00172 }
00173 else
00174 {
00175 isStraight = false;
00176 Center /= (NumberofPart - NumberofStraight);
00177 double meanR = 0.;
00178 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00179 meanR += getDistance(*iter, Center);
00180 meanR /= NumberofHitsinSeed;
00181 meanRadius = meanR;
00182 }
00183
00184
00185 isPatternChecked = false;
00186
00187
00188
00189
00190
00191 delete [] pt;
00192 delete [] pt_err;
00193 }
00194
00195 void RPCSeedPattern::MiddlePointsAlgorithm()
00196 {
00197 cout << "Using middle points algorithm" << endl;
00198 unsigned int NumberofHitsinSeed = nrhit();
00199
00200 if(NumberofHitsinSeed < 4)
00201 {
00202 isPatternChecked = true;
00203 isGoodPattern = -1;
00204 return;
00205 }
00206 double *X = new double[NumberofHitsinSeed];
00207 double *Y = new double[NumberofHitsinSeed];
00208 unsigned int n = 0;
00209 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00210 {
00211 X[n] = (*iter)->globalPosition().x();
00212 Y[n] = (*iter)->globalPosition().y();
00213 cout << "X[" << n <<"] = " << X[n] << ", Y[" << n <<"]= " << Y[n] << endl;
00214 n++;
00215 }
00216 unsigned int NumberofPoints = NumberofHitsinSeed;
00217 while(NumberofPoints > 3)
00218 {
00219 for(unsigned int i = 0; i <= (NumberofPoints - 2); i++)
00220 {
00221 X[i] = (X[i] + X[i+1]) / 2;
00222 Y[i] = (Y[i] + Y[i+1]) / 2;
00223 }
00224 NumberofPoints--;
00225 }
00226 double x[3], y[3];
00227 for(unsigned int i = 0; i < 3; i++)
00228 {
00229 x[i] = X[i];
00230 y[i] = Y[i];
00231 }
00232 double pt = 0;
00233 double pt_err = 0;
00234 bool checkStraight = checkStraightwithThreerecHits(x, y, MinDeltaPhi);
00235 if(!checkStraight)
00236 {
00237
00238 GlobalVector Center_temp = computePtWithThreerecHits(pt, pt_err, x, y);
00239 double meanR = 0.;
00240 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00241 meanR += getDistance(*iter, Center_temp);
00242 meanR /= NumberofHitsinSeed;
00243
00244 isStraight = false;
00245 Center = Center_temp;
00246 meanRadius = meanR;
00247 }
00248 else
00249 {
00250
00251 isStraight = true;
00252 meanRadius = -1;
00253 }
00254
00255
00256 isPatternChecked = false;
00257
00258 delete [] X;
00259 delete [] Y;
00260 }
00261
00262 void RPCSeedPattern::SegmentAlgorithm()
00263 {
00264 cout << "Using segments algorithm" << endl;
00265 unsigned int NumberofHitsinSeed = nrhit();
00266
00267 if(NumberofHitsinSeed < 4)
00268 {
00269 isPatternChecked = true;
00270 isGoodPattern = -1;
00271 return;
00272 }
00273
00274 RPCSegment* Segment;
00275 unsigned int NumberofSegment = NumberofHitsinSeed - 2;
00276 Segment = new RPCSegment[NumberofSegment];
00277 unsigned int n = 0;
00278 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != (theRecHits.end()-2); iter++)
00279 {
00280 Segment[n].first = (*iter);
00281 Segment[n].second = (*(iter + 2));
00282 n++;
00283 }
00284 unsigned int NumberofStraight = 0;
00285 for(unsigned int i = 0; i < NumberofSegment - 1; i++)
00286 {
00287 bool checkStraight = checkStraightwithSegment(Segment[i], Segment[i+1], MinDeltaPhi);
00288 if(checkStraight == true)
00289 {
00290
00291 NumberofStraight++;
00292 }
00293 else
00294 {
00295 GlobalVector Center_temp = computePtwithSegment(Segment[i], Segment[i+1]);
00296
00297 Center += Center_temp;
00298 }
00299 }
00300
00301 if((NumberofSegment-1-NumberofStraight) > 0)
00302 {
00303 isStraight = false;
00304 Center /= (NumberofSegment - 1 - NumberofStraight);
00305 double meanR = 0.;
00306 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00307 meanR += getDistance(*iter, Center);
00308 meanR /= NumberofHitsinSeed;
00309 meanRadius = meanR;
00310 }
00311 else
00312 {
00313 isStraight = true;
00314 meanRadius = -1;
00315 }
00316
00317
00318 isPatternChecked = false;
00319
00320 delete [] Segment;
00321 }
00322
00323 void RPCSeedPattern::SegmentAlgorithmSpecial(const edm::EventSetup& eSetup)
00324 {
00325
00326 edm::ESHandle<MagneticField> Field;
00327 eSetup.get<IdealMagneticFieldRecord>().get(Field);
00328
00329
00330 if(!checkSegment())
00331 {
00332 isPatternChecked = true;
00333 isGoodPattern = -1;
00334 return;
00335 }
00336
00337
00338 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != (theRecHits.end()-1); iter++)
00339 {
00340 GlobalPoint gpFirst = (*iter)->globalPosition();
00341 GlobalPoint gpLast = (*(iter+1))->globalPosition();
00342 GlobalPoint* gp = new GlobalPoint[sampleCount];
00343 double dx = (gpLast.x() - gpFirst.x()) / (sampleCount + 1);
00344 double dy = (gpLast.y() - gpFirst.y()) / (sampleCount + 1);
00345 double dz = (gpLast.z() - gpFirst.z()) / (sampleCount + 1);
00346 for(unsigned int index = 0; index < sampleCount; index++)
00347 {
00348 gp[index] = GlobalPoint((gpFirst.x()+dx*(index+1)), (gpFirst.y()+dy*(index+1)), (gpFirst.z()+dz*(index+1)));
00349 GlobalVector MagneticVec_temp = Field->inTesla(gp[index]);
00350 cout << "Sampling magnetic field : " << MagneticVec_temp << endl;
00351
00352 }
00353 delete [] gp;
00354 }
00355
00356
00357 ConstMuonRecHitContainer::const_iterator iter=theRecHits.begin();
00358 for(unsigned int n = 0; n <= 1; n++)
00359 {
00360 SegmentRB[n].first = (*iter);
00361 cout << "SegmentRB " << n << " recHit: " << (*iter)->globalPosition() << endl;
00362 iter++;
00363 SegmentRB[n].second = (*iter);
00364 cout << "SegmentRB " << n << " recHit: " << (*iter)->globalPosition() << endl;
00365 iter++;
00366 }
00367 GlobalVector segvec1 = (SegmentRB[0].second)->globalPosition() - (SegmentRB[0].first)->globalPosition();
00368 GlobalVector segvec2 = (SegmentRB[1].second)->globalPosition() - (SegmentRB[1].first)->globalPosition();
00369
00370
00371 entryPosition = (SegmentRB[0].second)->globalPosition();
00372 leavePosition = (SegmentRB[1].first)->globalPosition();
00373 while(fabs(Field->inTesla(entryPosition).z()) < MagnecticFieldThreshold)
00374 {
00375 cout << "Entry position is : " << entryPosition << ", and stepping into next point" << endl;
00376 entryPosition += segvec1.unit() * stepLength;
00377 }
00378
00379 while(fabs(Field->inTesla(entryPosition).z()) >= MagnecticFieldThreshold)
00380 {
00381 cout << "Entry position is : " << entryPosition << ", and stepping back into next point" << endl;
00382 entryPosition -= segvec1.unit() * stepLength / 10;
00383 }
00384 entryPosition += 0.5 * segvec1.unit() * stepLength / 10;
00385 cout << "Final entry position is : " << entryPosition << endl;
00386
00387 while(fabs(Field->inTesla(leavePosition).z()) < MagnecticFieldThreshold)
00388 {
00389 cout << "Leave position is : " << leavePosition << ", and stepping into next point" << endl;
00390 leavePosition -= segvec2.unit() * stepLength;
00391 }
00392
00393 while(fabs(Field->inTesla(leavePosition).z()) >= MagnecticFieldThreshold)
00394 {
00395 cout << "Leave position is : " << leavePosition << ", and stepping back into next point" << endl;
00396 leavePosition += segvec2.unit() * stepLength / 10;
00397 }
00398 leavePosition -= 0.5 * segvec2.unit() * stepLength / 10;
00399 cout << "Final leave position is : " << leavePosition << endl;
00400
00401
00402 GlobalPoint* gp = new GlobalPoint[sampleCount];
00403 double dx = (leavePosition.x() - entryPosition.x()) / (sampleCount + 1);
00404 double dy = (leavePosition.y() - entryPosition.y()) / (sampleCount + 1);
00405 double dz = (leavePosition.z() - entryPosition.z()) / (sampleCount + 1);
00406 std::vector<GlobalVector> BValue;
00407 BValue.clear();
00408 for(unsigned int index = 0; index < sampleCount; index++)
00409 {
00410 gp[index] = GlobalPoint((entryPosition.x()+dx*(index+1)), (entryPosition.y()+dy*(index+1)), (entryPosition.z()+dz*(index+1)));
00411 GlobalVector MagneticVec_temp = Field->inTesla(gp[index]);
00412 cout << "Sampling magnetic field : " << MagneticVec_temp << endl;
00413 BValue.push_back(MagneticVec_temp);
00414 }
00415 delete [] gp;
00416 GlobalVector meanB2(0, 0, 0);
00417 for(std::vector<GlobalVector>::const_iterator BIter = BValue.begin(); BIter != BValue.end(); BIter++)
00418 meanB2 += (*BIter);
00419 meanB2 /= BValue.size();
00420 cout << "Mean B field is " << meanB2 << endl;
00421 meanMagneticField2 = meanB2;
00422
00423 double meanBz2 = meanB2.z();
00424 double deltaBz2 = 0.;
00425 for(std::vector<GlobalVector>::const_iterator BIter = BValue.begin(); BIter != BValue.end(); BIter++)
00426 deltaBz2 += (BIter->z() - meanBz2) * (BIter->z() - meanBz2);;
00427 deltaBz2 /= BValue.size();
00428 deltaBz2 = sqrt(deltaBz2);
00429 cout<< "delta Bz is " << deltaBz2 << endl;
00430
00431
00432 S = 0;
00433 bool checkStraight = checkStraightwithSegment(SegmentRB[0], SegmentRB[1], MinDeltaPhi);
00434 if(checkStraight == true)
00435 {
00436
00437 isStraight2 = checkStraight;
00438 Center2 = GlobalVector(0, 0, 0);
00439 meanRadius2 = -1;
00440 GlobalVector MomentumVec = (SegmentRB[1].second)->globalPosition() - (SegmentRB[0].first)->globalPosition();
00441 S += MomentumVec.perp();
00442 lastPhi = MomentumVec.phi().value();
00443 }
00444 else
00445 {
00446 GlobalVector seg1 = entryPosition - (SegmentRB[0].first)->globalPosition();
00447 S += seg1.perp();
00448 GlobalVector seg2 = (SegmentRB[1].second)->globalPosition() - leavePosition;
00449 S += seg2.perp();
00450 GlobalVector vecZ(0, 0, 1);
00451 GlobalVector gvec1 = seg1.cross(vecZ);
00452 GlobalVector gvec2 = seg2.cross(vecZ);
00453 double A1 = gvec1.x();
00454 double B1 = gvec1.y();
00455 double A2 = gvec2.x();
00456 double B2 = gvec2.y();
00457 double X1 = entryPosition.x();
00458 double Y1 = entryPosition.y();
00459 double X2 = leavePosition.x();
00460 double Y2 = leavePosition.y();
00461 double XO = (A1*A2*(Y2-Y1)+A2*B1*X1-A1*B2*X2)/(A2*B1-A1*B2);
00462 double YO = (B1*B2*(X2-X1)+B2*A1*Y1-B1*A2*Y2)/(B2*A1-B1*A2);
00463 GlobalVector Center_temp(XO, YO, 0);
00464
00465 isStraight2 = checkStraight;
00466 Center2 = Center_temp;
00467
00468 cout << "entryPosition: " << entryPosition << endl;
00469 cout << "leavePosition: " << leavePosition << endl;
00470 cout << "Center2 is : " << Center_temp << endl;
00471
00472 double R1 = GlobalVector((entryPosition.x() - Center_temp.x()), (entryPosition.y() - Center_temp.y()), (entryPosition.z() - Center_temp.z())).perp();
00473 double R2 = GlobalVector((leavePosition.x() - Center_temp.x()), (leavePosition.y() - Center_temp.y()), (leavePosition.z() - Center_temp.z())).perp();
00474 double meanR = (R1 + R2) / 2;
00475 double deltaR = sqrt(((R1-meanR)*(R1-meanR)+(R2-meanR)*(R2-meanR))/2);
00476 meanRadius2 = meanR;
00477 cout << "R1 is " << R1 << ", R2 is " << R2 << endl;
00478 cout << "Mean radius is " << meanR << endl;
00479 cout << "Delta R is " << deltaR << endl;
00480 double deltaPhi = fabs(((leavePosition-GlobalPoint(XO, YO, 0)).phi()-(entryPosition-GlobalPoint(XO, YO, 0)).phi()).value());
00481 S += meanR * deltaPhi;
00482 lastPhi = seg2.phi().value();
00483 }
00484
00485
00486 isPatternChecked = false;
00487 }
00488
00489 bool RPCSeedPattern::checkSegment() const
00490 {
00491 bool isFit = true;
00492 unsigned int count = 0;
00493
00494 for(ConstMuonRecHitContainer::const_iterator iter=theRecHits.begin(); iter!=theRecHits.end(); iter++)
00495 {
00496 count++;
00497 const GeomDet* Detector = (*iter)->det();
00498 if(dynamic_cast<const RPCChamber*>(Detector) != 0)
00499 {
00500 const RPCChamber* RPCCh = dynamic_cast<const RPCChamber*>(Detector);
00501 RPCDetId RPCId = RPCCh->id();
00502 int Region = RPCId.region();
00503 int Station = RPCId.station();
00504
00505 if(count <= 4)
00506 {
00507 if(Region != 0)
00508 isFit = false;
00509 if(Station > 2)
00510 isFit = false;
00511 }
00512 }
00513 }
00514
00515 if(count <= 4)
00516 isFit = false;
00517 cout << "Check for segment fit: " << isFit << endl;
00518 return isFit;
00519 }
00520
00521 MuonTransientTrackingRecHit::ConstMuonRecHitPointer RPCSeedPattern::FirstRecHit() const
00522 {
00523 return theRecHits.front();
00524 }
00525
00526 MuonTransientTrackingRecHit::ConstMuonRecHitPointer RPCSeedPattern::BestRefRecHit() const
00527 {
00528 ConstMuonRecHitPointer best;
00529 int index = 0;
00530
00531
00532 for (ConstMuonRecHitContainer::const_iterator iter=theRecHits.begin(); iter!=theRecHits.end(); iter++)
00533 {
00534 if(AlgorithmType != 3)
00535 best = (*iter);
00536 else
00537 if(index < 4)
00538 best = (*iter);
00539 index++;
00540 }
00541 return best;
00542 }
00543
00544 double RPCSeedPattern::getDistance(const ConstMuonRecHitPointer& precHit, const GlobalVector& Center) const
00545 {
00546 return sqrt((precHit->globalPosition().x()-Center.x())*(precHit->globalPosition().x()-Center.x())+(precHit->globalPosition().y()-Center.y())*(precHit->globalPosition().y()-Center.y()));
00547 }
00548
00549 bool RPCSeedPattern::checkStraightwithThreerecHits(ConstMuonRecHitPointer (&precHit)[3], double MinDeltaPhi) const
00550 {
00551 GlobalVector segvec1 = precHit[1]->globalPosition() - precHit[0]->globalPosition();
00552 GlobalVector segvec2 = precHit[2]->globalPosition() - precHit[1]->globalPosition();
00553 double dPhi = (segvec2.phi() - segvec1.phi()).value();
00554 if(fabs(dPhi) > MinDeltaPhi)
00555 {
00556 cout << "Part is estimate to be not straight" << endl;
00557 return false;
00558 }
00559 else
00560 {
00561 cout << "Part is estimate to be straight" << endl;
00562 return true;
00563 }
00564 }
00565
00566 GlobalVector RPCSeedPattern::computePtwithThreerecHits(double& pt, double& pt_err, ConstMuonRecHitPointer (&precHit)[3]) const
00567 {
00568 double x[3], y[3];
00569 x[0] = precHit[0]->globalPosition().x();
00570 y[0] = precHit[0]->globalPosition().y();
00571 x[1] = precHit[1]->globalPosition().x();
00572 y[1] = precHit[1]->globalPosition().y();
00573 x[2] = precHit[2]->globalPosition().x();
00574 y[2] = precHit[2]->globalPosition().y();
00575 double A = (y[2]-y[1])/(x[2]-x[1]) - (y[1]-y[0])/(x[1]-x[0]);
00576 double TYO = (x[2]-x[0])/A + (y[2]*y[2]-y[1]*y[1])/((x[2]-x[1])*A) - (y[1]*y[1]-y[0]*y[0])/((x[1]-x[0])*A);
00577 double TXO = (x[2]+x[1]) + (y[2]*y[2]-y[1]*y[1])/(x[2]-x[1]) - TYO*(y[2]-y[1])/(x[2]-x[1]);
00578 double XO = 0.5 * TXO;
00579 double YO = 0.5 * TYO;
00580 double R2 = (x[0]-XO)*(x[0]-XO) + (y[0]-YO)*(y[0]-YO);
00581 cout << "R2 is " << R2 << endl;
00582
00583 pt = 0.01 * sqrt(R2) * 2 * 0.3;
00584 cout << "pt is " << pt << endl;
00585 GlobalVector Center(XO, YO, 0);
00586 return Center;
00587 }
00588
00589 bool RPCSeedPattern::checkStraightwithSegment(const RPCSegment& Segment1, const RPCSegment& Segment2, double MinDeltaPhi) const
00590 {
00591 GlobalVector segvec1 = (Segment1.second)->globalPosition() - (Segment1.first)->globalPosition();
00592 GlobalVector segvec2 = (Segment2.second)->globalPosition() - (Segment2.first)->globalPosition();
00593 GlobalVector segvec3 = (Segment2.first)->globalPosition() - (Segment1.first)->globalPosition();
00594
00595 double dPhi1 = (segvec2.phi() - segvec1.phi()).value();
00596 double dPhi2 = (segvec3.phi() - segvec1.phi()).value();
00597 cout << "Checking straight with 2 segments. dPhi1: " << dPhi1 << ", dPhi2: " << dPhi2 << endl;
00598 cout << "Checking straight with 2 segments. dPhi1 in degree: " << dPhi1*180/3.1415926 << ", dPhi2 in degree: " << dPhi2*180/3.1415926 << endl;
00599 if(fabs(dPhi1) > MinDeltaPhi || fabs(dPhi2) > MinDeltaPhi)
00600 {
00601 cout << "Segment is estimate to be not straight" << endl;
00602 return false;
00603 }
00604 else
00605 {
00606 cout << "Segment is estimate to be straight" << endl;
00607 return true;
00608 }
00609 }
00610
00611 GlobalVector RPCSeedPattern::computePtwithSegment(const RPCSegment& Segment1, const RPCSegment& Segment2) const
00612 {
00613 GlobalVector segvec1 = (Segment1.second)->globalPosition() - (Segment1.first)->globalPosition();
00614 GlobalVector segvec2 = (Segment2.second)->globalPosition() - (Segment2.first)->globalPosition();
00615 GlobalPoint Point1(((Segment1.second)->globalPosition().x() + (Segment1.first)->globalPosition().x()) / 2, ((Segment1.second)->globalPosition().y() + (Segment1.first)->globalPosition().y()) / 2, ((Segment1.second)->globalPosition().z() + (Segment1.first)->globalPosition().z()) / 2);
00616 GlobalPoint Point2(((Segment2.second)->globalPosition().x() + (Segment2.first)->globalPosition().x()) / 2, ((Segment2.second)->globalPosition().y() + (Segment2.first)->globalPosition().y()) / 2, ((Segment2.second)->globalPosition().z() + (Segment2.first)->globalPosition().z()) / 2);
00617 GlobalVector vecZ(0, 0, 1);
00618 GlobalVector gvec1 = segvec1.cross(vecZ);
00619 GlobalVector gvec2 = segvec2.cross(vecZ);
00620 double A1 = gvec1.x();
00621 double B1 = gvec1.y();
00622 double A2 = gvec2.x();
00623 double B2 = gvec2.y();
00624 double X1 = Point1.x();
00625 double Y1 = Point1.y();
00626 double X2 = Point2.x();
00627 double Y2 = Point2.y();
00628 double XO = (A1*A2*(Y2-Y1)+A2*B1*X1-A1*B2*X2)/(A2*B1-A1*B2);
00629 double YO = (B1*B2*(X2-X1)+B2*A1*Y1-B1*A2*Y2)/(B2*A1-B1*A2);
00630 GlobalVector Center(XO, YO, 0);
00631 return Center;
00632 }
00633
00634 bool RPCSeedPattern::checkStraightwithThreerecHits(double (&x)[3], double (&y)[3], double MinDeltaPhi) const
00635 {
00636 GlobalVector segvec1((x[1]-x[0]), (y[1]-y[0]), 0);
00637 GlobalVector segvec2((x[2]-x[1]), (y[2]-y[1]), 0);
00638 double dPhi = (segvec2.phi() - segvec1.phi()).value();
00639 if(fabs(dPhi) > MinDeltaPhi)
00640 {
00641 cout << "Part is estimate to be not straight" << endl;
00642 return false;
00643 }
00644 else
00645 {
00646 cout << "Part is estimate to be straight" << endl;
00647 return true;
00648 }
00649 }
00650
00651 GlobalVector RPCSeedPattern::computePtWithThreerecHits(double& pt, double& pt_err, double (&x)[3], double (&y)[3]) const
00652 {
00653 double A = (y[2]-y[1])/(x[2]-x[1]) - (y[1]-y[0])/(x[1]-x[0]);
00654 double TYO = (x[2]-x[0])/A + (y[2]*y[2]-y[1]*y[1])/((x[2]-x[1])*A) - (y[1]*y[1]-y[0]*y[0])/((x[1]-x[0])*A);
00655 double TXO = (x[2]+x[1]) + (y[2]*y[2]-y[1]*y[1])/(x[2]-x[1]) - TYO*(y[2]-y[1])/(x[2]-x[1]);
00656 double XO = 0.5 * TXO;
00657 double YO = 0.5 * TYO;
00658 double R2 = (x[0]-XO)*(x[0]-XO) + (y[0]-YO)*(y[0]-YO);
00659 cout << "R2 is " << R2 << endl;
00660
00661 pt = 0.01 * sqrt(R2) * 2 * 0.3;
00662 cout << "pt is " << pt << endl;
00663 GlobalVector Center(XO, YO, 0);
00664 return Center;
00665 }
00666
00667 void RPCSeedPattern::checkSimplePattern(const edm::EventSetup& eSetup)
00668 {
00669 if(isPatternChecked == true)
00670 return;
00671
00672
00673 edm::ESHandle<MagneticField> Field;
00674 eSetup.get<IdealMagneticFieldRecord>().get(Field);
00675
00676 unsigned int NumberofHitsinSeed = nrhit();
00677
00678
00679 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00680 cout << "Position of recHit is: " << (*iter)->globalPosition() << endl;
00681
00682
00683 std::vector<double> BzValue;
00684 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != (theRecHits.end()-1); iter++)
00685 {
00686 GlobalPoint gpFirst = (*iter)->globalPosition();
00687 GlobalPoint gpLast = (*(iter+1))->globalPosition();
00688 GlobalPoint *gp = new GlobalPoint[sampleCount];
00689 double dx = (gpLast.x() - gpFirst.x()) / (sampleCount + 1);
00690 double dy = (gpLast.y() - gpFirst.y()) / (sampleCount + 1);
00691 double dz = (gpLast.z() - gpFirst.z()) / (sampleCount + 1);
00692 for(unsigned int index = 0; index < sampleCount; index++)
00693 {
00694 gp[index] = GlobalPoint((gpFirst.x()+dx*(index+1)), (gpFirst.y()+dy*(index+1)), (gpFirst.z()+dz*(index+1)));
00695 GlobalVector MagneticVec_temp = Field->inTesla(gp[index]);
00696 cout << "Sampling magnetic field : " << MagneticVec_temp << endl;
00697 BzValue.push_back(MagneticVec_temp.z());
00698 }
00699 delete [] gp;
00700 }
00701 meanBz = 0.;
00702 for(unsigned int index = 0; index < BzValue.size(); index++)
00703 meanBz += BzValue[index];
00704 meanBz /= BzValue.size();
00705 cout << "Mean Bz is " << meanBz << endl;
00706 deltaBz = 0.;
00707 for(unsigned int index = 0; index < BzValue.size(); index++)
00708 deltaBz += (BzValue[index] - meanBz) * (BzValue[index] - meanBz);
00709 deltaBz /= BzValue.size();
00710 deltaBz = sqrt(deltaBz);
00711 cout<< "delata Bz is " << deltaBz << endl;
00712
00713
00714 isGoodPattern = 1;
00715
00716
00717 if(fabs((*(theRecHits.end()-1))->globalPosition().z() - (*(theRecHits.begin()))->globalPosition().z()) > ZError)
00718 {
00719 if(((*(theRecHits.end()-1))->globalPosition().z() - (*(theRecHits.begin()))->globalPosition().z()) > ZError)
00720 isParralZ = 1;
00721 else
00722 isParralZ = -1;
00723 }
00724 else
00725 isParralZ = 0;
00726
00727 cout << " Check isParralZ is :" << isParralZ << endl;
00728 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != (theRecHits.end()-1); iter++)
00729 {
00730 if(isParralZ == 0)
00731 {
00732 if(fabs((*(iter+1))->globalPosition().z()-(*iter)->globalPosition().z()) > ZError)
00733 {
00734 cout << "Pattern find error in Z direction: wrong perpendicular direction" << endl;
00735 isGoodPattern = 0;
00736 }
00737 }
00738 else
00739 {
00740 if((int)(((*(iter+1))->globalPosition().z()-(*iter)->globalPosition().z())/ZError)*isParralZ < 0)
00741 {
00742 cout << "Pattern find error in Z direction: wrong Z direction" << endl;
00743 isGoodPattern = 0;
00744 }
00745 }
00746 }
00747
00748
00749 if(isStraight == false)
00750 {
00751
00752 GlobalVector *vec = new GlobalVector[NumberofHitsinSeed];
00753 unsigned int index = 0;
00754 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00755 {
00756 GlobalVector vec_temp(((*iter)->globalPosition().x()-Center.x()), ((*iter)->globalPosition().y()-Center.y()), ((*iter)->globalPosition().z()-Center.z()));
00757 vec[index] = vec_temp;
00758 index++;
00759 }
00760 isClockwise = 0;
00761 for(unsigned int index = 0; index < (NumberofHitsinSeed-1); index++)
00762 {
00763
00764 if((vec[index+1].phi()-vec[index].phi()) > 0)
00765 isClockwise--;
00766 else
00767 isClockwise++;
00768 cout << "Current isClockwise is : " << isClockwise << endl;
00769 }
00770 cout << "Check isClockwise is : " << isClockwise << endl;
00771 if((unsigned int)abs(isClockwise) != (NumberofHitsinSeed-1))
00772 {
00773 cout << "Pattern find error in Phi direction" << endl;
00774 isGoodPattern = 0;
00775 isClockwise = 0;
00776 }
00777 else
00778 isClockwise /= abs(isClockwise);
00779 delete [] vec;
00780
00781
00782 double deltaRwithBz = fabs(deltaBz * meanRadius / meanBz);
00783 cout << "deltaR with Bz is " << deltaRwithBz << endl;
00784
00785 if(isClockwise == 0)
00786 meanPt = upper_limit_pt;
00787 else
00788 meanPt = 0.01 * meanRadius * meanBz * 0.3 * isClockwise;
00789 if(fabs(meanPt) > upper_limit_pt)
00790 meanPt = upper_limit_pt * meanPt / fabs(meanPt);
00791 cout << " meanRadius is " << meanRadius << endl;
00792 cout << " meanPt is " << meanPt << endl;
00793
00794 double deltaR = 0.;
00795 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00796 {
00797 deltaR += (getDistance(*iter, Center) - meanRadius) * (getDistance(*iter, Center) - meanRadius);
00798 }
00799 deltaR = deltaR / NumberofHitsinSeed;
00800 deltaR = sqrt(deltaR);
00801
00802 meanSpt = deltaR;
00803 cout << "DeltaR is " << deltaR << endl;
00804 if(deltaR > deltaRThreshold)
00805 {
00806 cout << "Pattern find error: deltaR over threshold" << endl;
00807 isGoodPattern = 0;
00808 }
00809 }
00810 else
00811 {
00812
00813 isClockwise =0;
00814 meanPt = upper_limit_pt;
00815
00816 meanSpt = deltaRThreshold;
00817 }
00818 cout << "III--> Seed Pt : " << meanPt << endl;
00819 cout << "III--> Pattern is: " << isGoodPattern << endl;
00820
00821
00822 isPatternChecked = true;
00823 }
00824
00825 void RPCSeedPattern::checkSegmentAlgorithmSpecial(const edm::EventSetup& eSetup)
00826 {
00827 if(isPatternChecked == true)
00828 return;
00829
00830 if(!checkSegment())
00831 {
00832 isPatternChecked = true;
00833 isGoodPattern = -1;
00834 return;
00835 }
00836
00837
00838 isGoodPattern = 1;
00839
00840
00841 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00842 cout << "Position of recHit is: " << (*iter)->globalPosition() << endl;
00843
00844
00845 if(fabs((*(theRecHits.end()-1))->globalPosition().z() - (*(theRecHits.begin()))->globalPosition().z()) > ZError)
00846 {
00847 if(((*(theRecHits.end()-1))->globalPosition().z() - (*(theRecHits.begin()))->globalPosition().z()) > ZError)
00848 isParralZ = 1;
00849 else
00850 isParralZ = -1;
00851 }
00852 else
00853 isParralZ = 0;
00854
00855 cout << " Check isParralZ is :" << isParralZ << endl;
00856 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != (theRecHits.end()-1); iter++)
00857 {
00858 if(isParralZ == 0)
00859 {
00860 if(fabs((*(iter+1))->globalPosition().z()-(*iter)->globalPosition().z()) > ZError)
00861 {
00862 cout << "Pattern find error in Z direction: wrong perpendicular direction" << endl;
00863 isGoodPattern = 0;
00864 }
00865 }
00866 else
00867 {
00868 if((int)(((*(iter+1))->globalPosition().z()-(*iter)->globalPosition().z())/ZError)*isParralZ < 0)
00869 {
00870 cout << "Pattern find error in Z direction: wrong Z direction" << endl;
00871 isGoodPattern = 0;
00872 }
00873 }
00874 }
00875
00876
00877 if(isStraight2 == true)
00878 {
00879
00880 isClockwise = 0;
00881 meanPt = upper_limit_pt;
00882
00883 meanSpt = deltaRThreshold;
00884
00885
00886 GlobalVector startSegment = (SegmentRB[1].second)->globalPosition() - (SegmentRB[1].first)->globalPosition();
00887 GlobalPoint startPosition = (SegmentRB[1].first)->globalPosition();
00888 GlobalVector startMomentum = startSegment*(meanPt/startSegment.perp());
00889 unsigned int index = 0;
00890 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00891 {
00892 if(index < 4)
00893 {
00894 index++;
00895 continue;
00896 }
00897 double tracklength = 0;
00898 cout << "Now checking recHit " << index << endl;
00899 double Distance = extropolateStep(startPosition, startMomentum, iter, isClockwise, tracklength, eSetup);
00900 cout << "Final distance is " << Distance << endl;
00901 if(Distance > MaxRSD)
00902 {
00903 cout << "Pattern find error in distance for other recHits: " << Distance << endl;
00904 isGoodPattern = 0;
00905 }
00906 index++;
00907 }
00908 }
00909 else
00910 {
00911
00912 GlobalVector vec[2];
00913 vec[0] = GlobalVector((entryPosition.x()-Center2.x()), (entryPosition.y()-Center2.y()), (entryPosition.z()-Center2.z()));
00914 vec[1] = GlobalVector((leavePosition.x()-Center2.x()), (leavePosition.y()-Center2.y()), (leavePosition.z()-Center2.z()));
00915 isClockwise = 0;
00916 if((vec[1].phi()-vec[0].phi()).value() > 0)
00917 isClockwise = -1;
00918 else
00919 isClockwise = 1;
00920
00921 cout << "Check isClockwise is : " << isClockwise << endl;
00922
00923
00924 meanPt = 0.01 * meanRadius2 * meanMagneticField2.z() * 0.3 * isClockwise;
00925
00926 cout << " meanRadius is " << meanRadius2 << ", with meanBz " << meanMagneticField2.z() << endl;
00927 cout << " meanPt is " << meanPt << endl;
00928 if(fabs(meanPt) > upper_limit_pt)
00929 meanPt = upper_limit_pt * meanPt / fabs(meanPt);
00930
00931
00932 cout << "entryPosition: " << entryPosition << endl;
00933 cout << "leavePosition: " << leavePosition << endl;
00934 cout << "Center2 is : " << Center2 << endl;
00935 double R1 = vec[0].perp();
00936 double R2 = vec[1].perp();
00937 double deltaR = sqrt(((R1-meanRadius2)*(R1-meanRadius2)+(R2-meanRadius2)*(R2-meanRadius2))/2);
00938 meanSpt = deltaR;
00939 cout << "R1 is " << R1 << ", R2 is " << R2 << endl;
00940 cout << "Delta R for the initial 3 segments is " << deltaR << endl;
00941 if(deltaR > deltaRThreshold)
00942 {
00943 cout << "Pattern find error in delta R for the initial 3 segments" << endl;
00944 isGoodPattern = 0;
00945 }
00946
00947
00948 GlobalVector startSegment = (SegmentRB[1].second)->globalPosition() - (SegmentRB[1].first)->globalPosition();
00949 GlobalPoint startPosition = (SegmentRB[1].first)->globalPosition();
00950 GlobalVector startMomentum = startSegment*(fabs(meanPt)/startSegment.perp());
00951 unsigned int index = 0;
00952 for(ConstMuonRecHitContainer::const_iterator iter = theRecHits.begin(); iter != theRecHits.end(); iter++)
00953 {
00954 if(index < 4)
00955 {
00956 index++;
00957 continue;
00958 }
00959 double tracklength = 0;
00960 cout << "Now checking recHit " << index << endl;
00961 double Distance = extropolateStep(startPosition, startMomentum, iter, isClockwise, tracklength, eSetup);
00962 cout << "Final distance is " << Distance << endl;
00963 if(Distance > MaxRSD)
00964 {
00965 cout << "Pattern find error in distance for other recHits: " << Distance << endl;
00966 isGoodPattern = 0;
00967 }
00968 index++;
00969 }
00970 }
00971
00972 cout << "Checking finish, isGoodPattern now is " << isGoodPattern << endl;
00973
00974 isPatternChecked = true;
00975 }
00976
00977 double RPCSeedPattern::extropolateStep(const GlobalPoint& startPosition, const GlobalVector& startMomentum, ConstMuonRecHitContainer::const_iterator iter, const int ClockwiseDirection, double& tracklength, const edm::EventSetup& eSetup)
00978 {
00979
00980 edm::ESHandle<MagneticField> Field;
00981 eSetup.get<IdealMagneticFieldRecord>().get(Field);
00982
00983 cout << "Extrapolating the track to check the pattern" << endl;
00984 tracklength = 0;
00985
00986 DetId hitDet = (*iter)->hit()->geographicalId();
00987 RPCDetId RPCId = RPCDetId(hitDet.rawId());
00988
00989 edm::ESHandle<RPCGeometry> pRPCGeom;
00990 eSetup.get<MuonGeometryRecord>().get(pRPCGeom);
00991 const RPCGeometry* rpcGeometry = (const RPCGeometry*)&*pRPCGeom;
00992
00993 const BoundPlane RPCSurface = rpcGeometry->chamber(RPCId)->surface();
00994 double startSide = RPCSurface.localZ(startPosition);
00995 cout << "Start side : " << startSide;
00996
00997 GlobalPoint currentPosition = startPosition;
00998 double currentSide = RPCSurface.localZ(currentPosition);
00999 GlobalVector currentMomentum = startMomentum;
01000 GlobalVector ZDirection(0, 0, 1);
01001
01002
01003 double currentDistance = ((GlobalVector)(currentPosition - (*iter)->globalPosition())).perp();
01004 cout << "Start current position is : " << currentPosition << endl;
01005 cout << "Start current Momentum is: " << currentMomentum.mag() << ", in vector: " << currentMomentum << endl;
01006 cout << "Start current distance is " << currentDistance << endl;
01007 cout << "Start current radius is " << currentPosition.perp() << endl;
01008 cout << "Destination radius is " << (*iter)->globalPosition().perp() << endl;
01009
01010
01011
01012 double currentDistance_next = currentDistance;
01013 do
01014 {
01015 currentDistance = currentDistance_next;
01016 if(ClockwiseDirection == 0)
01017 {
01018 currentPosition += currentMomentum.unit() * stepLength;
01019 }
01020 else
01021 {
01022 double Bz = Field->inTesla(currentPosition).z();
01023 double Radius = currentMomentum.perp()/fabs(Bz*0.01*0.3);
01024 double deltaPhi = (stepLength*currentMomentum.perp()/currentMomentum.mag())/Radius;
01025
01026
01027 GlobalVector currentPositiontoCenter = currentMomentum.unit().cross(ZDirection);
01028 currentPositiontoCenter *= Radius;
01029
01030 currentPositiontoCenter *= ClockwiseDirection;
01031
01032 GlobalPoint currentCenter = currentPosition;
01033 currentCenter += currentPositiontoCenter;
01034
01035
01036 GlobalVector CentertocurrentPosition = (GlobalVector)(currentPosition - currentCenter);
01037 double Phi = CentertocurrentPosition.phi().value();
01038 Phi += deltaPhi * (-1) * ClockwiseDirection;
01039 double deltaZ = stepLength*currentMomentum.z()/currentMomentum.mag();
01040 GlobalVector CentertonewPosition(GlobalVector::Cylindrical(CentertocurrentPosition.perp(), Phi, deltaZ));
01041 double PtPhi = currentMomentum.phi().value();
01042 PtPhi += deltaPhi * (-1) * ClockwiseDirection;
01043 currentMomentum = GlobalVector(GlobalVector::Cylindrical(currentMomentum.perp(), PtPhi, currentMomentum.z()));
01044 currentPosition = currentCenter + CentertonewPosition;
01045 }
01046
01047
01048 tracklength += stepLength * currentMomentum.perp() / currentMomentum.mag();
01049
01050
01051 currentSide = RPCSurface.localZ(currentPosition);
01052 cout << "Stepping current side : " << currentSide << endl;
01053 cout << "Stepping current position is: " << currentPosition << endl;
01054 cout << "Stepping current Momentum is: " << currentMomentum.mag() << ", in vector: " << currentMomentum << endl;
01055 currentDistance_next = ((GlobalVector)(currentPosition - (*iter)->globalPosition())).perp();
01056 cout << "Stepping current distance is " << currentDistance << endl;
01057 cout << "Stepping current radius is " << currentPosition.perp() << endl;
01058 }while(currentDistance_next < currentDistance);
01059
01060 return currentDistance;
01061 }
01062
01063 RPCSeedPattern::weightedTrajectorySeed RPCSeedPattern::createFakeSeed(int& isGoodSeed, const edm::EventSetup& eSetup)
01064 {
01065
01066 cout << "Now create a fake seed" << endl;
01067 isPatternChecked = true;
01068 isGoodPattern = -1;
01069 isStraight = true;
01070 meanPt = upper_limit_pt;
01071 meanSpt = 0;
01072 Charge = 0;
01073 isClockwise = 0;
01074 isParralZ = 0;
01075 meanRadius = -1;
01076
01077
01078
01079 const ConstMuonRecHitPointer best = BestRefRecHit();
01080
01081 Momentum = GlobalVector(0, 0, 0);
01082 LocalPoint segPos=best->localPosition();
01083 LocalVector segDirFromPos=best->det()->toLocal(Momentum);
01084 LocalTrajectoryParameters param(segPos, segDirFromPos, Charge);
01085
01086
01087 AlgebraicSymMatrix mat(5,0);
01088 mat = best->parametersError().similarityT(best->projectionMatrix());
01089 mat[0][0] = meanSpt;
01090 LocalTrajectoryError error(asSMatrix<5>(mat));
01091
01092 edm::ESHandle<MagneticField> Field;
01093 eSetup.get<IdealMagneticFieldRecord>().get(Field);
01094
01095 TrajectoryStateOnSurface tsos(param, error, best->det()->surface(), &*Field);
01096
01097 DetId id = best->geographicalId();
01098 TrajectoryStateTransform tsTransform;
01099 PTrajectoryStateOnDet *seedTSOS = tsTransform.persistentState(tsos, id.rawId());
01100
01101 edm::OwnVector<TrackingRecHit> container;
01102 for(ConstMuonRecHitContainer::const_iterator iter=theRecHits.begin(); iter!=theRecHits.end(); iter++)
01103 container.push_back((*iter)->hit()->clone());
01104
01105 TrajectorySeed theSeed(*seedTSOS, container, alongMomentum);
01106 weightedTrajectorySeed theweightedSeed;
01107 theweightedSeed.first = theSeed;
01108 theweightedSeed.second = meanSpt;
01109 isGoodSeed = isGoodPattern;
01110
01111 delete seedTSOS;
01112 return theweightedSeed;
01113 }
01114
01115 RPCSeedPattern::weightedTrajectorySeed RPCSeedPattern::createSeed(int& isGoodSeed, const edm::EventSetup& eSetup)
01116 {
01117 if(isPatternChecked == false || isGoodPattern == -1)
01118 {
01119 cout <<"Pattern is not yet checked! Create a fake seed instead!" << endl;
01120 return createFakeSeed(isGoodSeed, eSetup);
01121 }
01122
01123 edm::ESHandle<MagneticField> Field;
01124 eSetup.get<IdealMagneticFieldRecord>().get(Field);
01125
01126 MuonPatternRecoDumper debug;
01127
01128
01129
01130
01131
01132
01133 if(isClockwise == 0)
01134 Charge = 0;
01135 else
01136 Charge = (int)(meanPt / fabs(meanPt));
01137
01138
01139 const ConstMuonRecHitPointer best = BestRefRecHit();
01140 const ConstMuonRecHitPointer first = FirstRecHit();
01141
01142 if(isClockwise != 0)
01143 {
01144 if(AlgorithmType != 3)
01145 {
01146
01147 GlobalVector vecRef1((first->globalPosition().x()-Center.x()), (first->globalPosition().y()-Center.y()), (first->globalPosition().z()-Center.z()));
01148 GlobalVector vecRef2((best->globalPosition().x()-Center.x()), (best->globalPosition().y()-Center.y()), (best->globalPosition().z()-Center.z()));
01149
01150 double deltaPhi = (vecRef2.phi() - vecRef1.phi()).value();
01151 double deltaS = meanRadius * fabs(deltaPhi);
01152 double deltaZ = best->globalPosition().z() - first->globalPosition().z();
01153
01154 GlobalVector vecZ(0, 0, 1);
01155 GlobalVector vecPt = (vecRef2.unit()).cross(vecZ);
01156 if(isClockwise == -1)
01157 vecPt *= -1;
01158 vecPt *= deltaS;
01159 Momentum = GlobalVector(0, 0, deltaZ);
01160 Momentum += vecPt;
01161 Momentum *= fabs(meanPt / deltaS);
01162 }
01163 else
01164 {
01165 double deltaZ = best->globalPosition().z() - first->globalPosition().z();
01166 Momentum = GlobalVector(GlobalVector::Cylindrical(S, lastPhi, deltaZ));
01167 Momentum *= fabs(meanPt / S);
01168 }
01169 }
01170 else
01171 {
01172 Momentum = best->globalPosition() - first->globalPosition();
01173 double deltaS = Momentum.perp();
01174 Momentum *= fabs(meanPt / deltaS);
01175 }
01176 LocalPoint segPos = best->localPosition();
01177 LocalVector segDirFromPos = best->det()->toLocal(Momentum);
01178 LocalTrajectoryParameters param(segPos, segDirFromPos, Charge);
01179
01180 LocalTrajectoryError error = getSpecialAlgorithmErrorMatrix(first, best);
01181
01182 TrajectoryStateOnSurface tsos(param, error, best->det()->surface(), &*Field);
01183 cout << "Trajectory State on Surface before the extrapolation" << endl;
01184 cout << debug.dumpTSOS(tsos);
01185 DetId id = best->geographicalId();
01186 cout << "The RecSegment relies on: " << endl;
01187 cout << debug.dumpMuonId(id);
01188 cout << debug.dumpTSOS(tsos);
01189 TrajectoryStateTransform tsTransform;
01190 PTrajectoryStateOnDet *seedTSOS = tsTransform.persistentState(tsos, id.rawId());
01191
01192 edm::OwnVector<TrackingRecHit> container;
01193 for(ConstMuonRecHitContainer::const_iterator iter=theRecHits.begin(); iter!=theRecHits.end(); iter++)
01194 {
01195
01196
01197
01198
01199 container.push_back((*iter)->hit()->clone());
01200 }
01201
01202 TrajectorySeed theSeed(*seedTSOS, container, alongMomentum);
01203 weightedTrajectorySeed theweightedSeed;
01204 theweightedSeed.first = theSeed;
01205 theweightedSeed.second = meanSpt;
01206 isGoodSeed = isGoodPattern;
01207
01208 delete seedTSOS;
01209 return theweightedSeed;
01210 }
01211
01212 LocalTrajectoryError RPCSeedPattern::getSpecialAlgorithmErrorMatrix(const ConstMuonRecHitPointer& first, const ConstMuonRecHitPointer& best) {
01213
01214 LocalTrajectoryError Error;
01215 double dXdZ = 0;
01216 double dYdZ = 0;
01217 double dP = 0;
01218 AlgebraicSymMatrix mat(5, 0);
01219 mat = best->parametersError().similarityT(best->projectionMatrix());
01220 if(AlgorithmType != 3) {
01221 GlobalVector vecRef1((first->globalPosition().x()-Center.x()), (first->globalPosition().y()-Center.y()), (first->globalPosition().z()-Center.z()));
01222 GlobalVector vecRef2((best->globalPosition().x()-Center.x()), (best->globalPosition().y()-Center.y()), (best->globalPosition().z()-Center.z()));
01223 double deltaPhi = (vecRef2.phi() - vecRef1.phi()).value();
01224 double L = meanRadius * fabs(deltaPhi);
01225 double N = nrhit();
01226 double A_N = 180*N*N*N/((N-1)*(N+1)*(N+2)*(N+3));
01227 double sigma_x = sqrt(mat[3][3]);
01228 double betaovergame = Momentum.mag()/0.1066;
01229 double beta = sqrt((betaovergame*betaovergame)/(1+betaovergame*betaovergame));
01230 double dPt = meanPt*(0.0136*sqrt(1/100)*sqrt(4*A_N/N)/(beta*0.3*meanBz*L) + sigma_x*meanPt*sqrt(4*A_N)/(0.3*meanBz*L*L));
01231 double dP = dPt * Momentum.mag() / meanPt;
01232 mat[0][0] = (dP * dP) / (Momentum.mag() * Momentum.mag() * Momentum.mag() * Momentum.mag());
01233 mat[1][1] = dXdZ * dXdZ;
01234 mat[2][2] = dYdZ * dYdZ;
01235 Error = LocalTrajectoryError(asSMatrix<5>(mat));
01236 }
01237 else {
01238 AlgebraicSymMatrix mat0(5, 0);
01239 mat0 = (SegmentRB[0].first)->parametersError().similarityT((SegmentRB[0].first)->projectionMatrix());
01240 double dX0 = sqrt(mat0[3][3]);
01241 double dY0 = sqrt(mat0[4][4]);
01242 AlgebraicSymMatrix mat1(5, 0);
01243 mat1 = (SegmentRB[0].second)->parametersError().similarityT((SegmentRB[0].second)->projectionMatrix());
01244 double dX1 = sqrt(mat1[3][3]);
01245 double dY1 = sqrt(mat1[4][4]);
01246 AlgebraicSymMatrix mat2(5, 0);
01247 mat2 = (SegmentRB[1].first)->parametersError().similarityT((SegmentRB[1].first)->projectionMatrix());
01248 double dX2 = sqrt(mat2[3][3]);
01249 double dY2 = sqrt(mat2[4][4]);
01250 AlgebraicSymMatrix mat3(5, 0);
01251 mat3 = (SegmentRB[1].second)->parametersError().similarityT((SegmentRB[1].second)->projectionMatrix());
01252 double dX3 = sqrt(mat3[3][3]);
01253 double dY3 = sqrt(mat3[4][4]);
01254 cout << "Local error for 4 recHits are: " << dX0 << ", " << dY0 << ", " << dX1 << ", " << dY1 << ", " << dX2 << ", " << dY2 << ", " << dX3 << ", " << dY3 << endl;
01255 const GeomDetUnit* refRPC1 = (SegmentRB[0].second)->detUnit();
01256 LocalPoint recHit0 = refRPC1->toLocal((SegmentRB[0].first)->globalPosition());
01257 LocalPoint recHit1 = refRPC1->toLocal((SegmentRB[0].second)->globalPosition());
01258 LocalVector localSegment00 = (LocalVector)(recHit1 - recHit0);
01259 LocalVector localSegment01 = LocalVector(localSegment00.x()+dX0+dX1, localSegment00.y(), localSegment00.z());
01260 LocalVector localSegment02 = LocalVector(localSegment00.x()-dX0-dX1, localSegment00.y(), localSegment00.z());
01261 GlobalVector globalSegment00 = refRPC1->toGlobal(localSegment00);
01262 GlobalVector globalSegment01 = refRPC1->toGlobal(localSegment01);
01263 GlobalVector globalSegment02 = refRPC1->toGlobal(localSegment02);
01264
01265 const GeomDetUnit* refRPC2 = (SegmentRB[1].first)->detUnit();
01266 LocalPoint recHit2 = refRPC2->toLocal((SegmentRB[1].first)->globalPosition());
01267 LocalPoint recHit3 = refRPC2->toLocal((SegmentRB[1].second)->globalPosition());
01268 LocalVector localSegment10 = (LocalVector)(recHit3 - recHit2);
01269 LocalVector localSegment11 = LocalVector(localSegment10.x()+dX2+dX3, localSegment10.y(), localSegment10.z());
01270 LocalVector localSegment12 = LocalVector(localSegment10.x()-dX2-dX3, localSegment10.y(), localSegment10.z());
01271 GlobalVector globalSegment10 = refRPC2->toGlobal(localSegment10);
01272 GlobalVector globalSegment11 = refRPC2->toGlobal(localSegment11);
01273 GlobalVector globalSegment12 = refRPC2->toGlobal(localSegment12);
01274
01275 if(isClockwise != 0) {
01276 GlobalVector vec[2];
01277 vec[0] = GlobalVector((entryPosition.x()-Center2.x()), (entryPosition.y()-Center2.y()), (entryPosition.z()-Center2.z()));
01278 vec[1] = GlobalVector((leavePosition.x()-Center2.x()), (leavePosition.y()-Center2.y()), (leavePosition.z()-Center2.z()));
01279 double halfPhiCenter = fabs((vec[1].phi() - vec[0].phi()).value()) / 2;
01280
01281 double dPhi0 = (((globalSegment00.phi() - globalSegment01.phi()).value()*isClockwise) > 0) ? fabs((globalSegment00.phi() - globalSegment01.phi()).value()) : fabs((globalSegment00.phi() - globalSegment02.phi()).value());
01282 double dPhi1 = (((globalSegment10.phi() - globalSegment11.phi()).value()*isClockwise) < 0) ? fabs((globalSegment10.phi() - globalSegment11.phi()).value()) : fabs((globalSegment10.phi() - globalSegment12.phi()).value());
01283
01284 double dPhi = (dPhi0 <= dPhi1) ? dPhi0 : dPhi1;
01285 cout << "DPhi for new Segment is about " << dPhi << endl;
01286
01287 double newhalfPhiCenter = ((halfPhiCenter-dPhi) > 0 ? (halfPhiCenter-dPhi) : 0);
01288 if(newhalfPhiCenter != 0) {
01289 double newmeanPt = meanPt * halfPhiCenter / newhalfPhiCenter;
01290 if(fabs(newmeanPt) > upper_limit_pt)
01291 newmeanPt = upper_limit_pt * meanPt / fabs(meanPt);
01292 cout << "The error is inside range. Max meanPt could be " << newmeanPt << endl;
01293 dP = fabs(Momentum.mag() * (newmeanPt - meanPt) / meanPt);
01294 }
01295 else {
01296 double newmeanPt = upper_limit_pt * meanPt / fabs(meanPt);
01297 cout << "The error is outside range. Max meanPt could be " << newmeanPt << endl;
01298 dP = fabs(Momentum.mag() * (newmeanPt - meanPt) / meanPt);
01299 }
01300 }
01301 else {
01302 double dPhi0 = (fabs((globalSegment00.phi() - globalSegment01.phi()).value()) <= fabs((globalSegment00.phi() - globalSegment02.phi()).value())) ? fabs((globalSegment00.phi() - globalSegment01.phi()).value()) : fabs((globalSegment00.phi() - globalSegment02.phi()).value());
01303 double dPhi1 = (fabs((globalSegment10.phi() - globalSegment11.phi()).value()) <= fabs((globalSegment10.phi() - globalSegment12.phi()).value())) ? fabs((globalSegment10.phi() - globalSegment11.phi()).value()) : fabs((globalSegment10.phi() - globalSegment12.phi()).value());
01304 double dPhi = (dPhi0 <= dPhi1) ? dPhi0 : dPhi1;
01305 GlobalVector middleSegment = leavePosition - entryPosition;
01306 double halfDistance = middleSegment.perp() / 2;
01307 double newmeanPt = halfDistance / dPhi;
01308 cout << "The error is for straight. Max meanPt could be " << newmeanPt << endl;
01309 dP = fabs(Momentum.mag() * (newmeanPt - meanPt) / meanPt);
01310 }
01311
01312 double dXdZ1 = globalSegment11.x() / globalSegment11.z() - globalSegment10.x() / globalSegment10.z();
01313 double dXdZ2 = globalSegment12.x() / globalSegment12.z() - globalSegment10.x() / globalSegment10.z();
01314 dXdZ = (fabs(dXdZ1) >= fabs(dXdZ2)) ? dXdZ1 : dXdZ2;
01315
01316 LocalVector localSegment13 = LocalVector(localSegment10.x(), localSegment10.y()+dY2+dY3, localSegment10.z());
01317 LocalVector localSegment14 = LocalVector(localSegment10.x(), localSegment10.y()-dY2-dY3, localSegment10.z());
01318 GlobalVector globalSegment13 = refRPC2->toGlobal(localSegment13);
01319 GlobalVector globalSegment14 = refRPC2->toGlobal(localSegment14);
01320 double dYdZ1 = globalSegment13.y() / globalSegment13.z() - globalSegment10.y() / globalSegment10.z();
01321 double dYdZ2 = globalSegment14.y() / globalSegment14.z() - globalSegment10.y() / globalSegment10.z();
01322 dYdZ = (fabs(dYdZ1) >= fabs(dYdZ2)) ? dYdZ1 : dYdZ2;
01323
01324 mat[0][0] = (dP * dP) / (Momentum.mag() * Momentum.mag() * Momentum.mag() * Momentum.mag());
01325 mat[1][1] = dXdZ * dXdZ;
01326 mat[2][2] = dYdZ * dYdZ;
01327 Error = LocalTrajectoryError(asSMatrix<5>(mat));
01328 }
01329 return Error;
01330 }