tracking_validation.cpp 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411
  1. #include <iostream>
  2. #include <vector>
  3. #include <map>
  4. #include <utility>
  5. #include <numeric>
  6. #include <limits>
  7. #include <cmath>
  8. #include <TSystem.h>
  9. #include "filval/filval.hpp"
  10. #include "filval/root/filval.hpp"
  11. #include <boost/range/combine.hpp>
  12. #include <boost/format.hpp>
  13. #include "TrackingNtuple.h"
  14. #include "analysis/obj_types.cpp"
  15. using namespace std;
  16. using namespace fv;
  17. using namespace fv::root;
  18. typedef std::tuple<Seed, PixRecHit, Track, SimHit, SimTrack> MatchedTrack;
  19. typedef std::pair<std::vector<float>,std::vector<float>> pair_vec;
  20. #define HIT_TYPE_PIXEL 0
  21. #define HIT_TYPE_GLUED 1
  22. #define HIT_TYPE_STRIP 2
  23. #define PIXEL_BARREL 1
  24. #define PIXEL_ENDCAP 2
  25. std::map<int,string> subdet_names = {{1, "BPIX"}, {2, "FPIX"}};
  26. template<typename A, typename B>
  27. float displacement(const A& a, const B& b){
  28. return std::sqrt(pow(a.x-b.x, 2) +
  29. pow(a.x-b.x, 2) +
  30. pow(a.x-b.x, 2));
  31. }
  32. bool in_det(const MatchedTrack &mt, int &&det, int &&layer){
  33. auto& hit = std::get<PixRecHit>(mt);
  34. return hit.det == det && hit.lay == layer;
  35. };
  36. vector<string> seedTypes =
  37. {"initialStepSeeds",
  38. "highPtTripletStepSeeds",
  39. "mixedTripletStepSeeds",
  40. "pixelLessStepSeeds",
  41. "tripletElectronSeeds",
  42. "pixelPairElectronSeeds",
  43. "stripPairElectronSeeds"};
  44. Value<vector<Seed>>* seeds;
  45. Value<vector<PixRecHit>>* pixrec_hits;
  46. Value<vector<Track>>* tracks;
  47. Value<vector<SimHit>>* sim_hits;
  48. Value<vector<SimTrack>>* sim_tracks;
  49. void register_objects(TrackingDataSet& tds){
  50. seeds = register_seeds(tds);
  51. pixrec_hits = register_pixrec_hits(tds);
  52. tracks = register_tracks(tds);
  53. sim_hits = register_sim_hits(tds);
  54. sim_tracks = register_sim_tracks(tds);
  55. }
  56. void setup_first_hit_pairs(TrackingDataSet& tds){
  57. // Finds pairs of (rechit, simhit) where the rechit is the innermost hit on
  58. // a seed in a particular barrel layer.
  59. typedef std::tuple<PixRecHit, SimHit> HitPair;
  60. auto& matched_hits_in_layer =
  61. func<vector<HitPair>(vector<Seed>,
  62. vector<PixRecHit>,
  63. vector<SimHit>, int)>("find_matched_innermost_hits_in_layer",
  64. FUNC(([](const vector<Seed>& seeds,
  65. const vector<PixRecHit>& pixrec_hits,
  66. const vector<SimHit>& sim_hits,
  67. const int&& bpix_layer){
  68. vector<HitPair> matched_hits;
  69. for(const Seed &seed : seeds){
  70. if( seed.algoOriginal < 0 || seed.algoOriginal >= seedTypes.size()) continue;
  71. for(auto tup : boost::combine(seed.hitIdx, seed.hitType)){ //for hits in track's seed
  72. int hitIdx, hitType;
  73. boost::tie(hitIdx, hitType) = tup;
  74. if(hitType != HIT_TYPE_PIXEL) continue; // take only pixel hits for now
  75. const PixRecHit &rec_hit = pixrec_hits[hitIdx];
  76. if(rec_hit.det == PIXEL_BARREL && rec_hit.lay == bpix_layer){
  77. if(rec_hit.simHitIdx.size() > 0){
  78. matched_hits.push_back({rec_hit, sim_hits[rec_hit.simHitIdx[0]]});
  79. }
  80. }
  81. break;
  82. }
  83. }
  84. return matched_hits;
  85. })));
  86. auto first_hits_in_B1 = fv::apply(matched_hits_in_layer, fv::tuple(seeds, pixrec_hits, sim_hits, constant("1", 1)), "first_hits_in_B1");
  87. auto first_hits_in_B2 = fv::apply(matched_hits_in_layer, fv::tuple(seeds, pixrec_hits, sim_hits, constant("2", 2)), "first_hits_in_B2");
  88. auto& calc_dphi = func<float(HitPair)>("calc_dphi",
  89. FUNC(([](const HitPair& hit_pair){
  90. const auto &rec_hit = std::get<PixRecHit>(hit_pair);
  91. const auto &sim_hit = std::get<SimHit>(hit_pair);
  92. return atan2(rec_hit.x, rec_hit.y) - atan2(sim_hit.x, sim_hit.y);
  93. })));
  94. TH1Params params = {"$\\Delta \\phi$(rad)", 50, -0.001, 0.001, ""};
  95. tds.register_container<ContainerTH1Many<float>>("dphi_matched_hits_from_B1",
  96. fv::map(calc_dphi, first_hits_in_B1),
  97. "Matched Hits $\\Delta \\phi$ - B1", params);
  98. tds.register_container<ContainerTH1Many<float>>("dphi_matched_hits_from_B2",
  99. fv::map(calc_dphi, first_hits_in_B2),
  100. "Matched Hits $\\Delta \\phi$ - B2", params);
  101. auto& calc_dz = func<float(HitPair)>("calc_dz",
  102. FUNC(([](const HitPair& hit_pair){
  103. const auto &rec_hit = std::get<PixRecHit>(hit_pair);
  104. const auto &sim_hit = std::get<SimHit>(hit_pair);
  105. return rec_hit.z - sim_hit.z;
  106. })));
  107. params = {"$\\Delta z$", 50, -0.01, 0.01, ""};
  108. tds.register_container<ContainerTH1Many<float>>("dz_matched_hits_from_B1",
  109. fv::map(calc_dz, first_hits_in_B1),
  110. "Matched Hits $\\Delta z$ - B1", params);
  111. tds.register_container<ContainerTH1Many<float>>("dz_matched_hits_from_B2",
  112. fv::map(calc_dz, first_hits_in_B2),
  113. "Matched Hits $\\Delta z$ - B2", params);
  114. }
  115. void setup_matched_tracks(TrackingDataSet& tds){
  116. // Finds sets of (seed, rechit, track, simhit sim_track) where the cycle
  117. // track->seed->rec_hit->sim_hit->sim_track->track
  118. // is closed
  119. auto& find_matched_tracks =
  120. func<vector<MatchedTrack>(vector<Seed>,
  121. vector<PixRecHit>,
  122. vector<Track>,
  123. vector<SimHit>,
  124. vector<SimTrack>)>("find_matched_tracks",
  125. FUNC(([](const vector<Seed>& seeds,
  126. const vector<PixRecHit> pixrec_hits,
  127. const vector<Track>& tracks,
  128. const vector<SimHit> sim_hits,
  129. const vector<SimTrack> sim_tracks){
  130. INFO("New event");
  131. INFO(boost::format("Number of tracks is %d, number of seeds is %d, number of hits %d, number of simhits %d")
  132. % tracks.size() % seeds.size() % pixrec_hits.size() % sim_hits.size());
  133. vector<MatchedTrack> matched_tracks;
  134. for(const Track &track : tracks){
  135. const Seed& seed = seeds[track.seedIdx];
  136. if( seed.algoOriginal < 0 || seed.algoOriginal >= seedTypes.size()) continue;
  137. INFO(boost::format(" seed indx= %d algorithm= %d type %s\n")
  138. % track.seedIdx % seed.algoOriginal % seedTypes[seed.algoOriginal]);
  139. for(auto tup : boost::combine(seed.hitIdx, seed.hitType)){ //for hits in track's seed
  140. int hitIdx, hitType;
  141. boost::tie(hitIdx, hitType) = tup;
  142. if(hitType != HIT_TYPE_PIXEL) continue; // take only pixel hits for now
  143. const PixRecHit &rec_hit = pixrec_hits[hitIdx];
  144. TVector3 recoHitPoint;
  145. recoHitPoint.SetXYZ( rec_hit.x, rec_hit.y, rec_hit.z);
  146. float Rreco = recoHitPoint.Perp();
  147. INFO(boost::format(" hit= %d is in %s-L%d at radius= %f\n")
  148. % hitIdx % subdet_names[rec_hit.det] % rec_hit.lay % Rreco);
  149. for(const int& simHitIdx : rec_hit.simHitIdx){ // for sim-hits matching rec-hit
  150. const SimHit &sim_hit = sim_hits[simHitIdx];
  151. const SimTrack &sim_track = sim_tracks[sim_hit.simTrkIdx];
  152. for(const int &trkIdx : sim_track.trkIdx){ // for tracks matched to this sim_track
  153. if(trkIdx == track.idx){ // sim_track matches with our original track
  154. matched_tracks.push_back({seed, rec_hit, track, sim_hit, sim_track});
  155. break;
  156. }
  157. }
  158. }
  159. }
  160. }
  161. return matched_tracks;
  162. })));
  163. fv::apply(find_matched_tracks, fv::tuple(seeds, pixrec_hits, tracks, sim_hits, sim_tracks), "matched_tracks");
  164. }
  165. void setup_residuals(TrackingDataSet &tds){
  166. auto matched_tracks = lookup<vector<MatchedTrack>>("matched_tracks");
  167. // Break matched_tracks into catagories based on Barrel/Endcap and layer
  168. auto matched_tracks_B1 = filter(func<bool(MatchedTrack)>("matched_tracks_B1",
  169. FUNC(([](const MatchedTrack& matched_track){
  170. return in_det(matched_track, PIXEL_BARREL, 1);
  171. }))), matched_tracks);
  172. auto matched_tracks_B2 = filter(func<bool(MatchedTrack)>("matched_tracks_B2",
  173. FUNC(([](const MatchedTrack& matched_track){
  174. return in_det(matched_track, PIXEL_BARREL, 2);
  175. }))), matched_tracks);
  176. auto matched_tracks_B3 = filter(func<bool(MatchedTrack)>("matched_tracks_B3",
  177. FUNC(([](const MatchedTrack& matched_track){
  178. return in_det(matched_track, PIXEL_BARREL, 3);
  179. }))), matched_tracks);
  180. auto matched_tracks_B4 = filter(func<bool(MatchedTrack)>("matched_tracks_B4",
  181. FUNC(([](const MatchedTrack& matched_track){
  182. return in_det(matched_track, PIXEL_BARREL, 4);
  183. }))), matched_tracks);
  184. auto matched_tracks_F1 = filter(func<bool(MatchedTrack)>("matched_tracks_F1",
  185. FUNC(([](const MatchedTrack& matched_track){
  186. return in_det(matched_track, PIXEL_ENDCAP, 1);
  187. }))), matched_tracks);
  188. auto matched_tracks_F2 = filter(func<bool(MatchedTrack)>("matched_tracks_F2",
  189. FUNC(([](const MatchedTrack& matched_track){
  190. return in_det(matched_track, PIXEL_ENDCAP, 2);
  191. }))), matched_tracks);
  192. auto matched_tracks_F3 = filter(func<bool(MatchedTrack)>("matched_tracks_F3",
  193. FUNC(([](const MatchedTrack& matched_track){
  194. return in_det(matched_track, PIXEL_ENDCAP, 3);
  195. }))), matched_tracks);
  196. TH2Params params = {"$\\eta$", 100, -4, 4,
  197. "Residuals(cm)", 100, 0, .01};
  198. auto& calc_residuals_v_eta = func<pair_vec(vector<MatchedTrack>)>("matched_track_hit_residuals",
  199. FUNC(([](const vector<MatchedTrack>& matched_tracks){
  200. vector<float> residuals;
  201. vector<float> etas;
  202. for(auto matched_track : matched_tracks){
  203. auto& pixrec_hit = std::get<PixRecHit>(matched_track);
  204. auto& sim_hit = std::get<SimHit>(matched_track);
  205. auto& sim_track = std::get<SimTrack>(matched_track);
  206. residuals.push_back(displacement(pixrec_hit, sim_hit));
  207. etas.push_back(sim_track.eta);
  208. }
  209. return std::make_pair(etas, residuals);
  210. })));
  211. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_all",
  212. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks)),
  213. "Matched Track Residuals - All", params);
  214. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_B1",
  215. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks_B1)),
  216. "Matched Track Residuals - B1", params);
  217. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_B2",
  218. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks_B2)),
  219. "Matched Track Residuals - B2", params);
  220. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_B3",
  221. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks_B3)),
  222. "Matched Track Residuals - B3", params);
  223. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_B4",
  224. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks_B4)),
  225. "Matched Track Residuals - B4", params);
  226. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_F1",
  227. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks_F1)),
  228. "Matched Track Residuals - F1", params);
  229. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_F2",
  230. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks_F2)),
  231. "Matched Track Residuals - F2", params);
  232. tds.register_container<ContainerTH2Many<float>>("matched_track_residuals_v_eta_F3",
  233. fv::apply(calc_residuals_v_eta, fv::tuple(matched_tracks_F3)),
  234. "Matched Track Residuals - F3", params);
  235. params = {"$\\eta$", 100, -4, 4,
  236. "$\\Delta \\phi$(rad)", 100, -.002, .002};
  237. auto& calc_dphi_v_eta = func<pair_vec(vector<MatchedTrack>)>("matched_track_hit_dphis",
  238. FUNC(([](const vector<MatchedTrack>& matched_tracks){
  239. vector<float> dphis;
  240. vector<float> etas;
  241. for(auto matched_track : matched_tracks){
  242. auto& pixrec_hit = std::get<PixRecHit>(matched_track);
  243. auto& sim_hit = std::get<SimHit>(matched_track);
  244. auto& sim_track = std::get<SimTrack>(matched_track);
  245. dphis.push_back(atan2(sim_hit.x, sim_hit.y) - atan2(pixrec_hit.x, pixrec_hit.y));
  246. etas.push_back(sim_track.eta);
  247. }
  248. return std::make_pair(etas, dphis);
  249. })));
  250. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_all",
  251. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks)),
  252. "Matched Track $\\Delta \\phi$ - All", params);
  253. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_B1",
  254. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks_B1)),
  255. "Matched Track $\\Delta \\phi$ - B1", params);
  256. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_B2",
  257. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks_B2)),
  258. "Matched Track $\\Delta \\phi$ - B2", params);
  259. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_B3",
  260. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks_B3)),
  261. "Matched Track $\\Delta \\phi$ - B3", params);
  262. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_B4",
  263. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks_B4)),
  264. "Matched Track $\\Delta \\phi$ - B4", params);
  265. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_F1",
  266. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks_F1)),
  267. "Matched Track $\\Delta \\phi$ - F1", params);
  268. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_F2",
  269. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks_F2)),
  270. "Matched Track $\\Delta \\phi$ - F2", params);
  271. tds.register_container<ContainerTH2Many<float>>("matched_track_dphis_v_eta_F3",
  272. fv::apply(calc_dphi_v_eta, fv::tuple(matched_tracks_F3)),
  273. "Matched Track $\\Delta \\phi$ - F3", params);
  274. auto& calc_dz_v_eta = func<pair_vec(vector<MatchedTrack>)>("matched_track_hit_dphis",
  275. FUNC(([](const vector<MatchedTrack>& matched_tracks){
  276. vector<float> dzs;
  277. vector<float> etas;
  278. for(auto matched_track : matched_tracks){
  279. auto& pixrec_hit = std::get<PixRecHit>(matched_track);
  280. auto& sim_hit = std::get<SimHit>(matched_track);
  281. auto& sim_track = std::get<SimTrack>(matched_track);
  282. dzs.push_back(sim_hit.z - pixrec_hit.z);
  283. etas.push_back(sim_track.eta);
  284. }
  285. return std::make_pair(etas, dzs);
  286. })));
  287. params = {"$\\eta$", 100, -4, 4,
  288. "$\\Delta z$(cm)", 100, -.002, .002};
  289. tds.register_container<ContainerTH2Many<float>>("matched_track_dzs_v_eta_all",
  290. fv::apply(calc_dz_v_eta, fv::tuple(matched_tracks)),
  291. "Matched Track $\\Delta z$ - All", params);
  292. tds.register_container<ContainerTH2Many<float>>("matched_track_dzs_v_eta_B1",
  293. fv::apply(calc_dz_v_eta, fv::tuple(matched_tracks_B1)),
  294. "Matched Track $\\Delta z$ - B1", params);
  295. tds.register_container<ContainerTH2Many<float>>("matched_track_dzs_v_eta_B2",
  296. fv::apply(calc_dz_v_eta, fv::tuple(matched_tracks_B2)),
  297. "Matched Track $\\Delta z$ - B2", params);
  298. tds.register_container<ContainerTH2Many<float>>("matched_track_dzs_v_eta_B3",
  299. fv::apply(calc_dz_v_eta, fv::tuple(matched_tracks_B3)),
  300. "Matched Track $\\Delta z$ - B3", params);
  301. tds.register_container<ContainerTH2Many<float>>("matched_track_dzs_v_eta_B4",
  302. fv::apply(calc_dz_v_eta, fv::tuple(matched_tracks_B4)),
  303. "Matched Track $\\Delta z$ - B4", params);
  304. auto& calc_dr_v_eta = func<pair_vec(vector<MatchedTrack>)>("matched_track_hit_drs",
  305. FUNC(([](const vector<MatchedTrack>& matched_tracks){
  306. vector<float> drs;
  307. vector<float> etas;
  308. for(auto matched_track : matched_tracks){
  309. auto& rec_hit = std::get<PixRecHit>(matched_track);
  310. auto& sim_hit = std::get<SimHit>(matched_track);
  311. float sim_hit_r = sqrt(pow(sim_hit.x, 2) + pow(sim_hit.y, 2));
  312. float rec_hit_r = sqrt(pow(rec_hit.x, 2) + pow(rec_hit.y, 2));
  313. drs.push_back(sim_hit_r - rec_hit_r);
  314. etas.push_back(std::get<SimTrack>(matched_track).eta);
  315. }
  316. return std::make_pair(etas, drs);
  317. })));
  318. params = {"$\\eta$", 100, -4, 4,
  319. "$\\Delta r$(cm)", 100, -.002, .002};
  320. tds.register_container<ContainerTH2Many<float>>("matched_track_drs_v_eta_F1",
  321. fv::apply(calc_dr_v_eta, fv::tuple(matched_tracks_F1)),
  322. "Matched Track $\\Delta r$ - F1", params);
  323. tds.register_container<ContainerTH2Many<float>>("matched_track_drs_v_eta_F2",
  324. fv::apply(calc_dr_v_eta, fv::tuple(matched_tracks_F2)),
  325. "Matched Track $\\Delta r$ - F2", params);
  326. tds.register_container<ContainerTH2Many<float>>("matched_track_drs_v_eta_F3",
  327. fv::apply(calc_dr_v_eta, fv::tuple(matched_tracks_F3)),
  328. "Matched Track $\\Delta r$ - F3", params);
  329. }
  330. void run_analysis(const vector<fv::util::DataFileDescriptor>& dfds, const string output_filename, bool silent){
  331. gSystem->Load("libfilval.so");
  332. auto replace_suffix = [](const std::string& input, const std::string& new_suffix){
  333. return input.substr(0, input.find_last_of(".")) + new_suffix;
  334. };
  335. string log_filename = replace_suffix(output_filename, ".log");
  336. fv::util::Log::init_logger(log_filename, fv::util::LogPriority::kLogDebug);
  337. TrackingDataSet tds(output_filename, dfds, "trackingNtuple/tree");
  338. /* tds.set_max_events(10); */
  339. register_objects(tds);
  340. setup_matched_tracks(tds);
  341. setup_residuals(tds);
  342. setup_first_hit_pairs(tds);
  343. tds.process(silent);
  344. tds.save_all();
  345. }
  346. int main(int argc, char * argv[]){
  347. fv::util::ArgParser args(argc, argv);
  348. bool silent = args.cmdOptionExists("-s");
  349. string output_filename = args.cmdOptionExists("-o") ? args.getCmdOption("-o") : "output.root";
  350. if(args.cmdOptionExists("-F")){
  351. auto file_list = fv::util::read_input_list(args.getCmdOption("-F"));
  352. run_analysis(file_list, output_filename, silent);
  353. }else if(args.cmdOptionExists("-f")){
  354. string input_filename = args.getCmdOption("-f");
  355. string data_label = args.cmdOptionExists("-l") ? args.getCmdOption("-l") : "";
  356. fv::util::DataFileDescriptor dfd(input_filename, data_label);
  357. run_analysis(std::vector<fv::util::DataFileDescriptor>({dfd}), output_filename, silent);
  358. }else {
  359. cout << "Usage: ./tracking_validation (-s) {-o output_filename} -F datafiles.txt" << endl;
  360. cout << " ./tracking_validation (-s) {-l DATA_LABEL} {-o output_filename} -f treefile.root" << endl;
  361. }
  362. }