18#ifndef GPUCA_GPUCODE_DEVICE
22#ifndef GPUCA_ALIGPUCODE
23#include <fmt/printf.h>
30template <
typename value_T>
45template <
typename value_T>
51 value_t dx = xk - this->
getX();
52 if (gpu::CAMath::Abs(dx) < constants::math::Almost0) {
55 value_t crv = this->getCurvature(
b);
56 value_t x2r = crv * dx;
57 value_t
f1 = this->getSnp(),
f2 =
f1 + x2r;
58 if ((gpu::CAMath::Abs(
f1) > constants::math::Almost1) || (gpu::CAMath::Abs(
f2) > constants::math::Almost1)) {
61 value_t r1 = gpu::CAMath::Sqrt((1.f -
f1) * (1.f +
f1));
62 if (gpu::CAMath::Abs(r1) < constants::math::Almost0) {
65 value_t r2 = gpu::CAMath::Sqrt((1.f -
f2) * (1.f +
f2));
66 if (gpu::CAMath::Abs(r2) < constants::math::Almost0) {
69 double dy2dx = (
f1 +
f2) / (r1 + r2);
70 bool arcz = gpu::CAMath::Abs(x2r) > 0.05f;
81 auto arg = r1 *
f2 - r2 *
f1;
82 if (gpu::CAMath::Abs(arg) > constants::math::Almost1) {
85 value_t rot = gpu::CAMath::ASin(arg);
88 rot = constants::math::PI - rot;
90 rot = -constants::math::PI - rot;
93 dP[
kZ] = this->getTgl() / crv * rot;
95 dP[
kZ] = dx * (r2 +
f2 * dy2dx) * this->getTgl();
101 this->updateParams(dP);
109 double rinv = 1. / r1;
110 double r3inv = rinv * rinv * rinv;
111 double f24 = dx *
b * constants::math::B2C;
112 double f02 = dx * r3inv;
113 double f04 = 0.5 * f24 * f02;
114 double f12 = f02 * this->getTgl() *
f1;
115 double f14 = 0.5 * f24 * f12;
116 double f13 = dx * rinv;
119 double b00 = f02 * c20 + f04 * c40, b01 = f12 * c20 + f14 * c40 + f13 * c30;
120 double b02 = f24 * c40;
121 double b10 = f02 * c21 + f04 * c41, b11 = f12 * c21 + f14 * c41 + f13 * c31;
122 double b12 = f24 * c41;
123 double b20 = f02 * c22 + f04 * c42, b21 = f12 * c22 + f14 * c42 + f13 * c32;
124 double b22 = f24 * c42;
125 double b40 = f02 * c42 + f04 * c44, b41 = f12 * c42 + f14 * c44 + f13 * c43;
126 double b42 = f24 * c44;
127 double b30 = f02 * c32 + f04 * c43, b31 = f12 * c32 + f14 * c43 + f13 * c33;
128 double b32 = f24 * c43;
131 double a00 = f02 * b20 + f04 * b40, a01 = f02 * b21 + f04 * b41, a02 = f02 * b22 + f04 * b42;
132 double a11 = f12 * b21 + f14 * b41 + f13 * b31, a12 = f12 * b22 + f14 * b42 + f13 * b32;
133 double a22 = f24 * b42;
136 c00 += b00 + b00 + a00;
137 c10 += b10 + b01 + a01;
138 c20 += b20 + b02 + a02;
141 c11 += b11 + b11 + a11;
142 c21 += b21 + b12 + a12;
145 c22 += b22 + b22 + a22;
155template <
typename value_T>
162template <
typename value_T>
166 if (gpu::CAMath::Abs(this->getSnp()) > constants::math::Almost1) {
167 LOGP(
debug,
"Precondition is not satisfied: |sin(phi)|>1 ! {:f}", this->getSnp());
171 math_utils::detail::bringToPMPi<value_t>(
alpha);
173 value_t ca = 0, sa = 0;
174 math_utils::detail::sincos(
alpha - this->getAlpha(), sa, ca);
175 value_t snp = this->getSnp(), csp = gpu::CAMath::Sqrt((1.f - snp) * (1.f + snp));
178 if ((csp * ca + snp * sa) < 0) {
184 value_t updSnp = snp * ca - csp * sa;
185 if (gpu::CAMath::Abs(updSnp) > constants::math::Almost1) {
186 LOGP(
debug,
"Rotation failed: new snp {:.2f}", updSnp);
189 value_t xold = this->
getX(), yold = this->
getY();
190 this->setAlpha(
alpha);
191 this->setX(xold * ca + yold * sa);
192 this->setY(-xold * sa + yold * ca);
193 this->setSnp(updSnp);
195 if (gpu::CAMath::Abs(csp) < constants::math::Almost0) {
196 LOGP(
debug,
"Too small cosine value {:f}", csp);
197 csp = constants::math::Almost0;
200 value_t
rr = (ca + snp / csp * sa);
217template <
typename value_T>
221 value_t sn, cs, alp = this->getAlpha();
222 o2::math_utils::detail::sincos(alp, sn, cs);
223 value_t
x = this->
getX(),
y = this->
getY(), snp = this->getSnp(), csp = gpu::CAMath::Sqrt((1.f - snp) * (1.f + snp));
224 value_t xv = vtx.getX() * cs + vtx.getY() * sn, yv = -vtx.getX() * sn + vtx.getY() * cs, zv = vtx.getZ();
228 value_t d = gpu::CAMath::Abs(
x * snp -
y * csp);
232 value_t crv = this->getCurvature(
b);
233 value_t tgfv = -(crv *
x - snp) / (crv *
y + csp);
234 sn = tgfv / gpu::CAMath::Sqrt(1.f + tgfv * tgfv);
235 cs = gpu::CAMath::Sqrt((1.f - sn) * (1.f + sn));
236 cs = (gpu::CAMath::Abs(tgfv) > constants::math::Almost0) ? sn / tgfv : constants::math::
Almost1;
238 x = xv * cs + yv * sn;
239 yv = -xv * sn + yv * cs;
243 alp += gpu::CAMath::ASin(sn);
244 if (!tmpT.rotate(alp) || !tmpT.propagateTo(xv,
b)) {
245#if !defined(GPUCA_ALIGPUCODE)
246 LOG(
debug) <<
"failed to propagate to alpha=" << alp <<
" X=" << xv << vtx <<
" | Track is: " << tmpT.asString();
252 o2::math_utils::detail::sincos(alp, sn, cs);
253 auto s2ylocvtx = vtx.getSigmaX2() * sn * sn + vtx.getSigmaY2() * cs * cs - 2. * vtx.getSigmaXY() * cs * sn;
254 dca->set(this->
getY() - yv, this->getZ() - zv, getSigmaY2() + s2ylocvtx, getSigmaZY(), getSigmaZ2() + vtx.getSigmaZ2());
260template <
typename value_T>
269template <
typename value_T>
281 constexpr value_t kSafe = 1e-5f;
282 value_t radPos2 = xyz[0] * xyz[0] + xyz[1] * xyz[1];
284 if (sectorAlpha || radPos2 < 1) {
285 alp = gpu::CAMath::ATan2(pxpypz[1], pxpypz[0]);
287 alp = gpu::CAMath::ATan2(xyz[1], xyz[0]);
290 alp = math_utils::detail::angle2Alpha<value_t>(alp);
294 math_utils::detail::sincos(alp, sn, cs);
296 if (cs * pxpypz[0] + sn * pxpypz[1] < 0) {
297 LOG(
debug) <<
"alpha from phiPos() will invalidate this track parameters, overriding to alpha from phi()";
298 alp = gpu::CAMath::ATan2(pxpypz[1], pxpypz[0]);
300 alp = math_utils::detail::angle2Alpha<value_t>(alp);
302 math_utils::detail::sincos(alp, sn, cs);
305 if (gpu::CAMath::Abs(sn) < 2.f * kSafe) {
307 alp += alp < constants::math::PIHalf ? 2.f * kSafe : -2.f * kSafe;
309 alp += alp > -constants::math::PIHalf ? -2.f * kSafe : 2.f * kSafe;
311 math_utils::detail::sincos(alp, sn, cs);
312 }
else if (gpu::CAMath::Abs(cs) < 2.f * kSafe) {
314 alp += alp > constants::math::PIHalf ? 2.f * kSafe : -2.f * kSafe;
316 alp += alp > -constants::math::PIHalf ? 2.f * kSafe : -2.f * kSafe;
318 math_utils::detail::sincos(alp, sn, cs);
321 dim3_t ver{xyz[0], xyz[1], xyz[2]};
322 dim3_t mom{pxpypz[0], pxpypz[1], pxpypz[2]};
325 math_utils::detail::rotateZ<value_t>(ver, -alp);
326 math_utils::detail::rotateZ<value_t>(mom, -alp);
328 value_t pt = gpu::CAMath::Sqrt(mom[0] * mom[0] + mom[1] * mom[1]);
329 value_t ptI = 1.f / pt;
334 this->setSnp(mom[1] * ptI);
335 this->setTgl(mom[2] * ptI);
336 this->setAbsCharge(gpu::CAMath::Abs(
charge));
340 if (gpu::CAMath::Abs(1.f - this->getSnp()) < kSafe) {
341 this->setSnp(1.f - kSafe);
342 }
else if (gpu::CAMath::Abs(-1.f - this->getSnp()) < kSafe) {
343 this->setSnp(-1.f + kSafe);
347 value_t
r = mom[0] * ptI;
348 value_t cv34 = gpu::CAMath::Sqrt(cv[3] * cv[3] + cv[4] * cv[4]);
351 value_t sgcheck =
r * sn + this->getSnp() * cs;
352 if (gpu::CAMath::Abs(sgcheck) > 1 - kSafe) {
354 sgcheck = sgcheck < 0 ? -1.f : 1.f;
355 }
else if (gpu::CAMath::Abs(sgcheck) < kSafe) {
356 sgcheck = cs < 0 ? -1.0f : 1.0f;
360 mC[
kSigY2] = cv[0] + cv[2];
361 mC[
kSigZY] = (-cv[3] * sn) < 0 ? -cv34 : cv34;
364 value_t ptI2 = ptI * ptI;
365 value_t tgl2 = this->getTgl() * this->getTgl();
368 mC[
kSigSnpZ] = -sgcheck * cv[8] *
r * ptI;
369 mC[
kSigSnp2] = gpu::CAMath::Abs(cv[9] *
r *
r * ptI2);
370 mC[
kSigTglY] = (cv[10] * this->getTgl() - sgcheck * cv[15]) * ptI /
r;
371 mC[
kSigTglZ] = (cv[17] - sgcheck * cv[12] * this->getTgl()) * ptI;
372 mC[
kSigTglSnp] = (-sgcheck * cv[18] + cv[13] * this->getTgl()) *
r * ptI2;
373 mC[
kSigTgl2] = gpu::CAMath::Abs(cv[20] - 2 * sgcheck * cv[19] * mC[4] + cv[14] * tgl2) * ptI2;
377 mC[
kSigQ2PtTgl] = (-sgcheck * cv[19] + cv[14] * this->getTgl()) *
r * ptI2 * ptI;
378 mC[
kSigQ2Pt2] = gpu::CAMath::Abs(cv[14] * ptI2 * ptI2);
379 }
else if (special == 2) {
380 mC[
kSigSnpY] = -cv[10] * ptI * cs / sn;
382 mC[
kSigSnp2] = gpu::CAMath::Abs(cv[14] * cs * cs * ptI2);
383 mC[
kSigTglY] = (sgcheck * cv[6] * this->getTgl() - cv[15]) * ptI / sn;
384 mC[
kSigTglZ] = (cv[17] - sgcheck * cv[8] * this->getTgl()) * ptI;
385 mC[
kSigTglSnp] = (cv[19] - sgcheck * cv[13] * this->getTgl()) * cs * ptI2;
386 mC[
kSigTgl2] = gpu::CAMath::Abs(cv[20] - 2 * sgcheck * cv[18] * this->getTgl() + cv[9] * tgl2) * ptI2;
390 mC[
kSigQ2PtTgl] = (-sgcheck * cv[18] + cv[9] * this->getTgl()) * ptI2 * ptI *
charge;
391 mC[
kSigQ2Pt2] = gpu::CAMath::Abs(cv[9] * ptI2 * ptI2);
394 double m23 = -pt * (sn + this->getSnp() * cs /
r), m43 = -pt * pt * (
r * cs - this->getSnp() * sn);
395 double m24 = pt * (cs - this->getSnp() * sn /
r), m44 = -pt * pt * (
r * sn + this->getSnp() * cs);
396 double m35 = pt, m45 = -pt * pt * this->getTgl();
404 double a1 = cv[13] - cv[9] * (m23 * m44 + m43 * m24) / m23 / m43;
405 double a2 = m23 * m24 - m23 * (m23 * m44 + m43 * m24) / m43;
406 double a3 = m43 * m44 - m43 * (m23 * m44 + m43 * m24) / m23;
407 double a4 = cv[14] + 2. * cv[9];
408 double a5 = m24 * m24 - 2. * m24 * m44 * m23 / m43;
409 double a6 = m44 * m44 - 2. * m24 * m44 * m43 / m23;
411 mC[
kSigSnpY] = (cv[10] * m43 - cv[6] * m44) / (m24 * m43 - m23 * m44) / m00;
414 mC[
kSigSnpZ] = (cv[12] * m43 - cv[8] * m44) / (m24 * m43 - m23 * m44);
417 mC[
kSigSnp2] = gpu::CAMath::Abs((a4 * a3 - a6 *
a1) / (a5 * a3 - a6 * a2));
421 double b2 = m23 * m35;
422 double b3 = m43 * m35;
424 double b5 = m24 * m35;
425 double b6 = m44 * m35;
426 mC[
kSigTglSnp] = (b4 - b6 *
b1 / b3) / (b5 - b6 * b2 / b3);
434template <
typename value_T>
444 value_t dx = xk - this->
getX();
445 if (gpu::CAMath::Abs(dx) < constants::math::Almost0) {
449 if (gpu::CAMath::Abs(dx) > 1e5 || gpu::CAMath::Abs(this->
getY()) > 1e5 || gpu::CAMath::Abs(this->getZ()) > 1e5) {
450 LOG(warning) <<
"Anomalous track, target X:" << xk;
454 value_t crv = (gpu::CAMath::Abs(
b[2]) < constants::math::Almost0) ? 0.f :
this->getCurvature(
b[2]);
455 if (gpu::CAMath::Abs(crv) < constants::math::Almost0) {
456 return propagateTo(xk, 0.);
458 value_t x2r = crv * dx;
459 value_t
f1 = this->getSnp(),
f2 =
f1 + x2r;
460 if ((gpu::CAMath::Abs(
f1) > constants::math::Almost1) || (gpu::CAMath::Abs(
f2) > constants::math::Almost1)) {
463 value_t r1 = gpu::CAMath::Sqrt((1.f -
f1) * (1.f +
f1));
464 if (gpu::CAMath::Abs(r1) < constants::math::Almost0) {
467 value_t r2 = gpu::CAMath::Sqrt((1.f -
f2) * (1.f +
f2));
468 if (gpu::CAMath::Abs(r2) < constants::math::Almost0) {
472 value_t dy2dx = (
f1 +
f2) / (r1 + r2);
473 value_t
step = (gpu::CAMath::Abs(x2r) < 0.05f) ? dx * gpu::CAMath::Abs(r2 +
f2 * dy2dx)
474 : 2.f * gpu::CAMath::ASin(0.5f * dx * gpu::CAMath::Sqrt(1.f + dy2dx * dy2dx) * crv) / crv;
475 step *= gpu::CAMath::Sqrt(1.f + this->getTgl() * this->getTgl());
479 if (!this->getPosDirGlo(vecLab)) {
489 double rinv = 1. / r1;
490 double r3inv = rinv * rinv * rinv;
491 double f24 = dx *
b[2] * constants::math::B2C;
492 double f02 = dx * r3inv;
493 double f04 = 0.5 * f24 * f02;
494 double f12 = f02 * this->getTgl() *
f1;
495 double f14 = 0.5 * f24 * f12;
496 double f13 = dx * rinv;
499 double b00 = f02 * c20 + f04 * c40, b01 = f12 * c20 + f14 * c40 + f13 * c30;
500 double b02 = f24 * c40;
501 double b10 = f02 * c21 + f04 * c41, b11 = f12 * c21 + f14 * c41 + f13 * c31;
502 double b12 = f24 * c41;
503 double b20 = f02 * c22 + f04 * c42, b21 = f12 * c22 + f14 * c42 + f13 * c32;
504 double b22 = f24 * c42;
505 double b40 = f02 * c42 + f04 * c44, b41 = f12 * c42 + f14 * c44 + f13 * c43;
506 double b42 = f24 * c44;
507 double b30 = f02 * c32 + f04 * c43, b31 = f12 * c32 + f14 * c43 + f13 * c33;
508 double b32 = f24 * c43;
511 double a00 = f02 * b20 + f04 * b40, a01 = f02 * b21 + f04 * b41, a02 = f02 * b22 + f04 * b42;
512 double a11 = f12 * b21 + f14 * b41 + f13 * b31, a12 = f12 * b22 + f14 * b42 + f13 * b32;
513 double a22 = f24 * b42;
516 c00 += b00 + b00 + a00;
517 c10 += b10 + b01 + a01;
518 c20 += b20 + b02 + a02;
521 c11 += b11 + b11 + a11;
522 c21 += b21 + b12 + a12;
525 c22 += b22 + b22 + a22;
532 value_t bxy2 =
b[0] *
b[0] +
b[1] *
b[1];
533 value_t bt = gpu::CAMath::Sqrt(bxy2);
534 value_t cosphi = 1.f, sinphi = 0.f;
535 if (bt > constants::math::Almost0) {
539 value_t
bb = gpu::CAMath::Sqrt(bxy2 +
b[2] *
b[2]);
540 value_t costet = 1., sintet = 0.;
541 if (
bb > constants::math::Almost0) {
546 -sinphi * vecLab[0] + cosphi * vecLab[1],
547 sintet * cosphi * vecLab[0] + sintet * sinphi * vecLab[1] + costet * vecLab[2],
548 costet * cosphi * vecLab[3] + costet * sinphi * vecLab[4] - sintet * vecLab[5],
549 -sinphi * vecLab[3] + cosphi * vecLab[4],
550 sintet * cosphi * vecLab[3] + sintet * sinphi * vecLab[4] + costet * vecLab[5],
554 value_t q = this->getCharge();
558 vecLab[0] = cosphi * costet *
vect[0] - sinphi *
vect[1] + cosphi * sintet *
vect[2];
559 vecLab[1] = sinphi * costet *
vect[0] + cosphi *
vect[1] + sinphi * sintet *
vect[2];
560 vecLab[2] = -sintet *
vect[0] + costet *
vect[2];
562 vecLab[3] = cosphi * costet *
vect[3] - sinphi *
vect[4] + cosphi * sintet *
vect[5];
563 vecLab[4] = sinphi * costet *
vect[3] + cosphi *
vect[4] + sinphi * sintet *
vect[5];
564 vecLab[5] = -sintet *
vect[3] + costet *
vect[5];
567 value_t sinalp = -vecLab[7], cosalp = vecLab[8];
568 value_t t = cosalp * vecLab[0] - sinalp * vecLab[1];
569 vecLab[1] = sinalp * vecLab[0] + cosalp * vecLab[1];
571 t = cosalp * vecLab[3] - sinalp * vecLab[4];
572 vecLab[4] = sinalp * vecLab[3] + cosalp * vecLab[4];
576 value_t
x = vecLab[0],
y = vecLab[1],
z = vecLab[2];
577 if (gpu::CAMath::Abs(dx) > constants::math::Almost0) {
578 if (gpu::CAMath::Abs(vecLab[3]) < constants::math::Almost0) {
583 y += vecLab[4] / vecLab[3] * dx;
584 z += vecLab[5] / vecLab[3] * dx;
588 t = 1.f / gpu::CAMath::Sqrt(vecLab[3] * vecLab[3] + vecLab[4] * vecLab[4]);
592 this->setSnp(vecLab[4] * t);
593 this->setTgl(vecLab[5] * t);
594 this->setQ2Pt(q * t / vecLab[6]);
600template <
typename value_T>
604 constexpr float MaxCorr = 0.99;
606 for (
int j =
i;
j--;) {
607 auto sig2 = mC[DiagMap[
i]] * mC[DiagMap[
j]];
608 auto& cov = mC[CovarMap[
i][
j]];
609 if (cov * cov >= MaxCorr * sig2) {
610 cov = gpu::CAMath::Sqrt(sig2) * (cov > 0. ? MaxCorr : -MaxCorr);
617template <
typename value_T>
672template <
typename value_T>
677 if (s2 > constants::math::Almost0) {
678 d0 = getSigmaY2() * s2;
679 d1 = getSigmaZ2() * s2;
680 d2 = getSigmaSnp2() * s2;
681 d3 = getSigmaTgl2() * s2;
682 d4 = getSigma1Pt2() * s2;
710template <
typename value_T>
714 auto sdd =
static_cast<double>(getSigmaY2()) +
static_cast<double>(cov[0]);
715 auto sdz =
static_cast<double>(getSigmaZY()) +
static_cast<double>(cov[1]);
716 auto szz =
static_cast<double>(getSigmaZ2()) +
static_cast<double>(cov[2]);
717 auto det = sdd * szz - sdz * sdz;
719 if (gpu::CAMath::Abs(det) < constants::math::Almost0) {
720 return constants::math::VeryBig;
723 value_t d = this->
getY() - p[0];
724 value_t
z = this->getZ() - p[1];
725 auto chi2 = (d * (szz * d - sdz *
z) +
z * (sdd *
z - d * sdz)) / det;
727#ifndef GPUCA_ALIGPUCODE
728 LOGP(warning,
"Negative chi2={}, Cluster: {} {} {} Dy:{} Dz:{} | sdd:{} sdz:{} szz:{} det:{}", chi2, cov[0], cov[1], cov[2], d,
z, sdd, sdz, szz, det);
729 LOGP(warning,
"Track: {}",
asString());
736template <
typename value_T>
740 auto sdd =
static_cast<double>(getSigmaY2()) +
static_cast<double>(cov[0]);
741 auto sdz =
static_cast<double>(getSigmaZY()) +
static_cast<double>(cov[1]);
742 auto szz =
static_cast<double>(getSigmaZ2()) +
static_cast<double>(cov[2]);
743 auto det = sdd * szz - sdz * sdz;
745 if (gpu::CAMath::Abs(det) < constants::math::Almost0) {
746 return constants::math::VeryBig;
749 value_t d = this->
getY() - p[0];
750 value_t
z = this->getZ() - p[1];
752 return (d * (szz * d - sdz *
z) +
z * (sdd *
z - d * sdz)) / det;
756template <
typename value_T>
760 return getPredictedChi2(rhs, cov);
764template <
typename value_T>
768 cov(
kY,
kY) =
static_cast<double>(getSigmaY2()) +
static_cast<double>(
rhs.getSigmaY2());
769 cov(
kZ,
kY) =
static_cast<double>(getSigmaZY()) +
static_cast<double>(
rhs.getSigmaZY());
770 cov(
kZ,
kZ) =
static_cast<double>(getSigmaZ2()) +
static_cast<double>(
rhs.getSigmaZ2());
771 cov(
kSnp,
kY) =
static_cast<double>(getSigmaSnpY()) +
static_cast<double>(
rhs.getSigmaSnpY());
772 cov(
kSnp,
kZ) =
static_cast<double>(getSigmaSnpZ()) +
static_cast<double>(
rhs.getSigmaSnpZ());
773 cov(
kSnp,
kSnp) =
static_cast<double>(getSigmaSnp2()) +
static_cast<double>(
rhs.getSigmaSnp2());
774 cov(
kTgl,
kY) =
static_cast<double>(getSigmaTglY()) +
static_cast<double>(
rhs.getSigmaTglY());
775 cov(
kTgl,
kZ) =
static_cast<double>(getSigmaTglZ()) +
static_cast<double>(
rhs.getSigmaTglZ());
776 cov(
kTgl,
kSnp) =
static_cast<double>(getSigmaTglSnp()) +
static_cast<double>(
rhs.getSigmaTglSnp());
777 cov(
kTgl,
kTgl) =
static_cast<double>(getSigmaTgl2()) +
static_cast<double>(
rhs.getSigmaTgl2());
778 cov(
kQ2Pt,
kY) =
static_cast<double>(getSigma1PtY()) +
static_cast<double>(
rhs.getSigma1PtY());
779 cov(
kQ2Pt,
kZ) =
static_cast<double>(getSigma1PtZ()) +
static_cast<double>(
rhs.getSigma1PtZ());
780 cov(
kQ2Pt,
kSnp) =
static_cast<double>(getSigma1PtSnp()) +
static_cast<double>(
rhs.getSigma1PtSnp());
781 cov(
kQ2Pt,
kTgl) =
static_cast<double>(getSigma1PtTgl()) +
static_cast<double>(
rhs.getSigma1PtTgl());
782 cov(
kQ2Pt,
kQ2Pt) =
static_cast<double>(getSigma1Pt2()) +
static_cast<double>(
rhs.getSigma1Pt2());
786template <
typename value_T>
793 LOG(error) <<
"The reference Alpha of the tracks differ: " << this->getAlpha() <<
" : " <<
rhs.getAlpha();
797 LOG(error) <<
"The reference X of the tracks differ: " << this->
getX() <<
" : " <<
rhs.getX();
800 buildCombinedCovMatrix(rhs, covToSet);
801 if (!covToSet.Invert()) {
802 LOG(warning) <<
"Cov.matrix inversion failed: " << covToSet;
805 double chi2diag = 0., chi2ndiag = 0., diff[
kNParams];
807 diff[
i] = this->getParam(
i) -
rhs.getParam(
i);
808 chi2diag += diff[
i] * diff[
i] * covToSet(
i,
i);
811 for (
int j =
i;
j--;) {
812 chi2ndiag += diff[
i] * diff[
j] * covToSet(
i,
j);
815 return chi2diag + 2. * chi2ndiag;
819template <
typename value_T>
826 LOG(error) <<
"The reference Alpha of the tracks differ: " << this->getAlpha() <<
" : " <<
rhs.getAlpha();
830 LOG(error) <<
"The reference X of the tracks differ: " << this->
getX() <<
" : " <<
rhs.getX();
836 matC0(
kY,
kY) = getSigmaY2();
837 matC0(
kZ,
kY) = getSigmaZY();
838 matC0(
kZ,
kZ) = getSigmaZ2();
839 matC0(
kSnp,
kY) = getSigmaSnpY();
840 matC0(
kSnp,
kZ) = getSigmaSnpZ();
842 matC0(
kTgl,
kY) = getSigmaTglY();
843 matC0(
kTgl,
kZ) = getSigmaTglZ();
844 matC0(
kTgl,
kSnp) = getSigmaTglSnp();
846 matC0(
kQ2Pt,
kY) = getSigma1PtY();
847 matC0(
kQ2Pt,
kZ) = getSigma1PtZ();
851 MatrixD5 matK = matC0 * covInv;
857 diff[
i] =
rhs.getParam(
i) - this->getParam(
i);
861 this->updateParam(matK(
i,
j) * diff[
j],
i);
887template <
typename value_T>
892 buildCombinedCovMatrix(rhs, covI);
893 if (!covI.Invert()) {
894 LOG(warning) <<
"Cov.matrix inversion failed: " << covI;
897 return update(rhs, covI);
901template <
typename value_T>
913 double r00 =
static_cast<double>(cov[0]) +
static_cast<double>(cm00);
914 double r01 =
static_cast<double>(cov[1]) +
static_cast<double>(cm10);
915 double r11 =
static_cast<double>(cov[2]) +
static_cast<double>(cm11);
916 double det = r00 * r11 - r01 * r01;
918 if (gpu::CAMath::Abs(det) < constants::math::Almost0) {
921 double detI = 1. / det;
927 double k00 = cm00 * r00 + cm10 * r01, k01 = cm00 * r01 + cm10 * r11;
928 double k10 = cm10 * r00 + cm11 * r01, k11 = cm10 * r01 + cm11 * r11;
929 double k20 = cm20 * r00 + cm21 * r01, k21 = cm20 * r01 + cm21 * r11;
930 double k30 = cm30 * r00 + cm31 * r01, k31 = cm30 * r01 + cm31 * r11;
931 double k40 = cm40 * r00 + cm41 * r01, k41 = cm40 * r01 + cm41 * r11;
933 value_t dy = p[
kY] - this->
getY(), dz = p[
kZ] - this->getZ();
934 value_t dsnp = k20 * dy + k21 * dz;
935 if (gpu::CAMath::Abs(this->getSnp() + dsnp) > constants::math::Almost1) {
939 const params_t dP{value_t(k00 * dy + k01 * dz), value_t(k10 * dy + k11 * dz), dsnp, value_t(k30 * dy + k31 * dz),
940 value_t(k40 * dy + k41 * dz)};
941 this->updateParams(dP);
943 double c01 = cm10, c02 = cm20, c03 = cm30, c04 = cm40;
944 double c12 = cm21, c13 = cm31, c14 = cm41;
946 cm00 -= k00 * cm00 + k01 * cm10;
947 cm10 -= k00 * c01 + k01 * cm11;
948 cm20 -= k00 * c02 + k01 * c12;
949 cm30 -= k00 * c03 + k01 * c13;
950 cm40 -= k00 * c04 + k01 * c14;
952 cm11 -= k10 * c01 + k11 * cm11;
953 cm21 -= k10 * c02 + k11 * c12;
954 cm31 -= k10 * c03 + k11 * c13;
955 cm41 -= k10 * c04 + k11 * c14;
957 cm22 -= k20 * c02 + k21 * c12;
958 cm32 -= k20 * c03 + k21 * c13;
959 cm42 -= k20 * c04 + k21 * c14;
961 cm33 -= k30 * c03 + k31 * c13;
962 cm43 -= k30 * c04 + k31 * c14;
964 cm44 -= k40 * c04 + k41 * c14;
972template <
typename value_T>
977 auto vtLoc = this->getVertexInTrackFrame(vtx);
978 value_T chi2 = getPredictedChi2(vtLoc.yz, vtLoc.yzerr);
979 return chi2 < maxChi2 && update(vtLoc.yz, vtLoc.yzerr) ? chi2 : -chi2;
983template <
typename value_T>
995 constexpr value_t kMSConst2 = 0.0136f * 0.0136f;
996 constexpr value_t kMinP = 0.01f;
998 value_t csp2 = (1.f - this->getSnp()) * (1.f + this->getSnp());
999 value_t cst2I = (1.f + this->getTgl() * this->getTgl());
1001 value_t
angle = gpu::CAMath::Sqrt(cst2I / csp2);
1005 auto m = this->getPID().getMass();
1006 int charge2 = this->getAbsCharge() * this->getAbsCharge();
1007 value_t p = this->getP(),
p0 = p, p02 = p * p, e2 = p02 + this->getPID().getMass2(), massInv = 1. /
m,
bg = p * massInv, dETot = 0.;
1008 value_t e = gpu::CAMath::Sqrt(e2), e0 = e;
1009 if (
m > 0 && xrho != 0.f) {
1010 value_t ekin = e -
m,
dedx = this->getdEdxBBOpt(
bg);
1011#ifdef _BB_NONCONST_CORR_
1012 value_t dedxDer = 0., dedx1 =
dedx;
1017 value_t dE =
dedx * xrho;
1025#ifdef _BB_NONCONST_CORR_
1026 dedxDer = this->getBetheBlochSolidDerivativeApprox(dedx1,
bg);
1033#ifdef _BB_NONCONST_CORR_
1034 if (dedxDer != 0.) {
1038 auto corrC = (gpu::CAMath::Exp(dedxDer) - 1.) / dedxDer;
1044 p = gpu::CAMath::Sqrt(e * e - this->getPID().getMass2());
1050 dedx = this->getdEdxBBOpt(
bg);
1051#ifdef _BB_NONCONST_CORR_
1052 dedxDer = this->getBetheBlochSolidDerivativeApprox(
dedx,
bg);
1056#ifdef _BB_NONCONST_CORR_
1076 value_t cC22(0.f), cC33(0.f), cC43(0.f), cC44(0.f);
1078 value_t
beta2 = p02 / e2, theta2 = kMSConst2 / (
beta2 * p02) * gpu::CAMath::Abs(x2x0);
1079 value_t fp34 = this->getTgl();
1082 fp34 *= this->getCharge2Pt();
1084 if (theta2 > constants::math::PI * constants::math::PI) {
1087 value_t t2c2I = theta2 * cst2I;
1088 cC22 = t2c2I * csp2;
1089 cC33 = t2c2I * cst2I;
1090 cC43 = t2c2I * fp34;
1091 cC44 = theta2 * fp34 * fp34;
1100 constexpr value_t knst = 0.0007f;
1101 value_t sigmadE = knst * gpu::CAMath::Sqrt(gpu::CAMath::Abs(dETot)) * e0 / p02 * this->getCharge2Pt();
1102 cC44 += sigmadE * sigmadE;
1109 this->setQ2Pt(this->getQ2Pt() * p0 / p);
1117template <
typename value_T>
1132 if (gpu::CAMath::Abs(this->getQ2Pt()) <= constants::math::Almost0 || gpu::CAMath::Abs(this->getSnp()) > constants::math::Almost1) {
1133 for (
int i = 0;
i < 21;
i++) {
1139 auto pt = this->getPt();
1141 o2::math_utils::detail::sincos(this->getAlpha(), sn, cs);
1142 auto r = gpu::CAMath::Sqrt((1. - this->getSnp()) * (1. + this->getSnp()));
1143 auto m00 = -sn, m10 = cs;
1144 auto m23 = -pt * (sn + this->getSnp() * cs /
r), m43 = -pt * pt * (
r * cs - this->getSnp() * sn);
1145 auto m24 = pt * (cs - this->getSnp() * sn /
r), m44 = -pt * pt * (
r * sn + this->getSnp() * cs);
1146 auto m35 = pt, m45 = -pt * pt * this->getTgl();
1148 if (this->getSign() < 0) {
1154 cv[0] = mC[0] * m00 * m00;
1155 cv[1] = mC[0] * m00 * m10;
1156 cv[2] = mC[0] * m10 * m10;
1157 cv[3] = mC[1] * m00;
1158 cv[4] = mC[1] * m10;
1160 cv[6] = m00 * (mC[3] * m23 + mC[10] * m43);
1161 cv[7] = m10 * (mC[3] * m23 + mC[10] * m43);
1162 cv[8] = mC[4] * m23 + mC[11] * m43;
1163 cv[9] = m23 * (mC[5] * m23 + mC[12] * m43) + m43 * (mC[12] * m23 + mC[14] * m43);
1164 cv[10] = m00 * (mC[3] * m24 + mC[10] * m44);
1165 cv[11] = m10 * (mC[3] * m24 + mC[10] * m44);
1166 cv[12] = mC[4] * m24 + mC[11] * m44;
1167 cv[13] = m23 * (mC[5] * m24 + mC[12] * m44) + m43 * (mC[12] * m24 + mC[14] * m44);
1168 cv[14] = m24 * (mC[5] * m24 + mC[12] * m44) + m44 * (mC[12] * m24 + mC[14] * m44);
1169 cv[15] = m00 * (mC[6] * m35 + mC[10] * m45);
1170 cv[16] = m10 * (mC[6] * m35 + mC[10] * m45);
1171 cv[17] = mC[7] * m35 + mC[11] * m45;
1172 cv[18] = m23 * (mC[8] * m35 + mC[12] * m45) + m43 * (mC[13] * m35 + mC[14] * m45);
1173 cv[19] = m24 * (mC[8] * m35 + mC[12] * m45) + m44 * (mC[13] * m35 + mC[14] * m45);
1174 cv[20] = m35 * (mC[9] * m35 + mC[13] * m45) + m45 * (mC[13] * m35 + mC[14] * m45);
1179#ifndef GPUCA_ALIGPUCODE
1181template <
typename value_T>
1185 fmt::format(
" Cov: [{:+.3e}] [{:+.3e} {:+.3e}] [{:+.3e} {:+.3e} {:+.3e}] [{:+.3e} {:+.3e} {:+.3e} {:+.3e}] [{:+.3e} {:+.3e} {:+.3e} {:+.3e} {:+.3e}]",
1191template <
typename value_T>
1195 fmt::format(
" Cov: [{:x}] [{:x} {:x}] [{:x} {:x} {:x}] [{:x} {:x} {:x} {:x}] [{:x} {:x} {:x} {:x} {:x}]",
1196 reinterpret_cast<const unsigned int&
>(mC[
kSigY2]),
reinterpret_cast<const unsigned int&
>(mC[
kSigZY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigZ2]),
1197 reinterpret_cast<const unsigned int&
>(mC[
kSigSnpY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigSnpZ]),
reinterpret_cast<const unsigned int&
>(mC[
kSigSnp2]),
1198 reinterpret_cast<const unsigned int&
>(mC[
kSigTglY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigTglZ]),
reinterpret_cast<const unsigned int&
>(mC[
kSigTglSnp]),
1199 reinterpret_cast<const unsigned int&
>(mC[
kSigTgl2]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtZ]),
1200 reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtSnp]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtTgl]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2Pt2]));
1205template <
typename value_T>
1209#ifndef GPUCA_ALIGPUCODE
1210 printf(
"%s\n",
asString().c_str());
1211#elif !defined(GPUCA_GPUCODE_DEVICE) || (!defined(__OPENCL__) && defined(GPUCA_GPU_DEBUG_PRINT))
1214 " Cov: [%+.3e] [%+.3e %+.3e] [%+.3e %+.3e %+.3e] [%+.3e %+.3e %+.3e %+.3e] [%+.3e %+.3e %+.3e %+.3e %+.3e]\n",
1222template <
typename value_T>
1226#ifndef GPUCA_ALIGPUCODE
1227 printf(
"%s\n", asStringHexadecimal().c_str());
1228#elif !defined(GPUCA_GPUCODE_DEVICE) || (!defined(__OPENCL__) && defined(GPUCA_GPU_DEBUG_PRINT))
1231 " Cov: [%x] [%x %x] [%x %x %x] [%x %x %x %x] [%x %x %x %x %x]\n",
1232 gpu::CAMath::Float2UIntReint(mC[
kSigY2]),
1233 gpu::CAMath::Float2UIntReint(mC[
kSigZY]), gpu::CAMath::Float2UIntReint(mC[
kSigZ2]),
1234 gpu::CAMath::Float2UIntReint(mC[
kSigSnpY]), gpu::CAMath::Float2UIntReint(mC[
kSigSnpZ]), gpu::CAMath::Float2UIntReint(mC[
kSigSnp2]),
1235 gpu::CAMath::Float2UIntReint(mC[
kSigTglY]), gpu::CAMath::Float2UIntReint(mC[
kSigTglZ]), gpu::CAMath::Float2UIntReint(mC[
kSigTglSnp]), gpu::CAMath::Float2UIntReint(mC[
kSigTgl2]),
1242#if !defined(GPUCA_GPUCODE) || defined(GPUCA_GPUCODE_DEVICE)
1245#ifndef GPUCA_GPUCODE
std::string asString(TDataMember const &dm, char *pointer)
const GPUTPCGMMerger::trackCluster & b1
const GPUTPCGMMerger::trackCluster & a1
std::string asString() const
std::string asStringHexadecimal()
std::string asStringHexadecimal()
std::string asString() const
GLfloat GLfloat GLfloat alpha
GLboolean GLboolean GLboolean b
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
GLdouble GLdouble GLdouble z
typename trackParam_t::params_t params_t
typename trackParam_t::dim3_t dim3_t
D const SVectorGPU< T, D > & rhs
double * getX(double *xyDxy, int N)
double * getY(double *xyDxy, int N)
constexpr float kCTgl2max
constexpr int kCovMatSize
constexpr float kCSnp2max
constexpr int kLabCovMatSize
value_T gpu::gpustd::array< value_T, 7 > & vect
GPUd() value_T BetheBlochSolid(value_T bg
constexpr float kC1Pt2max
constexpr int MaxELossIter
constexpr float ELoss2EKinThreshInv
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"