1
1
/* * TRACCC library, part of the ACTS project (R&D line)
2
2
*
3
- * (c) 2022-2023 CERN for the benefit of the ACTS project
3
+ * (c) 2022-2024 CERN for the benefit of the ACTS project
4
4
*
5
5
* Mozilla Public License Version 2.0
6
6
*/
13
13
// System include(s).
14
14
#include < algorithm>
15
15
#include < cassert>
16
+ #include < iostream>
17
+ #include < map>
18
+ #include < set>
16
19
#include < stdexcept>
17
20
#include < utility>
18
21
#include < vector>
@@ -21,20 +24,28 @@ namespace {
21
24
22
25
// / Comparator used for sorting cells. This sorting is one of the assumptions
23
26
// / made in the clusterization algorithm
24
- const auto comp = [](const traccc::cell& c1, const traccc::cell& c2) {
25
- return c1.channel1 < c2.channel1 ;
26
- };
27
+ struct cell_order {
28
+ bool operator ()(const traccc::cell& lhs, const traccc::cell& rhs) const {
29
+ if (lhs.module_link != rhs.module_link ) {
30
+ return lhs.module_link < rhs.module_link ;
31
+ } else if (lhs.channel1 != rhs.channel1 ) {
32
+ return (lhs.channel1 < rhs.channel1 );
33
+ } else {
34
+ return (lhs.channel0 < rhs.channel0 );
35
+ }
36
+ }
37
+ }; // struct cell_order
27
38
28
39
// / Helper function which finds module from csv::cell in the geometry and
29
40
// / digitization config, and initializes the modules limits with the cell's
30
41
// / properties
31
- traccc::cell_module get_module (traccc::io::csv::cell c ,
42
+ traccc::cell_module get_module (const std:: uint64_t geometry_id ,
32
43
const traccc::geometry* geom,
33
44
const traccc::digitization_config* dconfig,
34
45
const std::uint64_t original_geometry_id) {
35
46
36
47
traccc::cell_module result;
37
- result.surface_link = detray::geometry::barcode{c. geometry_id };
48
+ result.surface_link = detray::geometry::barcode{geometry_id};
38
49
39
50
// Find/set the 3D position of the detector module.
40
51
if (geom != nullptr ) {
@@ -72,114 +83,119 @@ traccc::cell_module get_module(traccc::io::csv::cell c,
72
83
return result;
73
84
}
74
85
75
- } // namespace
76
-
77
- namespace traccc ::io::csv {
86
+ std::map<std::uint64_t , std::vector<traccc::cell> > read_deduplicated_cells (
87
+ std::string_view filename) {
78
88
79
- void read_cells (
80
- cell_reader_output& out, std::string_view filename, const geometry* geom,
81
- const digitization_config* dconfig,
82
- const std::map<std::uint64_t , detray::geometry::barcode>* barcode_map) {
89
+ // Temporary storage for all the cells and modules.
90
+ std::map<std::uint64_t , std::map<traccc::cell, float , ::cell_order> >
91
+ cellMap;
83
92
84
93
// Construct the cell reader object.
85
- auto reader = make_cell_reader (filename);
86
-
87
- // Create cell counter vector.
88
- std::vector<unsigned int > cellCounts;
89
- cellCounts.reserve (5000 );
90
-
91
- cell_module_collection_types::host& result_modules = out.modules ;
92
- result_modules.reserve (5000 );
93
-
94
- // Create a cell collection, which holds on to a flat list of all the cells
95
- // and the position of their respective cell counter & module.
96
- std::vector<std::pair<csv::cell, unsigned int >> allCells;
97
- allCells.reserve (50000 );
94
+ auto reader = traccc::io::csv::make_cell_reader (filename);
98
95
99
96
// Read all cells from input file.
100
- csv::cell iocell;
97
+ traccc::io::csv::cell iocell;
98
+ unsigned int nduplicates = 0 ;
101
99
while (reader.read (iocell)) {
102
100
103
- // Modify the geometry ID of the cell if a barcode map is provided.
104
- const std::uint64_t original_geometry_id = iocell.geometry_id ;
105
- if (barcode_map != nullptr ) {
106
- const auto it = barcode_map->find (iocell.geometry_id );
107
- if (it != barcode_map->end ()) {
108
- iocell.geometry_id = it->second .value ();
109
- } else {
110
- throw std::runtime_error (
111
- " Could not find barcode for geometry ID " +
112
- std::to_string (iocell.geometry_id ));
113
- }
101
+ // Construct a cell object.
102
+ const traccc::cell cell{iocell.channel0 , iocell.channel1 , iocell.value ,
103
+ iocell.timestamp , 0 };
104
+
105
+ // Add the cell to the module. At this point the module link of the
106
+ // cells is not set up correctly yet.
107
+ auto ret = cellMap[iocell.geometry_id ].insert ({cell, iocell.value });
108
+ if (ret.second == false ) {
109
+ cellMap[iocell.geometry_id ].at (cell) += iocell.value ;
110
+ ++nduplicates;
114
111
}
112
+ }
113
+ if (nduplicates > 0 ) {
114
+ std::cout << " WARNING: @traccc::io::csv::read_cells: " << nduplicates
115
+ << " duplicate cells found in " << filename << std::endl;
116
+ }
115
117
116
- // Look for current module in cell counter vector.
117
- auto rit = std::find_if (result_modules.rbegin (), result_modules.rend (),
118
- [&iocell](const cell_module& mod) {
119
- return mod.surface_link .value () ==
120
- iocell.geometry_id ;
121
- });
122
- if (rit == result_modules.rend ()) {
123
- // Add new cell and new cell counter if a new module is found
124
- const cell_module mod =
125
- get_module (iocell, geom, dconfig, original_geometry_id);
126
- allCells.push_back ({iocell, result_modules.size ()});
127
- result_modules.push_back (mod);
128
- cellCounts.push_back (1 );
129
- } else {
130
- // Add a new cell and update cell counter if repeat module is found
131
- const unsigned int pos =
132
- std::distance (result_modules.begin (), rit.base ()) - 1 ;
133
- allCells.push_back ({iocell, pos});
134
- ++(cellCounts[pos]);
118
+ // Create and fill the result container. With summed activation values.
119
+ std::map<std::uint64_t , std::vector<traccc::cell> > result;
120
+ for (const auto & [geometry_id, cells] : cellMap) {
121
+ for (const auto & [cell, value] : cells) {
122
+ traccc::cell summed_cell{cell};
123
+ summed_cell.activation = value;
124
+ result[geometry_id].push_back (summed_cell);
135
125
}
136
126
}
137
127
138
- // Transform the cellCounts vector into a prefix sum for accessing
139
- // positions in the result vector.
140
- std::partial_sum (cellCounts. begin (), cellCounts. end (), cellCounts. begin ());
128
+ // Return the container.
129
+ return result;
130
+ }
141
131
142
- // The total number cells.
143
- const unsigned int totalCells = allCells. size ();
132
+ std::map<std:: uint64_t , std::vector<traccc::cell> > read_all_cells (
133
+ std::string_view filename) {
144
134
145
- // Construct the result collection.
146
- cell_collection_types::host& result_cells = out.cells ;
147
- result_cells.resize (totalCells);
135
+ // The result container.
136
+ std::map<std::uint64_t , std::vector<traccc::cell> > result;
148
137
149
- // Member "-1" of the prefix sum vector
150
- unsigned int nCellsZero = 0 ;
151
- // Fill the result object with the read csv cells
152
- for (unsigned int i = 0 ; i < totalCells; ++i) {
153
- const csv::cell& c = allCells[i].first ;
138
+ // Construct the cell reader object.
139
+ auto reader = traccc::io::csv::make_cell_reader (filename);
154
140
155
- // The position of the cell counter this cell belongs to
156
- const unsigned int & counterPos = allCells[i].second ;
141
+ // Read all cells from input file.
142
+ traccc::io::csv::cell iocell;
143
+ while (reader.read (iocell)) {
157
144
158
- unsigned int & prefix_sum_previous =
159
- counterPos == 0 ? nCellsZero : cellCounts[counterPos - 1 ];
160
- result_cells[prefix_sum_previous++] = traccc::cell{
161
- c.channel0 , c.channel1 , c.value , c.timestamp , counterPos};
145
+ // Add the cell to the module. At this point the module link of the
146
+ // cells is not set up correctly yet.
147
+ result[iocell.geometry_id ].push_back ({iocell.channel0 , iocell.channel1 ,
148
+ iocell.value , iocell.timestamp ,
149
+ 0 });
162
150
}
163
151
164
- if (cellCounts.size () == 0 ) {
165
- return ;
152
+ // Sort the cells. Deduplication or not, they do need to be sorted.
153
+ for (auto & [_, cells] : result) {
154
+ std::sort (cells.begin (), cells.end (), ::cell_order ());
166
155
}
167
- /* This is might look a bit overcomplicated, and could be made simpler by
168
- * having a copy of the prefix sum vector before incrementing its value when
169
- * filling the vector. however this seems more efficient, but requires
170
- * manually setting the 1st & 2nd modules instead of just the 1st.
171
- */
172
-
173
- // Sort the cells belonging to the first module.
174
- std::sort (result_cells.begin (), result_cells.begin () + nCellsZero, comp);
175
- // Sort the cells belonging to the second module.
176
- std::sort (result_cells.begin () + nCellsZero,
177
- result_cells.begin () + cellCounts[0 ], comp);
178
-
179
- // Sort cells belonging to all other modules.
180
- for (unsigned int i = 1 ; i < cellCounts.size () - 1 ; ++i) {
181
- std::sort (result_cells.begin () + cellCounts[i - 1 ],
182
- result_cells.begin () + cellCounts[i], comp);
156
+
157
+ // Return the container.
158
+ return result;
159
+ }
160
+
161
+ } // namespace
162
+
163
+ namespace traccc ::io::csv {
164
+
165
+ void read_cells (
166
+ cell_reader_output& out, std::string_view filename, const geometry* geom,
167
+ const digitization_config* dconfig,
168
+ const std::map<std::uint64_t , detray::geometry::barcode>* barcode_map,
169
+ const bool deduplicate) {
170
+
171
+ // Get the cells and modules into an intermediate format.
172
+ auto cellsMap = (deduplicate ? read_deduplicated_cells (filename)
173
+ : read_all_cells (filename));
174
+
175
+ // Fill the output containers with the ordered cells and modules.
176
+ for (const auto & [original_geometry_id, cells] : cellsMap) {
177
+ // Modify the geometry ID of the module if a barcode map is
178
+ // provided.
179
+ std::uint64_t geometry_id = original_geometry_id;
180
+ if (barcode_map != nullptr ) {
181
+ const auto it = barcode_map->find (geometry_id);
182
+ if (it != barcode_map->end ()) {
183
+ geometry_id = it->second .value ();
184
+ } else {
185
+ throw std::runtime_error (
186
+ " Could not find barcode for geometry ID " +
187
+ std::to_string (geometry_id));
188
+ }
189
+ }
190
+
191
+ // Add the module and its cells to the output.
192
+ out.modules .push_back (
193
+ get_module (geometry_id, geom, dconfig, original_geometry_id));
194
+ for (auto & cell : cells) {
195
+ out.cells .push_back (cell);
196
+ // Set the module link.
197
+ out.cells .back ().module_link = out.modules .size () - 1 ;
198
+ }
183
199
}
184
200
}
185
201
0 commit comments