// $Id$ // // File: DFDCIntersection_factory.cc // Created: Tue Oct 30 11:24:53 EDT 2007 // Creator: davidl (on Darwin fwing-dhcp95.jlab.org 8.10.1 i386) // #include using namespace std; #include "DFDCIntersection_factory.h" #include "FDC/DFDCGeometry.h" #include "HDGEOMETRY/DGeometry.h" //------------------ // init //------------------ jerror_t DFDCIntersection_factory::init(void) { MAX_DIST2 = 2.0*2.0; DEBUG_LEVEL=0; // Create skeleton of data structure to hold hits vector mt_trkhits; vector > mt_trkhits_by_layer; for(int i=0; i<6; i++)mt_trkhits_by_layer.push_back(mt_trkhits); for(int i=0; i<4; i++)fdchits_by_package.push_back(mt_trkhits_by_layer); return NOERROR; } //------------------ // brun //------------------ jerror_t DFDCIntersection_factory::brun(JEventLoop *loop, int runnumber) { // Get pointer to DGeometry object DApplication* dapp=dynamic_cast(eventLoop->GetJApplication()); DGeometry *dgeom = dapp->GetDGeometry(runnumber); if (!dgeom->GetFDCWires(fdcwires)){ _DBG_<< "FDC geometry not available!" < fdchits; loop->Get(fdchits); // For events with a very large number of hits, assume // we can't reconstruct them so bail early // Feb. 8, 2008 D.L. if(fdchits.size()>(5+5+1)*24*4){ if (DEBUG_LEVEL>0){ _DBG_<<"Too many hits in FDC! Intersection point reconstruction in FDC bypassed for event "<GetJEvent().GetEventNumber()<type != 0)continue; // filter out cathode hits if(fdchit->gLayer<1 || fdchit->gLayer>24){ _DBG_<<"FDC gLayer out of range! ("<gLayer<<" is not between 1 and 24)"<element!=wire || fdchit->gLayer!=layer){ int package = (fdchit->gLayer-1)/6 + 1; fdchits_by_package[package-1][(fdchit->gLayer-1)%6].push_back(fdchit); } wire=fdchit->element; layer=fdchit->gLayer; } // Loop over packages, creating intersection points for(unsigned int package=1; package<=4; package++){ MakeIntersectionPoints(fdchits_by_package[package-1]); //MakeRestrictedIntersectionPoints(fdchits_by_package[package-1]); } return NOERROR; } //------------------ // MakeIntersectionPoints //------------------ void DFDCIntersection_factory::MakeIntersectionPoints(vector >&hits_by_layer) { // hits_by_layer should contain all of the wire hits in a single // FDC package. They are stored in set of nested vectors with the // outer vector being the list of layers (always 6 in length) // and the inner vector the hits within the layer. // Loop over layers for(unsigned int i=0; i &layer1 = hits_by_layer[i]; vector &layer2 = hits_by_layer[i+1]; FindIntersections(layer1, layer2, _data); } } //------------------ // MakeRestrictedIntersectionPoints //------------------ void DFDCIntersection_factory::MakeRestrictedIntersectionPoints(vector >&hits_by_layer) { // hits_by_layer should contain all of the wire hits in a single // FDC package. They are stored in a set of nested vectors with the // outer vector being the list of layers (always 6 in length) // and the inner vector the hits within the layer. // This method will first make a list of all the intersection points // between adjacent layers in the package. It will then make a list // which intersection points to keep by finding those that meet // one of the following criteria: // // 1. For one of the wires used, this is the only intersection // point at this z location. // // 2. The intersection point has a corresponding intersection // point in an adjacent plane (i.e. a triple wire coincidence) // // Note that there is a class of hits this method fails to accept // at the moment. This is when 2 wire planes have 2 hits each leading // to 4 intersections, only 2 of which are real. If the adjacent // plane has only one hit that is in coincidence with only // 1 of the points, the other "true" point could be inferred. At this // point, the second true point will not be kept in these cases // (or similar ones with more tracks). // First, get all of the intersection points vector > intersections(hits_by_layer.size()-1); for(unsigned int i=0; i > upstr; map > dnstr; for(unsigned int i=0; i &layer = intersections[i]; for(unsigned int j=0; jwire2].push_back(fdcinter); dnstr[fdcinter->wire1].push_back(fdcinter); } } // Keep track of the "good" intersection objects in a map. map intersects_to_keep; // Loop over both the upstr and dnstr lists to find wires // with only one hit in either its upstream or downstream // plane. map >::iterator iter; for(iter=upstr.begin(); iter!=upstr.end(); iter++){ vector &hits = iter->second; if(hits.size()==1)intersects_to_keep[hits[0]] = true; } for(iter=dnstr.begin(); iter!=dnstr.end(); iter++){ vector &hits = iter->second; if(hits.size()==1)intersects_to_keep[hits[0]] = true; } // Find triple wire plane coincidences // Loop over intersection layers for(unsigned int i=0; i &layer1 = intersections[i]; for(unsigned int j=0; j &layer2 = intersections[i+1]; for(unsigned int k=0; kwire2 != int2->wire1)continue; // Find distance between hits double dx = int1->pos.X() - int2->pos.X(); double dy = int1->pos.Y() - int2->pos.Y(); double dist2 = dx*dx + dy*dy; if(dist2 &layer1, vector &layer2, vector &intersections) { // Loop over hits in layer1 for(unsigned int j=0; jgLayer-1][hit1->element-1]; if(!wire1){ _DBG_<<"No wire for layer="<gLayer<<" wire="<element<<" !!"<origin.X(), wire1->origin.Y()); DVector2 b(wire1->udir.X(), wire1->udir.Y()); b /= b.Mod(); // Loop over hits in layer2 for(unsigned int k=0; kgLayer-1][hit2->element-1]; if(!wire2){ _DBG_<<"No wire for layer="<gLayer<<" wire="<element<<" !!"<origin.X(), wire2->origin.Y()); DVector2 d(wire2->udir.X(), wire2->udir.Y()); d /= d.Mod(); // Find intersection of the projections of wires 1 and 2 on the X/Y plane DVector2 e = a - c; double gamma = d*b; double beta = (d*e - gamma*(b*e))/(1.0-gamma*gamma); if(!isfinite(beta))continue; // wires must be parallel if(fabs(beta) > wire2->L/2.0)continue; // intersection is past end of wire double alpha = beta*gamma - b*e; if(fabs(alpha) > wire1->L/2.0)continue; // intersection is past end of wire DVector2 x = c + beta*d; // intersection point in X/Y // Add the intersection point DFDCIntersection *fdcint = new DFDCIntersection; fdcint->hit1 = hit1; fdcint->hit2 = hit2; fdcint->wire1 = wire1; fdcint->wire2 = wire2; fdcint->pos.SetXYZ(x.X(), x.Y(), (wire1->origin.Z()+wire2->origin.Z())/2.0); intersections.push_back(fdcint); } } }