58 static constexpr float kDeg2Rad = M_PI / 180.f;
59 CADEBUG(
static constexpr float kSectAngle = 2 * M_PI / 18.f);
66 prop.SetMaterialTPC();
67 prop.SetPolynomialField(&
param.polynomialField);
68 prop.SetMaxSinPhi(maxSinPhi);
69 prop.SetToyMCEventsFlag(
param.par.toyMCEventsFlag);
71 ShiftZ2(
clusters, clustersXYZ, merger, N);
73 if (
param.rec.tpc.mergerInterpolateErrors) {
74 for (int32_t
i = 0;
i < N;
i++) {
79 const int32_t nWays =
param.rec.tpc.nWays;
80 const int32_t maxN = N;
81 int32_t ihitStart = 0;
83 float lastUpdateX = -1.f;
84 uint8_t lastRow = 255;
85 uint8_t lastSector = 255;
86 uint8_t storeOuter = 0;
88 for (int32_t iWay = 0; iWay < nWays; iWay++) {
89 int32_t nMissed = 0, nMissed2 = 0;
90 float sumInvSqrtCharge = 0.f;
91 int32_t nAvgCharge = 0;
93 if (iWay && storeOuter != 255 &&
param.rec.tpc.nWaysOuter && outerParam) {
95 if (iWay == nWays - 1) {
96 StoreOuter(outerParam, prop, 0);
97 if (merger->OutputTracks()[iTrk].Looper()) {
100 }
else if (iWay == nWays - 2 && merger->OutputTracks()[iTrk].Looper()) {
105 int32_t resetT0 = initResetT0();
106 const bool refit = (nWays == 1 || iWay >= 1);
107 const float maxSinForUpdate = CAMath::Sin(70.f * kDeg2Rad);
110 prop.SetSeedingErrors(!(refit && attempt == 0));
111 prop.SetFitInProjections(
param.rec.fitInProjections == -1 ? (iWay != 0) :
param.rec.fitInProjections);
112 prop.SetPropagateBzOnly(
param.rec.fitPropagateBzOnly > iWay);
113 prop.SetMatLUT((
param.rec.useMatLUT && iWay == nWays - 1) ? merger->GetConstantMem()->calibObjects.matLUT :
nullptr);
114 prop.SetTrack(
this, iWay ? prop.GetAlpha() : Alpha);
116 CADEBUG(printf(
"Fitting track %d way %d (sector %d, alpha %f)\n", iTrk, iWay, CAMath::Float2IntRn(prop.GetAlpha() / kSectAngle) + (mP[1] < 0 ? 18 : 0), prop.GetAlpha()));
120 const bool inFlyDirection = iWay & 1;
121 uint8_t lastLeg =
clusters[ihitStart].leg;
122 const int32_t wayDirection = (iWay & 1) ? -1 : 1;
124 bool noFollowCircle =
false, noFollowCircle2 =
false;
125 int32_t goodRows = 0;
126 for (int32_t ihit = ihitStart; ihit >= 0 && ihit < maxN; ihit += wayDirection) {
127 const bool crossCE = lastSector != 255 && ((lastSector < 18) ^ (
clusters[ihit].sector < 18));
130 noFollowCircle2 =
true;
135 StoreOuter(outerParam, prop, 1);
143 CADEBUG(printf(
"\tSkipping hit, %d hits rejected, flag %X\n", nMissed, (int32_t)
clusters[ihit].
state));
150 const bool allowModification = refit && (iWay == 0 || (((nWays - iWay) & 1) ? (ihit >= CAMath::Min(maxN / 2, 30)) : (ihit <= CAMath::Max(maxN / 2, maxN - 30))));
152 int32_t ihitMergeFirst = ihit;
153 uint8_t clusterState =
clusters[ihit].state;
156 if (
param.par.earlyTpcTransform) {
157 const float zOffset = (
clusters[ihit].sector < 18) == (
clusters[0].sector < 18) ? mTZOffset : -mTZOffset;
158 xx = clustersXYZ[ihit].x;
159 yy = clustersXYZ[ihit].y;
160 zz = clustersXYZ[ihit].z - zOffset;
163 merger->GetConstantMem()->calibObjects.fastTransformHelper->Transform(
clusters[ihit].sector,
clusters[ihit].
row, cl.getPad(), cl.getTime(), xx, yy, zz, mTZOffset);
166 CADEBUG(printf(
"\tHit %3d/%3d Row %3d: Cluster Alpha %8.3f %3d, X %8.3f - Y %8.3f, Z %8.3f (Missed %d)\n", ihit, maxN, (int32_t)
clusters[ihit].
row, clAlpha, (int32_t)
clusters[ihit].sector, xx, yy, zz, nMissed));
170 if (MergeDoubleRowClusters(ihit, wayDirection,
clusters, clustersXYZ, merger, prop, xx, yy, zz, maxN, clAlpha, clusterState, allowModification) == -1) {
176 if (
param.rec.tpc.rejectIFCLowRadiusCluster) {
177 const float r2 = xx * xx + yy * yy;
178 const float rmax = (83.5f +
param.rec.tpc.sysClusErrorMinDist);
179 if (r2 < rmax * rmax) {
184 const auto& cluster =
clusters[ihit];
186 bool changeDirection = (cluster.leg - lastLeg) & 1;
188 CADEBUG(
if (changeDirection) printf(
"\t\tChange direction\n"));
189 CADEBUG(printf(
"\tLeg %3d Sector %2d %4sTrack Alpha %8.3f %s, X %8.3f - Y %8.3f, Z %8.3f - QPt %7.2f (%7.2f), SP %5.2f (%5.2f) %28s --- Cov sY %8.3f sZ %8.3f sSP %8.3f sPt %8.3f - YPt %8.3f\n", (int32_t)cluster.leg, (int32_t)cluster.sector,
"", prop.GetAlpha(), (CAMath::Abs(prop.GetAlpha() - clAlpha) < 0.01 ?
" " :
" R!"), mX, mP[0], mP[1], mP[4], prop.GetQPt0(), mP[2], prop.GetSinPhi0(),
"", sqrtf(mC[0]), sqrtf(mC[2]), sqrtf(mC[5]), sqrtf(mC[14]), mC[10]));
191 if (allowModification && changeDirection && !noFollowCircle && !noFollowCircle2) {
192 bool tryFollow = lastRow != 255;
195 const float backupAlpha = prop.GetAlpha();
196 if (FollowCircle<0>(merger, prop, lastSector, lastRow, iTrk, clAlpha, xx, yy, cluster.
sector, cluster.
row, inFlyDirection)) {
197 CADEBUG(printf(
"Error during follow circle, resetting track!\n"));
199 prop.SetTrack(
this, backupAlpha);
200 noFollowCircle =
true;
205 MirrorTo(prop, yy, zz, inFlyDirection,
param, cluster.row, clusterState,
false, cluster.
sector);
207 lastLeg = cluster.leg;
208 lastSector = cluster.sector;
211 resetT0 = initResetT0();
214 CADEBUG(printf(
"\t%21sMirror Alpha %8.3f , X %8.3f - Y %8.3f, Z %8.3f - QPt %7.2f (%7.2f), SP %5.2f (%5.2f) %28s --- Cov sY %8.3f sZ %8.3f sSP %8.3f sPt %8.3f - YPt %8.3f\n",
"", prop.GetAlpha(), mX, mP[0], mP[1], mP[4], prop.GetQPt0(), mP[2], prop.GetSinPhi0(),
"", sqrtf(mC[0]), sqrtf(mC[2]), sqrtf(mC[5]), sqrtf(mC[14]), mC[10]));
218 }
else if (allowModification && lastRow != 255 && CAMath::Abs(cluster.row - lastRow) > 1) {
219 bool dodEdx =
param.par.dodEdx &&
param.dodEdxDownscaled &&
param.rec.tpc.adddEdxSubThresholdClusters && iWay == nWays - 1 && CAMath::Abs(cluster.row - lastRow) == 2 && cluster.leg ==
clusters[maxN - 1].leg;
220 dodEdx = AttachClustersPropagate(merger, cluster.sector, lastRow, cluster.row, iTrk, cluster.leg ==
clusters[maxN - 1].leg, prop, inFlyDirection,
GPUCA_MAX_SIN_PHI, dodEdx);
222 dEdx.fillSubThreshold(lastRow - wayDirection);
223 dEdxAlt.fillSubThreshold(lastRow - wayDirection);
227 int32_t err = prop.PropagateToXAlpha(xx, clAlpha, inFlyDirection);
229 CADEBUG(
if (!CheckCov()){printf(
"INVALID COV AFTER PROPAGATE!!!\n");});
234 if (prop.PropagateToXAlpha(xx, prop.GetAlpha(), inFlyDirection) == 0) {
235 err = prop.PropagateToXAlpha(xx, clAlpha, inFlyDirection);
238 if (lastRow == 255 || CAMath::Abs((int32_t)lastRow - (int32_t)cluster.row) > 5 || lastSector != cluster.sector || (
param.rec.tpc.trackFitRejectMode < 0 && -nMissed <=
param.rec.tpc.trackFitRejectMode)) {
244 lastRow = cluster.row;
245 lastSector = cluster.
sector;
248 CADEBUG(printf(
"\t%21sPropaga Alpha %8.3f , X %8.3f - Y %8.3f, Z %8.3f - QPt %7.2f (%7.2f), SP %5.2f (%5.2f) --- Res %8.3f %8.3f --- Cov sY %8.3f sZ %8.3f sSP %8.3f sPt %8.3f - YPt %8.3f - Err %d",
"", prop.GetAlpha(), mX, mP[0], mP[1], mP[4], prop.GetQPt0(), mP[2], prop.GetSinPhi0(), mP[0] - yy, mP[1] - zz, sqrtf(mC[0]), sqrtf(mC[2]), sqrtf(mC[5]), sqrtf(mC[14]), mC[10], err));
252 if (
param.rec.tpc.addErrorsCECrossing) {
253 if (
param.rec.tpc.addErrorsCECrossing >= 2) {
254 AddCovDiagErrorsWithCorrelations(
param.rec.tpc.errorsCECrossing);
256 AddCovDiagErrors(
param.rec.tpc.errorsCECrossing);
258 }
else if (mC[2] < 0.5f) {
263 if (err == 0 && changeDirection) {
264 const float mirrordY = prop.GetMirroredYTrack();
265 CADEBUG(printf(
" -- MirroredY: %f --> %f", mP[0], mirrordY));
266 if (CAMath::Abs(yy - mP[0]) > CAMath::Abs(yy - mirrordY)) {
267 CADEBUG(printf(
" - Mirroring!!!"));
268 if (allowModification) {
269 AttachClustersMirror<0>(merger, cluster.sector, cluster.row, iTrk, yy, prop);
271 MirrorTo(prop, yy, zz, inFlyDirection,
param, cluster.row, clusterState,
true, cluster.
sector);
272 noFollowCircle =
false;
275 lastLeg = cluster.leg;
278 resetT0 = initResetT0();
281 CADEBUG(printf(
"\t%21sMirror Alpha %8.3f , X %8.3f - Y %8.3f, Z %8.3f - QPt %7.2f (%7.2f), SP %5.2f (%5.2f) %28s --- Cov sY %8.3f sZ %8.3f sSP %8.3f sPt %8.3f - YPt %8.3f\n",
"", prop.GetAlpha(), mX, mP[0], mP[1], mP[4], prop.GetQPt0(), mP[2], prop.GetSinPhi0(),
"", sqrtf(mC[0]), sqrtf(mC[2]), sqrtf(mC[5]), sqrtf(mC[14]), mC[10]));
287 float uncorrectedY = -1e6f;
288 if (allowModification) {
289 uncorrectedY = AttachClusters(merger, cluster.sector, cluster.row, iTrk, cluster.leg ==
clusters[maxN - 1].leg, prop);
292 const int32_t err2 = mNDF > 0 && CAMath::Abs(prop.GetSinPhi0()) >= maxSinForUpdate;
294 if (mC[0] >
param.rec.tpc.trackFitCovLimit || mC[2] >
param.rec.tpc.trackFitCovLimit) {
300 CADEBUG(printf(
" --- break (%d, %d)\n", err, err2));
306 float threshold = 3.f + (lastUpdateX >= 0 ? (CAMath::Abs(mX - lastUpdateX) / 2) : 0.f);
307 if (mNDF > 5 && (CAMath::Abs(yy - mP[0]) > threshold || CAMath::Abs(zz - mP[1]) > threshold)) {
310 int8_t rejectChi2 = attempt ? 0 : ((
param.rec.tpc.mergerInterpolateErrors && CAMath::Abs(ihit - ihitMergeFirst) <= 1) ? (refit ? (
GPUTPCGMPropagator::rejectInterFill + ((nWays - iWay) & 1)) : 0) : (allowModification && goodRows > 5));
311#if EXTRACT_RESIDUALS == 1
313 const float Iz0 = interpolation.
hit[ihit].
posY - mP[0];
314 const float Iz1 = interpolation.
hit[ihit].
posZ - mP[1];
315 float Iw0 = mC[2] + (float)interpolation.
hit[ihit].
errorZ;
316 float Iw2 = mC[0] + (float)interpolation.
hit[ihit].
errorY;
317 float Idet1 = 1.f / CAMath::Max(1e-10f, Iw0 * Iw2 - mC[1] * mC[1]);
318 const float Ik00 = (mC[0] * Iw0 + mC[1] * mC[1]) * Idet1;
319 const float Ik01 = (mC[0] * mC[1] + mC[1] * Iw2) * Idet1;
320 const float Ik10 = (mC[1] * Iw0 + mC[2] * mC[1]) * Idet1;
321 const float Ik11 = (mC[1] * mC[1] + mC[2] * Iw2) * Idet1;
322 const float ImP0 = mP[0] + Ik00 * Iz0 + Ik01 * Iz1;
323 const float ImP1 = mP[1] + Ik10 * Iz0 + Ik11 * Iz1;
324 const float ImC0 = mC[0] - Ik00 * mC[0] + Ik01 * mC[1];
325 const float ImC2 = mC[2] - Ik10 * mC[1] + Ik11 * mC[2];
326 auto& tup =
GPUROOTDump<TNtuple>::get(
"clusterres",
"row:clX:clY:clZ:angle:trkX:trkY:trkZ:trkSinPhi:trkDzDs:trkQPt:trkSigmaY2:trkSigmaZ2trkSigmaQPt2");
327 tup.Fill((
float)cluster.row, xx, yy, zz, clAlpha, mX, ImP0, ImP1, mP[2], mP[3], mP[4], ImC0, ImC2, mC[14]);
331 if (
param.rec.tpc.rejectEdgeClustersInTrackFit && uncorrectedY > -1e6f &&
param.rejectEdgeClusterByY(uncorrectedY, cluster.row, CAMath::Sqrt(mC[0]))) {
334 const float time = merger->GetConstantMem()->ioPtrs.clustersNative ? merger->GetConstantMem()->ioPtrs.clustersNative->clustersLinear[cluster.num].getTime() : -1.f;
335 const float invSqrtCharge = merger->GetConstantMem()->ioPtrs.clustersNative ? CAMath::InvSqrt(merger->GetConstantMem()->ioPtrs.clustersNative->clustersLinear[cluster.num].qMax) : 0.f;
336 const float invCharge = merger->GetConstantMem()->ioPtrs.clustersNative ? (1.f / merger->GetConstantMem()->ioPtrs.clustersNative->clustersLinear[cluster.num].qMax) : 0.f;
337 float invAvgCharge = (sumInvSqrtCharge += invSqrtCharge) / ++nAvgCharge;
338 invAvgCharge *= invAvgCharge;
339 retVal = prop.Update(yy, zz, cluster.row,
param, clusterState, rejectChi2, &interpolation.
hit[ihit], refit, cluster.sector,
time, invAvgCharge, invCharge
GPUCA_DEBUG_STREAMER_CHECK(, &debugVals));
342 merger->DebugStreamerUpdate(iTrk, ihit, xx, yy, zz, cluster, merger->GetConstantMem()->ioPtrs.clustersNative->clustersLinear[cluster.num], *
this, prop, interpolation.
hit[ihit], rejectChi2, refit,
retVal, sumInvSqrtCharge / nAvgCharge * sumInvSqrtCharge / nAvgCharge, yy, zz, clusterState, debugVals.retVal, debugVals.err2Y, debugVals.err2Z);
346 CADEBUG(
if (!CheckCov()) GPUError(
"INVALID COV AFTER UPDATE!!!"));
347 CADEBUG(printf(
"\t%21sFit Alpha %8.3f , X %8.3f - Y %8.3f, Z %8.3f - QPt %7.2f (%7.2f), SP %5.2f (%5.2f), DzDs %5.2f %16s --- Cov sY %8.3f sZ %8.3f sSP %8.3f sPt %8.3f - YPt %8.3f - Err %d\n",
"", prop.GetAlpha(), mX, mP[0], mP[1], mP[4], prop.GetQPt0(), mP[2], prop.GetSinPhi0(), mP[3],
"", sqrtf(mC[0]), sqrtf(mC[2]), sqrtf(mC[5]), sqrtf(mC[14]), mC[10],
retVal));
353 if (storeOuter == 1 && cluster.leg ==
clusters[maxN - 1].leg) {
354 StoreOuter(outerParam, prop, 2);
357 noFollowCircle2 =
false;
360 nMissed = nMissed2 = 0;
364 float dy = mP[0] - prop.Model().
Y();
365 float dz = mP[1] - prop.Model().
Z();
366 if (CAMath::Abs(mP[4]) *
param.qptB5Scaler > 10 && --resetT0 <= 0 && CAMath::Abs(mP[2]) < 0.15f && dy * dy + dz * dz > 1) {
367 CADEBUG(printf(
"Reinit linearization\n"));
368 prop.SetTrack(
this, prop.GetAlpha());
370 if (
param.par.dodEdx &&
param.dodEdxDownscaled && iWay == nWays - 1 && cluster.leg ==
clusters[maxN - 1].leg) {
371 bool acc = (clusterState &
param.rec.tpc.dEdxClusterRejectionFlagMask) == 0, accAlt = (clusterState &
param.rec.tpc.dEdxClusterRejectionFlagMaskAlt) == 0;
373 float qtot = 0, qmax = 0, pad = 0, relTime = 0;
374 const int32_t clusterCount = (ihit - ihitMergeFirst) * wayDirection + 1;
375 for (int32_t iTmp = ihitMergeFirst; iTmp != ihit + wayDirection; iTmp += wayDirection) {
376 if (merger->GetConstantMem()->ioPtrs.clustersNative ==
nullptr) {
377 qtot += clustersXYZ[ihit].amp;
379 const ClusterNative& cl = merger->GetConstantMem()->ioPtrs.clustersNative->clustersLinear[cluster.num];
381 qmax = CAMath::Max<float>(qmax, cl.
qMax);
383 relTime += cl.getTime();
386 qtot /= clusterCount;
388 relTime /= clusterCount;
389 relTime = relTime - CAMath::Round(relTime);
391 dEdx.fillCluster(qtot, qmax, cluster.row, cluster.sector, mP[2], mP[3], merger->GetConstantMem()->calibObjects, zz, pad, relTime);
394 dEdxAlt.fillCluster(qtot, qmax, cluster.row, cluster.sector, mP[2], mP[3], merger->GetConstantMem()->calibObjects, zz, pad, relTime);
399 if (allowModification) {
401 }
else if (iWay == nWays - 1) {
410 if (((nWays - iWay) & 1) && (
clusters[0].sector < 18) == (
clusters[maxN - 1].sector < 18)) {
411 ShiftZ2(
clusters, clustersXYZ, merger, maxN);
417 o2::utils::DebugStreamer::instance()->getStreamer(
"debug_accept_track",
"UPDATE") << o2::utils::DebugStreamer::instance()->getUniqueTreeName(
"debug_accept_track").data() <<
"iTrk=" << iTrk <<
"outerParam=" << *outerParam <<
"track=" <<
this <<
"ihitStart=" << ihitStart <<
"\n";
423 if (
param.rec.tpc.minNClustersFinalTrack != -1 && N + NTolerated <
param.rec.tpc.minNClustersFinalTrack) {
429 if (
param.par.dodEdx &&
param.dodEdxDownscaled) {
430 dEdx.computedEdx(merger->OutputTracksdEdx()[iTrk],
param);
431 dEdxAlt.computedEdx(merger->OutputTracksdEdxAlt()[iTrk],
param);
433 Alpha = prop.GetAlpha();
434 MoveToReference(prop,
param, Alpha);
435 NormalizeAlpha(Alpha);
754GPUdic(0, 1) int32_t
GPUTPCGMTrackParam::FollowCircle(const
GPUTPCGMMerger*
GPUrestrict() Merger,
GPUTPCGMPropagator&
GPUrestrict() prop, int32_t sector, int32_t iRow, int32_t iTrack,
float toAlpha,
float toX,
float toY, int32_t toSector, int32_t toRow,
bool inFlyDirection,
bool phase2)
756 static constexpr float kSectAngle = 2 * M_PI / 18.f;
757 if (Merger->Param().rec.tpc.disableRefitAttachment & 4) {
760 if (Merger->Param().rec.tpc.looperInterpolationInExtraPass && phase2 ==
false) {
761 StoreAttachMirror(Merger, sector, iRow, iTrack, toAlpha, toY, toX, toSector, toRow, inFlyDirection, prop.GetAlpha());
766 float dAlpha = toAlpha - prop.GetAlpha();
768 if (CAMath::Abs(dAlpha) > 0.001f) {
769 right = CAMath::Abs(dAlpha) < CAMath::Pi() ? (dAlpha > 0) : (dAlpha < 0);
773 bool up = (mP[2] < 0) ^
right;
775 float lrFactor = mP[2] < 0 ? -1.f : 1.f;
777 CADEBUG(printf(
"CIRCLE Track %d: Sector %d Alpha %f X %f Y %f Z %f SinPhi %f DzDs %f - Next hit: Sector %d Alpha %f X %f Y %f - Right %d Up %d dAlpha %f lrFactor %f\n", iTrack, sector, prop.GetAlpha(), mX, mP[0], mP[1], mP[2], mP[3], toSector, toAlpha, toX, toY, (int32_t)
right, (int32_t)up, dAlpha, lrFactor));
780 AttachClustersPropagate(Merger, sector, iRow, targetRow, iTrack,
false, prop, inFlyDirection, 0.7f);
781 if (prop.RotateToAlpha(prop.GetAlpha() + (CAMath::Pi() / 2.f) * lrFactor)) {
784 CADEBUG(printf(
"\tRotated: X %f Y %f Z %f SinPhi %f (Alpha %f / %f)\n", mP[0], mX, mP[1], mP[2], prop.GetAlpha(), prop.GetAlpha() + CAMath::Pi() / 2.f));
785 while (sector != toSector || FollowCircleChk(lrFactor, toY, toX, up,
right)) {
786 while ((sector != toSector) ? (CAMath::Abs(mX) <= CAMath::Abs(mP[0]) * CAMath::Tan(kSectAngle / 2.f)) : FollowCircleChk(lrFactor, toY, toX, up,
right)) {
787 int32_t err = prop.PropagateToXAlpha(mX + 1.f, prop.GetAlpha(), inFlyDirection);
789 CADEBUG(printf(
"\t\tpropagation error (%d)\n", err));
790 prop.RotateToAlpha(prop.GetAlpha() - (CAMath::Pi() / 2.f) * lrFactor);
793 CADEBUG(printf(
"\tPropagated to y = %f: X %f Z %f SinPhi %f\n", mX, mP[0], mP[1], mP[2]));
795 float rowX = GPUTPCGeometry::Row2X(
j);
796 if (CAMath::Abs(rowX - (-mP[0] * lrFactor)) < 1.5f) {
797 CADEBUG(printf(
"\t\tAttempt row %d (Y %f Z %f)\n",
j, mX * lrFactor, mP[1]));
798 AttachClusters(Merger, sector,
j, iTrack,
false, mX * lrFactor, mP[1]);
802 if (sector != toSector) {
804 if (++sector >= sectorSide + 18) {
808 if (--sector < sectorSide) {
812 CADEBUG(printf(
"\tRotating to sector %d\n", sector));
813 if (prop.RotateToAlpha(
param.Alpha(sector) + (CAMath::Pi() / 2.f) * lrFactor)) {
814 CADEBUG(printf(
"\t\trotation error\n"));
815 prop.RotateToAlpha(prop.GetAlpha() - (CAMath::Pi() / 2.f) * lrFactor);
818 CADEBUG(printf(
"\tAfter Rotatin Alpha %f Position X %f Y %f Z %f SinPhi %f\n", prop.GetAlpha(), mP[0], mX, mP[1], mP[2]));
821 CADEBUG(printf(
"\tRotating back\n"));
822 for (int32_t
i = 0;
i < 2;
i++) {
823 if (prop.RotateToAlpha(prop.GetAlpha() + (CAMath::Pi() / 2.f) * lrFactor) == 0) {
827 CADEBUG(printf(
"Final rotation failed\n"));
830 CADEBUG(printf(
"\tresetting physical model\n"));
831 prop.SetTrack(
this, prop.GetAlpha());
834 CADEBUG(printf(
"\tMirrored position: Alpha %f X %f Y %f Z %f SinPhi %f DzDs %f\n", prop.GetAlpha(), mX, mP[0], mP[1], mP[2], mP[3]));
836 float dx = toX - GPUTPCGeometry::Row2X(toRow);
837 if (up ^ (toX > mX)) {
839 while (iRow <
GPUCA_ROW_COUNT - 2 && GPUTPCGeometry::Row2X(iRow + 1) + dx <= mX) {
843 while (iRow > 1 && GPUTPCGeometry::Row2X(iRow - 1) + dx >= mX) {
847 prop.PropagateToXAlpha(GPUTPCGeometry::Row2X(iRow) + dx, prop.GetAlpha(), inFlyDirection);
848 AttachClustersPropagate(Merger, sector, iRow, toRow, iTrack,
false, prop, inFlyDirection);
850 if (prop.PropagateToXAlpha(toX, prop.GetAlpha(), inFlyDirection)) {
853 CADEBUG(printf(
"Final position: Alpha %f X %f Y %f Z %f SinPhi %f DzDs %f\n", prop.GetAlpha(), mX, mP[0], mP[1], mP[2], mP[3]));