LArSoft  v10_04_05
Liquid Argon Software toolkit - https://larsoft.org/
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
PFPUtils.cxx
Go to the documentation of this file.
2 
4 
5 #include "TDecompSVD.h"
6 #include "TMatrixD.h"
7 #include "TVectorD.h"
8 
9 #include <algorithm>
10 #include <array>
11 #include <bitset>
12 #include <cmath>
13 #include <iomanip>
14 #include <iostream>
15 #include <limits.h>
16 #include <limits>
17 #include <stdlib.h>
18 #include <utility>
19 #include <vector>
20 
36 
37 namespace tca {
38 
39  using namespace detail; // SortEntry, valsDecreasing(), valsIncreasing();
40 
42  void StitchPFPs()
43  {
44  // Stitch PFParticles in different TPCs. This does serious damage to PFPStruct and should
45  // only be called from TrajCluster module just before making PFParticles to put in the event
46  if (slices.size() < 2) return;
47  if (tcc.geom->NTPC() == 1) return;
48  if (tcc.pfpStitchCuts.size() < 2) return;
49  if (tcc.pfpStitchCuts[0] <= 0) return;
50 
51  bool prt = tcc.dbgStitch;
52 
53  if (prt) {
54  mf::LogVerbatim myprt("TC");
55  std::string fcnLabel = "SP";
56  myprt << fcnLabel << " cuts " << sqrt(tcc.pfpStitchCuts[0]) << " " << tcc.pfpStitchCuts[1]
57  << "\n";
58  bool printHeader = true;
59  for (size_t isl = 0; isl < slices.size(); ++isl) {
60  if (debug.Slice >= 0 && int(isl) != debug.Slice) continue;
61  auto& slc = slices[isl];
62  if (slc.pfps.empty()) continue;
63  for (auto& pfp : slc.pfps)
64  PrintP(myprt, pfp, printHeader);
65  } // slc
66  } // prt
67 
68  // lists of pfp UIDs to stitch
69  std::vector<std::vector<int>> stLists;
70  for (std::size_t sl1 = 0; sl1 < slices.size() - 1; ++sl1) {
71  auto& slc1 = slices[sl1];
72  for (std::size_t sl2 = sl1 + 1; sl2 < slices.size(); ++sl2) {
73  auto& slc2 = slices[sl2];
74  // look for PFParticles in the same recob::Slice
75  if (slc1.ID != slc2.ID) continue;
76  for (auto& p1 : slc1.pfps) {
77  if (p1.ID <= 0) continue;
78  // Can't stitch shower PFPs
79  if (p1.PDGCode == 1111) continue;
80  for (auto& p2 : slc2.pfps) {
81  if (p2.ID <= 0) continue;
82  // Can't stitch shower PFPs
83  if (p2.PDGCode == 1111) continue;
84  float maxSep2 = tcc.pfpStitchCuts[0];
85  float maxCth = tcc.pfpStitchCuts[1];
86  bool gotit = false;
87  for (unsigned short e1 = 0; e1 < 2; ++e1) {
88  auto pos1 = PosAtEnd(p1, e1);
89  // require the end to be close to a TPC boundary
90  if (InsideFV(slc1, p1, e1)) continue;
91  auto dir1 = DirAtEnd(p1, e1);
92  for (unsigned short e2 = 0; e2 < 2; ++e2) {
93  auto pos2 = PosAtEnd(p2, e2);
94  // require the end to be close to a TPC boundary
95  if (InsideFV(slc2, p2, e2)) continue;
96  auto dir2 = DirAtEnd(p2, e2);
97  float sep = PosSep2(pos1, pos2);
98  if (sep > maxSep2) continue;
99  float cth = std::abs(DotProd(dir1, dir2));
100  if (cth < maxCth) continue;
101  maxSep2 = sep;
102  maxCth = cth;
103  gotit = true;
104  } // e2
105  } // e1
106  if (!gotit) continue;
107  if (prt) {
108  mf::LogVerbatim myprt("TC");
109  myprt << "Stitch slice " << slc1.ID << " P" << p1.UID << " TPC " << p1.TPCID.TPC;
110  myprt << " and P" << p2.UID << " TPC " << p2.TPCID.TPC;
111  myprt << " sep " << sqrt(maxSep2) << " maxCth " << maxCth;
112  }
113  // see if either of these are in a list
114  bool added = false;
115  for (auto& pm : stLists) {
116  bool p1InList = (std::find(pm.begin(), pm.end(), p1.UID) != pm.end());
117  bool p2InList = (std::find(pm.begin(), pm.end(), p2.UID) != pm.end());
118  if (p1InList || p2InList) {
119  if (p1InList) pm.push_back(p2.UID);
120  if (p2InList) pm.push_back(p1.UID);
121  added = true;
122  }
123  } // pm
124  if (added) continue;
125  // start a new list
126  std::vector<int> tmp(2);
127  tmp[0] = p1.UID;
128  tmp[1] = p2.UID;
129  stLists.push_back(tmp);
130  break;
131  } // p2
132  } // p1
133  } // sl2
134  } // sl1
135  if (stLists.empty()) return;
136 
137  for (auto& stl : stLists) {
138  // Find the endpoints of the stitched pfp
139  float minZ = 1E6;
140  std::pair<unsigned short, unsigned short> minZIndx;
141  unsigned short minZEnd = 2;
142  for (auto puid : stl) {
143  auto slcIndex = GetSliceIndex("P", puid);
144  if (slcIndex.first == USHRT_MAX) continue;
145  auto& pfp = slices[slcIndex.first].pfps[slcIndex.second];
146  for (unsigned short end = 0; end < 2; ++end) {
147  auto pos = PosAtEnd(pfp, end);
148  if (pos[2] < minZ) {
149  minZ = pos[2];
150  minZIndx = slcIndex;
151  minZEnd = end;
152  }
153  } // end
154  } // puid
155  if (minZEnd > 1) continue;
156  // preserve the pfp with the min Z position
157  auto& pfp = slices[minZIndx.first].pfps[minZIndx.second];
158  if (prt) mf::LogVerbatim("TC") << "SP: P" << pfp.UID;
159  // add the Tjs in the other slices to it
160  for (auto puid : stl) {
161  if (puid == pfp.UID) continue;
162  auto sIndx = GetSliceIndex("P", puid);
163  if (sIndx.first == USHRT_MAX) continue;
164  auto& opfp = slices[sIndx.first].pfps[sIndx.second];
165  if (prt) mf::LogVerbatim("TC") << " +P" << opfp.UID;
166  pfp.TjUIDs.insert(pfp.TjUIDs.end(), opfp.TjUIDs.begin(), opfp.TjUIDs.end());
167  if (prt) mf::LogVerbatim();
168  // Check for parents and daughters
169  if (opfp.ParentUID > 0) {
170  auto pSlcIndx = GetSliceIndex("P", opfp.ParentUID);
171  if (pSlcIndx.first < slices.size()) {
172  auto& parpfp = slices[pSlcIndx.first].pfps[pSlcIndx.second];
173  std::replace(parpfp.DtrUIDs.begin(), parpfp.DtrUIDs.begin(), opfp.UID, pfp.UID);
174  } // valid pSlcIndx
175  } // has a parent
176  for (auto dtruid : opfp.DtrUIDs) {
177  auto dSlcIndx = GetSliceIndex("P", dtruid);
178  if (dSlcIndx.first < slices.size()) {
179  auto& dtrpfp = slices[dSlcIndx.first].pfps[dSlcIndx.second];
180  dtrpfp.ParentUID = pfp.UID;
181  } // valid dSlcIndx
182  } // dtruid
183  // declare it obsolete
184  opfp.ID = 0;
185  } // puid
186  } // stl
187 
188  } // StitchPFPs
189 
191  detinfo::DetectorPropertiesData const& detProp,
192  TCSlice& slc)
193  {
194  // Match Tjs in 3D and create PFParticles
195 
196  if (tcc.match3DCuts[0] <= 0) return;
197 
199 
200  // clear the TP -> P assn Tjs so that all are considered
201  for (auto& tj : slc.tjs) {
202  for (auto& tp : tj.Pts)
203  tp.InPFP = 0;
204  } // tj
205 
206  bool prt = (tcc.dbgPFP && tcc.dbgSlc);
207 
208  // Match these points in 3D
209  std::vector<MatchStruct> matVec;
210 
211  // iterate twice (at most), looking for 3-plane matches in long tjs in 3-plane TPCs on the
212  // first iteration, 3-plane matches in short tjs on the second iteration.
213  // and 2-plane matches + dead regions in 3-plane TPCs on the last iteration
214  slc.mallTraj.clear();
215 
216  unsigned short maxNit = 2;
217  if (slc.nPlanes == 2) maxNit = 1;
218  if (std::nearbyint(tcc.match3DCuts[2]) == 0) maxNit = 1;
219  // fill the mAllTraj vector with TPs if we aren't using SpacePoints
220  if (evt.sptHits.empty()) FillmAllTraj(detProp, slc);
221  for (unsigned short nit = 0; nit < maxNit; ++nit) {
222  matVec.clear();
223  if (slc.nPlanes == 3 && nit == 0) {
224  // look for match triplets
225  Match3Planes(slc, matVec);
226  }
227  else {
228  // look for match doublets requiring a dead region in the 3rd plane for 3-plane TPCs
229  Match2Planes(slc, matVec);
230  }
231  if (matVec.empty()) continue;
232  if (prt) {
233  mf::LogVerbatim myprt("TC");
234  myprt << "nit " << nit << " MVI Count Tjs\n";
235  for (unsigned int indx = 0; indx < matVec.size(); ++indx) {
236  auto& ms = matVec[indx];
237  myprt << std::setw(5) << indx << std::setw(6) << (int)ms.Count;
238  for (auto tid : ms.TjIDs)
239  myprt << " T" << tid;
240  // count the number of TPs in all Tjs
241  float tpCnt = 0;
242  for (auto tid : ms.TjIDs) {
243  auto& tj = slc.tjs[tid - 1];
244  tpCnt += NumPtsWithCharge(slc, tj, false);
245  } // tid
246  float frac = ms.Count / tpCnt;
247  myprt << " matFrac " << std::fixed << std::setprecision(3) << frac;
248  myprt << "\n";
249  } // indx
250  } // prt
251  MakePFParticles(clockData, detProp, slc, matVec, nit);
252  } // nit
253 
254  // a last debug print
255  if (tcc.dbgPFP && debug.MVI != UINT_MAX) {
256  for (auto& pfp : slc.pfps)
257  if (tcc.dbgPFP && pfp.MVI == debug.MVI)
258  PrintTP3Ds(clockData, detProp, "FPFP", slc, pfp, -1);
259  } // last debug print
260 
261  slc.mallTraj.resize(0);
262 
263  } // FindPFParticles
264 
267  detinfo::DetectorPropertiesData const& detProp,
268  TCSlice& slc,
269  std::vector<MatchStruct> matVec,
270  unsigned short matVec_Iter)
271  {
272  // Makes PFParticles using Tjs listed in matVec
273  if (matVec.empty()) return;
274 
275  bool prt = (tcc.dbgPFP && tcc.dbgSlc);
276 
277  // create a PFParticle for each valid match combination
278  for (std::size_t indx = 0; indx < matVec.size(); ++indx) {
279  // tone down the level of printing in ReSection
280  bool foundMVI = (tcc.dbgPFP && indx == debug.MVI && matVec_Iter == debug.MVI_Iter);
281  if (foundMVI) prt = true;
282  auto& ms = matVec[indx];
283  if (foundMVI) {
284  std::cout << "found MVI " << indx << " in MakePFParticles ms.Count = " << ms.Count << "\n";
285  }
286  // ignore dead matches
287  if (ms.Count == 0) continue;
288  // count the number of TPs that are available (not already 3D-matched) and used in a pfp
289  float npts = 0;
290  for (std::size_t itj = 0; itj < ms.TjIDs.size(); ++itj) {
291  auto& tj = slc.tjs[ms.TjIDs[itj] - 1];
292  for (unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt)
293  if (tj.Pts[ipt].InPFP == 0) ++npts;
294  } // tjID
295  // Create a vector of PFPs for this match so that we can split it later on if a kink is found
296  std::vector<PFPStruct> pfpVec(1);
297  pfpVec[0] = CreatePFP(slc);
298  // Define the starting set of tjs that were matched. TPs from other tjs may be added later
299  pfpVec[0].TjIDs = ms.TjIDs;
300  pfpVec[0].MVI = indx;
301  // fill the TP3D points using the 2D trajectory points for Tjs in TjIDs. All
302  // points are put in one section
303  if (!MakeTP3Ds(detProp, slc, pfpVec[0], foundMVI)) {
304  if (foundMVI) mf::LogVerbatim("TC") << " MakeTP3Ds failed. Too many points already used ";
305  continue;
306  }
307  // fit all the points to get the general direction
308  if (!FitSection(detProp, slc, pfpVec[0], 0)) continue;
309  if (pfpVec[0].SectionFits[0].ChiDOF > 500 && !pfpVec[0].Flags[kSmallAngle]) {
310  if (foundMVI)
311  mf::LogVerbatim("TC") << " crazy high ChiDOF P" << pfpVec[0].ID << " "
312  << pfpVec[0].SectionFits[0].ChiDOF << "\n";
313  Recover(detProp, slc, pfpVec[0], foundMVI);
314  continue;
315  }
316  // sort the points by the distance along the general direction vector
317  if (!SortSection(pfpVec[0], 0)) continue;
318  // define a junk pfp to be short with low MCSMom. These are likely to be shower-like
319  // pfps. A simple 3D line fit will be done. No attempt will be made to reconstruct it
320  // in sections or to look for kinks
321  npts = pfpVec[0].TP3Ds.size();
322  pfpVec[0].AlgMod[kJunk3D] = (npts < 20 && MCSMom(slc, pfpVec[0].TjIDs) < 50) || (npts < 10);
323  if (prt) {
324  auto& pfp = pfpVec[0];
325  mf::LogVerbatim myprt("TC");
326  myprt << " indx " << matVec_Iter << "/" << indx << " Count " << std::setw(5)
327  << (int)ms.Count;
328  myprt << " P" << pfpVec[0].ID;
329  myprt << " ->";
330  for (auto& tjid : pfp.TjIDs)
331  myprt << " T" << tjid;
332  myprt << " projInPlane";
333  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
334  CTP_t inCTP = EncodeCTP(pfp.TPCID.Cryostat, pfp.TPCID.TPC, plane);
335  auto tp = MakeBareTP(detProp, pfp.SectionFits[0].Pos, pfp.SectionFits[0].Dir, inCTP);
336  myprt << " " << std::setprecision(2) << tp.Delta;
337  } // plane
338  myprt << " maxTjLen " << (int)MaxTjLen(slc, pfp.TjIDs);
339  myprt << " MCSMom " << MCSMom(slc, pfp.TjIDs);
340  myprt << " PDGCodeVote " << PDGCodeVote(clockData, detProp, slc, pfp);
341  myprt << " nTP3Ds " << pfp.TP3Ds.size();
342  myprt << " Reco3DRange "
343  << Find3DRecoRange(slc, pfp, 0, (unsigned short)tcc.match3DCuts[3], 1);
344  } // prt
345  if (foundMVI) { PrintTP3Ds(clockData, detProp, "FF", slc, pfpVec[0], -1); }
346  for (unsigned short ip = 0; ip < pfpVec.size(); ++ip) {
347  auto& pfp = pfpVec[ip];
348  // set the end flag bits
349  geo::TPCID tpcid;
350  for (unsigned short end = 0; end < 2; ++end) {
351  // first set them all to 0
352  pfp.EndFlag[end].reset();
353  auto pos = PosAtEnd(pfp, end);
354  if (!InsideTPC(pos, tpcid)) pfp.EndFlag[end][kOutFV] = true;
355  } // end
356  // Set kink flag and create a vertex between this pfp and the previous one that was stored
357  if (ip > 0) {
358  pfp.EndFlag[0][kAtKink] = true;
359  Vtx3Store vx3;
360  vx3.TPCID = pfp.TPCID;
361  vx3.X = pfp.TP3Ds[0].Pos[0];
362  vx3.Y = pfp.TP3Ds[0].Pos[1];
363  vx3.Z = pfp.TP3Ds[0].Pos[2];
364  // TODO: Errors, Score?
365  vx3.Score = 100;
366  vx3.Vx2ID.resize(slc.nPlanes);
367  vx3.Wire = -2;
368  vx3.ID = slc.vtx3s.size() + 1;
369  vx3.Primary = false;
370  ++evt.global3V_UID;
371  vx3.UID = evt.global3V_UID;
372  slc.vtx3s.push_back(vx3);
373  pfp.Vx3ID[0] = vx3.ID;
374  auto& prevPFP = slc.pfps[slc.pfps.size() - 1];
375  prevPFP.Vx3ID[1] = vx3.ID;
376  } // ip > 0
377  // check for a valid two-plane match with a Tj in the third plane for long pfps.
378  // For short pfps, it is possible that a Tj would be too short to be reconstructed
379  // in the third plane.
380  if (pfp.TjIDs.size() == 2 && slc.nPlanes == 3 && pfp.TP3Ds.size() > 20 &&
381  !ValidTwoPlaneMatch(detProp, slc, pfp)) {
382  continue;
383  }
384  // Skip this combination if it isn't reconstructable in 3D
385  if (Find3DRecoRange(slc, pfp, 0, (unsigned short)tcc.match3DCuts[3], 1) == USHRT_MAX)
386  continue;
387  // See if it possible to reconstruct in more than one section
388  pfp.Flags[kCanSection] = CanSection(slc, pfp);
389  // Do a fit in multiple sections if the initial fit is poor
390  if (pfp.SectionFits[0].ChiDOF < tcc.match3DCuts[5]) {
391  // Good fit with one section
392  pfp.Flags[kNeedsUpdate] = false;
393  }
394  else if (pfp.Flags[kCanSection]) {
395  if (!ReSection(detProp, slc, pfp, foundMVI)) continue;
396  } // CanSection
397  if (foundMVI) { PrintTP3Ds(clockData, detProp, "RS", slc, pfp, -1); }
398  // FillGaps3D looks for gaps in the TP3Ds vector caused by broken trajectories and
399  // inserts new TP3Ds if there are hits in the gaps. This search is only done in a
400  // plane if the projection of the pfp results in a large angle where 2D reconstruction
401  // is likely to be poor - not true for TCWork2
402  FillGaps3D(clockData, detProp, slc, pfp, foundMVI);
403  // Check the TP3D -> TP assn, resolve conflicts and set TP -> InPFP
404  if (!ReconcileTPs(slc, pfp, foundMVI)) continue;
405  // Look for mis-placed 2D and 3D vertices
406  ReconcileVertices(slc, pfp, foundMVI);
407  // Set isGood
408  for (auto& tp3d : pfp.TP3Ds) {
409  if (tp3d.Flags[kTP3DBad]) continue;
410  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
411  if (tp.Environment[kEnvOverlap]) tp3d.Flags[kTP3DGood] = false;
412  } // tp3d
413  FilldEdx(clockData, detProp, slc, pfp);
414  pfp.PDGCode = PDGCodeVote(clockData, detProp, slc, pfp);
415  if (tcc.dbgPFP && pfp.MVI == debug.MVI)
416  PrintTP3Ds(clockData, detProp, "STORE", slc, pfp, -1);
417  if (!StorePFP(slc, pfp)) break;
418  } // ip (iterate over split pfps)
419  } // indx (iterate over matchVec entries)
420  slc.mallTraj.resize(0);
421  } // MakePFParticles
422 
424  bool ReconcileTPs(TCSlice& slc, PFPStruct& pfp, bool prt)
425  {
426  // Reconcile TP -> P assns before the pfp is stored. The TP3D -> TP is defined but
427  // the TP -> P assn may not have been done. This function overwrites the TjIDs
428  // vector to be the list of Tjs that contribute > 80% of their TPs to this pfp.
429  // This function returns true if the assns are consistent.
430 
431  if (!tcc.useAlg[kRTPs3D]) return true;
432  if (pfp.Flags[kSmallAngle]) return true;
433  if (pfp.TjIDs.empty()) return false;
434  if (pfp.TP3Ds.empty()) return false;
435  if (pfp.ID <= 0) return false;
436 
437  // Tj ID, TP count
438  std::vector<std::pair<int, float>> tjTPCnt;
439  for (auto& tp3d : pfp.TP3Ds) {
440  if (tp3d.Flags[kTP3DBad]) continue;
441  if (tp3d.TjID <= 0) return false;
442  // compare the TP3D -> TP -> P assn with the P -> TP assn
443  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
444  if (tp.InPFP > 0 && tp.InPFP != pfp.ID) return false;
445  // find the (Tj ID, TP count) pair in the list
446  unsigned short indx = 0;
447  for (indx = 0; indx < tjTPCnt.size(); ++indx)
448  if (tjTPCnt[indx].first == tp3d.TjID) break;
449  if (indx == tjTPCnt.size()) tjTPCnt.push_back(std::make_pair(tp3d.TjID, 0));
450  ++tjTPCnt[indx].second;
451  // make the TP -> P assn
452  tp.InPFP = pfp.ID;
453  } // tp3d
454 
455  std::vector<int> nTjIDs;
456  for (auto& tjtpcnt : tjTPCnt) {
457  auto& tj = slc.tjs[tjtpcnt.first - 1];
458  float npwc = NumPtsWithCharge(slc, tj, false);
459  if (tjtpcnt.second > 0.8 * npwc) nTjIDs.push_back(tjtpcnt.first);
460  } // tjtpcnt
461  if (prt) { mf::LogVerbatim("TC") << "RTPs3D: P" << pfp.ID << " nTjIDs " << nTjIDs.size(); }
462  // TODO: is this really a failure?
463  if (nTjIDs.size() < 2) { return false; }
464  pfp.TjIDs = nTjIDs;
465 
466  return true;
467  } // ReconcileTPs
468 
471  {
472  // Reconciles TP ownership conflicts between PFParticles
473  // Make a one-to-one TP -> P assn and look for one-to-many assns.
474  // Note: Comparing the pulls for a TP to two different PFParticles generally results
475  // in selecting the first PFParticle that was made which is not too surprising considering
476  // the order in which they were created. This comparison has been commented out in favor
477  // of simply keeping the old assn and removing the new one by setting IsBad true.
478 
479  if (!tcc.useAlg[kRTPs3D]) return;
480 
481  // make a list of T -> P assns
482  std::vector<int> TinP;
483  for (auto& pfp : slc.pfps) {
484  if (pfp.ID <= 0) continue;
485  if (pfp.Flags[kSmallAngle]) continue;
486  for (std::size_t ipt = 0; ipt < pfp.TP3Ds.size(); ++ipt) {
487  auto& tp3d = pfp.TP3Ds[ipt];
488  if (tp3d.TjID <= 0) continue;
489  if (std::find(TinP.begin(), TinP.end(), tp3d.TjID) == TinP.end()) TinP.push_back(tp3d.TjID);
490  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
491  if (tp.InPFP > 0) {
492  // an assn exists. Set the overlap bit and check consistency
493  tp.Environment[kEnvOverlap] = true;
494  // keep the previous assn (since it was created earlier and is more credible) and remove the new one
495  tp3d.Flags[kTP3DBad] = true;
496  tp3d.Flags[kTP3DGood] = false;
497  tp.InPFP = 0;
498  }
499  else {
500  // no assn exists
501  tp.InPFP = pfp.ID;
502  } // tp.InPFP > 0
503  } // ipt
504  } // pfp
505  } // ReconcileTPs
506 
508  void MakePFPTjs(TCSlice& slc)
509  {
510  // This function clobbers all of the tjs that are used in TP3Ds in the pfp and replaces
511  // them with new tjs that have a consistent set of TPs to prepare for putting them
512  // into the event. Note that none of the Tjs are attached to 2D vertices.
513  if (!tcc.useAlg[kMakePFPTjs]) return;
514 
515  // kill trajectories
516  std::vector<int> killme;
517  for (auto& pfp : slc.pfps) {
518  if (pfp.ID <= 0) continue;
519  for (auto& tp3d : pfp.TP3Ds) {
520  if (tp3d.TjID <= 0) continue;
521  if (tp3d.Flags[kTP3DBad]) continue;
522  if (std::find(killme.begin(), killme.end(), tp3d.TjID) == killme.end())
523  killme.push_back(tp3d.TjID);
524  } // tp3d
525  } // pfp
526 
527  bool prt = (tcc.dbgPFP);
528 
529  for (auto tid : killme)
530  MakeTrajectoryObsolete(slc, (unsigned int)(tid - 1));
531 
532  // Make template trajectories in each plane. These will be re-used by
533  // each PFParticle
534  std::vector<Trajectory> ptjs(slc.nPlanes);
535  // define the basic tj variables
536  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
537  ptjs[plane].Pass = 0;
538  ptjs[plane].CTP = EncodeCTP(slc.TPCID.Cryostat, slc.TPCID.TPC, plane);
539  // This Tj wasn't created by stepping
540  ptjs[plane].StepDir = 0;
541  // It was created by this function however
542  ptjs[plane].AlgMod[kMakePFPTjs] = true;
543  // and is 3D matched
544  ptjs[plane].AlgMod[kMat3D] = true;
545  } // plane
546 
547  // now make the new Tjs
548  for (auto& pfp : slc.pfps) {
549  if (pfp.ID <= 0) continue;
550  // initialize the tjs
551  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
552  ptjs[plane].Pts.clear();
553  --evt.WorkID;
554  if (evt.WorkID == INT_MIN) evt.WorkID = -1;
555  ptjs[plane].ID = evt.WorkID;
556  } // plane
557  pfp.TjIDs.clear();
558  // iterate through all of the TP3Ds, adding TPs to the TJ in the appropriate plane.
559  // The assumption here is that TP order reflects the TP3D order
560  for (auto& tp3d : pfp.TP3Ds) {
561  if (tp3d.TjID <= 0) continue;
562  if (tp3d.Flags[kTP3DBad]) continue;
563  // make a copy of the 2D TP
564  auto tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
565  if (tp.InPFP > 0 && tp.InPFP != pfp.ID) continue;
566  tp.InPFP = pfp.ID;
567  // the TP Step isn't useful anymore, so stash the original TJ ID into it
568  tp.Step = tp3d.TjID;
569  unsigned short plane = DecodeCTP(tp.CTP).Plane;
570  // append it to Pts
571  ptjs[plane].Pts.push_back(tp);
572  } // tp3d
573  // finish defining each of the Tjs and store them
574  // new tj ID indexed by plane
575  std::vector<int> tids(ptjs.size(), 0);
576  for (unsigned short plane = 0; plane < ptjs.size(); ++plane) {
577  auto& tj = ptjs[plane];
578  if (tj.Pts.size() < 2) continue;
579  tj.PDGCode = pfp.PDGCode;
580  tj.MCSMom = MCSMom(slc, tj);
581  if (!StoreTraj(slc, tj)) continue;
582  // associate it with the pfp
583  auto& newTj = slc.tjs.back();
584  pfp.TjIDs.push_back(newTj.ID);
585  tids[plane] = newTj.ID;
586  } // tj
587  // preserve the PFP -> 3V -> 2V -> T assns
588  for (unsigned short end = 0; end < 2; ++end) {
589  if (pfp.Vx3ID[end] <= 0) continue;
590  auto& vx3 = slc.vtx3s[pfp.Vx3ID[end] - 1];
591  for (unsigned short plane = 0; plane < ptjs.size(); ++plane) {
592  if (tids[plane] == 0) continue;
593  if (vx3.Vx2ID[plane] <= 0) continue;
594  auto& vx2 = slc.vtxs[vx3.Vx2ID[plane] - 1];
595  auto& tj = slc.tjs[tids[plane] - 1];
596  auto tend = CloseEnd(tj, vx2.Pos);
597  tj.VtxID[tend] = vx2.ID;
598  if (prt)
599  mf::LogVerbatim("TC") << "MPFPTjs: 3V" << vx3.ID << " -> 2V" << vx2.ID << " -> T"
600  << tj.ID << "_" << tend << " in plane " << plane;
601  } // plane
602  } // end
603  } // pfp
604  } // MakePFPTjs
605 
608  {
609  // Find wire intersections and put them in evt.wireIntersections
610 
611  // see if anything needs to be done
612  if (!evt.wireIntersections.empty() && evt.wireIntersections[0].tpc == slc.TPCID.TPC) return;
613 
614  evt.wireIntersections.clear();
615 
616  unsigned int cstat = slc.TPCID.Cryostat;
617  unsigned int tpc = slc.TPCID.TPC;
618  // find the minMax number of wires in each plane of the TPC
619  unsigned int maxWire = slc.nWires[0];
620  for (auto nw : slc.nWires)
621  if (nw < maxWire) maxWire = nw;
622  // Start looking for intersections in the middle
623  unsigned int firstWire = maxWire / 2;
624 
625  // find a valid wire intersection in all plane combinations
626  std::vector<std::pair<unsigned short, unsigned short>> pln1pln2;
627  for (unsigned short pln1 = 0; pln1 < slc.nPlanes - 1; ++pln1) {
628  for (unsigned short pln2 = pln1 + 1; pln2 < slc.nPlanes; ++pln2) {
629  auto p1p2 = std::make_pair(pln1, pln2);
630  if (std::find(pln1pln2.begin(), pln1pln2.end(), p1p2) != pln1pln2.end()) continue;
631  // find two wires that have a valid intersection
632  for (unsigned int wire = firstWire; wire < maxWire; ++wire) {
633  auto const intersection00 = tcc.wireReadoutGeom->WireIDsIntersect(
634  geo::WireID{cstat, tpc, pln1, wire}, geo::WireID{cstat, tpc, pln2, wire});
635  if (!intersection00) continue;
636 
637  // increment by one wire in pln1 and find another valid intersection
638  auto const intersection10 = tcc.wireReadoutGeom->WireIDsIntersect(
639  geo::WireID{cstat, tpc, pln1, wire + 10}, geo::WireID{cstat, tpc, pln2, wire});
640  if (!intersection10) continue;
641 
642  // increment by one wire in pln2 and find another valid intersection
643  auto const intersection01 = tcc.wireReadoutGeom->WireIDsIntersect(
644  geo::WireID{cstat, tpc, pln1, wire}, geo::WireID{cstat, tpc, pln2, wire + 10});
645  if (!intersection01) continue;
646 
647  TCWireIntersection tcwi;
648  tcwi.tpc = tpc;
649  tcwi.pln1 = pln1;
650  tcwi.pln2 = pln2;
651  tcwi.wir1 = wire;
652  tcwi.wir2 = wire;
653  tcwi.y = intersection00->y;
654  tcwi.z = intersection00->z;
655  tcwi.dydw1 = (intersection10->y - intersection00->y) / 10;
656  tcwi.dzdw1 = (intersection10->z - intersection00->z) / 10;
657  tcwi.dydw2 = (intersection01->y - intersection00->y) / 10;
658  tcwi.dzdw2 = (intersection01->z - intersection00->z) / 10;
659  evt.wireIntersections.push_back(tcwi);
660  break;
661  } // wire
662  } // pln2
663  } // pln1
664  } // FillWireIntersections
665 
667  bool TCIntersectionPoint(unsigned int wir1,
668  unsigned int wir2,
669  unsigned int pln1,
670  unsigned int pln2,
671  float& y,
672  float& z)
673  {
674  // A TrajCluster analog of geometry IntersectionPoint that uses local wireIntersections with
675  // float precision. The (y,z) position is only used to match TPs between planes - not for 3D fitting
676  if (evt.wireIntersections.empty()) return false;
677  if (pln1 == pln2) return false;
678 
679  if (pln1 > pln2) {
680  std::swap(pln1, pln2);
681  std::swap(wir1, wir2);
682  }
683 
684  for (auto& wi : evt.wireIntersections) {
685  if (wi.pln1 != pln1) continue;
686  if (wi.pln2 != pln2) continue;
687  // estimate the position using the wire differences
688  double dw1 = wir1 - wi.wir1;
689  double dw2 = wir2 - wi.wir2;
690  y = (float)(wi.y + dw1 * wi.dydw1 + dw2 * wi.dydw2);
691  z = (float)(wi.z + dw1 * wi.dzdw1 + dw2 * wi.dzdw2);
692  return true;
693  } // wi
694  return false;
695  } // TCIntersectionPoint
696 
698  void Match3PlanesSpt(TCSlice& slc, std::vector<MatchStruct>& matVec)
699  {
700  // fill matVec using SpacePoint -> Hit -> TP -> tj assns
701  if (evt.sptHits.empty()) return;
702 
703  // create a local vector of allHit -> Tj assns and populate it
704  std::vector<int> inTraj((*evt.allHits).size(), 0);
705  for (auto& tch : slc.slHits)
706  inTraj[tch.allHitsIndex] = tch.InTraj;
707 
708  // the TJ IDs for one match
709  std::array<int, 3> tIDs;
710  // vector for matched Tjs
711  std::vector<std::array<int, 3>> mtIDs;
712  // and a matching vector for the count
713  std::vector<unsigned short> mCnt;
714  // ignore Tj matches after hitting a user-defined limit
715  unsigned short maxCnt = USHRT_MAX;
716  if (tcc.match3DCuts[1] < (float)USHRT_MAX) maxCnt = (unsigned short)tcc.match3DCuts[1];
717  // a list of those Tjs
718  std::vector<unsigned short> tMaxed;
719 
720  unsigned int tpc = slc.TPCID.TPC;
721 
722  for (auto& sptHits : evt.sptHits) {
723  if (sptHits.size() != 3) continue;
724  // ensure that the SpacePoint is in the requested TPC
725  if (!SptInTPC(sptHits, tpc)) continue;
726  unsigned short cnt = 0;
727  for (unsigned short plane = 0; plane < 3; ++plane) {
728  unsigned int iht = sptHits[plane];
729  if (iht == UINT_MAX) continue;
730  if (inTraj[iht] <= 0) continue;
731  tIDs[plane] = inTraj[iht];
732  ++cnt;
733  } // iht
734  if (cnt != 3) continue;
735  // look for it in the list of tj combinations
736  unsigned short indx = 0;
737  for (indx = 0; indx < mtIDs.size(); ++indx)
738  if (tIDs == mtIDs[indx]) break;
739  if (indx == mtIDs.size()) {
740  // not found so add it to mtIDs and add another element to mCnt
741  mtIDs.push_back(tIDs);
742  mCnt.push_back(0);
743  }
744  ++mCnt[indx];
745  if (mCnt[indx] == maxCnt) {
746  // add the Tjs to the list
747  tMaxed.insert(tMaxed.end(), tIDs[0]);
748  tMaxed.insert(tMaxed.end(), tIDs[1]);
749  tMaxed.insert(tMaxed.end(), tIDs[2]);
750  break;
751  } // hit maxCnt
752  ++cnt;
753  } // sptHit
754 
755  std::vector<SortEntry> sortVec;
756  for (unsigned short indx = 0; indx < mCnt.size(); ++indx) {
757  auto& tIDs = mtIDs[indx];
758  // find the fraction of TPs on the shortest tj that are matched
759  float minTPCnt = USHRT_MAX;
760  for (auto tid : tIDs) {
761  auto& tj = slc.tjs[tid - 1];
762  float tpcnt = NumPtsWithCharge(slc, tj, false);
763  if (tpcnt < minTPCnt) minTPCnt = tpcnt;
764  } // tid
765  float frac = (float)mCnt[indx] / minTPCnt;
766  // ignore matches with a very low match fraction
767  if (frac < 0.05) continue;
768  SortEntry se;
769  se.index = indx;
770  se.val = mCnt[indx];
771  sortVec.push_back(se);
772  } // ii
773  if (sortVec.size() > 1) std::sort(sortVec.begin(), sortVec.end(), valsDecreasing);
774 
775  matVec.resize(sortVec.size());
776 
777  for (unsigned short ii = 0; ii < sortVec.size(); ++ii) {
778  unsigned short indx = sortVec[ii].index;
779  auto& ms = matVec[ii];
780  ms.Count = mCnt[indx];
781  ms.TjIDs.resize(3);
782  for (unsigned short plane = 0; plane < 3; ++plane)
783  ms.TjIDs[plane] = mtIDs[indx][plane];
784  } // indx
785 
786  } // Match3PlanesSpt
787 
789  bool SptInTPC(const std::array<unsigned int, 3>& sptHits, unsigned int tpc)
790  {
791  // returns true if a hit referenced in sptHits resides in the requested tpc. We assume
792  // that if one does, then all of them do
793 
794  unsigned int ahi = UINT_MAX;
795  for (auto ii : sptHits)
796  if (ii != UINT_MAX) {
797  ahi = ii;
798  break;
799  }
800  if (ahi >= (*evt.allHits).size()) return false;
801  // get a reference to the hit and see if it is in the desired tpc
802  auto& hit = (*evt.allHits)[ahi];
803  if (hit.WireID().TPC == tpc) return true;
804  return false;
805 
806  } // SptInTPC
807 
809  void Match3Planes(TCSlice& slc, std::vector<MatchStruct>& matVec)
810  {
811  // A simpler and faster version of MatchPlanes that only creates three plane matches
812 
813  if (slc.nPlanes != 3) return;
814 
815  // use SpacePoint -> Hit -> TP assns?
816  if (!evt.sptHits.empty()) {
817  Match3PlanesSpt(slc, matVec);
818  return;
819  }
820 
821  if (slc.mallTraj.empty()) return;
822  float xcut = tcc.match3DCuts[0];
823  double yzcut = 1.5 * tcc.wirePitch;
824 
825  // the TJ IDs for one match
826  std::array<unsigned short, 3> tIDs;
827  // vector for matched Tjs
828  std::vector<std::array<unsigned short, 3>> mtIDs;
829  // and a matching vector for the count
830  std::vector<unsigned short> mCnt;
831  // ignore Tj matches after hitting a user-defined limit
832  unsigned short maxCnt = USHRT_MAX;
833  if (tcc.match3DCuts[1] < (float)USHRT_MAX) maxCnt = (unsigned short)tcc.match3DCuts[1];
834  // a list of those Tjs
835  std::vector<unsigned short> tMaxed;
836 
837  for (std::size_t ipt = 0; ipt < slc.mallTraj.size() - 1; ++ipt) {
838  auto& iTjPt = slc.mallTraj[ipt];
839  // see if we hit the maxCnt limit
840  if (std::find(tMaxed.begin(), tMaxed.end(), iTjPt.id) != tMaxed.end()) continue;
841  auto& itp = slc.tjs[iTjPt.id - 1].Pts[iTjPt.ipt];
842  unsigned int iPlane = iTjPt.plane;
843  unsigned int iWire = std::nearbyint(itp.Pos[0]);
844  tIDs[iPlane] = iTjPt.id;
845  bool hitMaxCnt = false;
846  for (std::size_t jpt = ipt + 1; jpt < slc.mallTraj.size() - 1; ++jpt) {
847  auto& jTjPt = slc.mallTraj[jpt];
848  // ensure that the planes are different
849  if (jTjPt.plane == iTjPt.plane) continue;
850  // check for x range overlap. We know that jTjPt.xlo is >= iTjPt.xlo because of the sort
851  if (jTjPt.xlo > iTjPt.xhi) continue;
852  // break out if the x range difference becomes large
853  if (jTjPt.xlo > iTjPt.xhi + xcut) break;
854  // see if we hit the maxCnt limit
855  if (std::find(tMaxed.begin(), tMaxed.end(), jTjPt.id) != tMaxed.end()) continue;
856  auto& jtp = slc.tjs[jTjPt.id - 1].Pts[jTjPt.ipt];
857  unsigned short jPlane = jTjPt.plane;
858  unsigned int jWire = jtp.Pos[0];
859  Point2_t ijPos;
860  if (!TCIntersectionPoint(iWire, jWire, iPlane, jPlane, ijPos[0], ijPos[1])) continue;
861  tIDs[jPlane] = jTjPt.id;
862  for (std::size_t kpt = jpt + 1; kpt < slc.mallTraj.size(); ++kpt) {
863  auto& kTjPt = slc.mallTraj[kpt];
864  // ensure that the planes are different
865  if (kTjPt.plane == iTjPt.plane || kTjPt.plane == jTjPt.plane) continue;
866  if (kTjPt.xlo > iTjPt.xhi) continue;
867  // break out if the x range difference becomes large
868  if (kTjPt.xlo > iTjPt.xhi + xcut) break;
869  // see if we hit the maxCnt limit
870  if (std::find(tMaxed.begin(), tMaxed.end(), kTjPt.id) != tMaxed.end()) continue;
871  auto& ktp = slc.tjs[kTjPt.id - 1].Pts[kTjPt.ipt];
872  unsigned short kPlane = kTjPt.plane;
873  unsigned int kWire = ktp.Pos[0];
874  Point2_t ikPos;
875  if (!TCIntersectionPoint(iWire, kWire, iPlane, kPlane, ikPos[0], ikPos[1])) continue;
876  if (std::abs(ijPos[0] - ikPos[0]) > yzcut) continue;
877  if (std::abs(ijPos[1] - ikPos[1]) > yzcut) continue;
878  // we have a match
879  tIDs[kPlane] = kTjPt.id;
880  // look for it in the list
881  unsigned int indx = 0;
882  for (indx = 0; indx < mtIDs.size(); ++indx)
883  if (tIDs == mtIDs[indx]) break;
884  if (indx == mtIDs.size()) {
885  // not found so add it to mtIDs and add another element to mCnt
886  mtIDs.push_back(tIDs);
887  mCnt.push_back(0);
888  }
889  ++mCnt[indx];
890  if (mCnt[indx] == maxCnt) {
891  // add the Tjs to the list
892  tMaxed.insert(tMaxed.end(), tIDs[0]);
893  tMaxed.insert(tMaxed.end(), tIDs[1]);
894  tMaxed.insert(tMaxed.end(), tIDs[2]);
895  hitMaxCnt = true;
896  break;
897  } // hit maxCnt
898  } // kpt
899  if (hitMaxCnt) break;
900  } // jpt
901  } // ipt
902 
903  if (mCnt.empty()) return;
904 
905  std::vector<SortEntry> sortVec;
906  for (std::size_t indx = 0; indx < mCnt.size(); ++indx) {
907  auto& tIDs = mtIDs[indx];
908  // count the number of TPs in all Tjs
909  float tpCnt = 0;
910  for (auto tid : tIDs) {
911  auto& tj = slc.tjs[tid - 1];
912  tpCnt += NumPtsWithCharge(slc, tj, false);
913  } // tid
914  float frac = mCnt[indx] / tpCnt;
915  frac /= 3;
916  // ignore matches with a very low match fraction
917  if (frac < 0.05) continue;
918  SortEntry se;
919  se.index = indx;
920  se.val = mCnt[indx];
921  sortVec.push_back(se);
922  } // ii
923  if (sortVec.size() > 1) std::sort(sortVec.begin(), sortVec.end(), valsDecreasing);
924 
925  matVec.resize(sortVec.size());
926 
927  for (std::size_t ii = 0; ii < sortVec.size(); ++ii) {
928  unsigned short indx = sortVec[ii].index;
929  auto& ms = matVec[ii];
930  ms.Count = mCnt[indx];
931  ms.TjIDs.resize(3);
932  for (unsigned short plane = 0; plane < 3; ++plane)
933  ms.TjIDs[plane] = (int)mtIDs[indx][plane];
934  } // indx
935 
936  } // Match3Planes
937 
939  void Match2Planes(TCSlice& slc, std::vector<MatchStruct>& matVec)
940  {
941  // A simpler faster version of MatchPlanes that only creates two plane matches
942 
943  matVec.clear();
944  if (slc.mallTraj.empty()) return;
945 
946  int cstat = slc.TPCID.Cryostat;
947  int tpc = slc.TPCID.TPC;
948 
949  float xcut = tcc.match3DCuts[0];
950 
951  // the TJ IDs for one match
952  std::array<unsigned short, 2> tIDs;
953  // vector for matched Tjs
954  std::vector<std::array<unsigned short, 2>> mtIDs;
955  // and a matching vector for the count
956  std::vector<unsigned short> mCnt;
957  // ignore Tj matches after hitting a user-defined limit
958  unsigned short maxCnt = USHRT_MAX;
959  if (tcc.match3DCuts[1] < (float)USHRT_MAX) maxCnt = (unsigned short)tcc.match3DCuts[1];
960  // a list of those Tjs
961  std::vector<unsigned short> tMaxed;
962 
963  for (std::size_t ipt = 0; ipt < slc.mallTraj.size() - 1; ++ipt) {
964  auto& iTjPt = slc.mallTraj[ipt];
965  // see if we hit the maxCnt limit
966  if (std::find(tMaxed.begin(), tMaxed.end(), iTjPt.id) != tMaxed.end()) continue;
967  auto& itp = slc.tjs[iTjPt.id - 1].Pts[iTjPt.ipt];
968  unsigned short iPlane = iTjPt.plane;
969  unsigned int iWire = itp.Pos[0];
970  bool hitMaxCnt = false;
971  for (std::size_t jpt = ipt + 1; jpt < slc.mallTraj.size() - 1; ++jpt) {
972  auto& jTjPt = slc.mallTraj[jpt];
973  // ensure that the planes are different
974  if (jTjPt.plane == iTjPt.plane) continue;
975  // check for x range overlap. We know that jTjPt.xlo is >= iTjPt.xlo because of the sort
976  if (jTjPt.xlo > iTjPt.xhi) continue;
977  // break out if the x range difference becomes large
978  if (jTjPt.xlo > iTjPt.xhi + xcut) break;
979  // see if we hit the maxCnt limit
980  if (std::find(tMaxed.begin(), tMaxed.end(), jTjPt.id) != tMaxed.end()) continue;
981  auto& jtp = slc.tjs[jTjPt.id - 1].Pts[jTjPt.ipt];
982  unsigned short jPlane = jTjPt.plane;
983  unsigned int jWire = jtp.Pos[0];
984  if (!tcc.wireReadoutGeom->WireIDsIntersect(geo::WireID(cstat, tpc, iPlane, iWire),
985  geo::WireID(cstat, tpc, jPlane, jWire)))
986  continue;
987  tIDs[0] = iTjPt.id;
988  tIDs[1] = jTjPt.id;
989  // swap the order so that the == operator works correctly
990  if (tIDs[0] > tIDs[1]) std::swap(tIDs[0], tIDs[1]);
991  // look for it in the list
992  std::size_t indx = 0;
993  for (indx = 0; indx < mtIDs.size(); ++indx)
994  if (tIDs == mtIDs[indx]) break;
995  if (indx == mtIDs.size()) {
996  // not found so add it to mtIDs and add another element to mCnt
997  mtIDs.push_back(tIDs);
998  mCnt.push_back(0);
999  }
1000  ++mCnt[indx];
1001  if (mCnt[indx] == maxCnt) {
1002  // add the Tjs to the list
1003  tMaxed.insert(tMaxed.end(), tIDs[0]);
1004  tMaxed.insert(tMaxed.end(), tIDs[1]);
1005  hitMaxCnt = true;
1006  break;
1007  } // hit maxCnt
1008  if (hitMaxCnt) break;
1009  } // jpt
1010  } // ipt
1011 
1012  if (mCnt.empty()) return;
1013 
1014  std::vector<SortEntry> sortVec;
1015  for (std::size_t indx = 0; indx < mCnt.size(); ++indx) {
1016  auto& tIDs = mtIDs[indx];
1017  // count the number of TPs in all Tjs
1018  float tpCnt = 0;
1019  for (auto tid : tIDs) {
1020  auto& tj = slc.tjs[tid - 1];
1021  tpCnt += NumPtsWithCharge(slc, tj, false);
1022  } // tid
1023  float frac = mCnt[indx] / tpCnt;
1024  frac /= 2;
1025  // ignore matches with a very low match fraction
1026  if (frac < 0.05) continue;
1027  SortEntry se;
1028  se.index = indx;
1029  se.val = mCnt[indx];
1030  sortVec.push_back(se);
1031  } // ii
1032  if (sortVec.size() > 1) std::sort(sortVec.begin(), sortVec.end(), valsDecreasing);
1033 
1034  matVec.resize(sortVec.size());
1035 
1036  for (std::size_t ii = 0; ii < sortVec.size(); ++ii) {
1037  unsigned short indx = sortVec[ii].index;
1038  auto& ms = matVec[ii];
1039  ms.Count = mCnt[indx];
1040  ms.TjIDs.resize(2);
1041  for (unsigned short plane = 0; plane < 2; ++plane)
1042  ms.TjIDs[plane] = (int)mtIDs[indx][plane];
1043  } // indx
1044 
1045  } // Match2Planes
1046 
1048  bool Update(detinfo::DetectorPropertiesData const& detProp, const TCSlice& slc, PFPStruct& pfp)
1049  {
1050  // This function only updates SectionFits that need to be re-sorted or re-fit. It returns
1051  // false if there was a serious error indicating that the pfp should be abandoned
1052  if (pfp.TP3Ds.empty() || pfp.SectionFits.empty()) return false;
1053 
1054  // special handling for small angle tracks
1055  if (pfp.AlgMod[kSmallAngle]) {
1056  for (unsigned short sfi = 0; sfi < pfp.SectionFits.size(); ++sfi) {
1057  auto& sf = pfp.SectionFits[sfi];
1058  if (!sf.NeedsUpdate) continue;
1059  if (!SortSection(pfp, sfi)) return false;
1060  sf.NPts = 0;
1061  sf.ChiDOF = 0;
1062  for (unsigned short ipt = 0; ipt < pfp.TP3Ds.size(); ++ipt) {
1063  auto& tp3d = pfp.TP3Ds[ipt];
1064  if (tp3d.SFIndex < sfi) continue;
1065  if (tp3d.SFIndex > sfi) break;
1066  ++sf.NPts;
1067  double delta = tp3d.Pos[0] - tp3d.TPX;
1068  sf.ChiDOF += delta * delta / tp3d.TPXErr2;
1069  } // ipt
1070  if (sf.NPts < 5) { sf.ChiDOF = 0; }
1071  else {
1072  sf.ChiDOF /= (float)(sf.NPts - 4);
1073  }
1074  sf.NeedsUpdate = false;
1075  } // sfi
1076  pfp.Flags[kNeedsUpdate] = false;
1077  return true;
1078  } // kSmallAngle
1079 
1080  for (std::size_t sfi = 0; sfi < pfp.SectionFits.size(); ++sfi) {
1081  auto& sf = pfp.SectionFits[sfi];
1082  if (!sf.NeedsUpdate) continue;
1083  if (!FitSection(detProp, slc, pfp, sfi)) return false;
1084  if (!SortSection(pfp, sfi)) return false;
1085  sf.NeedsUpdate = false;
1086  } // sfi
1087 
1088  // ensure that all points (good or not) have a valid SFIndex
1089  for (auto& tp3d : pfp.TP3Ds) {
1090  if (tp3d.SFIndex >= pfp.SectionFits.size()) SetSection(detProp, pfp, tp3d);
1091  } // tp3d
1092  pfp.Flags[kNeedsUpdate] = false;
1093  return true;
1094  } // Update
1095 
1098  const TCSlice& slc,
1099  PFPStruct& pfp,
1100  bool prt)
1101  {
1102  // Re-fit the TP3Ds in sections and add/remove sections to keep ChiDOF of each section close to 1.
1103  // This function only fails when there is a serious error, otherwise if reasonable fits cannot be
1104  // achieved, the CanSection flag is set false.
1105  if (pfp.SectionFits.empty()) return false;
1106  // This function shouldn't be called if this is the case but it isn't a major failure if it is
1107  if (!pfp.Flags[kCanSection]) return true;
1108  if (pfp.Flags[kSmallAngle]) return true;
1109  // Likewise this shouldn't be attempted if there aren't at least 3 points in 2 planes in 2 sections
1110  // but it isn't a failure
1111  if (pfp.TP3Ds.size() < 12) {
1112  pfp.Flags[kCanSection] = false;
1113  return true;
1114  }
1115 
1116  prt = (pfp.MVI == debug.MVI);
1117 
1118  // try to keep ChiDOF between chiLo and chiHi
1119  float chiLo = 0.5 * tcc.match3DCuts[5];
1120  float chiHi = 1.5 * tcc.match3DCuts[5];
1121 
1122  // clobber the old sections if more than one exists
1123  if (pfp.SectionFits.size() > 1) {
1124  // make one section
1125  pfp.SectionFits.resize(1);
1126  // put all of the points in it and fit
1127  for (auto& tp3d : pfp.TP3Ds) {
1128  tp3d.SFIndex = 0;
1129  tp3d.Flags[kTP3DGood] = true;
1130  }
1131  auto& sf = pfp.SectionFits[0];
1132  if (!FitSection(detProp, slc, pfp, 0)) { return false; }
1133  if (sf.ChiDOF < tcc.match3DCuts[5]) return true;
1134  } // > 1 SectionFit
1135  // sort by distance from the start
1136  if (!SortSection(pfp, 0)) return false;
1137  // require a minimum of 3 points in 2 planes
1138  unsigned short min2DPts = 3;
1139  unsigned short fromPt = 0;
1140  // set the section index to invalid for all points
1141  for (auto& tp3d : pfp.TP3Ds)
1142  tp3d.SFIndex = USHRT_MAX;
1143  // Guess how many points should be added in each iteration
1144  unsigned short nPtsToAdd = pfp.TP3Ds.size() / 4;
1145  // the actual number of points that will be fit in the section
1146  unsigned short nPts = nPtsToAdd;
1147  // the minimum number of points
1148  unsigned short nPtsMin = Find3DRecoRange(slc, pfp, fromPt, min2DPts, 1) - fromPt + 1;
1149  if (nPtsMin >= pfp.TP3Ds.size()) {
1150  pfp.Flags[kCanSection] = false;
1151  return true;
1152  }
1153  float chiDOF = 0;
1154  if (nPts < nPtsMin) nPts = nPtsMin;
1155  // Try to reduce the number of iterations for long pfps
1156  if (pfp.TP3Ds.size() > 100) {
1157  unsigned short nhalf = pfp.TP3Ds.size() / 2;
1158  FitTP3Ds(detProp, slc, pfp, fromPt, nhalf, USHRT_MAX);
1159  if (chiDOF < tcc.match3DCuts[5]) nPts = nhalf;
1160  }
1161  bool lastSection = false;
1162  for (unsigned short sfIndex = 0; sfIndex < 20; ++sfIndex) {
1163  // Try to add/remove points in each section no more than 20 times
1164  float chiDOFPrev = 0;
1165  short nHiChi = 0;
1166  for (unsigned short nit = 0; nit < 10; ++nit) {
1167  // Decide how many points to add or subtract after doing the fit
1168  unsigned short nPtsNext = nPts;
1169  if (!FitTP3Ds(detProp, slc, pfp, fromPt, nPts, USHRT_MAX)) { nPtsNext += 1.5 * nPtsToAdd; }
1170  else if (chiDOF < chiLo) {
1171  // low chiDOF
1172  if (nHiChi > 2) {
1173  // declare it close enough if several attempts were made
1174  nPtsNext = 0;
1175  }
1176  else {
1177  nPtsNext += nPtsToAdd;
1178  } // nHiChi < 2
1179  nHiChi = 0;
1180  }
1181  else if (chiDOF > chiHi) {
1182  // high chiDOF
1183  ++nHiChi;
1184  if (nHiChi == 1 && chiDOFPrev > tcc.match3DCuts[5]) {
1185  // reduce the number of points by 1/2 on the first attempt
1186  nPtsNext /= 2;
1187  }
1188  else {
1189  // that didn't work so start subtracting groups of points
1190  short npnext = (short)nPts - nHiChi * 5;
1191  // assume this won't work
1192  nPtsNext = 0;
1193  if (npnext > nPtsMin) nPtsNext = npnext;
1194  }
1195  }
1196  else {
1197  // just right
1198  nPtsNext = 0;
1199  }
1200  // check for passing the end
1201  if (fromPt + nPtsNext >= pfp.TP3Ds.size()) {
1202  nPtsNext = pfp.TP3Ds.size() - fromPt;
1203  lastSection = true;
1204  }
1205  if (prt) {
1206  mf::LogVerbatim myprt("TC");
1207  myprt << " RS: P" << pfp.ID << " sfi/nit/npts " << sfIndex << "/" << nit << "/" << nPts;
1208  myprt << std::fixed << std::setprecision(1) << " chiDOF " << chiDOF;
1209  myprt << " fromPt " << fromPt;
1210  myprt << " nPtsNext " << nPtsNext;
1211  myprt << " nHiChi " << nHiChi;
1212  myprt << " lastSection? " << lastSection;
1213  }
1214  if (nPtsNext == 0) break;
1215  // see if this is the last section
1216  if (lastSection) break;
1217  if (chiDOF == chiDOFPrev) {
1218  if (prt) mf::LogVerbatim("TC") << " MVI " << pfp.MVI << " chiDOF not changing\n";
1219  break;
1220  }
1221  nPts = nPtsNext;
1222  chiDOFPrev = chiDOF;
1223  } // nit
1224  // finished this section. Assign the points to it
1225  unsigned short toPt = fromPt + nPts;
1226  if (toPt > pfp.TP3Ds.size()) toPt = pfp.TP3Ds.size();
1227  for (unsigned short ipt = fromPt; ipt < toPt; ++ipt)
1228  pfp.TP3Ds[ipt].SFIndex = sfIndex;
1229  // See if there are enough points remaining to reconstruct another section if this isn't known
1230  // to be the last section
1231  if (!lastSection) {
1232  // this will be the first point in the next section
1233  unsigned short nextFromPt = fromPt + nPts;
1234  // See if it will have enough points to be reconstructed
1235  unsigned short nextToPtMin = Find3DRecoRange(slc, pfp, nextFromPt, min2DPts, 1);
1236  if (nextToPtMin == USHRT_MAX) {
1237  // not enough points so this is the last section
1238  lastSection = true;
1239  // assign the remaining points to the last section
1240  for (std::size_t ipt = nextFromPt; ipt < pfp.TP3Ds.size(); ++ipt)
1241  pfp.TP3Ds[ipt].SFIndex = sfIndex;
1242  }
1243  } // !lastSection
1244  // Do a final fit and update the points. Don't worry about a poor ChiDOF
1245  FitSection(detProp, slc, pfp, sfIndex);
1246  if (!SortSection(pfp, 0)) { return false; }
1247  if (lastSection) break;
1248  // Prepare for the next section.
1249  fromPt = fromPt + nPts;
1250  nPts = nPtsToAdd;
1251  nPtsMin = Find3DRecoRange(slc, pfp, fromPt, min2DPts, 1) - fromPt + 1;
1252  if (nPtsMin >= pfp.TP3Ds.size()) break;
1253  // add a new section
1254  pfp.SectionFits.resize(pfp.SectionFits.size() + 1);
1255  } // snit
1256 
1257  // see if the last sf is valid
1258  if (pfp.SectionFits.size() > 1 && pfp.SectionFits.back().ChiDOF < 0) {
1259  unsigned short badSFI = pfp.SectionFits.size() - 1;
1260  // remove it
1261  pfp.SectionFits.pop_back();
1262  for (std::size_t ipt = pfp.TP3Ds.size() - 1; ipt > 0; --ipt) {
1263  auto& tp3d = pfp.TP3Ds[ipt];
1264  if (tp3d.SFIndex < badSFI) break;
1265  --tp3d.SFIndex;
1266  }
1267  pfp.SectionFits.back().NeedsUpdate = true;
1268  } // bad last SF
1269 
1270  // Ensure that the points at the end are in the last section
1271  for (std::size_t ipt = pfp.TP3Ds.size() - 1; ipt > 0; --ipt) {
1272  auto& tp3d = pfp.TP3Ds[ipt];
1273  if (tp3d.SFIndex < pfp.SectionFits.size()) break;
1274  tp3d.SFIndex = pfp.SectionFits.size() - 1;
1275  pfp.Flags[kNeedsUpdate] = true;
1276  pfp.SectionFits[tp3d.SFIndex].NeedsUpdate = true;
1277  } // tp3d
1278 
1279  Update(detProp, slc, pfp);
1280 
1281  // set CanSection false if the chisq is poor in any section
1282  for (auto& sf : pfp.SectionFits) {
1283  if (sf.ChiDOF > tcc.match3DCuts[5]) pfp.Flags[kCanSection] = false;
1284  }
1285 
1286  return true;
1287  } // resection
1288 
1290  void CountBadPoints(const TCSlice& slc,
1291  const PFPStruct& pfp,
1292  unsigned short fromPt,
1293  unsigned short toPt,
1294  unsigned short& nBadPts,
1295  unsigned short& firstBadPt)
1296  {
1297  // Count the number of points whose pull exceeds tcc.match3DCuts[4]
1298  firstBadPt = USHRT_MAX;
1299  nBadPts = 0;
1300  if (fromPt > pfp.TP3Ds.size() - 1) {
1301  nBadPts = USHRT_MAX;
1302  return;
1303  }
1304  if (toPt > pfp.TP3Ds.size()) toPt = pfp.TP3Ds.size();
1305  bool first = true;
1306  for (unsigned short ipt = fromPt; ipt < toPt; ++ipt) {
1307  auto& tp3d = pfp.TP3Ds[ipt];
1308  if (!tp3d.Flags[kTP3DGood]) continue;
1309  // don't clobber a point if it is on a TP that is overlapping another Tj. This will
1310  // happen for points close to a vertex and when trajectories cross
1311  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
1312  if (tp.Environment[kEnvOverlap]) continue;
1313  if (PointPull(tp3d) < tcc.match3DCuts[4]) continue;
1314  ++nBadPts;
1315  if (first) {
1316  first = false;
1317  firstBadPt = ipt;
1318  }
1319  } // ipt
1320  } // CountBadPoints
1321 
1323  bool CanSection(const TCSlice& slc, const PFPStruct& pfp)
1324  {
1325  // analyze the TP3D vector to determine if it can be reconstructed in 3D in more than one section with
1326  // the requirement that there are at least 3 points in two planes
1327  if (pfp.AlgMod[kJunk3D]) return false;
1328  if (pfp.AlgMod[kSmallAngle]) return false;
1329  if (pfp.TP3Ds.size() < 12) return false;
1330  unsigned short toPt = Find3DRecoRange(slc, pfp, 0, 3, 1);
1331  if (toPt > pfp.TP3Ds.size()) return false;
1332  unsigned short nextToPt = Find3DRecoRange(slc, pfp, toPt, 3, 1);
1333  if (nextToPt > pfp.TP3Ds.size()) return false;
1334  return true;
1335  } // CanSection
1336 
1338  unsigned short Find3DRecoRange(const TCSlice& slc,
1339  const PFPStruct& pfp,
1340  unsigned short fromPt,
1341  unsigned short min2DPts,
1342  short dir)
1343  {
1344  // Scans the TP3Ds vector starting at fromPt until it finds min2DPts in two planes. It returns
1345  // with the index of that point (+1) in the TP3Ds vector. The dir variable defines the scan direction in
1346  // the TP3Ds vector
1347  if (fromPt > pfp.TP3Ds.size() - 1) return USHRT_MAX;
1348  if (pfp.TP3Ds.size() < 2 * min2DPts) return USHRT_MAX;
1349  if (dir == 0) return USHRT_MAX;
1350 
1351  std::vector<unsigned short> cntInPln(slc.nPlanes);
1352  for (std::size_t ii = 0; ii < pfp.TP3Ds.size(); ++ii) {
1353  unsigned short ipt = fromPt + ii;
1354  if (dir < 0) ipt = fromPt - ii;
1355  if (ipt >= pfp.TP3Ds.size()) break;
1356  auto& tp3d = pfp.TP3Ds[ipt];
1357  if (!tp3d.Flags[kTP3DGood]) continue;
1358  unsigned short plane = DecodeCTP(slc.tjs[tp3d.TjID - 1].CTP).Plane;
1359  ++cntInPln[plane];
1360  unsigned short enufInPlane = 0;
1361  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane)
1362  if (cntInPln[plane] >= min2DPts) ++enufInPlane;
1363  if (enufInPlane > 1) return ipt + 1;
1364  if (dir < 0 && ipt == 0) break;
1365  } // ipt
1366  return USHRT_MAX;
1367  } // Find3DRecoRange
1368 
1370  void GetRange(const PFPStruct& pfp,
1371  unsigned short sfIndex,
1372  unsigned short& fromPt,
1373  unsigned short& npts)
1374  {
1375  fromPt = USHRT_MAX;
1376  if (sfIndex >= pfp.SectionFits.size()) return;
1377  if (pfp.TP3Ds.empty()) return;
1378  fromPt = USHRT_MAX;
1379  npts = 0;
1380  // Note that no test is made for not-good TP3Ds here since that would give a wrong npts count
1381  for (std::size_t ipt = 0; ipt < pfp.TP3Ds.size(); ++ipt) {
1382  auto& tp3d = pfp.TP3Ds[ipt];
1383  if (tp3d.SFIndex < sfIndex) continue;
1384  if (tp3d.SFIndex > sfIndex) break;
1385  if (fromPt == USHRT_MAX) fromPt = ipt;
1386  ++npts;
1387  } // ipt
1388  } // GetRange
1389 
1392  const TCSlice& slc,
1393  PFPStruct& pfp,
1394  unsigned short sfIndex)
1395  {
1396  // Fits the TP3D points in the selected section to a 3D line with the origin at the center of
1397  // the section
1398  if (pfp.TP3Ds.size() < 4) return false;
1399  if (sfIndex >= pfp.SectionFits.size()) return false;
1400  // don't fit a small angle PFP
1401  if (pfp.Flags[kSmallAngle]) return true;
1402 
1403  unsigned short fromPt = USHRT_MAX;
1404  unsigned short npts = 0;
1405  GetRange(pfp, sfIndex, fromPt, npts);
1406  if (fromPt == USHRT_MAX) return false;
1407  if (npts < 4) return false;
1408 
1409  // check for errors
1410  for (unsigned short ipt = fromPt; ipt < fromPt + npts; ++ipt) {
1411  auto& tp3d = pfp.TP3Ds[ipt];
1412  if (tp3d.SFIndex != sfIndex) return false;
1413  } // ipt
1414 
1415  // fit these points and update
1416  return FitTP3Ds(detProp, slc, pfp, fromPt, npts, sfIndex);
1417 
1418  } // FitSection
1419 
1422  const TCSlice& slc,
1423  const std::vector<TP3D>& tp3ds,
1424  unsigned short fromPt,
1425  short fitDir,
1426  unsigned short nPtsFit)
1427  {
1428  // fits the points and returns the fit results in a SectionFit struct. This function assumes that the
1429  // vector of TP3Ds exists in the slc.TPCID
1430 
1431  SectionFit sf;
1432  sf.ChiDOF = 999;
1433  if (nPtsFit < 5) return sf;
1434  if (!(fitDir == -1 || fitDir == 1)) return sf;
1435  if (fitDir == 1 && fromPt + nPtsFit > tp3ds.size()) return sf;
1436  if (fitDir == -1 && fromPt < 3) return sf;
1437 
1438  // put the offset, cosine-like and sine-like components in a vector
1439  std::vector<std::array<double, 3>> ocs(slc.nPlanes);
1440  for (unsigned short p = 0; p < slc.nPlanes; ++p) {
1441  auto planeID = geo::PlaneID{slc.TPCID, p};
1442  auto const& plane = tcc.wireReadoutGeom->Plane(planeID);
1443  // plane offset
1444  ocs[p][0] = plane.WireCoordinate(geo::Point_t{0, 0, 0});
1445  // get the "cosine-like" component
1446  ocs[p][1] = plane.WireCoordinate(geo::Point_t{0, 1, 0}) - ocs[p][0];
1447  // the "sine-like" component
1448  ocs[p][2] = plane.WireCoordinate(geo::Point_t{0, 0, 1}) - ocs[p][0];
1449  } // plane
1450 
1451  const unsigned int nvars = 4;
1452  unsigned int npts = 0;
1453 
1454  // count the number of TPs in each plane
1455  std::vector<unsigned short> cntInPln(slc.nPlanes, 0);
1456  // and define the X position for the fit origin
1457  double x0 = 0.;
1458  for (short ii = 0; ii < nPtsFit; ++ii) {
1459  short ipt = fromPt + fitDir * ii;
1460  if (ipt < 0 || ipt >= (short)tp3ds.size()) break;
1461  auto& tp3d = tp3ds[ipt];
1462  if (!tp3d.Flags[kTP3DGood]) continue;
1463  if (tp3d.TPXErr2 < 0.0001) return sf;
1464  x0 += tp3d.TPX;
1465  unsigned short plane = DecodeCTP(tp3d.CTP).Plane;
1466  ++cntInPln[plane];
1467  ++npts;
1468  } // ipt
1469  if (npts < 6) return sf;
1470  // ensure there are at least three points in at least two planes
1471  unsigned short enufInPlane = 0;
1472  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane)
1473  if (cntInPln[plane] > 2) ++enufInPlane;
1474  if (enufInPlane < 2) return sf;
1475 
1476  x0 /= (double)npts;
1477 
1478  TMatrixD A(npts, nvars);
1479  // vector holding the Wire number
1480  TVectorD w(npts);
1481  unsigned short cnt = 0;
1482  double weight = 1;
1483  for (short ii = 0; ii < nPtsFit; ++ii) {
1484  short ipt = fromPt + fitDir * ii;
1485  auto& tp3d = tp3ds[ipt];
1486  if (!tp3d.Flags[kTP3DGood]) continue;
1487  unsigned short plane = DecodeCTP(tp3d.CTP).Plane;
1488  double x = tp3d.TPX - x0;
1489  A[cnt][0] = weight * ocs[plane][1];
1490  A[cnt][1] = weight * ocs[plane][2];
1491  A[cnt][2] = weight * ocs[plane][1] * x;
1492  A[cnt][3] = weight * ocs[plane][2] * x;
1493  w[cnt] = weight * (tp3d.Wire - ocs[plane][0]);
1494  ++cnt;
1495  } // ipt
1496 
1497  TDecompSVD svd(A);
1498  bool ok;
1499  TVectorD tVec = svd.Solve(w, ok);
1500  if (!ok) return sf;
1501  double norm = sqrt(1 + tVec[2] * tVec[2] + tVec[3] * tVec[3]);
1502 
1503  norm *= -1;
1504 
1505  sf.Dir[0] = 1 / norm;
1506  sf.Dir[1] = tVec[2] / norm;
1507  sf.Dir[2] = tVec[3] / norm;
1508  sf.Pos[0] = x0;
1509  sf.Pos[1] = tVec[0];
1510  sf.Pos[2] = tVec[1];
1511  sf.NPts = npts;
1512 
1513  // Calculate errors from sigma * (A^T * A)^(-1) where sigma is the
1514  // error on the wire number (= 1)
1515  TMatrixD AT(nvars, npts);
1516  AT.Transpose(A);
1517  TMatrixD ATA = AT * A;
1518  double* det = 0;
1519  try {
1520  ATA.Invert(det);
1521  }
1522  catch (...) {
1523  return sf;
1524  }
1525  sf.DirErr[1] = -sqrt(ATA[2][2]) / norm;
1526  sf.DirErr[2] = -sqrt(ATA[3][3]) / norm;
1527 
1528  // calculate ChiDOF
1529  sf.ChiDOF = 0;
1530  // project this 3D vector into a TP in every plane
1531  std::vector<TrajPoint> plnTP(slc.nPlanes);
1532  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
1533  CTP_t inCTP = EncodeCTP(slc.TPCID.Cryostat, slc.TPCID.TPC, plane);
1534  plnTP[plane] = MakeBareTP(detProp, sf.Pos, sf.Dir, inCTP);
1535  } // plane
1536  // a local position
1537  Point3_t pos;
1538  sf.DirErr[0] = 0.;
1539  for (short ii = 0; ii < nPtsFit; ++ii) {
1540  short ipt = fromPt + fitDir * ii;
1541  auto& tp3d = tp3ds[ipt];
1542  if (!tp3d.Flags[kTP3DGood]) continue;
1543  unsigned short plane = DecodeCTP(tp3d.CTP).Plane;
1544  double dw = tp3d.Wire - plnTP[plane].Pos[0];
1545  // dt/dW was stored in DeltaRMS by MakeBareTP
1546  double t = dw * plnTP[plane].DeltaRMS;
1547  for (unsigned short xyz = 0; xyz < 3; ++xyz)
1548  pos[xyz] = sf.Pos[xyz] + t * sf.Dir[xyz];
1549  // Note that the tp3d position is directly above the wire position and not the
1550  // point at the distance of closest approach. Delta is the difference in the
1551  // drift direction in cm
1552  double delta = pos[0] - tp3d.TPX;
1553  sf.ChiDOF += delta * delta / tp3d.TPXErr2;
1554  // estimate the X slope error ~ X direction vector with an overly simple average
1555  double dangErr = delta / dw;
1556  sf.DirErr[0] += dangErr * dangErr;
1557  } // indx
1558  sf.DirErr[0] = sqrt(sf.DirErr[0]) / (double)nPtsFit;
1559  sf.ChiDOF /= (float)(npts - 4);
1560  return sf;
1561 
1562  } // FitTP3Ds
1563 
1566  const TCSlice& slc,
1567  PFPStruct& pfp,
1568  unsigned short fromPt,
1569  unsigned short nPtsFit,
1570  unsigned short sfIndex)
1571  {
1572  // Fit points in the pfp.TP3Ds vector fromPt. This function
1573  // doesn't update the TP3Ds unless sfIndex refers to a valid SectionFit in the pfp.
1574  // No check is made to ensure that the TP3D SFIndex variable is compatible with sfIndex
1575 
1576  if (nPtsFit < 5) return false;
1577  if (fromPt + nPtsFit > pfp.TP3Ds.size()) return false;
1578 
1579  auto sf = FitTP3Ds(detProp, slc, pfp.TP3Ds, fromPt, 1, nPtsFit);
1580  if (sf.ChiDOF > 900) return false;
1581 
1582  // don't update the pfp?
1583  if (sfIndex >= pfp.SectionFits.size()) return true;
1584 
1585  // update the pfp Sectionfit
1586  pfp.SectionFits[sfIndex] = sf;
1587  // update the TP3Ds
1588  // project this 3D vector into a TP in every plane
1589  std::vector<TrajPoint> plnTP(slc.nPlanes);
1590  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
1591  CTP_t inCTP = EncodeCTP(pfp.TPCID.Cryostat, pfp.TPCID.TPC, plane);
1592  plnTP[plane] = MakeBareTP(detProp, sf.Pos, sf.Dir, inCTP);
1593  } // plane
1594  Point3_t pos;
1595  bool needsSort = false;
1596  double prevAlong = 0;
1597  for (unsigned short ipt = fromPt; ipt < fromPt + nPtsFit; ++ipt) {
1598  auto& tp3d = pfp.TP3Ds[ipt];
1599  unsigned short plane = DecodeCTP(tp3d.CTP).Plane;
1600  double dw = tp3d.Wire - plnTP[plane].Pos[0];
1601  // dt/dW was stored in DeltaRMS by MakeBareTP
1602  double t = dw * plnTP[plane].DeltaRMS;
1603  if (ipt == fromPt) { prevAlong = t; }
1604  else {
1605  if (t < prevAlong) needsSort = true;
1606  prevAlong = t;
1607  }
1608  for (unsigned short xyz = 0; xyz < 3; ++xyz)
1609  pos[xyz] = sf.Pos[xyz] + t * sf.Dir[xyz];
1610  // Note that the tp3d position is directly above the wire position and not the
1611  // distance of closest approach. The Delta variable is the difference in the
1612  // drift direction in cm
1613  double delta = pos[0] - tp3d.TPX;
1614  tp3d.Pos = pos;
1615  tp3d.Dir = sf.Dir;
1616  tp3d.along = t;
1617  if (tp3d.Flags[kTP3DGood]) sf.ChiDOF += delta * delta / tp3d.TPXErr2;
1618  } // ipt
1619  if (needsSort) SortSection(pfp, sfIndex);
1620  pfp.SectionFits[sfIndex].NeedsUpdate = false;
1621  return true;
1622 
1623  } // FitTP3Ds
1624 
1626  void ReconcileVertices(TCSlice& slc, PFPStruct& pfp, bool prt)
1627  {
1628  // Checks for mis-placed 2D and 3D vertices and either attaches them
1629  // to a vertex or deletes(?) the vertex while attempting to preserve or
1630  // correct the P -> T -> 2V -> 3V assn. After this is done, the function
1631  // TCVertex/AttachToAnyVertex is called.
1632  // This function returns true if something was done to the pfp that requires
1633  // a re-definition of the pfp, e.g. adding or removing TP3Ds. Note that this
1634  // never occurs as the function is currently written
1635 
1636  if (tcc.vtx3DCuts.size() < 3) return;
1637  if (pfp.TP3Ds.empty()) return;
1638  if (pfp.Flags[kJunk3D]) return;
1639  if (pfp.Flags[kSmallAngle]) return;
1640 
1641  // first make a list of all Tjs
1642  std::vector<int> tjList;
1643  for (auto& tp3d : pfp.TP3Ds) {
1644  if (!tp3d.Flags[kTP3DGood]) continue;
1645  // ignore single hits
1646  if (tp3d.TjID <= 0) continue;
1647  if (std::find(tjList.begin(), tjList.end(), tp3d.TjID) == tjList.end())
1648  tjList.push_back(tp3d.TjID);
1649  } // tp3d
1650  // look for 3D vertices associated with these Tjs and list of
1651  // orphan 2D vertices - those that are not matched to 3D vertices
1652  std::vector<int> vx2List, vx3List;
1653  for (auto tid : tjList) {
1654  auto& tj = slc.tjs[tid - 1];
1655  for (unsigned short end = 0; end < 2; ++end) {
1656  if (tj.VtxID[end] <= 0) continue;
1657  auto& vx2 = slc.vtxs[tj.VtxID[end] - 1];
1658  if (vx2.Vx3ID > 0) {
1659  if (std::find(vx3List.begin(), vx3List.end(), vx2.Vx3ID) == vx3List.end())
1660  vx3List.push_back(vx2.Vx3ID);
1661  // 3D vertex exists
1662  }
1663  else {
1664  // no 3D vertex
1665  if (std::find(vx2List.begin(), vx2List.end(), tj.VtxID[end]) == vx2List.end())
1666  vx2List.push_back(tj.VtxID[end]);
1667  } // no 3D vertex
1668  } // end
1669  } // tid
1670  // no vertex reconciliation is necessary
1671  if (vx2List.empty() && vx3List.empty()) return;
1672  if (prt) {
1673  mf::LogVerbatim myprt("TC");
1674  myprt << "RV: P" << pfp.ID << " ->";
1675  for (auto tid : tjList)
1676  myprt << " T" << tid;
1677  myprt << " ->";
1678  for (auto vid : vx3List)
1679  myprt << " 3V" << vid;
1680  if (!vx2List.empty()) {
1681  myprt << " orphan";
1682  for (auto vid : vx2List)
1683  myprt << " 2V" << vid;
1684  }
1685  } // prt
1686  // Just kill the orphan 2D vertices regardless of their score.
1687  // This is an indicator that the vertex was created between two tjs
1688  // that maybe should have been reconstructed as one or alternatively
1689  // as two Tjs. This decision presumes the existence of a 3D kink
1690  // algorithm that doesn't yet exist...
1691  for (auto vid : vx2List) {
1692  auto& vx2 = slc.vtxs[vid - 1];
1693  MakeVertexObsolete("RV", slc, vx2, true);
1694  } // vx2List
1695  // ignore the T -> 2V -> 3V assns (if any exist) and try to directly
1696  // attach to 3D vertices at both ends
1697  AttachToAnyVertex(slc, pfp, tcc.vtx3DCuts[2], prt);
1698  // check for differences and while we are here, see if the pfp was attached
1699  // to a neutrino vertex and the direction is wrong
1700  int neutrinoVx = 0;
1701  if (!slc.pfps.empty()) {
1702  auto& npfp = slc.pfps[0];
1703  bool neutrinoPFP = (npfp.PDGCode == 12 || npfp.PDGCode == 14);
1704  if (neutrinoPFP) neutrinoVx = npfp.Vx3ID[0];
1705  } // pfps exist
1706  unsigned short neutrinoVxEnd = 2;
1707  for (unsigned short end = 0; end < 2; ++end) {
1708  // see if a vertex got attached
1709  if (pfp.Vx3ID[end] <= 0) continue;
1710  if (pfp.Vx3ID[end] == neutrinoVx) neutrinoVxEnd = end;
1711  // see if this is a vertex in the list using the T -> 2V -> 3V assns
1712  if (std::find(vx3List.begin(), vx3List.end(), pfp.Vx3ID[end]) != vx3List.end()) continue;
1713  } // end
1714  if (neutrinoVxEnd < 2 && neutrinoVxEnd != 0) Reverse(pfp);
1715 
1716  return;
1717  } // ReconcileVertices
1718 
1721  detinfo::DetectorPropertiesData const& detProp,
1722  TCSlice& slc,
1723  PFPStruct& pfp,
1724  bool prt)
1725  {
1726  // Look for gaps in each plane in the TP3Ds vector in planes in which
1727  // the projection of the pfp angle is large (~> 60 degrees). Hits
1728  // reconstructed at large angles are poorly reconstructed which results
1729  // in poorly reconstructed 2D trajectories
1730 
1731  if (pfp.ID <= 0) return;
1732  if (pfp.TP3Ds.empty()) return;
1733  if (pfp.SectionFits.empty()) return;
1734  if (!tcc.useAlg[kFillGaps3D]) return;
1735  if (pfp.Flags[kJunk3D]) return;
1736  if (pfp.Flags[kSmallAngle]) return;
1737 
1738  // Only print APIR details if MVI is set
1739  bool foundMVI = (tcc.dbgPFP && pfp.MVI == debug.MVI);
1740 
1741  // make a copy in case something goes wrong
1742  auto pWork = pfp;
1743 
1744  unsigned short nPtsAdded = 0;
1745  unsigned short fromPt = 0;
1746  unsigned short toPt = pWork.TP3Ds.size();
1747  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
1748  CTP_t inCTP = EncodeCTP(pWork.TPCID.Cryostat, pWork.TPCID.TPC, plane);
1749  unsigned short nWires, nAdd;
1750  AddPointsInRange(clockData,
1751  detProp,
1752  slc,
1753  pWork,
1754  fromPt,
1755  toPt,
1756  inCTP,
1757  tcc.match3DCuts[4],
1758  nWires,
1759  nAdd,
1760  foundMVI);
1761  if (pWork.Flags[kNeedsUpdate]) Update(detProp, slc, pWork);
1762  nPtsAdded += nAdd;
1763  } // plane
1764  if (prt) mf::LogVerbatim("TC") << "FG3D P" << pWork.ID << " added " << nPtsAdded << " points";
1765  if (pWork.Flags[kNeedsUpdate] && !Update(detProp, slc, pWork)) return;
1766  pfp = pWork;
1767  return;
1768  } // FillGaps3D
1769 
1772  const TCSlice& slc,
1773  const PFPStruct& pfp)
1774  {
1775  // This function checks the third plane in the PFP when only two Tjs are 3D-matched to
1776  // ensure that the reason for the lack of a 3rd plane match is that it is in a dead region.
1777  // This function should be used after an initial fit is done and the TP3Ds are sorted
1778  if (pfp.TjIDs.size() != 2) return false;
1779  if (slc.nPlanes != 3) return false;
1780  if (pfp.TP3Ds.empty()) return false;
1781 
1782  // find the third plane
1783  std::vector<unsigned short> planes;
1784  for (auto tid : pfp.TjIDs)
1785  planes.push_back(DecodeCTP(slc.tjs[tid - 1].CTP).Plane);
1786  unsigned short thirdPlane = 3 - planes[0] - planes[1];
1787  CTP_t inCTP = EncodeCTP(slc.TPCID.Cryostat, slc.TPCID.TPC, thirdPlane);
1788  // Project the 3D position at the start into the third plane
1789  auto tp = MakeBareTP(detProp, pfp.TP3Ds[0].Pos, inCTP);
1790  unsigned int wire0 = 0;
1791  if (tp.Pos[0] > 0) wire0 = std::nearbyint(tp.Pos[0]);
1792  if (wire0 > slc.nWires[thirdPlane]) wire0 = slc.nWires[thirdPlane];
1793  // Do the same for the end
1794  unsigned short lastPt = pfp.TP3Ds.size() - 1;
1795  tp = MakeBareTP(detProp, pfp.TP3Ds[lastPt].Pos, inCTP);
1796  unsigned int wire1 = 0;
1797  if (tp.Pos[0] > 0) wire1 = std::nearbyint(tp.Pos[0]);
1798  if (wire1 > slc.nWires[thirdPlane]) wire1 = slc.nWires[thirdPlane];
1799  if (wire0 == wire1) return !evt.goodWire[thirdPlane][wire0];
1800  if (wire1 < wire0) std::swap(wire0, wire1);
1801  // count the number of good wires
1802  int dead = 0;
1803  int wires = wire1 - wire0;
1804  for (unsigned int wire = wire0; wire < wire1; ++wire)
1805  if (!evt.goodWire[thirdPlane][wire]) ++dead;
1806  // require that most of the wires are dead
1807  return (dead > 0.8 * wires);
1808  } // ValidTwoPlaneMatch
1809 
1812  detinfo::DetectorPropertiesData const& detProp,
1813  TCSlice& slc,
1814  PFPStruct& pfp,
1815  unsigned short fromPt,
1816  unsigned short toPt,
1817  CTP_t inCTP,
1818  float maxPull,
1819  unsigned short& nWires,
1820  unsigned short& nAdd,
1821  bool prt)
1822  {
1823  // Try to insert 2D trajectory points into the 3D trajectory point vector pfp.TP3Ds.
1824  // This function inserts new TP3Ds and sets the NeedsUpdate flags true.
1825  // The calling function should call Update
1826  // Note that maxPull is used for the charge pull as well as the position pull
1827  nWires = 0;
1828  nAdd = 0;
1829  if (fromPt > toPt) return;
1830  if (toPt >= pfp.TP3Ds.size()) toPt = pfp.TP3Ds.size() - 1;
1831 
1832  // Find the average dE/dx so we can apply a generous min/max dE/dx cut
1833  float dEdXAve = 0;
1834  float dEdXRms = 0;
1835  Average_dEdX(clockData, detProp, slc, pfp, dEdXAve, dEdXRms);
1836  float dEdxMin = 0.5, dEdxMax = 50.;
1837  if (dEdXAve > 0.5) {
1838  dEdxMin = dEdXAve - maxPull * dEdXRms;
1839  if (dEdxMin < 0.5) dEdxMin = 0.5;
1840  dEdxMax = dEdXAve + maxPull * dEdXRms;
1841  if (dEdxMax > 50.) dEdxMax = 50.;
1842  } // dEdXAve > 0.5
1843 
1844  // Split the range into sub-ranges; one for each SectionFit and make 2D TPs at the
1845  // start of each sub-range.
1846  std::vector<TrajPoint> sfTPs;
1847  // form a list of TPs that are used in this CTP
1848  std::vector<std::pair<int, unsigned short>> tpUsed;
1849  for (auto& tp3d : pfp.TP3Ds) {
1850  if (tp3d.CTP != inCTP) continue;
1851  tpUsed.push_back(std::make_pair(tp3d.TjID, tp3d.TPIndex));
1852  } // tp3d
1853  unsigned int toWire = 0;
1854  unsigned int fromWire = UINT_MAX;
1855 
1856  unsigned short inSF = USHRT_MAX;
1857  unsigned short pln = DecodeCTP(inCTP).Plane;
1858  for (unsigned short ipt = 0; ipt < pfp.TP3Ds.size(); ++ipt) {
1859  auto& tp3d = pfp.TP3Ds[ipt];
1860  // Skip if not the last point and we already found a point in this SectionFit
1861  if (ipt < pfp.TP3Ds.size() - 1 && tp3d.SFIndex == inSF) continue;
1862  unsigned int wire;
1863  if (tp3d.CTP == inCTP) {
1864  // Found the first tp3d in a new SectionFit and it is in the right CTP
1865  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
1866  sfTPs.push_back(tp);
1867  wire = std::nearbyint(tp.Pos[0]);
1868  if (wire >= slc.nWires[pln]) break;
1869  }
1870  else {
1871  // Found the first tp3d in a new SectionFit and it is in a different CTP.
1872  // Make a TP in the right CTP
1873  auto tp = MakeBareTP(detProp, tp3d.Pos, tp3d.Dir, inCTP);
1874  wire = std::nearbyint(tp.Pos[0]);
1875  if (wire >= slc.nWires[pln]) break;
1876  sfTPs.push_back(tp);
1877  }
1878  if (wire < fromWire) fromWire = wire;
1879  if (wire > toWire) toWire = wire;
1880  inSF = tp3d.SFIndex;
1881  } // tp3d
1882  if (sfTPs.empty()) return;
1883  // reverse the vector if necessary so the wires will be in increasing order
1884  if (sfTPs.size() > 1 && sfTPs[0].Pos[0] > sfTPs.back().Pos[0]) {
1885  std::reverse(sfTPs.begin(), sfTPs.end());
1886  }
1887 
1888  // set a generous search window in WSE units
1889  float window = 50;
1890 
1891  if (prt)
1892  mf::LogVerbatim("TC") << "APIR: inCTP " << inCTP << " fromWire " << fromWire << " toWire "
1893  << toWire;
1894 
1895  // iterate over the sub-ranges
1896  for (unsigned short subr = 0; subr < sfTPs.size(); ++subr) {
1897  auto& fromTP = sfTPs[subr];
1898  unsigned int toWireInSF = toWire;
1899  if (subr < sfTPs.size() - 1) toWireInSF = std::nearbyint(sfTPs[subr + 1].Pos[0]);
1900  SetAngleCode(fromTP);
1901  unsigned int fromWire = std::nearbyint(fromTP.Pos[0]);
1902  if (fromWire > toWire) continue;
1903  if (prt)
1904  mf::LogVerbatim("TC") << " inCTP " << inCTP << " subr " << subr << " fromWire " << fromWire
1905  << " toWireInSF " << toWireInSF;
1906  for (unsigned int wire = fromWire; wire <= toWireInSF; ++wire) {
1907  MoveTPToWire(fromTP, (float)wire);
1908  if (!FindCloseHits(slc, fromTP, window, kUsedHits)) continue;
1909  if (fromTP.Environment[kEnvNotGoodWire]) continue;
1910  float bestPull = maxPull;
1911  TP3D bestTP3D;
1912  for (auto iht : fromTP.Hits) {
1913  if (slc.slHits[iht].InTraj <= 0) continue;
1914  // this hit is used in a TP so find the tpIndex
1915  auto& utj = slc.tjs[slc.slHits[iht].InTraj - 1];
1916  unsigned short tpIndex = 0;
1917  for (tpIndex = utj.EndPt[0]; tpIndex <= utj.EndPt[1]; ++tpIndex) {
1918  auto& utp = utj.Pts[tpIndex];
1919  if (utp.Chg <= 0) continue;
1920  // This doesn't check for UseHit true but that is probably ok here
1921  if (std::find(utp.Hits.begin(), utp.Hits.end(), iht) != utp.Hits.end()) break;
1922  } // ipt
1923  if (tpIndex > utj.EndPt[1]) continue;
1924  // see if it is already used in this pfp
1925  std::pair<int, unsigned short> tppr = std::make_pair(utj.ID, tpIndex);
1926  if (std::find(tpUsed.begin(), tpUsed.end(), tppr) != tpUsed.end()) continue;
1927  tpUsed.push_back(tppr);
1928  auto& utp = utj.Pts[tpIndex];
1929  // see if it is used in a different PFP
1930  if (utp.InPFP > 0) continue;
1931  // or if it overlaps another trajectory near a 2D vertex
1932  if (utp.Environment[kEnvOverlap]) continue;
1933  auto newTP3D = CreateTP3D(detProp, slc, utj.ID, tpIndex);
1934  if (!SetSection(detProp, pfp, newTP3D)) continue;
1935  // set the direction to the direction of the SectionFit it is in so we can calculate dE/dx
1936  newTP3D.Dir = pfp.SectionFits[newTP3D.SFIndex].Dir;
1937  float pull = PointPull(newTP3D);
1938  float dedx = dEdx(clockData, detProp, slc, newTP3D);
1939  // Require a good pull and a consistent dE/dx (MeV/cm)
1940  bool useIt = (pull < bestPull && dedx > dEdxMin && dedx < dEdxMax);
1941  if (!useIt) continue;
1942  bestTP3D = newTP3D;
1943  bestPull = pull;
1944  } // iht
1945  if (bestPull < maxPull) {
1946  if (prt && bestPull < 10) {
1947  mf::LogVerbatim myprt("TC");
1948  auto& tp = slc.tjs[bestTP3D.TjID - 1].Pts[bestTP3D.TPIndex];
1949  myprt << "APIR: P" << pfp.ID << " added TP " << PrintPos(tp);
1950  myprt << " pull " << std::fixed << std::setprecision(2) << bestPull;
1951  myprt << " dx " << bestTP3D.TPX - bestTP3D.Pos[0] << " in section " << bestTP3D.SFIndex;
1952  }
1953  if (InsertTP3D(pfp, bestTP3D) == USHRT_MAX) continue;
1954  ++nAdd;
1955  } // bestPull < maxPull
1956  } // wire
1957  } // subr
1958  } // AddPointsInRange
1959 
1961  unsigned short InsertTP3D(PFPStruct& pfp, TP3D& tp3d)
1962  {
1963  // inserts the tp3d into the section defined by tp3d.SFIndex
1964  if (tp3d.SFIndex >= pfp.SectionFits.size()) return USHRT_MAX;
1965  // Find the first occurrence of this SFIndex
1966  std::size_t ipt = 0;
1967  for (ipt = 0; ipt < pfp.TP3Ds.size(); ++ipt)
1968  if (tp3d.SFIndex == pfp.TP3Ds[ipt].SFIndex) break;
1969  if (ipt == pfp.TP3Ds.size()) return USHRT_MAX;
1970  // next see if we can insert it so that re-sorting of this section isn't required
1971  auto lastTP3D = pfp.TP3Ds.back();
1972  if (ipt == 0 && tp3d.along < pfp.TP3Ds[0].along) {
1973  // insert at the beginning. No search needs to be done
1974  }
1975  else if (tp3d.SFIndex == lastTP3D.SFIndex && tp3d.along > lastTP3D.along) {
1976  // insert at the end. Use push_back and return
1977  pfp.TP3Ds.push_back(tp3d);
1978  pfp.SectionFits[tp3d.SFIndex].NeedsUpdate = true;
1979  pfp.Flags[kNeedsUpdate] = true;
1980  return pfp.TP3Ds.size() - 1;
1981  }
1982  else {
1983  for (std::size_t iipt = ipt; iipt < pfp.TP3Ds.size() - 1; ++iipt) {
1984  // break out if the next point is in a different section
1985  if (pfp.TP3Ds[iipt + 1].SFIndex != tp3d.SFIndex) break;
1986  if (tp3d.along > pfp.TP3Ds[iipt].along && tp3d.along < pfp.TP3Ds[iipt + 1].along) {
1987  ipt = iipt + 1;
1988  break;
1989  }
1990  } // iipt
1991  } // insert in the middle
1992  pfp.TP3Ds.insert(pfp.TP3Ds.begin() + ipt, tp3d);
1993  pfp.SectionFits[tp3d.SFIndex].NeedsUpdate = true;
1994  pfp.Flags[kNeedsUpdate] = true;
1995  return ipt;
1996  } // InsertTP3D
1997 
1999  bool SortSection(PFPStruct& pfp, unsigned short sfIndex)
2000  {
2001  // sorts the TP3Ds by the distance from the start of a fit section
2002 
2003  if (sfIndex > pfp.SectionFits.size() - 1) return false;
2004  auto& sf = pfp.SectionFits[sfIndex];
2005  if (sf.Pos[0] == 0.0 && sf.Pos[1] == 0.0 && sf.Pos[2] == 0.0) return false;
2006 
2007  // a temp vector of points in this section
2008  std::vector<TP3D> temp;
2009  // and the index into TP3Ds
2010  std::vector<unsigned short> indx;
2011  // See if the along variable is monotonically increasing
2012  float prevAlong = 0;
2013  bool first = true;
2014  bool needsSort = false;
2015  for (std::size_t ii = 0; ii < pfp.TP3Ds.size(); ++ii) {
2016  auto& tp3d = pfp.TP3Ds[ii];
2017  if (tp3d.SFIndex != sfIndex) continue;
2018  if (first) {
2019  first = false;
2020  prevAlong = tp3d.along;
2021  }
2022  else {
2023  if (tp3d.along < prevAlong) needsSort = true;
2024  prevAlong = tp3d.along;
2025  }
2026  temp.push_back(tp3d);
2027  indx.push_back(ii);
2028  } // tp3d
2029  if (temp.empty()) return false;
2030  // no sort needed?
2031  if (temp.size() == 1) return true;
2032  if (!needsSort) {
2033  sf.NeedsUpdate = false;
2034  return true;
2035  }
2036  // see if the points are not-contiguous
2037  bool contiguous = true;
2038  for (std::size_t ipt = 1; ipt < indx.size(); ++ipt) {
2039  if (indx[ipt] != indx[ipt - 1] + 1) contiguous = false;
2040  } // ipt
2041  if (!contiguous) { return false; }
2042 
2043  std::vector<SortEntry> sortVec(temp.size());
2044  for (std::size_t ii = 0; ii < temp.size(); ++ii) {
2045  sortVec[ii].index = ii;
2046  sortVec[ii].val = temp[ii].along;
2047  } // ipt
2048  std::sort(sortVec.begin(), sortVec.end(), valsIncreasing);
2049  for (std::size_t ii = 0; ii < temp.size(); ++ii) {
2050  // overwrite the tp3d
2051  auto& tp3d = pfp.TP3Ds[indx[ii]];
2052  tp3d = temp[sortVec[ii].index];
2053  } // ii
2054  sf.NeedsUpdate = false;
2055  return true;
2056  } // SortSection
2057 
2060  TCSlice& slc,
2061  PFPStruct& pfp,
2062  bool prt)
2063  {
2064  // try to recover from a poor initial fit
2065  if (pfp.AlgMod[kSmallAngle]) return;
2066  if (pfp.SectionFits.size() != 1) return;
2067  if (pfp.TP3Ds.size() < 20) return;
2068  if (!CanSection(slc, pfp)) return;
2069 
2070  // make a copy
2071  auto p2 = pfp;
2072  // try two sections
2073  p2.SectionFits.resize(2);
2074  unsigned short halfPt = p2.TP3Ds.size() / 2;
2075  for (unsigned short ipt = halfPt; ipt < p2.TP3Ds.size(); ++ipt)
2076  p2.TP3Ds[ipt].SFIndex = 1;
2077  // Confirm that both sections can be reconstructed
2078  unsigned short toPt = Find3DRecoRange(slc, p2, 0, 3, 1);
2079  if (toPt > p2.TP3Ds.size()) return;
2080  toPt = Find3DRecoRange(slc, p2, halfPt, 3, 1);
2081  if (toPt > p2.TP3Ds.size()) return;
2082  if (!FitSection(detProp, slc, p2, 0) || !FitSection(detProp, slc, p2, 1)) {
2083  if (prt) {
2084  mf::LogVerbatim myprt("TC");
2085  myprt << "Recover failed MVI " << p2.MVI << " in TPC " << p2.TPCID.TPC;
2086  for (auto tid : p2.TjIDs)
2087  myprt << " T" << tid;
2088  } // prt
2089  return;
2090  }
2091  if (prt) mf::LogVerbatim("TC") << "Recover: P" << pfp.ID << " success";
2092  pfp = p2;
2093 
2094  } // Recover
2095 
2098  TCSlice& slc,
2099  PFPStruct& pfp,
2100  bool prt)
2101  {
2102  // Create and populate the TP3Ds vector. This function is called before the first
2103  // fit is done so the TP3D along variable can't be determined. It returns false
2104  // if a majority of the tj points in TjIDs are already assigned to a different pfp
2105  pfp.TP3Ds.clear();
2106  if (!pfp.TP3Ds.empty() || pfp.SectionFits.size() != 1) return false;
2107 
2108  // Look for TPs that are ~parallel to the wire plane and trajectory points
2109  // where the min/max Pos[1] value is not near an end.
2110  // number of Small Angle Tjs
2111  // stash the inflection point index in the TjUIDs vector
2112  pfp.TjUIDs.resize(pfp.TjIDs.size(), -1);
2113  // count TPs with charge in all of the Tjs
2114  float cnt = 0;
2115  // and the number of TPs available for use
2116  float avail = 0;
2117  // and the number of junk Tjs
2118  unsigned short nJunk = 0;
2119  unsigned short nSA = 0;
2120  for (unsigned short itj = 0; itj < pfp.TjIDs.size(); ++itj) {
2121  auto& tj = slc.tjs[pfp.TjIDs[itj] - 1];
2122  if (tj.AlgMod[kJunkTj]) ++nJunk;
2123  float posMin = 1E6;
2124  unsigned short iptMin = USHRT_MAX;
2125  float posMax = -1E6;
2126  unsigned short iptMax = USHRT_MAX;
2127  float aveAng = 0;
2128  float npwc = 0;
2129  for (unsigned short ipt = tj.EndPt[0]; ipt < tj.EndPt[1]; ++ipt) {
2130  auto& tp = tj.Pts[ipt];
2131  if (tp.Chg <= 0) continue;
2132  ++cnt;
2133  if (tp.InPFP > 0) continue;
2134  ++avail;
2135  if (tp.Pos[1] > posMax) {
2136  posMax = tp.Pos[1];
2137  iptMax = ipt;
2138  }
2139  if (tp.Pos[1] < posMin) {
2140  posMin = tp.Pos[1];
2141  iptMin = ipt;
2142  }
2143  aveAng += tp.Ang;
2144  ++npwc;
2145  } // ipt
2146  if (npwc == 0) continue;
2147  aveAng /= npwc;
2148  if (std::abs(aveAng) < 0.05) ++nSA;
2149  // No problem if the min/max points are near the ends
2150  if (iptMin > tj.EndPt[0] + 4 && iptMin < tj.EndPt[1] - 4) pfp.TjUIDs[itj] = iptMin;
2151  if (iptMax > tj.EndPt[0] + 4 && iptMax < tj.EndPt[1] - 4) pfp.TjUIDs[itj] = iptMax;
2152  } // tid
2153  if (avail < 0.8 * cnt) return false;
2154  // small angle trajectory?
2155  if (nSA > 1) pfp.AlgMod[kSmallAngle] = true;
2156  if (prt)
2157  mf::LogVerbatim("TC") << " P" << pfp.ID << " MVI " << pfp.MVI << " nJunkTj " << nJunk
2158  << " SmallAngle? " << pfp.AlgMod[kSmallAngle];
2159 
2160  if (pfp.AlgMod[kSmallAngle]) return MakeSmallAnglePFP(detProp, slc, pfp, prt);
2161 
2162  // Add the points associated with the Tjs that were used to create the PFP
2163  for (auto tid : pfp.TjIDs) {
2164  auto& tj = slc.tjs[tid - 1];
2165  // There is one TP for every hit in a junk Tj so we can skip one, if there is only one
2166  if (nJunk == 1 && tj.AlgMod[kJunkTj]) continue;
2167  // All of the Tj's may be junk, especially for those at very high angle, so the
2168  // X position of the TP's isn't high quality. Inflate the errors below.
2169  bool isJunk = tj.AlgMod[kJunkTj];
2170  for (unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2171  auto& tp = tj.Pts[ipt];
2172  if (tp.Chg <= 0) continue;
2173  if (tp.InPFP > 0) continue;
2174  ++avail;
2175  auto tp3d = CreateTP3D(detProp, slc, tid, ipt);
2176  if (tp3d.Flags[kTP3DBad]) continue;
2177  tp3d.SFIndex = 0;
2178  if (isJunk) tp3d.TPXErr2 *= 4;
2179  // We need to assume that all points are good or the first fit will fail
2180  tp3d.Flags[kTP3DGood] = true;
2181  pfp.TP3Ds.push_back(tp3d);
2182  } // ipt
2183  } // tid
2184  if (prt) mf::LogVerbatim("TC") << " has " << pfp.TP3Ds.size() << " TP3Ds";
2185  return true;
2186  } // MakeTP3Ds
2187 
2190  TCSlice& slc,
2191  PFPStruct& pfp,
2192  bool prt)
2193  {
2194  // Create and populate the TP3Ds vector for a small-angle track. The standard track fit
2195  // will fail for these tracks. The kSmallAngle AlgMod bit
2196  // is set true. Assume that the calling function, MakeTP3Ds, has decided that this is a
2197  // small-angle track.
2198 
2199  if (!tcc.useAlg[kSmallAngle]) return false;
2200  if (pfp.TjIDs.size() < 2) return false;
2201 
2202  std::vector<SortEntry> sortVec(pfp.TjIDs.size());
2203  unsigned short sbCnt = 0;
2204  for (unsigned short itj = 0; itj < pfp.TjIDs.size(); ++itj) {
2205  sortVec[itj].index = itj;
2206  auto& tj = slc.tjs[pfp.TjIDs[itj] - 1];
2207  sortVec[itj].val = NumPtsWithCharge(slc, tj, false);
2208  if (pfp.TjUIDs[itj] > 0) ++sbCnt;
2209  } // ipt
2210  std::sort(sortVec.begin(), sortVec.end(), valsDecreasing);
2211 
2212  // Decide whether to use the inflection points to add another section. Inflection
2213  // points must exist in the two longest Tjs
2214  unsigned short tlIndex = sortVec[0].index;
2215  unsigned short nlIndex = sortVec[1].index;
2216  auto& tlong = slc.tjs[pfp.TjIDs[tlIndex] - 1];
2217  auto& nlong = slc.tjs[pfp.TjIDs[nlIndex] - 1];
2218  bool twoSections = (sbCnt > 1 && pfp.TjUIDs[tlIndex] > 0 && pfp.TjUIDs[nlIndex] > 0);
2219  unsigned short tStartPt = tlong.EndPt[0];
2220  unsigned short tEndPt = tlong.EndPt[1];
2221  unsigned short nStartPt = nlong.EndPt[0];
2222  unsigned short nEndPt = nlong.EndPt[1];
2223  if (twoSections) {
2224  pfp.SectionFits.resize(2);
2225  tEndPt = pfp.TjUIDs[tlIndex];
2226  nEndPt = pfp.TjUIDs[nlIndex];
2227  if (prt) {
2228  mf::LogVerbatim myprt("TC");
2229  myprt << "MakeSmallAnglePFP: creating two sections using points";
2230  myprt << " T" << tlong.ID << "_" << tEndPt;
2231  myprt << " T" << nlong.ID << "_" << nEndPt;
2232  } // prt
2233  } // two Sections
2234  std::vector<Point3_t> sfEndPos;
2235  for (unsigned short isf = 0; isf < pfp.SectionFits.size(); ++isf) {
2236  // get the start and end TPs in this section
2237  auto& ltp0 = tlong.Pts[tStartPt];
2238  auto& ltp1 = tlong.Pts[tEndPt];
2239  auto& ntp0 = nlong.Pts[nStartPt];
2240  auto& ntp1 = nlong.Pts[nEndPt];
2241  // Get the 3D end points
2242  auto start = MakeTP3D(detProp, ltp0, ntp0);
2243  auto end = MakeTP3D(detProp, ltp1, ntp1);
2244  if (!start.Flags[kTP3DGood] || !end.Flags[kTP3DGood]) {
2245  std::cout << " Start/end fail in section " << isf << ". Add recovery code\n";
2246  return false;
2247  } // failure
2248  if (!InsideTPC(start.Pos, pfp.TPCID)) {
2249  mf::LogVerbatim("TC") << " Start is outside the TPC " << start.Pos[0] << " " << start.Pos[1]
2250  << " " << start.Pos[2];
2251  }
2252  if (!InsideTPC(end.Pos, pfp.TPCID)) {
2253  mf::LogVerbatim("TC") << " End is outside the TPC " << end.Pos[0] << " " << end.Pos[1]
2254  << " " << end.Pos[2];
2255  }
2256  if (isf == 0) sfEndPos.push_back(start.Pos);
2257  sfEndPos.push_back(end.Pos);
2258  auto& sf = pfp.SectionFits[isf];
2259  // Find the start and end positions
2260  for (unsigned short xyz = 0; xyz < 3; ++xyz) {
2261  sf.Dir[xyz] = end.Pos[xyz] - start.Pos[xyz];
2262  sf.Pos[xyz] = (end.Pos[xyz] + start.Pos[xyz]) / 2.;
2263  }
2264  SetMag(sf.Dir, 1.);
2265  sf.ChiDOF = 0.;
2266  sf.NPts = 0;
2267  // move the start/end point indices
2268  tStartPt = tEndPt + 1;
2269  tEndPt = tlong.EndPt[1];
2270  nStartPt = nEndPt + 1;
2271  nEndPt = nlong.EndPt[1];
2272  } // isf
2273  // Create TP3Ds
2274  // a temporary vector to hold TP3Ds for the second SectionFit
2275  std::vector<TP3D> sf2pts;
2276  for (unsigned short itj = 0; itj < sortVec.size(); ++itj) {
2277  int tid = pfp.TjIDs[sortVec[itj].index];
2278  // don't add points for the Tj that doesn't have an inflection point. It is
2279  // probably broken and would probably be put in the wrong section
2280  if (twoSections && pfp.TjUIDs[sortVec[itj].index] < 0) continue;
2281  auto& tj = slc.tjs[tid - 1];
2282  unsigned short sb = tj.EndPt[1];
2283  if (twoSections && pfp.TjUIDs[sortVec[itj].index] > 0) sb = pfp.TjUIDs[sortVec[itj].index];
2284  // count the number of good TPs in each section
2285  std::vector<double> npwc(pfp.SectionFits.size(), 0);
2286  for (unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2287  auto& tp = tj.Pts[ipt];
2288  if (tp.Chg <= 0) continue;
2289  if (ipt > sb) { ++npwc[1]; }
2290  else {
2291  ++npwc[0];
2292  }
2293  } // ipt
2294  double length = PosSep(sfEndPos[0], sfEndPos[1]);
2295  double step = length / npwc[0];
2296  double along = -length / 2;
2297  unsigned short sfi = 0;
2298  for (unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2299  auto& tp = tj.Pts[ipt];
2300  if (tp.Chg <= 0) continue;
2301  auto tp3d = CreateTP3D(detProp, slc, tid, ipt);
2302  if (tp3d.Flags[kTP3DBad]) continue;
2303  if (ipt == sb + 1) {
2304  sfi = 1;
2305  length = PosSep(sfEndPos[1], sfEndPos[2]);
2306  step = length / npwc[1];
2307  along = -length / 2;
2308  }
2309  tp3d.SFIndex = sfi;
2310  auto& sf = pfp.SectionFits[sfi];
2311  ++sf.NPts;
2312  tp3d.along = along;
2313  for (unsigned short xyz = 0; xyz < 3; ++xyz)
2314  tp3d.Pos[xyz] = sf.Pos[xyz] + along * sf.Dir[xyz];
2315  tp3d.Dir = sf.Dir;
2316  along += step;
2317  double delta = tp3d.Pos[0] - tp3d.TPX;
2318  sf.ChiDOF += delta * delta / tp3d.TPXErr2;
2319  // Assume that all points are good
2320  tp3d.Flags[kTP3DGood] = true;
2321  if (sfi == 0) { pfp.TP3Ds.push_back(tp3d); }
2322  else {
2323  sf2pts.push_back(tp3d);
2324  }
2325  } // ipt
2326  } // tid
2327  if (pfp.TP3Ds.size() < 4) return false;
2328  for (auto& sf : pfp.SectionFits) {
2329  if (sf.NPts < 5) return false;
2330  sf.ChiDOF /= (float)(sf.NPts - 4);
2331  } // sf
2332  if (!SortSection(pfp, 0)) return false;
2333  if (!sf2pts.empty()) {
2334  // append the points and sort
2335  pfp.TP3Ds.insert(pfp.TP3Ds.end(), sf2pts.begin(), sf2pts.end());
2336  if (!SortSection(pfp, 1)) return false;
2337  } // two sections
2338  pfp.Flags[kCanSection] = false;
2339  pfp.AlgMod[kSmallAngle] = true;
2340  if (prt) {
2341  mf::LogVerbatim("TC") << "Created SmallAngle P" << pfp.ID << " with " << pfp.TP3Ds.size()
2342  << " points in " << pfp.SectionFits.size() << " sections\n";
2343  }
2344  return true;
2345  } // MakeSmallAnglePFP
2346 
2348  void Reverse(PFPStruct& pfp)
2349  {
2350  // reverse the PFParticle
2351  std::reverse(pfp.TP3Ds.begin(), pfp.TP3Ds.end());
2352  std::reverse(pfp.SectionFits.begin(), pfp.SectionFits.end());
2353  for (std::size_t sfi = 0; sfi < pfp.SectionFits.size(); ++sfi) {
2354  auto& sf = pfp.SectionFits[sfi];
2355  // flip the direction vector
2356  for (unsigned short xyz = 0; xyz < 3; ++xyz)
2357  sf.Dir[xyz] *= -1;
2358  } // sf
2359  // correct the along variable
2360  for (auto& tp3d : pfp.TP3Ds)
2361  tp3d.along *= -1;
2362  std::swap(pfp.dEdx[0], pfp.dEdx[1]);
2363  std::swap(pfp.dEdxErr[0], pfp.dEdxErr[1]);
2364  std::swap(pfp.Vx3ID[0], pfp.Vx3ID[1]);
2365  std::swap(pfp.EndFlag[0], pfp.EndFlag[1]);
2366  } // Reverse
2367 
2370  {
2371  // Fills the mallTraj vector with trajectory points in the tpc and sorts
2372  // them by increasing X
2373 
2374  int cstat = slc.TPCID.Cryostat;
2375  int tpc = slc.TPCID.TPC;
2376 
2377  // define mallTraj
2378  slc.mallTraj.clear();
2379  Tj2Pt tj2pt;
2380  unsigned short cnt = 0;
2381 
2382  // try to reduce CPU time by not attempting to match tjs that are near muons
2383  bool muFuzzCut = (tcc.match3DCuts.size() > 6 && tcc.match3DCuts[6] > 0);
2384 
2385  float rms = tcc.match3DCuts[0];
2386  for (auto& tj : slc.tjs) {
2387  if (tj.AlgMod[kKilled] || tj.AlgMod[kHaloTj]) continue;
2388  // ignore already matched
2389  if (tj.AlgMod[kMat3D]) continue;
2390  geo::PlaneID planeID = DecodeCTP(tj.CTP);
2391  if ((int)planeID.Cryostat != cstat) continue;
2392  if ((int)planeID.TPC != tpc) continue;
2393  int plane = planeID.Plane;
2394  if (tj.ID <= 0) continue;
2395  unsigned short tjID = tj.ID;
2396  for (unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2397  auto& tp = tj.Pts[ipt];
2398  if (tp.Chg <= 0) continue;
2399  if (tp.Pos[0] < -0.4) continue;
2400  // ignore already matched
2401  if (tp.InPFP > 0) continue;
2402  if (muFuzzCut && tp.Environment[kEnvNearMuon]) continue;
2403  tj2pt.wire = std::nearbyint(tp.Pos[0]);
2404  ++cnt;
2405  // don't try matching if the wire doesn't exist
2406  if (!tcc.wireReadoutGeom->HasWire(geo::WireID(cstat, tpc, plane, tj2pt.wire))) continue;
2407  float xpos = detProp.ConvertTicksToX(tp.Pos[1] / tcc.unitsPerTick, plane, tpc, cstat);
2408  tj2pt.xlo = xpos - rms;
2409  tj2pt.xhi = xpos + rms;
2410  tj2pt.plane = plane;
2411  tj2pt.id = tjID;
2412  tj2pt.ipt = ipt;
2413  tj2pt.npts = tj.EndPt[1] - tj.EndPt[0] + 1;
2414  slc.mallTraj.push_back(tj2pt);
2415  } // tp
2416  } // tj
2417 
2418  // sort by increasing x
2419  std::vector<SortEntry> sortVec(slc.mallTraj.size());
2420  for (std::size_t ipt = 0; ipt < slc.mallTraj.size(); ++ipt) {
2421  // populate the sort vector
2422  sortVec[ipt].index = ipt;
2423  sortVec[ipt].val = slc.mallTraj[ipt].xlo;
2424  } // ipt
2425  // sort by increasing xlo
2426  std::sort(sortVec.begin(), sortVec.end(), valsIncreasing);
2427  // put slc.mallTraj into sorted order
2428  auto tallTraj = slc.mallTraj;
2429  for (std::size_t ii = 0; ii < sortVec.size(); ++ii)
2430  slc.mallTraj[ii] = tallTraj[sortVec[ii].index];
2431 
2432  } // FillmAllTraj
2433 
2436  const TrajPoint& itp,
2437  const TrajPoint& jtp)
2438  {
2439  // Make a 3D trajectory point using two 2D trajectory points. The TP3D Pos and Wire
2440  // variables are defined using itp. The SectionFit variables are un-defined
2441  TP3D tp3d;
2442  tp3d.TPIndex = 0;
2443  tp3d.TjID = 0;
2444  tp3d.CTP = itp.CTP;
2445  // assume failure
2446  tp3d.Flags[kTP3DGood] = false;
2447  tp3d.Dir = {{0.0, 0.0, 1.0}};
2448  tp3d.Pos = {{999.0, 999.0, 999.0}};
2449  geo::PlaneID iPlnID = DecodeCTP(itp.CTP);
2450  geo::PlaneID jPlnID = DecodeCTP(jtp.CTP);
2451  if (iPlnID == jPlnID) return tp3d;
2452  double upt = tcc.unitsPerTick;
2453  double ix = detProp.ConvertTicksToX(itp.Pos[1] / upt, iPlnID);
2454  double jx = detProp.ConvertTicksToX(jtp.Pos[1] / upt, jPlnID);
2455 
2456  // don't continue if the points are wildly far apart in X
2457  double dx = std::abs(ix - jx);
2458  if (dx > 20) return tp3d;
2459  tp3d.Pos[0] = (ix + jx) / 2;
2460  tp3d.TPX = ix;
2461  // Fake the error
2462  tp3d.TPXErr2 = dx;
2463  // determine the wire orientation and offsets using WireCoordinate
2464  // wire = yp * OrthY + zp * OrthZ - Wire0 = cs * yp + sn * zp - wire0
2465  // wire offset
2466  auto const& iplane = tcc.wireReadoutGeom->Plane(iPlnID);
2467  auto const& jplane = tcc.wireReadoutGeom->Plane(jPlnID);
2468  double iw0 = iplane.WireCoordinate(geo::Point_t{0, 0, 0});
2469  // cosine-like component
2470  double ics = iplane.WireCoordinate(geo::Point_t{0, 1, 0}) - iw0;
2471  // sine-like component
2472  double isn = iplane.WireCoordinate(geo::Point_t{0, 0, 1}) - iw0;
2473  double jw0 = jplane.WireCoordinate(geo::Point_t{0, 0, 0});
2474  double jcs = jplane.WireCoordinate(geo::Point_t{0, 1, 0}) - jw0;
2475  double jsn = jplane.WireCoordinate(geo::Point_t{0, 0, 1}) - jw0;
2476  double den = isn * jcs - ics * jsn;
2477  if (den == 0) return tp3d;
2478  double iPos0 = itp.Pos[0];
2479  double jPos0 = jtp.Pos[0];
2480  // Find the Z position of the intersection
2481  tp3d.Pos[2] = (jcs * (iPos0 - iw0) - ics * (jPos0 - jw0)) / den;
2482  // and the Y position
2483  bool useI = std::abs(ics) > std::abs(jcs);
2484  if (useI) { tp3d.Pos[1] = (iPos0 - iw0 - isn * tp3d.Pos[2]) / ics; }
2485  else {
2486  tp3d.Pos[1] = (jPos0 - jw0 - jsn * tp3d.Pos[2]) / jcs;
2487  }
2488 
2489  // Now find the direction. Protect against large angles first
2490  if (jtp.Dir[1] == 0) {
2491  // Going either in the +X direction or -X direction
2492  if (jtp.Dir[0] > 0) { tp3d.Dir[0] = 1; }
2493  else {
2494  tp3d.Dir[0] = -1;
2495  }
2496  tp3d.Dir[1] = 0;
2497  tp3d.Dir[2] = 0;
2498  return tp3d;
2499  } // jtp.Dir[1] == 0
2500 
2501  tp3d.Wire = iPos0;
2502 
2503  // make a copy of itp and shift it by many wires to avoid precision problems
2504  double itp2_0 = itp.Pos[0] + 100;
2505  double itp2_1 = itp.Pos[1];
2506  if (std::abs(itp.Dir[0]) > 0.01) itp2_1 += 100 * itp.Dir[1] / itp.Dir[0];
2507  // Create a second Point3 for the shifted point
2508  Point3_t pos2;
2509  // Find the X position corresponding to the shifted point
2510  pos2[0] = detProp.ConvertTicksToX(itp2_1 / upt, iPlnID);
2511  // Convert X to Ticks in the j plane and then to WSE units
2512  double jtp2Pos1 = detProp.ConvertXToTicks(pos2[0], jPlnID) * upt;
2513  // Find the wire position (Pos0) in the j plane that this corresponds to
2514  double jtp2Pos0 = (jtp2Pos1 - jtp.Pos[1]) * (jtp.Dir[0] / jtp.Dir[1]) + jtp.Pos[0];
2515  // Find the Y,Z position using itp2 and jtp2Pos0
2516  pos2[2] = (jcs * (itp2_0 - iw0) - ics * (jtp2Pos0 - jw0)) / den;
2517  if (useI) { pos2[1] = (itp2_0 - iw0 - isn * pos2[2]) / ics; }
2518  else {
2519  pos2[1] = (jtp2Pos0 - jw0 - jsn * pos2[2]) / jcs;
2520  }
2521  double sep = PosSep(tp3d.Pos, pos2);
2522  if (sep == 0) return tp3d;
2523  for (unsigned short ixyz = 0; ixyz < 3; ++ixyz)
2524  tp3d.Dir[ixyz] = (pos2[ixyz] - tp3d.Pos[ixyz]) / sep;
2525  tp3d.Flags[kTP3DGood] = true;
2526  return tp3d;
2527 
2528  } // MakeTP3D
2529 
2531  double DeltaAngle(const Vector3_t v1, const Vector3_t v2)
2532  {
2533  if (v1[0] == v2[0] && v1[1] == v2[1] && v1[2] == v2[2]) return 0;
2534  return acos(DotProd(v1, v2));
2535  }
2536 
2539  {
2540  // Finds the direction vector between the two points from p1 to p2
2541  Vector3_t dir;
2542  for (unsigned short xyz = 0; xyz < 3; ++xyz)
2543  dir[xyz] = p2[xyz] - p1[xyz];
2544  if (dir[0] == 0 && dir[1] == 0 && dir[2] == 0) return dir;
2545  if (!SetMag(dir, 1)) {
2546  dir[0] = 0;
2547  dir[1] = 0;
2548  dir[3] = 0;
2549  }
2550  return dir;
2551  } // PointDirection
2552 
2554  double PosSep(const Point3_t& pos1, const Point3_t& pos2)
2555  {
2556  return sqrt(PosSep2(pos1, pos2));
2557  } // PosSep
2558 
2560  double PosSep2(const Point3_t& pos1, const Point3_t& pos2)
2561  {
2562  // returns the separation distance^2 between two positions in 3D
2563  double d0 = pos1[0] - pos2[0];
2564  double d1 = pos1[1] - pos2[1];
2565  double d2 = pos1[2] - pos2[2];
2566  return d0 * d0 + d1 * d1 + d2 * d2;
2567  } // PosSep2
2568 
2570  bool SetMag(Vector3_t& v1, double mag)
2571  {
2572  double den = v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2];
2573  if (den == 0) return false;
2574  den = sqrt(den);
2575 
2576  v1[0] *= mag / den;
2577  v1[1] *= mag / den;
2578  v1[2] *= mag / den;
2579  return true;
2580  } // SetMag
2581 
2583  void FilldEdx(detinfo::DetectorClocksData const& clockData,
2584  detinfo::DetectorPropertiesData const& detProp,
2585  const TCSlice& slc,
2586  PFPStruct& pfp)
2587  {
2588  // Fills dE/dx variables in the pfp struct
2589 
2590  // don't attempt to find dE/dx at the end of a shower
2591  unsigned short numEnds = 2;
2592  if (pfp.PDGCode == 1111) numEnds = 1;
2593 
2594  // set dE/dx to 0 to indicate that a valid dE/dx is expected
2595  for (unsigned short end = 0; end < numEnds; ++end) {
2596  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane)
2597  pfp.dEdx[end][plane] = 0;
2598  } // end
2599 
2600  // square of the maximum length that is used for finding the average dE/dx
2601  float maxSep2 = 5 * tcc.wirePitch;
2602  maxSep2 *= maxSep2;
2603 
2604  for (unsigned short end = 0; end < numEnds; ++end) {
2605  std::vector<float> cnt(slc.nPlanes);
2606  short dir = 1 - 2 * end;
2607  auto endPos = PosAtEnd(pfp, end);
2608  for (std::size_t ii = 0; ii < pfp.TP3Ds.size(); ++ii) {
2609  unsigned short ipt;
2610  if (dir > 0) { ipt = ii; }
2611  else {
2612  ipt = pfp.TP3Ds.size() - ii - 1;
2613  }
2614  if (ipt >= pfp.TP3Ds.size()) break;
2615  auto& tp3d = pfp.TP3Ds[ipt];
2616  if (tp3d.Flags[kTP3DBad]) continue;
2617  if (PosSep2(tp3d.Pos, endPos) > maxSep2) break;
2618  // require good points
2619  if (!tp3d.Flags[kTP3DGood]) continue;
2620  float dedx = dEdx(clockData, detProp, slc, tp3d);
2621  if (dedx < 0.5) continue;
2622  unsigned short plane = DecodeCTP(tp3d.CTP).Plane;
2623  pfp.dEdx[end][plane] += dedx;
2624  ++cnt[plane];
2625  } // ii
2626  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
2627  if (cnt[plane] == 0) continue;
2628  pfp.dEdx[end][plane] /= cnt[plane];
2629  } // plane
2630  } // end
2631 
2632  } // FilldEdx
2633 
2636  detinfo::DetectorPropertiesData const& detProp,
2637  const TCSlice& slc,
2638  PFPStruct& pfp,
2639  float& dEdXAve,
2640  float& dEdXRms)
2641  {
2642  // Return a simple average of dE/dx and rms using ALL points in all planes, not
2643  // just those at the ends ala FilldEdx
2644  dEdXAve = -1.;
2645  dEdXRms = -1.;
2646 
2647  double sum = 0;
2648  double sum2 = 0;
2649  double cnt = 0;
2650  for (auto& tp3d : pfp.TP3Ds) {
2651  if (!tp3d.Flags[kTP3DGood] || tp3d.Flags[kTP3DBad]) continue;
2652  double dedx = dEdx(clockData, detProp, slc, tp3d);
2653  if (dedx < 0.5 || dedx > 80.) continue;
2654  sum += dedx;
2655  sum2 += dedx * dedx;
2656  ++cnt;
2657  } // tp3d
2658  if (cnt < 3) return;
2659  dEdXAve = sum / cnt;
2660  // Use a default rms of 30% of the average
2661  dEdXRms = 0.3 * dEdXAve;
2662  double arg = sum2 - cnt * dEdXAve * dEdXAve;
2663  if (arg < 0) return;
2664  dEdXRms = sqrt(arg) / (cnt - 1);
2665  // don't return a too-small rms
2666  double minRms = 0.05 * dEdXAve;
2667  if (dEdXRms < minRms) dEdXRms = minRms;
2668  } // Average_dEdX
2669 
2671  float dEdx(detinfo::DetectorClocksData const& clockData,
2672  detinfo::DetectorPropertiesData const& detProp,
2673  const TCSlice& slc,
2674  TP3D& tp3d)
2675  {
2676  if (!tp3d.Flags[kTP3DGood]) return 0;
2677  if (tp3d.TjID > (int)slc.slHits.size()) return 0;
2678  if (tp3d.TjID <= 0) return 0;
2679 
2680  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
2681  if (tp.Environment[kEnvOverlap]) return 0;
2682 
2683  double dQ = 0.;
2684  double time = 0;
2685  for (unsigned short ii = 0; ii < tp.Hits.size(); ++ii) {
2686  if (!tp.UseHit[ii]) continue;
2687  auto& hit = (*evt.allHits)[slc.slHits[tp.Hits[ii]].allHitsIndex];
2688  dQ += hit.Integral();
2689  } // ii
2690  time = tp.Pos[1] / tcc.unitsPerTick;
2691  geo::PlaneID plnID = DecodeCTP(tp.CTP);
2692  if (dQ == 0) return 0;
2693  double angleToVert = tcc.wireReadoutGeom->Plane(plnID).ThetaZ() - 0.5 * ::util::pi<>();
2694  double cosgamma =
2695  std::abs(std::sin(angleToVert) * tp3d.Dir[1] + std::cos(angleToVert) * tp3d.Dir[2]);
2696  if (cosgamma < 1.E-5) return 0;
2697  double dx = tcc.wireReadoutGeom->Plane(plnID).WirePitch() / cosgamma;
2698  double dQdx = dQ / dx;
2699  double t0 = 0;
2700  float dedx = tcc.caloAlg->dEdx_AREA(clockData, detProp, dQdx, time, plnID.Plane, t0);
2701  if (std::isinf(dedx)) dedx = 0;
2702  return dedx;
2703  } // dEdx
2704 
2707  const TCSlice& slc,
2708  int tjID,
2709  unsigned short tpIndex)
2710  {
2711  // create a TP3D with a single TP. Note that the SectionFit in which it
2712  // should be placed and the 3D position can't be determined until the the TP3D is
2713  // associated with a pfp. See SetSection()
2714 
2715  TP3D tp3d;
2716  tp3d.Flags[kTP3DBad] = true;
2717  if (tjID <= 0 || tjID > (int)slc.tjs.size()) return tp3d;
2718  auto& tj = slc.tjs[tjID - 1];
2719  if (tpIndex < tj.EndPt[0] || tpIndex > tj.EndPt[1]) return tp3d;
2720  tp3d.TjID = tjID;
2721  tp3d.TPIndex = tpIndex;
2722  auto& tp2 = tj.Pts[tp3d.TPIndex];
2723  auto plnID = DecodeCTP(tp2.CTP);
2724  tp3d.CTP = tp2.CTP;
2725  double tick = tp2.HitPos[1] / tcc.unitsPerTick;
2726  tp3d.TPX = detProp.ConvertTicksToX(tick, plnID);
2727  // Get the RMS of the TP in WSE units and convert to cm
2728  float rms = TPHitsRMSTime(slc, tp2, kAllHits) * tcc.wirePitch;
2729  // inflate the error for large angle TPs
2730  if (tp2.AngleCode == 1) rms *= 2;
2731  // a more careful treatment for long-pulse hits
2732  if (tp2.AngleCode > 1) {
2733  std::vector<unsigned int> hitMultiplet;
2734  for (std::size_t ii = 0; ii < tp2.Hits.size(); ++ii) {
2735  if (!tp2.UseHit[ii]) continue;
2736  GetHitMultiplet(slc, tp2.Hits[ii], hitMultiplet, true);
2737  if (hitMultiplet.size() > 1) break;
2738  } // ii
2739  rms = HitsRMSTime(slc, hitMultiplet, kAllHits) * tcc.wirePitch;
2740  // the returned RMS is closer to the FWHM, so divide by 2
2741  rms /= 2;
2742  } // tp2.AngleCode > 1
2743  tp3d.TPXErr2 = rms * rms;
2744  tp3d.Wire = tp2.Pos[0];
2745  // Can't declare it good since Pos and SFIndex aren't defined
2746  tp3d.Flags[kTP3DGood] = false;
2747  tp3d.Flags[kTP3DBad] = false;
2748  return tp3d;
2749  } // CreateTP3D
2750 
2752  bool SetSection(detinfo::DetectorPropertiesData const& detProp, PFPStruct& pfp, TP3D& tp3d)
2753  {
2754  // Determine which SectionFit this tp3d should reside in, then calculate
2755  // the 3D position and the distance from the center of the SectionFit
2756 
2757  if (tp3d.Wire < 0) return false;
2758  if (pfp.SectionFits.empty()) return false;
2759  if (pfp.SectionFits[0].Pos[0] == -10.0) return false;
2760  if (pfp.Flags[kSmallAngle]) return true;
2761 
2762  auto plnID = DecodeCTP(tp3d.CTP);
2763 
2764  if (pfp.SectionFits.size() == 1) { tp3d.SFIndex = 0; }
2765  else {
2766  // Find the section center that is closest to this point in the wire coordinate
2767  float best = 1E6;
2768  for (std::size_t sfi = 0; sfi < pfp.SectionFits.size(); ++sfi) {
2769  auto& sf = pfp.SectionFits[sfi];
2770  float sfWire =
2771  tcc.wireReadoutGeom->Plane(plnID).WireCoordinate(geo::Point_t{0, sf.Pos[1], sf.Pos[2]});
2772  float sep = std::abs(sfWire - tp3d.Wire);
2773  if (sep < best) {
2774  best = sep;
2775  tp3d.SFIndex = sfi;
2776  }
2777  } // sfi
2778  } // pfp.SectionFits.size() > 1
2779  auto& sf = pfp.SectionFits[tp3d.SFIndex];
2780  auto plnTP = MakeBareTP(detProp, sf.Pos, sf.Dir, tp3d.CTP);
2781  // the number of wires relative to the SectionFit center
2782  double dw = tp3d.Wire - plnTP.Pos[0];
2783  // dt/dW was stored in DeltaRMS
2784  double t = dw * plnTP.DeltaRMS;
2785  // define the 3D position
2786  for (unsigned short xyz = 0; xyz < 3; ++xyz)
2787  tp3d.Pos[xyz] = sf.Pos[xyz] + t * sf.Dir[xyz];
2788  tp3d.along = t;
2789  tp3d.Flags[kTP3DGood] = true;
2790  return true;
2791  } // SetSection
2792 
2794  float PointPull(const TP3D& tp3d)
2795  {
2796  // returns the pull that the tp3d will cause in the pfp section fit. This
2797  // currently only uses position but eventually will include charge
2798  return std::abs(tp3d.Pos[0] - tp3d.TPX) / sqrt(tp3d.TPXErr2);
2799  } // PointPull
2800 
2803  {
2804  // The calling function should define the size of pfp.TjIDs
2805  PFPStruct pfp;
2806  pfp.ID = slc.pfps.size() + 1;
2807  pfp.ParentUID = 0;
2808  pfp.TPCID = slc.TPCID;
2809  // initialize arrays for both ends
2810  if (slc.nPlanes < 4) {
2811  pfp.dEdx[0].resize(slc.nPlanes, -1);
2812  pfp.dEdx[1].resize(slc.nPlanes, -1);
2813  pfp.dEdxErr[0].resize(slc.nPlanes, -1);
2814  pfp.dEdxErr[1].resize(slc.nPlanes, -1);
2815  }
2816  // create a single section fit to hold the start/end positions and direction
2817  pfp.SectionFits.resize(1);
2818  return pfp;
2819  } // CreatePFP
2820 
2823  {
2824  // Ensure that all PFParticles have a start vertex. It is possible for
2825  // PFParticles to be attached to a 3D vertex that is later killed.
2826  if (!slc.isValid) return;
2827  if (slc.pfps.empty()) return;
2828 
2829  for (auto& pfp : slc.pfps) {
2830  if (pfp.ID == 0) continue;
2831  if (pfp.Vx3ID[0] > 0) continue;
2832  if (pfp.SectionFits.empty()) continue;
2833  Vtx3Store vx3;
2834  vx3.TPCID = pfp.TPCID;
2835  vx3.Vx2ID.resize(slc.nPlanes);
2836  // Flag it as a PFP vertex that isn't required to have matched 2D vertices
2837  vx3.Wire = -2;
2838  Point3_t startPos;
2839  if (pfp.TP3Ds.empty()) {
2840  // must be a neutrino pfp
2841  startPos = pfp.SectionFits[0].Pos;
2842  }
2843  else if (!pfp.TP3Ds.empty()) {
2844  // normal pfp
2845  startPos = pfp.TP3Ds[0].Pos;
2846  }
2847  vx3.X = startPos[0];
2848  vx3.Y = startPos[1];
2849  vx3.Z = startPos[2];
2850  vx3.ID = slc.vtx3s.size() + 1;
2851  vx3.Primary = false;
2852  ++evt.global3V_UID;
2853  vx3.UID = evt.global3V_UID;
2854  slc.vtx3s.push_back(vx3);
2855  pfp.Vx3ID[0] = vx3.ID;
2856  } // pfp
2857  } // PFPVertexCheck
2858 
2861  {
2862  /*
2863  This function reconciles vertices, PFParticles and slc, then
2864  defines the parent (j) - daughter (i) relationship and PDGCode. Here is a
2865  description of the conventions:
2866 
2867  V1 is the highest score 3D vertex in this tpcid so a neutrino PFParticle P1 is defined.
2868  V4 is a high-score vertex that has lower score than V1. It is declared to be a
2869  primary vertex because its score is higher than V5 and it is not associated with the
2870  neutrino interaction
2871  V6 was created to adhere to the convention that all PFParticles, in this case P9,
2872  be associated with a start vertex. There is no score for V6. P9 is it's own parent
2873  but is not a primary PFParticle.
2874 
2875  P1 - V1 - P2 - V2 - P4 - V3 - P5 V4 - P6 V6 - P9
2876  \ \
2877  P3 P7 - V5 - P8
2878 
2879  The PrimaryID in this table is the ID of the PFParticle that is attached to the
2880  primary vertex, which may or may not be a neutrino interaction vertex.
2881  The PrimaryID is returned by the PrimaryID function
2882  PFP parentID DtrIDs PrimaryID
2883  -----------------------------------
2884  P1 P1 P2, P3 P1
2885  P2 P1 P4 P2
2886  P3 P1 none P3
2887  P4 P2 P5 P2
2888  P5 P4 none P2
2889 
2890  P6 P6 none P6
2891  P7 P7 P8 P7
2892 
2893  P9 P9 none 0
2894 
2895  */
2896  if (slc.pfps.empty()) return;
2897  if (tcc.modes[kTestBeam]) return;
2898 
2899  int neutrinoPFPID = 0;
2900  for (auto& pfp : slc.pfps) {
2901  if (pfp.ID == 0) continue;
2902  if (!tcc.modes[kTestBeam] && neutrinoPFPID == 0 && (pfp.PDGCode == 12 || pfp.PDGCode == 14)) {
2903  neutrinoPFPID = pfp.ID;
2904  break;
2905  }
2906  } // pfp
2907 
2908  // define the end vertex if the Tjs have end vertices
2909  constexpr unsigned short end1 = 1;
2910  for (auto& pfp : slc.pfps) {
2911  if (pfp.ID == 0) continue;
2912  // already done?
2913  if (pfp.Vx3ID[end1] > 0) continue;
2914  // ignore shower-like pfps
2915  if (IsShowerLike(slc, pfp.TjIDs)) continue;
2916  // count 2D -> 3D matched vertices
2917  unsigned short cnt3 = 0;
2918  unsigned short vx3id = 0;
2919  // list of unmatched 2D vertices that should be merged
2920  std::vector<int> vx2ids;
2921  for (auto tjid : pfp.TjIDs) {
2922  auto& tj = slc.tjs[tjid - 1];
2923  if (tj.VtxID[end1] == 0) continue;
2924  auto& vx2 = slc.vtxs[tj.VtxID[end1] - 1];
2925  if (vx2.Vx3ID == 0) {
2926  if (vx2.Topo == 1 && vx2.NTraj == 2) vx2ids.push_back(vx2.ID);
2927  continue;
2928  }
2929  if (vx3id == 0) vx3id = vx2.Vx3ID;
2930  if (vx2.Vx3ID == vx3id) ++cnt3;
2931  } // tjid
2932  if (cnt3 > 1) {
2933  // ensure it isn't attached at the other end
2934  if (pfp.Vx3ID[1 - end1] == vx3id) continue;
2935  pfp.Vx3ID[end1] = vx3id;
2936  } // cnt3 > 1
2937  } // pfp
2938 
2939  // Assign a PDGCode to each PFParticle and look for a parent
2940  for (auto& pfp : slc.pfps) {
2941  if (pfp.ID == 0) continue;
2942  // skip a neutrino PFParticle
2943  if (pfp.PDGCode == 12 || pfp.PDGCode == 14 || pfp.PDGCode == 22) continue;
2944  // Define a PFP parent if there are two or more Tjs that are daughters of
2945  // Tjs that are used by the same PFParticle
2946  int pfpParentID = INT_MAX;
2947  unsigned short nParent = 0;
2948  for (auto tjid : pfp.TjIDs) {
2949  auto& tj = slc.tjs[tjid - 1];
2950  if (tj.ParentID <= 0) continue;
2951  auto parPFP = GetAssns(slc, "T", tj.ParentID, "P");
2952  if (parPFP.empty()) continue;
2953  if (pfpParentID == INT_MAX) pfpParentID = parPFP[0];
2954  if (parPFP[0] == pfpParentID) ++nParent;
2955  } // ii
2956  if (nParent > 1) {
2957  auto& ppfp = slc.pfps[pfpParentID - 1];
2958  // set the parent UID
2959  pfp.ParentUID = ppfp.UID;
2960  // add to the parent daughters list
2961  ppfp.DtrUIDs.push_back(pfp.UID);
2962  } // nParent > 1
2963  } // ipfp
2964  // associate primary PFParticles with a neutrino PFParticle
2965  if (neutrinoPFPID > 0) {
2966  auto& neutrinoPFP = slc.pfps[neutrinoPFPID - 1];
2967  int vx3id = neutrinoPFP.Vx3ID[1];
2968  for (auto& pfp : slc.pfps) {
2969  if (pfp.ID == 0 || pfp.ID == neutrinoPFPID) continue;
2970  if (pfp.Vx3ID[0] != vx3id) continue;
2971  pfp.ParentUID = (size_t)neutrinoPFPID;
2972  neutrinoPFP.DtrUIDs.push_back(pfp.ID);
2973  if (pfp.PDGCode == 111) neutrinoPFP.PDGCode = 12;
2974  } // pfp
2975  } // neutrino PFP exists
2976  } // DefinePFPParents
2977 
2979  bool StorePFP(TCSlice& slc, PFPStruct& pfp)
2980  {
2981  // stores the PFParticle in the slice
2982  bool neutrinoPFP = (pfp.PDGCode == 12 || pfp.PDGCode == 14);
2983  if (!neutrinoPFP) {
2984  if (pfp.TjIDs.empty()) return false;
2985  if (pfp.PDGCode != 1111 && pfp.TP3Ds.size() < 2) return false;
2986  }
2987 
2988  if (pfp.Flags[kSmallAngle]) {
2989  // Make the PFP -> TP assn
2990  for (auto& tp3d : pfp.TP3Ds) {
2991  if (tp3d.TPIndex != USHRT_MAX) slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex].InPFP = pfp.ID;
2992  }
2993  }
2994 
2995  // ensure that the InPFP flag is set
2996  unsigned short nNotSet = 0;
2997  for (auto& tp3d : pfp.TP3Ds) {
2998  if (tp3d.Flags[kTP3DBad]) continue;
2999  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
3000  if (tp.InPFP != pfp.ID) ++nNotSet;
3001  } // tp3d
3002  if (nNotSet > 0) return false;
3003  // check the ID and correct it if it is wrong
3004  if (pfp.ID != (int)slc.pfps.size() + 1) pfp.ID = slc.pfps.size() + 1;
3005  ++evt.globalP_UID;
3006  pfp.UID = evt.globalP_UID;
3007 
3008  // set the 3D match flag
3009  for (auto tjid : pfp.TjIDs) {
3010  auto& tj = slc.tjs[tjid - 1];
3011  tj.AlgMod[kMat3D] = true;
3012  } // tjid
3013 
3014  slc.pfps.push_back(pfp);
3015  return true;
3016  } // StorePFP
3017 
3019  bool InsideFV(const TCSlice& slc, const PFPStruct& pfp, unsigned short end)
3020  {
3021  // returns true if the end of the pfp is inside the fiducial volume of the TPC
3022  if (pfp.ID <= 0) return false;
3023  if (end > 1) return false;
3024  if (pfp.SectionFits.empty()) return false;
3025  // require that the points are sorted which ensures that the start and end points
3026  // are the first and last points in the TP3Ds vector
3027  if (pfp.Flags[kNeedsUpdate]) return false;
3028  bool neutrinoPFP = pfp.PDGCode == 12 || pfp.PDGCode == 14;
3029 
3030  float abit = 5;
3031  Point3_t pos;
3032  if (neutrinoPFP) { pos = pfp.SectionFits[0].Pos; }
3033  else if (end == 0) {
3034  pos = pfp.TP3Ds[0].Pos;
3035  }
3036  else {
3037  pos = pfp.TP3Ds[pfp.TP3Ds.size() - 1].Pos;
3038  }
3039  return (pos[0] > slc.xLo + abit && pos[0] < slc.xHi - abit && pos[1] > slc.yLo + abit &&
3040  pos[1] < slc.yHi - abit && pos[2] > slc.zLo + abit && pos[2] < slc.zHi - abit);
3041 
3042  } // InsideFV
3043 
3045  bool InsideTPC(const Point3_t& pos, geo::TPCID& inTPCID)
3046  {
3047  // determine which TPC this point is in. This function returns false
3048  // if the point is not inside any TPC
3049  float abit = 5;
3050  for (const geo::TPCGeo& TPC : tcc.geom->Iterate<geo::TPCGeo>()) {
3051  auto const world = TPC.GetCenter();
3052  // reduce the active area of the TPC by a bit to be consistent with FillWireHitRange
3053  if (pos[0] < world.X() - TPC.HalfWidth() + abit) continue;
3054  if (pos[0] > world.X() + TPC.HalfWidth() - abit) continue;
3055  if (pos[1] < world.Y() - TPC.HalfHeight() + abit) continue;
3056  if (pos[1] > world.Y() + TPC.HalfHeight() - abit) continue;
3057  if (pos[2] < world.Z() - TPC.Length() / 2 + abit) continue;
3058  if (pos[2] > world.Z() + TPC.Length() / 2 - abit) continue;
3059  inTPCID = TPC.ID();
3060  return true;
3061  } // tpcid
3062  return false;
3063  } // InsideTPC
3064 
3066  void FindAlongTrans(Point3_t pos1, Vector3_t dir1, Point3_t pos2, Point2_t& alongTrans)
3067  {
3068  // Calculate the distance along and transvers to the direction vector from pos1 to pos2
3069  alongTrans[0] = 0;
3070  alongTrans[1] = 0;
3071  if (pos1[0] == pos2[0] && pos1[1] == pos2[1] && pos1[2] == pos2[2]) return;
3072  auto ptDir = PointDirection(pos1, pos2);
3073  SetMag(dir1, 1.0);
3074  double costh = DotProd(dir1, ptDir);
3075  if (costh > 1) costh = 1;
3076  double sep = PosSep(pos1, pos2);
3077  alongTrans[0] = costh * sep;
3078  double sinth = sqrt(1 - costh * costh);
3079  alongTrans[1] = sinth * sep;
3080  } // FindAlongTrans
3081 
3084  Vector3_t p1Dir,
3085  Point3_t p2,
3086  Vector3_t p2Dir,
3087  Point3_t& intersect,
3088  float& doca)
3089  {
3090  // Point - vector version
3091  Point3_t p1End, p2End;
3092  for (unsigned short xyz = 0; xyz < 3; ++xyz) {
3093  p1End[xyz] = p1[xyz] + 10 * p1Dir[xyz];
3094  p2End[xyz] = p2[xyz] + 10 * p2Dir[xyz];
3095  }
3096  return LineLineIntersect(p1, p1End, p2, p2End, intersect, doca);
3097  } // PointDirIntersect
3098 
3101  Point3_t p2,
3102  Point3_t p3,
3103  Point3_t p4,
3104  Point3_t& intersect,
3105  float& doca)
3106  {
3107  /*
3108  Calculate the line segment PaPb that is the shortest route between
3109  two lines P1P2 and P3P4. Calculate also the values of mua and mub where
3110  Pa = P1 + mua (P2 - P1)
3111  Pb = P3 + mub (P4 - P3)
3112  Return FALSE if no solution exists.
3113  http://paulbourke.net/geometry/pointlineplane/
3114  */
3115 
3116  Point3_t p13, p43, p21;
3117  double d1343, d4321, d1321, d4343, d2121;
3118  double numer, denom;
3119  constexpr double EPS = std::numeric_limits<double>::min();
3120 
3121  p13[0] = p1[0] - p3[0];
3122  p13[1] = p1[1] - p3[1];
3123  p13[2] = p1[2] - p3[2];
3124  p43[0] = p4[0] - p3[0];
3125  p43[1] = p4[1] - p3[1];
3126  p43[2] = p4[2] - p3[2];
3127  if (std::abs(p43[0]) < EPS && std::abs(p43[1]) < EPS && std::abs(p43[2]) < EPS) return (false);
3128  p21[0] = p2[0] - p1[0];
3129  p21[1] = p2[1] - p1[1];
3130  p21[2] = p2[2] - p1[2];
3131  if (std::abs(p21[0]) < EPS && std::abs(p21[1]) < EPS && std::abs(p21[2]) < EPS) return (false);
3132 
3133  d1343 = p13[0] * p43[0] + p13[1] * p43[1] + p13[2] * p43[2];
3134  d4321 = p43[0] * p21[0] + p43[1] * p21[1] + p43[2] * p21[2];
3135  d1321 = p13[0] * p21[0] + p13[1] * p21[1] + p13[2] * p21[2];
3136  d4343 = p43[0] * p43[0] + p43[1] * p43[1] + p43[2] * p43[2];
3137  d2121 = p21[0] * p21[0] + p21[1] * p21[1] + p21[2] * p21[2];
3138 
3139  denom = d2121 * d4343 - d4321 * d4321;
3140  if (std::abs(denom) < EPS) return (false);
3141  numer = d1343 * d4321 - d1321 * d4343;
3142 
3143  double mua = numer / denom;
3144  double mub = (d1343 + d4321 * mua) / d4343;
3145 
3146  intersect[0] = p1[0] + mua * p21[0];
3147  intersect[1] = p1[1] + mua * p21[1];
3148  intersect[2] = p1[2] + mua * p21[2];
3149  Point3_t pb;
3150  pb[0] = p3[0] + mub * p43[0];
3151  pb[1] = p3[1] + mub * p43[1];
3152  pb[2] = p3[2] + mub * p43[2];
3153  doca = PosSep(intersect, pb);
3154  // average the closest points
3155  for (unsigned short xyz = 0; xyz < 3; ++xyz)
3156  intersect[xyz] += pb[xyz];
3157  for (unsigned short xyz = 0; xyz < 3; ++xyz)
3158  intersect[xyz] /= 2;
3159  return true;
3160  } // LineLineIntersect
3161 
3164  const TCSlice& slc,
3165  Point3_t pos1,
3166  Point3_t pos2)
3167  {
3168  // Step between pos1 and pos2 and find the fraction of the points that have nearby hits
3169  // in each plane. This function returns -1 if something is fishy, but this doesn't mean
3170  // that there is no charge. Note that there is no check for charge precisely at the pos1 and pos2
3171  // positions
3172  float sep = PosSep(pos1, pos2);
3173  if (sep == 0) return -1;
3174  unsigned short nstep = sep / tcc.wirePitch;
3175  auto dir = PointDirection(pos1, pos2);
3176  float sum = 0;
3177  float cnt = 0;
3178  TrajPoint tp;
3179  for (unsigned short step = 0; step < nstep; ++step) {
3180  for (unsigned short xyz = 0; xyz < 3; ++xyz)
3181  pos1[xyz] += tcc.wirePitch * dir[xyz];
3182  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
3183  tp.CTP = EncodeCTP(slc.TPCID.Cryostat, slc.TPCID.TPC, plane);
3184  geo::PlaneID const planeID{slc.TPCID, plane};
3185  tp.Pos[0] =
3186  tcc.wireReadoutGeom->Plane(planeID).WireCoordinate(geo::Point_t{0, pos1[1], pos1[2]});
3187  tp.Pos[1] = detProp.ConvertXToTicks(pos1[0], planeID) * tcc.unitsPerTick;
3188  ++cnt;
3189  if (SignalAtTp(tp)) ++sum;
3190  } // plane
3191  } // step
3192  if (cnt == 0) return -1;
3193  return sum / cnt;
3194  } // ChgFracBetween
3195 
3198  const TCSlice& slc,
3199  const PFPStruct& pfp,
3200  unsigned short end)
3201  {
3202  // returns the charge fraction near the end of the pfp. Note that this function
3203  // assumes that there is only one Tj in a plane.
3204  if (pfp.ID == 0) return 0;
3205  if (pfp.TjIDs.empty()) return 0;
3206  if (end > 1) return 0;
3207  if (pfp.TPCID != slc.TPCID) return 0;
3208  if (pfp.SectionFits.empty()) return 0;
3209 
3210  float sum = 0;
3211  float cnt = 0;
3212  // keep track of the lowest value and maybe reject it
3213  float lo = 1;
3214  float hi = 0;
3215  auto pos3 = PosAtEnd(pfp, end);
3216  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
3217  CTP_t inCTP = EncodeCTP(pfp.TPCID.Cryostat, pfp.TPCID.TPC, plane);
3218  std::vector<int> tjids(1);
3219  for (auto tjid : pfp.TjIDs) {
3220  auto& tj = slc.tjs[tjid - 1];
3221  if (tj.CTP != inCTP) continue;
3222  tjids[0] = tjid;
3223  Point2_t pos2;
3224  geo::PlaneID planeID = geo::PlaneID{pfp.TPCID, plane};
3225  pos2[0] =
3226  tcc.wireReadoutGeom->Plane(planeID).WireCoordinate(geo::Point_t{0, pos3[1], pos3[2]});
3227  if (pos2[0] < -0.4) continue;
3228  // check for dead wires
3229  unsigned int wire = std::nearbyint(pos2[0]);
3230  if (wire > slc.nWires[plane]) continue;
3231  if (slc.wireHitRange[plane][wire].first == UINT_MAX) continue;
3232  pos2[1] = detProp.ConvertXToTicks(pos3[0], planeID) * tcc.unitsPerTick;
3233  float cf = ChgFracNearPos(slc, pos2, tjids);
3234  if (cf < lo) lo = cf;
3235  if (cf > hi) hi = cf;
3236  sum += cf;
3237  ++cnt;
3238  } // tjid
3239  } // plane
3240  if (cnt == 0) return 0;
3241  if (cnt > 1 && lo < 0.3 && hi > 0.8) {
3242  sum -= lo;
3243  --cnt;
3244  }
3245  return sum / cnt;
3246  } // ChgFracNearEnd
3247 
3249  Vector3_t DirAtEnd(const PFPStruct& pfp, unsigned short end)
3250  {
3251  if (end > 1 || pfp.SectionFits.empty()) return {{0., 0., 0.}};
3252  if (end == 0) return pfp.SectionFits[0].Dir;
3253  return pfp.SectionFits[pfp.SectionFits.size() - 1].Dir;
3254  } // PosAtEnd
3255 
3257  Point3_t PosAtEnd(const PFPStruct& pfp, unsigned short end)
3258  {
3259  if (end > 1 || pfp.SectionFits.empty()) return {{0., 0., 0.}};
3260  // handle a neutrino pfp that doesn't have any TP3Ds
3261  if (pfp.TP3Ds.empty()) return pfp.SectionFits[0].Pos;
3262  if (end == 0) return pfp.TP3Ds[0].Pos;
3263  return pfp.TP3Ds[pfp.TP3Ds.size() - 1].Pos;
3264  } // PosAtEnd
3265 
3267  float Length(const PFPStruct& pfp)
3268  {
3269  if (pfp.TP3Ds.empty()) return 0;
3270  return PosSep(pfp.TP3Ds[0].Pos, pfp.TP3Ds[pfp.TP3Ds.size() - 1].Pos);
3271  } // Length
3272 
3274  bool SectionStartEnd(const PFPStruct& pfp,
3275  unsigned short sfIndex,
3276  unsigned short& startPt,
3277  unsigned short& endPt)
3278  {
3279  // this assumes that the TP3Ds vector is sorted
3280  startPt = USHRT_MAX;
3281  endPt = USHRT_MAX;
3282  if (sfIndex >= pfp.SectionFits.size()) return false;
3283 
3284  bool first = true;
3285  for (std::size_t ipt = 0; ipt < pfp.TP3Ds.size(); ++ipt) {
3286  auto& tp3d = pfp.TP3Ds[ipt];
3287  if (tp3d.SFIndex < sfIndex) continue;
3288  if (first) {
3289  first = false;
3290  startPt = ipt;
3291  } // first
3292  if (tp3d.SFIndex > sfIndex) break;
3293  endPt = ipt;
3294  } // ipt
3295  return true;
3296 
3297  } // SectionStartEnd
3298 
3300  unsigned short FarEnd(const PFPStruct& pfp, const Point3_t& pos)
3301  {
3302  // Returns the end (0 or 1) of the pfp that is furthest away from the position pos
3303  if (pfp.ID == 0) return 0;
3304  if (pfp.TP3Ds.empty()) return 0;
3305  auto& pos0 = pfp.TP3Ds[0].Pos;
3306  auto& pos1 = pfp.TP3Ds[pfp.TP3Ds.size() - 1].Pos;
3307  if (PosSep2(pos1, pos) > PosSep2(pos0, pos)) return 1;
3308  return 0;
3309  } // FarEnd
3310 
3313  detinfo::DetectorPropertiesData const& detProp,
3314  const TCSlice& slc,
3315  PFPStruct& pfp)
3316  {
3317  // returns a vote using PDG code assignments from dE/dx. A PDGCode of -1 is
3318  // returned if there was a failure and returns 0 if no decision can be made
3319  if (pfp.TP3Ds.empty()) return -1;
3320 
3321  // try to do better using dE/dx
3322  float dEdXAve = 0;
3323  float dEdXRms = 0;
3324  Average_dEdX(clockData, detProp, slc, pfp, dEdXAve, dEdXRms);
3325  if (dEdXAve < 0) return 0;
3326  // looks like a proton if dE/dx is high and the rms is low
3327  dEdXRms /= dEdXAve;
3328  float length = Length(pfp);
3329  float mcsmom = 0;
3330  float chgrms = 0;
3331  float cnt = 0;
3332  for (auto tjid : pfp.TjIDs) {
3333  auto& tj = slc.tjs[tjid - 1];
3334  float el = ElectronLikelihood(slc, tj);
3335  if (el <= 0) continue;
3336  mcsmom += MCSMom(slc, tj);
3337  chgrms += tj.ChgRMS;
3338  ++cnt;
3339  } // tjid
3340  if (cnt < 2) return 0;
3341  mcsmom /= cnt;
3342  chgrms /= cnt;
3343  int vote = 0;
3344  // call anything longer than 150 cm a muon
3345  if (length > 150) vote = 13;
3346  // or shorter with low dE/dx and really straight
3347  if (vote == 0 && length > 50 && dEdXAve < 2.5 && mcsmom > 500) vote = 13;
3348  // protons have high dE/dx, high MCSMom and low charge rms
3349  if (vote == 0 && dEdXAve > 3.0 && mcsmom > 200 && chgrms < 0.4) vote = 2212;
3350  // electrons have low MCSMom and large charge RMS
3351  if (vote == 0 && mcsmom < 50 && chgrms > 0.4) vote = 11;
3352  return vote;
3353  } // PDGCodeVote
3354 
3357  detinfo::DetectorPropertiesData const& detProp,
3358  std::string someText,
3359  const TCSlice& slc,
3360  const PFPStruct& pfp,
3361  short printPts)
3362  {
3363  if (pfp.TP3Ds.empty()) return;
3364  mf::LogVerbatim myprt("TC");
3365  myprt << someText << " pfp P" << pfp.ID << " MVI " << pfp.MVI;
3366  for (auto tid : pfp.TjIDs)
3367  myprt << " T" << tid;
3368  myprt << " Flags: CanSection? " << pfp.Flags[kCanSection];
3369  myprt << " NeedsUpdate? " << pfp.Flags[kNeedsUpdate];
3370  myprt << " Algs:";
3371  for (unsigned short ib = 0; ib < pAlgModSize; ++ib) {
3372  if (pfp.AlgMod[ib]) myprt << " " << AlgBitNames[ib];
3373  } // ib
3374  myprt << "\n";
3375  if (!pfp.SectionFits.empty()) {
3376  myprt << someText
3377  << " SFI ________Pos________ ________Dir_______ _____EndPos________ ChiDOF NPts "
3378  "NeedsUpdate?\n";
3379  for (std::size_t sfi = 0; sfi < pfp.SectionFits.size(); ++sfi) {
3380  myprt << someText << std::setw(4) << sfi;
3381  auto& sf = pfp.SectionFits[sfi];
3382  myprt << std::fixed << std::setprecision(1);
3383  unsigned short startPt = 0, endPt = 0;
3384  if (SectionStartEnd(pfp, sfi, startPt, endPt)) {
3385  auto& start = pfp.TP3Ds[startPt].Pos;
3386  myprt << std::setw(7) << start[0] << std::setw(7) << start[1] << std::setw(7) << start[2];
3387  }
3388  else {
3389  myprt << " Invalid";
3390  }
3391  myprt << std::fixed << std::setprecision(2);
3392  myprt << std::setw(7) << sf.Dir[0] << std::setw(7) << sf.Dir[1] << std::setw(7)
3393  << sf.Dir[2];
3394  myprt << std::fixed << std::setprecision(1);
3395  if (endPt < pfp.TP3Ds.size()) {
3396  auto& end = pfp.TP3Ds[endPt].Pos;
3397  myprt << std::setw(7) << end[0] << std::setw(7) << end[1] << std::setw(7) << end[2];
3398  }
3399  else {
3400  myprt << " Invalid";
3401  }
3402  myprt << std::setprecision(1) << std::setw(6) << sf.ChiDOF;
3403  myprt << std::setw(6) << sf.NPts;
3404  myprt << std::setw(6) << sf.NeedsUpdate;
3405  myprt << "\n";
3406  } // sec
3407  } // SectionFits
3408  if (printPts < 0) {
3409  // print the head if we print all points
3410  myprt << someText << " Note: GBH = TP3D Flags. G = Good, B = Bad, H = High dE/dx \n";
3411  myprt
3412  << someText
3413  << " ipt SFI ________Pos________ Delta Pull GBH Path along dE/dx S? T_ipt_P:W:T\n";
3414  }
3415  unsigned short fromPt = 0;
3416  unsigned short toPt = pfp.TP3Ds.size() - 1;
3417  if (printPts >= 0) fromPt = toPt;
3418  // temp kink angle for each point
3419  std::vector<float> dang(pfp.TP3Ds.size(), -1);
3420  for (unsigned short ipt = fromPt; ipt <= toPt; ++ipt) {
3421  auto tp3d = pfp.TP3Ds[ipt];
3422  myprt << someText << std::setw(4) << ipt;
3423  myprt << std::setw(4) << tp3d.SFIndex;
3424  myprt << std::fixed << std::setprecision(1);
3425  myprt << std::setw(7) << tp3d.Pos[0] << std::setw(7) << tp3d.Pos[1] << std::setw(7)
3426  << tp3d.Pos[2];
3427  myprt << std::setprecision(1) << std::setw(6) << (tp3d.Pos[0] - tp3d.TPX);
3428  float pull = PointPull(tp3d);
3429  myprt << std::setprecision(1) << std::setw(6) << pull;
3430  myprt << std::setw(3) << tp3d.Flags[kTP3DGood] << tp3d.Flags[kTP3DBad];
3431  myprt << std::setw(7) << std::setprecision(1) << PosSep(tp3d.Pos, pfp.TP3Ds[0].Pos);
3432  myprt << std::setw(7) << std::setprecision(1) << tp3d.along;
3433  myprt << std::setw(6) << std::setprecision(2) << dEdx(clockData, detProp, slc, tp3d);
3434  // print SignalAtTP in each plane
3435  myprt << " ";
3436  for (unsigned short plane = 0; plane < slc.nPlanes; ++plane) {
3437  CTP_t inCTP = EncodeCTP(pfp.TPCID.Cryostat, pfp.TPCID.TPC, plane);
3438  auto tp = MakeBareTP(detProp, tp3d.Pos, inCTP);
3439  myprt << " " << SignalAtTp(tp);
3440  } // plane
3441  if (tp3d.TjID > 0) {
3442  auto& tp = slc.tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
3443  myprt << " T" << tp3d.TjID << "_" << tp3d.TPIndex << "_" << PrintPos(tp) << " "
3444  << TPEnvString(tp);
3445  }
3446  else {
3447  myprt << " UNDEFINED";
3448  }
3449  myprt << "\n";
3450  } // ipt
3451  } // PrintTP3Ds
3452 } // namespace
Expect tracks entering from the front face. Don&#39;t create neutrino PFParticles.
Definition: DataStructs.h:523
Float_t x
Definition: compare.C:6
calo::CalorimetryAlg * caloAlg
Definition: DataStructs.h:569
code to link reconstructed objects back to the MC truth information
Vector2_t Dir
Definition: DataStructs.h:151
Point2_t Pos
Definition: DataStructs.h:150
unsigned short NPts
Definition: DataStructs.h:242
short MCSMom(const TCSlice &slc, const std::vector< int > &tjIDs)
Definition: Utils.cxx:3381
MaybeLogger_< ELseverityLevel::ELsev_info, true > LogVerbatim
std::vector< Trajectory > tjs
vector of all trajectories in each plane
Definition: DataStructs.h:662
float Length(const PFPStruct &pfp)
Definition: PFPUtils.cxx:3267
float Wire
Definition: DataStructs.h:252
bool dbgStitch
debug PFParticle stitching
Definition: DataStructs.h:595
void Average_dEdX(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp, float &dEdXAve, float &dEdXRms)
Definition: PFPUtils.cxx:2635
void MakePFParticles(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, std::vector< MatchStruct > matVec, unsigned short matVec_Iter)
Definition: PFPUtils.cxx:266
std::array< std::vector< float >, 2 > dEdxErr
Definition: DataStructs.h:284
bool MakeVertexObsolete(std::string fcnLabel, TCSlice &slc, VtxStore &vx2, bool forceKill)
Definition: TCVertex.cxx:2698
int TjID
ID of the trajectory -> TP3D assn.
Definition: DataStructs.h:254
bool ReconcileTPs(TCSlice &slc, PFPStruct &pfp, bool prt)
Definition: PFPUtils.cxx:424
bool InsideFV(const TCSlice &slc, const PFPStruct &pfp, unsigned short end)
Definition: PFPUtils.cxx:3019
static unsigned int kWire
const std::vector< std::string > AlgBitNames
Definition: DataStructs.cxx:16
CTP_t CTP
Cryostat, TPC, Plane code.
Definition: DataStructs.h:148
bool InsideTPC(const Point3_t &pos, geo::TPCID &inTPCID)
Definition: PFPUtils.cxx:3045
const geo::WireReadoutGeom * wireReadoutGeom
Definition: DataStructs.h:568
std::array< double, 3 > Point3_t
Definition: DataStructs.h:39
bool SignalAtTp(TrajPoint &tp)
Definition: Utils.cxx:1964
std::vector< float > vtx3DCuts
2D vtx -> 3D vtx matching cuts
Definition: DataStructs.h:542
double WireCoordinate(Point_t const &point) const
Returns the coordinate of the point on the plane, in wire units.
Definition: PlaneGeo.h:704
bool SortSection(PFPStruct &pfp, unsigned short sfIndex)
Definition: PFPUtils.cxx:1999
TCConfig tcc
Definition: DataStructs.cxx:9
unsigned short id
Definition: DataStructs.h:141
Float_t den
Definition: plot.C:35
unsigned int NTPC(CryostatID const &cryoid=details::cryostat_zero) const
Returns the total number of TPCs in the specified cryostat.
Definition: GeometryCore.h:416
Declaration of signal hit object.
void FindAlongTrans(Point3_t pos1, Vector3_t dir1, Point3_t pos2, Point2_t &alongTrans)
Definition: PFPUtils.cxx:3066
Float_t y
Definition: compare.C:6
bool HasWire(WireID const &wireid) const
Returns whether we have the specified wire.
std::bitset< pAlgModSize > AlgMod
Definition: DataStructs.h:297
Double_t z
Definition: plot.C:276
std::vector< std::vector< std::pair< unsigned int, unsigned int > > > wireHitRange
Definition: DataStructs.h:667
std::vector< int > Vx2ID
Definition: DataStructs.h:109
The data type to uniquely identify a Plane.
Definition: geo_types.h:364
Geometry information for a single TPC.
Definition: TPCGeo.h:33
Point3_t Pos
center position of this section
Definition: DataStructs.h:238
bool StoreTraj(TCSlice &slc, Trajectory &tj)
Definition: Utils.cxx:1074
bool LineLineIntersect(Point3_t p1, Point3_t p2, Point3_t p3, Point3_t p4, Point3_t &intersect, float &doca)
Definition: PFPUtils.cxx:3100
bool IsShowerLike(TCSlice const &slc, std::vector< int > const &TjIDs)
Definition: TCShower.cxx:1891
constexpr auto abs(T v)
Returns the absolute value of the argument.
CryostatID_t Cryostat
Index of cryostat.
Definition: geo_types.h:195
for(Int_t i=0;i< nentries;i++)
Definition: comparison.C:30
static unsigned int kPlane
void SetAngleCode(TrajPoint &tp)
Definition: Utils.cxx:765
PFPStruct CreatePFP(const TCSlice &slc)
Definition: PFPUtils.cxx:2802
Float_t tmp
Definition: plot.C:35
Point3_t PosAtEnd(const PFPStruct &pfp, unsigned short end)
Definition: PFPUtils.cxx:3257
double DeltaAngle(const Vector3_t v1, const Vector3_t v2)
Definition: PFPUtils.cxx:2531
bool dbgSlc
debug only in the user-defined slice? default is all slices
Definition: DataStructs.h:582
void Match2Planes(TCSlice &slc, std::vector< MatchStruct > &matVec)
Definition: PFPUtils.cxx:939
bool CanSection(const TCSlice &slc, const PFPStruct &pfp)
Definition: PFPUtils.cxx:1323
std::array< int, 2 > Vx3ID
Definition: DataStructs.h:285
float TPHitsRMSTime(const TCSlice &slc, const TrajPoint &tp, HitStatus_t hitRequest)
Definition: Utils.cxx:4092
unsigned int MVI
MatchVec Index for detailed 3D matching.
Definition: DebugStruct.h:29
void FilldEdx(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp)
Definition: PFPUtils.cxx:2583
void ReconcileVertices(TCSlice &slc, PFPStruct &pfp, bool prt)
Definition: PFPUtils.cxx:1626
std::string TPEnvString(const TrajPoint &tp)
Definition: Utils.cxx:6190
std::string PrintPos(const TrajPoint &tp)
Definition: Utils.cxx:6356
std::vector< std::array< unsigned int, 3 > > sptHits
SpacePoint -> Hits assns by plane.
Definition: DataStructs.h:625
float MaxTjLen(const TCSlice &slc, std::vector< int > &tjIDs)
Definition: Utils.cxx:2567
bool Update(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp)
Definition: PFPUtils.cxx:1048
void MakePFPTjs(TCSlice &slc)
Definition: PFPUtils.cxx:508
unsigned short Find3DRecoRange(const TCSlice &slc, const PFPStruct &pfp, unsigned short fromPt, unsigned short min2DPts, short dir)
Definition: PFPUtils.cxx:1338
bool MakeSmallAnglePFP(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
Definition: PFPUtils.cxx:2189
Vector3_t DirErr
and direction error
Definition: DataStructs.h:240
double ThetaZ() const
Angle of the wires from positive z axis; .
Definition: PlaneGeo.cxx:549
unsigned short plane
Definition: DataStructs.h:139
void AddPointsInRange(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, unsigned short fromPt, unsigned short toPt, CTP_t inCTP, float maxPull, unsigned short &nWires, unsigned short &nAdd, bool prt)
Definition: PFPUtils.cxx:1811
decltype(auto) constexpr end(T &&obj)
ADL-aware version of std::end.
Definition: StdUtils.h:77
unsigned short npts
Definition: DataStructs.h:144
std::bitset< 8 > Flags
Definition: DataStructs.h:295
float HitsRMSTime(const TCSlice &slc, const std::vector< unsigned int > &hitsInMultiplet, HitStatus_t hitRequest)
Definition: Utils.cxx:4125
Access the description of the physical detector geometry.
std::vector< int > TjUIDs
Definition: DataStructs.h:279
unsigned short ipt
Definition: DataStructs.h:142
CTP_t CTP
Definition: DataStructs.h:255
struct of temporary 3D vertices
Definition: DataStructs.h:99
Vector3_t Dir
Definition: DataStructs.h:249
geo::TPCID TPCID
Definition: DataStructs.h:290
if(nlines<=0)
void PFPVertexCheck(TCSlice &slc)
Definition: PFPUtils.cxx:2822
Vector3_t Dir
and direction
Definition: DataStructs.h:239
std::vector< Tj2Pt > mallTraj
vector of trajectory points ordered by increasing X
Definition: DataStructs.h:663
void FillGaps3D(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
Definition: PFPUtils.cxx:1720
std::array< float, 2 > Point2_t
Definition: DataStructs.h:41
int PDGCodeVote(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp)
Definition: PFPUtils.cxx:3312
float unitsPerTick
scale factor from Tick to WSE equivalent units
Definition: DataStructs.h:562
geo::TPCID TPCID
Definition: DataStructs.h:108
DebugStuff debug
Definition: DebugStruct.cxx:4
Vector3_t PointDirection(const Point3_t p1, const Point3_t p2)
Definition: PFPUtils.cxx:2538
void FillWireIntersections(TCSlice &slc)
Definition: PFPUtils.cxx:607
Float_t E
Definition: plot.C:20
bool MakeTP3Ds(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
Definition: PFPUtils.cxx:2097
int UID
unique global ID
Definition: DataStructs.h:111
unsigned short TPIndex
and the TP index
Definition: DataStructs.h:256
unsigned int wire
Definition: DataStructs.h:135
void Match3Planes(TCSlice &slc, std::vector< MatchStruct > &matVec)
Definition: PFPUtils.cxx:809
void GetHitMultiplet(const TCSlice &slc, unsigned int theHit, std::vector< unsigned int > &hitsInMultiplet, bool useLongPulseHits)
Definition: StepUtils.cxx:1521
bool TCIntersectionPoint(unsigned int wir1, unsigned int wir2, unsigned int pln1, unsigned int pln2, float &y, float &z)
Definition: PFPUtils.cxx:667
float ElectronLikelihood(const TCSlice &slc, const Trajectory &tj)
Definition: Utils.cxx:3144
bool FitSection(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp, unsigned short sfIndex)
Definition: PFPUtils.cxx:1391
std::vector< std::vector< bool > > goodWire
Definition: DataStructs.h:622
Point3_t Pos
position of the trajectory
Definition: DataStructs.h:248
float ChgFracNearPos(const TCSlice &slc, const Point2_t &pos, const std::vector< int > &tjIDs)
Definition: Utils.cxx:3163
std::vector< unsigned int > FindCloseHits(const TCSlice &slc, std::array< int, 2 > const &wireWindow, Point2_t const &timeWindow, const unsigned short plane, HitStatus_t hitRequest, bool usePeakTime, bool &hitsNear)
Definition: Utils.cxx:2774
float ChgFracBetween(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, Point3_t pos1, Point3_t pos2)
Definition: PFPUtils.cxx:3163
TrajPoint MakeBareTP(detinfo::DetectorPropertiesData const &detProp, const Point3_t &pos, CTP_t inCTP)
Definition: Utils.cxx:3928
unsigned short pln2
Definition: DataStructs.h:120
std::bitset< 8 > Flags
Definition: DataStructs.h:258
std::vector< VtxStore > vtxs
2D vertices
Definition: DataStructs.h:668
tick_as<> tick
Tick number, represented by std::ptrdiff_t.
Definition: electronics.h:73
double ConvertXToTicks(double X, int p, int t, int c) const
void CountBadPoints(const TCSlice &slc, const PFPStruct &pfp, unsigned short fromPt, unsigned short toPt, unsigned short &nBadPts, unsigned short &firstBadPt)
Definition: PFPUtils.cxx:1290
bool ReSection(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp, bool prt)
Definition: PFPUtils.cxx:1097
std::vector< float > match3DCuts
3D matching cuts
Definition: DataStructs.h:551
std::vector< SectionFit > SectionFits
Definition: DataStructs.h:281
void MakeTrajectoryObsolete(TCSlice &slc, unsigned int itj)
Definition: Utils.cxx:2138
double TPXErr2
(X position error)^2
Definition: DataStructs.h:251
double PosSep2(const Point3_t &pos1, const Point3_t &pos2)
Definition: PFPUtils.cxx:2560
const geo::GeometryCore * geom
Definition: DataStructs.h:567
bool SectionStartEnd(const PFPStruct &pfp, unsigned short sfIndex, unsigned short &startPt, unsigned short &endPt)
Definition: PFPUtils.cxx:3274
bool PointDirIntersect(Point3_t p1, Vector3_t p1Dir, Point3_t p2, Vector3_t p2Dir, Point3_t &intersect, float &doca)
Definition: PFPUtils.cxx:3083
The data type to uniquely identify a TPC.
Definition: geo_types.h:306
PlaneID_t Plane
Index of the plane within its TPC.
Definition: geo_types.h:373
constexpr unsigned int pAlgModSize
Definition: DataStructs.h:275
void PrintP(mf::LogVerbatim &myprt, PFPStruct const &pfp, bool &printHeader)
Definition: Utils.cxx:5453
bool valsIncreasing(const SortEntry &c1, const SortEntry &c2)
Definition: Utils.cxx:43
bool SetMag(Vector3_t &v1, double mag)
Definition: PFPUtils.cxx:2570
Definition of data types for geometry description.
unsigned short FarEnd(const PFPStruct &pfp, const Point3_t &pos)
Definition: PFPUtils.cxx:3300
float dEdx(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, TP3D &tp3d)
Definition: PFPUtils.cxx:2671
std::vector< TCSlice > slices
Definition: DataStructs.cxx:13
Detector simulation of raw signals on wires.
void GetRange(const PFPStruct &pfp, unsigned short sfIndex, unsigned short &fromPt, unsigned short &npts)
Definition: PFPUtils.cxx:1370
bool WireIDsIntersect(WireID const &wid1, WireID const &wid2, Point_t &intersection) const
Computes the intersection between two wires.
unsigned short InsertTP3D(PFPStruct &pfp, TP3D &tp3d)
Definition: PFPUtils.cxx:1961
SectionFit FitTP3Ds(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, const std::vector< TP3D > &tp3ds, unsigned short fromPt, short fitDir, unsigned short nPtsFit)
Definition: PFPUtils.cxx:1421
double ConvertTicksToX(double ticks, int p, int t, int c) const
std::bitset< 16 > modes
number of points to find AveChg
Definition: DataStructs.h:599
std::vector< TCHit > slHits
Definition: DataStructs.h:661
void Recover(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
Definition: PFPUtils.cxx:2059
TP3D CreateTP3D(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, int tjID, unsigned short tpIndex)
Definition: PFPUtils.cxx:2706
ROOT::Math::PositionVector3D< ROOT::Math::Cartesian3D< double >, ROOT::Math::GlobalCoordinateSystemTag > Point_t
Type for representation of position in physical 3D space.
Definition: geo_vectors.h:180
std::vector< int > GetAssns(TCSlice const &slc, std::string type1Name, int id, std::string type2Name)
Definition: Utils.cxx:4704
double weight
Definition: plottest35.C:25
bool ValidTwoPlaneMatch(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, const PFPStruct &pfp)
Definition: PFPUtils.cxx:1771
double DotProd(const Vector3_t &v1, const Vector3_t &v2)
Definition: PFPUtils.h:126
unsigned int CTP_t
Definition: DataStructs.h:45
std::vector< Vtx3Store > vtx3s
3D vertices
Definition: DataStructs.h:669
geo::TPCID TPCID
Definition: DataStructs.h:654
void FillmAllTraj(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc)
Definition: PFPUtils.cxx:2369
std::bitset< 128 > useAlg
Max hit separation for making junk trajectories. < 0 to turn off.
Definition: DataStructs.h:578
Float_t norm
Encapsulate the construction of a single detector plane .
Contains all timing reference information for the detector.
TDirectory * dir
Definition: macro.C:5
std::vector< float > pfpStitchCuts
cuts for stitching between TPCs
Definition: DataStructs.h:555
TP3D MakeTP3D(detinfo::DetectorPropertiesData const &detProp, const TrajPoint &itp, const TrajPoint &jtp)
Definition: PFPUtils.cxx:2435
geo::PlaneID DecodeCTP(CTP_t CTP)
float along
distance from the start Pos of the section
Definition: DataStructs.h:253
double PosSep(const Point3_t &pos1, const Point3_t &pos2)
Definition: PFPUtils.cxx:2554
std::vector< int > TjIDs
Definition: DataStructs.h:278
range_type< T > Iterate() const
Definition: Iterable.h:121
float PointPull(const TP3D &tp3d)
Definition: PFPUtils.cxx:2794
std::vector< TCWireIntersection > wireIntersections
Definition: DataStructs.h:631
std::vector< recob::Hit > const * allHits
Definition: DataStructs.h:614
bool valsDecreasing(const SortEntry &c1, const SortEntry &c2)
Definition: Utils.cxx:39
CTP_t EncodeCTP(unsigned int cryo, unsigned int tpc, unsigned int plane)
Definition: DataStructs.h:49
double dEdx_AREA(detinfo::DetectorClocksData const &clock_data, detinfo::DetectorPropertiesData const &det_prop, recob::Hit const &hit, double pitch, double T0=0) const
std::pair< unsigned short, unsigned short > GetSliceIndex(std::string typeName, int uID)
Definition: Utils.cxx:4942
unsigned short MVI_Iter
MVI iteration - see FindPFParticles.
Definition: DebugStruct.h:30
std::vector< unsigned int > nWires
Definition: DataStructs.h:645
std::array< double, 3 > Vector3_t
Definition: DataStructs.h:40
std::vector< TP3D > TP3Ds
Definition: DataStructs.h:280
int ID
ID of the recob::Slice (not the sub-slice)
Definition: DataStructs.h:656
std::array< std::vector< float >, 2 > dEdx
Definition: DataStructs.h:283
void Match3PlanesSpt(TCSlice &slc, std::vector< MatchStruct > &matVec)
Definition: PFPUtils.cxx:698
unsigned short pln1
Definition: DataStructs.h:119
unsigned short nPlanes
Definition: DataStructs.h:655
void FindPFParticles(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc)
Definition: PFPUtils.cxx:190
std::vector< PFPStruct > pfps
Definition: DataStructs.h:670
void MoveTPToWire(TrajPoint &tp, float wire)
Definition: Utils.cxx:2763
unsigned short NumPtsWithCharge(const TCSlice &slc, const Trajectory &tj, bool includeDeadWires)
Definition: Utils.cxx:2072
unsigned short CloseEnd(const Trajectory &tj, const Point2_t &pos)
Definition: Utils.cxx:2495
PlaneGeo const & Plane(TPCID const &tpcid, View_t view) const
Returns the specified wire.
TCEvent evt
Definition: DataStructs.cxx:8
unsigned short MVI
Definition: DataStructs.h:294
double TPX
X position of the TP or the single hit.
Definition: DataStructs.h:250
TPCID_t TPC
Index of the TPC within its cryostat.
Definition: geo_types.h:315
bool StorePFP(TCSlice &slc, PFPStruct &pfp)
Definition: PFPUtils.cxx:2979
Collection of Physical constants used in LArSoft.
Double_t sum
Definition: plot.C:31
float ChgFracNearEnd(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, const PFPStruct &pfp, unsigned short end)
Definition: PFPUtils.cxx:3197
void PrintTP3Ds(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, std::string someText, const TCSlice &slc, const PFPStruct &pfp, short printPts)
Definition: PFPUtils.cxx:3356
size_t ParentUID
Definition: DataStructs.h:289
Vector3_t DirAtEnd(const PFPStruct &pfp, unsigned short end)
Definition: PFPUtils.cxx:3249
void StitchPFPs()
Definition: PFPUtils.cxx:42
Float_t w
Definition: plot.C:20
bool SetSection(detinfo::DetectorPropertiesData const &detProp, PFPStruct &pfp, TP3D &tp3d)
Definition: PFPUtils.cxx:2752
void DefinePFPParents(TCSlice &slc)
Definition: PFPUtils.cxx:2860
bool AttachToAnyVertex(TCSlice &slc, PFPStruct &pfp, float maxSep, bool prt)
Definition: TCVertex.cxx:1580
Interface to geometry for wire readouts .
constexpr TPCID()=default
Default constructor: an invalid TPC ID.
double WirePitch() const
Return the wire pitch (in centimeters). It is assumed constant.
Definition: PlaneGeo.h:312
Encapsulate the construction of a single detector plane .
std::array< std::bitset< 8 >, 2 > EndFlag
Definition: DataStructs.h:296
bool SptInTPC(const std::array< unsigned int, 3 > &sptHits, unsigned int tpc)
Definition: PFPUtils.cxx:789
void Reverse(PFPStruct &pfp)
Definition: PFPUtils.cxx:2348
unsigned short SFIndex
and the section fit index
Definition: DataStructs.h:257
unsigned int index
Definition: DataStructs.h:34