CMS 3D CMS Logo

 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Pages
MuonSelectors.cc
Go to the documentation of this file.
7 
8 namespace muon {
10 {
11  static SelectionTypeStringToEnum selectionTypeStringToEnumMap[] = {
12  { "All", All },
13  { "AllGlobalMuons", AllGlobalMuons },
14  { "AllStandAloneMuons", AllStandAloneMuons },
15  { "AllTrackerMuons", AllTrackerMuons },
16  { "TrackerMuonArbitrated", TrackerMuonArbitrated },
17  { "AllArbitrated", AllArbitrated },
18  { "GlobalMuonPromptTight", GlobalMuonPromptTight },
19  { "TMLastStationLoose", TMLastStationLoose },
20  { "TMLastStationTight", TMLastStationTight },
21  { "TM2DCompatibilityLoose", TM2DCompatibilityLoose },
22  { "TM2DCompatibilityTight", TM2DCompatibilityTight },
23  { "TMOneStationLoose", TMOneStationLoose },
24  { "TMOneStationTight", TMOneStationTight },
25  { "TMLastStationOptimizedLowPtLoose", TMLastStationOptimizedLowPtLoose },
26  { "TMLastStationOptimizedLowPtTight", TMLastStationOptimizedLowPtTight },
27  { "GMTkChiCompatibility", GMTkChiCompatibility },
28  { "GMStaChiCompatibility", GMStaChiCompatibility},
29  { "GMTkKinkTight", GMTkKinkTight},
30  { "TMLastStationAngLoose", TMLastStationAngLoose },
31  { "TMLastStationAngTight", TMLastStationAngTight },
32  { "TMOneStationAngLoose", TMOneStationAngLoose },
33  { "TMOneStationAngTight", TMOneStationAngTight },
34  { "TMLastStationOptimizedBarrelLowPtLoose", TMLastStationOptimizedBarrelLowPtLoose },
35  { "TMLastStationOptimizedBarrelLowPtTight", TMLastStationOptimizedBarrelLowPtTight },
36  { "RPCMuLoose", RPCMuLoose },
37  { 0, (SelectionType)-1 }
38  };
39 
41  bool found = false;
42  for(int i = 0; selectionTypeStringToEnumMap[i].label && (! found); ++i)
43  if (! strcmp(label.c_str(), selectionTypeStringToEnumMap[i].label)) {
44  found = true;
45  value = selectionTypeStringToEnumMap[i].value;
46  }
47 
48  // in case of unrecognized selection type
49  if (! found) throw cms::Exception("MuonSelectorError") << label << " is not a recognized SelectionType";
50  return value;
51 }
52 }
53 
55  double maxChamberDist,
56  double maxChamberDistPull,
57  reco::Muon::ArbitrationType arbitrationType )
58 {
59  unsigned int theMask = 0;
60 
61  for(int stationIdx = 1; stationIdx < 5; ++stationIdx)
62  for(int detectorIdx = 1; detectorIdx < 3; ++detectorIdx)
63  if(muon.trackDist(stationIdx,detectorIdx,arbitrationType) < maxChamberDist &&
64  muon.trackDist(stationIdx,detectorIdx,arbitrationType)/muon.trackDistErr(stationIdx,detectorIdx,arbitrationType) < maxChamberDistPull)
65  theMask += 1<<((stationIdx-1)+4*(detectorIdx-1));
66 
67  return theMask;
68 }
69 
70 // ------------ method to calculate the calo compatibility for a track with matched muon info ------------
72  return muon.caloCompatibility();
73 }
74 
75 // ------------ method to calculate the segment compatibility for a track with matched muon info ------------
77  bool use_weight_regain_at_chamber_boundary = true;
78  bool use_match_dist_penalty = true;
79 
80  int nr_of_stations_crossed = 0;
81  int nr_of_stations_with_segment = 0;
82  std::vector<int> stations_w_track(8);
83  std::vector<int> station_has_segmentmatch(8);
84  std::vector<int> station_was_crossed(8);
85  std::vector<float> stations_w_track_at_boundary(8);
86  std::vector<float> station_weight(8);
87  int position_in_stations = 0;
88  float full_weight = 0.;
89 
90  for(int i = 1; i<=8; ++i) {
91  // ********************************************************;
92  // *** fill local info for this muon (do some counting) ***;
93  // ************** begin ***********************************;
94  if(i<=4) { // this is the section for the DTs
95  if( muon.trackDist(i,1,arbitrationType) < 999999 ) { //current "raw" info that a track is close to a chamber
96  ++nr_of_stations_crossed;
97  station_was_crossed[i-1] = 1;
98  if(muon.trackDist(i,1,arbitrationType) > -10. ) stations_w_track_at_boundary[i-1] = muon.trackDist(i,1,arbitrationType);
99  else stations_w_track_at_boundary[i-1] = 0.;
100  }
101  if( muon.segmentX(i,1,arbitrationType) < 999999 ) { //current "raw" info that a segment is matched to the current track
102  ++nr_of_stations_with_segment;
103  station_has_segmentmatch[i-1] = 1;
104  }
105  }
106  else { // this is the section for the CSCs
107  if( muon.trackDist(i-4,2,arbitrationType) < 999999 ) { //current "raw" info that a track is close to a chamber
108  ++nr_of_stations_crossed;
109  station_was_crossed[i-1] = 1;
110  if(muon.trackDist(i-4,2,arbitrationType) > -10. ) stations_w_track_at_boundary[i-1] = muon.trackDist(i-4,2,arbitrationType);
111  else stations_w_track_at_boundary[i-1] = 0.;
112  }
113  if( muon.segmentX(i-4,2,arbitrationType) < 999999 ) { //current "raw" info that a segment is matched to the current track
114  ++nr_of_stations_with_segment;
115  station_has_segmentmatch[i-1] = 1;
116  }
117  }
118  // rough estimation of chamber border efficiency (should be parametrized better, this is just a quick guess):
119  // TF1 * merf = new TF1("merf","-0.5*(TMath::Erf(x/6.)-1)",-100,100);
120  // use above value to "unpunish" missing segment if close to border, i.e. rather than not adding any weight, add
121  // the one from the function. Only for dist ~> -10 cm, else full punish!.
122 
123  // ********************************************************;
124  // *** fill local info for this muon (do some counting) ***;
125  // ************** end *************************************;
126  }
127 
128  // ********************************************************;
129  // *** calculate weights for each station *****************;
130  // ************** begin ***********************************;
131  // const float slope = 0.5;
132  // const float attenuate_weight_regain = 1.;
133  // if attenuate_weight_regain < 1., additional punishment if track is close to boundary and no segment
134  const float attenuate_weight_regain = 0.5;
135 
136  for(int i = 1; i<=8; ++i) { // loop over all possible stations
137 
138  // first set all weights if a station has been crossed
139  // later penalize if a station did not have a matching segment
140 
141  //old logic if(station_has_segmentmatch[i-1] > 0 ) { // the track has an associated segment at the current station
142  if( station_was_crossed[i-1] > 0 ) { // the track crossed this chamber (or was nearby)
143  // - Apply a weight depending on the "depth" of the muon passage.
144  // - The station_weight is later reduced for stations with badly matched segments.
145  // - Even if there is no segment but the track passes close to a chamber boundary, the
146  // weight is set non zero and can go up to 0.5 of the full weight if the track is quite
147  // far from any station.
148  ++position_in_stations;
149 
150  switch ( nr_of_stations_crossed ) { // define different weights depending on how many stations were crossed
151  case 1 :
152  station_weight[i-1] = 1.;
153  break;
154  case 2 :
155  if ( position_in_stations == 1 ) station_weight[i-1] = 0.33;
156  else station_weight[i-1] = 0.67;
157  break;
158  case 3 :
159  if ( position_in_stations == 1 ) station_weight[i-1] = 0.23;
160  else if( position_in_stations == 2 ) station_weight[i-1] = 0.33;
161  else station_weight[i-1] = 0.44;
162  break;
163  case 4 :
164  if ( position_in_stations == 1 ) station_weight[i-1] = 0.10;
165  else if( position_in_stations == 2 ) station_weight[i-1] = 0.20;
166  else if( position_in_stations == 3 ) station_weight[i-1] = 0.30;
167  else station_weight[i-1] = 0.40;
168  break;
169 
170  default :
171 // LogTrace("MuonIdentification")<<" // Message: A muon candidate track has more than 4 stations with matching segments.";
172 // LogTrace("MuonIdentification")<<" // Did not expect this - please let me know: ibloch@fnal.gov";
173  // for all other cases
174  station_weight[i-1] = 1./nr_of_stations_crossed;
175  }
176 
177  if( use_weight_regain_at_chamber_boundary ) { // reconstitute some weight if there is no match but the segment is close to a boundary:
178  if(station_has_segmentmatch[i-1] <= 0 && stations_w_track_at_boundary[i-1] != 0. ) {
179  // if segment is not present but track in inefficient region, do not count as "missing match" but add some reduced weight.
180  // original "match weight" is currently reduced by at least attenuate_weight_regain, variing with an error function down to 0 if the track is
181  // inside the chamber.
182  station_weight[i-1] = station_weight[i-1]*attenuate_weight_regain*0.5*(TMath::Erf(stations_w_track_at_boundary[i-1]/6.)+1.); // remark: the additional scale of 0.5 normalizes Err to run from 0 to 1 in y
183  }
184  else if(station_has_segmentmatch[i-1] <= 0 && stations_w_track_at_boundary[i-1] == 0.) { // no segment match and track well inside chamber
185  // full penalization
186  station_weight[i-1] = 0.;
187  }
188  }
189  else { // always fully penalize tracks with no matching segment, whether the segment is close to the boundary or not.
190  if(station_has_segmentmatch[i-1] <= 0) station_weight[i-1] = 0.;
191  }
192 
193  if( station_has_segmentmatch[i-1] > 0 && 42 == 42 ) { // if track has matching segment, but the matching is not high quality, penalize
194  if(i<=4) { // we are in the DTs
195  if( muon.dY(i,1,arbitrationType) < 999999 && muon.dX(i,1,arbitrationType) < 999999) { // have both X and Y match
196  if(
197  TMath::Sqrt(TMath::Power(muon.pullX(i,1,arbitrationType),2.)+TMath::Power(muon.pullY(i,1,arbitrationType),2.))> 1. ) {
198  // reduce weight
199  if(use_match_dist_penalty) {
200  // only use pull if 3 sigma is not smaller than 3 cm
201  if(TMath::Sqrt(TMath::Power(muon.dX(i,1,arbitrationType),2.)+TMath::Power(muon.dY(i,1,arbitrationType),2.)) < 3. && TMath::Sqrt(TMath::Power(muon.pullX(i,1,arbitrationType),2.)+TMath::Power(muon.pullY(i,1,arbitrationType),2.)) > 3. ) {
202  station_weight[i-1] *= 1./TMath::Power(
203  TMath::Max((double)TMath::Sqrt(TMath::Power(muon.dX(i,1,arbitrationType),2.)+TMath::Power(muon.dY(i,1,arbitrationType),2.)),(double)1.),.25);
204  }
205  else {
206  station_weight[i-1] *= 1./TMath::Power(
207  TMath::Sqrt(TMath::Power(muon.pullX(i,1,arbitrationType),2.)+TMath::Power(muon.pullY(i,1,arbitrationType),2.)),.25);
208  }
209  }
210  }
211  }
212  else if (muon.dY(i,1,arbitrationType) >= 999999) { // has no match in Y
213  if( muon.pullX(i,1,arbitrationType) > 1. ) { // has a match in X. Pull larger that 1 to avoid increasing the weight (just penalize, don't anti-penalize)
214  // reduce weight
215  if(use_match_dist_penalty) {
216  // only use pull if 3 sigma is not smaller than 3 cm
217  if( muon.dX(i,1,arbitrationType) < 3. && muon.pullX(i,1,arbitrationType) > 3. ) {
218  station_weight[i-1] *= 1./TMath::Power(TMath::Max((double)muon.dX(i,1,arbitrationType),(double)1.),.25);
219  }
220  else {
221  station_weight[i-1] *= 1./TMath::Power(muon.pullX(i,1,arbitrationType),.25);
222  }
223  }
224  }
225  }
226  else { // has no match in X
227  if( muon.pullY(i,1,arbitrationType) > 1. ) { // has a match in Y. Pull larger that 1 to avoid increasing the weight (just penalize, don't anti-penalize)
228  // reduce weight
229  if(use_match_dist_penalty) {
230  // only use pull if 3 sigma is not smaller than 3 cm
231  if( muon.dY(i,1,arbitrationType) < 3. && muon.pullY(i,1,arbitrationType) > 3. ) {
232  station_weight[i-1] *= 1./TMath::Power(TMath::Max((double)muon.dY(i,1,arbitrationType),(double)1.),.25);
233  }
234  else {
235  station_weight[i-1] *= 1./TMath::Power(muon.pullY(i,1,arbitrationType),.25);
236  }
237  }
238  }
239  }
240  }
241  else { // We are in the CSCs
242  if(
243  TMath::Sqrt(TMath::Power(muon.pullX(i-4,2,arbitrationType),2.)+TMath::Power(muon.pullY(i-4,2,arbitrationType),2.)) > 1. ) {
244  // reduce weight
245  if(use_match_dist_penalty) {
246  // only use pull if 3 sigma is not smaller than 3 cm
247  if(TMath::Sqrt(TMath::Power(muon.dX(i-4,2,arbitrationType),2.)+TMath::Power(muon.dY(i-4,2,arbitrationType),2.)) < 3. && TMath::Sqrt(TMath::Power(muon.pullX(i-4,2,arbitrationType),2.)+TMath::Power(muon.pullY(i-4,2,arbitrationType),2.)) > 3. ) {
248  station_weight[i-1] *= 1./TMath::Power(
249  TMath::Max((double)TMath::Sqrt(TMath::Power(muon.dX(i-4,2,arbitrationType),2.)+TMath::Power(muon.dY(i-4,2,arbitrationType),2.)),(double)1.),.25);
250  }
251  else {
252  station_weight[i-1] *= 1./TMath::Power(
253  TMath::Sqrt(TMath::Power(muon.pullX(i-4,2,arbitrationType),2.)+TMath::Power(muon.pullY(i-4,2,arbitrationType),2.)),.25);
254  }
255  }
256  }
257  }
258  }
259 
260  // Thoughts:
261  // - should penalize if the segment has only x OR y info
262  // - should also use the segment direction, as it now works!
263 
264  }
265  else { // track did not pass a chamber in this station - just reset weight
266  station_weight[i-1] = 0.;
267  }
268 
269  //increment final weight for muon:
270  full_weight += station_weight[i-1];
271  }
272 
273  // if we don't expect any matches, we set the compatibility to
274  // 0.5 as the track is as compatible with a muon as it is with
275  // background - we should maybe rather set it to -0.5!
276  if( nr_of_stations_crossed == 0 ) {
277  // full_weight = attenuate_weight_regain*0.5;
278  full_weight = 0.5;
279  }
280 
281  // ********************************************************;
282  // *** calculate weights for each station *****************;
283  // ************** end *************************************;
284 
285  return full_weight;
286 
287 }
288 
291  double minCompatibility,
292  reco::Muon::ArbitrationType arbitrationType ) {
293  if (!muon.isMatchesValid()) return false;
294  bool goodMuon = false;
295 
296  switch( type ) {
297  case TM2DCompatibility:
298  // Simplistic first cut in the 2D segment- vs calo-compatibility plane. Will have to be refined!
299  if( ( (0.8*caloCompatibility( muon ))+(1.2*segmentCompatibility( muon, arbitrationType )) ) > minCompatibility ) goodMuon = true;
300  else goodMuon = false;
301  return goodMuon;
302  break;
303  default :
304  // LogTrace("MuonIdentification")<<" // Invalid Algorithm Type called!";
305  goodMuon = false;
306  return goodMuon;
307  break;
308  }
309 }
310 
313  int minNumberOfMatches,
314  double maxAbsDx,
315  double maxAbsPullX,
316  double maxAbsDy,
317  double maxAbsPullY,
318  double maxChamberDist,
319  double maxChamberDistPull,
320  reco::Muon::ArbitrationType arbitrationType,
321  bool syncMinNMatchesNRequiredStationsInBarrelOnly,
322  bool applyAlsoAngularCuts)
323 {
324  if (!muon.isMatchesValid()) return false;
325  bool goodMuon = false;
326 
327  if (type == TMLastStation) {
328  // To satisfy my own paranoia, if the user specifies that the
329  // minimum number of matches is zero, then return true.
330  if(minNumberOfMatches == 0) return true;
331 
332  unsigned int theStationMask = muon.stationMask(arbitrationType);
333  unsigned int theRequiredStationMask = RequiredStationMask(muon, maxChamberDist, maxChamberDistPull, arbitrationType);
334 
335  // Require that there be at least a minimum number of segments
336  int numSegs = 0;
337  int numRequiredStations = 0;
338  for(int it = 0; it < 8; ++it) {
339  if(theStationMask & 1<<it) ++numSegs;
340  if(theRequiredStationMask & 1<<it) ++numRequiredStations;
341  }
342 
343  // Make sure the minimum number of matches is not greater than
344  // the number of required stations but still greater than zero
345  if (syncMinNMatchesNRequiredStationsInBarrelOnly) {
346  // Note that we only do this in the barrel region!
347  if (fabs(muon.eta()) < 1.2) {
348  if(minNumberOfMatches > numRequiredStations)
349  minNumberOfMatches = numRequiredStations;
350  if(minNumberOfMatches < 1) //SK: this only happens for negative values
351  minNumberOfMatches = 1;
352  }
353  } else {
354  if(minNumberOfMatches > numRequiredStations)
355  minNumberOfMatches = numRequiredStations;
356  if(minNumberOfMatches < 1) //SK: this only happens for negative values
357  minNumberOfMatches = 1;
358  }
359 
360  if(numSegs >= minNumberOfMatches) goodMuon = 1;
361 
362  // Require that last required station have segment
363  // If there are zero required stations keep track
364  // of the last station with a segment so that we may
365  // apply the quality cuts below to it instead
366  int lastSegBit = 0;
367  if(theRequiredStationMask) {
368  for(int stationIdx = 7; stationIdx >= 0; --stationIdx)
369  if(theRequiredStationMask & 1<<stationIdx){
370  if(theStationMask & 1<<stationIdx) {
371  lastSegBit = stationIdx;
372  goodMuon &= 1;
373  break;
374  } else {
375  goodMuon = false;
376  break;
377  }
378  }
379  } else {
380  for(int stationIdx = 7; stationIdx >= 0; --stationIdx)
381  if(theStationMask & 1<<stationIdx) {
382  lastSegBit = stationIdx;
383  break;
384  }
385  }
386 
387  if(!goodMuon) return false;
388 
389  // Impose pull cuts on last segment
390  int station = 0, detector = 0;
391  station = lastSegBit < 4 ? lastSegBit+1 : lastSegBit-3;
392  detector = lastSegBit < 4 ? 1 : 2;
393 
394  // Check x information
395  if(fabs(muon.pullX(station,detector,arbitrationType,1)) > maxAbsPullX &&
396  fabs(muon.dX(station,detector,arbitrationType)) > maxAbsDx)
397  return false;
398 
399  if(applyAlsoAngularCuts && fabs(muon.pullDxDz(station,detector,arbitrationType,1)) > maxAbsPullX)
400  return false;
401 
402  // Is this a tight algorithm, i.e. do we bother to check y information?
403  if (maxAbsDy < 999999) { // really if maxAbsDy < 1E9 as currently defined
404 
405  // Check y information
406  if (detector == 2) { // CSC
407  if(fabs(muon.pullY(station,2,arbitrationType,1)) > maxAbsPullY &&
408  fabs(muon.dY(station,2,arbitrationType)) > maxAbsDy)
409  return false;
410 
411  if(applyAlsoAngularCuts && fabs(muon.pullDyDz(station,2,arbitrationType,1)) > maxAbsPullY)
412  return false;
413  } else {
414  //
415  // In DT, if this is a "Tight" algorithm and the last segment is
416  // missing y information (always the case in station 4!!!), impose
417  // respective cuts on the next station in the stationMask that has
418  // a segment with y information. If there are no segments with y
419  // information then there is nothing to penalize. Should we
420  // penalize in Tight for having zero segments with y information?
421  // That is the fundamental question. Of course I am being uber
422  // paranoid; if this is a good muon then there will probably be at
423  // least one segment with y information but not always. Suppose
424  // somehow a muon only creates segments in station 4, then we
425  // definitely do not want to require that there be at least one
426  // segment with y information because we will lose it completely.
427  //
428 
429  for (int stationIdx = station; stationIdx > 0; --stationIdx) {
430  if(! (theStationMask & 1<<(stationIdx-1))) // don't bother if the station is not in the stationMask
431  continue;
432 
433  if(muon.dY(stationIdx,1,arbitrationType) > 999998) // no y-information
434  continue;
435 
436  if(fabs(muon.pullY(stationIdx,1,arbitrationType,1)) > maxAbsPullY &&
437  fabs(muon.dY(stationIdx,1,arbitrationType)) > maxAbsDy) {
438  return false;
439  }
440 
441  if(applyAlsoAngularCuts && fabs(muon.pullDyDz(stationIdx,1,arbitrationType,1)) > maxAbsPullY)
442  return false;
443 
444  // If we get this far then great this is a good muon
445  return true;
446  }
447  }
448  }
449 
450  return goodMuon;
451  } // TMLastStation
452 
453  // TMOneStation requires only that there be one "good" segment, regardless
454  // of the required stations. We do not penalize if there are absolutely zero
455  // segments with y information in the Tight algorithm. Maybe I'm being
456  // paranoid but so be it. If it's really a good muon then we will probably
457  // find at least one segment with both x and y information but you never
458  // know, and I don't want to deal with a potential inefficiency in the DT
459  // like we did with the original TMLastStation. Incidentally, not penalizing
460  // for total lack of y information in the Tight algorithm is what is done in
461  // the new TMLastStation
462  //
463  if (type == TMOneStation) {
464  unsigned int theStationMask = muon.stationMask(arbitrationType);
465 
466  // Of course there must be at least one segment
467  if (! theStationMask) return false;
468 
469  int station = 0, detector = 0;
470  // Keep track of whether or not there is a DT segment with y information.
471  // In the end, if it turns out there are absolutely zero DT segments with
472  // y information, require only that there was a segment with good x info.
473  // This of course only applies to the Tight algorithms.
474  bool existsGoodDTSegX = false;
475  bool existsDTSegY = false;
476 
477  // Impose cuts on the segments in the station mask until we find a good one
478  // Might as well start with the lowest bit to speed things up.
479  for(int stationIdx = 0; stationIdx <= 7; ++stationIdx)
480  if(theStationMask & 1<<stationIdx) {
481  station = stationIdx < 4 ? stationIdx+1 : stationIdx-3;
482  detector = stationIdx < 4 ? 1 : 2;
483 
484  if((fabs(muon.pullX(station,detector,arbitrationType,1)) > maxAbsPullX &&
485  fabs(muon.dX(station,detector,arbitrationType)) > maxAbsDx) ||
486  (applyAlsoAngularCuts && fabs(muon.pullDxDz(station,detector,arbitrationType,1)) > maxAbsPullX))
487  continue;
488  else if (detector == 1)
489  existsGoodDTSegX = true;
490 
491  // Is this a tight algorithm? If yes, use y information
492  if (maxAbsDy < 999999) {
493  if (detector == 2) { // CSC
494  if((fabs(muon.pullY(station,2,arbitrationType,1)) > maxAbsPullY &&
495  fabs(muon.dY(station,2,arbitrationType)) > maxAbsDy) ||
496  (applyAlsoAngularCuts && fabs(muon.pullDyDz(station,2,arbitrationType,1)) > maxAbsPullY))
497  continue;
498  } else {
499 
500  if(muon.dY(station,1,arbitrationType) > 999998) // no y-information
501  continue;
502  else
503  existsDTSegY = true;
504 
505  if((fabs(muon.pullY(station,1,arbitrationType,1)) > maxAbsPullY &&
506  fabs(muon.dY(station,1,arbitrationType)) > maxAbsDy) ||
507  (applyAlsoAngularCuts && fabs(muon.pullDyDz(station,1,arbitrationType,1)) > maxAbsPullY)) {
508  continue;
509  }
510  }
511  }
512 
513  // If we get this far then great this is a good muon
514  return true;
515  }
516 
517  // If we get this far then for sure there are no "good" CSC segments. For
518  // DT, check if there were any segments with y information. If there
519  // were none, but there was a segment with good x, then we're happy. If
520  // there WERE segments with y information, then they must have been shit
521  // since we are here so fail it. Of course, if this is a Loose algorithm
522  // then fail immediately since if we had good x we would already have
523  // returned true
524  if (maxAbsDy < 999999) {
525  if (existsDTSegY)
526  return false;
527  else if (existsGoodDTSegX)
528  return true;
529  } else
530  return false;
531  } // TMOneStation
532 
533  if ( type == RPCMu )
534  {
535  if ( minNumberOfMatches == 0 ) return true;
536 
537  int nMatch = 0;
538  for ( std::vector<reco::MuonChamberMatch>::const_iterator chamberMatch = muon.matches().begin();
539  chamberMatch != muon.matches().end(); ++chamberMatch )
540  {
541  if ( chamberMatch->detector() != 3 ) continue;
542 
543  const double trkX = chamberMatch->x;
544  const double errX = chamberMatch->xErr;
545 
546  for ( std::vector<reco::MuonRPCHitMatch>::const_iterator rpcMatch = chamberMatch->rpcMatches.begin();
547  rpcMatch != chamberMatch->rpcMatches.end(); ++rpcMatch )
548  {
549  const double rpcX = rpcMatch->x;
550 
551  const double dX = fabs(rpcX-trkX);
552  if ( dX < maxAbsDx or dX/errX < maxAbsPullX )
553  {
554  ++nMatch;
555  break;
556  }
557  }
558  }
559 
560  if ( nMatch >= minNumberOfMatches ) return true;
561  else return false;
562  } // RPCMu
563 
564  return goodMuon;
565 }
566 
568  reco::Muon::ArbitrationType arbitrationType)
569 {
570  switch (type)
571  {
572  case muon::All:
573  return true;
574  break;
576  return muon.isGlobalMuon();
577  break;
579  return muon.isTrackerMuon();
580  break;
582  return muon.isStandAloneMuon();
583  break;
585  return muon.isTrackerMuon() && muon.numberOfMatches(arbitrationType)>0;
586  break;
587  case muon::AllArbitrated:
588  return ! muon.isTrackerMuon() || muon.numberOfMatches(arbitrationType)>0;
589  break;
591  return muon.isGlobalMuon() && muon.globalTrack()->normalizedChi2()<10. && muon.globalTrack()->hitPattern().numberOfValidMuonHits() >0;
592  break;
593  // For "Loose" algorithms we choose maximum y quantity cuts of 1E9 instead of
594  // 9999 as before. We do this because the muon methods return 999999 (note
595  // there are six 9's) when the requested information is not available. For
596  // example, if a muon fails to traverse the z measuring superlayer in a station
597  // in the DT, then all methods involving segmentY in this station return
598  // 999999 to demonstrate that the information is missing. In order to not
599  // penalize muons for missing y information in Loose algorithms where we do
600  // not care at all about y information, we raise these limits. In the
601  // TMLastStation and TMOneStation algorithms we actually use this huge number
602  // to determine whether to consider y information at all.
604  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,1E9,1E9,-3,-3,arbitrationType,true,false);
605  break;
607  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,3,3,-3,-3,arbitrationType,true,false);
608  break;
610  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,1E9,1E9,1E9,1E9,arbitrationType,false,false);
611  break;
613  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,3,3,1E9,1E9,arbitrationType,false,false);
614  break;
616  if (muon.pt() < 8. && fabs(muon.eta()) < 1.2)
617  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,1E9,1E9,1E9,1E9,arbitrationType,false,false);
618  else
619  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,1E9,1E9,-3,-3,arbitrationType,false,false);
620  break;
622  if (muon.pt() < 8. && fabs(muon.eta()) < 1.2)
623  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,3,3,1E9,1E9,arbitrationType,false,false);
624  else
625  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,3,3,-3,-3,arbitrationType,false,false);
626  break;
627  //compatibility loose
629  return muon.isTrackerMuon() && isGoodMuon(muon,TM2DCompatibility,0.7,arbitrationType);
630  break;
631  //compatibility tight
633  return muon.isTrackerMuon() && isGoodMuon(muon,TM2DCompatibility,1.0,arbitrationType);
634  break;
636  return muon.isGlobalMuon() && muon.isQualityValid() && fabs(muon.combinedQuality().trkRelChi2 - muon.innerTrack()->normalizedChi2()) < 2.0;
637  break;
639  return muon.isGlobalMuon() && muon.isQualityValid() && fabs(muon.combinedQuality().staRelChi2 - muon.outerTrack()->normalizedChi2()) < 2.0;
640  break;
641  case muon::GMTkKinkTight:
642  return muon.isGlobalMuon() && muon.isQualityValid() && muon.combinedQuality().trkKink < 100.0;
643  break;
645  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,1E9,1E9,-3,-3,arbitrationType,false,true);
646  break;
648  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,3,3,-3,-3,arbitrationType,false,true);
649  break;
651  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,1E9,1E9,1E9,1E9,arbitrationType,false,true);
652  break;
654  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,3,3,1E9,1E9,arbitrationType,false,true);
655  break;
657  if (muon.pt() < 8. && fabs(muon.eta()) < 1.2)
658  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,1E9,1E9,1E9,1E9,arbitrationType,false,false);
659  else
660  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,1E9,1E9,-3,-3,arbitrationType,true,false);
661  break;
663  if (muon.pt() < 8. && fabs(muon.eta()) < 1.2)
664  return muon.isTrackerMuon() && isGoodMuon(muon,TMOneStation,1,3,3,3,3,1E9,1E9,arbitrationType,false,false);
665  else
666  return muon.isTrackerMuon() && isGoodMuon(muon,TMLastStation,2,3,3,3,3,-3,-3,arbitrationType,true,false);
667  break;
668  case muon::RPCMuLoose:
669  return muon.isRPCMuon() && isGoodMuon(muon, RPCMu, 2, 20, 4, 1e9, 1e9, 1e9, 1e9, arbitrationType, false, false);
670  break;
671  default:
672  return false;
673  }
674 }
675 
676 bool muon::overlap( const reco::Muon& muon1, const reco::Muon& muon2,
677  double pullX, double pullY, bool checkAdjacentChambers)
678 {
679  unsigned int nMatches1 = muon1.numberOfMatches(reco::Muon::SegmentAndTrackArbitration);
680  unsigned int nMatches2 = muon2.numberOfMatches(reco::Muon::SegmentAndTrackArbitration);
681  unsigned int betterMuon = ( muon1.pt() > muon2.pt() ? 1 : 2 );
682  for ( std::vector<reco::MuonChamberMatch>::const_iterator chamber1 = muon1.matches().begin();
683  chamber1 != muon1.matches().end(); ++chamber1 )
684  for ( std::vector<reco::MuonChamberMatch>::const_iterator chamber2 = muon2.matches().begin();
685  chamber2 != muon2.matches().end(); ++chamber2 )
686  {
687 
688  // if ( (chamber1->segmentMatches.empty() || chamber2->segmentMatches.empty()) ) continue;
689 
690  // handle case where both muons have information about the same chamber
691  // here we know how close they are
692  if ( chamber1->id == chamber2->id ){
693  // found the same chamber
694  if ( fabs(chamber1->x-chamber2->x) <
695  pullX * sqrt(chamber1->xErr*chamber1->xErr+chamber2->xErr*chamber2->xErr) )
696  {
697  if ( betterMuon == 1 )
698  nMatches2--;
699  else
700  nMatches1--;
701  if ( nMatches1==0 || nMatches2==0 ) return true;
702  continue;
703  }
704  if ( fabs(chamber1->y-chamber2->y) <
705  pullY * sqrt(chamber1->yErr*chamber1->yErr+chamber2->yErr*chamber2->yErr) )
706  {
707  if ( betterMuon == 1 )
708  nMatches2--;
709  else
710  nMatches1--;
711  if ( nMatches1==0 || nMatches2==0 ) return true;
712  }
713  } else {
714  if ( ! checkAdjacentChambers ) continue;
715  // check if tracks are pointing into overlaping region of the CSC detector
716  if ( chamber1->id.subdetId() != MuonSubdetId::CSC ||
717  chamber2->id.subdetId() != MuonSubdetId::CSC ) continue;
718  CSCDetId id1(chamber1->id);
719  CSCDetId id2(chamber2->id);
720  if ( id1.endcap() != id2.endcap() ) continue;
721  if ( id1.station() != id2.station() ) continue;
722  if ( id1.ring() != id2.ring() ) continue;
723  if ( abs(id1.chamber() - id2.chamber())>1 ) continue;
724  // FIXME: we don't handle 18->1; 36->1 transitions since
725  // I don't know how to check for sure how many chambers
726  // are there. Probably need to hard code some checks.
727 
728  // Now we have to make sure that both tracks are close to an edge
729  // FIXME: ignored Y coordinate for now
730  if ( fabs(chamber1->edgeX) > chamber1->xErr*pullX ) continue;
731  if ( fabs(chamber2->edgeX) > chamber2->xErr*pullX ) continue;
732  if ( chamber1->x * chamber2->x < 0 ) { // check if the same edge
733  if ( betterMuon == 1 )
734  nMatches2--;
735  else
736  nMatches1--;
737  if ( nMatches1==0 || nMatches2==0 ) return true;
738  }
739  }
740  }
741  return false;
742 }
743 
744 
746 
747  if(!muon.isPFMuon() || !muon.isGlobalMuon() ) return false;
748 
749  bool muID = isGoodMuon(muon,GlobalMuonPromptTight) && (muon.numberOfMatchedStations() > 1);
750 
751 
752  bool hits = muon.innerTrack()->hitPattern().trackerLayersWithMeasurement() > 5 &&
753  muon.innerTrack()->hitPattern().numberOfValidPixelHits() > 0;
754 
755 
756  bool ip = fabs(muon.muonBestTrack()->dxy(vtx.position())) < 0.2 && fabs(muon.muonBestTrack()->dz(vtx.position())) < 0.5;
757 
758  return muID && hits && ip;
759 }
760 
761 
763  return muon.isPFMuon() && ( muon.isGlobalMuon() || muon.isTrackerMuon());
764 }
765 
766 
767 bool muon::isSoftMuon(const reco::Muon& muon, const reco::Vertex& vtx){
768 
769  bool muID = muon::isGoodMuon(muon, TMOneStationTight);
770 
771  if(!muID) return false;
772 
773  bool layers = muon.innerTrack()->hitPattern().trackerLayersWithMeasurement() > 5 &&
774  muon.innerTrack()->hitPattern().pixelLayersWithMeasurement() > 1;
775 
776  bool chi2 = muon.innerTrack()->normalizedChi2() < 1.8;
777 
778  bool ip = fabs(muon.innerTrack()->dxy(vtx.position())) < 3. && fabs(muon.innerTrack()->dz(vtx.position())) < 30.;
779 
780  return muID && layers && ip && chi2 ;
781 }
782 
783 
784 
786  bool muID = muon.isGlobalMuon() && muon.globalTrack()->hitPattern().numberOfValidMuonHits() >0 && (muon.numberOfMatchedStations() > 1);
787  if(!muID) return false;
788 
789  bool hits = muon.innerTrack()->hitPattern().trackerLayersWithMeasurement() > 5 &&
790  muon.innerTrack()->hitPattern().numberOfValidPixelHits() > 0;
791 
792  bool momQuality = muon.muonBestTrack()->ptError()/muon.muonBestTrack()->pt() < 0.3;
793 
794  bool ip = fabs(muon.muonBestTrack()->dxy(vtx.position())) < 0.2 && fabs(muon.bestTrack()->dz(vtx.position())) < 0.5;
795 
796  return muID && hits && momQuality && ip;
797 
798 }
799 
800 int muon::sharedSegments( const reco::Muon& mu, const reco::Muon& mu2, unsigned int segmentArbitrationMask ) {
801  int ret = 0;
802 
803  // Will do with a stupid double loop, since creating and filling a map is probably _more_ inefficient for a single lookup.
804  for(std::vector<reco::MuonChamberMatch>::const_iterator chamberMatch = mu.matches().begin();
805  chamberMatch != mu.matches().end(); ++chamberMatch) {
806  if (chamberMatch->segmentMatches.empty()) continue;
807  for(std::vector<reco::MuonChamberMatch>::const_iterator chamberMatch2 = mu2.matches().begin();
808  chamberMatch2 != mu2.matches().end(); ++chamberMatch2) {
809  if (chamberMatch2->segmentMatches.empty()) continue;
810  if (chamberMatch2->id() != chamberMatch->id()) continue;
811  for(std::vector<reco::MuonSegmentMatch>::const_iterator segmentMatch = chamberMatch->segmentMatches.begin();
812  segmentMatch != chamberMatch->segmentMatches.end(); ++segmentMatch) {
813  if (!segmentMatch->isMask(segmentArbitrationMask)) continue;
814  for(std::vector<reco::MuonSegmentMatch>::const_iterator segmentMatch2 = chamberMatch2->segmentMatches.begin();
815  segmentMatch2 != chamberMatch2->segmentMatches.end(); ++segmentMatch2) {
816  if (!segmentMatch2->isMask(segmentArbitrationMask)) continue;
817  if ((segmentMatch->cscSegmentRef.isNonnull() && segmentMatch->cscSegmentRef == segmentMatch2->cscSegmentRef) ||
818  (segmentMatch-> dtSegmentRef.isNonnull() && segmentMatch-> dtSegmentRef == segmentMatch2-> dtSegmentRef) ) {
819  ++ret;
820  } // is the same
821  } // segment of mu2 in chamber
822  } // segment of mu1 in chamber
823  } // chamber of mu2
824  } // chamber of mu1
825 
826  return ret;
827 }
int chamber() const
Definition: CSCDetId.h:70
type
Definition: HCALResponse.h:21
int i
Definition: DBlmapReader.cc:9
The Signals That Services Can Subscribe To This is based on ActivityRegistry and is current per Services can connect to the signals distributed by the ActivityRegistry in order to monitor the activity of the application Each possible callback has some defined which we here list in angle e< void, edm::EventIDconst &, edm::Timestampconst & > We also list in braces which AR_WATCH_USING_METHOD_ is used for those or
Definition: Activities.doc:12
AlgorithmType
Definition: MuonSelectors.h:64
float trackDistErr(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration) const
Definition: Muon.cc:784
virtual TrackRef innerTrack() const
Definition: Muon.h:49
bool isTrackerMuon() const
Definition: Muon.h:220
#define abs(x)
Definition: mlp_lapack.h:159
bool isGlobalMuon() const
Definition: Muon.h:219
float trkKink
value of the kink algorithm applied to the inner track stub
Definition: MuonQuality.h:11
bool isMatchesValid() const
Definition: Muon.h:139
float dY(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration) const
Definition: Muon.cc:421
bool isStandAloneMuon() const
Definition: Muon.h:221
float trkRelChi2
chi2 value for the inner track stub with respect to the global track
Definition: MuonQuality.h:15
const Point & position() const
position
Definition: Vertex.h:93
float caloCompatibility(const reco::Muon &muon)
bool isLooseMuon(const reco::Muon &)
SelectionType
Selector type.
Definition: MuonSelectors.h:18
bool overlap(const reco::Muon &muon1, const reco::Muon &muon2, double pullX=1.0, double pullY=1.0, bool checkAdjacentChambers=false)
bool isSoftMuon(const reco::Muon &, const reco::Vertex &)
int endcap() const
Definition: CSCDetId.h:95
bool isRPCMuon() const
Definition: Muon.h:224
virtual const Track * bestTrack() const
best track pointer
Definition: Muon.h:62
static const int CSC
Definition: MuonSubdetId.h:15
virtual TrackRef muonBestTrack() const
Definition: Muon.h:64
ArbitrationType
define arbitration schemes
Definition: Muon.h:179
T sqrt(T t)
Definition: SSEVec.h:48
MuonQuality combinedQuality() const
get energy deposition information
Definition: Muon.h:122
float pullDxDz(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration, bool includeSegmentError=true) const
Definition: Muon.cc:468
float segmentCompatibility(const reco::Muon &muon, reco::Muon::ArbitrationType arbitrationType=reco::Muon::SegmentAndTrackArbitration)
const int mu
Definition: Constants.h:23
bool isQualityValid() const
Definition: Muon.h:120
unsigned int stationMask(ArbitrationType type=SegmentAndTrackArbitration) const
Definition: Muon.cc:123
virtual TrackRef outerTrack() const
reference to Track reconstructed in the muon detector only
Definition: Muon.h:52
float pullY(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration, bool includeSegmentError=true) const
Definition: Muon.cc:457
float segmentX(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration) const
Definition: Muon.cc:489
bool isGoodMuon(const reco::Muon &muon, SelectionType type, reco::Muon::ArbitrationType arbitrationType=reco::Muon::SegmentAndTrackArbitration)
main GoodMuon wrapper call
int numberOfMatches(ArbitrationType type=SegmentAndTrackArbitration) const
get number of chambers with matched segments
Definition: Muon.cc:60
virtual float eta() const GCC11_FINAL
momentum pseudorapidity
double dz() const
dz parameter (= dsz/cos(lambda)). This is the track z0 w.r.t (0,0,0) only if the refPoint is close to...
Definition: TrackBase.h:127
int ring() const
Definition: CSCDetId.h:77
float staRelChi2
chi2 value for the outer track stub with respect to the global track
Definition: MuonQuality.h:17
int sharedSegments(const reco::Muon &muon1, const reco::Muon &muon2, unsigned int segmentArbitrationMask=reco::MuonSegmentMatch::BestInChamberByDR)
bool isHighPtMuon(const reco::Muon &, const reco::Vertex &)
SelectionType selectionTypeFromString(const std::string &label)
Definition: MuonSelectors.cc:9
a lightweight &quot;map&quot; for selection type string label and enum value
Definition: MuonSelectors.h:54
unsigned int RequiredStationMask(const reco::Muon &muon, double maxChamberDist, double maxChamberDistPull, reco::Muon::ArbitrationType arbitrationType)
std::vector< MuonChamberMatch > & matches()
get muon matching information
Definition: Muon.h:141
float trackDist(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration) const
Definition: Muon.cc:767
float dX(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration) const
Definition: Muon.cc:413
int numberOfMatchedStations(ArbitrationType type=SegmentAndTrackArbitration) const
Definition: Muon.cc:110
bool isPFMuon() const
Definition: Muon.h:223
int station() const
Definition: CSCDetId.h:88
bool isTightMuon(const reco::Muon &, const reco::Vertex &)
virtual float pt() const GCC11_FINAL
transverse momentum
float caloCompatibility() const
Definition: Muon.h:151
float pullDyDz(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration, bool includeSegmentError=true) const
Definition: Muon.cc:478
virtual TrackRef globalTrack() const
reference to Track reconstructed in both tracked and muon detector
Definition: Muon.h:55
float pullX(int station, int muonSubdetId, ArbitrationType type=SegmentAndTrackArbitration, bool includeSegmentError=true) const
Definition: Muon.cc:447