rivet is hosted by Hepforge, IPPP Durham
Rivet 4.1.0
ParticleIdUtils.hh
1// -*- C++ -*-
2//
3// This file is part of MCUtils -- https://gitlab.com/hepcedar/mcutils/
4// Copyright (C) 2013-2025 Andy Buckley <andy.buckley@cern.ch>
5//
6// Embedding of MCUtils code in other projects is permitted provided this
7// notice is retained and the MCUtils namespace and include path are changed.
8//
9#ifndef RIVET_PARTICLEIDUTILS_HH
10#define RIVET_PARTICLEIDUTILS_HH
11
14
15#include "Rivet/Tools/ParticleName.hh"
16#include "Rivet/Math/MathUtils.hh"
17#include <cassert>
18
19namespace Rivet {
20 namespace PID {
21
22
25
28 inline int abspid(int pid) {
29 return abs(pid);
30 }
31
32
33 // /// Compile-time int^int power-raising function
34 // template <size_t N>
35 // inline int _intpow(int x) { return x * _intpow<N-1>(x); }
36 // template <>
37 // inline int _intpow<0>(int x) { return 1; }
38
40 // inline size_t _pow10(unsigned int power) {
41 // return (size_t) std::pow(10.0, power);
42 // }
44 inline size_t _pow10(unsigned int power) {
45 //assert(power >= 0 && "_pow10 only defined for positive powers");
46 assert(power < 16 && "_pow10 only defined for powers < 16");
47 static const size_t POWS10[] = {1, 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000, 1000000000, 10000000000, 100000000000, 1000000000000, 10000000000000, 100000000000000, 1000000000000000, 10000000000000000};
48 return POWS10[power];
49 }
50 // /// Raise 10 to an integer power (constexpr)
51 // inline constexpr size_t _pow10(unsigned int power) {
52 // assert(power >= 0 && "_pow10 only defined for positive powers");
53 // return (power > 0) ? 10*_pow10(power-1) : 1;
54 // }
55
58 enum Location { nj=1, nq3, nq2, nq1, nl, nr, n, n8, n9, n10 };
59
65 inline unsigned short _digit(Location loc, int pid) {
66 const int div = _pow10(loc-1);
67 return (abs(pid)/div) % 10;
68 }
69
71 inline int _extraBits(int pid) {
72 return abs(pid)/10000000;
73 }
74
77 inline int _fundamentalID(int pid) {
78 if (_extraBits(pid) > 0) return 0;
79 if (_digit(nq2,pid) == 0 && _digit(nq1,pid) == 0) {
80 return abs(pid) % 10000;
81 } else if (abs(pid) <= 100) {
82 return abs(pid);
83 } else {
84 return 0;
85 }
86 }
87
89
90
91 // Forward declaration
92 inline bool isBSM(int pid);
93
94
97
106 inline bool isNucleus(int pid) {
107 // A proton can also be a Hydrogen nucleus
108 if (abs(pid) == 2212) return true;
109 // New standard: +/- 10LZZZAAAI
110 if (_digit(n10,pid) == 1 && _digit(n9,pid) == 0) {
111 // charge should always be less than or equal to baryon number
112 // the following line is A >= Z
113 if ((abs(pid)/10) % 1000 >= (abs(pid)/10000) % 1000) return true;
114 }
115 return false;
116 }
117
120 inline int nuclZ(int pid) {
121 // A proton can also be a Hydrogen nucleus
122 if (abs(pid) == 2212) {
123 return 1;
124 }
125 if (isNucleus(pid)) return (abs(pid)/10000) % 1000;
126 return 0;
127 }
129 inline int Z(int pid) {
130 return nuclZ(pid);
131 }
132
135 inline int nuclA(int pid) {
136 // a proton can also be a Hydrogen nucleus
137 if (abs(pid) == 2212) {
138 return 1;
139 }
140 if (isNucleus(pid)) return (abs(pid)/10) % 1000;
141 return 0;
142 }
144 inline int A(int pid) {
145 return nuclA(pid);
146 }
147
150 inline int nuclNlambda(int pid) {
151 // a proton can also be a Hydrogen nucleus
152 if (abs(pid) == 2212) {
153 return 0;
154 }
155 if (isNucleus(pid)) return _digit(n8,pid);
156 return 0;
157 }
159 inline int lambda(int pid) {
160 return nuclNlambda(pid);
161 }
162
164
165
168
170 inline bool isQuark(int pid) {
171 return in_closed_range(abs(pid), 1, 8);
172 }
173
175 inline bool isGluon(int pid) {
176 return pid == GLUON;
177 }
178
180 inline bool isParton(int pid) {
181 return isGluon(pid) || isQuark(pid);
182 }
183
184
186 inline bool isPhoton(int pid) {
187 return pid == PHOTON;
188 }
189
191 inline bool isElectron(int pid) {
192 return abs(pid) == ELECTRON;
193 }
194
196 inline bool isMuon(int pid) {
197 return abs(pid) == MUON;
198 }
199
201 inline bool isTau(int pid) {
202 return abs(pid) == TAU;
203 }
204
206 inline bool isChargedLepton(int pid) {
207 const long apid = abs(pid);
208 return apid == 11 || apid == 13 || apid == 15 || apid == 17;
209 }
212 inline bool isChLepton(int pid) {
213 return isChargedLepton(pid);
214 }
215
217 inline bool isNeutrino(int pid) {
218 const long apid = abs(pid);
219 return apid == 12 || apid == 14 || apid == 16 || apid == 18;
220 }
221
222
224 inline bool isWplus(int pid) {
225 return pid == WPLUSBOSON;
226 }
227
229 inline bool isWminus(int pid) {
230 return pid == WMINUSBOSON;
231 }
232
234 inline bool isW(int pid) {
235 return abs(pid) == WPLUSBOSON;
236 }
237
239 inline bool isZ(int pid) {
240 return pid == Z0BOSON;
241 }
242
244 inline bool isHiggs(int pid) {
245 return pid == HIGGSBOSON || pid == 26; //< @todo Check on 26 still needed? (used in HERWIG SUSY, for example)
246 }
247
249
250
252 inline bool isGraviton(int pid) {
253 return pid == GRAVITON;
254 }
255
256
257 // /// Determine if the PID is that of a d/dbar
258 // inline bool isDown(int pid) { return abs(pid) == DQUARK; }
259
260 // /// Determine if the PID is that of a u/ubar
261 // inline bool isUp(int pid) { return abs(pid) == UQUARK; }
262
264 inline bool isStrange(int pid) {
265 return abs(pid) == SQUARK;
266 }
267
269 inline bool isCharm(int pid) {
270 return abs(pid) == CQUARK;
271 }
272
274 inline bool isBottom(int pid) {
275 return abs(pid) == BQUARK;
276 }
277
279 inline bool isTop(int pid) {
280 return abs(pid) == TQUARK;
281 }
282
284
285
288
290 inline bool isReggeon(int pid) {
291 return pid == 110 || pid == 990 || pid == 9990;
292 }
293
295 inline bool isMeson(int pid) {
296 if (_extraBits(pid) > 0) return false;
297 if (isBSM(pid)) return false;
298 const int aid = abs(pid);
299 if (aid == 130 || aid == 310 || aid == 210) return true; //< special cases for kaons
300 if (aid <= 100) return false;
301 if (_digit(nq1,pid) != 0) return false;
302 if (_digit(nq2,pid) == 0) return false;
303 if (_digit(nq3,pid) == 0) return false;
304 if (_digit(nq2,pid) < _digit(nq3,pid)) return false;
305 // EvtGen uses some odd numbers
307 if (aid == 150 || aid == 350 || aid == 510 || aid == 530) return true;
308 // Pomeron, Reggeon, etc.
309 if (isReggeon(pid)) return false; //true; //< WTF?
310 // Check for illegal antiparticles
311 if (_digit(nj,pid) > 0 && _digit(nq3,pid) > 0 && _digit(nq2,pid) > 0 && _digit(nq1,pid) == 0) {
312 return !(_digit(nq3,pid) == _digit(nq2,pid) && pid < 0);
313 }
314 return false;
315 }
316
318 inline bool isBaryon(int pid) {
319 if (_extraBits(pid) > 0) return false;
320 if (isBSM(pid)) return false;
321 if (abs(pid) <= 100) return false;
322 if (_fundamentalID(pid) <= 100 && _fundamentalID(pid) > 0) return false;
323 if (abs(pid) == 2110 || abs(pid) == 2210) return true;
324 if (_digit(nj,pid) == 0) return false;
325 if (_digit(nq1,pid) == 0 || _digit(nq2,pid) == 0 || _digit(nq3,pid) == 0) return false;
326 return true;
328 // if ((_digit(nq1,pid) >= _digit(nq2,pid) && _digit(nq2,pid) >= _digit(nq3,pid)) ||
329 // (_digit(nq1,pid) > _digit(nq3,pid) && _digit(nq3,pid) > _digit(nq2,pid)) || //< case 6b for lighter quarks in J=1
330 // (_digit(nq3,pid) > _digit(nq1,pid) && _digit(nq1,pid) > _digit(nq2,pid))) //< case 6e for extra states in excited multiplets
331 // return true;
332 // return false;
333 }
334
335 // Check to see if this is a valid diquark
336 inline bool isDiquark(int pid) {
337 if (_extraBits(pid) > 0) return false;
338 if (isBSM(pid)) return false;
339 if (abs(pid) <= 100) return false;
340 if (_fundamentalID(pid) <= 100 && _fundamentalID(pid) > 0) return false;
341 if (_digit(nq1,pid) == 0) return false;
342 if (_digit(nq2,pid) == 0) return false;
343 if (_digit(nq3,pid) != 0) return false;
344 if (_digit(nq1,pid) < _digit(nq2,pid)) return false;
345 if (_digit(nj,pid) > 0 && _digit(nq3,pid) == 0 && _digit(nq2,pid) > 0 && _digit(nq1,pid) > 0) return true; // diquark signature
346 // EvtGen uses the diquarks for quark pairs, so, for instance, 5501 is a valid "diquark" for EvtGen
347 // if (_digit(nj) == 1 && _digit(nq2) == _digit(nq1)) { // illegal
348 // return false;
349 // } else {
350 // return true;
351 // }
352 return false;
353 }
355 inline bool isDiQuark(int pid) {
356 return isDiquark(pid);
357 }
358
360 inline bool isPentaquark(int pid) {
361 // a pentaquark is of the form 9abcdej,
362 // where j is the spin and a, b, c, d, and e are quarks
363 if (_extraBits(pid) > 0) return false;
364 if (isBSM(pid)) return false;
365 if (_digit(n,pid) != 9) return false;
366 if (_digit(nr,pid) == 9 || _digit(nr,pid) == 0) return false;
367 if (_digit(nj,pid) == 9 || _digit(nl,pid) == 0) return false;
368 if (_digit(nq1,pid) == 0) return false;
369 if (_digit(nq2,pid) == 0) return false;
370 if (_digit(nq3,pid) == 0) return false;
371 if (_digit(nj,pid) == 0) return false;
372 // check ordering
373 if (_digit(nq2,pid) > _digit(nq1,pid)) return false;
374 if (_digit(nq1,pid) > _digit(nl,pid)) return false;
375 if (_digit(nl,pid) > _digit(nr,pid)) return false;
376 return true;
377 }
378
384 inline bool isQuarkonium(int pid) {
385 if (!isMeson(pid)) return false; //< all quarkonia are mesons
386 if (_digit(nq1, pid) != 0) return false;
387 const int fq = _digit(nq2, pid);
388 return (fq > 3 && _digit(nj, pid) > 0 && _digit(nq3, pid) == fq);
389 }
390
394 inline bool isHadron(int pid) {
395 if (_extraBits(pid) > 0) return false;
396 if (isBSM(pid) > 0) return false;
397 if (isMeson(pid)) return true;
398 if (isBaryon(pid)) return true;
399 if (isPentaquark(pid)) return true;
400 return false;
401 }
402
404
405
408
412 inline bool isLepton(int pid) {
413 if (_extraBits(pid) > 0) return false;
414 if (isBSM(pid) > 0) return false;
415 if (_fundamentalID(pid) >= 11 && _fundamentalID(pid) <= 18) return true;
416 return false;
417 }
418
420 inline bool isBSMBoson(int pid) {
421 return in_closed_range(abs(pid), 32, 37);
422 }
423
425 inline bool isSMFundamental(int pid) {
426 return isQuark(pid) || isLepton(pid) ||
427 isGluon(pid) || isPhoton(pid) || isW(pid) || isZ(pid) || isHiggs(pid) ||
429 }
430
434 inline bool isSUSY(int pid) {
435 // Fundamental SUSY particles have n = 1 or 2
436 if (_extraBits(pid) > 0) return false;
437 if (_digit(n,pid) != 1 && _digit(n,pid) != 2) return false;
438 if (_digit(nr,pid) != 0) return false;
439 // Check fundamental part for SM PID on which it is based
440 const int fundId = _fundamentalID(pid);
441 if (fundId == 0) return false;
442 if (_digit(n,pid) == 1) { // most superpartners, incl LH sfermions
443 return isSMFundamental(fundId);
444 } else if (_digit(n,pid) == 2) { // RH sfermions
445 return isQuark(fundId) || isChargedLepton(fundId);
446 }
447 return true;
448 }
449
451 inline bool isRHadron(int pid) {
452 // An R-hadron is of the form 10abcdj,
453 // where j is the spin and a, b, c, and d are quarks or gluons
454 if (_extraBits(pid) > 0) return false;
455 if (_digit(n,pid) != 1) return false;
456 if (_digit(nr,pid) != 0) return false;
457 // Make sure this isn't a SUSY particle
458 if (isSUSY(pid)) return false;
459 // All R-hadrons have at least 3 core digits
460 if (_digit(nq2,pid) == 0) return false;
461 if (_digit(nq3,pid) == 0) return false;
462 if (_digit(nj,pid) == 0) return false;
463 return true;
464 }
466 inline bool isRhadron(int pid) {
467 return isRHadron(pid);
468 }
469
471 inline bool isTechnicolor(int pid) {
472 if (_extraBits(pid) > 0) return false;
473 return _digit(n,pid) == 3;
474 }
475
477 inline bool isExcited(int pid) {
478 if (_extraBits(pid) > 0) return false;
479 return _digit(n,pid) == 4 && _digit(nr,pid) == 0;
480 }
481
483 inline bool isKK(int pid) {
484 if (_extraBits(pid) > 0) return false;
485 const int ndigit = _digit(n,pid);
486 return ndigit == 5 || ndigit == 6;
487 }
488
490 inline bool isLeptoQuark(int pid) {
491 // Many UFO models are extending the PDG standards... is this going to be official?
492 return abs(pid) == 42;
493 }
494
499 inline bool isDarkMatter(int pid) {
500 const int ndigit = _digit(n,pid);
501 const int nrdigit = _digit(nr,pid);
502 if ((ndigit == 0 && nrdigit == 0) || (ndigit == 5 && nrdigit == 9))
503 return in_closed_range(abs(_fundamentalID(pid)),50,60);
504 return false;
505 }
507 inline bool isDM(int pid) {
508 return isDarkMatter(pid);
509 }
510
512 inline bool isHiddenValley(int pid) {
513 return (_digit(n,pid) == 4 && _digit(nr,pid) == 9);
514 }
515
517 inline bool isExotic(int pid) {
518 // From the PDG definition, 40-80 reserved for exotic particles
519 // Some overlap with ranges from other functions (e.g. isDM)
520 // Also covers R0 (41)
521 return in_closed_range(abs(pid),40,80);
522 }
523
525 inline bool isFourthGen(int pid) {
526 return abs(pid) == BPRIME || abs(pid) == TPRIME || abs(pid) == LPRIME || abs(pid) == NUPRIME;
527 }
528
530 inline bool isMagMonopole(int pid) {
531 if (_digit(n,pid) != 4) return false;
532 if (_digit(nr,pid) != 1) return false;
533 if (_digit(nl,pid) != 1 && _digit(nl,pid) != 2) return false;
534 // Require at least 1 core digit
535 // NOT TRUE! Electrically neutral monopoles are possible
536 // if (_digit(nq3,pid) == 0) return false;
537 // Always have spin zero for now
538 if (_digit(nj,pid) != 0) return false;
539 return true;
540 }
542 inline bool isDyon(int pid) {
543 return isMagMonopole(pid);
544 }
545
548 inline bool isQBall(int pid) {
549 if (_extraBits(pid) != 1) return false;
550 if (_digit(n,pid) != 0) return false;
551 if (_digit(nr,pid) != 0) return false;
552 // Check the core number
553 if ((abs(pid)/10) % 10000 == 0) return false;
554 // These particles have spin zero for now
555 if (_digit(nj,pid) != 0) return false;
556 return true;
557 }
559 inline bool isQball(int pid) {
560 return isQBall(pid);
561 }
562
564 inline bool isExcitedLepton(int pid) {
565 if (!isExcited(pid)) return false;
566 return isLepton( _fundamentalID(pid) );
567 }
568
570 inline bool isBlackHole(int pid) {
571 if (_digit(n,pid) != 5 && _digit(n,pid) != 6) return false;
572 if (_digit(nl,pid) != 0) return false;
573 return _fundamentalID(pid)==40;
574 }
575
577 inline bool isAECO(int pid) {
578 if (_digit( n,pid) != 1) return false;
579 if (_digit(nr,pid) != 0) return false;
580 if (_digit(nl,pid) != 0) return false;
581 if (_digit(nj,pid) != 0) return false;
582 return true;
583 }
584
586 inline bool isBSM(int pid) {
587 return isSUSY(pid) || isRHadron(pid) || isTechnicolor(pid) ||
588 isExcited(pid) || isKK(pid) || isGraviton(pid) ||
591 isDyon(pid) || isQball(pid) || isAECO(pid);
592 }
593
595 inline bool _isValid(int pid) {
596 // Starting with 99 means anything goes (but nothing is known)
597 if (_digit(n, pid) == 9 && _digit(nr, pid) == 9) return true;
598 // Check that extra bits are only used for nuclei
599 if (_extraBits(pid) > 0) return (isNucleus(pid) || isQball(pid));
600 // Check that it fits into a standard non-nucleus convention
601 if (isBSM(pid)) return true;
602 if (isHadron(pid)) return true;
603 if (_digit(n,pid) == 9 && _digit(nr,pid) == 0) return false; // could only have been a tentative hadron, but !isHadron
604 if (isDiquark(pid)) return true;
605 if (isPentaquark(pid)) return true;
606 if (isReggeon(pid)) return true;
607 // // Quark digit orderings required by the standard
608 // if (_digit(nq1,pid) != 0 && _digit(nq1,pid) < _digit(nq2,pid)) return false;
609 // if (_digit(nq2,pid) != 0 && _digit(nq2,pid) < _digit(nq3,pid)) return false;
610 // Final check on fundamental ID
611 return (_fundamentalID(pid) > 0);
612 }
613 inline bool isValid(int pid) {
614 return _isValid(pid);
615 }
616
618
619
622
623 inline bool _hasQ(int pid, int q) {
624 if (abs(pid) == q) return true; //< trivial case!
625 if (!_isValid(pid)) return false;
626 // if (_extraBits(pid) > 0) return false;
627 // if (_fundamentalID(pid) > 0) return false;
628 if (isMagMonopole(pid)) return false;
629 if (isRHadron(pid)) {
630 int iz = 7;
631 for (int i = 6; i > 1; --i) {
632 if (_digit(Location(i), pid) == 0) {
633 iz = i;
634 } else if ( i == iz-1 ) {
635 // ignore squark or gluino
636 } else {
637 if (_digit(Location(i),pid) == q) return true;
638 }
639 }
640 return false;
641 }
642 if (_digit(nq3,pid) == q || _digit(nq2,pid) == q || _digit(nq1,pid) == q ) return true;
643 if (isPentaquark(pid)) {
644 if (_digit(nl,pid) == q || _digit(nr,pid) == q) return true;
645 }
646 return false;
647 }
648
650 inline bool hasDown(int pid) {
651 return (isHadron(pid) || isQuark(pid)) && _hasQ(pid, 1);
652 }
654 inline bool hasUp(int pid) {
655 return (isHadron(pid) || isQuark(pid)) && _hasQ(pid, 2);
656 }
658 inline bool hasStrange(int pid) {
659 return (isHadron(pid) || isQuark(pid)) && _hasQ(pid, 3);
660 }
662 inline bool hasCharm(int pid) {
663 return (isHadron(pid) || isQuark(pid)) && _hasQ(pid, 4);
664 }
666 inline bool hasBottom(int pid) {
667 return (isHadron(pid) || isQuark(pid)) && _hasQ(pid, 5);
668 }
670 inline bool hasTop(int pid) {
671 return (isHadron(pid) || isQuark(pid)) && _hasQ(pid, 6);
672 }
673
675
676
679
681 inline bool isHeavyFlavor(int pid) {
682 if (!isHadron(pid) && !isQuark(pid)) return false;
683 return hasCharm(pid) || hasBottom(pid) || hasTop(pid);
684 }
686 inline bool isHeavyFlavour(int pid) {
687 return isHeavyFlavor(pid);
688 }
689
690
691 // /// Determine if the particle is a light-flavour hadron or parton
692 // inline bool isLightFlavor(int pid) {
693 // return !isHeavyFlavor();
694 // }
695
696
698 inline bool isHeavyParton(int pid) {
699 return isParton(pid) && isHeavyFlavor(pid);
700 }
701
703 inline bool isLightParton(int pid) {
704 return isParton(pid) && !isHeavyFlavor(pid);
705 }
706
707
709 inline bool isHeavyMeson(int pid) {
710 return isMeson(pid) && isHeavyFlavor(pid);
711 }
712
714 inline bool isHeavyBaryon(int pid) {
715 return isBaryon(pid) && isHeavyFlavor(pid);
716 }
717
719 inline bool isHeavyHadron(int pid) {
720 return isHadron(pid) && isHeavyFlavor(pid);
721 }
722
724 inline bool isLightMeson(int pid) {
725 return isMeson(pid) && !isHeavyFlavor(pid);
726 }
727
729 inline bool isLightBaryon(int pid) {
730 return isBaryon(pid) && !isHeavyFlavor(pid);
731 }
732
734 inline bool isLightHadron(int pid) {
735 return isHadron(pid) && !isHeavyFlavor(pid);
736 }
737
738
740 inline bool isBottomMeson(int pid) {
741 return hasBottom(pid) && isMeson(pid);
742 }
743
745 inline bool isBottomBaryon(int pid) {
746 return hasBottom(pid) && isBaryon(pid);
747 }
748
750 inline bool isBottomHadron(int pid) {
751 return hasBottom(pid) && isHadron(pid);
752 }
753
754
759 inline bool isCharmMeson(int pid) {
760 return isMeson(pid) && hasCharm(pid) &&
761 !hasBottom(pid);
762 }
763
769 inline bool isCharmBaryon(int pid) {
770 return isBaryon(pid) && hasCharm(pid) &&
771 !hasBottom(pid);
772 }
773
779 inline bool isCharmHadron(int pid) {
780 return isHadron(pid) && hasCharm(pid) &&
781 !hasBottom(pid);
782 }
783
784
789 inline bool isStrangeMeson(int pid) {
790 return isMeson(pid) && hasStrange(pid) &&
791 !(hasBottom(pid) || hasCharm(pid));
792 }
793
798 inline bool isStrangeBaryon(int pid) {
799 return isBaryon(pid) && hasStrange(pid) &&
800 !(hasBottom(pid) || hasCharm(pid));
801 }
802
807 inline bool isStrangeHadron(int pid) {
808 return isHadron(pid) && hasStrange(pid) &&
809 !(hasBottom(pid) || hasCharm(pid));
810 }
811
813
814
815
818
820 inline int jSpin(int pid) {
821 const int fund = _fundamentalID(pid);
822 if (fund > 0) {
823 // some of these are known
824 if (fund > 0 && fund < 7) return 2;
825 if (fund == 9) return 3;
826 if (fund > 10 && fund < 17) return 2;
827 if (fund > 20 && fund < 25) return 3;
828 return 0;
829 } else if (_extraBits(pid) > 0) {
830 return 0;
831 }
832 return abs(pid) % 10;
833 }
834
836 inline int sSpin(int pid) {
837 // Handle invalid cases first
838 if (!isMeson(pid)) return 0;
839 if (_digit(n,pid) == 9 && _digit(nr,pid) == 0) return 0; // tentative ID
840 // Special generic DM particles with defined spins
841 const int fund = _fundamentalID(pid);
842 if (fund == 51 || fund == 54) return 1;
843 if (fund == 52) return 2;
844 if (fund == 53 || fund == 55) return 3;
845 // Calculate from nl and nj digits
846 const int inl = _digit(nl,pid);
847 const int js = _digit(nj,pid);
848 if (inl == 0 && js >= 3) return 1;
849 else if (inl == 0 && js == 1) return 0;
850 else if (inl == 1 && js >= 3) return 0;
851 else if (inl == 2 && js >= 3) return 1;
852 else if (inl == 1 && js == 1) return 1;
853 else if (inl == 3 && js >= 3) return 1;
854 // Default to zero
855 return 0;
856 }
857
859 inline int lSpin(int pid) {
860 // Handle invalid cases first
861 if (!isMeson(pid)) return 0;
862 if (_digit(n,pid) == 9 && _digit(nr,pid) == 0) return 0; // tentative ID
863 // Calculate from nl and nj digits
864 const int inl = _digit(nl,pid);
865 const int js = _digit(nj,pid);
866 if (inl == 0 && js == 3) return 0;
867 else if (inl == 0 && js == 5) return 1;
868 else if (inl == 0 && js == 7) return 2;
869 else if (inl == 0 && js == 9) return 3;
870 else if (inl == 0 && js == 1) return 0;
871 else if (inl == 1 && js == 3) return 1;
872 else if (inl == 1 && js == 5) return 2;
873 else if (inl == 1 && js == 7) return 3;
874 else if (inl == 1 && js == 9) return 4;
875 else if (inl == 2 && js == 3) return 1;
876 else if (inl == 2 && js == 5) return 2;
877 else if (inl == 2 && js == 7) return 3;
878 else if (inl == 2 && js == 9) return 4;
879 else if (inl == 1 && js == 1) return 1;
880 else if (inl == 3 && js == 3) return 2;
881 else if (inl == 3 && js == 5) return 3;
882 else if (inl == 3 && js == 7) return 4;
883 else if (inl == 3 && js == 9) return 5;
884 // Default to zero
885 return 0;
886 }
887
889
890
893
895 inline int charge3(int pid) {
896 static int ch100[100] = { -1, 2, -1, 2, -1, 2, -1, 2, 0, 0,
897 -3, 0, -3, 0, -3, 0, -3, 0, 0, 0,
898 0, 0, 0, 3, 0, 0, 0, 0, 0, 0,
899 0, 0, 0, 3, 0, 0, 3, 0, 0, 0,
900 0, -1, 0, 0, 0, 0, 0, 0, 0, 0,
901 0, 6, 3, 6, 0, 0, 0, 0, 0, 0,
902 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
903 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
904 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
905 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
906 };
907 const int ida = abs(pid);
908 // Shortcuts for common particles
909 if (pid == 21 || pid == 22) return 0; // gluon and photon
910 if (ida == 211) return std::signbit(pid) ? -3 : 3; // charged pion
911 if (pid == 111) return 0; // neutral pion
912 // if (ida == 12 || ida == 14 || ida == 16) return 0; // neutrinos
913 // if (ida == 11 || ida == 13 || ida == 15) return std::signbit(pid) ? +3 : -3; // leptons
914 // if (ida == 1 || ida == 3 || ida == 5) return std::signbit(pid) ? +1 : -1; // quarks
915 // if (ida == 2 || ida == 4 || ida == 6) return std::signbit(pid) ? -2 : +2; // quarks
916 // Standard decoding
917 const unsigned short q1 = _digit(nq1,pid);
918 const unsigned short q2 = _digit(nq2,pid);
919 const unsigned short q3 = _digit(nq3,pid);
920 const unsigned short ql = _digit(nl,pid);
921 const int sid = _fundamentalID(pid);
922 int ch3 = 0;
923 if (ida == 0 || _extraBits(pid) > 0) { // ion or illegal
924 return 0;
925 } else if (sid > 0 && sid <= 100) { // Use table
926 if (ida == 1000017 || ida == 1000018 || ida == 1000034) ch3 = 0;
927 else if (ida > 1000050 && ida <= 1000060) ch3 = 0; // ?
928 else if (ida > 50 && ida <= 60) ch3 = 0; // Generic DM
929 else if (ida == 5100061 || ida == 5100062) ch3 = 6;
930 else ch3 = ch100[sid-1];
931 } else if (_digit(nj,pid) == 0) { // KL, Ks, or undefined
932 return 0;
933 } else if (isMeson(pid)) { // Mesons
934 ch3 = ((q2 == 3 || q2 == 5) ? -1 : 1) * (ch100[q2-1] - ch100[q3-1]);
935 } else if (isBaryon(pid)) { // Baryons
936 ch3 = ch100[q3-1] + ch100[q2-1] + ch100[q1-1];
937 } else if (isQBall(pid) ) { // QBall
938 ch3 = 3*( (ida/10) % 10000);
939 } else if (isHiddenValley(pid) ) { // Hidden Valley
940 return 0;
941 } else if (isDyon(pid) ) { // Dyon
942 ch3 = 3*( (ida/10) % 1000) * (ql == 2 ? -1 : 1); //< NB. charge is flipped at the end if pid < 0
943 } else if (isRHadron(pid) ) { // R-hadron
945 if (q1 == 0 || q1 == 9) { //< gluino+q+qbar
946 if (q2 == 3 || q2 == 5) {
947 ch3 = ch100[q3-1] - ch100[q2-1];
948 } else {
949 ch3 = ch100[q2-1] - ch100[q3-1];
950 }
951 } else if (ql == 0) { //< squark+q+q
952 ch3 = ch100[q3-1] + ch100[q2-1] + ch100[q1-1];
953 } else if (_digit(nr,pid) == 0) { //< squark+q+q+q
954 ch3 = ch100[q3-1] + ch100[q2-1] + ch100[q1-1] + ch100[ql-1];
955 }
956 } else if (isDiQuark(pid)) { // Diquarks
957 ch3 = ch100[q2-1] + ch100[q1-1];
958 } else { // Unknown
959 return 0;
960 }
961 if (pid < 0) ch3 *= -1;
962 return ch3;
963 }
964
967 inline int threeCharge(int pid) {
968 return charge3(pid);
969 }
970
972 inline int abscharge3(int pid) {
973 return std::abs(charge3(pid));
974 }
975
977 inline double charge(int pid) {
978 return charge3(pid)/3.0;
979 }
980
982 inline double abscharge(int pid) {
983 return std::abs(charge(pid));
984 }
985
987
988
991
993 inline bool isCharged(int pid) {
994 return charge3(pid) != 0;
995 }
996
998 inline bool isNeutral(int pid) {
999 return charge3(pid) == 0;
1000 }
1001
1003
1004
1007
1009 inline bool isStrongInteracting(int pid) {
1010 return isParton(pid) || isHadron(pid);
1011 }
1012
1014 inline bool isEMInteracting(int pid) {
1015 return isCharged(pid) || isPhoton(pid);
1016 }
1017
1022 inline bool isWeakInteracting(int pid) {
1023 return !isGluon(pid) && !isGraviton(pid);
1024 }
1025
1027
1028
1031
1033 inline bool isGenSpecific(int pid) {
1034 return in_range(pid, 80, 101);
1035 }
1036
1040 inline bool isResonance(int pid) {
1041 return isW(pid) || isZ(pid) || isHiggs(pid) || isTop(pid);
1042 }
1043
1048 inline bool isTransportable(int pid) {
1049 // return !isResonance(pid) && !isParton(pid) && !isGenSpecific(pid);
1050 return isPhoton(pid) || isHadron(pid) || isLepton(pid);
1051 }
1052
1054
1055
1056
1057
1061
1062 inline bool isSameSign(PdgId a, PdgId b) { return a*b >= 0; }
1063 inline bool isOppSign(PdgId a, PdgId b) { return !isSameSign(a, b); }
1064 inline bool isSameFlav(PdgId a, PdgId b) { return abs(a) == abs(b); }
1065 inline bool isOppFlav(PdgId a, PdgId b) { return !isSameFlav(a, b); }
1066
1067 inline bool isOSSF(PdgId a, PdgId b) { return isOppSign(a, b) && isSameFlav(a, b); }
1068 inline bool isSSSF(PdgId a, PdgId b) { return isSameSign(a, b) && isSameFlav(a, b); }
1069 inline bool isOSOF(PdgId a, PdgId b) { return isOppSign(a, b) && isOppFlav(a, b); }
1070 inline bool isSSOF(PdgId a, PdgId b) { return isSameSign(a, b) && isOppFlav(a, b); }
1071
1073
1074
1075 }
1076}
1077
1078#endif
int sSpin(int pid)
sSpin returns 2S+1, where S is the spin
Definition ParticleIdUtils.hh:836
int jSpin(int pid)
jSpin returns 2J+1, where J is the total spin
Definition ParticleIdUtils.hh:820
int lSpin(int pid)
lSpin returns 2L+1, where L is the orbital angular momentum
Definition ParticleIdUtils.hh:859
bool isCharged(int pid)
Determine if the particle is electrically charged.
Definition ParticleIdUtils.hh:993
bool isNeutral(int pid)
Determine if the particle is electrically neutral.
Definition ParticleIdUtils.hh:998
int threeCharge(int pid)
Definition ParticleIdUtils.hh:967
int abscharge3(int pid)
Return the absolute value of 3 times the EM charge.
Definition ParticleIdUtils.hh:972
double charge(int pid)
Return the EM charge (as floating point)
Definition ParticleIdUtils.hh:977
double abscharge(int pid)
Return the EM charge (as floating point)
Definition ParticleIdUtils.hh:982
int charge3(int pid)
Three times the EM charge (as integer)
Definition ParticleIdUtils.hh:895
bool isBottom(int pid)
Determine if the PID is that of a b/bbar.
Definition ParticleIdUtils.hh:274
bool isParton(int pid)
Determine if the PID is that of a parton (quark or gluon)
Definition ParticleIdUtils.hh:180
bool isW(int pid)
Determine if the PID is that of a W+-.
Definition ParticleIdUtils.hh:234
bool isWminus(int pid)
Determine if the PID is that of a W-.
Definition ParticleIdUtils.hh:229
bool isZ(int pid)
Determine if the PID is that of a Z0.
Definition ParticleIdUtils.hh:239
bool isMuon(int pid)
Determine if the PID is that of an muon or antimuon.
Definition ParticleIdUtils.hh:196
bool isHiggs(int pid)
Determine if the PID is that of an SM/lightest SUSY Higgs.
Definition ParticleIdUtils.hh:244
bool isCharm(int pid)
Determine if the PID is that of a c/cbar.
Definition ParticleIdUtils.hh:269
bool isGraviton(int pid)
Is this a graviton?
Definition ParticleIdUtils.hh:252
bool isChLepton(int pid)
Definition ParticleIdUtils.hh:212
bool isTau(int pid)
Determine if the PID is that of an tau or antitau.
Definition ParticleIdUtils.hh:201
bool isGluon(int pid)
Determine if the PID is that of a gluon.
Definition ParticleIdUtils.hh:175
bool isWplus(int pid)
Determine if the PID is that of a W+.
Definition ParticleIdUtils.hh:224
bool isPhoton(int pid)
Determine if the PID is that of a photon.
Definition ParticleIdUtils.hh:186
bool isNeutrino(int pid)
Determine if the PID is that of a neutrino.
Definition ParticleIdUtils.hh:217
bool isElectron(int pid)
Determine if the PID is that of an electron or positron.
Definition ParticleIdUtils.hh:191
bool isChargedLepton(int pid)
Determine if the PID is that of a charged lepton.
Definition ParticleIdUtils.hh:206
bool isQuark(int pid)
Determine if the PID is that of a quark.
Definition ParticleIdUtils.hh:170
bool isTop(int pid)
Determine if the PID is that of a t/tbar.
Definition ParticleIdUtils.hh:279
bool isStrange(int pid)
Determine if the PID is that of an s/sbar.
Definition ParticleIdUtils.hh:264
bool isRHadron(int pid)
Is this an R-hadron?
Definition ParticleIdUtils.hh:451
bool isQBall(int pid)
Definition ParticleIdUtils.hh:548
bool isSMFundamental(int pid)
Is this an SM fundamental particle?
Definition ParticleIdUtils.hh:425
bool isMagMonopole(int pid)
Is this from a magnetic monopole or dyon?
Definition ParticleIdUtils.hh:530
bool isLepton(int pid)
Definition ParticleIdUtils.hh:412
bool isSUSY(int pid)
Is this a fundamental SUSY particle?
Definition ParticleIdUtils.hh:434
bool isKK(int pid)
Is this a Kaluza-Klein excitation?
Definition ParticleIdUtils.hh:483
bool isRhadron(int pid)
Alias.
Definition ParticleIdUtils.hh:466
bool isExotic(int pid)
Is this an exotic particle?
Definition ParticleIdUtils.hh:517
bool isTechnicolor(int pid)
Is this a technicolor particle?
Definition ParticleIdUtils.hh:471
bool isLeptoQuark(int pid)
Is this a lepto-quark?
Definition ParticleIdUtils.hh:490
bool isAECO(int pid)
Is this an anomalously electrically charged particle (AECO)?
Definition ParticleIdUtils.hh:577
bool isExcitedLepton(int pid)
Is this an excited lepton?
Definition ParticleIdUtils.hh:564
bool isBSM(int pid)
Is this a BSM particle (including graviton)?
Definition ParticleIdUtils.hh:586
bool isFourthGen(int pid)
Is this a 4th generation particle?
Definition ParticleIdUtils.hh:525
bool isBSMBoson(int pid)
Is this a valid BSM boson (SUSY Higgs, W', Z')?
Definition ParticleIdUtils.hh:420
bool isExcited(int pid)
Is this an excited (composite) quark or lepton?
Definition ParticleIdUtils.hh:477
bool isDM(int pid)
Convenience alias.
Definition ParticleIdUtils.hh:507
bool isBlackHole(int pid)
Is this a black hole?
Definition ParticleIdUtils.hh:570
bool isHiddenValley(int pid)
Is this a Hidden Valley particle?
Definition ParticleIdUtils.hh:512
bool isQball(int pid)
Alias.
Definition ParticleIdUtils.hh:559
bool isDyon(int pid)
Just treat a dyon as an alias for magmonopole for now.
Definition ParticleIdUtils.hh:542
bool isDarkMatter(int pid)
Definition ParticleIdUtils.hh:499
bool isEMInteracting(int pid)
Determine if the PID is that of a electromagnetically interacting particle.
Definition ParticleIdUtils.hh:1014
bool isWeakInteracting(int pid)
Definition ParticleIdUtils.hh:1022
bool isStrongInteracting(int pid)
Determine if the PID is that of a strongly interacting particle.
Definition ParticleIdUtils.hh:1009
int lambda(int pid)
Definition ParticleIdUtils.hh:159
int nuclA(int pid)
Definition ParticleIdUtils.hh:135
int Z(int pid)
Definition ParticleIdUtils.hh:129
int A(int pid)
Definition ParticleIdUtils.hh:144
int nuclNlambda(int pid)
Definition ParticleIdUtils.hh:150
bool isNucleus(int pid)
Is this a nucleus PID?
Definition ParticleIdUtils.hh:106
int nuclZ(int pid)
Definition ParticleIdUtils.hh:120
bool isResonance(int pid)
Definition ParticleIdUtils.hh:1040
bool isGenSpecific(int pid)
Determine if the PID is in the generator-specific range.
Definition ParticleIdUtils.hh:1033
bool isTransportable(int pid)
Definition ParticleIdUtils.hh:1048
bool isLightHadron(int pid)
Determine if the PID is that of a light flavour (not b or c) hadron.
Definition ParticleIdUtils.hh:734
bool isCharmHadron(int pid)
Definition ParticleIdUtils.hh:779
bool isStrangeBaryon(int pid)
Definition ParticleIdUtils.hh:798
bool isHeavyFlavor(int pid)
Determine if the particle is a heavy flavour hadron or parton.
Definition ParticleIdUtils.hh:681
bool isHeavyHadron(int pid)
Determine if the PID is that of a heavy flavour (b or c) hadron.
Definition ParticleIdUtils.hh:719
bool isCharmMeson(int pid)
Determine if the PID is that of a c-meson.
Definition ParticleIdUtils.hh:759
bool isStrangeHadron(int pid)
Definition ParticleIdUtils.hh:807
bool isCharmBaryon(int pid)
Determine if the PID is that of a c-baryon.
Definition ParticleIdUtils.hh:769
bool isHeavyBaryon(int pid)
Determine if the PID is that of a heavy flavour (b or c) baryon.
Definition ParticleIdUtils.hh:714
bool isHeavyMeson(int pid)
Determine if the PID is that of a heavy flavour (b or c) meson.
Definition ParticleIdUtils.hh:709
bool isLightBaryon(int pid)
Determine if the PID is that of a light flavour (not b or c) baryon.
Definition ParticleIdUtils.hh:729
bool isBottomBaryon(int pid)
Determine if the PID is that of a b-baryon.
Definition ParticleIdUtils.hh:745
bool isLightParton(int pid)
Determine if the PID is that of a light parton (u,d,s)
Definition ParticleIdUtils.hh:703
bool isStrangeMeson(int pid)
Definition ParticleIdUtils.hh:789
bool isHeavyParton(int pid)
Determine if the PID is that of a heavy parton (c,b,t)
Definition ParticleIdUtils.hh:698
bool isBottomHadron(int pid)
Determine if the PID is that of a b-hadron.
Definition ParticleIdUtils.hh:750
bool isHeavyFlavour(int pid)
British-spelling alias for isHeavyFlavor.
Definition ParticleIdUtils.hh:686
bool isBottomMeson(int pid)
Determine if the PID is that of a b-meson.
Definition ParticleIdUtils.hh:740
bool isLightMeson(int pid)
Determine if the PID is that of a light flavour (not b or c) meson.
Definition ParticleIdUtils.hh:724
bool hasUp(int pid)
Does this particle contain an up quark?
Definition ParticleIdUtils.hh:654
bool hasTop(int pid)
Does this particle contain a top quark?
Definition ParticleIdUtils.hh:670
bool hasBottom(int pid)
Does this particle contain a bottom quark?
Definition ParticleIdUtils.hh:666
bool hasStrange(int pid)
Does this particle contain a strange quark?
Definition ParticleIdUtils.hh:658
bool hasDown(int pid)
Does this particle contain a down quark?
Definition ParticleIdUtils.hh:650
bool hasCharm(int pid)
Does this particle contain a charm quark?
Definition ParticleIdUtils.hh:662
bool isBaryon(int pid)
Check to see if this is a valid baryon.
Definition ParticleIdUtils.hh:318
bool isDiQuark(int pid)
Definition ParticleIdUtils.hh:355
bool isPentaquark(int pid)
Check to see if this is a valid pentaquark.
Definition ParticleIdUtils.hh:360
bool isMeson(int pid)
Check to see if this is a valid meson.
Definition ParticleIdUtils.hh:295
bool isHadron(int pid)
Definition ParticleIdUtils.hh:394
bool isReggeon(int pid)
Is this a pomeron, odderon, or generic reggeon?
Definition ParticleIdUtils.hh:290
bool isQuarkonium(int pid)
Definition ParticleIdUtils.hh:384
Location
Definition ParticleIdUtils.hh:58
int abspid(int pid)
Definition ParticleIdUtils.hh:28
int pid(const Particle &p)
Unbound function access to PID code.
Definition ParticleUtils.hh:23
Definition MC_CENT_PPB_Projections.hh:10
std::enable_if_t< std::is_arithmetic_v< N1 > &&std::is_arithmetic_v< N2 > &&std::is_arithmetic_v< N3 >, bool > in_range(N1 val, N2 low, N3 high)
Boolean function to determine if value is within the given range.
Definition MathUtils.hh:185
std::enable_if_t< std::is_arithmetic_v< N1 > &&std::is_arithmetic_v< N2 > &&std::is_arithmetic_v< N3 >, bool > in_closed_range(N1 val, N2 low, N3 high)
Boolean function to determine if value is within the given range.
Definition MathUtils.hh:194