ATLAS Offline Software
Loading...
Searching...
No Matches
MuonDetDescr/MuonGeoModelTest/src/NSWGeoPlottingAlg.cxx
Go to the documentation of this file.
1/*
2 Copyright (C) 2002-2025 CERN for the benefit of the ATLAS collaboration
3*/
4#include "NSWGeoPlottingAlg.h"
5
6#include <cmath>
7#include <format>
8
9#include "GaudiKernel/SystemOfUnits.h"
14#include "TFile.h"
15#include "TGraph.h"
16#include "TH1.h"
17#include "TH2D.h"
19#include "TrkSurfaces/Surface.h"
20
21namespace {
22std::string to_string(const Amg::Vector3D& v) {
23 std::stringstream sstr{};
24 sstr << "[x,y,z]=(" << v.x() << "," << v.y() << "," << v.z()
25 << ") [theta/eta/phi]=(" << (v.theta() / Gaudi::Units::degree) << ","
26 << v.eta() << "," << v.phi() << ")";
27 return sstr.str();
28}
29
30} // namespace
31namespace MuonGM{
33 auto out_file = std::make_unique<TFile>(m_outFile.value().c_str(), "RECREATE");
34 if (!out_file || !out_file->IsOpen() || out_file->IsZombie()) {
35
36 ATH_MSG_FATAL("Failed to create the output file " << m_outFile);
37 return StatusCode::FAILURE;
38 }
39 out_file->mkdir("SinglePads");
40 out_file->mkdir("ActiveSurfaces");
41 const MmIdHelper& mm_helper = m_idHelperSvc->mmIdHelper();
42 const sTgcIdHelper& st_helper = m_idHelperSvc->stgcIdHelper();
43 for (auto& id_graph : m_nswPads) {
44 std::unique_ptr<TGraph>& graph = id_graph.second;
45 const Identifier& id = id_graph.first;
46 bool is_mm = m_idHelperSvc->isMM(id);
47 const int stEta = m_idHelperSvc->stationEta(id);
48 const int ml = is_mm ? mm_helper.multilayer(id) : st_helper.multilayer(id);
49 const int lay = is_mm ? mm_helper.gasGap(id) : st_helper.gasGap(id);
50 const std::string ch_name =
51 (is_mm ? mm_helper.stationNameString(m_idHelperSvc->stationName(id))
52 : st_helper.stationNameString(m_idHelperSvc->stationName(id))) +
53 std::to_string(std::abs(stEta)) + (stEta > 0 ? "A" : "C") +
54 std::to_string(m_idHelperSvc->stationPhi(id)) + "W" +
55 std::to_string(ml) + "L" + std::to_string(lay);
56 TDirectory* dir = out_file->GetDirectory("SinglePads");
57 dir->WriteObject(graph.get(), ch_name.c_str());
58 graph.reset();
59 const int signed_lay = layerId(id);
60 std::unique_ptr<TGraph>& lay_graph = m_nswLayers[signed_lay];
61 if (!lay_graph)
62 continue;
63 std::string lay_name = std::string{is_mm ? "MMG" : "STG"} + "W" +
64 std::to_string(ml) + (stEta > 0 ? "A" : "C") +
65 std::to_string(lay);
66 out_file->WriteObject(lay_graph.get(), lay_name.c_str());
67 lay_graph.reset();
68 std::unique_ptr<TH1>& active_area = m_nswActiveAreas[signed_lay];
69 if (!active_area)
70 continue;
71 dir = out_file->GetDirectory("ActiveSurfaces");
72
73 dir->WriteObject(active_area.get(), lay_name.c_str());
74 active_area.reset();
75 }
76 return StatusCode::SUCCESS;
77}
79 ATH_CHECK(m_DetectorManagerKey.initialize());
80 ATH_CHECK(m_idHelperSvc.retrieve());
83 return StatusCode::SUCCESS;
84}
86 if (m_alg_run)
87 return StatusCode::SUCCESS;
88 ATH_MSG_INFO("Executing NSWGeoPlottingAlg for the first time");
89 const EventContext& ctx = Gaudi::Hive::currentContext();
90
91 const MuonGM::MuonDetectorManager* detMgr{nullptr};
92
94
95 for (auto& id_graph : m_nswPads) {
96 const Identifier& id = id_graph.first;
97 std::unique_ptr<TGraph>& pad_graph = id_graph.second;
98 const bool is_mm = m_idHelperSvc->isMM(id);
99 const int signed_lay = layerId(id);
100 std::unique_ptr<TGraph>& wheel_graph = m_nswLayers[signed_lay];
101
102 const MuonGM::MMReadoutElement* mm_roe =
103 is_mm ? detMgr->getMMReadoutElement(id) : nullptr;
104 const MuonGM::sTgcReadoutElement* st_roe =
105 is_mm ? nullptr : detMgr->getsTgcReadoutElement(id);
106
107 const MuonGM::MuonChannelDesign* design =
108 is_mm ? mm_roe->getDesign(id) : st_roe->getDesign(id);
109
110 const Trk::TrkDetElementBase* roe =
111 (mm_roe ? static_cast<const Trk::TrkDetElementBase*>(mm_roe)
112 : static_cast<const Trk::TrkDetElementBase*>(st_roe));
113 auto fill_graphs = [&](const Amg::Vector2D& locPos, int strip) {
114 if ((is_mm || std::abs(locPos.y()) < 1.e-3) &&
115 design->channelNumber(locPos) != strip) {
116 ATH_MSG_ALWAYS(" Backmapping of the strip number did not work for "
117 << m_idHelperSvc->toString(id) << " local pos: "
118 << locPos.x() << " " << locPos.y() << " "
119 << " " << design->channelNumber(locPos) << " vs. "
120 << strip);
121 return false;
122 }
123
124 Amg::Vector3D globPos{Amg::Vector3D::Zero()};
125 roe->surface(id).localToGlobal(locPos, Amg::Vector3D::Zero(), globPos);
126 if (pad_graph)
127 pad_graph->SetPoint(pad_graph->GetN(), globPos.x(), globPos.y());
128 if (wheel_graph)
129 wheel_graph->SetPoint(wheel_graph->GetN(), globPos.x(), globPos.y());
130
131 return true;
132 };
133 const MmIdHelper& id_helper = m_idHelperSvc->mmIdHelper();
134 auto global_points = [&](const Identifier& id, Amg::Vector3D& left,
135 Amg::Vector3D& center, Amg::Vector3D& right) {
136 Amg::Vector2D l_cen{Amg::Vector2D::Zero()}, l_left{Amg::Vector2D::Zero()},
137 l_right{Amg::Vector2D::Zero()};
138 const MuonGM::MuonChannelDesign* design = nullptr;
139 if (mm_roe)
140 design = mm_roe->getDesign(id);
141 else if (st_roe)
142 design = st_roe->getDesign(id);
143 const int chan = id_helper.channel(id);
144 design->leftEdge(chan, l_left);
145 design->center(chan, l_cen);
146 design->rightEdge(chan, l_right);
147
148 roe->surface(id).localToGlobal(l_left, Amg::Vector3D::Zero(), left);
149 roe->surface(id).localToGlobal(l_cen, Amg::Vector3D::Zero(), center);
150 roe->surface(id).localToGlobal(l_right, Amg::Vector3D::Zero(), right);
151 };
152
153 const int n_strips =
154 (is_mm ? mm_roe->numberOfStrips(id) : st_roe->numberOfStrips(id));
155 for (int strip = design->numberOfMissingBottomStrips() + 1;
156 strip <= n_strips; strip += 1) {
157 {
158 Amg::Vector2D locPos{Amg::Vector2D::Zero()};
159 if (design->leftEdge(strip, locPos) && !fill_graphs(locPos, strip))
160 ATH_MSG_DEBUG("left edge -- channel " << m_idHelperSvc->toString(id)
161 << " does not have strip "
162 << strip << "... Why?");
163 }
164 {
165 Amg::Vector2D locPos{Amg::Vector2D::Zero()};
166 if (design->center(strip, locPos) && !fill_graphs(locPos, strip))
167 ATH_MSG_DEBUG("center -- channel " << m_idHelperSvc->toString(id)
168 << " does not have strip " << strip
169 << "... Why?");
170 }
171 {
172 Amg::Vector2D locPos{Amg::Vector2D::Zero()};
173 if (design->rightEdge(strip, locPos) && !fill_graphs(locPos, strip))
174 ATH_MSG_DEBUG("right edge -- channel " << m_idHelperSvc->toString(id)
175 << " does not have strip "
176 << strip << "... Why?");
177 }
178 }
179 auto uv_intersects = [&]() {
181 if (design->hasStereoAngle() || m_idHelperSvc->issTgc(id))
182 return StatusCode::SUCCESS;
183 const int gap = id_helper.gasGap(id);
184 const int ml = id_helper.multilayer(id);
185 if ((ml == 1 && gap == 2) || (ml == 2 && gap == 4))
186 return StatusCode::SUCCESS;
187
188 for (int strip = design->numberOfMissingBottomStrips() + 1;
189 strip <= n_strips; strip += 1) {
190 const int u_gap = ml == 1 ? 3 : 1;
191 const int v_gap = ml == 1 ? 4 : 2;
192 const Identifier x_id = id_helper.channelID(id, ml, gap, strip);
193 const Identifier u_id = id_helper.channelID(id, ml, u_gap, strip);
194 const Identifier v_id = id_helper.channelID(id, ml, v_gap, strip);
195 Amg::Vector3D x_center{Amg::Vector3D::Zero()},
196 u_center{Amg::Vector3D::Zero()}, v_center{Amg::Vector3D::Zero()};
197 Amg::Vector3D x_left{Amg::Vector3D::Zero()},
198 u_left{Amg::Vector3D::Zero()}, v_left{Amg::Vector3D::Zero()};
199 Amg::Vector3D x_right{Amg::Vector3D::Zero()},
200 u_right{Amg::Vector3D::Zero()}, v_right{Amg::Vector3D::Zero()};
201
202 global_points(x_id, x_left, x_center, x_right);
203 global_points(u_id, u_left, u_center, u_right);
204 global_points(v_id, v_left, v_center, v_right);
205
206 const Amg::Vector3D x_dir = (x_right - x_left).unit();
207 const Amg::Vector3D v_dir = (v_left - v_right).unit();
208 const Amg::Vector3D u_dir = (u_left - u_right).unit();
209
210 std::optional<double> uv_isect =
211 Amg::intersect<3>(v_center, v_dir, u_center, u_dir);
212
213 if (!uv_isect) {
214 ATH_MSG_ERROR("Failed to intersect the uv strips for identifiers "
215 << std::endl
216 << " *** " << m_idHelperSvc->toString(u_id) << " "
217 << to_string(u_dir) << std::endl
218 << " *** " << m_idHelperSvc->toString(v_id) << " "
219 << to_string(v_dir));
220 return StatusCode::FAILURE;
221 }
222 const Amg::Vector3D uv_ipoint = u_center + (*uv_isect) * u_dir;
223 const Amg::Vector2D cen_diff = (uv_ipoint - x_center).block<2, 1>(0, 0);
224 if (cen_diff.dot(cen_diff) > std::numeric_limits<float>::epsilon()) {
225 ATH_MSG_ERROR("Expect that the uv strips "
226 << std::endl
227 << " *** " << m_idHelperSvc->toString(u_id) << " "
228 << to_string(u_dir) << std::endl
229 << " *** " << m_idHelperSvc->toString(v_id) << " "
230 << to_string(v_dir)
231 << " intersect at the center of the corresponding x "
232 "strip. But they don't. "
233 << std::endl
234 << to_string(uv_ipoint) << std::endl
235 << " vs." << std::endl
236 << to_string(x_center));
237 }
238 ATH_MSG_DEBUG("Intersection of uv is in " << to_string(uv_ipoint) << " "
239 << to_string(x_center));
240
241 std::optional<double> ux_isect =
242 Amg::intersect<3>(u_center, u_dir, x_center, x_dir);
243 if (!ux_isect) {
244 ATH_MSG_ERROR("Failed to intersect the ux strips for identifiers "
245 << std::endl
246 << " *** " << m_idHelperSvc->toString(v_id) << " "
247 << to_string(u_dir) << std::endl
248 << " *** " << m_idHelperSvc->toString(x_id) << " "
249 << to_string(x_dir));
250 return StatusCode::FAILURE;
251 }
252 ATH_MSG_DEBUG("Intersection of xu is in "
253 << to_string(x_center + (*ux_isect) * x_dir) << " "
254 << to_string(x_center));
255
256 std::optional<double> vx_isect =
257 Amg::intersect<3>(v_center, v_dir, x_center, x_dir);
258 if (!ux_isect) {
259 ATH_MSG_ERROR("Failed to intersect the vx strips for identifiers "
260 << std::endl
261 << " *** " << m_idHelperSvc->toString(v_id) << " "
262 << to_string(v_dir) << std::endl
263 << " *** " << m_idHelperSvc->toString(x_id) << " "
264 << to_string(x_dir));
265 return StatusCode::FAILURE;
266 }
267 ATH_MSG_DEBUG("Intersection of vu is in "
268 << to_string(x_center + (*vx_isect) * x_dir) << " "
269 << to_string(x_center));
270 }
271 return StatusCode::SUCCESS;
272 };
273 if (false)
274 ATH_CHECK(uv_intersects());
275
276 std::unique_ptr<TH1>& surface_histo = m_nswActiveAreas[signed_lay];
277 if (!surface_histo) {
278 ATH_MSG_WARNING("No surface histo has been made for "
279 << m_idHelperSvc->toString(id) << " " << signed_lay);
280 continue;
281 }
282 const Amg::Vector3D& surf_cent = roe->center(id);
283 double d = std::max(std::max(design->xSize(), design->maxYSize()),
284 design->minYSize());
285 for (double x = surf_cent.x() - d; x <= surf_cent.x() + d; x += 1) {
286 for (double y = surf_cent.y() - d; y <= surf_cent.y() + d; y += 1) {
287 const Amg::Vector3D glob_pos{x, y, surf_cent.z()};
288 Amg::Vector2D lpos{Amg::Vector2D::Zero()};
289 if (!roe->surface(id).globalToLocal(glob_pos, glob_pos, lpos))
290 continue;
291 if ((is_mm && mm_roe->insideActiveBounds(id, lpos, 10., 10.)) ||
292 (!is_mm && st_roe->surface(id).insideBounds(lpos, 10., 10.))
293
294 ) {
295 surface_histo->Fill(x, y);
296 }
297 }
298 }
299 }
300 m_alg_run = true;
301 return StatusCode::SUCCESS;
302}
304 const MuonGM::MuonDetectorManager* detMgr{nullptr};
305 ATH_CHECK(detStore()->retrieve(detMgr));
306 const MmIdHelper& id_helper = m_idHelperSvc->mmIdHelper();
307 for (const std::string station : {"MML", "MMS"}) {
308 for (int phi = id_helper.stationPhiMin();
309 phi <= id_helper.stationPhiMax(); ++phi) {
310 for (int eta = -2; eta <= 2; ++eta) {
311 if (eta == 0) {
312 continue;
313 }
314 bool is_valid{false};
315
316 const Identifier station_id = id_helper.elementID(station, eta, phi, is_valid);
317 if (!is_valid) {
318 continue;
319 }
320 for (int ml = id_helper.multilayerMin();
321 ml <= id_helper.multilayerMax(); ++ml) {
322 const Identifier module_id = id_helper.multilayerID(station_id, ml);
323 for (int i_layer = 1; i_layer <= 4; ++i_layer) {
324 const Identifier id =
325 id_helper.channelID(module_id, ml, i_layer, 10);
326 m_nswPads[id] = std::make_unique<TGraph>();
328 int signed_layer = layerId(id);
329 if (!m_nswLayers[signed_layer]) {
330 m_nswLayers[signed_layer] = std::make_unique<TGraph>();
331 }
332 if (!m_nswActiveAreas[signed_layer]) {
333 m_nswActiveAreas[signed_layer] = std::make_unique<TH2D>(
334 std::to_string(m_nswActiveAreas.size()).c_str(),
335 "ActiveNSW;x [mm]; y [mm]", 1000, -5001, 5001., 1000,
336 -5001., 5001.);
337 }
338 }
339 }
340 }
341 }
342 }
343 return StatusCode::SUCCESS;
344}
346 int eta = m_idHelperSvc->stationEta(id);
347 if (m_idHelperSvc->issTgc(id)) {
348 const sTgcIdHelper& id_helper = m_idHelperSvc->stgcIdHelper();
349 int ml = id_helper.multilayer(id);
350 int lay = id_helper.gasGap(id);
351 int type = id_helper.channelType(id);
352 return (8 * (type == sTgcIdHelper::sTgcChannelTypes::Strip) + 4 * (ml - 1) +
353 (lay - 1)) *
354 (eta > 0 ? 1 : -1);
355 }
356 if (m_idHelperSvc->isMM(id)) {
357 const MmIdHelper& id_helper = m_idHelperSvc->mmIdHelper();
358 int ml = id_helper.multilayer(id);
359 int lay = id_helper.gasGap(id);
360 return (16 + 4 * (ml - 1) + (lay - 1)) * (eta > 0 ? 1 : -1);
361 }
362 return -666;
363}
365 const MuonGM::MuonDetectorManager* detMgr{nullptr};
366 ATH_CHECK(detStore()->retrieve(detMgr));
367 const sTgcIdHelper& id_helper = m_idHelperSvc->stgcIdHelper();
368 for (const std::string station : {"STS", "STL"}) {
369 for (int eta = id_helper.stationEtaMin(); eta <= id_helper.stationEtaMax();
370 ++eta) {
371 if (eta == 0)
372 continue;
373 for (int phi = id_helper.stationPhiMin();
374 phi <= id_helper.stationPhiMax(); ++phi) {
375 for (int ml = id_helper.multilayerMin();
376 ml <= id_helper.multilayerMax(); ++ml) {
377 bool is_valid{false};
378 Identifier station_id =
379 id_helper.elementID(station, eta, phi, is_valid);
380 if (!is_valid)
381 continue;
382 const Identifier module_id = id_helper.multilayerID(station_id, ml);
383 if (!detMgr->getsTgcReadoutElement(module_id))
384 continue;
385 for (int lay = 1; lay <= 4; ++lay) {
386 const Identifier strip_id = id_helper.channelID(
387 module_id, ml, lay, sTgcIdHelper::sTgcChannelTypes::Strip, 10);
388 const Identifier wire_id = id_helper.channelID(
389 module_id, ml, lay, sTgcIdHelper::sTgcChannelTypes::Wire, 10);
390 for (const Identifier& id : {strip_id, wire_id}) {
391 m_nswPads[id] = std::make_unique<TGraph>();
392 int signed_layer = layerId(id);
393 if (!m_nswLayers[signed_layer]) {
394 m_nswLayers[signed_layer] = std::make_unique<TGraph>();
395 }
396 if (!m_nswActiveAreas[signed_layer]) {
397 m_nswActiveAreas[signed_layer] = std::make_unique<TH2D>(
398 std::to_string(m_nswActiveAreas.size()).c_str(),
399 "ActiveNSW;x [mm]; y [mm]", 1000, -5001, 5001., 1000,
400 -5001., 5001.);
401 }
402 }
403 }
404 }
405 }
406 }
407 }
408 return StatusCode::SUCCESS;
409}
410}
Scalar eta() const
pseudorapidity method
Scalar phi() const
phi method
const PlainObject unit() const
This is a plugin that makes Eigen look like CLHEP & defines some convenience methods.
#define ATH_CHECK
Evaluate an expression and check for errors.
#define ATH_MSG_ERROR(x)
#define ATH_MSG_FATAL(x)
#define ATH_MSG_INFO(x)
#define ATH_MSG_ALWAYS(x)
#define ATH_MSG_WARNING(x)
#define ATH_MSG_DEBUG(x)
static std::string to_string(const std::vector< T > &v)
#define y
#define x
const ServiceHandle< StoreGateSvc > & detStore() const
TGraph * graph(const std::string &graphName, const std::string &tDir="", const std::string &stream="")
Simplify the retrieval of registered TGraphs.
Identifier channelID(int stationName, int stationEta, int stationPhi, int multilayer, int gasGap, int channel) const
Identifier elementID(int stationName, int stationEta, int stationPhi) const
Identifier multilayerID(const Identifier &channeldID) const
int channel(const Identifier &id) const override
static int stationPhiMin()
int gasGap(const Identifier &id) const override
get the hashes
static int multilayerMin()
static int multilayerMax()
static int stationPhiMax()
int multilayer(const Identifier &id) const
An MMReadoutElement corresponds to a single STGC module; therefore typicaly a barrel muon station con...
virtual int numberOfStrips(const Identifier &layerId) const override final
number of strips per layer
bool insideActiveBounds(const Identifier &id, const Amg::Vector2D &locpos, double tol1=0., double tol2=0.) const
boundary check Wrapper Trk::PlaneSurface::insideBounds() taking into account the passivated width
const MuonChannelDesign * getDesign(const Identifier &id) const
returns the MuonChannelDesign class for the given identifier
virtual const Trk::PlaneSurface & surface() const override
access to chamber surface (phi orientation), uses the first gas gap
The MuonDetectorManager stores the transient representation of the Muon Spectrometer geometry and pro...
const MMReadoutElement * getMMReadoutElement(const Identifier &id) const
access via extended identifier (requires unpacking)
const sTgcReadoutElement * getsTgcReadoutElement(const Identifier &id) const
access via extended identifier (requires unpacking)
std::map< int, std::unique_ptr< TGraph > > m_nswLayers
Map showing the edges of the 16 layers of the NSW.
std::map< Identifier, std::unique_ptr< TGraph > > m_nswPads
Map containing each PCB of the NSW seperately.
std::map< int, std::unique_ptr< TH1 > > m_nswActiveAreas
Map showing the active areas of the NSW to show the passivation.
SG::ReadCondHandleKey< MuonGM::MuonDetectorManager > m_DetectorManagerKey
ServiceHandle< Muon::IMuonIdHelperSvc > m_idHelperSvc
An sTgcReadoutElement corresponds to a single STGC module; therefore typicaly a barrel muon station c...
const MuonChannelDesign * getDesign(const Identifier &id) const
returns the MuonChannelDesign class for the given identifier
virtual int numberOfStrips(const Identifier &layerId) const override final
number of strips per layer
const std::string & stationNameString(const int &index) const
virtual bool insideBounds(const Amg::Vector2D &locpos, double tol1=0., double tol2=0.) const override
This method calls the inside() method of the Bounds.
virtual bool globalToLocal(const Amg::Vector3D &glob, const Amg::Vector3D &mom, Amg::Vector2D &loc) const =0
Specified by each surface type: GlobalToLocal method without dynamic memory allocation - boolean chec...
virtual void localToGlobal(const Amg::Vector2D &locp, const Amg::Vector3D &mom, Amg::Vector3D &glob) const =0
Specified by each surface type: LocalToGlobal method without dynamic memory allocation.
This is the base class for all tracking detector elements with read-out relevant information.
virtual const Amg::Vector3D & center() const =0
Return the center of the element.
virtual const Surface & surface() const =0
Return surface associated with this detector element.
int multilayer(const Identifier &id) const
static int stationPhiMax()
int channelType(const Identifier &id) const
Identifier elementID(int stationName, int stationEta, int stationPhi) const
static int multilayerMax()
static int stationPhiMin()
static int stationEtaMin()
static int multilayerMin()
static int stationEtaMax()
int gasGap(const Identifier &id) const override
get the hashes
Identifier channelID(int stationName, int stationEta, int stationPhi, int multilayer, int gasGap, int channelType, int channel) const
Identifier multilayerID(const Identifier &channeldID) const
std::optional< double > intersect(const AmgVector(N)&posA, const AmgVector(N)&dirA, const AmgVector(N)&posB, const AmgVector(N)&dirB)
Calculates the point B' along the line B that's closest to a second line A.
Eigen::Matrix< double, 2, 1 > Vector2D
Eigen::Matrix< double, 3, 1 > Vector3D
Ensure that the Athena extensions are properly loaded.
Definition GeoMuonHits.h:27
const T * get(const ReadCondHandleKey< T > &key, const EventContext &ctx)
Convenience function to retrieve an object given a ReadCondHandleKey.
bool center(int channel, Amg::Vector2D &pos) const
STRIPS ONLY: Returns the center on the strip.
bool leftEdge(int channel, Amg::Vector2D &pos) const
STRIPS ONLY: Returns the left edge of the strip.
int numberOfMissingBottomStrips() const
Returns the number of missing bottom strips.
bool rightEdge(int channel, Amg::Vector2D &pos) const
STRIPS ONLY: Returns the right edge of the strip.
double hasStereoAngle() const
returns whether the stereo angle is non-zero
int channelNumber(const Amg::Vector2D &pos) const
calculate local channel number, range 1=nstrips like identifiers. Returns -1 if out of range