QwAnalysis
QwBPMStripline.cc
Go to the documentation of this file.
1 /**********************************************************\
2 * File: QwBPMStripline.cc *
3 * *
4 * Author: *
5 * Time-stamp: *
6 \**********************************************************/
7 
8 #include "QwBPMStripline.h"
9 
10 // System headers
11 #include <stdexcept>
12 
13 // Qweak headers
14 #include "QwDBInterface.h"
15 #include "QwVQWK_Channel.h"
16 #include "QwScaler_Channel.h"
17 
18 
19 /* Position calibration factor, transform ADC counts in mm*/
20 //const Double_t QwBPStripline::kQwStriplineCalibration = 18.77;
21 //const Double_t QwBPMStripline::kRotationCorrection = 1./1.414;
22 template<typename T>
23 const TString QwBPMStripline<T>::subelement[4]={"XP","XM","YP","YM"};
24 
25 template<typename T>
27 {
28  Short_t i=0;
29  Bool_t localdebug = kFALSE;
30 
31 
33 
34  for(i=kXAxis;i<kNumAxes;i++)
35  fAbsPos[i].InitializeChannel(name+kAxisLabel[i],"derived");
36 
37  fEffectiveCharge.InitializeChannel(name+"_EffectiveCharge","derived");
38 
39  for(i=0;i<4;i++) {
40  fWire[i].InitializeChannel(name+subelement[i],"raw");
41  if(localdebug)
42  std::cout<<" Wire ["<<i<<"]="<<fWire[i].GetElementName()<<"\n";
43  }
44 
45  for(i=kXAxis;i<kNumAxes;i++) fRelPos[i].InitializeChannel(name+"Rel"+kAxisLabel[i],"derived");
46 
47  bFullSave=kTRUE;
48 
49  return;
50 }
51 
52 template<typename T>
53 void QwBPMStripline<T>::InitializeChannel(TString subsystem, TString name,
54  TString type)
55 {
56  SetModuleType(type);
57  InitializeChannel(subsystem, name);
58 }
59 
60 template<typename T>
61 void QwBPMStripline<T>::InitializeChannel(TString subsystem, TString name)
62 {
63  Short_t i=0;
64  Bool_t localdebug = kFALSE;
65 
67 
68  for(i=kXAxis;i<kNumAxes;i++)
69  fAbsPos[i].InitializeChannel(subsystem, "QwBPMStripline", name+kAxisLabel[i],"derived");
70 
71  fEffectiveCharge.InitializeChannel(subsystem, "QwBPMStripline", name+"_EffectiveCharge","derived");
72 
73  for(i=0;i<4;i++) {
74  fWire[i].InitializeChannel(subsystem, "QwBPMStripline", name+subelement[i],"raw");
75  if(localdebug)
76  std::cout<<" Wire ["<<i<<"]="<<fWire[i].GetElementName()<<"\n";
77  }
78 
79  for(i=kXAxis;i<kNumAxes;i++) fRelPos[i].InitializeChannel(subsystem, "QwBPMStripline", name+"Rel"+kAxisLabel[i],"derived");
80 
81  bFullSave=kTRUE;
82 
83  return;
84 }
85 
86 template<typename T>
88 {
89  Short_t i=0;
90 
91  for(i=0;i<4;i++) fWire[i].ClearEventData();
92 
93  for(i=kXAxis;i<kNumAxes;i++){
94  fAbsPos[i].ClearEventData();
95  fRelPos[i].ClearEventData();
96  }
97  fEffectiveCharge.ClearEventData();
98 
99  return;
100 }
101 
102 
103 template<typename T>
105 {
106  Bool_t eventokay=kTRUE;
107 
108  UInt_t deviceerror=0;
109  for(Short_t i=0;i<4;i++)
110  {
111  deviceerror|= fWire[i].ApplyHWChecks(); //OR the error code from each wire
112  eventokay &= (deviceerror & 0x0);//AND with 0 since zero means HW is good.
113 
114  // if (bDEBUG) std::cout<<" Inconsistent within BPM terminals wire[ "<<i<<" ] "<<std::endl;
115  // if (bDEBUG) std::cout<<" wire[ "<<i<<" ] sequence num "<<fWire[i].GetSequenceNumber()<<" sample size "<<fWire[i].GetNumberOfSamples()<<std::endl;
116  }
117 
118  return eventokay;
119 }
120 
121 
122 template<typename T>
124 {
125  Short_t i=0;
126 
127  for(i=0;i<4;i++) fWire[i].IncrementErrorCounters();
128  for(i=kXAxis;i<kNumAxes;i++) {
129  fRelPos[i].IncrementErrorCounters();
130  fAbsPos[i].IncrementErrorCounters();
131  }
132  fEffectiveCharge.IncrementErrorCounters();
133 }
134 
135 template<typename T>
137 {
138  Short_t i=0;
139 
140  for(i=0;i<4;i++) fWire[i].PrintErrorCounters();
141  for(i=kXAxis;i<kNumAxes;i++) {
142  fRelPos[i].PrintErrorCounters();
143  fAbsPos[i].PrintErrorCounters();
144  }
145  fEffectiveCharge.PrintErrorCounters();
146 }
147 
148 template<typename T>
150  Short_t i=0;
151  UInt_t error=0;
152  for(i=0;i<4;i++) error|=fWire[i].GetEventcutErrorFlag();
153  for(i=kXAxis;i<kNumAxes;i++) {
154  error|=fRelPos[i].GetEventcutErrorFlag();
155  error|=fAbsPos[i].GetEventcutErrorFlag();
156  }
157  error|=fEffectiveCharge.GetEventcutErrorFlag();
158  return error;
159 }
160 
161 template<typename T>
163 {
164  Short_t i=0;
165  UInt_t error1=0;
166  UInt_t error2=0;
167  for(i=0;i<4;i++){
168  error1 |= fWire[i].GetErrorCode();
169  error2 |= fWire[i].GetEventcutErrorFlag();
170  }
171  for(i=kXAxis;i<kNumAxes;i++) {
172  fRelPos[i].UpdateErrorFlag(error1);
173  fAbsPos[i].UpdateErrorFlag(error1);
174  error2|=fRelPos[i].GetEventcutErrorFlag();
175  error2|=fAbsPos[i].GetEventcutErrorFlag();
176  }
177  fEffectiveCharge.UpdateErrorFlag(error1);
178  error2|=fEffectiveCharge.GetEventcutErrorFlag();
179  return error2;
180 };
181 
182 
183 template<typename T>
185  Short_t i=0;
186  try {
187  if(typeid(*ev_error)==typeid(*this)) {
188  // std::cout<<" Here in QwBPMStripline::UpdateErrorFlag \n";
189  if (this->GetElementName()!="") {
190  const QwBPMStripline<T>* value_bpm = dynamic_cast<const QwBPMStripline<T>* >(ev_error);
191  for(i=0;i<4;i++){
192  fWire[i].UpdateErrorFlag(value_bpm->fWire[i]);
193  }
194  for(i=kXAxis;i<kNumAxes;i++) {
195  fRelPos[i].UpdateErrorFlag(value_bpm->fRelPos[i]);
196  fAbsPos[i].UpdateErrorFlag(value_bpm->fAbsPos[i]);
197  }
198  fEffectiveCharge.UpdateErrorFlag(value_bpm->fEffectiveCharge);
199  }
200  } else {
201  TString loc="Standard exception from QwBPMStripline::UpdateErrorFlag :"+
202  ev_error->GetElementName()+" "+this->GetElementName()+" are not of the "
203  +"same type";
204  throw std::invalid_argument(loc.Data());
205  }
206  } catch (std::exception& e) {
207  std::cerr<< e.what()<<std::endl;
208  }
209 };
210 
211 
212 template<typename T>
214 {
215  Bool_t status=kTRUE;
216  Int_t i=0;
217 
218  UInt_t element_error_code[2];
219  //Event cuts for four wires
220  for(i=0;i<4;i++){
221  if (fWire[i].ApplySingleEventCuts()){ //for RelX
222  status&=kTRUE;
223  }
224  else{
225  status&=kFALSE;
226  if (bDEBUG) std::cout<<" Abs X event cut failed ";
227  }
228  }
229 
230  //Get the rex/abs X event cut error flag from xm and xp
231  element_error_code[kXAxis] = GetSubelementByName("xm")->GetErrorCode() | GetSubelementByName("xp")->GetErrorCode();
232  //Get the rex/abs Y event cut error flag from ym and yp
233  element_error_code[kYAxis] = GetSubelementByName("ym")->GetErrorCode() | GetSubelementByName("yp")->GetErrorCode();
234  //Update the error flags for rel and abs positions
235  fRelPos[kXAxis].UpdateErrorFlag(element_error_code[kXAxis]);
236  fRelPos[kYAxis].UpdateErrorFlag(element_error_code[kYAxis]);
237  fAbsPos[kXAxis].UpdateErrorFlag(element_error_code[kXAxis]);
238  fAbsPos[kYAxis].UpdateErrorFlag(element_error_code[kYAxis]);
239  //update the sum of error flags of all wires to the charge element
240  fEffectiveCharge.UpdateErrorFlag(element_error_code[kXAxis]|element_error_code[kYAxis]);
241 
242 
243 
244  //Event cuts for Relative X & Y
245  for(i=kXAxis;i<kNumAxes;i++){
246  if (fRelPos[i].ApplySingleEventCuts()){ //for RelX
247  status&=kTRUE;
248  }
249  else{
250  status&=kFALSE;
251  if (bDEBUG) std::cout<<" Rel X event cut failed ";
252  }
253  }
254 
255  for(i=kXAxis;i<kNumAxes;i++){
256  if (fAbsPos[i].ApplySingleEventCuts()){ //for RelX
257  status&=kTRUE;
258  }
259  else{
260  status&=kFALSE;
261  if (bDEBUG) std::cout<<" Abs X event cut failed ";
262  }
263  }
264 
265  //Event cuts for four wire sum (EffectiveCharge) are already ORed when EffectiveCharge is calculated
266  if (fEffectiveCharge.ApplySingleEventCuts()){
267  status&=kTRUE;
268  }
269  else{
270  status&=kFALSE;
271  if (bDEBUG) std::cout<<"EffectiveCharge event cut failed ";
272  }
273  return status;
274 }
275 
276 template<typename T>
278 {
279  VQwHardwareChannel* tmpptr = NULL;
280  ch_name.ToLower();
281  if (ch_name=="xp"){
282  tmpptr = &fWire[0];
283  }else if (ch_name=="xm"){
284  tmpptr = &fWire[1];
285  }else if (ch_name=="yp"){
286  tmpptr = &fWire[2];
287  }else if (ch_name=="ym"){
288  tmpptr = &fWire[3];
289  }else if (ch_name=="relx"){
290  tmpptr = &fRelPos[0];
291  }else if (ch_name=="rely"){
292  tmpptr = &fRelPos[1];
293  }else if (ch_name=="absx" || ch_name=="x" ){
294  tmpptr = &fAbsPos[0];
295  }else if (ch_name=="absy" || ch_name=="y"){
296  tmpptr = &fAbsPos[1];
297  }else if (ch_name=="effectivecharge" || ch_name=="charge"){
298  tmpptr = &fEffectiveCharge;
299  } else {
300  TString loc="QwBPMStripline::GetSubelementByName for"
301  + this->GetElementName() + " was passed "
302  + ch_name + ", which is an unrecognized subelement name.";
303  throw std::invalid_argument(loc.Data());
304  }
305  return tmpptr;
306 }
307 
308 /*
309 template<typename T>
310 void QwBPMStripline<T>::SetSingleEventCuts(TString ch_name, Double_t minX, Double_t maxX)
311 {
312  VQwHardwareChannel* tmpptr = GetSubelementByName(ch_name);
313  QwMessage << ch_name
314  << " LL " << minX <<" UL " << maxX <<QwLog::endl;
315  tmpptr->SetSingleEventCuts(minX,maxX);
316 }
317 
318 
319 template<typename T>
320 void QwBPMStripline<T>::SetSingleEventCuts(TString ch_name, UInt_t errorflag,Double_t minX, Double_t maxX, Double_t stability){
321  errorflag|=kBPMErrorFlag;//update the device flag
322  if (ch_name=="xp"){
323  QwMessage<<"XP LL " << minX <<" UL " << maxX <<QwLog::endl;
324  fWire[0].SetSingleEventCuts(errorflag,minX,maxX,stability);
325 
326  }else if (ch_name=="xm"){
327  QwMessage<<"XM LL " << minX <<" UL " << maxX <<QwLog::endl;
328  fWire[1].SetSingleEventCuts(errorflag,minX,maxX,stability);
329 
330  }else if (ch_name=="yp"){
331  QwMessage<<"YP LL " << minX <<" UL " << maxX <<QwLog::endl;
332  fWire[2].SetSingleEventCuts(errorflag,minX,maxX,stability);
333 
334  }else if (ch_name=="ym"){
335  QwMessage<<"YM LL " << minX <<" UL " << maxX <<QwLog::endl;
336  fWire[3].SetSingleEventCuts(errorflag,minX,maxX,stability);
337 
338  }else if (ch_name=="relx"){
339  QwMessage<<"RelX LL " << minX <<" UL " << maxX <<QwLog::endl;
340  fRelPos[0].SetSingleEventCuts(errorflag,minX,maxX,stability);
341 
342  }else if (ch_name=="rely"){
343  QwMessage<<"RelY LL " << minX <<" UL " << maxX <<QwLog::endl;
344  fRelPos[1].SetSingleEventCuts(errorflag,minX,maxX,stability);
345 
346  } else if (ch_name=="absx"){
347  //cuts for the absolute x and y
348  QwMessage<<"AbsX LL " << minX <<" UL " << maxX <<QwLog::endl;
349  fAbsPos[0].SetSingleEventCuts(errorflag,minX,maxX,stability);
350 
351  }else if (ch_name=="absy"){
352  QwMessage<<"AbsY LL " << minX <<" UL " << maxX <<QwLog::endl;
353  fAbsPos[1].SetSingleEventCuts(errorflag,minX,maxX,stability);
354 
355  }else if (ch_name=="effectivecharge"){
356  QwMessage<<"EffectveQ LL " << minX <<" UL " << maxX <<QwLog::endl;
357  fEffectiveCharge.SetSingleEventCuts(errorflag,minX,maxX,stability);
358 
359  }
360 }
361 
362 */
363 
364 template<typename T>
366 {
367  Bool_t localdebug = kFALSE;
368  static T numer("numerator","derived"), denom("denominator","derived");
369  static T tmp1("tmp1","derived"), tmp2("tmp2","derived");
370  static T rawpos[2] = {T("rawpos_0","derived"),T("rawpos_1","derived")};
371 
372  Short_t i = 0;
373 
374  ApplyHWChecks();
375  /**First apply HW checks and update HW error flags.
376  Calling this routine here and not in ApplySingleEventCuts
377  makes a difference for a BPMs because they have derrived devices.
378  */
379 
380  fEffectiveCharge.ClearEventData();
381 
382  for(i=0;i<4;i++)
383  {
384  fWire[i].ProcessEvent();
385  fEffectiveCharge+=fWire[i];
386  }
387 
388 
389  /**
390  To obtain the beam position in X and Y in the CEBAF coordinates, we use the following equations
391 
392  (XP - AlphaX XM)
393  RelX (bpm coordinates) = fQwStriplineCalibration x GainX x ----------------
394  (XP + AlphaX XM)
395 
396  (YP - AplhaY YM)
397  RelY (bpm coordinates) = fQwStriplineCalibration x GainY x ----------------
398  (YP + AlphaY YM)
399 
400  To get back to accelerator coordinates, rotate anti-clockwise around +Z by phi degrees (angle w.r.t X axis).
401 
402  RelX (accelarator coordinates) = cos(phi) RelX - sin(phi)RelY
403 
404  RelY (accelarator coordinates) = sin(phi) RelX + cos(Phi)RelY
405 
406  */
407 
408  for(i=kXAxis;i<kNumAxes;i++)
409  {
410  fWire[i*2+1].Scale(fRelativeGains[i]);
411  numer.Difference(fWire[i*2],fWire[i*2+1]);
412  denom.Sum(fWire[i*2],fWire[i*2+1]);
413  rawpos[i].Ratio(numer,denom);
414  rawpos[i].Scale(fQwStriplineCalibration);
415 
416  if(localdebug)
417  {
418  std::cout<<" stripline name="<<fElementName<<std::endl;
419  // std::cout<<" event number= "<<fWire[i*2].GetSequenceNumber()<<std::endl;
420  std::cout<<" hw Wire["<<i*2<<"]="<<fWire[i*2].GetValue()<<" ";
421  std::cout<<" hw relative gain * Wire["<<i*2+1<<"]="<<fWire[i*2+1].GetValue()<<"\n";
422  std::cout<<" Relative gain["<<i<<"]="<<fRelativeGains[i]<<"\n";
423  std::cout<<" hw numerator= "<<numer.GetValue()<<" ";
424  std::cout<<" hw denominator= "<<denom.GetValue()<<"\n";
425  std::cout<<" Rotation = "<<fRotationAngle<<std::endl;
426  }
427  }
428 
429  for(i=kXAxis;i<kNumAxes;i++){
430  tmp1.AssignScaledValue(rawpos[i], fCosRotation);
431  tmp2.AssignScaledValue(rawpos[1-i], fSinRotation);
432  if (i == kXAxis) {
433  fRelPos[i].Difference(tmp1,tmp2);
434  } else {
435  fRelPos[i].Sum(tmp1,tmp2);
436  }
437  }
438 
439 
440  for(i=kXAxis;i<kNumAxes;i++){
441  fAbsPos[i] = fRelPos[i];
442  fAbsPos[i].AddChannelOffset(fPositionCenter[i]);
443  fAbsPos[i].Scale(1.0/fGains[i]);
444 
445  if(localdebug)
446  {
447  std::cout<<" hw fRelPos["<<kAxisLabel[i]<<"]="<<fRelPos[i].GetValue()<<"\n";
448  std::cout<<" hw fOffset["<<kAxisLabel[i]<<"]="<<fPositionCenter[i]<<"\n";
449  std::cout<<" hw fAbsPos["<<kAxisLabel[i]<<"]="<<fAbsPos[i].GetValue()<<"\n \n";
450  }
451 
452  }
453 
454  return;
455 }
456 
457 
458 template<typename T>
459 Int_t QwBPMStripline<T>::ProcessEvBuffer(UInt_t* buffer, UInt_t word_position_in_buffer,UInt_t index)
460 {
461  if(index<4)
462  {
463  fWire[index].ProcessEvBuffer(buffer,word_position_in_buffer);
464  }
465  else
466  {
467  std::cerr <<
468  "QwBPMStripline::ProcessEvBuffer(): attemp to fill in raw date for a wire that doesn't exist \n";
469  }
470  return word_position_in_buffer;
471 }
472 
473 
474 
475 template<typename T>
477 {
478  for (Short_t i = 0; i < 2; i++) {
479  fAbsPos[i].PrintValue();
480  fRelPos[i].PrintValue();
481  }
482  return;
483 }
484 
485 
486 template<typename T>
488 {
489 
490  QwMessage << "void QwBPMStripline<T>::WritePromptSummary() const test " << QwLog::endl;
491  // for (Short_t i = 0; i < 2; i++) {
492  // fAbsPos[i].PrintValue();
493  // fRelPos[i].PrintValue();
494  // }
495  return;
496 }
497 
498 
499 template<typename T>
501 {
502  Short_t i = 0;
503  for (i = 0; i < 4; i++) fWire[i].PrintInfo();
504  for (i = 0; i < 2; i++) {
505  fRelPos[i].PrintInfo();
506  fAbsPos[i].PrintInfo();
507  }
508  fEffectiveCharge.PrintInfo();
509 }
510 
511 
512 template<typename T>
514 {
515  TString thisname;
516  if(subindex<4&&subindex>-1)
517  thisname=fWire[subindex].GetElementName();
518  else
519  std::cerr<<"QwBPMStripline::GetSubElementName for "<<
520  GetElementName()<<" this subindex doesn't exists \n";
521 
522  return thisname;
523 }
524 
525 template<typename T>
527 {
528  subname.ToUpper();
529  UInt_t localindex = kInvalidSubelementIndex;
530  for(Short_t i=0;i<4;i++) if(subname==subelement[i])localindex=i;
531 
532  if(localindex==kInvalidSubelementIndex)
533  std::cerr << "QwBPMStripline::GetSubElementIndex is unable to associate the string -"
534  <<subname<<"- to any index"<<std::endl;
535  return localindex;
536 }
537 
538 template<typename T>
540 {
541  for(Short_t i=kXAxis;i<kNumAxes;i++){
542  fAbsPos[i]= fRelPos[i];
543  fAbsPos[i].AddChannelOffset(fPositionCenter[i]);
544  }
545  // For Z, the absolute position will be the offset we are reading from the
546  // geometry map file. Since we are not putting that to the tree it is not
547  // treated as a vqwk channel.
548 }
549 
550 
551 template<typename T>
553 {
554  *(dynamic_cast<QwBPMStripline<T>*>(this)) =
555  *(dynamic_cast<const QwBPMStripline<T>*>(&value));
556  return *this;
557 }
558 
559 template<typename T>
561 {
562  VQwBPM::operator= (value);
563 
564  this->bRotated=value.bRotated;
565  if (GetElementName()!=""){
566  Short_t i = 0;
567  this->fEffectiveCharge=value.fEffectiveCharge;
568  for(i=0;i<4;i++) this->fWire[i]=value.fWire[i];
569  for(i=kXAxis;i<kNumAxes;i++) {
570  this->fRelPos[i]=value.fRelPos[i];
571  this->fAbsPos[i]=value.fAbsPos[i];
572  }
573  }
574  return *this;
575 }
576 
577 template<typename T>
579 {
580  *(dynamic_cast<QwBPMStripline<T>*>(this)) +=
581  *(dynamic_cast<const QwBPMStripline<T>*>(&value));
582  return *this;
583 }
584 
585 template<typename T>
587 {
588 
589  if (GetElementName()!=""){
590  Short_t i = 0;
591  this->fEffectiveCharge+=value.fEffectiveCharge;
592  for(i=0;i<4;i++) this->fWire[i]+=value.fWire[i];
593  for(i=kXAxis;i<kNumAxes;i++) {
594  this->fRelPos[i]+=value.fRelPos[i];
595  this->fAbsPos[i]+=value.fAbsPos[i];
596  }
597  }
598  return *this;
599 }
600 
601 template<typename T>
603 {
604  *(dynamic_cast<QwBPMStripline<T>*>(this)) -=
605  *(dynamic_cast<const QwBPMStripline<T>*>(&value));
606  return *this;
607 }
608 template<typename T>
610 {
611 
612  if (GetElementName()!=""){
613  Short_t i = 0;
614  this->fEffectiveCharge-=value.fEffectiveCharge;
615  for(i=0;i<4;i++) this->fWire[i]-=value.fWire[i];
616  for(i=kXAxis;i<kNumAxes;i++) {
617  this->fRelPos[i]-=value.fRelPos[i];
618  this->fAbsPos[i]-=value.fAbsPos[i];
619  }
620  }
621  return *this;
622 }
623 
624 template<typename T>
626 {
627  Ratio(*dynamic_cast<QwBPMStripline<T>*>(&numer),
628  *dynamic_cast<QwBPMStripline<T>*>(&denom));
629 }
630 
631 template<typename T>
633 {
634  // this function is called when forming asymmetries. In this case waht we actually want for the
635  // stripline is the difference only not the asymmetries
636 
637  *this=numer;
638  this->fEffectiveCharge.Ratio(numer.fEffectiveCharge,denom.fEffectiveCharge);
639  return;
640 }
641 
642 
643 
644 template<typename T>
645 void QwBPMStripline<T>::Scale(Double_t factor)
646 {
647  Short_t i = 0;
648  fEffectiveCharge.Scale(factor);
649 
650  for(i=0;i<4;i++) fWire[i].Scale(factor);
651  for(Short_t i=kXAxis;i<kNumAxes;i++){
652  fRelPos[i].Scale(factor);
653  fAbsPos[i].Scale(factor);
654  }
655  return;
656 }
657 
658 
659 template<typename T>
661 {
662  Short_t i = 0;
663  for (i = 0; i < 4; i++){
664  fWire[i].CalculateRunningAverage();
665  }
666 
667  for (i = 0; i < 2; i++){
668  fRelPos[i].CalculateRunningAverage();
669  fAbsPos[i].CalculateRunningAverage();
670  }
671  fEffectiveCharge.CalculateRunningAverage();
672  return;
673 }
674 
675 
676 template<typename T>
678 {
679  AccumulateRunningSum(*dynamic_cast<const QwBPMStripline<T>* >(&value));
680 }
681 
682 template<typename T>
684 {
685  Short_t i = 0;
686  for (i = 0; i < 4; i++){
687  fWire[i].AccumulateRunningSum(value.fWire[i]);
688  }
689  for (i = 0; i < 2; i++){
690  fRelPos[i].AccumulateRunningSum(value.fRelPos[i]);
691  fAbsPos[i].AccumulateRunningSum(value.fAbsPos[i]);
692  }
693  fEffectiveCharge.AccumulateRunningSum(value.fEffectiveCharge);
694  return;
695 }
696 template<typename T>
698  DeaccumulateRunningSum(*dynamic_cast<QwBPMStripline<T>* >(&value));
699 };
700 
701 template<typename T>
703  Short_t i = 0;
704  for (i = 0; i < 4; i++){
705  fWire[i].DeaccumulateRunningSum(value.fWire[i]);
706  }
707  for (i = 0; i < 2; i++){
708  fRelPos[i].DeaccumulateRunningSum(value.fRelPos[i]);
709  fAbsPos[i].DeaccumulateRunningSum(value.fAbsPos[i]);
710  }
711  fEffectiveCharge.DeaccumulateRunningSum(value.fEffectiveCharge); return;
712 
713 };
714 
715 template<typename T>
716 void QwBPMStripline<T>::ConstructHistograms(TDirectory *folder, TString &prefix)
717 {
718 
719  if (GetElementName()=="") {
720  // This channel is not used, so skip filling the histograms.
721  } else {
722  fEffectiveCharge.ConstructHistograms(folder, prefix);
723  TString thisprefix=prefix;
724 
725  if(prefix=="asym_")
726  thisprefix="diff_";
727  this->SetRootSaveStatus(prefix);
728  Short_t i = 0;
729  if(bFullSave) {
730  for(i=0;i<4;i++) fWire[i].ConstructHistograms(folder, thisprefix);
731  }
732  for(i=kXAxis;i<kNumAxes;i++) {
733  fRelPos[i].ConstructHistograms(folder, thisprefix);
734  fAbsPos[i].ConstructHistograms(folder, thisprefix);
735  }
736  }
737  return;
738 }
739 
740 template<typename T>
742 {
743  if (GetElementName()=="") {
744  // This channel is not used, so skip filling the histograms.
745  }
746  else {
747  fEffectiveCharge.FillHistograms();
748  Short_t i = 0;
749  if(bFullSave) {
750  for(i=0;i<4;i++) fWire[i].FillHistograms();
751  }
752  for(i=kXAxis;i<kNumAxes;i++){
753  fRelPos[i].FillHistograms();
754  fAbsPos[i].FillHistograms();
755  }
756  //No data for z position
757  }
758  return;
759 }
760 
761 template<typename T>
762 void QwBPMStripline<T>::ConstructBranchAndVector(TTree *tree, TString &prefix, std::vector<Double_t> &values)
763 {
764  if (GetElementName()==""){
765  // This channel is not used, so skip constructing trees.
766  }
767  else {
768  TString thisprefix=prefix;
769  if(prefix=="asym_")
770  thisprefix="diff_";
771 
772  this->SetRootSaveStatus(prefix);
773 
774  fEffectiveCharge.ConstructBranchAndVector(tree,prefix,values);
775  Short_t i = 0;
776  if(bFullSave) {
777  for(i=0;i<4;i++) fWire[i].ConstructBranchAndVector(tree,thisprefix,values);
778  }
779  for(i=kXAxis;i<kNumAxes;i++) {
780  fRelPos[i].ConstructBranchAndVector(tree,thisprefix,values);
781  fAbsPos[i].ConstructBranchAndVector(tree,thisprefix,values);
782  }
783 
784  }
785  return;
786 }
787 
788 template<typename T>
789 void QwBPMStripline<T>::ConstructBranch(TTree *tree, TString &prefix)
790 {
791  if (GetElementName()==""){
792  // This channel is not used, so skip constructing trees.
793  }
794  else {
795  TString thisprefix=prefix;
796  if(prefix=="asym_")
797  thisprefix="diff_";
798 
799  this->SetRootSaveStatus(prefix);
800 
801  fEffectiveCharge.ConstructBranch(tree,prefix);
802  Short_t i = 0;
803  if(bFullSave) {
804  for(i=0;i<4;i++) fWire[i].ConstructBranch(tree,thisprefix);
805  }
806  for(i=kXAxis;i<kNumAxes;i++) {
807  fRelPos[i].ConstructBranch(tree,thisprefix);
808  fAbsPos[i].ConstructBranch(tree,thisprefix);
809  }
810 
811  }
812  return;
813 }
814 
815 template<typename T>
816 void QwBPMStripline<T>::ConstructBranch(TTree *tree, TString &prefix, QwParameterFile& modulelist)
817 {
818  TString devicename;
819  /*
820  QwMessage <<" QwBCM::ConstructBranch "<<QwLog::endl;
821  modulelist.RewindToFileStart();
822  while (modulelist.ReadNextLine()){
823  modulelist.TrimComment('!'); // Remove everything after a '!' character.
824  modulelist.TrimWhitespace(); // Get rid of leading and trailing spaces.
825  QwMessage <<" "<<modulelist.GetLine()<<" ";
826  }
827  QwMessage <<QwLog::endl;
828  */
829  devicename=GetElementName();
830  devicename.ToLower();
831  if (GetElementName()==""){
832  // This channel is not used, so skip filling the histograms.
833  } else
834  {
835  if (modulelist.HasValue(devicename)){
836  TString thisprefix=prefix;
837  if(prefix=="asym_")
838  thisprefix="diff_";
839 
840  this->SetRootSaveStatus(prefix);
841 
842  fEffectiveCharge.ConstructBranch(tree,prefix);
843  Short_t i = 0;
844  if(bFullSave) {
845  for(i=0;i<4;i++) fWire[i].ConstructBranch(tree,thisprefix);
846  }
847  for(i=kXAxis;i<kNumAxes;i++) {
848  fRelPos[i].ConstructBranch(tree,thisprefix);
849  fAbsPos[i].ConstructBranch(tree,thisprefix);
850  }
851 
852  QwMessage <<" Tree leaves added to "<<devicename<<" Corresponding channels"<<QwLog::endl;
853  }
854  // this functions doesn't do anything yet
855  }
856 
857 
858 
859 
860 
861  return;
862 }
863 
864 template<typename T>
865 void QwBPMStripline<T>::FillTreeVector(std::vector<Double_t> &values) const
866 {
867  if (GetElementName()=="") {
868  // This channel is not used, so skip filling the tree.
869  }
870  else {
871  fEffectiveCharge.FillTreeVector(values);
872  Short_t i = 0;
873  if(bFullSave) {
874  for(i=0;i<4;i++) fWire[i].FillTreeVector(values);
875  }
876  for(i=kXAxis;i<kNumAxes;i++){
877  fRelPos[i].FillTreeVector(values);
878  fAbsPos[i].FillTreeVector(values);
879  }
880  }
881  return;
882 }
883 
884 template<typename T>
886 {
887  Short_t i = 0;
888  // bEVENTCUTMODE=bcuts;
889  for (i=0;i<4;i++) fWire[i].SetEventCutMode(bcuts);
890  for (i=kXAxis;i<kNumAxes;i++) {
891  fRelPos[i].SetEventCutMode(bcuts);
892  fAbsPos[i].SetEventCutMode(bcuts);
893  }
894  fEffectiveCharge.SetEventCutMode(bcuts);
895 }
896 
897 
898 template<typename T>
900 {
901  for(size_t i=kXAxis;i<kNumAxes;i++) {
902  T relpos(fRelPos[i]);
903  relpos = fRelPos[i]; // data
904  fBPMElementList.push_back(relpos);
905  T abspos(fAbsPos[i]);
906  abspos = fAbsPos[i]; // data
907  fBPMElementList.push_back(abspos);
908  }
909  T bpm_sub_element(fEffectiveCharge);
910  bpm_sub_element = fEffectiveCharge;
911  fBPMElementList.push_back(bpm_sub_element);
912 }
913 
914 
915 template<typename T>
916 std::vector<QwDBInterface> QwBPMStripline<T>::GetDBEntry()
917 {
918  std::vector <QwDBInterface> row_list;
919  row_list.clear();
920 
921  for(size_t i=0;i<2;i++) {
922  fRelPos[i].AddEntriesToList(row_list);
923  fAbsPos[i].AddEntriesToList(row_list);
924  }
925  fEffectiveCharge.AddEntriesToList(row_list);
926  return row_list;
927 }
928 
929 
930 template<typename T>
931 std::vector<QwErrDBInterface> QwBPMStripline<T>::GetErrDBEntry()
932 {
933  std::vector <QwErrDBInterface> row_list;
934  row_list.clear();
935 
936  for(size_t i=0;i<2;i++) {
937  fRelPos[i].AddErrEntriesToList(row_list);
938  fAbsPos[i].AddErrEntriesToList(row_list);
939  }
940  fEffectiveCharge.AddErrEntriesToList(row_list);
941  return row_list;
942 }
943 
944 
945 
946 /**********************************
947  * Mock data generation routines
948  **********************************/
949 
950 template<typename T>
951 void QwBPMStripline<T>::SetRandomEventParameters(Double_t meanX, Double_t sigmaX, Double_t meanY, Double_t sigmaY)
952 {
953  // Average values of the signals in the stripline ADCs
954  Double_t sumX = 1.1e8; // These are just guesses, but I made X and Y different
955  Double_t sumY = 0.9e8; // to make it more interesting for the analyzer...
956 
957  // Rotate the requested position if necessary (this is not tested yet)
958  if (bRotated) {
959  Double_t rotated_meanX = (meanX*fCosRotation - meanY*fSinRotation);// / fRotationCorrection;
960  Double_t rotated_meanY = (meanX*fSinRotation + meanY*fCosRotation);// / fRotationCorrection;
961  meanX = rotated_meanX;
962  meanY = rotated_meanY;
963  }
964 
965  // Determine the asymmetry from the position
966  Double_t meanXP = (1.0 + meanX / fQwStriplineCalibration) * sumX / 2.0;
967  Double_t meanXM = (1.0 - meanX / fQwStriplineCalibration) * sumX / 2.0; // = sumX - meanXP;
968  Double_t meanYP = (1.0 + meanY / fQwStriplineCalibration) * sumY / 2.0;
969  Double_t meanYM = (1.0 - meanY / fQwStriplineCalibration) * sumY / 2.0; // = sumY - meanYP;
970 
971  // Determine the spread of the asymmetry (this is not tested yet)
972  // (negative sigma should work in the QwVQWK_Channel, but still using fabs)
973  Double_t sigmaXP = fabs(sumX * sigmaX / meanX);
974  Double_t sigmaXM = sigmaXP;
975  Double_t sigmaYP = fabs(sumY * sigmaY / meanY);
976  Double_t sigmaYM = sigmaYP;
977 
978  // Propagate these parameters to the ADCs
979  fWire[0].SetRandomEventParameters(meanXP, sigmaXP);
980  fWire[1].SetRandomEventParameters(meanXM, sigmaXM);
981  fWire[2].SetRandomEventParameters(meanYP, sigmaYP);
982  fWire[3].SetRandomEventParameters(meanYM, sigmaYM);
983 }
984 
985 
986 template<typename T>
987 void QwBPMStripline<T>::RandomizeEventData(int helicity, double time)
988 {
989  for (Short_t i=0; i<4; i++) fWire[i].RandomizeEventData(helicity, time);
990 
991  return;
992 }
993 
994 
995 template<typename T>
996 void QwBPMStripline<T>::SetEventData(Double_t* relpos, UInt_t sequencenumber)
997 {
998  for (Short_t i=0; i<2; i++)
999  {
1000  //fRelPos[i].SetHardwareSum(relpos[i], sequencenumber);
1001  }
1002 
1003  return;
1004 }
1005 
1006 
1007 template<typename T>
1008 void QwBPMStripline<T>::EncodeEventData(std::vector<UInt_t> &buffer)
1009 {
1010  for (Short_t i=0; i<4; i++) fWire[i].EncodeEventData(buffer);
1011 }
1012 
1013 
1014 template<typename T>
1016 {
1017  for(Short_t i=0;i<4;i++) fWire[i].SetDefaultSampleSize((size_t)sample_size);
1018  return;
1019 }
1020 
1021 
1022 template<typename T>
1023 void QwBPMStripline<T>::SetSubElementPedestal(Int_t j, Double_t value)
1024 {
1025  fWire[j].SetPedestal(value);
1026  return;
1027 }
1028 
1029 template<typename T>
1031 {
1032  fWire[j].SetCalibrationFactor(value);
1033  return;
1034 }
1035 
1036 
1037 template class QwBPMStripline<QwVQWK_Channel>;
1038 template class QwBPMStripline<QwSIS3801_Channel>;
#define QwMessage
Predefined log drain for regular messages.
Definition: QwLog.h:50
Bool_t ApplySingleEventCuts()
void AccumulateRunningSum(const QwBPMStripline &value)
VQwBPM & operator=(const VQwBPM &value)
void CalculateRunningAverage()
void SetSubElementCalibrationFactor(Int_t j, Double_t value)
void AccumulateRunningSum(const QwEnergyCalculator &value)
void FillHistograms()
Fill the histograms for this data element.
void ConstructBranch(TTree *tree, TString &prefix)
void SetRootSaveStatus(TString &prefix)
void ConstructHistograms(TDirectory *folder, TString &prefix)
Construct the histograms for this data element.
void IncrementErrorCounters()
void InitializeChannel(TString name, TString datatosave)
void Ratio(QwEnergyCalculator &numer, QwEnergyCalculator &denom)
std::vector< QwDBInterface > GetDBEntry()
Bool_t bRotated
Definition: VQwBPM.h:263
void SetSubElementPedestal(Int_t j, Double_t value)
void Ratio(VQwBPM &numer, VQwBPM &denom)
static const double e
Definition: QwUnits.h:91
UInt_t UpdateErrorFlag()
Update the error flag based on the error flags of internally contained objects Return paramter is the...
void FillHistograms()
Fill the histograms for this data element.
void DeaccumulateRunningSum(VQwBPM &value)
static const UInt_t kInvalidSubelementIndex
Definition: QwTypes.h:191
void PrintErrorCounters() const
report number of events failed due to HW and event cut failure
void PrintInfo() const
Print multiple lines of information about this data element.
void DeaccumulateRunningSum(QwEnergyCalculator &value)
void ClearEventData()
Clear the event data in this element.
void GetAbsolutePosition()
Bool_t HasValue(TString &vname)
void PrintErrorCounters() const
report number of events failed due to HW and event cut failure
void ConstructHistograms(TDirectory *folder, TString &prefix)
Construct the histograms for this data element.
void PrintInfo(TStopwatch &timer)
void RandomizeEventData(int helicity=0, double time=0.0)
void Scale(Double_t factor)
void SetRandomEventParameters(Double_t meanX, Double_t sigmaX, Double_t meanY, Double_t sigmaY)
static const double T
Magnetic field: base unit is T.
Definition: QwUnits.h:111
void InitializeChannel(TString name)
Definition: VQwBPM.cc:25
void PrintValue() const
Print single line of value and error of this data element.
Bool_t ApplyHWChecks()
void Scale(Double_t factor)
void ConstructBranchAndVector(TTree *tree, TString &prefix, std::vector< Double_t > &values)
UInt_t GetEventcutErrorFlag()
return the error flag on this channel/device
void SetEventCutMode(Int_t bcuts)
Int_t ProcessEvBuffer(UInt_t *buffer, UInt_t word_position_in_buffer, UInt_t indexnumber)
Process the CODA event buffer for this element.
void FillTreeVector(std::vector< Double_t > &values) const
std::vector< QwErrDBInterface > GetErrDBEntry()
VQwBPM & operator-=(const VQwBPM &value)
static std::ostream & endl(std::ostream &)
End of the line.
Definition: QwLog.cc:299
Definition: VQwBPM.h:34
void SetModuleType(TString ModuleType)
set the type of the beam instrument
void ConstructBranch(TTree *tree, TString &prefix)
TString fElementName
Name of this data element.
static UInt_t GetSubElementIndex(TString subname)
virtual const TString & GetElementName() const
Get the name of this element.
void SetEventCutMode(Int_t bcuts)
void EncodeEventData(std::vector< UInt_t > &buffer)
void FillTreeVector(std::vector< Double_t > &values) const
void ClearEventData()
Clear the event data in this element.
virtual VQwBPM & operator=(const VQwBPM &value)=0
Definition: VQwBPM.cc:115
UInt_t GetEventcutErrorFlag()
return the error flag on this channel/device
void InitializeChannel(TString name)
void ConstructBranchAndVector(TTree *tree, TString &prefix, std::vector< Double_t > &values)
TString GetSubElementName(Int_t subindex)
void SetDefaultSampleSize(Int_t sample_size)
VQwBPM & operator+=(const VQwBPM &value)
void WritePromptSummary(QwPromptSummary *ps, TString type)
void SetEventData(Double_t *block, UInt_t sequencenumber)
VQwHardwareChannel * GetSubelementByName(TString ch_name)