Class Reference for E1039 Core & Analysis Software
FastTracklet.cxx
Go to the documentation of this file.
1 /*
2 FastTracklet.cxx
3 
4 Implementation of class Tracklet
5 
6 Author: Kun Liu, liuk@fnal.gov
7 Created: 05-28-2013
8 */
9 
10 #include <phool/recoConsts.h>
11 
12 #include <iostream>
13 #include <algorithm>
14 #include <cmath>
15 
16 #include <TMath.h>
17 #include <TMatrixD.h>
18 
19 #include "FastTracklet.h"
20 #include "TriggerRoad.h"
21 
26 
27 namespace
28 {
29  //static flag to indicate the initialized has been done
30  static bool inited = false;
31 
32  //static pointer to geomtry service
33  static GeomSvc* p_geomSvc = nullptr;
34 
35  //static flag of kmag on/off
36  static bool KMAG_ON;
37 
38  //corase geometry
39  static bool COARSE_MODE;
40 
41  //static flag of kmag strength
42  static double FMAGSTR;
43  static double KMAGSTR;
44 
45  static double PT_KICK_FMAG;
46  static double PT_KICK_KMAG;
47 
48  //Track quality cuts
49  static double TX_MAX;
50  static double TY_MAX;
51  static double X0_MAX;
52  static double Y0_MAX;
53  static double INVP_MAX;
54  static double INVP_MIN;
55  static double PROB_LOOSE;
56  static double PROB_TIGHT;
57 
58  //Geometric positions
59  static double Z_KMAG_BEND;
60  static double Z_ABSORBER;
61  static double Z_FMAG_BEND;
62  static double Z_KFMAG_BEND;
63  static double ELOSS_KFMAG;
64  static double ELOSS_ABSORBER;
65 
66  //initialize global variables
67  void initGlobalVariables()
68  {
69  if(!inited)
70  {
71  inited = true;
72  p_geomSvc = GeomSvc::instance();
73 
75  KMAG_ON = rc->get_BoolFlag("KMAG_ON");
76  COARSE_MODE = rc->get_BoolFlag("COARSE_MODE");
77 
78  FMAGSTR = rc->get_DoubleFlag("FMAGSTR");
79  KMAGSTR = rc->get_DoubleFlag("KMAGSTR");
80  PT_KICK_FMAG = rc->get_DoubleFlag("PT_KICK_FMAG")*FMAGSTR;
81  PT_KICK_KMAG = rc->get_DoubleFlag("PT_KICK_KMAG")*KMAGSTR;
82 
83  TX_MAX = rc->get_DoubleFlag("TX_MAX");
84  TY_MAX = rc->get_DoubleFlag("TY_MAX");
85  X0_MAX = rc->get_DoubleFlag("X0_MAX");
86  Y0_MAX = rc->get_DoubleFlag("Y0_MAX");
87  INVP_MAX = rc->get_DoubleFlag("INVP_MAX");
88  INVP_MIN = rc->get_DoubleFlag("INVP_MIN");
89  PROB_LOOSE = rc->get_DoubleFlag("PROB_LOOSE");
90  PROB_TIGHT = rc->get_DoubleFlag("PROB_TIGHT");
91 
92  Z_KMAG_BEND = p_geomSvc->Z_KMAG_BEND();
93  Z_ABSORBER = p_geomSvc->Z_ABSORBER();
94  Z_FMAG_BEND = p_geomSvc->Z_FMAG_BEND();
95  Z_KFMAG_BEND = p_geomSvc->Z_KFMAG_BEND();
96  ELOSS_KFMAG = p_geomSvc->ELOSS_KFMAG();
97  ELOSS_ABSORBER = p_geomSvc->ELOSS_ABSORBER();
98  }
99  }
100 }
101 
102 //Signed hit definition
104 {
105 }
106 
107 SignedHit::SignedHit(int detectorID) : sign(0)
108 {
109  hit.index = -1;
110  hit.detectorID = detectorID;
111 }
112 
113 SignedHit::SignedHit(Hit hit_input, int sign_input) : hit(hit_input), sign(sign_input)
114 {
115 }
116 
117 void SignedHit::identify(std::ostream& os) const
118 {
119  if(sign > 0) os << "L - ";
120  if(sign < 0) os << "R - ";
121  if(sign == 0) os << "U - ";
122 
123  os << hit.index << " " << hit.detectorID << " " << hit.elementID << std::endl;
124 }
125 
126 //Proptube segment definition
127 //const GeomSvc* PropSegment::p_geomSvc = GeomSvc::instance();
128 
129 PropSegment::PropSegment() : a(-999.), b(-999.), err_a(100.), err_b(100.), chisq(1.E6), nHodoHits(0)
130 {
131  for(int i = 0; i < 4; ++i) hits[i].hit.index = -1;
132  for(int i = 0; i < 4; ++i) hodoHits[i].index = -1;
133  initGlobalVariables();
134 }
135 
137 {
138  a = -999.;
139  b = -999.;
140  err_a = 100;
141  err_b = 100;
142 
143  for(int i = 0; i < 4; ++i) hits[i].hit.index = -1;
144 
145  chisq = 1E6;
146 }
147 
148 void PropSegment::print(std::ostream& os) const
149 {
150  using namespace std;
151 
152  os << "nHits: " << getNHits() << ", nPlanes: " << getNPlanes() << endl;
153  os << "a = " << a << ", b = " << b << ", chisq = " << chisq << endl;
154  os << "TX_MAX = " << TX_MAX << ", X0_MAX = " << X0_MAX << ", chisq max = " << 5 << endl;
155  //os << "Absorber projection: " << getExpPosition(MUID_Z_REF) << endl;
156 
157  for(int i = 0; i < 4; ++i)
158  {
159  if(hits[i].sign > 0) os << "L: ";
160  if(hits[i].sign < 0) os << "R: ";
161  if(hits[i].sign == 0) os << "U: ";
162 
163  os << hits[i].hit.index << " " << hits[i].hit.detectorID << " " << hits[i].hit.elementID << " === ";
164  }
165  os << endl;
166 }
167 
168 double PropSegment::getClosestApproach(double z, double pos)
169 {
170  return (a*z + b - pos)/sqrt(a*a + 1.);
171 }
172 
173 double PropSegment::getPosRef(double default_val)
174 {
175  if(hits[0].hit.index < 0 && hits[1].hit.index < 0) return default_val;
176 
177  int nRefPoints = 0;
178  double pos_exp = 0.;
179  for(int i = 0; i < 2; ++i)
180  {
181  if(hits[i].hit.index < 0) continue;
182 
183  pos_exp += hits[i].hit.pos;
184  ++nRefPoints;
185  }
186 
187  return pos_exp/nRefPoints;
188 }
189 
191 {
192  int nHits = 0;
193  for(int i = 0; i < 4; ++i)
194  {
195  if(hits[i].hit.index >= 0) ++nHits;
196  }
197  return nHits;
198 }
199 
201 {
202  int nPlanes = 0;
203  for(int i = 0; i < 2; ++i)
204  {
205  if(hits[i].hit.index >= 0 && hits[i+1].hit.index >= 0) ++nPlanes;
206  }
207  return nPlanes;
208 }
209 
211 {
212  if(getNHits() < 2) return 0;
213  if(getNPlanes() != 2) return 0;
214  if(chisq > 5.) return 0;
215 
216  //May need optimization
217  if(fabs(a) > TX_MAX) return 0;
218  if(fabs(b) > X0_MAX) return 0;
219 
220  return 1;
221 }
222 
224 {
225  //Sign assignment for 1st and 2nd hits
226  if(hits[0].hit.index > 0 && hits[1].hit.index > 0)
227  {
228  if(hits[0].hit.elementID == hits[1].hit.elementID)
229  {
230  hits[0].sign = -1;
231  hits[1].sign = 1;
232  }
233  else
234  {
235  hits[0].sign = 1;
236  hits[1].sign = -1;
237  }
238  }
239  else
240  {
241  hits[0].sign = 0;
242  hits[1].sign = 0;
243  }
244 
245  //Sign assignment for 3rd and 4th hits
246  if(hits[2].hit.index > 0 && hits[3].hit.index > 0)
247  {
248  if(hits[2].hit.elementID == hits[3].hit.elementID)
249  {
250  hits[2].sign = -1;
251  hits[3].sign = 1;
252  }
253  else
254  {
255  hits[2].sign = 1;
256  hits[3].sign = -1;
257  }
258  }
259  else
260  {
261  hits[2].sign = 0;
262  hits[3].sign = 0;
263  }
264 }
265 
266 void PropSegment::resolveLR(int settings)
267 {
268  hits[0].sign = 2*(settings & 1) - 1;
269  hits[1].sign = 2*((settings & 2) >> 1) - 1;
270  hits[2].sign = 2*((settings & 4) >> 1) - 1;
271  hits[3].sign = 2*((settings & 8) >> 1) - 1;
272 }
273 
275 {
276  int nHits = getNHits();
277  if(nHits == 2)
278  {
279  fit_2hits();
280  }
281  else
282  {
283  fit_34hits();
284  }
285 
286  //remove one bad hits if possible/needed
287  while(nHits > 2 && chisq > 5.)
288  {
289  int index = -1;
290  double res_max = 0.;
291  for(int i = 0; i < 4; ++i)
292  {
293  if(hits[i].hit.index < 0) continue;
294 
295  double res = fabs(hits[i].pos() - a*p_geomSvc->getPlanePosition(hits[i].hit.detectorID) - b);
296  if(res > res_max)
297  {
298  index = i;
299  res_max = res;
300  }
301  }
302 
303 #ifdef _DEBUG_ON
304  LogInfo("Invoking prop segment re-fitting...");
305  LogInfo("Removing hit " << index << " with residual = " << res_max);
306  LogInfo("Before removing a = " << a << ", b = " << b << ", chisq = " << chisq);
307 #endif
308 
309  //remove the bad hit
310  hits[index].hit.index = -1;
311 
312  //fit again
313  --nHits;
314  if(nHits == 2)
315  {
316  fit_2hits();
317  }
318  else
319  {
320  fit_34hits();
321  }
322 
323 #ifdef _DEBUG_ON
324  LogInfo("After removing a = " << a << ", b = " << b << ", chisq = " << chisq);
325 #endif
326  }
327 }
328 
330 {
331  double z[2], pos[2];
332 
333  int idx = 0;
334  for(int i = 0; i < 4; ++i)
335  {
336  if(hits[i].hit.index < 0) continue;
337 
338  z[idx] = p_geomSvc->getPlanePosition(hits[i].hit.detectorID);
339  pos[idx] = hits[i].hit.pos;
340 
341  ++idx;
342  }
343 
344  a = (pos[1] - pos[0])/(z[1] - z[0]);
345  b = pos[0] - a*z[0];
346  err_a = 1.5;
347  err_b = 1.5;
348  chisq = 0.;
349 }
350 
351 /*
352 void PropSegment::fit_34hits()
353 {
354  int index_min = -1;
355  double chisq_min = 1E8;
356  for(int i = 0; i < 16; ++i)
357  {
358  resolveLR(i);
359 
360  linearFit_iterative();
361  if(chisq < chisq_min)
362  {
363  chisq_min = chisq;
364  index_min = i;
365  }
366  }
367 
368  resolveLR(index_min);
369  linearFit_iterative();
370 }
371 */
372 
374 {
375  resolveLR();
377 }
378 
380 {
381  double sum = 0.;
382  double sx = 0.;
383  double sy = 0.;
384  double sxx = 0.;
385  double syy = 0.;
386  double sxy = 0.;
387 
388  double x[14], y[14];
389  for(int i = 0; i < 4; ++i)
390  {
391  if(hits[i].hit.index < 0) continue;
392 
393  y[i] = hits[i].pos();
394  x[i] = p_geomSvc->getPlanePosition(hits[i].hit.detectorID);
395 
396  ++sum;
397  sx += x[i];
398  sy += y[i];
399  sxx += (x[i]*x[i]);
400  syy += (y[i]*y[i]);
401  sxy += (x[i]*y[i]);
402  }
403 
404  /*
405  for(int i = 0; i < nHodoHits; ++i)
406  {
407  int idx = i + 4;
408  y[idx] = hodoHits[i].pos;
409  x[idx] = p_geomSvc->getPlanePosition(hodoHits[i].detectorID);
410 
411  ++sum;
412  sx += x[idx];
413  sy += y[idx];
414  sxx += (x[idx]*x[idx]);
415  syy += (y[idx]*y[idx]);
416  sxy += (x[idx]*y[idx]);
417  }
418  */
419 
420  double det = sum*sxx - sx*sx;
421  if(fabs(det) < 1E-20) return;
422 
423  a = (sum*sxy - sx*sy)/det;
424  b = (sy*sxx - sxy*sx)/det;
425  err_a = sqrt(fabs(sum/det));
426  err_b = sqrt(fabs(sxx/det));
427 
428  chisq = 0.;
429  for(int i = 0; i < 4; ++i)
430  {
431  if(hits[i].hit.index < 0) continue;
432  chisq += ((y[i] - a*x[i] -b)*(y[i] - a*x[i] -b));
433  }
434 }
435 
437 {
438  //Algorithm refer to Kun's PhD thesis
439 
440  //prepare the raw data
441  int len = 0;
442  double x[4], y[4], r[4]; //note r here is signed drift distance
443  for(int i = 0; i < 4; ++i)
444  {
445  if(hits[i].hit.index < 0) continue;
446 
447  y[len] = hits[i].pos();
448  x[len] = p_geomSvc->getPlanePosition(hits[i].hit.detectorID);
449  r[len] = hits[i].sign*hits[i].hit.driftDistance;
450 
451  ++len;
452  }
453 
454  //while loop with less than 100 iterations
455  a = 0;
456  int iter = 0;
457  double a_prev = -999.; // value of a in previous iteration
458  double _x[4], _y[4]; // corrected hit pos
459  while(fabs(a_prev - a) > 1E-4 && ++iter < 100)
460  {
461  a_prev = a;
462 
463  //prepare corrected hit pos
464  double C, D, E, F, G, H;
465  C = 0.;
466  D = 0.;
467  E = 0.;
468  F = 0.;
469  G = 0.;
470  H = 0.;
471 
472  for(int i = 0; i < len; ++i)
473  {
474  _x[i] = x[i] - r[i]*a/sqrt(a*a + 1.);
475  _y[i] = y[i] + r[i]/sqrt(a*a + 1.);
476 
477  C += 1.;
478  D += _x[i];
479  E += _y[i];
480  F += _x[i]*_x[i];
481  G += _y[i]*_y[i];
482  H += _x[i]*_y[i];
483  }
484 
485  double alpha = H - D*E/C;
486  double beta = F - G -D*D/C + E*E/C;
487 
488  a = (-beta + sqrt(beta*beta + 4*alpha*alpha))/alpha/2.;
489  b = (E - a*D)/C;
490 
491  chisq = 0.;
492  for(int i = 0; i < len; ++i)
493  {
494  chisq += (a*_x[i] + b - _y[i])*(a*_x[i] + b - _y[i])/(a*a + 1.);
495  }
496  }
497 }
498 
499 
500 Tracklet::Tracklet() : stationID(-1), nXHits(0), nUHits(0), nVHits(0), chisq(9999.), chisq_vtx(9999.), tx(0.), ty(0.), x0(0.), y0(0.), invP(0.1), err_tx(-1.), err_ty(-1.), err_x0(-1.), err_y0(-1.), err_invP(-1.)
501 {
502  for(int i = 0; i < nChamberPlanes; i++) residual[i] = 999.;
503  initGlobalVariables();
504 }
505 
506 int Tracklet::isValid() const
507 {
508  if(stationID < 1 || stationID > nStations) return 0;
509  if(fabs(tx) > TX_MAX || fabs(x0) > X0_MAX) return 0;
510  if(fabs(ty) > TY_MAX || fabs(y0) > Y0_MAX) return 0;
511  if(err_tx < 0 || err_ty < 0 || err_x0 < 0 || err_y0 < 0) return 0;
512 
513  double prob = getProb();
514  if(stationID != nStations && prob < PROB_LOOSE) return 0;
515 
516  //Tracklets in each station
517  int nHits = nXHits + nUHits + nVHits;
518  if(stationID < nStations-1)
519  {
520  if(nXHits < 1 || nUHits < 1 || nVHits < 1) return 0;
521  if(nHits < 4) return 0;
522  if(chisq > 40.) return 0;
523  }
524  else
525  {
526  //Number of hits cuts, second index is X, U, V, first index is station-1, 2, 3
527  int nRealHits[3][3] = {{0, 0, 0}, {0, 0, 0}, {0, 0, 0}};
528  nXHits = 0; nUHits = 0; nVHits = 0;
529  for(std::list<SignedHit>::const_iterator iter = hits.begin(); iter != hits.end(); ++iter)
530  {
531  if(iter->hit.index < 0) continue;
532 
533  int idx1 = iter->hit.detectorID <= 12 ? 0 : (iter->hit.detectorID <= 18 ? 1 : 2);
534  int idx2 = p_geomSvc->getPlaneType(iter->hit.detectorID) - 1;
535 
536  ++nRealHits[idx1][idx2];
537  if(idx2 == 0)
538  ++nXHits;
539  else if(idx2 == 1)
540  ++nUHits;
541  else
542  ++nVHits;
543  }
544 
545  //Number of hits cut after removing bad hits
546  for(int i = 1; i < 3; ++i)
547  {
548  if(nRealHits[i][0] < 1 || nRealHits[i][1] < 1 || nRealHits[i][2] < 1) return 0;
549  if(nRealHits[i][0] + nRealHits[i][1] + nRealHits[i][2] < 4) return 0;
550  }
551 
552  //for global tracks only -- TODO: may need to set a new station-1 cut
553  if(stationID == nStations)
554  {
555  if(nRealHits[0][0] < 1 || nRealHits[0][1] < 1 || nRealHits[0][2] < 1) return 0;
556  if(nRealHits[0][0] + nRealHits[0][1] + nRealHits[0][2] < 4) return 0;
557 
558  if(prob < PROB_TIGHT) return 0;
559  if(KMAG_ON)
560  {
561  if(invP < INVP_MIN || invP > INVP_MAX) return 0;
562  }
563  }
564  }
565 
566  return 1;
567 }
568 
569 double Tracklet::getProb() const
570 {
571  int ndf;
572  if(stationID == nStations && KMAG_ON)
573  {
574  ndf = getNHits() - 5;
575  }
576  else
577  {
578  ndf = getNHits() - 4;
579  }
580 
581  return TMath::Prob(chisq, ndf);
582  //return -chisq/ndf;
583 }
584 
585 double Tracklet::getMomProb() const
586 {
587  double weights[40] = {0, 0, 3.1292e-05, 0.00203877, 0.0147764, 0.0417885, 0.08536, 0.152212, 0.250242, 0.322011, 0.327125, 0.275443, 0.220316, 0.189932, 0.162112, 0.131674, 0.100102, 0.0736696, 0.0510353, 0.0364215, 0.0233914, 0.0152907, 0.00992716, 0.00601322, 0.00382757, 0.00239005, 0.00137169, 0.000768309, 0.000413311, 0.00019659, 8.31216e-05, 2.77721e-05, 7.93826e-06, 7.45884e-07, 2.57648e-08, 0, 6.00088e-09, 0, 0, 0};
588  int index = int(1./invP/2.5);
589 
590  return (index >= 0 && index < 40) ? (weights[index] < 1.E-5 ? 1.E-5 : weights[index]) : 1.E-5;
591 }
592 
593 TVector3 Tracklet::getExpMomentum(double z) const
594 {
595  return (KMAG_ON && stationID >= nStations-1 && z < Z_KMAG_BEND - 1.) ? getMomentumSt1() : getMomentumSt3();
596 }
597 
598 double Tracklet::getExpPositionX(double z) const
599 {
600  if(KMAG_ON && stationID >= nStations-1 && z < Z_KMAG_BEND - 1.)
601  {
602  double tx_st1 = tx + PT_KICK_KMAG*invP*getCharge();
603  double x0_st1 = tx*Z_KMAG_BEND + x0 - tx_st1*Z_KMAG_BEND;
604 
605  return x0_st1 + tx_st1*z;
606  }
607  else
608  {
609  return x0 + tx*z;
610  }
611 }
612 
613 double Tracklet::getExpPosErrorX(double z) const
614 {
615  double err_x;
616  if(KMAG_ON && stationID >= nStations-1 && z < Z_KMAG_BEND - 1.)
617  {
618  double err_kick = fabs(err_invP*PT_KICK_KMAG);
619  double err_tx_st1 = err_tx + err_kick;
620  double err_x0_st1 = err_x0 + err_kick*Z_KMAG_BEND;
621 
622  err_x = err_x0_st1 + fabs(err_tx_st1*z);
623  }
624  else
625  {
626  err_x = fabs(err_tx*z) + err_x0;
627  }
628 
629  if(z > Z_ABSORBER) err_x += 1.;
630  return err_x;
631 }
632 
633 double Tracklet::getExpPositionY(double z) const
634 {
635  return y0 + ty*z;
636 }
637 
638 double Tracklet::getExpPosErrorY(double z) const
639 {
640  double err_y = fabs(err_ty*z) + err_y0;
641  if(z > Z_ABSORBER) err_y += 1.;
642 
643  return err_y;
644 }
645 
646 double Tracklet::getExpPositionW(int detectorID) const
647 {
648  double z = p_geomSvc->getPlanePosition(detectorID);
649 
650  double x_exp = getExpPositionX(z);
651  double y_exp = getExpPositionY(z);
652 
653  if(!p_geomSvc->isInPlane(detectorID, x_exp, y_exp)) return 999999.;
654  return p_geomSvc->getCostheta(detectorID)*x_exp + p_geomSvc->getSintheta(detectorID)*y_exp;
655 }
656 
657 int Tracklet::getExpElementID(int detectorID) const
658 {
659  return p_geomSvc->getExpElementID(detectorID, getExpPositionW(detectorID));
660 }
661 
662 bool Tracklet::operator<(const Tracklet& elem) const
663 {
664  //return nXHits + nUHits + nVHits - 0.4*chisq > elem.nXHits + elem.nUHits + elem.nVHits - 0.4*elem.chisq;
665  if(getNHits() == elem.getNHits())
666  {
667  return chisq < elem.chisq;
668  }
669  else
670  {
671  return getProb() > elem.getProb();
672  }
673 }
674 
675 bool Tracklet::similarity(const Tracklet& elem) const
676 {
677  int nCommonHits = 0;
678  std::list<SignedHit>::const_iterator first = hits.begin();
679  std::list<SignedHit>::const_iterator second = elem.hits.begin();
680 
681  while(first != hits.end() && second != elem.hits.end())
682  {
683  if((*first) < (*second))
684  {
685  ++first;
686  }
687  else if((*second) < (*first))
688  {
689  ++second;
690  }
691  else
692  {
693  if((*first) == (*second)) nCommonHits++;
694  ++first;
695  ++second;
696  }
697  }
698 
699  if(nCommonHits/double(elem.getNHits()) > 0.33333) return true;
700  return false;
701 }
702 
703 double Tracklet::getMomentum() const
704 {
705  //Ref. SEAQUEST-doc-453-v3 by Don. Geesaman
706  //if(KMAG_ON == 0) return 1E8;
707 
708  double p = 50.;
709  double charge = getCharge();
710 
711  double c1 = Z_FMAG_BEND*PT_KICK_FMAG*charge;
712  double c2 = Z_KMAG_BEND*PT_KICK_KMAG*charge;
713  double c3 = -x0;
714  double c4 = ELOSS_KFMAG/2.;
715  double c5 = ELOSS_KFMAG;
716 
717  double b = c1/c3 + c2/c3 - c4 - c5;
718  double c = c4*c5 - c1*c5/c3 - c2*c4/c3;
719 
720  double disc = b*b - 4*c;
721  if(disc > 0.)
722  {
723  p = (-b + sqrt(disc))/2. - ELOSS_KFMAG;
724  }
725 
726  if(p < 10. || p > 120. || disc < 0)
727  {
728  double k = fabs(getExpPositionX(Z_KFMAG_BEND)/Z_KFMAG_BEND - tx);
729  p = 1./(0.00832161 + 0.184186*k - 0.104132*k*k) + ELOSS_ABSORBER;
730  }
731 
732  return p;
733 }
734 
736 
746 {
747  return -3e-3 * copysign(1.0, FMAGSTR) * x0 < tx ? +1 : -1;
748 }
749 
750 void Tracklet::getXZInfoInSt1(double& tx_st1, double& x0_st1) const
751 {
752  if(KMAG_ON)
753  {
754  tx_st1 = tx + PT_KICK_KMAG*invP*getCharge();
755  x0_st1 = tx*Z_KMAG_BEND + x0 - tx_st1*Z_KMAG_BEND;
756  }
757  else
758  {
759  tx_st1 = tx;
760  x0_st1 = x0;
761  }
762 }
763 
764 void Tracklet::getXZErrorInSt1(double& err_tx_st1, double& err_x0_st1) const
765 {
766  if(KMAG_ON)
767  {
768  double err_kick = fabs(err_invP*PT_KICK_KMAG);
769  err_tx_st1 = err_tx + err_kick;
770  err_x0_st1 = err_x0 + err_kick*Z_KMAG_BEND;
771  }
772  else
773  {
774  err_tx_st1 = err_tx;
775  err_x0_st1 = err_x0;
776  }
777 }
778 
780 {
781  Tracklet tracklet;
782  tracklet.stationID = nStations - 1;
783 
784  tracklet.nXHits = nXHits + elem.nXHits;
785  tracklet.nUHits = nUHits + elem.nUHits;
786  tracklet.nVHits = nVHits + elem.nVHits;
787 
788  tracklet.hits.assign(hits.begin(), hits.end());
789  if(elem.stationID > stationID)
790  {
791  tracklet.hits.insert(tracklet.hits.end(), elem.hits.begin(), elem.hits.end());
792  }
793  else
794  {
795  tracklet.hits.insert(tracklet.hits.begin(), elem.hits.begin(), elem.hits.end());
796  }
797 
798  tracklet.err_tx = 1./sqrt(1./err_tx/err_tx + 1./elem.err_tx/elem.err_tx);
799  tracklet.err_ty = 1./sqrt(1./err_ty/err_ty + 1./elem.err_ty/elem.err_ty);
800  tracklet.err_x0 = 1./sqrt(1./err_x0/err_x0 + 1./elem.err_x0/elem.err_x0);
801  tracklet.err_y0 = 1./sqrt(1./err_y0/err_y0 + 1./elem.err_y0/elem.err_y0);
802 
803  tracklet.tx = (tx/err_tx/err_tx + elem.tx/elem.err_tx/elem.err_tx)*tracklet.err_tx*tracklet.err_tx;
804  tracklet.ty = (ty/err_ty/err_ty + elem.ty/elem.err_ty/elem.err_ty)*tracklet.err_ty*tracklet.err_ty;
805  tracklet.x0 = (x0/err_x0/err_x0 + elem.x0/elem.err_x0/elem.err_x0)*tracklet.err_x0*tracklet.err_x0;
806  tracklet.y0 = (y0/err_y0/err_y0 + elem.y0/elem.err_y0/elem.err_y0)*tracklet.err_y0*tracklet.err_y0;
807 
808  tracklet.invP = 1./tracklet.getMomentum();
809  tracklet.err_invP = 0.25*tracklet.invP;
810 
811  tracklet.calcChisq();
812  return tracklet;
813 }
814 
816 {
817  Tracklet tracklet;
818  tracklet.stationID = nStations;
819 
820  tracklet.nXHits = nXHits + elem.nXHits;
821  tracklet.nUHits = nUHits + elem.nUHits;
822  tracklet.nVHits = nVHits + elem.nVHits;
823 
824  tracklet.hits.assign(hits.begin(), hits.end());
825  if(elem.stationID > stationID)
826  {
827  tracklet.hits.insert(tracklet.hits.end(), elem.hits.begin(), elem.hits.end());
828  }
829  else
830  {
831  tracklet.hits.insert(tracklet.hits.begin(), elem.hits.begin(), elem.hits.end());
832  }
833 
834  if(elem.stationID == nStations - 1)
835  {
836  tracklet.tx = elem.tx;
837  tracklet.ty = elem.ty;
838  tracklet.x0 = elem.x0;
839  tracklet.y0 = elem.y0;
840  tracklet.invP = 1./elem.getMomentum();
841 
842  tracklet.err_tx = elem.err_tx;
843  tracklet.err_ty = elem.err_ty;
844  tracklet.err_x0 = elem.err_x0;
845  tracklet.err_y0 = elem.err_y0;
846  tracklet.err_invP = 0.25*tracklet.invP;
847  }
848  else
849  {
850  tracklet.tx = tx;
851  tracklet.ty = ty;
852  tracklet.x0 = x0;
853  tracklet.y0 = y0;
854  tracklet.invP = 1./getMomentum();
855 
856  tracklet.err_tx = err_tx;
857  tracklet.err_ty = err_ty;
858  tracklet.err_x0 = err_x0;
859  tracklet.err_y0 = err_y0;
860  tracklet.err_invP = 0.25*tracklet.invP;
861  }
862 
863  tracklet.calcChisq();
864  return tracklet;
865 }
866 
868 {
869  Tracklet tracklet;
870  tracklet.stationID = stationID;
871 
872  tracklet.tx = tx;
873  tracklet.ty = ty;
874  tracklet.x0 = x0;
875  tracklet.y0 = y0;
876  tracklet.invP = 1./getMomentum();
877 
878  tracklet.err_tx = err_tx;
879  tracklet.err_ty = err_ty;
880  tracklet.err_x0 = err_x0;
881  tracklet.err_y0 = err_y0;
882  tracklet.err_invP = 0.25*tracklet.invP;
883 
884  tracklet.chisq_vtx = chisq_vtx < 999 ? chisq_vtx : elem.chisq_vtx;
885 
886  tracklet.seg_x = seg_x;
887  tracklet.seg_y = seg_y;
888 
889  std::list<SignedHit>::iterator elemend = elem.hits.begin();
890  for(int i = 0; i < 6; ++i) ++elemend;
891  tracklet.hits.insert(tracklet.hits.begin(), hits.begin(), hits.end());
892  tracklet.hits.insert(tracklet.hits.begin(), elem.hits.begin(), elemend);
893  tracklet.hits.sort();
894 
895  tracklet.calcChisq();
896  tracklet.isValid();
897  return tracklet;
898 }
899 
901 {
902  std::vector<int> detectorIDs_all;
903  for(int i = stationID*6 - 5; i <= stationID*6; ++i) detectorIDs_all.push_back(i);
904 
905  std::vector<int> detectorIDs_now;
906  for(std::list<SignedHit>::const_iterator iter = hits.begin(); iter != hits.end(); ++iter)
907  {
908  detectorIDs_now.push_back(iter->hit.detectorID);
909  }
910 
911  std::vector<int> detectorIDs_miss(6);
912  std::vector<int>::iterator iter = std::set_difference(detectorIDs_all.begin(), detectorIDs_all.end(), detectorIDs_now.begin(), detectorIDs_now.end(), detectorIDs_miss.begin());
913  detectorIDs_miss.resize(iter - detectorIDs_miss.begin());
914 
915  for(std::vector<int>::iterator iter = detectorIDs_miss.begin(); iter != detectorIDs_miss.end(); ++iter)
916  {
917  SignedHit dummy;
918  dummy.hit.detectorID = *iter;
919 
920  hits.push_back(dummy);
921  }
922 
923  sortHits();
924 }
925 
930 {
931  chisq = 0.;
932 
933  double tx_st1, x0_st1;
934  if(stationID == nStations && KMAG_ON)
935  {
936  getXZInfoInSt1(tx_st1, x0_st1);
937  }
938 
939  for(std::list<SignedHit>::const_iterator iter = hits.begin(); iter != hits.end(); ++iter)
940  {
941  if(iter->hit.index < 0) continue;
942 
943  int detectorID = iter->hit.detectorID;
944  int index = detectorID - 1;
945 
946  double sigma;
947  if(iter->sign == 0 || COARSE_MODE)
948  sigma = p_geomSvc->getPlaneSpacing(detectorID)/sqrt(12.);
949  //sigma = fabs(iter->hit.driftDistance)/sqrt(12.);
950  else
951  sigma = p_geomSvc->getPlaneResolution(detectorID);
952 
953  //double p = iter->hit.pos + iter->sign*fabs(iter->hit.driftDistance);
954  if(KMAG_ON && stationID == nStations && detectorID <= 12)
955  {
956  //residual[index] = p - p_geomSvc->getInterception(detectorID, tx_st1, ty, x0_st1, y0);
957  residual[index] = iter->sign*fabs(iter->hit.driftDistance) - p_geomSvc->getDCA(detectorID, iter->hit.elementID, tx_st1, ty, x0_st1, y0);
958  }
959  else
960  {
961  //residual[index] = p - p_geomSvc->getInterception(detectorID, tx, ty, x0, y0);
962  residual[index] = iter->sign*fabs(iter->hit.driftDistance) - p_geomSvc->getDCA(detectorID, iter->hit.elementID, tx, ty, x0, y0);
963  }
964 
965  chisq += (residual[index]*residual[index]/sigma/sigma);
966  //std::cout << iter->hit.detectorID << " " << iter->hit.elementID << " " << iter->sign << " " << iter->hit.pos << " " << iter->hit.driftDistance << " " << residual[index] << " " << sigma << " " << chisq << std::endl;
967  }
968 
969  //std::cout << chisq << std::endl;
970  return chisq;
971 }
972 
974 {
975  int id = 0;
976  for(std::list<SignedHit>::const_iterator iter = hits.begin(); iter != hits.end(); ++iter)
977  {
978  if(id == index) return *iter;
979  id++;
980  }
981 
982  SignedHit dummy;
983  return dummy;
984 }
985 
986 double Tracklet::Eval(const double* par)
987 {
988  tx = par[0];
989  ty = par[1];
990  x0 = par[2];
991  y0 = par[3];
992  if(KMAG_ON) invP = par[4];
993 
994  //std::cout << tx << " " << ty << " " << x0 << " " << y0 << " " << 1./invP << std::endl;
995  return calcChisq();
996 }
997 
998 double Tracklet::Eval4(const double* par)
999 {
1000  tx = par[0];
1001  ty = par[1];
1002  x0 = par[2];
1003  y0 = par[3];
1004 
1005  //std::cout << tx << " " << ty << " " << x0 << " " << y0 << " " << 1./invP << std::endl;
1006  return calcChisq();
1007 }
1008 
1010 {
1011  SRecTrack strack;
1012  strack.setChisq(chisq);
1013  for(std::list<SignedHit>::iterator iter = hits.begin(); iter != hits.end(); ++iter)
1014  {
1015  if(iter->hit.index < 0) continue;
1016 
1017  double z = p_geomSvc->getPlanePosition(iter->hit.detectorID);
1018  double tx_val, tx_err, x0_val, x0_err;
1019  if(iter->hit.detectorID <= 12)
1020  {
1021  getXZInfoInSt1(tx_val, x0_val);
1022  getXZErrorInSt1(tx_err, x0_err);
1023  }
1024  else
1025  {
1026  tx_val = tx;
1027  x0_val = x0;
1028  tx_err = err_tx;
1029  x0_err = err_x0;
1030  }
1031 
1032  TMatrixD state(5, 1), covar(5, 5);
1033  state[0][0] = getCharge()*invP*sqrt((1. + tx_val*tx_val)/(1. + tx_val*tx_val + ty*ty));
1034  state[1][0] = tx_val;
1035  state[2][0] = ty;
1036  state[3][0] = getExpPositionX(z);
1037  state[4][0] = getExpPositionY(z);
1038 
1039  covar.Zero();
1040  covar[0][0] = err_invP*err_invP;
1041  covar[1][1] = tx_err*tx_err;
1042  covar[2][2] = err_ty*err_ty;
1043  covar[3][3] = getExpPosErrorX(z)*getExpPosErrorX(z);
1044  covar[4][4] = getExpPosErrorY(z)*getExpPosErrorY(z);
1045 
1046  strack.insertHitIndex(iter->hit.index*iter->sign);
1047  strack.insertStateVector(state);
1048  strack.insertCovariance(covar);
1049  strack.insertZ(z);
1050  }
1051 
1052  //Set single vertex swimming
1053  strack.swimToVertex(nullptr, nullptr, hyptest);
1054 
1055  //Set trigger road info
1056  TriggerRoad road(*this);
1057  strack.setTriggerRoad(road.getRoadID());
1058 
1059  //Set prop tube slopes
1060  strack.setNHitsInPT(seg_x.getNHits(), seg_y.getNHits());
1061  strack.setPTSlope(seg_x.a, seg_y.a);
1062 
1063  return strack;
1064 }
1065 
1067 {
1068  double tx_st1, x0_st1;
1069  getXZInfoInSt1(tx_st1, x0_st1);
1070 
1071  double pz = 1./invP/sqrt(1. + tx_st1*tx_st1);
1072  return TVector3(pz*tx_st1, pz*ty, pz);
1073 }
1074 
1076 {
1077  double pz = 1./invP/sqrt(1. + tx*tx);
1078  return TVector3(pz*tx, pz*ty, pz);
1079 }
1080 
1081 void Tracklet::print(std::ostream& os)
1082 {
1083  using namespace std;
1084 
1085  calcChisq();
1086  TVector3 mom1 = getMomentumSt1();
1087  TVector3 mom3 = getMomentumSt3();
1088 
1089  os << "Tracklet in station " << stationID << endl;
1090  os << nXHits + nUHits + nVHits << " hits in this station with chisq = " << chisq << endl;
1091  os << "Momentum in z @st1: " << mom1.Pz() << " +/- " << mom1.Pz()*err_invP/invP << endl;
1092  os << "Momentum in z @st3: " << mom3.Pz() << " +/- " << mom3.Pz()*err_invP/invP << endl;
1093  os << "Charge: " << getCharge() << endl;
1094  for(std::list<SignedHit>::iterator iter = hits.begin(); iter != hits.end(); ++iter)
1095  {
1096  if(iter->sign > 0) os << "L: ";
1097  if(iter->sign < 0) os << "R: ";
1098  if(iter->sign == 0) os << "U: ";
1099 
1100  os << iter->hit.index << " " << p_geomSvc->getDetectorName(iter->hit.detectorID) << "(" << iter->hit.detectorID << ") " << iter->hit.elementID << " " << residual[iter->hit.detectorID-1] << " = ";
1101  }
1102  os << endl;
1103 
1104  os << "X-Z: (" << tx << " +/- " << err_tx << ")*z + (" << x0 << " +/- " << err_x0 << ")" << endl;
1105  os << "Y-Z: (" << ty << " +/- " << err_ty << ")*z + (" << y0 << " +/- " << err_y0 << ")" << endl;
1106 
1107  os << "KMAG projection: X = " << getExpPositionX(Z_KMAG_BEND) << " +/- " << getExpPosErrorX(Z_KMAG_BEND) << endl;
1108  os << "KMAG projection: Y = " << getExpPositionY(Z_KMAG_BEND) << " +/- " << getExpPosErrorY(Z_KMAG_BEND) << endl;
1109  os << "KMAGSTR = " << KMAGSTR << endl;
1110 }
1111 
1113 {}
1114 
1116 {
1117  Reset();
1118 }
1119 
1120 void TrackletVector::identify(std::ostream& os) const
1121 {
1122  os << "TrackletVector with " << size() << " entries" << std::endl;
1123 }
1124 
1126 {
1127  for(auto iter = trackletVec.begin(); iter != trackletVec.end(); ++iter) delete (*iter);
1128  trackletVec.clear();
1129 }
1130 
1131 const Tracklet* TrackletVector::at(const size_t index) const
1132 {
1133  if(index >= size()) return nullptr;
1134  return trackletVec[index];
1135 }
1136 
1137 Tracklet* TrackletVector::at(const size_t index)
1138 {
1139  if(index >= size()) return nullptr;
1140  return trackletVec[index];
1141 }
1142 
1144 {
1145  trackletVec.push_back(tracklet->Clone());
1146 }
1147 
1148 size_t TrackletVector::erase(const size_t index)
1149 {
1150  if(index >= size()) return size();
1151 
1152  delete trackletVec[index];
1153  trackletVec.erase(trackletVec.begin() + index);
1154  return trackletVec.size();
1155 }
#define LogInfo(message)
Definition: DbSvc.cc:15
ClassImp(SignedHit) ClassImp(PropSegment) ClassImp(Tracklet) ClassImp(TrackletVector) namespace
#define nChamberPlanes
Definition: GlobalConsts.h:6
#define nStations
Definition: GlobalConsts.h:5
const double FMAGSTR
Definition: MacroCommon.h:2
const double KMAGSTR
Definition: MacroCommon.h:3
User interface class about the geometry of detector planes.
Definition: GeomSvc.h:164
double ELOSS_KFMAG() const
Definition: GeomSvc.h:317
double Z_KMAG_BEND() const
Getter/setters for a set of fixed parameters - should not be changed unless absolutely necessary.
Definition: GeomSvc.h:311
double ELOSS_ABSORBER() const
Definition: GeomSvc.h:319
static GeomSvc * instance()
singlton instance
Definition: GeomSvc.cxx:212
double Z_FMAG_BEND() const
Definition: GeomSvc.h:313
double Z_ABSORBER() const
Definition: GeomSvc.h:323
double Z_KFMAG_BEND() const
Definition: GeomSvc.h:315
Definition of hit structure.
Definition: SRawEvent.h:35
Int_t index
Definition: SRawEvent.h:77
Float_t pos
Definition: SRawEvent.h:82
Short_t elementID
Definition: SRawEvent.h:79
Short_t detectorID
Definition: SRawEvent.h:78
Float_t driftDistance
Definition: SRawEvent.h:81
virtual double get_DoubleFlag(const std::string &name) const
Definition: PHFlag.cc:49
virtual bool get_BoolFlag(const std::string &name) const
Definition: PHFlag.cc:151
void linearFit_iterative()
SignedHit hits[4]
Definition: FastTracklet.h:118
Hit hodoHits[4]
Definition: FastTracklet.h:115
double getClosestApproach(double z, double pos)
int getNPlanes() const
double chisq
Definition: FastTracklet.h:111
double err_a
Definition: FastTracklet.h:107
void resolveLR()
void print(std::ostream &os=std::cout) const
int getNHits() const
void fit_34hits()
int isValid() const
isValid returns non zero if object contains vailid data
void fit_2hits()
double getPosRef(double default_val=-9999.)
double err_b
Definition: FastTracklet.h:108
void linearFit_simple()
void setNHitsInPT(Int_t nHitsX, Int_t nHitsY)
Definition: SRecEvent.h:226
void setChisq(Double_t chisq)
Sets.
Definition: SRecEvent.h:154
void insertCovariance(TMatrixD covar)
Definition: SRecEvent.h:157
void setPTSlope(Double_t slopeX, Double_t slopeY)
Prop. tube muon ID info.
Definition: SRecEvent.h:225
void setTriggerRoad(Int_t roadID)
Trigger road info.
Definition: SRecEvent.h:221
void insertZ(Double_t z)
Definition: SRecEvent.h:158
void swimToVertex(TVector3 *pos=nullptr, TVector3 *mom=nullptr, bool hyptest=true)
Simple swim to vertex.
Definition: SRecEvent.cxx:402
void insertHitIndex(Int_t index)
Definition: SRecEvent.h:155
void insertStateVector(TMatrixD state)
Definition: SRecEvent.h:156
double pos()
Definition: FastTracklet.h:44
void identify(std::ostream &os=std::cout) const
void identify(std::ostream &os=std::cout) const
const Tracklet * at(const size_t index) const
virtual ~TrackletVector()
size_t erase(const size_t index)
void Reset()
Clear Event.
void push_back(const Tracklet *tracklet)
size_t size() const
Definition: FastTracklet.h:265
PropSegment seg_x
Definition: FastTracklet.h:231
double invP
Definition: FastTracklet.h:239
double err_invP
Definition: FastTracklet.h:245
double getExpPositionY(double z) const
std::list< SignedHit > hits
Definition: FastTracklet.h:228
double y0
Definition: FastTracklet.h:238
Tracklet * Clone() const
Definition: FastTracklet.h:132
bool operator<(const Tracklet &elem) const
int stationID
Definition: FastTracklet.h:214
TVector3 getMomentumSt1() const
PropSegment seg_y
Definition: FastTracklet.h:232
int isValid() const
isValid returns non zero if object contains vailid data
double ty
Definition: FastTracklet.h:236
double err_y0
Definition: FastTracklet.h:244
double err_x0
Definition: FastTracklet.h:243
double tx
Definition: FastTracklet.h:235
double Eval(const double *par)
double x0
Definition: FastTracklet.h:237
void sortHits()
Definition: FastTracklet.h:142
double getExpPosErrorY(double z) const
double Eval4(const double *par)
void getXZErrorInSt1(double &err_tx_st1, double &err_x0_st1) const
Tracklet operator*(const Tracklet &elem) const
int getNHits() const
Definition: FastTracklet.h:145
TVector3 getMomentumSt3() const
double getExpPosErrorX(double z) const
double chisq_vtx
Definition: FastTracklet.h:225
int getExpElementID(int detectorID) const
double getProb() const
TVector3 getExpMomentum(double z) const
double getMomProb() const
bool similarity(const Tracklet &elem) const
double chisq
Definition: FastTracklet.h:222
double residual[nChamberPlanes]
Definition: FastTracklet.h:248
double getExpPositionX(double z) const
SRecTrack getSRecTrack(bool hyptest=true)
SignedHit getSignedHit(int index)
double getExpPositionW(int detectorID) const
double getMomentum() const
double err_tx
Definition: FastTracklet.h:241
Tracklet operator+(const Tracklet &elem) const
double err_ty
Definition: FastTracklet.h:242
void print(std::ostream &os=std::cout)
Tracklet merge(Tracklet &elem)
int getCharge() const
Return the charge (+1 or -1) of this tracklet.
double calcChisq()
void addDummyHits()
void getXZInfoInSt1(double &tx_st1, double &x0_st1) const
static recoConsts * instance()
Definition: recoConsts.cc:7