Belle II Software development
AxialTrackMerger.cc
1/**************************************************************************
2 * basf2 (Belle II Analysis Software Framework) *
3 * Author: The Belle II Collaboration *
4 * *
5 * See git log for contributors and copyright holders. *
6 * This file is licensed under LGPL-3.0, see LICENSE.md. *
7 **************************************************************************/
8#include <tracking/trackFindingCDC/findlets/minimal/AxialTrackMerger.h>
9
10#include <tracking/trackFindingCDC/processing/AxialTrackUtil.h>
11
12#include <tracking/trackFindingCDC/fitting/CDCKarimakiFitter.h>
13
14#include <tracking/trackingUtilities/eventdata/tracks/CDCTrack.h>
15#include <tracking/trackingUtilities/eventdata/hits/CDCWireHit.h>
16#include <tracking/trackingUtilities/eventdata/trajectories/CDCTrajectory2D.h>
17
18#include <tracking/trackingUtilities/numerics/WeightComperator.h>
19
20#include <tracking/trackingUtilities/utilities/StringManipulation.h>
21
22#include <framework/core/ModuleParamList.templateDetails.h>
23
24using namespace Belle2;
25using namespace TrackFindingCDC;
26using namespace TrackingUtilities;
27
29{
30 return "Merges axial tracks found in the Legendre search";
31}
32
33void AxialTrackMerger::exposeParameters(ModuleParamList* moduleParamList, const std::string& prefix)
34{
35 moduleParamList->addParameter(prefixed(prefix, "minFitProb"),
37 "Minimal fit probability of the common fit "
38 "of two tracks to be eligible for merging",
40}
41
42void AxialTrackMerger::apply(std::vector<CDCTrack>& axialTracks,
43 const std::vector<const CDCWireHit*>& allAxialWireHits)
44{
45 // Check quality of the track basing on holes on the trajectory;
46 // if holes exist then track is split
47 for (CDCTrack& track : axialTracks) {
48 if (track.size() < 5) continue;
51 }
52
53 // Update tracks before storing to DataStore
54 for (CDCTrack& track : axialTracks) {
56 }
57
58 // Remove bad tracks
61
62 // Perform tracks merging
63 this->doTracksMerging(axialTracks, allAxialWireHits);
64
65 // Remove the consumed, now empty tracks.
67}
68
69void AxialTrackMerger::doTracksMerging(std::vector<CDCTrack>& axialTracks,
70 const std::vector<const CDCWireHit*>& allAxialWireHits)
71{
72 // Search for best matches - cannot use range for here :(.
73 for (auto itTrack = axialTracks.begin(); itTrack != axialTracks.end(); ++itTrack) {
74 CDCTrack& track = *itTrack;
75 auto followingTracks = asRange(std::next(itTrack), axialTracks.end());
76
77 WithWeight<MayBePtr<CDCTrack> > bestTrack = calculateBestTrackToMerge(track, followingTracks);
78 double fitProb = bestTrack.getWeight();
79
80 if (bestTrack != nullptr and fitProb > m_param_minFitProb) {
81 mergeTracks(track, *bestTrack, allAxialWireHits);
82 }
83 }
84
86}
87
89template <class ACDCTracks>
91{
92 std::vector<WithWeight<CDCTrack*>> weightedTracks;
93 for (CDCTrack& track2 : tracks) {
94 if (&track == &track2) continue;
95 if (track2.size() < 3) continue;
96
97 double fitProb = doTracksFitTogether(track, track2);
98 if (std::isnan(fitProb)) continue;
99
100 weightedTracks.emplace_back(&track2, fitProb);
101 }
102
103 auto bestMatch = std::max_element(weightedTracks.begin(), weightedTracks.end(), LessWeight());
104 if (bestMatch == weightedTracks.end()) return {nullptr, 0};
105 else return *bestMatch;
106}
107
109{
110 // First check whether most of the hits from the tracks lie in the backward direction
111 // even if though track is not curling -> tracks should not be merged
112 const CDCTrajectory3D& trajectory3D1 = track1.getStartTrajectory3D();
113 const CDCTrajectory3D& trajectory3D2 = track2.getStartTrajectory3D();
114
115 int fbVote12 = 0;
116 int fbVote21 = 0;
117
118 for (const CDCRecoHit3D& recoHit3D : track1) {
119 EForwardBackward fbInfo = trajectory3D2.getFlightDirection3DAtSupport().xy().isForwardOrBackwardOf(recoHit3D.getRecoPos2D());
120 if (not isValid(fbInfo)) continue;
121 fbVote12 += fbInfo;
122 }
123
124 for (const CDCRecoHit3D& recoHit3D : track2) {
125 EForwardBackward fbInfo = trajectory3D1.getFlightDirection3DAtSupport().xy().isForwardOrBackwardOf(recoHit3D.getRecoPos2D());
126 if (not isValid(fbInfo)) continue;
127 fbVote21 += fbInfo;
128 }
129
130 if (not trajectory3D1.isCurler() and fbVote12 < 0) return NAN;
131 if (not trajectory3D2.isCurler() and fbVote21 < 0) return NAN;
132
133 // Build common hit list by copying the wire hits into one large list
134 // We use the wire hits here as we do not want them to bring
135 // their "old" reconstructed position when fitting.
136 std::vector<const CDCWireHit*> combinedWireHits;
137 combinedWireHits.reserve(track1.size() + track2.size());
138 for (const CDCRecoHit3D& hit : track1) {
139 combinedWireHits.push_back(&(hit.getWireHit()));
140 }
141 for (const CDCRecoHit3D& hit : track2) {
142 combinedWireHits.push_back(&(hit.getWireHit()));
143 }
144
145 // Sorting is done via pointer addresses (!!).
146 // This is not very stable and also not very meaningful (in terms of ordering in the track),
147 // but it does the job for unique.
148 // (the ordering is still outwards though since the wire hits are ordered like that in continuous memory)
149 std::sort(combinedWireHits.begin(), combinedWireHits.end());
150 erase_unique(combinedWireHits);
151
152 // Calculate track parameters
153 CDCTrajectory2D commonTrajectory2D;
155
156 // Approach the best fit
157 commonTrajectory2D = fitter.fit(combinedWireHits);
158 removeStrangeHits(5, combinedWireHits, commonTrajectory2D);
159 commonTrajectory2D = fitter.fit(combinedWireHits);
160 removeStrangeHits(3, combinedWireHits, commonTrajectory2D);
161 commonTrajectory2D = fitter.fit(combinedWireHits);
162 removeStrangeHits(1, combinedWireHits, commonTrajectory2D);
163 commonTrajectory2D = fitter.fit(combinedWireHits);
164 removeStrangeHits(1, combinedWireHits, commonTrajectory2D);
165 commonTrajectory2D = fitter.fit(combinedWireHits);
166
167 // Dismiss this possibility if the hit list size after all the removing of hits is even smaller
168 // than the two lists before or if the list is too small
169 if (combinedWireHits.size() <= std::max(track1.size(), track2.size())
170 or combinedWireHits.size() < 15) {
171 return NAN;
172 }
173
174 return commonTrajectory2D.getPValue();
175}
176
178 std::vector<const CDCWireHit*>& wireHits,
179 CDCTrajectory2D& trajectory2D)
180{
181 auto farFromTrajectory = [&trajectory2D, &factor](const CDCWireHit * wireHit) {
182 Vector2D pos2D = wireHit->getRefPos2D();
183 double driftLength = wireHit->getRefDriftLength();
184 double dist = std::fabs(trajectory2D.getDist2D(pos2D)) - driftLength;
185 return std::fabs(dist) > driftLength * factor;
186 };
187 erase_remove_if(wireHits, farFromTrajectory);
188}
189
191 CDCTrack& track2,
192 const std::vector<const CDCWireHit*>& allAxialWireHits)
193{
194 if (&track1 == &track2) return;
195
196 CDCTrajectory2D trajectory2D = track1.getStartTrajectory3D().getTrajectory2D();
197 for (const CDCRecoHit3D& orgRecoHit3D : track2) {
198 CDCRecoHit3D recoHit3D = CDCRecoHit3D::reconstruct(orgRecoHit3D.getRLWireHit(), trajectory2D);
199 track1.push_back(std::move(recoHit3D));
200 }
201 track2.clear();
202
204
206
208
209 for (CDCRecoHit3D& recoHit3D : track2) {
210 recoHit3D.setRecoPos3D({recoHit3D.getRefPos2D(), 0});
211 recoHit3D.setRLInfo(ERightLeft::c_Unknown);
212 }
213
215 bool success = AxialTrackUtil::postprocessTrack(track2, allAxialWireHits);
216 if (not success) {
217 for (const CDCRecoHit3D& recoHit3D : track2) {
218 recoHit3D.getWireHit()->setTakenFlag(false);
219 }
220 track2.clear();
221 }
222}
The Module parameter list class.
void apply(std::vector< TrackingUtilities::CDCTrack > &axialTracks, const std::vector< const TrackingUtilities::CDCWireHit * > &axialWireHits) final
Merge tracks together. Allows for axial hits to be added as it may see fit.
static void removeStrangeHits(double factor, std::vector< const TrackingUtilities::CDCWireHit * > &wireHits, TrackingUtilities::CDCTrajectory2D &trajectory)
Remove all hits that are further than factor * driftlength away from the trajectory.
void doTracksMerging(std::vector< TrackingUtilities::CDCTrack > &axialTracks, const std::vector< const TrackingUtilities::CDCWireHit * > &allAxialWireHits)
The track finding often finds two curling tracks, originating from the same particle.
std::string getDescription() final
Short description of the findlet.
static void mergeTracks(TrackingUtilities::CDCTrack &track1, TrackingUtilities::CDCTrack &track2, const std::vector< const TrackingUtilities::CDCWireHit * > &allAxialWireHits)
Function to merge two track candidates.
static double doTracksFitTogether(TrackingUtilities::CDCTrack &track1, TrackingUtilities::CDCTrack &track2)
Fits the hit content of both tracks in a common fit repeated with an annealing schedule removing far ...
void exposeParameters(ModuleParamList *moduleParamList, const std::string &prefix) final
Expose the parameters to a module.
static TrackingUtilities::WithWeight< TrackingUtilities::MayBePtr< TrackingUtilities::CDCTrack > > calculateBestTrackToMerge(TrackingUtilities::CDCTrack &track, ACDCTracks &tracks)
Searches for the best candidate to merge this track to.
double m_param_minFitProb
Parameter : Minimal fit probability of the common fit of two tracks to be eligible for merging.
Class implementing the fitter using Karimakis method.
static const CDCKarimakiFitter & getNoDriftVarianceFitter()
Static getter for a general fitter that does not use the drift length variances.
Class representing a three dimensional reconstructed hit.
static CDCRecoHit3D reconstruct(const CDCRecoHit2D &recoHit2D, const CDCTrajectory2D &trajectory2D)
Reconstructs the three dimensional hit from the two dimensional and the two dimensional trajectory.
Class representing a sequence of three dimensional reconstructed hits.
Definition CDCTrack.h:39
Particle trajectory as it is seen in xy projection represented as a circle.
double getPValue() const
Getter for p-value.
double getDist2D(const Vector2D &point) const
Calculates the distance from the point to the trajectory as seen from the xy projection.
Particle full three dimensional trajectory.
bool isCurler(double factor=1) const
Checks if the trajectory leaves the outer radius of the CDC times the given tolerance factor.
Vector3D getFlightDirection3DAtSupport() const
Get the unit momentum at the start point of the trajectory.
Class representing a hit wire in the central drift chamber.
Definition CDCWireHit.h:58
A two dimensional vector which is equipped with functions for correct handling of orientation relate...
Definition Vector2D.h:36
EForwardBackward isForwardOrBackwardOf(const Vector2D &rhs) const
Indicates if the given vector is more coaligned or reverse if you looked in the direction of this vec...
Definition Vector2D.h:524
const Vector2D & xy() const
Getter for the xy projected vector ( reference ! )
Definition Vector3D.h:513
A mixin class to attach a weight to an object.
Definition WithWeight.h:24
Weight getWeight() const
Getter for the weight.
Definition WithWeight.h:58
void addParameter(const std::string &name, T &paramVariable, const std::string &description, const T &defaultValue)
Adds a new parameter to the module list.
Abstract base class for different kinds of events.
static void normalizeTrack(TrackingUtilities::CDCTrack &track)
Refit and resort the track. Unmask all hits.
static void deleteShortTracks(std::vector< TrackingUtilities::CDCTrack > &axialTracks, double minimal_size=5)
Remove tracks that are shorter than the given number of hits.
static void deleteTracksWithLowFitProbability(std::vector< TrackingUtilities::CDCTrack > &axialTracks, double minimal_probability_for_good_fit=0.4)
Check an (improper) p-values of the tracks. If they are below the given value, delete the track from ...
static void removeHitsAfterSuperLayerBreak(TrackingUtilities::CDCTrack &track)
Searches for a break in the super layer chain and remove all hits that come after that.
static std::vector< TrackingUtilities::CDCRecoHit3D > splitBack2BackTrack(TrackingUtilities::CDCTrack &track)
Tries to split back-to-back tracks into two different tracks.
static bool postprocessTrack(TrackingUtilities::CDCTrack &track, const std::vector< const TrackingUtilities::CDCWireHit * > &allAxialWireHits)
Perform all track postprocessing - return whether the track is considered good after the postprocessi...