// $Id$ // // Created June 22, 2005 David Lawrence #include #include #include using namespace std; #include #include #include "HDDM/hddm_s.h" float RANDOM_MAX = (float)(0x7FFFFFFF); #ifndef _DBG_ #define _DBG_ cout<<__FILE__<<":"<<__LINE__<<" " #define _DBG__ cout<<__FILE__<<":"<<__LINE__< NCDC_STRAWS; vector CDC_RING_RADIUS; DFCALGeometry *fcalGeom = NULL; bool FDC_GEOMETRY_INITIALIZED = false; unsigned int NFDC_WIRES_PER_PLANE; vector FDC_LAYER_Z; double FDC_RATE_COEFFICIENT; double SampleGaussian(double sigma); double SampleRange(double x1, double x2); // Do we or do we not add noise hits bool ADD_NOISE = true; // Do we or do we not smear real hits bool SMEAR_HITS = true; // If the following flag is true, then include the drift-distance // dependency on the error in the CDC position. Otherwise, use a // flat distribution given by the CDC_TDRIFT_SIGMA below. bool CDC_USE_PARAMETERIZED_SIGMA = true; // The error on the drift time in the CDC. The drift times // for the actual CDC hits coming from the input file // are smeared by a gaussian with this sigma. double CDC_TDRIFT_SIGMA = 150.0/55.0*1E-9; // in seconds // The time window for which CDC hits are accumulated. // This is used to determine the number of background // hits in the CDC for a given event. double CDC_TIME_WINDOW = 1000.0E-9; // in seconds // The error on the drift time in the CDC. The drift times // for the actual CDC hits coming from the input file // are smeared by a gaussian with this sigma. double FDC_TDRIFT_SIGMA = 200.0/55.0*1.0E-9; // in seconds // The error in the distance along the wire as measured by // the cathodes. This should NOT include the Lorentz // effect which is already included in hdgeant. It // should include any fluctuations due to ion trail density // etc. double FDC_CATHODE_SIGMA = 150.0; // in microns // The FDC pedestal noise is used to smear the cathode ADC // values such that the position along the wire has the resolution // specified by FDC_CATHODE_SIGMA. double FDC_PED_NOISE; //pC (calculated from FDC_CATHODE_SIGMA in SmearFDC) // If energy loss was turned off in the FDC then the pedestal // noise will not be scaled properly to give the nominal 200 micron // resolution along the wires. This flag is used to indicated // the magic scale factor should be applied to FDC_PED_NOISE // when it is calculated below to get the correct resolution. bool FDC_ELOSS_OFF = false; // Time window for acceptance of FDC hits double FDC_TIME_WINDOW = 1000.0E-9; // in seconds // Photon-statistics factor for smearing hit energy (from Criss's MC) // (copied from DFCALMCResponse_factory.cc 7/2/2009 DL) double FCAL_PHOT_STAT_COEF = 0.035; // Single block energy threshold (applied after smearing) double FCAL_BLOCK_THRESHOLD = 20.0*k_MeV; // Forward TOF resolution double TOF_SIGMA = 100.*k_psec; //----------- // Smear //----------- void Smear(s_HDDM_t *hddm_s) { if(SMEAR_HITS)SmearCDC(hddm_s); if(ADD_NOISE)AddNoiseHitsCDC(hddm_s); if(SMEAR_HITS)SmearFDC(hddm_s); if(ADD_NOISE)AddNoiseHitsFDC(hddm_s); if(SMEAR_HITS)SmearFCAL(hddm_s); if(SMEAR_HITS)SmearBCAL(hddm_s); if(SMEAR_HITS)SmearTOF(hddm_s); if(SMEAR_HITS)SmearUPV(hddm_s); if(SMEAR_HITS)SmearCherenkov(hddm_s); } //----------- // SmearCDC //----------- void SmearCDC(s_HDDM_t *hddm_s) { /// Smear the drift times of all CDC hits. // Acquire the pointer to the physics events s_PhysicsEvents_t* allEvents = hddm_s->physicsEvents; if(!allEvents)return; for (unsigned int m=0; m < allEvents->mult; m++) { // Acquire the pointer to the overall hits section of the data s_HitView_t *hits = allEvents->in[m].hitView; if (hits == HDDM_NULL)return; if (hits->centralDC == HDDM_NULL)return; if (hits->centralDC->cdcStraws == HDDM_NULL)return; for(unsigned int k=0; kcentralDC->cdcStraws->mult; k++){ s_CdcStraw_t *cdcstraw = &hits->centralDC->cdcStraws->in[k]; for(unsigned int j=0; jcdcStrawHits->mult; j++){ s_CdcStrawHit_t *strawhit = &cdcstraw->cdcStrawHits->in[j]; double sigma_t = CDC_TDRIFT_SIGMA; if(CDC_USE_PARAMETERIZED_SIGMA){ // Convert drift time back to drift distance assuming standard 55 um/ns double drift_d = strawhit->t*55.0E-3; // use mm since that's what the error function was paramaterized // The following is from a fit to Yves' numbers circa Aug 2009. The values fit were // resolution (microns) vs. drift distance (mm). // par[8] = {699.875, -559.056, 149.391, 25.6929, -22.0238, 4.75091, -0.452373, 0.0163858}; double x = drift_d; //double sigma_d = (699.875) + x*((-559.056) + x*((149.391) + x*((25.6929) + x*((-22.0238) + x*((4.75091) + x*((-0.452373) + x*((0.0163858)))))))); double sigma_d = 108.55 + 7.62391*x + 556.176*exp(-(1.12566)*pow(x,1.29645)); sigma_t = sigma_d/55.0; // remember that sigma_d is already in microns here! sigma_t *= 1.0E-9; // convert sigma_t to seconds } // Smear out the CDC drift time using the specified sigma. // This should include both timing resolution and ion trail // density effects. double delta_t = SampleGaussian(sigma_t)*1.0E9; // delta_t is in ns strawhit->t += delta_t; // If the time is negative, reject this smear and try again //if(strawhit->t<0)j--; } } } } //----------- // AddNoiseHitsCDC //----------- void AddNoiseHitsCDC(s_HDDM_t *hddm_s) { if(!CDC_GEOMETRY_INITIALIZED)InitCDCGeometry(); // Calculate the number of noise hits for each straw and store // them in a sparse map. We must do it this way since we have to know // the total number of CdcStraw_t structures to allocate in our // call to make_s_CdcStraws. // // The straw rates are obtained using a parameterization done // to calculate the event size for the August 29, 2007 online // meeting. This parameterization is almost already obsolete. // 10/12/2007 D. L. vector Nstraw_hits; vector straw_number; vector ring_number; int Nnoise_straws = 0; int Nnoise_hits = 0; for(unsigned int ring=1; ring<=NCDC_STRAWS.size(); ring++){ double p[2] = {10.4705, -0.103046}; double r_prime = (double)(ring+3); double N = exp(p[0] + r_prime*p[1]); N *= CDC_TIME_WINDOW; for(unsigned int straw=1; straw<=NCDC_STRAWS[ring-1]; straw++){ // Indivdual straw rates should be way less than 1/event so // we just use the rate as a probablity. double Nhits = SampleRange(0.0, 1.0)physicsEvents; if(!PE) return; for(unsigned int i=0; imult; i++){ s_HitView_t *hits = PE->in[i].hitView; if (hits == HDDM_NULL)continue; // If no CDC hits were produced by HDGeant, then we need to add // the branches needed to the HDDM tree if(hits->centralDC == HDDM_NULL){ hits->centralDC = make_s_CentralDC(); hits->centralDC->cdcStraws = (s_CdcStraws_t*)HDDM_NULL; hits->centralDC->cdcTruthPoints = (s_CdcTruthPoints_t*)HDDM_NULL; } if(hits->centralDC->cdcStraws == HDDM_NULL){ hits->centralDC->cdcStraws = make_s_CdcStraws(0); hits->centralDC->cdcStraws->mult=0; } // Get existing hits s_CdcStraws_t *old_cdcstraws = hits->centralDC->cdcStraws; unsigned int Nold = old_cdcstraws->mult; // Create CdcStraws structure that has enough slots for // both the real and noise hits. s_CdcStraws_t* cdcstraws = make_s_CdcStraws((unsigned int)Nnoise_straws + Nold); // Add real hits back in first cdcstraws->mult = 0; for(unsigned int j=0; jin[cdcstraws->mult++] = old_cdcstraws->in[j]; // We need to transfer ownership of the hits to the new cdcstraws // branch so they don't get deleted when old_cdcstraws is freed. s_CdcStraw_t *cdcstraw = &old_cdcstraws->in[j]; cdcstraw->cdcStrawHits = (s_CdcStrawHits_t *)HDDM_NULL; } // Delete memory used for old hits structure and // replace pointer in HDDM tree with ours free(old_cdcstraws); hits->centralDC->cdcStraws = cdcstraws; // Loop over straws with noise hits for(unsigned int j=0; jin[cdcstraws->mult++]; s_CdcStrawHits_t *strawhits = make_s_CdcStrawHits(Nstraw_hits[j]); cdcstraw->cdcStrawHits = strawhits; cdcstraw->ring = ring_number[j]; cdcstraw->straw = straw_number[j]; strawhits->mult = 0; for(int k=0; kin[strawhits->mult++]; strawhit->dE = 1.0; strawhit->t = SampleRange(-CDC_TIME_WINDOW/2.0, +CDC_TIME_WINDOW/2.0)*1.e9; } } } } //----------- // SmearFDC //----------- void SmearFDC(s_HDDM_t *hddm_s) { // Calculate ped noise level based on position resolution FDC_PED_NOISE=-0.004594+0.008711*FDC_CATHODE_SIGMA+0.000010*FDC_CATHODE_SIGMA*FDC_CATHODE_SIGMA; //pC if(FDC_ELOSS_OFF)FDC_PED_NOISE*=7.0; // empirical 4/29/2009 DL // Loop over Physics Events s_PhysicsEvents_t* PE = hddm_s->physicsEvents; if(!PE) return; for(unsigned int i=0; imult; i++){ s_HitView_t *hits = PE->in[i].hitView; if (hits == HDDM_NULL || hits->forwardDC == HDDM_NULL || hits->forwardDC->fdcChambers == HDDM_NULL)continue; s_FdcChambers_t* fdcChambers = hits->forwardDC->fdcChambers; s_FdcChamber_t *fdcChamber = fdcChambers->in; for(unsigned int j=0; jmult; j++, fdcChamber++){ // Add pedestal noise to strip charge data s_FdcCathodeStrips_t *strips= fdcChamber->fdcCathodeStrips; if (strips!=HDDM_NULL){ s_FdcCathodeStrip_t *strip=strips->in; for (unsigned int k=0;kmult;k++,strip++){ s_FdcCathodeHits_t *hits=strip->fdcCathodeHits; if (hits==HDDM_NULL)continue; s_FdcCathodeHit_t *hit=hits->in; for (unsigned int s=0;smult;s++,hit++){ hit->q+=SampleGaussian(FDC_PED_NOISE); } } } // Add drift time varation to the anode data s_FdcAnodeWires_t *wires=fdcChamber->fdcAnodeWires; if (wires!=HDDM_NULL){ s_FdcAnodeWire_t *wire=wires->in; for (unsigned int k=0;kmult;k++,wire++){ s_FdcAnodeHits_t *hits=wire->fdcAnodeHits; if (hits==HDDM_NULL)continue; s_FdcAnodeHit_t *hit=hits->in; for (unsigned int s=0;smult;s++,hit++){ hit->t+=SampleGaussian(FDC_TDRIFT_SIGMA)*1.0E9; } } } } } } //----------- // AddNoiseHitsFDC //----------- void AddNoiseHitsFDC(s_HDDM_t *hddm_s) { if(!FDC_GEOMETRY_INITIALIZED)InitFDCGeometry(); // Calculate the number of noise hits for each FDC wire and store // them in a sparse map. We must do it this way since we have to know // the total number of s_FdcAnodeWire_t structures to allocate in our // call to make_s_FdcAnodeWires. // // We do this using the individual wire rates to calculate the probability // of the wire firing for a single event. For the FDC, we calculate the // wire rates as a function of both wire number (distance from beam line) // and layer (position in z). We want a roughly 1/r distribution in the // radial direction and a roughly exponential rise in rate in the // +z direction. // // The wire rates are obtained using a parameterization done // to calculate the event size for the August 29, 2007 online // meeting. This parameterization is almost already obsolete. // In rough terms, the layer rate (integrated over all wires) // is about 1 MHz. For a 24 layer chamber with a 1us time window, // we should have approximately 24 background hits per event. // 11/9/2007 D. L. vector Nwire_hits; vector wire_number; vector layer_number; int Nnoise_wires = 0; int Nnoise_hits = 0; for(unsigned int layer=1; layer<=FDC_LAYER_Z.size(); layer++){ double No = FDC_RATE_COEFFICIENT*exp((double)layer*log(4.0)/24.0); for(unsigned int wire=1; wire<=96; wire++){ double rwire = fabs(96.0/2.0 - (double)wire); double N = No*log((rwire+0.5)/(rwire-0.5)); // Indivdual wire rates should be way less than 1/event so // we just use the rate as a probablity. double Nhits = SampleRange(0.0, 1.0)physicsEvents; if(!PE) return; for(unsigned int i=0; imult; i++){ s_HitView_t *hits = PE->in[i].hitView; if (hits == HDDM_NULL)continue; // If no FDC hits were produced by HDGeant, then we need to add // the branches needed to the HDDM tree if(hits->forwardDC == HDDM_NULL){ hits->forwardDC = make_s_ForwardDC(); hits->forwardDC->fdcChambers = (s_FdcChambers_t*)HDDM_NULL; } if(hits->forwardDC->fdcChambers == HDDM_NULL){ hits->forwardDC->fdcChambers = make_s_FdcChambers(0); hits->forwardDC->fdcChambers->mult=0; } // Get existing hits s_FdcChambers_t *old_fdcchambers = hits->forwardDC->fdcChambers; unsigned int Nold = old_fdcchambers->mult; // If we were doing this "right" we'd conglomerate all of the noise // hits from the same chamber into the same s_FdcChamber_t structure. // That's a pain in the butt and really may only save a tiny bit of disk // space so we just add each noise hit back in as another chamber // structure. // Create FdcChambers structure that has enough slots for // both the real and noise hits. s_FdcChambers_t* fdcchambers = make_s_FdcChambers(Nwire_hits.size() + Nold); // Add real hits back in first fdcchambers->mult = 0; for(unsigned int j=0; jin[fdcchambers->mult++] = old_fdcchambers->in[j]; // We need to transfer ownership of the hits to the new fdcchambers // branch so they don't get deleted when old_fdcchambers is freed. s_FdcChamber_t *fdcchamber = &old_fdcchambers->in[j]; fdcchamber->fdcAnodeWires = (s_FdcAnodeWires_t *)HDDM_NULL; fdcchamber->fdcCathodeStrips = (s_FdcCathodeStrips_t *)HDDM_NULL; } // Delete memory used for old hits structure and // replace pointer in HDDM tree with ours free(old_fdcchambers); hits->forwardDC->fdcChambers = fdcchambers; // Loop over wires with noise hits for(unsigned int j=0; jin[fdcchambers->mult++]; // Create structure for anode wires s_FdcAnodeWires_t *fdcAnodeWires = make_s_FdcAnodeWires(Nwire_hits[j]); fdcchamber->fdcAnodeWires = fdcAnodeWires; fdcAnodeWires->mult = 0; for(int k=0; kin[fdcAnodeWires->mult++]; // Create anode hits structure s_FdcAnodeHits_t *fdcanodehits = make_s_FdcAnodeHits(1); fdcAnodeWire->fdcAnodeHits = fdcanodehits; // Get pointer to anode hit structure fdcanodehits->mult = 1; s_FdcAnodeHit_t *fdcanodehit = &fdcanodehits->in[0]; fdcanodehit->dE = 0.1; // what should this be? fdcanodehit->t = SampleRange(-FDC_TIME_WINDOW/2., +FDC_TIME_WINDOW/2.)*1.e9; fdcAnodeWire->wire = wire_number[j]; fdcchamber->layer = (layer_number[j]-1)%3 + 1; fdcchamber->module = (layer_number[j]-1)/3 + 1; } } } } //----------- // SmearFCAL //----------- void SmearFCAL(s_HDDM_t *hddm_s) { /// Smear the FCAL hits using the nominal resolution of the individual blocks. /// The way this works is a little funny and warrants a little explanation. /// The information coming from hdgeant is truth information indexed by /// row and column, but containing energy deposited and time. The mcsmear /// program will copy the truth information from the FcalTruthBlock to the /// FcalBlock branch, smearing the values with the appropriate detector /// resolution. /// /// To access the "truth" values in DANA, get the DFCALHit objects using the /// "TRUTH" tag. // The code below is perhaps slightly odd in that it simultaneously creates // and fills the mirror (truth) branch while smearing the values in the // nominal hit branch. // Loop over Physics Events s_PhysicsEvents_t* PE = hddm_s->physicsEvents; if(!PE) return; if(!fcalGeom)fcalGeom = new DFCALGeometry(); for(unsigned int i=0; imult; i++){ s_HitView_t *hits = PE->in[i].hitView; if (hits == HDDM_NULL || hits->forwardEMcal == HDDM_NULL || hits->forwardEMcal->fcalBlocks == HDDM_NULL)continue; s_FcalBlocks_t *blocks = hits->forwardEMcal->fcalBlocks; for(unsigned int j=0; jmult; j++){ s_FcalBlock_t *block = &blocks->in[j]; // Create FCAL hits structures to put smeared data into if(block->fcalHits!=HDDM_NULL)free(block->fcalHits); block->fcalHits = make_s_FcalHits(block->fcalTruthHits->mult); block->fcalHits->mult = block->fcalTruthHits->mult; for(unsigned int k=0; kfcalTruthHits->mult; k++){ s_FcalTruthHit_t *fcaltruthhit = &block->fcalTruthHits->in[k]; s_FcalHit_t *fcalhit = &block->fcalHits->in[k]; // Copy info from truth stream before doing anything else fcalhit->E = fcaltruthhit->E; fcalhit->t = fcaltruthhit->t; // Simulation simulates a grid of blocks for simplicity. // Do not bother smearing inactive blocks. They will be // discarded in DEventSourceHDDM.cc while being read in // anyway. if(!fcalGeom->isBlockActive( block->row, block->column ))continue; // Smear the energy and timing of the hit double sigma = FCAL_PHOT_STAT_COEF/sqrt(fcalhit->E) ; fcalhit->E *= 1.0 + SampleGaussian(sigma); fcalhit->t += SampleGaussian(200.0E-3); // smear by 200 ps fixed for now 7/2/2009 DL // Apply a single block threshold. If the (smeared) energy is below this, // then set the energy and time to zero. if(fcalhit->E < FCAL_BLOCK_THRESHOLD){fcalhit->E = fcalhit->t = 0.0;} } // k (fcalhits) } // j (blocks) } // i (physicsEvents) } //----------- // SmearBCAL //----------- void SmearBCAL(s_HDDM_t *hddm_s) { } //----------- // SmearTOF //----------- void SmearTOF(s_HDDM_t *hddm_s) { // Loop over Physics Events s_PhysicsEvents_t* PE = hddm_s->physicsEvents; if(!PE) return; for(unsigned int i=0; imult; i++){ s_HitView_t *hits = PE->in[i].hitView; if (hits == HDDM_NULL || hits->forwardTOF == HDDM_NULL || hits->forwardTOF->ftofCounters == HDDM_NULL)continue; s_FtofCounters_t* ftofCounters = hits->forwardTOF->ftofCounters; // Loop over counters s_FtofCounter_t *ftofCounter = ftofCounters->in; for(unsigned int j=0;jmult; j++, ftofCounter++){ // Loop over north AND south hits s_FtofNorthHits_t *ftofNorthHits = ftofCounter->ftofNorthHits; s_FtofNorthHit_t *ftofNorthHit = ftofNorthHits->in; s_FtofSouthHits_t *ftofSouthHits = ftofCounter->ftofSouthHits; s_FtofSouthHit_t *ftofSouthHit = ftofSouthHits->in; for (unsigned int m=0;mmult;m++,ftofNorthHit++){ // Smear the time ftofNorthHit->t +=SampleGaussian(TOF_SIGMA); } for (unsigned int m=0;mmult;m++,ftofSouthHit++){ // Smear the time ftofSouthHit->t +=SampleGaussian(TOF_SIGMA); } } } } //----------- // SmearUPV //----------- void SmearUPV(s_HDDM_t *hddm_s) { } //----------- // SmearCherenkov //----------- void SmearCherenkov(s_HDDM_t *hddm_s) { } //----------- // InitCDCGeometry //----------- void InitCDCGeometry(void) { CDC_GEOMETRY_INITIALIZED = true; CDC_MAX_RINGS = 25; //-- This was cut and pasted from DCDCTrackHit_factory.cc on 10/11/2007 -- float degrees0 = 0.0; float degrees6 = 6.0*M_PI/180.0; for(int ring=1; ring<=CDC_MAX_RINGS; ring++){ int myNstraws=0; float radius = 0.0; float stereo=0.0; float phi_shift=0.0; float deltaX=0.0, deltaY=0.0; float rotX=0.0, rotY=0.0; switch(ring){ // axial case 1: myNstraws= 42; radius= 10.7219; stereo= degrees0; phi_shift= 0.00000; break; case 2: myNstraws= 42; radius= 12.097; stereo= degrees0; phi_shift= 4.285714; break; case 3: myNstraws= 54; radius= 13.7803; stereo= degrees0; phi_shift= 2.00000; break; case 4: myNstraws= 54; radius= 15.1621; stereo= degrees0; phi_shift= 5.3333333; break; // -stereo case 5: myNstraws= 66; radius= 16.9321; stereo= -degrees6; phi_shift= 0.33333; break; case 6: myNstraws= 66; phi_shift= 0.33333; deltaX= 18.2948 ; deltaY= 0.871486; rotX=-6.47674; rotY=-0.302853; break; case 7: myNstraws= 80; radius= 20.5213; stereo= -degrees6; phi_shift= -0.5000; break; case 8: myNstraws= 80; phi_shift= -0.5000; deltaX= 21.8912; deltaY= 0.860106; rotX=-6.39548; rotY=-0.245615; break; // +stereo case 9: myNstraws= 93; radius= 23.8544; stereo= +degrees6; phi_shift= 1.1000; break; case 10: myNstraws= 93; phi_shift= 1.1000; deltaX= 25.229; deltaY= 0.852573; rotX=+6.34142; rotY=+0.208647; break; case 11: myNstraws= 106; radius= 27.1877; stereo= +degrees6; phi_shift= -1.40; break; case 12: myNstraws= 106; phi_shift= -1.400; deltaX= 28.5658; deltaY= 0.846871; rotX=+6.30035; rotY=+0.181146; break; // axial case 13: myNstraws= 123; radius= 31.3799; stereo= degrees0; phi_shift= 0.5000000; break; case 14: myNstraws= 123; radius= 32.7747; stereo= degrees0; phi_shift= 1.9634146; break; case 15: myNstraws= 135; radius= 34.4343; stereo= degrees0; phi_shift= 1.0000000; break; case 16: myNstraws= 135; radius= 35.8301; stereo= degrees0; phi_shift= 2.3333333; break; // -stereo case 17: myNstraws= 146; radius= 37.4446; stereo= -degrees6; phi_shift= 0.2; break; case 18: myNstraws= 146; phi_shift= 0.2; deltaX= 38.8295; deltaY= 0.835653; rotX=-6.21919 ; rotY=-0.128247; break; case 19: myNstraws= 158; radius= 40.5364; stereo= -degrees6; phi_shift= 0.7; break; case 20: myNstraws= 158; phi_shift= 0.7; deltaX=41.9225 ; deltaY= 0.833676; rotX=-6.20274; rotY=-0.118271; break; // +stereo case 21: myNstraws= 170; radius= 43.6152; stereo= +degrees6; phi_shift= 1.1000; break; case 22: myNstraws= 170; phi_shift= 1.1000; deltaX=45.0025 ; deltaY= 0.831738; rotX=+6.18859; rotY=+0.109325; break; case 23: myNstraws= 182; radius= 46.6849; stereo= +degrees6; phi_shift= 1.40; break; case 24: myNstraws= 182; phi_shift= 1.400; deltaX= 48.0733; deltaY= 0.829899; rotX=+6.1763; rotY=+0.101315; break; // axial case 25: myNstraws= 197; radius= 50.37; stereo= degrees0; phi_shift= 0.200000000; break; case 26: myNstraws= 197; radius= 51.77; stereo= degrees0; phi_shift= 1.113705000; break; case 27: myNstraws= 209; radius= 53.363; stereo= degrees0; phi_shift= 0.800000000; break; case 28: myNstraws= 209; radius= 54.76; stereo= degrees0; phi_shift= 1.661244; break; default: cerr<<__FILE__<<":"<<__LINE__<<" Invalid value for CDC ring ("<