Skip to content
Snippets Groups Projects
THcDC.cxx 52.5 KiB
Newer Older
/** \class THcDC
    \ingroup Detectors

\brief Analyze a package of horizontal drift chambers.

Uses the
first letter of the apparatus name as a prefix to parameter names.  The
paramters, read in the Setup method, determine the number of chambers and
the number of parameters per plane.

\author S. A. Wood, based on Fortran ENGINE

*/

#include "THcDC.h"
Sylvester Joosten's avatar
Sylvester Joosten committed
#include "TClonesArray.h"
#include "THaApparatus.h"
#include "THaCutList.h"
#include "THaDetMap.h"
Sylvester Joosten's avatar
Sylvester Joosten committed
#include "THaEvData.h"
#include "THaTrack.h"
#include "THcDCTrack.h"
#include "THcDetectorMap.h"
#include "THcGlobals.h"
Sylvester Joosten's avatar
Sylvester Joosten committed
#include "THcHallCSpectrometer.h"
#include "THcParmList.h"
#include "TMath.h"
Sylvester Joosten's avatar
Sylvester Joosten committed
#include "VarDef.h"
#include "VarType.h"

#include <cstdio>
#include <cstdlib>
Sylvester Joosten's avatar
Sylvester Joosten committed
#include <cstring>
#include <iostream>

