18#ifndef GPUCA_GPUCODE_DEVICE
22#ifndef GPUCA_ALIGPUCODE
23#include <fmt/printf.h>
30template <
typename value_T>
45template <
typename value_T>
52 if (gpu::CAMath::Abs(dx) < constants::math::Almost0) {
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);
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);
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;
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);
236 value_t crv = this->getCurvature(
b);
237 value_t tgfv = -(crv *
x - snp) / (crv *
y + csp);
238 sn = tgfv / gpu::CAMath::Sqrt(1.f + tgfv * tgfv);
239 cs = gpu::CAMath::Sqrt((1.f - sn) * (1.f + sn));
240 cs = (gpu::CAMath::Abs(tgfv) > constants::math::Almost0) ? sn / tgfv : constants::math::
Almost1;
242 x = xv * cs + yv * sn;
243 yv = -xv * sn + yv * cs;
247 alp += gpu::CAMath::ASin(sn);
248 if (!tmpT.rotate(alp) || !tmpT.propagateTo(xv,
b)) {
249#if !defined(GPUCA_ALIGPUCODE)
250 LOG(
debug) <<
"failed to propagate to alpha=" << alp <<
" X=" << xv << vtx <<
" | Track is: " << tmpT.asString();
260 o2::math_utils::detail::sincos(alp, sn, cs);
261 auto s2ylocvtx = vtx.getSigmaX2() * sn * sn + vtx.getSigmaY2() * cs * cs - 2. * vtx.getSigmaXY() * cs * sn;
262 dca->set(this->
getY() - yv, this->getZ() - zv, getSigmaY2() + s2ylocvtx, getSigmaZY(), getSigmaZ2() + vtx.getSigmaZ2());
268template <
typename value_T>
277template <
typename value_T>
289 constexpr value_t kSafe = 1e-5f;
290 value_t radPos2 = xyz[0] * xyz[0] + xyz[1] * xyz[1];
292 if (sectorAlpha || radPos2 < 1) {
293 alp = gpu::CAMath::ATan2(pxpypz[1], pxpypz[0]);
295 alp = gpu::CAMath::ATan2(xyz[1], xyz[0]);
298 alp = math_utils::detail::angle2Alpha<value_t>(alp);
302 math_utils::detail::sincos(alp, sn, cs);
304 if (cs * pxpypz[0] + sn * pxpypz[1] < 0) {
305 LOG(
debug) <<
"alpha from phiPos() will invalidate this track parameters, overriding to alpha from phi()";
306 alp = gpu::CAMath::ATan2(pxpypz[1], pxpypz[0]);
308 alp = math_utils::detail::angle2Alpha<value_t>(alp);
310 math_utils::detail::sincos(alp, sn, cs);
313 if (gpu::CAMath::Abs(sn) < 2.f * kSafe) {
315 alp += alp < constants::math::PIHalf ? 2.f * kSafe : -2.f * kSafe;
317 alp += alp > -constants::math::PIHalf ? -2.f * kSafe : 2.f * kSafe;
319 math_utils::detail::sincos(alp, sn, cs);
320 }
else if (gpu::CAMath::Abs(cs) < 2.f * kSafe) {
322 alp += alp > constants::math::PIHalf ? 2.f * kSafe : -2.f * kSafe;
324 alp += alp > -constants::math::PIHalf ? 2.f * kSafe : -2.f * kSafe;
326 math_utils::detail::sincos(alp, sn, cs);
329 dim3_t ver{xyz[0], xyz[1], xyz[2]};
330 dim3_t mom{pxpypz[0], pxpypz[1], pxpypz[2]};
333 math_utils::detail::rotateZ<value_t>(ver, -alp);
334 math_utils::detail::rotateZ<value_t>(mom, -alp);
336 value_t pt = gpu::CAMath::Sqrt(mom[0] * mom[0] + mom[1] * mom[1]);
342 this->setSnp(mom[1] * ptI);
343 this->setTgl(mom[2] * ptI);
344 this->setAbsCharge(gpu::CAMath::Abs(
charge));
348 if (gpu::CAMath::Abs(1.f - this->getSnp()) < kSafe) {
349 this->setSnp(1.f - kSafe);
350 }
else if (gpu::CAMath::Abs(-1.f - this->getSnp()) < kSafe) {
351 this->setSnp(-1.f + kSafe);
356 value_t cv34 = gpu::CAMath::Sqrt(cv[3] * cv[3] + cv[4] * cv[4]);
359 value_t sgcheck =
r * sn + this->getSnp() * cs;
360 if (gpu::CAMath::Abs(sgcheck) > 1 - kSafe) {
362 sgcheck = sgcheck < 0 ? -1.f : 1.f;
363 }
else if (gpu::CAMath::Abs(sgcheck) < kSafe) {
364 sgcheck = cs < 0 ? -1.0f : 1.0f;
368 mC[
kSigY2] = cv[0] + cv[2];
369 mC[
kSigZY] = (-cv[3] * sn) < 0 ? -cv34 : cv34;
373 value_t tgl2 = this->getTgl() * this->getTgl();
376 mC[
kSigSnpZ] = -sgcheck * cv[8] *
r * ptI;
377 mC[
kSigSnp2] = gpu::CAMath::Abs(cv[9] *
r *
r * ptI2);
378 mC[
kSigTglY] = (cv[10] * this->getTgl() - sgcheck * cv[15]) * ptI /
r;
379 mC[
kSigTglZ] = (cv[17] - sgcheck * cv[12] * this->getTgl()) * ptI;
380 mC[
kSigTglSnp] = (-sgcheck * cv[18] + cv[13] * this->getTgl()) *
r * ptI2;
381 mC[
kSigTgl2] = gpu::CAMath::Abs(cv[20] - 2 * sgcheck * cv[19] * mC[4] + cv[14] * tgl2) * ptI2;
385 mC[
kSigQ2PtTgl] = (-sgcheck * cv[19] + cv[14] * this->getTgl()) *
r * ptI2 * ptI;
386 mC[
kSigQ2Pt2] = gpu::CAMath::Abs(cv[14] * ptI2 * ptI2);
387 }
else if (special == 2) {
388 mC[
kSigSnpY] = -cv[10] * ptI * cs / sn;
390 mC[
kSigSnp2] = gpu::CAMath::Abs(cv[14] * cs * cs * ptI2);
391 mC[
kSigTglY] = (sgcheck * cv[6] * this->getTgl() - cv[15]) * ptI / sn;
392 mC[
kSigTglZ] = (cv[17] - sgcheck * cv[8] * this->getTgl()) * ptI;
393 mC[
kSigTglSnp] = (cv[19] - sgcheck * cv[13] * this->getTgl()) * cs * ptI2;
394 mC[
kSigTgl2] = gpu::CAMath::Abs(cv[20] - 2 * sgcheck * cv[18] * this->getTgl() + cv[9] * tgl2) * ptI2;
398 mC[
kSigQ2PtTgl] = (-sgcheck * cv[18] + cv[9] * this->getTgl()) * ptI2 * ptI *
charge;
399 mC[
kSigQ2Pt2] = gpu::CAMath::Abs(cv[9] * ptI2 * ptI2);
402 double m23 = -pt * (sn + this->getSnp() * cs /
r), m43 = -pt * pt * (
r * cs - this->getSnp() * sn);
403 double m24 = pt * (cs - this->getSnp() * sn /
r), m44 = -pt * pt * (
r * sn + this->getSnp() * cs);
404 double m35 = pt, m45 = -pt * pt * this->getTgl();
412 double a1 = cv[13] - cv[9] * (m23 * m44 + m43 * m24) / m23 / m43;
413 double a2 = m23 * m24 - m23 * (m23 * m44 + m43 * m24) / m43;
414 double a3 = m43 * m44 - m43 * (m23 * m44 + m43 * m24) / m23;
415 double a4 = cv[14] + 2. * cv[9];
416 double a5 = m24 * m24 - 2. * m24 * m44 * m23 / m43;
417 double a6 = m44 * m44 - 2. * m24 * m44 * m43 / m23;
419 mC[
kSigSnpY] = (cv[10] * m43 - cv[6] * m44) / (m24 * m43 - m23 * m44) / m00;
422 mC[
kSigSnpZ] = (cv[12] * m43 - cv[8] * m44) / (m24 * m43 - m23 * m44);
425 mC[
kSigSnp2] = gpu::CAMath::Abs((a4 * a3 - a6 * a1) / (a5 * a3 - a6 * a2));
429 double b2 = m23 * m35;
430 double b3 = m43 * m35;
432 double b5 = m24 * m35;
433 double b6 = m44 * m35;
434 mC[
kSigTglSnp] = (b4 - b6 * b1 / b3) / (b5 - b6 * b2 / b3);
442template <
typename value_T>
453 if (gpu::CAMath::Abs(dx) < constants::math::Almost0) {
457 if (gpu::CAMath::Abs(dx) > 1e5 || gpu::CAMath::Abs(this->
getY()) > 1e5 || gpu::CAMath::Abs(this->getZ()) > 1e5) {
458 LOG(warning) <<
"Anomalous track, target X:" << xk;
462 value_t crv = (gpu::CAMath::Abs(
b[2]) < constants::math::Almost0) ? 0.f :
this->getCurvature(
b[2]);
463 if (gpu::CAMath::Abs(crv) < constants::math::Almost0) {
464 return propagateTo(xk, 0.);
468 if ((gpu::CAMath::Abs(
f1) > constants::math::Almost1) || (gpu::CAMath::Abs(
f2) > constants::math::Almost1)) {
471 value_t r1 = gpu::CAMath::Sqrt((1.f -
f1) * (1.f +
f1));
472 if (gpu::CAMath::Abs(r1) < constants::math::Almost0) {
475 value_t r2 = gpu::CAMath::Sqrt((1.f -
f2) * (1.f +
f2));
476 if (gpu::CAMath::Abs(r2) < constants::math::Almost0) {
481 value_t step = (gpu::CAMath::Abs(x2r) < 0.05f) ? dx * gpu::CAMath::Abs(r2 +
f2 * dy2dx)
482 : 2.f * gpu::CAMath::ASin(0.5f * dx * gpu::CAMath::Sqrt(1.f + dy2dx * dy2dx) * crv) / crv;
483 step *= gpu::CAMath::Sqrt(1.f + this->getTgl() * this->getTgl());
486 std::array<value_t, 9> vecLab{0.f};
487 if (!this->getPosDirGlo(vecLab)) {
497 double rinv = 1. / r1;
498 double r3inv = rinv * rinv * rinv;
499 double f24 = dx *
b[2] * constants::math::B2C;
500 double f02 = dx * r3inv;
501 double f04 = 0.5 * f24 * f02;
502 double f12 = f02 * this->getTgl() *
f1;
503 double f14 = 0.5 * f24 * f12;
504 double f13 = dx * rinv;
507 double b00 = f02 * c20 + f04 * c40, b01 = f12 * c20 + f14 * c40 + f13 * c30;
508 double b02 = f24 * c40;
509 double b10 = f02 * c21 + f04 * c41, b11 = f12 * c21 + f14 * c41 + f13 * c31;
510 double b12 = f24 * c41;
511 double b20 = f02 * c22 + f04 * c42, b21 = f12 * c22 + f14 * c42 + f13 * c32;
512 double b22 = f24 * c42;
513 double b40 = f02 * c42 + f04 * c44, b41 = f12 * c42 + f14 * c44 + f13 * c43;
514 double b42 = f24 * c44;
515 double b30 = f02 * c32 + f04 * c43, b31 = f12 * c32 + f14 * c43 + f13 * c33;
516 double b32 = f24 * c43;
519 double a00 = f02 * b20 + f04 * b40, a01 = f02 * b21 + f04 * b41, a02 = f02 * b22 + f04 * b42;
520 double a11 = f12 * b21 + f14 * b41 + f13 * b31, a12 = f12 * b22 + f14 * b42 + f13 * b32;
521 double a22 = f24 * b42;
524 c00 += b00 + b00 + a00;
525 c10 += b10 + b01 + a01;
526 c20 += b20 + b02 + a02;
529 c11 += b11 + b11 + a11;
530 c21 += b21 + b12 + a12;
533 c22 += b22 + b22 + a22;
541 value_t bt = gpu::CAMath::Sqrt(bxy2);
542 value_t cosphi = 1.f, sinphi = 0.f;
543 if (bt > constants::math::Almost0) {
547 value_t bb = gpu::CAMath::Sqrt(bxy2 +
b[2] *
b[2]);
548 value_t costet = 1., sintet = 0.;
549 if (
bb > constants::math::Almost0) {
553 std::array<value_t, 7>
vect{costet * cosphi * vecLab[0] + costet * sinphi * vecLab[1] - sintet * vecLab[2],
554 -sinphi * vecLab[0] + cosphi * vecLab[1],
555 sintet * cosphi * vecLab[0] + sintet * sinphi * vecLab[1] + costet * vecLab[2],
556 costet * cosphi * vecLab[3] + costet * sinphi * vecLab[4] - sintet * vecLab[5],
557 -sinphi * vecLab[3] + cosphi * vecLab[4],
558 sintet * cosphi * vecLab[3] + sintet * sinphi * vecLab[4] + costet * vecLab[5],
566 vecLab[0] = cosphi * costet *
vect[0] - sinphi *
vect[1] + cosphi * sintet *
vect[2];
567 vecLab[1] = sinphi * costet *
vect[0] + cosphi *
vect[1] + sinphi * sintet *
vect[2];
568 vecLab[2] = -sintet *
vect[0] + costet *
vect[2];
570 vecLab[3] = cosphi * costet *
vect[3] - sinphi *
vect[4] + cosphi * sintet *
vect[5];
571 vecLab[4] = sinphi * costet *
vect[3] + cosphi *
vect[4] + sinphi * sintet *
vect[5];
572 vecLab[5] = -sintet *
vect[3] + costet *
vect[5];
575 value_t sinalp = -vecLab[7], cosalp = vecLab[8];
576 value_t t = cosalp * vecLab[0] - sinalp * vecLab[1];
577 vecLab[1] = sinalp * vecLab[0] + cosalp * vecLab[1];
579 t = cosalp * vecLab[3] - sinalp * vecLab[4];
580 vecLab[4] = sinalp * vecLab[3] + cosalp * vecLab[4];
584 value_t x = vecLab[0],
y = vecLab[1],
z = vecLab[2];
585 if (gpu::CAMath::Abs(dx) > constants::math::Almost0) {
586 if (gpu::CAMath::Abs(vecLab[3]) < constants::math::Almost0) {
591 y += vecLab[4] / vecLab[3] * dx;
592 z += vecLab[5] / vecLab[3] * dx;
596 t = 1.f / gpu::CAMath::Sqrt(vecLab[3] * vecLab[3] + vecLab[4] * vecLab[4]);
600 this->setSnp(vecLab[4] * t);
601 this->setTgl(vecLab[5] * t);
602 this->setQ2Pt(q * t / vecLab[6]);
608template <
typename value_T>
612 constexpr float MaxCorr = 0.99;
614 for (
int j =
i;
j--;) {
615 auto sig2 = mC[DiagMap[
i]] * mC[DiagMap[
j]];
616 auto& cov = mC[CovarMap[
i][
j]];
617 if (cov * cov >= MaxCorr * sig2) {
618 cov = gpu::CAMath::Sqrt(sig2) * (cov > 0. ? MaxCorr : -MaxCorr);
625template <
typename value_T>
680template <
typename value_T>
685 if (s2 > constants::math::Almost0) {
686 d0 = getSigmaY2() * s2;
687 d1 = getSigmaZ2() * s2;
688 d2 = getSigmaSnp2() * s2;
689 d3 = getSigmaTgl2() * s2;
690 d4 = getSigma1Pt2() * s2;
718template <
typename value_T>
722 auto sdd =
static_cast<double>(getSigmaY2()) +
static_cast<double>(cov[0]);
723 auto sdz =
static_cast<double>(getSigmaZY()) +
static_cast<double>(cov[1]);
724 auto szz =
static_cast<double>(getSigmaZ2()) +
static_cast<double>(cov[2]);
725 auto det = sdd * szz - sdz * sdz;
727 if (gpu::CAMath::Abs(det) < constants::math::Almost0) {
728 return constants::math::VeryBig;
733 auto chi2 = (d * (szz * d - sdz *
z) +
z * (sdd *
z - d * sdz)) / det;
735#ifndef GPUCA_ALIGPUCODE
736 LOGP(warning,
"Negative chi2={}, Cluster: {} {} {} Dy:{} Dz:{} | sdd:{} sdz:{} szz:{} det:{}", chi2, cov[0], cov[1], cov[2], d,
z, sdd, sdz, szz, det);
737 LOGP(warning,
"Track: {}",
asString());
744template <
typename value_T>
748 auto sdd =
static_cast<double>(getSigmaY2()) +
static_cast<double>(cov[0]);
749 auto sdz =
static_cast<double>(getSigmaZY()) +
static_cast<double>(cov[1]);
750 auto szz =
static_cast<double>(getSigmaZ2()) +
static_cast<double>(cov[2]);
751 auto det = sdd * szz - sdz * sdz;
753 if (gpu::CAMath::Abs(det) < constants::math::Almost0) {
754 return constants::math::VeryBig;
760 return (d * (szz * d - sdz *
z) +
z * (sdd *
z - d * sdz)) / det;
764template <
typename value_T>
768 return getPredictedChi2(rhs, cov);
772template <
typename value_T>
776 cov(
kY,
kY) =
static_cast<double>(getSigmaY2()) +
static_cast<double>(
rhs.getSigmaY2());
777 cov(
kZ,
kY) =
static_cast<double>(getSigmaZY()) +
static_cast<double>(
rhs.getSigmaZY());
778 cov(
kZ,
kZ) =
static_cast<double>(getSigmaZ2()) +
static_cast<double>(
rhs.getSigmaZ2());
779 cov(
kSnp,
kY) =
static_cast<double>(getSigmaSnpY()) +
static_cast<double>(
rhs.getSigmaSnpY());
780 cov(
kSnp,
kZ) =
static_cast<double>(getSigmaSnpZ()) +
static_cast<double>(
rhs.getSigmaSnpZ());
781 cov(
kSnp,
kSnp) =
static_cast<double>(getSigmaSnp2()) +
static_cast<double>(
rhs.getSigmaSnp2());
782 cov(
kTgl,
kY) =
static_cast<double>(getSigmaTglY()) +
static_cast<double>(
rhs.getSigmaTglY());
783 cov(
kTgl,
kZ) =
static_cast<double>(getSigmaTglZ()) +
static_cast<double>(
rhs.getSigmaTglZ());
784 cov(
kTgl,
kSnp) =
static_cast<double>(getSigmaTglSnp()) +
static_cast<double>(
rhs.getSigmaTglSnp());
785 cov(
kTgl,
kTgl) =
static_cast<double>(getSigmaTgl2()) +
static_cast<double>(
rhs.getSigmaTgl2());
786 cov(
kQ2Pt,
kY) =
static_cast<double>(getSigma1PtY()) +
static_cast<double>(
rhs.getSigma1PtY());
787 cov(
kQ2Pt,
kZ) =
static_cast<double>(getSigma1PtZ()) +
static_cast<double>(
rhs.getSigma1PtZ());
788 cov(
kQ2Pt,
kSnp) =
static_cast<double>(getSigma1PtSnp()) +
static_cast<double>(
rhs.getSigma1PtSnp());
789 cov(
kQ2Pt,
kTgl) =
static_cast<double>(getSigma1PtTgl()) +
static_cast<double>(
rhs.getSigma1PtTgl());
790 cov(
kQ2Pt,
kQ2Pt) =
static_cast<double>(getSigma1Pt2()) +
static_cast<double>(
rhs.getSigma1Pt2());
794template <
typename value_T>
801 LOG(error) <<
"The reference Alpha of the tracks differ: " << this->getAlpha() <<
" : " <<
rhs.getAlpha();
805 LOG(error) <<
"The reference X of the tracks differ: " << this->
getX() <<
" : " <<
rhs.getX();
808 buildCombinedCovMatrix(rhs, covToSet);
809 if (!covToSet.Invert()) {
810 LOG(warning) <<
"Cov.matrix inversion failed: " << covToSet;
813 double chi2diag = 0., chi2ndiag = 0., diff[
kNParams];
815 diff[
i] = this->getParam(
i) -
rhs.getParam(
i);
816 chi2diag += diff[
i] * diff[
i] * covToSet(
i,
i);
819 for (
int j =
i;
j--;) {
820 chi2ndiag += diff[
i] * diff[
j] * covToSet(
i,
j);
823 return chi2diag + 2. * chi2ndiag;
827template <
typename value_T>
834 LOG(error) <<
"The reference Alpha of the tracks differ: " << this->getAlpha() <<
" : " <<
rhs.getAlpha();
838 LOG(error) <<
"The reference X of the tracks differ: " << this->
getX() <<
" : " <<
rhs.getX();
844 matC0(
kY,
kY) = getSigmaY2();
845 matC0(
kZ,
kY) = getSigmaZY();
846 matC0(
kZ,
kZ) = getSigmaZ2();
847 matC0(
kSnp,
kY) = getSigmaSnpY();
848 matC0(
kSnp,
kZ) = getSigmaSnpZ();
850 matC0(
kTgl,
kY) = getSigmaTglY();
851 matC0(
kTgl,
kZ) = getSigmaTglZ();
852 matC0(
kTgl,
kSnp) = getSigmaTglSnp();
854 matC0(
kQ2Pt,
kY) = getSigma1PtY();
855 matC0(
kQ2Pt,
kZ) = getSigma1PtZ();
859 MatrixD5 matK = matC0 * covInv;
865 diff[
i] =
rhs.getParam(
i) - this->getParam(
i);
869 this->updateParam(matK(
i,
j) * diff[
j],
i);
895template <
typename value_T>
900 buildCombinedCovMatrix(rhs, covI);
901 if (!covI.Invert()) {
902 LOG(warning) <<
"Cov.matrix inversion failed: " << covI;
905 return update(rhs, covI);
909template <
typename value_T>
921 double r00 =
static_cast<double>(cov[0]) +
static_cast<double>(cm00);
922 double r01 =
static_cast<double>(cov[1]) +
static_cast<double>(cm10);
923 double r11 =
static_cast<double>(cov[2]) +
static_cast<double>(cm11);
924 double det = r00 * r11 - r01 * r01;
926 if (gpu::CAMath::Abs(det) < constants::math::Almost0) {
929 double detI = 1. / det;
935 double k00 = cm00 * r00 + cm10 * r01, k01 = cm00 * r01 + cm10 * r11;
936 double k10 = cm10 * r00 + cm11 * r01, k11 = cm10 * r01 + cm11 * r11;
937 double k20 = cm20 * r00 + cm21 * r01, k21 = cm20 * r01 + cm21 * r11;
938 double k30 = cm30 * r00 + cm31 * r01, k31 = cm30 * r01 + cm31 * r11;
939 double k40 = cm40 * r00 + cm41 * r01, k41 = cm40 * r01 + cm41 * r11;
942 value_t dsnp = k20 * dy + k21 * dz;
943 if (gpu::CAMath::Abs(this->getSnp() + dsnp) > constants::math::Almost1) {
949 this->updateParams(dP);
951 double c01 = cm10, c02 = cm20, c03 = cm30, c04 = cm40;
952 double c12 = cm21, c13 = cm31, c14 = cm41;
954 cm00 -= k00 * cm00 + k01 * cm10;
955 cm10 -= k00 * c01 + k01 * cm11;
956 cm20 -= k00 * c02 + k01 * c12;
957 cm30 -= k00 * c03 + k01 * c13;
958 cm40 -= k00 * c04 + k01 * c14;
960 cm11 -= k10 * c01 + k11 * cm11;
961 cm21 -= k10 * c02 + k11 * c12;
962 cm31 -= k10 * c03 + k11 * c13;
963 cm41 -= k10 * c04 + k11 * c14;
965 cm22 -= k20 * c02 + k21 * c12;
966 cm32 -= k20 * c03 + k21 * c13;
967 cm42 -= k20 * c04 + k21 * c14;
969 cm33 -= k30 * c03 + k31 * c13;
970 cm43 -= k30 * c04 + k31 * c14;
972 cm44 -= k40 * c04 + k41 * c14;
980template <
typename value_T>
985 auto vtLoc = this->getVertexInTrackFrame(vtx);
986 value_T chi2 = getPredictedChi2(vtLoc.yz, vtLoc.yzerr);
987 return chi2 < maxChi2 && update(vtLoc.yz, vtLoc.yzerr) ? chi2 : -chi2;
991template <
typename value_T>
1003 constexpr value_t kMSConst2 = 0.0136f * 0.0136f;
1004 constexpr value_t kMinP = 0.01f;
1006 value_t csp2 = (1.f - this->getSnp()) * (1.f + this->getSnp());
1007 value_t cst2I = (1.f + this->getTgl() * this->getTgl());
1013 auto m = this->getPID().getMass();
1014 int charge2 = this->getAbsCharge() * this->getAbsCharge();
1015 value_t p = this->getP(),
p0 =
p, p02 =
p *
p, e2 = p02 + this->getPID().getMass2(), massInv = 1. /
m,
bg =
p * massInv, dETot = 0.;
1016 value_t e = gpu::CAMath::Sqrt(e2), e0 = e;
1017 if (
m > 0 && xrho != 0.f) {
1019#ifdef _BB_NONCONST_CORR_
1033#ifdef _BB_NONCONST_CORR_
1034 dedxDer = this->getBetheBlochSolidDerivativeApprox(dedx1,
bg);
1041#ifdef _BB_NONCONST_CORR_
1042 if (dedxDer != 0.) {
1046 auto corrC = (gpu::CAMath::Exp(dedxDer) - 1.) / dedxDer;
1052 p = gpu::CAMath::Sqrt(e * e - this->getPID().getMass2());
1058 dedx = this->getdEdxBBOpt(
bg);
1059#ifdef _BB_NONCONST_CORR_
1060 dedxDer = this->getBetheBlochSolidDerivativeApprox(
dedx,
bg);
1064#ifdef _BB_NONCONST_CORR_
1084 value_t cC22(0.f), cC33(0.f), cC43(0.f), cC44(0.f);
1086 value_t beta2 = p02 / e2, theta2 = kMSConst2 / (
beta2 * p02) * gpu::CAMath::Abs(x2x0);
1087 value_t fp34 = this->getTgl();
1090 fp34 *= this->getCharge2Pt();
1092 if (theta2 > constants::math::PI * constants::math::PI) {
1095 value_t t2c2I = theta2 * cst2I;
1096 cC22 = t2c2I * csp2;
1097 cC33 = t2c2I * cst2I;
1098 cC43 = t2c2I * fp34;
1099 cC44 = theta2 * fp34 * fp34;
1108 constexpr value_t knst = 0.0007f;
1109 value_t sigmadE = knst * gpu::CAMath::Sqrt(gpu::CAMath::Abs(dETot)) * e0 / p02 * this->getCharge2Pt();
1110 cC44 += sigmadE * sigmadE;
1117 this->setQ2Pt(this->getQ2Pt() * p0 / p);
1125template <
typename value_T>
1140 if (gpu::CAMath::Abs(this->getQ2Pt()) <= constants::math::Almost0 || gpu::CAMath::Abs(this->getSnp()) > constants::math::Almost1) {
1141 for (
int i = 0;
i < 21;
i++) {
1147 auto pt = this->getPt();
1149 o2::math_utils::detail::sincos(this->getAlpha(), sn, cs);
1150 auto r = gpu::CAMath::Sqrt((1. - this->getSnp()) * (1. + this->getSnp()));
1151 auto m00 = -sn, m10 = cs;
1152 auto m23 = -pt * (sn + this->getSnp() * cs /
r), m43 = -pt * pt * (
r * cs - this->getSnp() * sn);
1153 auto m24 = pt * (cs - this->getSnp() * sn /
r), m44 = -pt * pt * (
r * sn + this->getSnp() * cs);
1154 auto m35 = pt, m45 = -pt * pt * this->getTgl();
1156 if (this->getSign() < 0) {
1162 cv[0] = mC[0] * m00 * m00;
1163 cv[1] = mC[0] * m00 * m10;
1164 cv[2] = mC[0] * m10 * m10;
1165 cv[3] = mC[1] * m00;
1166 cv[4] = mC[1] * m10;
1168 cv[6] = m00 * (mC[3] * m23 + mC[10] * m43);
1169 cv[7] = m10 * (mC[3] * m23 + mC[10] * m43);
1170 cv[8] = mC[4] * m23 + mC[11] * m43;
1171 cv[9] = m23 * (mC[5] * m23 + mC[12] * m43) + m43 * (mC[12] * m23 + mC[14] * m43);
1172 cv[10] = m00 * (mC[3] * m24 + mC[10] * m44);
1173 cv[11] = m10 * (mC[3] * m24 + mC[10] * m44);
1174 cv[12] = mC[4] * m24 + mC[11] * m44;
1175 cv[13] = m23 * (mC[5] * m24 + mC[12] * m44) + m43 * (mC[12] * m24 + mC[14] * m44);
1176 cv[14] = m24 * (mC[5] * m24 + mC[12] * m44) + m44 * (mC[12] * m24 + mC[14] * m44);
1177 cv[15] = m00 * (mC[6] * m35 + mC[10] * m45);
1178 cv[16] = m10 * (mC[6] * m35 + mC[10] * m45);
1179 cv[17] = mC[7] * m35 + mC[11] * m45;
1180 cv[18] = m23 * (mC[8] * m35 + mC[12] * m45) + m43 * (mC[13] * m35 + mC[14] * m45);
1181 cv[19] = m24 * (mC[8] * m35 + mC[12] * m45) + m44 * (mC[13] * m35 + mC[14] * m45);
1182 cv[20] = m35 * (mC[9] * m35 + mC[13] * m45) + m45 * (mC[13] * m35 + mC[14] * m45);
1187#ifndef GPUCA_ALIGPUCODE
1189template <
typename value_T>
1193 fmt::format(
" Cov: [{:+.3e}] [{:+.3e} {:+.3e}] [{:+.3e} {:+.3e} {:+.3e}] [{:+.3e} {:+.3e} {:+.3e} {:+.3e}] [{:+.3e} {:+.3e} {:+.3e} {:+.3e} {:+.3e}]",
1199template <
typename value_T>
1203 fmt::format(
" Cov: [{:x}] [{:x} {:x}] [{:x} {:x} {:x}] [{:x} {:x} {:x} {:x}] [{:x} {:x} {:x} {:x} {:x}]",
1204 reinterpret_cast<const unsigned int&
>(mC[
kSigY2]),
reinterpret_cast<const unsigned int&
>(mC[
kSigZY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigZ2]),
1205 reinterpret_cast<const unsigned int&
>(mC[
kSigSnpY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigSnpZ]),
reinterpret_cast<const unsigned int&
>(mC[
kSigSnp2]),
1206 reinterpret_cast<const unsigned int&
>(mC[
kSigTglY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigTglZ]),
reinterpret_cast<const unsigned int&
>(mC[
kSigTglSnp]),
1207 reinterpret_cast<const unsigned int&
>(mC[
kSigTgl2]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtY]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtZ]),
1208 reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtSnp]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2PtTgl]),
reinterpret_cast<const unsigned int&
>(mC[
kSigQ2Pt2]));
1213template <
typename value_T>
1217#ifndef GPUCA_ALIGPUCODE
1218 printf(
"%s\n",
asString().c_str());
1219#elif !defined(GPUCA_GPUCODE_DEVICE) || (!defined(__OPENCL__) && defined(GPUCA_GPU_DEBUG_PRINT))
1222 " Cov: [%+.3e] [%+.3e %+.3e] [%+.3e %+.3e %+.3e] [%+.3e %+.3e %+.3e %+.3e] [%+.3e %+.3e %+.3e %+.3e %+.3e]\n",
1230template <
typename value_T>
1234#ifndef GPUCA_ALIGPUCODE
1235 printf(
"%s\n", asStringHexadecimal().c_str());
1236#elif !defined(GPUCA_GPUCODE_DEVICE) || (!defined(__OPENCL__) && defined(GPUCA_GPU_DEBUG_PRINT))
1239 " Cov: [%x] [%x %x] [%x %x %x] [%x %x %x %x] [%x %x %x %x %x]\n",
1240 gpu::CAMath::Float2UIntReint(mC[
kSigY2]),
1241 gpu::CAMath::Float2UIntReint(mC[
kSigZY]), gpu::CAMath::Float2UIntReint(mC[
kSigZ2]),
1242 gpu::CAMath::Float2UIntReint(mC[
kSigSnpY]), gpu::CAMath::Float2UIntReint(mC[
kSigSnpZ]), gpu::CAMath::Float2UIntReint(mC[
kSigSnp2]),
1243 gpu::CAMath::Float2UIntReint(mC[
kSigTglY]), gpu::CAMath::Float2UIntReint(mC[
kSigTglZ]), gpu::CAMath::Float2UIntReint(mC[
kSigTglSnp]), gpu::CAMath::Float2UIntReint(mC[
kSigTgl2]),
1250#if !defined(GPUCA_GPUCODE) || defined(GPUCA_GPUCODE_DEVICE)
1253#ifndef GPUCA_GPUCODE
std::string asString(TDataMember const &dm, char *pointer)
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
typename trackParam_t::value_t value_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
constexpr float DefaultDCACov
value_T std::array< value_T, 7 > & vect
GPUd() value_T BetheBlochSolid(value_T bg
constexpr float kC1Pt2max
constexpr float DefaultDCA
constexpr int MaxELossIter
constexpr float ELoss2EKinThreshInv
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
Defining DataPointCompositeObject explicitly as copiable.
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"