/* * hitCDC - registers hits for Central Drift Chamber * * This is a part of the hits package for the * HDGeant simulation program for Hall D. * * version 1.0 -Richard Jones July 16, 2001 * * changes: Wed Jun 20 13:19:56 EDT 2007 B. Zihlmann * add ipart to the function call hitCentralDC */ #include #include #include #include #include #include #include #include "calibDB.h" extern s_HDDM_t* thisInputEvent; typedef struct { int writeenohits; int showersincol; int driftclusters; } controlparams_t; extern controlparams_t controlparams_; void gpoiss_(float*,int*,const int*); // avoid solaris compiler warnings // Drift speed 2.2cm/us is appropriate for a 90/10 Argon/Methane mixture static float DRIFT_SPEED = 0.0055; static float TWO_HIT_RESOL = 25.; static int MAX_HITS = 1000; static float THRESH_KEV = 1.; static float THRESH_MV = 1.; static float STRAW_RADIUS = 0.776; static float CDC_TIME_WINDOW = 1000.0; //time window for accepting CDC hits, ns static float ELECTRON_CHARGE =1.6022e-4; /* fC */ static float GAS_GAIN = 1e5; binTree_t* centralDCTree = 0; static int strawCount = 0; static int pointCount = 0; static int stripCount = 0; static int initialized = 0; static float cdc_drift_time[78]; static float cdc_drift_distance[78]; static float BSCALE_PAR1=0.; static float BSCALE_PAR2=0.; /* void GetDOCA(int ipart, float x[3], float p[5], float doca[3]); disabled 6/24/2009 */ typedef int (*compfn)(const void*, const void*); extern void polint(float *xa, float *ya,int n,float x, float *y,float *dy); // Sort function for sorting clusters int cdc_cluster_sort(const void *a,const void *b) { const s_CdcStrawTruthHit_t *ca=a; const s_CdcStrawTruthHit_t *cb=b; if (ca->t < cb->t) return -1; else if (ca->t > cb->t) return 1; else return 0; } // Simulation of the ASIC response to a pulse due to a cluster double asic_response(double t) { double func=0; double par[11]={-0.01986,0.01802,-0.001097,10.3,11.72,-0.03701,35.84, 15.93,0.006141,80.95,24.77}; if (t < par[3]) { func=par[0]*t+par[1]*t*t+par[2]*t*t*t; } else { func+=(par[0]*par[3]+par[1]*par[3]*par[3]+par[2]*par[3]*par[3]*par[3]) *exp(-(t-par[3])*(t-par[3])/(par[4]*par[4])); func+=par[5]*exp(-(t-par[6])*(t-par[6])/(par[7]*par[7])); func+=par[8]*exp(-(t-par[9])*(t-par[9])/(par[10]*par[10])); } return func; } // Simulation of signal on a wire double cdc_wire_signal(double t,s_CdcStrawTruthHits_t* chits) { int m; double asic_gain=0.5; // mV/fC double func=0; for (m=0; m < chits->mult; m++) { if (t > chits->in[m].t) { double my_time=t-chits->in[m].t; func+=asic_gain*chits->in[m].q*asic_response(my_time); } } return func; } void AddCDCCluster(s_CdcStrawTruthHits_t* hits, int ipart, int track, int n_p, float t, float xyzcluster[3]) { // measured charge float q=0.; // drift radius float dradius=sqrt(xyzcluster[0]*xyzcluster[0]+xyzcluster[1]*xyzcluster[1]); // Find the drift time for this cluster. Drift time depends on B: // (dependence derived from Garfield calculations) float B[3],Bmag,x[3]; transformCoord(xyzcluster,"local",x,"global"); gufld_db_(x,B); Bmag=sqrt(B[0]*B[0]+B[1]*B[1]+B[2]*B[2]); float d2=dradius*dradius; float d3=dradius*d2; int i=(int)(dradius/0.01); float my_t,my_t_err; // Check for closeness to boundaries of the drift table if (i>=75){ // Do a crude linear extrapolation my_t=cdc_drift_time[75]+((cdc_drift_time[77]-cdc_drift_time[75])/0.02) *(dradius-cdc_drift_distance[75]); } else{ int index; if (i<1) index=0; else index=i-1; // Interpolate over the drift table to find an approximation for the drift // time polint(&cdc_drift_distance[index],&cdc_drift_time[index],4,dradius,&my_t, &my_t_err); } float tdrift=my_t/(1.-BSCALE_PAR1-BSCALE_PAR2*Bmag); //Longitudinal diffusion int two=2; float rndno[2]; grndm_(rndno,&two); float rho = sqrt(-2*log(rndno[0])); float phi = rndno[1]*2*M_PI; float dt=(7.515*dradius-2.139*d2+12.63*d3)*rho*cos(phi); tdrift+=dt; // Prevent unphysical times (drift electrons arriving at wire before particle // passes the doca to the wire) double v_max=0.08; // guess for now based on Garfield, near wire double tmin=dradius/v_max; if (tdrift < tmin) { tdrift=tmin; } float total_time=t+tdrift; // Skip cluster if the time would go beyond readout window if (total_time > CDC_TIME_WINDOW) return; // Average number of secondary ion pairs for 50/50 Ar/CO2 mixture float n_s_per_p=1.94; if (controlparams_.driftclusters == 0) { /* Total number of ion pairs. On average for each primary ion pair produced there are n_s secondary ion pairs produced. The probability distribution is a compound poisson distribution that requires generating two Poisson variables. */ int n_s,one=1; float n_s_mean = ((float)n_p)*n_s_per_p; gpoiss_(&n_s_mean,&n_s,&one); int n_t = n_s+n_p; q = ((float)n_t)*GAS_GAIN*ELECTRON_CHARGE; } else { // Distribute the number of secondary ionizations for this primary // ionization according to a Poisson distribution with mean n_s_over_p. // For simplicity we assume these secondary electrons and the primary // electron stay together as a cluster. int n_s, one=1; gpoiss_(&n_s_per_p,&n_s,&one); // Energy deposition, equivalent to anode charge, in units of fC q = GAS_GAIN*ELECTRON_CHARGE*(float)(1+n_s); } // Add the hit info int nhit; for (nhit = 0; nhit < hits->mult; nhit++) { if (fabs(hits->in[nhit].t - total_time) < TWO_HIT_RESOL) { break; } } if (nhit < hits->mult) { /* merge with former hit */ /* Use the time from the earlier hit but add the charge*/ hits->in[nhit].q += q; if (hits->in[nhit].t > total_time) { hits->in[nhit].t = total_time; hits->in[nhit].d = dradius; hits->in[nhit].itrack = gidGetId(track); hits->in[nhit].ptype = ipart; } /* hits->in[nhit].t = (hits->in[nhit].t * hits->in[nhit].q + tdrift * dEsum) / (hits->in[nhit].q += dEsum); */ } else if (nhit < MAX_HITS) { /* create new hit */ hits->in[nhit].t = total_time; hits->in[nhit].q = q; hits->in[nhit].d = dradius; hits->in[nhit].itrack = gidGetId(track); hits->in[nhit].ptype = ipart; hits->mult++; } else { fprintf(stderr,"HDGeant error in hitCentralDC: "); fprintf(stderr,"max hit count %d exceeded, truncating!\n",MAX_HITS); } } /* register hits during tracking (from gustep) */ void hitCentralDC (float xin[4], float xout[4], float pin[5], float pout[5], float dEsum, int track, int stack, int history, int ipart ) { float x[3], t; float dx[3], dr; float dEdx; float xlocal[3]; float xinlocal[3]; float xoutlocal[3]; float dradius,drin,drout; float trackdir[3]; float alpha; if (!initialized) { mystr_t strings[50]; float values[50]; int nvalues = 50; int status = GetConstants("CDC/cdc_parms", &nvalues, values, strings); if (!status) { int ncounter = 0; int i; for ( i=0;i<(int)nvalues;i++) { //printf("%d %s \n",i,strings[i].str); if (!strcmp(strings[i].str,"CDC_DRIFT_SPEED")) { DRIFT_SPEED = values[i]; ncounter++; } if (!strcmp(strings[i].str,"CDC_TWO_HIT_RESOL")) { TWO_HIT_RESOL = values[i]; ncounter++; } if (!strcmp(strings[i].str,"CDC_MAX_HITS")) { MAX_HITS = (int)values[i]; ncounter++; } if (!strcmp(strings[i].str,"CDC_THRESH_KEV")) { THRESH_KEV = values[i]; ncounter++; } } if (ncounter==4) { printf("CDC: ALL parameters loaded from Data Base\n"); } else if (ncounter<5) { printf("CDC: NOT ALL necessary parameters found in Data Base %d out of 5\n",ncounter); } else { printf("CDC: SOME parameters found more than once in Data Base\n"); } } // // Get drift table and scale factors from the database // // First check for non-zero field in the magnet bore float x[3]={0.,0.,65}; float B[3]; gufld_db_(x,B); if (fabs(B[2])>1e-3){ nvalues=78; status=GetColumn("CDC/cdc_drift_table",&nvalues,cdc_drift_time,"t"); int k; for (k=0;k 1.0) t = xin[3] * 1e9; drin = sqrt(xinlocal[0]*xinlocal[0] + xinlocal[1]*xinlocal[1]); drout = sqrt(xoutlocal[0]*xoutlocal[0] + xoutlocal[1]*xoutlocal[1]); trackdir[0] =-xinlocal[0] + xoutlocal[0]; trackdir[1] =-xinlocal[1] + xoutlocal[1]; trackdir[2] =-xinlocal[2] + xoutlocal[2]; alpha=-(xinlocal[0]*trackdir[0]+xinlocal[1]*trackdir[1]) /(trackdir[0]*trackdir[0]+trackdir[1]*trackdir[1]); xlocal[0]=xinlocal[0]+trackdir[0]*alpha; xlocal[1]=xinlocal[1]+trackdir[1]*alpha; xlocal[2]=xinlocal[2]+trackdir[2]*alpha; // Deal with tracks exiting the ends of the straws if (fabs(xlocal[2]) >= 75.45) { float sign = (xoutlocal[2] > 0)? 1. : -1.; int ring = getring_wrapper_(); if (ring <= 4 || (ring >= 13 && ring <= 16) || ring >= 25) { alpha=(sign*75.45-xinlocal[2])/trackdir[2]; xlocal[0]=xinlocal[0]+trackdir[0]*alpha; xlocal[1]=xinlocal[1]+trackdir[1]*alpha; xlocal[2]=sign*75.45; } else if (fabs(xlocal[2]) >= 75.575) { alpha=(sign*75.575-xinlocal[2])/trackdir[2]; xlocal[0]=xinlocal[0]+trackdir[0]*alpha; xlocal[1]=xinlocal[1]+trackdir[1]*alpha; xlocal[2]=sign*75.575; } } /* This will get called when the particle actually passes through * the wire volume itself. For these cases, we should set the * location of the hit to be the point on the wire itself. Do * determine if this is what is happening, we check drout to * see if it is very close to the wire and drin to see if it is * close to the tube. * * For the other case, when drin is close to the wire, we assume * it is because it is emerging from the wire volume and * automatically ignore those hits by returning immediately. */ if (drin < 0.0050) return; /* entering straw within 50 microns of wire. ignore */ if ((drin > (STRAW_RADIUS-0.0200) && drout<0.0050) || (drin < 0.274 && drin > 0.234 && drout<0.0050)) { /* Either we entered within 200 microns of the straw tube and left * within 50 microns of the wire or we entered the stub region near the * donuts at either end of the straw (the inner radius of the feedthrough * region is 0.254 cm) and passed near the wire. Assume the track passed * through the wire volume. */ x[0] = xout[0]; x[1] = xout[1]; x[2] = xout[2]; t = xout[3] * 1e9; xlocal[0] = xoutlocal[0]; xlocal[1] = xoutlocal[1]; xlocal[2] = xoutlocal[2]; /* For dx, we will just assume it is twice the distance from * the straw to wire. */ dx[0] *= 2.0; dx[1] *= 2.0; dx[2] *= 2.0; /* We will approximate the energy loss in the straw to be twice the energy loss in the first half of the straw */ dEsum *= 2.0; } /* Distance of hit from center of wire */ dradius = sqrt(xlocal[0]*xlocal[0] + xlocal[1]*xlocal[1]); /* Calculate dE/dx */ dr = sqrt(dx[0]*dx[0] + dx[1]*dx[1] + dx[2]*dx[2]); if (dr > 1e-3) { dEdx = dEsum/dr; } else { dEdx = 0; } /* post the hit to the truth tree */ if (history == 0) { int mark = (1<<30) + pointCount; void** twig = getTwig(¢ralDCTree, mark); if (*twig == 0) { s_CentralDC_t* cdc = *twig = make_s_CentralDC(); s_CdcTruthPoints_t* points = make_s_CdcTruthPoints(1); int a = thisInputEvent->physicsEvents->in[0].reactions->in[0].vertices->in[0].products->mult; points->in[0].primary = (stack <= a); points->in[0].track = track; points->in[0].t = t; points->in[0].z = x[2]; points->in[0].r = sqrt(x[0]*x[0] + x[1]*x[1]); points->in[0].phi = atan2(x[1],x[0]); points->in[0].dradius = dradius; points->in[0].px = pin[0]*pin[4]; points->in[0].py = pin[1]*pin[4]; points->in[0].pz = pin[2]*pin[4]; points->in[0].dEdx = dEdx; points->in[0].ptype = ipart; points->in[0].trackID = make_s_TrackID(); points->in[0].trackID->itrack = gidGetId(track); points->mult = 1; cdc->cdcTruthPoints = points; pointCount++; } } /* post the hit to the hits tree, mark sector as hit */ if (dEsum > 0) { s_CdcStrawTruthHits_t* hits; int layer = getlayer_wrapper_(); int ring = getring_wrapper_(); int sector = getsector_wrapper_(); if (layer == 0) /* in a straw */ { int mark = (ring<<20) + sector; void** twig = getTwig(¢ralDCTree, mark); if (*twig == 0) { s_CentralDC_t* cdc = *twig = make_s_CentralDC(); s_CdcStraws_t* straws = make_s_CdcStraws(1); straws->mult = 1; straws->in[0].ring = ring; straws->in[0].straw = sector; straws->in[0].cdcStrawTruthHits = hits = make_s_CdcStrawTruthHits(MAX_HITS); cdc->cdcStraws = straws; strawCount++; } else { s_CentralDC_t* cdc = (s_CentralDC_t*) *twig; hits = cdc->cdcStraws->in[0].cdcStrawTruthHits; } /* Simulate number of primary ion pairs*/ /* The total number of ion pairs depends on the energy deposition and the effective average energy to produce a pair, w_eff. On average for each primary ion pair produced there are n_s_per_p secondary ion pairs produced. */ int one=1; // Average number of secondary ion pairs for 50/50 Ar/CO2 mixture float n_s_per_p=1.94; //Average energy needed to produce an ion pair for 50/50 mixture float w_eff=29.5e-9; // GeV // Average number of primary ion pairs float n_p_mean = dEsum/w_eff/(1.+n_s_per_p); int n_p; // number of primary ion pairs gpoiss_(&n_p_mean,&n_p,&one); if (controlparams_.driftclusters==0) { AddCDCCluster(hits,ipart,track,n_p,t,xlocal); } else{ // Loop over the number of primary ion pairs int n; for (n=0; n < n_p; n++) { // Generate a cluster at a random position along the path within the // straw int one=2; float rndno[1]; grndm_(rndno,&one); xlocal[0]=xinlocal[0]+trackdir[0]*rndno[0]; xlocal[1]=xinlocal[1]+trackdir[1]*rndno[0]; xlocal[2]=xinlocal[2]+trackdir[2]*rndno[0]; AddCDCCluster(hits,ipart,track,n_p,t,xlocal); } } } } } /* entry points from fortran */ void hitcentraldc_(float* xin, float* xout, float* pin, float* pout, float* dEsum, int* track, int* stack, int* history, int* ipart) { hitCentralDC(xin,xout,pin,pout,*dEsum,*track,*stack,*history, *ipart); } /* pick and package the hits for shipping */ s_CentralDC_t* pickCentralDC () { s_CentralDC_t* box; s_CentralDC_t* item; if ((strawCount == 0) && (stripCount == 0) && (pointCount == 0)) { return HDDM_NULL; } box = make_s_CentralDC(); box->cdcStraws = make_s_CdcStraws(strawCount); box->cdcTruthPoints = make_s_CdcTruthPoints(pointCount); while ((item = (s_CentralDC_t*) pickTwig(¢ralDCTree))) { s_CdcStraws_t* straws = item->cdcStraws; int straw; s_CdcTruthPoints_t* points = item->cdcTruthPoints; int point; for (straw=0; straw < straws->mult; ++straw) { int m = box->cdcStraws->mult; s_CdcStrawTruthHits_t* hits = straws->in[straw].cdcStrawTruthHits; // Sort the clusters by time qsort(hits->in,hits->mult,sizeof(s_CdcStrawTruthHit_t),(compfn)cdc_cluster_sort); /* compress out the hits below threshold */ int i,iok=0; if (controlparams_.driftclusters == 0) { for (iok=i=0; i < hits->mult; i++) { if (hits->in[i].q >0.) { if (iok < i) { hits->in[iok] = hits->in[i]; } ++iok; } } } else{ // Temporary histogram in 1 ns bins to store waveform data int num_samples=(int)CDC_TIME_WINDOW; float *samples=(float *)malloc(num_samples*sizeof(float)); for (i=0;i= THRESH_MV) { if (returned_to_baseline == 0) { hits->in[iok].itrack = hits->in[0].itrack; hits->in[iok].ptype = hits->in[0].ptype; hits->in[iok].t=(float) i; returned_to_baseline = 1; iok++; } q += (float)FADC_BIN_SIZE*samples[i]; } if (returned_to_baseline && (samples[i] < THRESH_MV)) { returned_to_baseline = 0; if (iok > 0 && q > 0.) { hits->in[iok-1].q=q; q=0.; } //break; } } if (q > 0) { hits->in[iok-1].q = q; } free(samples); } if (iok) { hits->mult = iok; box->cdcStraws->in[m] = straws->in[straw]; box->cdcStraws->mult++; } else if (hits != HDDM_NULL) { FREE(hits); } } if (straws != HDDM_NULL) { FREE(straws); } for (point=0; point < points->mult; ++point) { int m = box->cdcTruthPoints->mult++; box->cdcTruthPoints->in[m] = points->in[point]; } if (points != HDDM_NULL) { FREE(points); } FREE(item); } strawCount = stripCount = pointCount = 0; if ((box->cdcStraws != HDDM_NULL) && (box->cdcStraws->mult == 0)) { FREE(box->cdcStraws); box->cdcStraws = HDDM_NULL; } if ((box->cdcTruthPoints != HDDM_NULL) && (box->cdcTruthPoints->mult == 0)) { FREE(box->cdcTruthPoints); box->cdcTruthPoints = HDDM_NULL; } if ((box->cdcStraws->mult == 0) && (box->cdcTruthPoints->mult == 0)) { FREE(box); box = HDDM_NULL; } return box; }