Sylvester Joosten's avatar
Sylvester Joosten committed
#include "spdlog/sinks/basic_file_sink.h"    // support for basic file logging
#include "spdlog/sinks/stdout_color_sinks.h" //support for stdout logging
Sylvester Joosten's avatar
Sylvester Joosten committed
#include "spdlog/spdlog.h"
using namespace std;

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
THcDC::THcDC(const char* name, const char* description, THaApparatus* apparatus)
    : THaTrackingDetector(name, description, apparatus) {
  // Constructor

Sylvester Joosten's avatar
Sylvester Joosten committed
  fNPlanes               = 0; // No planes until we make them
  _sub_logger            = _det_logger->clone("THcDC");
  fXCenter               = NULL;
  fYCenter               = NULL;
  fMinHits               = NULL;
  fMaxHits               = NULL;
  fMinCombos             = NULL;
  fSpace_Point_Criterion = NULL;
Sylvester Joosten's avatar
Sylvester Joosten committed
  fTdcWinMin     = NULL;
  fTdcWinMax     = NULL;
  fCentralTime   = NULL;
  fNWires        = NULL;
  fNChamber      = NULL;
  fWireOrder     = NULL;
Sylvester Joosten's avatar
Sylvester Joosten committed
  fReadoutTB     = NULL;
  fReadoutLR     = NULL;

  fXPos          = NULL;
  fYPos          = NULL;
  fZPos          = NULL;
  fAlphaAngle    = NULL;
  fBetaAngle     = NULL;
  fGammaAngle    = NULL;
  fPitch         = NULL;
  fCentralWire   = NULL;
Sylvester Joosten's avatar
Sylvester Joosten committed
  fSigma         = NULL;
  // These should be set to zero (in a parameter file) in order to
  // replicate historical ENGINE behavior
Sylvester Joosten's avatar
Sylvester Joosten committed
  fFixLR                    = 1;
Sylvester Joosten's avatar
Sylvester Joosten committed
  fProjectToChamber         = 0; // Use 1 for SOS chambers
Sylvester Joosten's avatar
Sylvester Joosten committed
  fDCTracks = new TClonesArray("THcDCTrack", 20);
Sylvester Joosten's avatar
Sylvester Joosten committed
  fNChamHits   = 0;
  fPlaneEvents = 0;
Sylvester Joosten's avatar
Sylvester Joosten committed
  // The version defaults to 0 (old HMS style). 1 is new HMS style and 2 is SHMS style.
Sylvester Joosten's avatar
Sylvester Joosten committed
  // Create and return a shared_ptr to a multithreaded console logger.
  //_logger = spdlog::get("config");
Sylvester Joosten's avatar
Sylvester Joosten committed
  // if(!_logger) {
  //  _logger = spdlog::stdout_color_mt("config");
  //}
Whitney Armstrong's avatar
Whitney Armstrong committed
  _sub_logger->debug("THcDC constructor");
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
void THcDC::Setup(const char* name, const char* description) {
Sylvester Joosten's avatar
Sylvester Joosten committed
  Bool_t optional = true;
  // Create the chamber and plane objects using parameters.
  static const char* const here = "Setup";
Sylvester Joosten's avatar
Sylvester Joosten committed
  THaApparatus* app = GetApparatus();
  if (app) {
Eric Pooser's avatar
Eric Pooser committed
    // cout << app->GetName() << endl;
Sylvester Joosten's avatar
Sylvester Joosten committed
    fPrefix[0] = tolower(app->GetName()[0]);
    fPrefix[1] = '\0';
  } else {
    cout << "No apparatus found" << endl;
Sylvester Joosten's avatar
Sylvester Joosten committed
    fPrefix[0] = '\0';
  // For now, decide chamber style from the spectrometer name.
  // Should override with a paramter
Sylvester Joosten's avatar
Sylvester Joosten committed
  // cout<<"HMS Style??\t"<<fHMSStyleChambers<<endl;
  string    planenamelist;
  DBRequest list[] = {{"dc_num_planes", &fNPlanes, kInt},
                      {"dc_num_chambers", &fNChambers, kInt},
                      {"dc_tdc_time_per_channel", &fNSperChan, kDouble},
                      {"dc_wire_velocity", &fWireVelocity, kDouble},
                      {"dc_plane_names", &planenamelist, kString},
                      {"dc_version", &fVersion, kInt, 0, optional},
                      {"dc_tdcrefcut", &fTDC_RefTimeCut, kInt, 0, 1},
                      {0}};

  fTDC_RefTimeCut = 0; // Minimum allowed reference times
  gHcParms->LoadParmValues((DBRequest*)&list, fPrefix);

  if (fVersion == 0) {
Mark Jones's avatar
Mark Jones committed
    fHMSStyleChambers = 1;
  } else {
    fHMSStyleChambers = 0;
  }

Whitney Armstrong's avatar
Whitney Armstrong committed
  _det_logger->info("Plane Name List: {}", planenamelist);
  _det_logger->info("Drift Chambers: {} planes in {} chambers", fNPlanes, fNChambers);
Sylvester Joosten's avatar
Sylvester Joosten committed
  // cout << "Plane Name List: " << planenamelist << endl;
  // cout << "Drift Chambers: " <<  fNPlanes << " planes in " << fNChambers << " chambers" << endl;

  vector<string> plane_names = vsplit(planenamelist);

Sylvester Joosten's avatar
Sylvester Joosten committed
  if (plane_names.size() != (UInt_t)fNPlanes) {
    // cout << "ERROR: Number of planes " << fNPlanes << " doesn't agree with number of plane names
    // " << plane_names.size() << endl;
Whitney Armstrong's avatar
Whitney Armstrong committed
    _det_logger->error("ERROR: Number of planes {} doesn't agree with number of plane names {}",
Sylvester Joosten's avatar
Sylvester Joosten committed
                       fNPlanes, plane_names.size());
    // Should quit.  Is there an official way to quit?
  }
Sylvester Joosten's avatar
Sylvester Joosten committed
  fPlaneNames = new char*[fNPlanes];
  for (Int_t i = 0; i < fNPlanes; i++) {
    fPlaneNames[i] = new char[plane_names[i].length() + 1];
    strcpy(fPlaneNames[i], plane_names[i].c_str());
  }

Sylvester Joosten's avatar
Sylvester Joosten committed
  char* desc  = new char[strlen(description) + 100];
  char* desc1 = new char[strlen(description) + 100];
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (Int_t i = 0; i < fNPlanes; i++) {
    strcpy(desc, description);
    strcat(desc, " Plane ");
    strcat(desc, fPlaneNames[i]);

Sylvester Joosten's avatar
Sylvester Joosten committed
    THcDriftChamberPlane* newplane = new THcDriftChamberPlane(fPlaneNames[i], desc, i + 1, this);
    if (!newplane or newplane->IsZombie()) {
      _det_logger->error("{} Error creating Drift Chamber plane {}. Call expert.", Here(here),
                         name);
Sylvester Joosten's avatar
Sylvester Joosten committed
      delete[] desc;
      delete[] desc1;
      return;
    }
    fPlanes.push_back(newplane);
    newplane->SetDebug(fDebug);
Eric Pooser's avatar
Eric Pooser committed
    // cout << "Created Drift Chamber Plane " << fPlaneNames[i] << ", " << desc << endl;
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (UInt_t i = 0; i < fNChambers; i++) {
    sprintf(desc1, "Ch%d", i + 1);

    // Should construct a better chamber name
Sylvester Joosten's avatar
Sylvester Joosten committed
    THcDriftChamber* newchamber = new THcDriftChamber(desc1, desc, i + 1, this);
Sylvester Joosten's avatar
Sylvester Joosten committed
    // cout << "Created Drift Chamber " << i+1 << ", " << desc1 << endl;
    _det_logger->info("Created Drift Chamber {}, {}", i + 1, desc1);
    newchamber->SetHMSStyleFlag(fHMSStyleChambers); // Tell the chamber its style
Sylvester Joosten's avatar
Sylvester Joosten committed
  delete[] desc;
  delete[] desc1;
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
THcDC::THcDC() {
  // Constructor
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
THaAnalysisObject::EStatus THcDC::Init(const TDatime& date) {
  // Register the plane objects with the appropriate chambers.
  // Trigger ReadDatabase to load the remaining parameters
Sylvester Joosten's avatar
Sylvester Joosten committed
  Setup(GetName(), GetTitle()); // Create the subdetectors here
Sylvester Joosten's avatar
Sylvester Joosten committed
  EngineDID[0]     = toupper(GetApparatus()->GetName()[0]);
  if (gHcDetectorMap->FillMap(fDetMap, EngineDID) < 0) {
    static const char* const here = "Init()";
Sylvester Joosten's avatar
Sylvester Joosten committed
    _det_logger->error("{} Error filling detectormap for {}.", Here(here), EngineDID);
  // Should probably put this in ReadDatabase as we will know the
  // maximum number of hits after setting up the detector map
Whitney Armstrong's avatar
Whitney Armstrong committed
  _det_logger->info("DC tdc ref time cut = {} ", fTDC_RefTimeCut);
Sylvester Joosten's avatar
Sylvester Joosten committed
  // cout << " DC tdc ref time cut = " << fTDC_RefTimeCut  << endl;
  InitHitList(fDetMap, "THcRawDCHit", fDetMap->GetTotNumChan() + 1, fTDC_RefTimeCut, 0);
Sylvester Joosten's avatar
Sylvester Joosten committed
  CreateMissReportParms(Form("%sdc", fPrefix));
  EStatus status;
  // This triggers call of ReadDatabase and DefineVariables
Sylvester Joosten's avatar
Sylvester Joosten committed
  if ((status = THaTrackingDetector::Init(date)))
    return fStatus = status;

  // Initialize planes and add them to chambers
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (Int_t ip = 0; ip < fNPlanes; ip++) {
    if ((status = fPlanes[ip]->Init(date))) {
      return fStatus = status;
Sylvester Joosten's avatar
Sylvester Joosten committed
      Int_t chamber = fNChamber[ip];
      fChambers[chamber - 1]->AddPlane(fPlanes[ip]);
    }
  }
  // Initialize chambers
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (UInt_t ic = 0; ic < fNChambers; ic++) {
    if ((status = fChambers[ic]->Init(date))) {
      return fStatus = status;
  // Retrieve the fiting coefficients
Sylvester Joosten's avatar
Sylvester Joosten committed
  fPlaneCoeffs = new Double_t*[fNPlanes];
  for (Int_t ip = 0; ip < fNPlanes; ip++) {
    fPlaneCoeffs[ip] = fPlanes[ip]->GetPlaneCoef();
  }
Sylvester Joosten's avatar
Sylvester Joosten committed
  fResiduals          = new Double_t[fNPlanes];
  fResidualsExclPlane = new Double_t[fNPlanes];
  fWire_hit_did       = new Double_t[fNPlanes];
  fWire_hit_should    = new Double_t[fNPlanes];
  // Replace with what we need for Hall C
  //  const DataDest tmp[NDEST] = {
  //    { &fRTNhit, &fRANhit, fRT, fRT_c, fRA, fRA_p, fRA_c, fROff, fRPed, fRGain },
  //    { &fLTNhit, &fLANhit, fLT, fLT_c, fLA, fLA_p, fLA_c, fLOff, fLPed, fLGain }
  //  };
  //  memcpy( fDataDest, tmp, NDEST*sizeof(DataDest) );
Sylvester Joosten's avatar
Sylvester Joosten committed

  fPresentP        = 0;
  THaVar* vpresent = gHaVars->Find(Form("%s.present", GetApparatus()->GetName()));
  if (vpresent) {
    fPresentP = (Bool_t*)vpresent->GetValuePointer();
  return fStatus = kOK;
}
//_____________________________________________________________________________

Sylvester Joosten's avatar
Sylvester Joosten committed
Int_t THcDC::ManualInitTree(TTree* t) {
  std::string app_name    = GetApparatus()->GetName();
  std::string det_name    = GetName();
  std::string branch_name = (app_name + "_" + det_name + "_data");
  if (t) {
    _det_logger->info("THcDC::ManualInitTree : Adding branch, {}, to output tree", branch_name);
    t->Branch(branch_name.c_str(), &_basic_data, 32000, 99);
  }
  return 0;
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
Int_t THcDC::ReadDatabase(const TDatime& date) {
  /**
  Read this detector's parameters from the ThcParmList
  This function is called by THaDetectorBase::Init() once at the
  beginning of the analysis.
  //  static const char* const here = "ReadDatabase()";

Sylvester Joosten's avatar
Sylvester Joosten committed
  delete[] fXCenter;
  fXCenter = new Double_t[fNChambers];
  delete[] fYCenter;
  fYCenter = new Double_t[fNChambers];
  delete[] fMinHits;
  fMinHits = new Int_t[fNChambers];
  delete[] fMaxHits;
  fMaxHits = new Int_t[fNChambers];
  delete[] fMinCombos;
  fMinCombos = new Int_t[fNChambers];
  delete[] fSpace_Point_Criterion;
  fSpace_Point_Criterion = new Double_t[fNChambers];

  delete[] fTdcWinMin;
  fTdcWinMin = new Int_t[fNPlanes];
  delete[] fTdcWinMax;
  fTdcWinMax = new Int_t[fNPlanes];
  delete[] fCentralTime;
  fCentralTime = new Double_t[fNPlanes];
  delete[] fNWires;
  fNWires = new Int_t[fNPlanes];
  delete[] fNChamber;
  fNChamber = new Int_t[fNPlanes]; // Which chamber is this plane
  delete[] fWireOrder;
  fWireOrder = new Int_t[fNPlanes]; // Wire readout order
  delete[] fDriftTimeSign;
  fDriftTimeSign = new Int_t[fNPlanes];
  delete[] fReadoutLR;
  fReadoutLR = new Int_t[fNPlanes];
  delete[] fReadoutTB;
  fReadoutTB = new Int_t[fNPlanes];

  delete[] fXPos;
  fXPos = new Double_t[fNPlanes];
  delete[] fYPos;
  fYPos = new Double_t[fNPlanes];
  delete[] fZPos;
  fZPos = new Double_t[fNPlanes];
  delete[] fAlphaAngle;
  fAlphaAngle = new Double_t[fNPlanes];
  delete[] fBetaAngle;
  fBetaAngle = new Double_t[fNPlanes];
  delete[] fGammaAngle;
  fGammaAngle = new Double_t[fNPlanes];
  delete[] fPitch;
  fPitch = new Double_t[fNPlanes];
  delete[] fCentralWire;
  fCentralWire = new Double_t[fNPlanes];
  delete[] fPlaneTimeZero;
  fPlaneTimeZero = new Double_t[fNPlanes];
  delete[] fSigma;
  fSigma = new Double_t[fNPlanes];
Sylvester Joosten's avatar
Sylvester Joosten committed
  DBRequest list[] = {{"dc_tdc_time_per_channel", &fNSperChan, kDouble},
                      {"dc_wire_velocity", &fWireVelocity, kDouble},

                      {"dc_xcenter", fXCenter, kDouble, fNChambers},
                      {"dc_ycenter", fYCenter, kDouble, fNChambers},
                      {"min_hit", fMinHits, kInt, fNChambers},
                      {"max_pr_hits", fMaxHits, kInt, fNChambers},
                      {"min_combos", fMinCombos, kInt, fNChambers},
                      {"space_point_criterion", fSpace_Point_Criterion, kDouble, fNChambers},

                      {"dc_tdc_min_win", fTdcWinMin, kInt, (UInt_t)fNPlanes},
                      {"dc_tdc_max_win", fTdcWinMax, kInt, (UInt_t)fNPlanes},
                      {"dc_central_time", fCentralTime, kDouble, (UInt_t)fNPlanes},
                      {"dc_nrwire", fNWires, kInt, (UInt_t)fNPlanes},
                      {"dc_chamber_planes", fNChamber, kInt, (UInt_t)fNPlanes},
                      {"dc_wire_counting", fWireOrder, kInt, (UInt_t)fNPlanes},
                      {"dc_drifttime_sign", fDriftTimeSign, kInt, (UInt_t)fNPlanes},
                      {"dc_readoutLR", fReadoutLR, kInt, (UInt_t)fNPlanes, optional},
                      {"dc_readoutTB", fReadoutTB, kInt, (UInt_t)fNPlanes, optional},

                      {"dc_zpos", fZPos, kDouble, (UInt_t)fNPlanes},
                      {"dc_alpha_angle", fAlphaAngle, kDouble, (UInt_t)fNPlanes},
                      {"dc_beta_angle", fBetaAngle, kDouble, (UInt_t)fNPlanes},
                      {"dc_gamma_angle", fGammaAngle, kDouble, (UInt_t)fNPlanes},
                      {"dc_pitch", fPitch, kDouble, (UInt_t)fNPlanes},
                      {"dc_central_wire", fCentralWire, kDouble, (UInt_t)fNPlanes},
                      {"dc_plane_time_zero", fPlaneTimeZero, kDouble, (UInt_t)fNPlanes},
                      {"dc_sigma", fSigma, kDouble, (UInt_t)fNPlanes},
                      {"single_stub", &fSingleStub, kInt, 0, 1},
                      {"ntracks_max_fp", &fNTracksMaxFP, kInt},
                      {"xt_track_criterion", &fXtTrCriterion, kDouble},
                      {"yt_track_criterion", &fYtTrCriterion, kDouble},
                      {"xpt_track_criterion", &fXptTrCriterion, kDouble},
                      {"ypt_track_criterion", &fYptTrCriterion, kDouble},
                      {"dc_fix_lr", &fFixLR, kInt},
                      {"dc_fix_propcorr", &fFixPropagationCorrection, kInt},
                      {"debuglinkstubs", &fdebuglinkstubs, kInt},
                      {"debugprintrawdc", &fdebugprintrawdc, kInt},
                      {"debugprintdecodeddc", &fdebugprintdecodeddc, kInt},
                      {"debugflagpr", &fdebugflagpr, kInt},
                      {"debugflagstubs", &fdebugflagstubs, kInt},
                      {"debugtrackprint", &fdebugtrackprint, kInt},
                      {0}};
  fSingleStub      = 0;
  for (Int_t ip = 0; ip < fNPlanes; ip++) {
    fReadoutLR[ip] = 0.0;
    fReadoutTB[ip] = 0.0;
Sylvester Joosten's avatar
Sylvester Joosten committed
  }
Sylvester Joosten's avatar
Sylvester Joosten committed
  gHcParms->LoadParmValues((DBRequest*)&list, fPrefix);
Sylvester Joosten's avatar
Sylvester Joosten committed
  // Set the default plane x,y positions to those of the chamber
  for (Int_t ip = 0; ip < fNPlanes; ip++) {
    fXPos[ip] = fXCenter[GetNChamber(ip + 1) - 1];
    fYPos[ip] = fYCenter[GetNChamber(ip + 1) - 1];
  }
Sylvester Joosten's avatar
Sylvester Joosten committed
  // Load the x,y positions of the planes if they exist (overwrites defaults)
  DBRequest listOpt[] = {{"dc_xpos", fXPos, kDouble, (UInt_t)fNPlanes, optional},
                         {"dc_ypos", fYPos, kDouble, (UInt_t)fNPlanes, optional},
                         {0}};
  gHcParms->LoadParmValues((DBRequest*)&listOpt, fPrefix);
  if (fNTracksMaxFP <= 0)
    fNTracksMaxFP = 10;
  // if(fNTracksMaxFP > HNRACKS_MAX) fNTracksMaxFP = NHTRACKS_MAX;
Sylvester Joosten's avatar
Sylvester Joosten committed

  std::string plane_counts_string;
Sylvester Joosten's avatar
Sylvester Joosten committed
  // cout << "Plane counts:";
  for (Int_t i = 0; i < fNPlanes; i++) {
    // cout << " " << fNWires[i];
    plane_counts_string += std::string(" ");
    plane_counts_string += std::to_string(fNWires[i]);
Sylvester Joosten's avatar
Sylvester Joosten committed
  // cout << endl;
Whitney Armstrong's avatar
Whitney Armstrong committed
  _det_logger->info("Plane counts: {}", plane_counts_string);

  fIsInit = true;

  return kOK;
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
Int_t THcDC::DefineVariables(EMode mode) {
  /**
    Initialize global variables for histograms and Root tree
  */
Sylvester Joosten's avatar
Sylvester Joosten committed
  if (mode == kDefine && fIsSetup)
    return kOK;
  fIsSetup = (mode == kDefine);

  // Register variables in global list

  RVarDef vars[] = {
Sylvester Joosten's avatar
Sylvester Joosten committed
      {"stubtest", "stub test", "fStubTest"},
      {"nhit", "Number of DC hits", "fNhits"},
      {"tnhit", "Number of good DC hits", "fNthits"},
      {"trawhit", "Number of true raw DC hits", "fN_True_RawHits"},
      {"ntrack", "Number of Tracks", "fNDCTracks"},
      {"nsp", "Number of Space Points", "fNSp"},
      {"track_nsp", "Number of spacepoints in track", "fDCTracks.THcDCTrack.GetNSpacePoints()"},
      {"x", "X at focal plane", "fDCTracks.THcDCTrack.GetX()"},
      {"y", "Y at focal plane", "fDCTracks.THcDCTrack.GetY()"},
      {"xp", "XP at focal plane", "fDCTracks.THcDCTrack.GetXP()"},
      {"yp", "YP at focal plane", "fDCTracks.THcDCTrack.GetYP()"},
      {"x_fp", "X at focal plane (golden track)", "fX_fp_best"},
      {"y_fp", "Y at focal plane( golden track)", "fY_fp_best"},
      {"xp_fp", "XP at focal plane (golden track)", "fXp_fp_best"},
      {"yp_fp", "YP at focal plane(golden track) ", "fYp_fp_best"},
      {"chisq", "chisq/dof (golden track) ", "fChisq_best"},
      {"sp1_id", " (golden track) ", "fSp1_ID_best"},
      {"sp2_id", " (golden track) ", "fSp2_ID_best"},
      {"InsideDipoleExit", " ", "fInSideDipoleExit_best"},
      {"gtrack_nsp", " Number of space points in golden track ", "fNsp_best"},
      {"residual", "Residuals", "fResiduals"},
      {"residualExclPlane", "Residuals", "fResidualsExclPlane"},
      {"wireHitDid", "Wire did have  matched track hit", "fWire_hit_did"},
      {"wireHitShould", "Wire should have matched track hit", "fWire_hit_should"},
      {0}};
  return DefineVarsFromList(vars, mode);
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
THcDC::~THcDC() {
  // Destructor. Remove variables from global list.

Sylvester Joosten's avatar
Sylvester Joosten committed
  if (fIsSetup)
    RemoveVariables();
Sylvester Joosten's avatar
Sylvester Joosten committed
  if (fIsInit)
    DeleteArrays();
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (vector<THcDriftChamberPlane*>::iterator ip = fPlanes.begin(); ip != fPlanes.end(); ++ip)
    delete *ip;
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (vector<THcDriftChamber*>::iterator ip = fChambers.begin(); ip != fChambers.end(); ++ip)
    delete *ip;
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
void THcDC::DeleteArrays() {
  // Delete member arrays. Used by destructor.

Sylvester Joosten's avatar
Sylvester Joosten committed
  delete[] fXCenter;
  fXCenter = NULL;
  delete[] fYCenter;
  fYCenter = NULL;
  delete[] fMinHits;
  fMinHits = NULL;
  delete[] fMaxHits;
  fMaxHits = NULL;
  delete[] fMinCombos;
  fMinCombos = NULL;
  delete[] fSpace_Point_Criterion;
  fSpace_Point_Criterion = NULL;

  delete[] fTdcWinMin;
  fTdcWinMin = NULL;
  delete[] fTdcWinMax;
  fTdcWinMax = NULL;
  delete[] fCentralTime;
  fCentralTime = NULL;
  delete[] fNWires;
  fNWires = NULL;
  delete[] fNChamber;
  fNChamber = NULL;
  delete[] fWireOrder;
  fWireOrder = NULL;
  delete[] fDriftTimeSign;
  fDriftTimeSign = NULL;
  delete[] fReadoutLR;
  fReadoutLR = NULL;
  delete[] fReadoutTB;
  fReadoutTB = NULL;

  delete[] fXPos;
  fXPos = NULL;
  delete[] fYPos;
  fYPos = NULL;
  delete[] fZPos;
  fZPos = NULL;
  delete[] fAlphaAngle;
  fAlphaAngle = NULL;
  delete[] fBetaAngle;
  fBetaAngle = NULL;
  delete[] fGammaAngle;
  fGammaAngle = NULL;
  delete[] fPitch;
  fPitch = NULL;
  delete[] fCentralWire;
  fCentralWire = NULL;
  delete[] fPlaneTimeZero;
  fPlaneTimeZero = NULL;
  delete[] fSigma;
  fSigma = NULL;
  // Efficiency arrays
Sylvester Joosten's avatar
Sylvester Joosten committed
  delete[] fNChamHits;
  fNChamHits = NULL;
  delete[] fPlaneEvents;
  fPlaneEvents = NULL;

  for (Int_t i = 0; i < fNPlanes; ++i)
    delete[] fPlaneNames[i];
  delete[] fPlaneNames;

  delete[] fPlaneCoeffs;
  fPlaneCoeffs = 0;
  delete[] fResiduals;
  fResiduals = 0;
  delete[] fResidualsExclPlane;
  fResidualsExclPlane = 0;
  delete[] fWire_hit_did;
  fWire_hit_did = 0;
  delete[] fWire_hit_should;
  fWire_hit_should = 0;
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
inline void THcDC::ClearEvent() {
  // Reset per-event data.
Sylvester Joosten's avatar
Sylvester Joosten committed
  fStubTest              = 0;
  fNhits                 = 0;
  fNthits                = 0;
  fN_True_RawHits        = 0;
  fX_fp_best             = -10000.;
  fY_fp_best             = -10000.;
  fXp_fp_best            = -10000.;
  fYp_fp_best            = -10000.;
  fChisq_best            = kBig;
  fNsp_best              = 0;
  fInSideDipoleExit_best = kTRUE;
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (UInt_t i = 0; i < fNChambers; i++) {
    fChambers[i]->Clear();
  }

Sylvester Joosten's avatar
Sylvester Joosten committed
  for (Int_t i = 0; i < fNPlanes; i++) {
    fResiduals[i]          = 1000.0;
    fResidualsExclPlane[i] = 1000.0;
Sylvester Joosten's avatar
Sylvester Joosten committed
    fWire_hit_did[i]       = 1000.0;
    fWire_hit_should[i]    = 1000.0;
  //  fTrackProj->Clear();
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
Int_t THcDC::Decode(const THaEvData& evdata) {
  /**
    Decode event into hit list.
    Pass hit list to the planes.
    Load hits from planes into chamber objects
  */
  ClearEvent();
  Int_t num_event = evdata.GetEvNum();
Sylvester Joosten's avatar
Sylvester Joosten committed
  if (fdebugprintrawdc || fdebugprintdecodeddc || fdebuglinkstubs || fdebugtrackprint)
    cout << " event num = " << num_event << endl;
  // Get the Hall C style hitlist (fRawHitList) for this event
Sylvester Joosten's avatar
Sylvester Joosten committed
  Bool_t present = kTRUE; // Suppress reference time warnings
  if (fPresentP) {        // if this spectrometer not part of trigger
    present = *fPresentP;
  }
  fNhits = DecodeToHitList(evdata, !present);
Sylvester Joosten's avatar
Sylvester Joosten committed
  if (!gHaCuts->Result("Pedestal_event")) {
    // Let each plane get its hits
    Int_t nexthit = 0;
Sylvester Joosten's avatar
Sylvester Joosten committed
    for (Int_t ip = 0; ip < fNPlanes; ip++) {
      nexthit = fPlanes[ip]->ProcessHits(fRawHitList, nexthit);
      fN_True_RawHits += fPlanes[ip]->GetNRawhits();
    }
    // fRawHitList is TClones array of THcRawDCHit objects
Sylvester Joosten's avatar
Sylvester Joosten committed
    Int_t counter = 0;
    if (fdebugprintrawdc) {
Sylvester Joosten's avatar
Sylvester Joosten committed
      cout << " RAW_TOT_HITS = " << fNRawHits << endl;
      cout << " Hit #  "
           << "Plane  "
           << " Wire "
           << " Raw TDC " << endl;
      for (UInt_t ihit = 0; ihit < fNRawHits; ihit++) {
        THcRawDCHit* hit = (THcRawDCHit*)fRawHitList->At(ihit);
        for (UInt_t imhit = 0; imhit < hit->GetRawTdcHit().GetNHits(); imhit++) {
          counter++;
          cout << counter << "      " << hit->fPlane << "     " << hit->fCounter << "     "
               << hit->GetRawTdcHit().GetTimeRaw(imhit) << endl;
        }
Sylvester Joosten's avatar
Sylvester Joosten committed
    Eff(); // Accumlate statistics
  return fNhits;
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
Int_t THcDC::ApplyCorrections(void) { return (0); }

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
Int_t THcDC::CoarseTrack(TClonesArray& tracks) {
  /**
     Find a set of tracks through the drift chambers and put them
     into the tracks TClonesArray.
     Tracks are in the detector coordinate system.
  */

  // Subtract starttimes from each plane hit
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (Int_t ip = 0; ip < fNPlanes; ip++) {
    fPlanes[ip]->SubtractStartTime();
  }
Sylvester Joosten's avatar
Sylvester Joosten committed
  // Let each chamber get its hits
  for (UInt_t ic = 0; ic < fNChambers; ic++) {
    fChambers[ic]->ProcessHits();
    fNthits += fChambers[ic]->GetNHits();
    if (fdebugprintdecodeddc)
      fChambers[ic]->PrintDecode();
  }
  //
  for (UInt_t i = 0; i < fNChambers; i++) {
    fChambers[i]->FindSpacePoints();
Sylvester Joosten's avatar
Sylvester Joosten committed
  if (fdebugflagstubs)
    PrintSpacePoints();
  if (fdebugflagstubs)
    PrintStubs();
  // Now link the stubs between chambers
  LinkStubs();
Sylvester Joosten's avatar
Sylvester Joosten committed
  if (fNDCTracks > 0) {
    TrackFit();
    // Copy tracks into podd tracks list
Sylvester Joosten's avatar
Sylvester Joosten committed
    for (UInt_t itrack = 0; itrack < fNDCTracks; itrack++) {
Sylvester Joosten's avatar
Sylvester Joosten committed
      theTrack           = AddTrack(tracks, 0.0, 0.0, 0.0, 0.0); // Leaving off trackID
      // Should we add stubs with AddCluster?  Could we do this
      // by having stubs inherit from cluster
Sylvester Joosten's avatar
Sylvester Joosten committed
      THcDCTrack* tr = static_cast<THcDCTrack*>(fDCTracks->At(itrack));
      theTrack->Set(tr->GetX(), tr->GetY(), tr->GetXP(), tr->GetYP());
Sylvester Joosten's avatar
Sylvester Joosten committed
      theTrack->SetFlag((UInt_t)0);
      // Need to look at how engine does chi2 and track selection.  Reduced?
Sylvester Joosten's avatar
Sylvester Joosten committed
      theTrack->SetChi2(tr->GetChisq(), tr->GetNFree());
      // CalcFocalPlaneCoords.  Aren't our tracks already in focal plane coords
      // We should have some kind of track ID so that the THaTrack can be
      // associate back with the DC track
      // Assign the track number
Sylvester Joosten's avatar
Sylvester Joosten committed
      theTrack->SetTrkNum(itrack + 1);
Sylvester Joosten's avatar
Sylvester Joosten committed
  }

  ApplyCorrections();

  return 0;
}

//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
Int_t THcDC::FineTrack(TClonesArray& tracks) { return 0; }
Sylvester Joosten's avatar
Sylvester Joosten committed
void THcDC::SetFocalPlaneBestTrack(Int_t golden_track_index) {
  THcDCTrack* tr1           = static_cast<THcDCTrack*>(fDCTracks->At(golden_track_index));
  fX_fp_best                = tr1->GetX();
  fY_fp_best                = tr1->GetY();
  fXp_fp_best               = tr1->GetXP();
  fYp_fp_best               = tr1->GetYP();
  THcHallCSpectrometer* app = dynamic_cast<THcHallCSpectrometer*>(GetApparatus());
  fInSideDipoleExit_best =
      app->InsideDipoleExitWindow(fX_fp_best, fXp_fp_best, fY_fp_best, fYp_fp_best);
  fSp1_ID_best = tr1->GetSp1_ID();
  fSp2_ID_best = tr1->GetSp2_ID();
  fChisq_best  = tr1->GetChisq();
  fNsp_best    = tr1->GetNSpacePoints();
  for (UInt_t ihit = 0; ihit < UInt_t(tr1->GetNHits()); ihit++) {
    THcDCHit* hit                          = tr1->GetHit(ihit);
    Int_t     plane                        = hit->GetPlaneNum() - 1;
    _basic_data._Residuals[plane]          = tr1->GetResidual(plane);
    fResiduals[plane]                      = tr1->GetResidual(plane);
    _basic_data._ResidualsExclPlane[plane] = tr1->GetResidualExclPlane(plane);
    fResidualsExclPlane[plane]             = tr1->GetResidualExclPlane(plane);
  }
  EfficiencyPerWire(golden_track_index);
Sylvester Joosten's avatar
Sylvester Joosten committed
void THcDC::EfficiencyPerWire(Int_t golden_track_index) {
  THcDCTrack* tr1 = static_cast<THcDCTrack*>(fDCTracks->At(golden_track_index));
  Double_t    track_pos;
  for (UInt_t ihit = 0; ihit < UInt_t(tr1->GetNHits()); ihit++) {
    THcDCHit* hit        = tr1->GetHit(ihit);
    Int_t     plane      = hit->GetPlaneNum() - 1;
    track_pos            = tr1->GetCoord(plane);
    Int_t wire_num       = hit->GetWireNum();
    Int_t wire_track_num = round(fPlanes[plane]->CalcWireFromPos(track_pos));
    if ((wire_num - wire_track_num) == 0)
      fWire_hit_did[plane] = wire_num;
  }
  for (Int_t ip = 0; ip < fNPlanes; ip++) {
    track_pos            = tr1->GetCoord(ip);
    Int_t wire_should    = round(fPlanes[ip]->CalcWireFromPos(track_pos));
    fWire_hit_should[ip] = wire_should;
Sylvester Joosten's avatar
Sylvester Joosten committed
void THcDC::PrintSpacePoints() {
  for (UInt_t ich = 0; ich < fNChambers; ich++) {
    printf("%s %2d %s %3d %s %3d \n", " chamber = ", fChambers[ich]->GetChamberNum(),
           " number of hits = ", fChambers[ich]->GetNHits(),
           " number of spacepoints = ", fChambers[ich]->GetNSpacePoints());
    printf("%6s %-8s %-8s %6s %6s %10s \n", "     ", " ", " ", "Number", "Number", "Plane Wire");
    printf("%6s %-8s %-8s %6s %6s %10s \n", "Point", "x", "y", " hits ", "combos", " for each hit");
    TClonesArray* spacepointarray = fChambers[ich]->GetSpacePointsP();
Sylvester Joosten's avatar
Sylvester Joosten committed
    for (Int_t isp = 0; isp < fChambers[ich]->GetNSpacePoints(); isp++) {
      THcSpacePoint* sp = (THcSpacePoint*)(spacepointarray->At(isp));
Sylvester Joosten's avatar
Sylvester Joosten committed
      printf("%5d %8.5f %8.5f %5d  %5d ", isp + 1, sp->GetX(), sp->GetY(), sp->GetNHits(),
             sp->GetCombos());
      for (Int_t ii = 0; ii < sp->GetNHits(); ii++) {
        THcDCHit* hittemp = (THcDCHit*)(sp->GetHit(ii));
        printf("%3d %3d %3d", hittemp->GetPlaneNum(), hittemp->GetWireNum(), hittemp->GetLR());
Sylvester Joosten's avatar
Sylvester Joosten committed
void THcDC::PrintStubs() {
  for (UInt_t ich = 0; ich < fNChambers; ich++) {
    printf("%s %3d \n", " Stub fit results Chamber = ", ich + 1);
    printf("%-5s %-18s %-18s %-18s %-18s\n", "point", "x_t", "y_t", "xp_t", "yp_t");
    printf("%-5s %-18s %-18s %-18s %-18s\n", "     ", "[cm]", "[cm]", "[cm]", "[cm]");
    TClonesArray* spacepointarray = fChambers[ich]->GetSpacePointsP();
Sylvester Joosten's avatar
Sylvester Joosten committed
    for (Int_t isp = 0; isp < fChambers[ich]->GetNSpacePoints(); isp++) {
      THcSpacePoint* sp      = (THcSpacePoint*)(spacepointarray->At(isp));
      Double_t*      spstubt = sp->GetStubP();
      printf("%-5d % 15.10e % 15.10e % 15.10e % 15.10e \n", isp + 1, spstubt[0], spstubt[1],
             spstubt[2], spstubt[3]);
//_____________________________________________________________________________
Sylvester Joosten's avatar
Sylvester Joosten committed
void THcDC::LinkStubs() {
  /**
       The logic is
                    0) Put all space points in a single list
                    1) loop over all space points as seeds  isp1
                    2) Check if this space point is all ready in a track
                    3) loop over all succeeding space pointss   isp2
                    4)  check if there is a track-criterion match
                         either add to existing track
                         or if there is another point in same chamber
                            make a copy containing isp2 rather than
                              other point in same chamber
                    5) If hsingle_stub is set, make a track of all single
                       stubs.
  */

  std::vector<THcSpacePoint*> fSp;
Sylvester Joosten's avatar
Sylvester Joosten committed
  fNSp = 0;
Mark Jones's avatar
Mark Jones committed
  fSp.reserve(100);
Sylvester Joosten's avatar
Sylvester Joosten committed
  fNDCTracks = 0; // Number of Focal Plane tracks found
  fDCTracks->Delete();
  // Make a vector of pointers to the SpacePoints
Sylvester Joosten's avatar
Sylvester Joosten committed
  // if (fChambers[0]->GetNSpacePoints()+fChambers[1]->GetNSpacePoints()>10) return;
Sylvester Joosten's avatar
Sylvester Joosten committed
  for (UInt_t ich = 0; ich < fNChambers; ich++) {
    Int_t         nchamber        = fChambers[ich]->GetChamberNum();
    TClonesArray* spacepointarray = fChambers[ich]->GetSpacePointsP();
Sylvester Joosten's avatar
Sylvester Joosten committed
    for (Int_t isp = 0; isp < fChambers[ich]->GetNSpacePoints(); isp++) {
      fSp.push_back(static_cast<THcSpacePoint*>(spacepointarray->At(isp)));
Sylvester Joosten's avatar
Sylvester Joosten committed
      fSp[fNSp]->fNChamber       = nchamber;
      fSp[fNSp]->fNChamber_spnum = isp;
Sylvester Joosten's avatar
Sylvester Joosten committed
      if (ich == 0 && fNSp > 50)
        break;
      if (fNSp > 100)
        break;
Sylvester Joosten's avatar
Sylvester Joosten committed
  Double_t stubminx  = 999999;
  Double_t stubminy  = 999999;
  Double_t stubminxp = 999999;
  Double_t stubminyp = 999999;
Sylvester Joosten's avatar
Sylvester Joosten committed
  Int_t    stub_tracks[MAXTRACKS];
  if (fSingleStub == 0) {
    for (Int_t isp1 = 0; isp1 < fNSp - 1;
         isp1++) { // isp1 is index/id in total list of space points
      THcSpacePoint* sp1      = fSp[isp1];
      Int_t          sptracks = 0;
      // Now make sure this sp is not already used in a sp.
      // Could this be done by having a sp point to the track it is in?
Sylvester Joosten's avatar
Sylvester Joosten committed
      Int_t tryflag = 1;
      for (UInt_t itrack = 0; itrack < fNDCTracks; itrack++) {
        THcDCTrack* theDCTrack = static_cast<THcDCTrack*>(fDCTracks->At(itrack));
        for (Int_t isp = 0; isp < theDCTrack->GetNSpacePoints(); isp++) {
          // isp is index into list of space points attached to theDCTrack
          if (theDCTrack->GetSpacePoint(isp) == sp1) {
            tryflag = 0;
          }
        }
Sylvester Joosten's avatar
Sylvester Joosten committed
      if (tryflag) { // SP not already part of a track
        Int_t newtrack = 1;
        for (Int_t isp2 = isp1 + 1; isp2 < fNSp; isp2++) {
          THcSpacePoint* sp2 = fSp[isp2];
          if (sp1->fNChamber != sp2->fNChamber && sp1->GetSetStubFlag() && sp2->GetSetStubFlag()) {
            Double_t* spstub1 = sp1->GetStubP();
            Double_t* spstub2 = sp2->GetStubP();
            Double_t  dposx   = spstub1[0] - spstub2[0];
            Double_t  dposy;
            if (fProjectToChamber) { // From SOS s_link_stubs
              // Since single chamber resolution is ~50mr, and the maximum y`
              // angle is about 30mr, use differenece between y AT CHAMBERS, rather
              // than at focal plane.  (Project back to chamber, to take out y' uncertainty)
              // (Should this be done for SHMS and HMS too?)
              Double_t y1 = spstub1[1] + fChambers[sp1->fNChamber]->GetZPos() * spstub1[3];
              Double_t y2 = spstub2[1] + fChambers[sp2->fNChamber]->GetZPos() * spstub2[3];
              dposy       = y1 - y2;
            } else {
              dposy = spstub1[1] - spstub2[1];
            }
            Double_t dposxp = spstub1[2] - spstub2[2];
            Double_t dposyp = spstub1[3] - spstub2[3];

            // What is the point of saving these stubmin values.  They
            // Don't seem to be used anywhere except that they can be
            // printed out if hbypass_track_eff_files is zero.
            if (TMath::Abs(dposx) < TMath::Abs(stubminx))
              stubminx = dposx;
            if (TMath::Abs(dposy) < TMath::Abs(stubminy))
              stubminy = dposy;
            if (TMath::Abs(dposxp) < TMath::Abs(stubminxp))
              stubminxp = dposxp;
            if (TMath::Abs(dposyp) < TMath::Abs(stubminyp))
              stubminyp = dposyp;

            // if hbypass_track_eff_files == 0 then
            // Print out each stubminX that is less that its criterion

            if ((TMath::Abs(dposx) < fXtTrCriterion) && (TMath::Abs(dposy) < fYtTrCriterion) &&
                (TMath::Abs(dposxp) < fXptTrCriterion) && (TMath::Abs(dposyp) < fYptTrCriterion)) {
              if (newtrack) {
                assert(sptracks == 0);
                fStubTest = 1;
                // stubtest=1;  Used in h_track_tests.f
                // Make a new track if there are not to many
                if (fNDCTracks < fNTracksMaxFP) {
                  sptracks                = 0; // Number of tracks with this seed
                  stub_tracks[sptracks++] = fNDCTracks;
                  THcDCTrack* theDCTrack  = new ((*fDCTracks)[fNDCTracks++]) THcDCTrack(fNPlanes);
                  theDCTrack->AddSpacePoint(sp1);
                  theDCTrack->AddSpacePoint(sp2);
                  if (sp1->fNChamber == 1)
                    theDCTrack->SetSp1_ID(sp1->fNChamber_spnum);
                  if (sp1->fNChamber == 2)
                    theDCTrack->SetSp2_ID(sp1->fNChamber_spnum);
                  if (sp2->fNChamber == 1)
                    theDCTrack->SetSp1_ID(sp2->fNChamber_spnum);
                  if (sp2->fNChamber == 2)
                    theDCTrack->SetSp2_ID(sp2->fNChamber_spnum);
                  newtrack = 0; // Make no more tracks in this loop
                  // (But could replace a SP?)
                } else {
                  if (fHMSStyleChambers) {
Sylvester Joosten's avatar
Sylvester Joosten committed
                    fNDCTracks = 0;
                    return;
Sylvester Joosten's avatar
Sylvester Joosten committed
                }
              } else {
                // Check if there is another space point in the same chamber
                for (Int_t itrack = 0; itrack < sptracks; itrack++) {
                  Int_t       track      = stub_tracks[itrack];
                  THcDCTrack* theDCTrack = static_cast<THcDCTrack*>(fDCTracks->At(track));
                  Int_t       spoint     = -1;
                  Int_t       duppoint   = 0;
                  for (Int_t isp = 0; isp < theDCTrack->GetNSpacePoints(); isp++) {
                    // isp is index of space points in theDCTrack
                    if (sp2->fNChamber == theDCTrack->GetSpacePoint(isp)->fNChamber) {
                      spoint = isp;
                    }
                    if (sp2 == theDCTrack->GetSpacePoint(isp)) {
                      duppoint = 1;
                    }
                  } // End loop over sp in tracks with isp1
                    // If there is no other space point in this chamber
                    // add this space point to current track(2)
                  if (!duppoint) {
                    if (spoint < 0) {
                      theDCTrack->AddSpacePoint(sp2);
                      if (sp2->fNChamber == 1)
                        theDCTrack->SetSp1_ID(sp2->fNChamber_spnum);
                      if (sp2->fNChamber == 2)
                        theDCTrack->SetSp2_ID(sp2->fNChamber_spnum);
                    } else {
                      // If there is another point in the same chamber
                      // in this track create a new track with all the
                      // same space points except spoint
                      if (fNDCTracks < MAXTRACKS) {
                        stub_tracks[sptracks++] = fNDCTracks;
                        THcDCTrack* newDCTrack =
                            new ((*fDCTracks)[fNDCTracks++]) THcDCTrack(fNPlanes);
                        for (Int_t isp = 0; isp < theDCTrack->GetNSpacePoints(); isp++) {
                          if (isp != spoint) {
                            newDCTrack->AddSpacePoint(theDCTrack->GetSpacePoint(isp));
                            if (theDCTrack->GetSpacePoint(isp)->fNChamber == 1)
                              newDCTrack->SetSp1_ID(
                                  theDCTrack->GetSpacePoint(isp)->fNChamber_spnum);
                            if (theDCTrack->GetSpacePoint(isp)->fNChamber == 2)
                              newDCTrack->SetSp2_ID(
                                  theDCTrack->GetSpacePoint(isp)->fNChamber_spnum);
                          } else {
                            newDCTrack->AddSpacePoint(sp2);
                            if (sp2->fNChamber == 1)
                              newDCTrack->SetSp1_ID(sp2->fNChamber_spnum);
                            if (sp2->fNChamber == 2)
                              newDCTrack->SetSp2_ID(sp2->fNChamber_spnum);
                          } // End check for dup on copy
                        }   // End copy of track
                      } else {
                        if (fHMSStyleChambers) {
                          if (fdebuglinkstubs)
                            cout << "EPIC FAIL 2:  Too many tracks found in THcDC::LinkStubs "
                                    "maxtracks = "
                                 << MAXTRACKS << endl;
                          fNDCTracks = 0;
                          return; // Max # of allowed tracks
Sylvester Joosten's avatar
Sylvester Joosten committed
                      }
                    } // end if on same chamber
                  }   // end if on duplicate point
                }     // end for over tracks with isp1
              }       // else newtrack
            }         // criterion
          }           // end test on same chamber
        }             // end isp2 loop over new space points
      }               // end test on tryflag
    }                 // end isp1 outer loop over space points
    //
  } else { // Make track out of each single space point
Sylvester Joosten's avatar
Sylvester Joosten committed
    for (Int_t isp = 0; isp < fNSp; isp++) {
      if (fNDCTracks < MAXTRACKS) {
        // Need some constructed t thingy
        if (fSp[isp]->GetSetStubFlag()) {
Sylvester Joosten's avatar
Sylvester Joosten committed
          THcDCTrack* newDCTrack = new ((*fDCTracks)[fNDCTracks++]) THcDCTrack(fNPlanes);
          newDCTrack->AddSpacePoint(fSp[isp]);
        }