source: trunk/FACT++/src/ftmctrl.cc@ 16774

Last change on this file since 16774 was 16771, checked in by tbretz, 12 years ago
When compiling a new static buffer to be sent to the FTM, not the last received buffer has to be used but the last sent buffer, this is because a new configuration might be in the queue already, but we have not yet received the answer. To avoid race conditions, the sent buffer is now filled at connection time and used to update the configuration; also a new state was introduced to signal (for debugging mainly) whether the last sent and the last received static data are different (this does not include the thresholds); for convenience in the ratecontrol: added a SET_ALL_THRESHOLDS
File size: 99.6 KB
Line 
1#include <array>
2
3#include "Dim.h"
4#include "Event.h"
5#include "Shell.h"
6#include "StateMachineDim.h"
7#include "StateMachineAsio.h"
8#include "Connection.h"
9#include "LocalControl.h"
10#include "Configuration.h"
11#include "Console.h"
12#include "Converter.h"
13
14#include "tools.h"
15
16#include "HeadersFTM.h"
17
18
19namespace ba = boost::asio;
20namespace bs = boost::system;
21
22using namespace std;
23
24// ------------------------------------------------------------------------
25
26class ConnectionFTM : public Connection
27{
28public:
29 enum States
30 {
31 // State Machine states
32 kDisconnected = StateMachineImp::kSM_UserMode,
33 kConnected,
34 kIdle,
35 kConfigured, // Returned if idle and fBufStaticData==fStaticData
36 kTriggerOn,
37 };
38
39private:
40 vector<uint16_t> fBuffer;
41
42 bool fHasHeader;
43
44 bool fIsVerbose;
45 bool fIsDynamicOut;
46 bool fIsHexOutput;
47
48protected:
49 map<uint16_t, uint32_t> fCounter;
50
51 FTM::Header fHeader;
52 FTM::FtuList fFtuList;
53 FTM::StaticData fStaticData; // fStaticBufferTx
54 FTM::DynamicData fDynamicData;
55 FTM::Error fError;
56
57 FTM::StaticData fBufStaticData; // fStaticBufferRx
58
59 virtual void UpdateFirstHeader()
60 {
61 // FIXME: Message() ?
62 Out() << endl << kBold << "First header received:" << endl;
63 Out() << fHeader;
64 if (fIsHexOutput)
65 Out() << Converter::GetHex<uint16_t>(fHeader, 16) << endl;
66 }
67
68 virtual void UpdateHeader()
69 {
70 // emit service with trigger counter from header
71 if (!fIsVerbose)
72 return;
73
74 if (fHeader.fType==FTM::kDynamicData && !fIsDynamicOut)
75 return;
76
77 Out() << endl << kBold << "Header received:" << endl;
78 Out() << fHeader;
79 if (fIsHexOutput)
80 Out() << Converter::GetHex<uint16_t>(fHeader, 16) << endl;
81 }
82
83 virtual void UpdateFtuList()
84 {
85 if (!fIsVerbose)
86 return;
87
88 Out() << endl << kBold << "FtuList received:" << endl;
89 Out() << fFtuList;
90 if (fIsHexOutput)
91 Out() << Converter::GetHex<uint16_t>(fFtuList, 16) << endl;
92 }
93
94 virtual void UpdateStaticData()
95 {
96 if (!fIsVerbose)
97 return;
98
99 Out() << endl << kBold << "Static data received:" << endl;
100 Out() << fStaticData;
101 if (fIsHexOutput)
102 Out() << Converter::GetHex<uint16_t>(fStaticData, 16) << endl;
103 }
104
105 virtual void UpdateDynamicData()
106 {
107 if (!fIsDynamicOut)
108 return;
109
110 Out() << endl << kBold << "Dynamic data received:" << endl;
111 Out() << fDynamicData;
112 if (fIsHexOutput)
113 Out() << Converter::GetHex<uint16_t>(fDynamicData, 16) << endl;
114 }
115
116 virtual void UpdateError()
117 {
118 if (!fIsVerbose)
119 return;
120
121 Out() << endl << kRed << "Error received:" << endl;
122 Out() << fError;
123 if (fIsHexOutput)
124 Out() << Converter::GetHex<uint16_t>(fError, 16) << endl;
125 }
126
127 virtual void UpdateCounter()
128 {
129 if (!fIsVerbose)
130 return;
131
132 if (!fIsDynamicOut)
133 return;
134
135 Out() << "Received: ";
136 Out() << "H=" << fCounter[FTM::kHeader] << " ";
137 Out() << "S=" << fCounter[FTM::kStaticData] << " ";
138 Out() << "D=" << fCounter[FTM::kDynamicData] << " ";
139 Out() << "F=" << fCounter[FTM::kFtuList] << " ";
140 Out() << "E=" << fCounter[FTM::kErrorList] << " ";
141 Out() << "R=" << fCounter[FTM::kRegister] << endl;
142 }
143
144 bool CheckConsistency(FTM::StaticData &data)
145 {
146 bool warn1 = false;
147 if (data.IsEnabled(FTM::StaticData::kPedestal) != (data.GetSequencePed() >0) ||
148 data.IsEnabled(FTM::StaticData::kLPint) != (data.GetSequenceLPint()>0) ||
149 data.IsEnabled(FTM::StaticData::kLPext) != (data.GetSequenceLPext()>0))
150 {
151 warn1 = true;
152 data.Enable(FTM::StaticData::kPedestal, data.GetSequencePed()>0);
153 data.Enable(FTM::StaticData::kLPint, data.GetSequenceLPint()>0);
154 data.Enable(FTM::StaticData::kLPext, data.GetSequenceLPext()>0);
155 }
156
157 bool warn2 = false;
158 const uint16_t ref = data[0].fPrescaling;
159 for (int i=1; i<40; i++)
160 {
161 if (data[i].fPrescaling != ref)
162 {
163 warn2 = true;
164 data[i].fPrescaling = ref;
165 }
166 }
167
168 bool warn3 = false;
169 for (int i=0; i<4; i++)
170 if (data.fActiveFTU[i]!=0x3ff)
171 {
172 warn3 = true;
173 data.fActiveFTU[i]=0x3ff;
174 }
175
176
177
178 if (warn1)
179 Warn("GeneralSettings not consistent with trigger sequence.");
180 if (warn2)
181 Warn("Prescaling not consistent for all boards.");
182 if (warn3)
183 Warn("Not all FTUs are enabled - enable all FTUs.");
184
185 return !warn1 && !warn2 && !warn3;
186 }
187
188private:
189 void HandleReceivedData(const bs::error_code& err, size_t bytes_received, int /*type*/)
190 {
191 // Do not schedule a new read if the connection failed.
192 if (bytes_received==0 || err)
193 {
194 if (err==ba::error::eof)
195 Warn("Connection closed by remote host (FTM).");
196
197 // 107: Transport endpoint is not connected (bs::error_code(107, bs::system_category))
198 // 125: Operation canceled
199 if (err && err!=ba::error::eof && // Connection closed by remote host
200 err!=ba::error::basic_errors::not_connected && // Connection closed by remote host
201 err!=ba::error::basic_errors::operation_aborted) // Connection closed by us
202 {
203 ostringstream str;
204 str << "Reading from " << URL() << ": " << err.message() << " (" << err << ")";// << endl;
205 Error(str);
206 }
207 PostClose(err!=ba::error::basic_errors::operation_aborted);
208 return;
209 }
210
211 // If we have not yet received a header we expect one now
212 // This could be moved to a HandleReceivedHeader function
213 if (!fHasHeader)
214 {
215 if (bytes_received!=sizeof(FTM::Header))
216 {
217 ostringstream str;
218 str << "Excepted " << sizeof(FTM::Header) << " bytes (FTM::Header) but received " << bytes_received << ".";
219 Error(str);
220 PostClose(false);
221 return;
222 }
223
224 fHeader = fBuffer;
225
226 // Check the data integrity
227 if (fHeader.fDelimiter!=FTM::kDelimiterStart)
228 {
229 ostringstream str;
230 str << "Invalid header received: start delimiter wrong, received ";
231 str << hex << fHeader.fDelimiter << ", expected " << FTM::kDelimiterStart << ".";
232 Error(str);
233 PostClose(false);
234 return;
235 }
236
237 fHasHeader = true;
238
239 // Convert FTM state into FtmCtrl state
240 if (++fCounter[FTM::kHeader]==1)
241 UpdateFirstHeader();
242
243 UpdateCounter();
244 UpdateHeader();
245
246 // Start reading of data
247 switch (fHeader.fType)
248 {
249 case FTM::kStaticData:
250 case FTM::kDynamicData:
251 case FTM::kFtuList:
252 case FTM::kRegister:
253 case FTM::kErrorList:
254 // This is not very efficient because the space is reallocated
255 // maybe we can check if the capacity of the std::vector
256 // is ever decreased. If not, everythign is fine.
257 fBuffer.resize(fHeader.fDataSize);
258 AsyncRead(ba::buffer(fBuffer));
259 AsyncWait(fInTimeout, 50, &Connection::HandleReadTimeout);
260 return;
261
262 default:
263 ostringstream str;
264 str << "Unknonw type " << fHeader.fType << " in received header." << endl;
265 Error(str);
266 PostClose(false);
267 return;
268 }
269
270 return;
271 }
272
273 // Check the data integrity (check end delimiter)
274 if (ntohs(fBuffer.back())!=FTM::kDelimiterEnd)
275 {
276 ostringstream str;
277 str << "Invalid data received: end delimiter wrong, received ";
278 str << hex << ntohs(fBuffer.back()) << ", expected " << FTM::kDelimiterEnd << ".";
279 Error(str);
280 PostClose(false);
281 return;
282 }
283
284 // Remove end delimiter
285 fBuffer.pop_back();
286
287 try
288 {
289 // If we have already received a header this is the data now
290 // This could be moved to a HandleReceivedData function
291
292 fCounter[fHeader.fType]++;
293 UpdateCounter();
294
295 switch (fHeader.fType)
296 {
297 case FTM::kFtuList:
298 fFtuList = fBuffer;
299 UpdateFtuList();
300 break;
301
302 case FTM::kStaticData:
303 if (fCounter[FTM::kStaticData]==1)
304 {
305 // This check is only done at startup
306 FTM::StaticData data(fBuffer);
307 if (!CheckConsistency(data))
308 {
309 CmdSendStatDat(data);
310 CmdPing(); // FIXME: Only needed in case of warn3
311 break;
312 }
313 }
314
315 fStaticData = fBuffer;
316
317 // is this the first received static data block?
318 if (!fBufStaticData.valid())
319 fBufStaticData = fStaticData;
320
321 UpdateStaticData();
322 break;
323
324 case FTM::kDynamicData:
325 fDynamicData = fBuffer;
326 UpdateDynamicData();
327 break;
328
329 case FTM::kRegister:
330 if (fIsVerbose)
331 {
332 Out() << endl << kBold << "Register received: " << endl;
333 Out() << "Addr: " << ntohs(fBuffer[0]) << endl;
334 Out() << "Value: " << ntohs(fBuffer[1]) << endl;
335 }
336 break;
337
338 case FTM::kErrorList:
339 fError = fBuffer;
340 UpdateError();
341 break;
342
343 default:
344 ostringstream str;
345 str << "Unknonw type " << fHeader.fType << " in header." << endl;
346 Error(str);
347 PostClose(false);
348 return;
349 }
350 }
351 catch (const logic_error &e)
352 {
353 ostringstream str;
354 str << "Exception converting buffer into data structure: " << e.what();
355 Error(str);
356 PostClose(false);
357 return;
358 }
359
360 fInTimeout.cancel();
361
362 //fHeader.clear();
363 fHasHeader = false;
364 fBuffer.resize(sizeof(FTM::Header)/2);
365 AsyncRead(ba::buffer(fBuffer));
366 }
367
368 // This is called when a connection was established
369 void ConnectionEstablished()
370 {
371 fCounter.clear();
372 fBufStaticData.clear();
373
374 fHeader.clear();
375 fHasHeader = false;
376 fBuffer.resize(sizeof(FTM::Header)/2);
377 AsyncRead(ba::buffer(fBuffer));
378
379// if (!fDefaultSetup.empty())
380// LoadStaticData(fDefaultSetup);
381
382 // Get a header and configdata!
383 CmdReqStatDat();
384
385 // get the DNA of the FTUs
386 CmdPing();
387 }
388
389 void HandleReadTimeout(const bs::error_code &error)
390 {
391 if (error==ba::error::basic_errors::operation_aborted)
392 return;
393
394 if (error)
395 {
396 ostringstream str;
397 str << "Read timeout of " << URL() << ": " << error.message() << " (" << error << ")";// << endl;
398 Error(str);
399
400 PostClose();
401 return;
402
403 }
404
405 if (!is_open())
406 {
407 // For example: Here we could schedule a new accept if we
408 // would not want to allow two connections at the same time.
409 return;
410 }
411
412 // Check whether the deadline has passed. We compare the deadline
413 // against the current time since a new asynchronous operation
414 // may have moved the deadline before this actor had a chance
415 // to run.
416 if (fInTimeout.expires_at() > ba::deadline_timer::traits_type::now())
417 return;
418
419 Error("Timeout reading data from "+URL());
420
421 PostClose();
422 }
423
424
425 template<size_t N>
426 void PostCmd(array<uint16_t, N> dat, uint16_t u1=0, uint16_t u2=0, uint16_t u3=0, uint16_t u4=0)
427 {
428 array<uint16_t, 5> cmd = {{ '@', u1, u2, u3, u4 }};
429
430 if (fIsVerbose)
431 {
432 ostringstream msg;
433 msg << "Sending command:" << hex;
434 msg << " 0x" << setw(4) << setfill('0') << cmd[0];
435 msg << " 0x" << setw(4) << setfill('0') << u1;
436 msg << " 0x" << setw(4) << setfill('0') << u2;
437 msg << " 0x" << setw(4) << setfill('0') << u3;
438 msg << " 0x" << setw(4) << setfill('0') << u4;
439 msg << " (+" << dec << dat.size() << " words)";
440 Message(msg);
441 }
442
443 vector<uint16_t> out(cmd.size()+dat.size());
444
445 transform(cmd.begin(), cmd.end(), out.begin(), htons);
446 transform(dat.begin(), dat.end(), out.begin()+cmd.size(), htons);
447
448 PostMessage(out);
449 }
450
451 void PostCmd(vector<uint16_t> dat, uint16_t u1=0, uint16_t u2=0, uint16_t u3=0, uint16_t u4=0)
452 {
453 array<uint16_t, 5> cmd = {{ '@', u1, u2, u3, u4 }};
454
455 if (fIsVerbose)
456 {
457 ostringstream msg;
458 msg << "Sending command:" << hex;
459 msg << " 0x" << setw(4) << setfill('0') << cmd[0];
460 msg << " 0x" << setw(4) << setfill('0') << u1;
461 msg << " 0x" << setw(4) << setfill('0') << u2;
462 msg << " 0x" << setw(4) << setfill('0') << u3;
463 msg << " 0x" << setw(4) << setfill('0') << u4;
464 msg << " (+" << dec << dat.size() << " words)";
465 Message(msg);
466 }
467
468 vector<uint16_t> out(cmd.size()+dat.size());
469
470 transform(cmd.begin(), cmd.end(), out.begin(), htons);
471 copy(dat.begin(), dat.end(), out.begin()+cmd.size());
472
473 PostMessage(out);
474 }
475
476 void PostCmd(uint16_t u1=0, uint16_t u2=0, uint16_t u3=0, uint16_t u4=0)
477 {
478 PostCmd(array<uint16_t, 0>(), u1, u2, u3, u4);
479 }
480public:
481
482// static const uint16_t kMaxAddr;
483
484public:
485 ConnectionFTM(ba::io_service& ioservice, MessageImp &imp) : Connection(ioservice, imp()),
486 fIsVerbose(true), fIsDynamicOut(true), fIsHexOutput(true)
487 {
488 SetLogStream(&imp);
489 }
490
491 void CmdToggleLed()
492 {
493 PostCmd(FTM::kCmdToggleLed);
494 }
495
496 void CmdPing()
497 {
498 PostCmd(FTM::kCmdPing);
499 }
500
501 void CmdReqDynDat()
502 {
503 PostCmd(FTM::kCmdRead, FTM::kCmdDynamicData);
504 }
505
506 void CmdReqStatDat()
507 {
508 PostCmd(FTM::kCmdRead, FTM::kCmdStaticData);
509 }
510
511 void CmdSendStatDat(const FTM::StaticData &data)
512 {
513 fBufStaticData = data;
514
515 PostCmd(data.HtoN(), FTM::kCmdWrite, FTM::kCmdStaticData);
516
517 // Request the changed configuration to ensure the
518 // change is distributed in the network
519 CmdReqStatDat();
520 }
521
522 void CmdStartRun()
523 {
524 PostCmd(FTM::kCmdStartRun, FTM::kStartRun);
525 CmdGetRegister(0);
526
527 Info("Sending start trigger.");
528 }
529
530 void CmdStopRun()
531 {
532 PostCmd(FTM::kCmdStopRun);
533 CmdGetRegister(0);
534
535 Info("Sending stop trigger.");
536 }
537
538 void CmdTakeNevents(uint32_t n)
539 {
540 const array<uint16_t, 2> data = {{ uint16_t(n>>16), uint16_t(n&0xffff) }};
541 PostCmd(data, FTM::kCmdStartRun, FTM::kTakeNevents);
542
543 // Update state information by requesting a new header
544 CmdGetRegister(0);
545 }
546
547 bool CmdSetRegister(uint16_t addr, uint16_t val)
548 {
549 if (addr>FTM::StaticData::kMaxAddr)
550 return false;
551
552 const array<uint16_t, 2> data = {{ addr, val }};
553 PostCmd(data, FTM::kCmdWrite, FTM::kCmdRegister);
554
555 reinterpret_cast<uint16_t*>(&fBufStaticData)[addr] = val;
556
557 // Request the changed configuration to ensure the
558 // change is distributed in the network
559 CmdReqStatDat();
560
561 return true;
562 }
563
564 bool CmdGetRegister(uint16_t addr)
565 {
566 if (addr>FTM::StaticData::kMaxAddr)
567 return false;
568
569 const array<uint16_t, 1> data = {{ addr }};
570 PostCmd(data, FTM::kCmdRead, FTM::kCmdRegister);
571
572 return true;
573 }
574
575 bool CmdResetCrate(uint16_t addr)
576 {
577 if (addr>3)
578 return false;
579
580 PostCmd(FTM::kCmdCrateReset, 1<<addr);
581 Info("Sending crate reset for crate "+to_string(addr));
582
583 return true;
584 }
585
586 bool CmdResetCamera()
587 {
588 PostCmd(FTM::kCmdCrateReset, FTM::kResetCrate0);
589 PostCmd(FTM::kCmdCrateReset, FTM::kResetCrate1);
590 PostCmd(FTM::kCmdCrateReset, FTM::kResetCrate2);
591 PostCmd(FTM::kCmdCrateReset, FTM::kResetCrate3);
592
593 Info("Sending camera reset");
594
595 return true;
596 }
597
598 bool CmdDisableReports(bool b)
599 {
600 PostCmd(FTM::kCmdDisableReports, b ? uint16_t(0) : uint16_t(1));
601 return true;
602 }
603
604
605 void SetVerbose(bool b)
606 {
607 fIsVerbose = b;
608 }
609
610 void SetHexOutput(bool b)
611 {
612 fIsHexOutput = b;
613 }
614
615 void SetDynamicOut(bool b)
616 {
617 fIsDynamicOut = b;
618 }
619/*
620 void SetDefaultSetup(const string &file)
621 {
622 fDefaultSetup = file;
623 }
624*/
625
626 bool LoadStaticData(string name)
627 {
628 if (name.rfind(".bin")!=name.length()-4)
629 name += ".bin";
630
631 ifstream fin(name);
632 if (!fin)
633 return false;
634
635 FTM::StaticData data;
636
637 fin.read(reinterpret_cast<char*>(&data), sizeof(FTM::StaticData));
638
639 if (fin.gcount()<streamsize(sizeof(FTM::StaticData)))
640 return false;
641
642 if (fin.fail() || fin.eof())
643 return false;
644
645 if (fin.peek()!=-1)
646 return false;
647
648 CmdSendStatDat(data);
649
650 return true;
651 }
652
653 bool SaveStaticData(string name) const
654 {
655 if (name.rfind(".bin")!=name.length()-4)
656 name += ".bin";
657
658 ofstream fout(name);
659 if (!fout)
660 return false;
661
662 fout.write(reinterpret_cast<const char*>(&fStaticData), sizeof(FTM::StaticData));
663
664 return !fout.bad();
665 }
666
667 bool SetThreshold(int32_t patch, int32_t value)
668 {
669 if (patch>FTM::StaticData::kMaxPatchIdx)
670 return false;
671
672 if (value<0 || value>FTM::StaticData::kMaxDAC)
673 return false;
674
675 if (patch<0)
676 {
677 FTM::StaticData data(fBufStaticData);
678
679 bool ident = true;
680 for (int i=0; i<=FTM::StaticData::kMaxPatchIdx; i++)
681 if (data[i/4].fDAC[i%4] != value)
682 {
683 ident = false;
684 break;
685 }
686
687 if (ident)
688 return true;
689
690 for (int i=0; i<=FTM::StaticData::kMaxPatchIdx; i++)
691 data[i/4].fDAC[i%4] = value;
692
693 // Maybe move to a "COMMIT" command?
694 CmdSendStatDat(data);
695
696 return true;
697 }
698
699 /*
700 if (data[patch/4].fDAC[patch%4] == value)
701 return true;
702 */
703
704 // Calculate offset in static data block
705 const uint16_t addr = (uintptr_t(&fStaticData[patch/4].fDAC[patch%4])-uintptr_t(&fStaticData))/2;
706
707 // From CmdSetRegister
708 const array<uint16_t, 2> data = {{ addr, uint16_t(value) }};
709 PostCmd(data, FTM::kCmdWrite, FTM::kCmdRegister);
710
711 reinterpret_cast<uint16_t*>(&fBufStaticData)[addr] = value;
712
713 // Now execute change before the static data is requested back
714 PostCmd(FTM::kCmdConfigFTU, (patch/40) | (((patch/4)%10)<<8));
715
716 //CmdGetRegister(addr);
717 CmdReqStatDat();
718
719 return true;
720 }
721 /*
722 bool SetSelectedThresholds(const int32_t *th)
723 {
724 for (int i=0; i<FTM::StaticData::kMaxPatchIdx; i++)
725 if (th[i]<0 || th[i]>FTM::StaticData::kMaxDAC)
726 return false;
727
728 FTM::StaticData data(fBufStaticData);
729
730 for (int i=0; i<=FTM::StaticData::kMaxPatchIdx; i++)
731 {
732 if (fBufStaticData[i/4].fDac[i%4]!=th[i])
733 {
734
735 // Calculate offset in static data block
736 const uint16_t addr = (uintptr_t(&fStaticData[i/4].fDAC[i%4])-uintptr_t(&fStaticData))/2;
737
738 // From CmdSetRegister
739 const array<uint16_t, 2> data = {{ addr, uint16_t(th[i]) }};
740 PostCmd(data, FTM::kCmdWrite, FTM::kCmdRegister);
741
742 reinterpret_cast<uint16_t*>(&fBufStaticData)[addr] = th[i];
743 }
744 }
745
746 // Now execute change before the static data is requested back
747 PostCmd(FTM::kCmdConfigFTU, (patch/40) | (((patch/4)%10)<<8));
748
749 //CmdGetRegister(addr);
750 CmdReqStatDat();
751
752 return true;
753 }*/
754
755
756 bool SetAllThresholds(const int32_t *th)
757 {
758 for (int i=0; i<FTM::StaticData::kMaxPatchIdx; i++)
759 if (th[i]<0 || th[i]>FTM::StaticData::kMaxDAC)
760 return false;
761
762 FTM::StaticData data(fBufStaticData);
763
764 for (int i=0; i<=FTM::StaticData::kMaxPatchIdx; i++)
765 data[i/4].fDAC[i%4] = th[i];
766
767 CmdSendStatDat(data);
768
769 return true;
770 }
771
772 bool SetNoutof4(int32_t patch, int32_t value)
773 {
774 if (patch>=FTM::StaticData::kMaxMultiplicity)
775 return false;
776
777 if (value<0 || value>FTM::StaticData::kMaxDAC)
778 return false;
779
780 if (patch<0)
781 {
782 FTM::StaticData data(fBufStaticData);
783
784 bool ident = true;
785 for (int i=0; i<FTM::StaticData::kMaxMultiplicity; i++)
786 if (data[i].fDAC[4] != value)
787 {
788 ident = false;
789 break;
790 }
791
792 if (ident)
793 return true;
794
795 for (int i=0; i<=FTM::StaticData::kMaxMultiplicity; i++)
796 data[i].fDAC[4] = value;
797
798 // Maybe move to a "COMMIT" command?
799 CmdSendStatDat(data);
800
801 return true;
802 }
803
804 /*
805 if (data[patch/4].fDAC[patch%4] == value)
806 return true;
807
808 data[patch/4].fDAC[patch%4] = value;
809
810 CmdSendStatDat(data);
811 return true;
812 */
813
814 // Calculate offset in static data block
815 const uint16_t addr = (uintptr_t(&fStaticData[patch].fDAC[4])-uintptr_t(&fStaticData))/2;
816
817 // From CmdSetRegister
818 const array<uint16_t, 2> data = {{ addr, uint16_t(value) }};
819 PostCmd(data, FTM::kCmdWrite, FTM::kCmdRegister);
820
821 reinterpret_cast<uint16_t*>(&fBufStaticData)[addr] = value;
822
823 // Now execute change before the static data is requested back
824 PostCmd(FTM::kCmdConfigFTU, (patch/40) | (((patch/4)%10)<<8));
825
826 //CmdGetRegister(addr);
827 CmdReqStatDat();
828
829 return true;
830 }
831
832 bool SetPrescaling(uint32_t value)
833 {
834 if (value>0xffff)
835 return false;
836
837 FTM::StaticData data(fBufStaticData);
838
839 bool ident = true;
840 for (int i=0; i<40; i++)
841 if (data[i].fPrescaling != value)
842 {
843 ident = false;
844 break;
845 }
846
847 if (ident)
848 return true;
849
850 data.SetPrescaling(value);
851
852 // Maybe move to a "COMMIT" command?
853 CmdSendStatDat(data);
854
855 return true;
856 }
857
858 bool EnableFTU(int32_t board, bool enable)
859 {
860 if (board>39)
861 return false;
862
863 FTM::StaticData data(fBufStaticData);
864
865 if (board<0)
866 {
867 if (enable)
868 data.EnableAllFTU();
869 else
870 data.DisableAllFTU();
871 }
872 else
873 {
874 if (enable)
875 data.EnableFTU(board);
876 else
877 data.DisableFTU(board);
878
879 }
880
881 // Maybe move to a "COMMIT" command?
882 CmdSendStatDat(data);
883
884 return true;
885 }
886
887 bool ToggleFTU(uint32_t board)
888 {
889 if (board>39)
890 return false;
891
892 FTM::StaticData data(fBufStaticData);
893
894 data.ToggleFTU(board);
895
896 // Maybe move to a "COMMIT" command?
897 CmdSendStatDat(data);
898
899 return true;
900 }
901
902 bool SetVal(uint16_t *dest, uint32_t val, uint32_t max)
903 {
904 if (val>max)
905 return false;
906
907 if (*dest==val)
908 return true;
909
910 FTM::StaticData data(fBufStaticData);
911
912 dest = reinterpret_cast<uint16_t*>(&data) + (dest - reinterpret_cast<uint16_t*>(&fStaticData));
913
914 *dest = val;
915
916 CmdSendStatDat(data);
917
918 return true;
919 }
920
921 bool SetTriggerInterval(uint32_t val)
922 {
923 return SetVal(&fBufStaticData.fTriggerInterval, val,
924 FTM::StaticData::kMaxTriggerInterval);
925 }
926
927 bool SetTriggerDelay(uint32_t val)
928 {
929 return SetVal(&fBufStaticData.fDelayTrigger, val,
930 FTM::StaticData::kMaxDelayTrigger);
931 }
932
933 bool SetTimeMarkerDelay(uint32_t val)
934 {
935 return SetVal(&fBufStaticData.fDelayTimeMarker, val,
936 FTM::StaticData::kMaxDelayTimeMarker);
937 }
938
939 bool SetDeadTime(uint32_t val)
940 {
941 return SetVal(&fBufStaticData.fDeadTime, val,
942 FTM::StaticData::kMaxDeadTime);
943 }
944
945 void Enable(FTM::StaticData::GeneralSettings type, bool enable)
946 {
947 //if (fStaticData.IsEnabled(type)==enable)
948 // return;
949
950 FTM::StaticData data(fBufStaticData);
951 data.Enable(type, enable);
952 CmdSendStatDat(data);
953 }
954
955 bool SetTriggerSeq(const uint16_t d[3])
956 {
957 if (d[0]>FTM::StaticData::kMaxSequence ||
958 d[1]>FTM::StaticData::kMaxSequence ||
959 d[2]>FTM::StaticData::kMaxSequence)
960 return false;
961
962 FTM::StaticData data(fBufStaticData);
963
964 /*
965 data.Enable(FTM::StaticData::kPedestal, d[0]>0);
966 data.Enable(FTM::StaticData::kLPext, d[1]>0);
967 data.Enable(FTM::StaticData::kLPint, d[2]>0);
968 */
969
970 data.SetSequence(d[0], d[2], d[1]);
971
972 //if (fStaticData.fTriggerSeq !=data.fTriggerSequence ||
973 // fStaticData.fGeneralSettings!=data.fGeneralSettings)
974 // CmdSendStatDat(data);
975
976 CmdSendStatDat(data);
977
978 return true;
979 }
980
981 bool SetTriggerMultiplicity(uint16_t n)
982 {
983 if (n==0 || n>FTM::StaticData::kMaxMultiplicity)
984 return false;
985
986 //if (n==fBufStaticData.fMultiplicityPhysics)
987 // return true;
988
989 FTM::StaticData data(fBufStaticData);
990
991 data.fMultiplicityPhysics = n;
992
993 CmdSendStatDat(data);
994
995 return true;
996 }
997
998 bool SetTriggerWindow(uint16_t win)
999 {
1000 if (win>FTM::StaticData::kMaxWindow)
1001 return false;
1002
1003 //if (win==fStaticData.fWindowPhysics)
1004 // return true;
1005
1006 FTM::StaticData data(fBufStaticData);
1007
1008 data.fWindowPhysics = win;
1009
1010 CmdSendStatDat(data);
1011
1012 return true;
1013 }
1014
1015 bool SetCalibMultiplicity(uint16_t n)
1016 {
1017 if (n==0 || n>FTM::StaticData::kMaxMultiplicity)
1018 return false;
1019
1020 //if (n==fStaticData.fMultiplicityCalib)
1021 // return true;
1022
1023 FTM::StaticData data(fBufStaticData);
1024
1025 data.fMultiplicityCalib = n;
1026
1027 CmdSendStatDat(data);
1028
1029 return true;
1030 }
1031
1032 bool SetCalibWindow(uint16_t win)
1033 {
1034 if (win>FTM::StaticData::kMaxWindow)
1035 return false;
1036
1037 //if (win==fStaticData.fWindowCalib)
1038 // return true;
1039
1040 FTM::StaticData data(fBufStaticData);
1041
1042 data.fWindowCalib = win;
1043
1044 CmdSendStatDat(data);
1045
1046 return true;
1047 }
1048
1049 bool SetClockRegister(const uint64_t reg[])
1050 {
1051 FTM::StaticData data(fBufStaticData);
1052
1053 for (int i=0; i<8; i++)
1054 if (reg[i]>0xffffffff)
1055 return false;
1056
1057 data.SetClockRegister(reg);
1058
1059 CmdSendStatDat(data);
1060
1061 return true;
1062 }
1063
1064 bool EnableLP(FTM::StaticData::GeneralSettings lp, FTM::StaticData::LightPulserEnable group, bool enable)
1065 {
1066 if (lp!=FTM::StaticData::kLPint && lp!=FTM::StaticData::kLPext)
1067 return false;
1068
1069 FTM::StaticData data(fBufStaticData);
1070
1071 if (lp==FTM::StaticData::kLPint)
1072 data.EnableLPint(group, enable);
1073
1074 if (lp==FTM::StaticData::kLPext)
1075 data.EnableLPext(group, enable);
1076
1077 CmdSendStatDat(data);
1078
1079 return true;
1080 }
1081
1082 bool SetIntensity(FTM::StaticData::GeneralSettings lp, uint16_t intensity)
1083 {
1084 if (intensity>FTM::StaticData::kMaxIntensity)
1085 return false;
1086
1087 if (lp!=FTM::StaticData::kLPint && lp!=FTM::StaticData::kLPext)
1088 return false;
1089
1090 FTM::StaticData data(fBufStaticData);
1091
1092 if (lp==FTM::StaticData::kLPint)
1093 data.fIntensityLPint = intensity;
1094
1095 if (lp==FTM::StaticData::kLPext)
1096 data.fIntensityLPext = intensity;
1097
1098 CmdSendStatDat(data);
1099
1100 return true;
1101 }
1102
1103 bool EnablePixel(int16_t idx, bool enable)
1104 {
1105 if (idx<-1 || idx>FTM::StaticData::kMaxPixelIdx)
1106 return false;
1107
1108 if (idx==-1)
1109 {
1110 FTM::StaticData data(fBufStaticData);
1111
1112 for (int i=0; i<=FTM::StaticData::kMaxPixelIdx; i++)
1113 data.EnablePixel(i, enable);
1114
1115 CmdSendStatDat(data);
1116
1117 return true;
1118 }
1119
1120 /*
1121 data.EnablePixel(idx, enable);
1122 CmdSendStatDat(data);
1123 return true;
1124 */
1125
1126 FTM::StaticData data(fBufStaticData);
1127
1128 const uintptr_t base = uintptr_t(&data);
1129 const uint16_t *mem = data.EnablePixel(idx, enable);
1130
1131 // Calculate offset in static data block
1132 const uint16_t addr = (uintptr_t(mem)-base)/2;
1133
1134 // From CmdSetRegister
1135 const array<uint16_t, 2> cmd = {{ addr, *mem }};
1136 PostCmd(cmd, FTM::kCmdWrite, FTM::kCmdRegister);
1137
1138 reinterpret_cast<uint16_t*>(&fBufStaticData)[addr] = *mem;
1139
1140 // Now execute change before the static data is requested back
1141 PostCmd(FTM::kCmdConfigFTU, (idx/360) | (((idx/36)%10)<<8));
1142
1143 // Now request the register back to ensure consistency
1144 //CmdGetRegister(addr);
1145 CmdReqStatDat();
1146
1147 return true;
1148 }
1149
1150 bool DisableAllPixelsExcept(uint16_t idx)
1151 {
1152 if (idx>FTM::StaticData::kMaxPixelIdx)
1153 return false;
1154
1155 FTM::StaticData data(fBufStaticData);
1156
1157 for (int i=0; i<=FTM::StaticData::kMaxPixelIdx; i++)
1158 data.EnablePixel(i, i==idx);
1159
1160 CmdSendStatDat(data);
1161
1162 return true;
1163 }
1164
1165 bool DisableAllPatchesExcept(int16_t idx)
1166 {
1167 if (idx>FTM::StaticData::kMaxPatchIdx)
1168 return false;
1169
1170 FTM::StaticData data(fBufStaticData);
1171
1172 for (int i=0; i<=FTM::StaticData::kMaxPixelIdx; i++)
1173 data.EnablePixel(i, i/9==idx);
1174
1175 CmdSendStatDat(data);
1176
1177 return true;
1178 }
1179
1180 bool EnablePatch(int16_t idx, bool enable)
1181 {
1182 if (idx>FTM::StaticData::kMaxPatchIdx)
1183 return false;
1184
1185 FTM::StaticData data(fBufStaticData);
1186
1187 for (int i=0; i<=FTM::StaticData::kMaxPixelIdx; i++)
1188 if (i/9==idx)
1189 data.EnablePixel(i, enable);
1190
1191 CmdSendStatDat(data);
1192
1193 return true;
1194 }
1195
1196 bool TogglePixel(uint16_t idx)
1197 {
1198 if (idx>FTM::StaticData::kMaxPixelIdx)
1199 return false;
1200
1201 FTM::StaticData data(fBufStaticData);
1202
1203 data.EnablePixel(idx, !fBufStaticData.Enabled(idx));
1204
1205 CmdSendStatDat(data);
1206
1207 return true;
1208 }
1209
1210 States GetState() const
1211 {
1212 if (!IsConnected())
1213 return kDisconnected; // rc=1
1214
1215 switch (fHeader.fState&FTM::kFtmStates)
1216 {
1217 case FTM::kFtmUndefined: // 0
1218 return fBufStaticData.valid() ? kConnected : kDisconnected; // rc=2
1219
1220 case FTM::kFtmRunning: // 3
1221 case FTM::kFtmCalib: // 4
1222 return kTriggerOn; // rc=4
1223
1224 case FTM::kFtmIdle: // 1
1225 case FTM::kFtmConfig: // 2 // rc=7 // rc=3
1226 return fStaticData == fBufStaticData ? kConfigured : kIdle;
1227 }
1228
1229 throw runtime_error("ConnectionFTM::GetState - Impossible code reached.");
1230 }
1231
1232 // If fState==2, the clock conditioner will always be reported as unlocked
1233 //bool IsLocked() const { return fHeader.fState&FTM::kFtmLocked; }
1234
1235 uint32_t GetCounter(FTM::Types type) { return fCounter[type]; }
1236
1237 const FTM::StaticData &GetStaticData() const { return fStaticData; }
1238};
1239
1240//const uint16_t ConnectionFTM::kMaxAddr = 0xfff;
1241
1242// ------------------------------------------------------------------------
1243
1244#include "DimDescriptionService.h"
1245
1246class ConnectionDimFTM : public ConnectionFTM
1247{
1248private:
1249
1250 DimDescribedService fDimPassport;
1251 DimDescribedService fDimTriggerRates;
1252 DimDescribedService fDimError;
1253 DimDescribedService fDimFtuList;
1254 DimDescribedService fDimStaticData;
1255 DimDescribedService fDimDynamicData;
1256 DimDescribedService fDimCounter;
1257
1258 uint64_t fTimeStamp;
1259 uint64_t fTimeStampOn;
1260 uint32_t fTriggerCounter;
1261 uint64_t fPrevState;
1262
1263 void UpdateFirstHeader()
1264 {
1265 ConnectionFTM::UpdateFirstHeader();
1266
1267 const FTM::DimPassport data(fHeader);
1268 fDimPassport.Update(data);
1269 }
1270
1271 /*
1272 void UpdateHeader()
1273 {
1274 ConnectionFTM::UpdateHeader();
1275
1276 if (fHeader.fType!=FTM::kDynamicData)
1277 return;
1278
1279 const FTM::DimTriggerCounter data(fHeader);
1280 fDimTriggerCounter.Update(data);
1281 }*/
1282
1283 void UpdateFtuList()
1284 {
1285 ConnectionFTM::UpdateFtuList();
1286
1287 const FTM::DimFtuList data(fHeader, fFtuList);
1288 fDimFtuList.Update(data);
1289 }
1290
1291 void UpdateStaticData()
1292 {
1293 ConnectionFTM::UpdateStaticData();
1294
1295 const FTM::DimStaticData data(fHeader, fStaticData);
1296 fDimStaticData.setQuality(fHeader.fState);
1297 fDimStaticData.Update(data);
1298 }
1299
1300 void UpdateDynamicData()
1301 {
1302 ConnectionFTM::UpdateDynamicData();
1303
1304 const FTM::DimDynamicData data(fHeader, fDynamicData, fStaticData);
1305 fDimDynamicData.setQuality(fHeader.fState);
1306 fDimDynamicData.Update(data);
1307
1308 uint64_t odiff = fDynamicData.fOnTimeCounter;
1309 uint32_t cdiff = fHeader.fTriggerCounter;
1310 uint64_t tdiff = fHeader.fTimeStamp;
1311
1312 // The easiest way to detect whether the counters have been
1313 // reset or not is to detect a state change, because with
1314 // every state change they are reset. However, there are cases
1315 // when the trigger is switched on already (data run) and
1316 // the trigger is turned off ans switched on again within
1317 // a very short time, that the state of the previous and the
1318 // new report is the same. So in addition we have to check
1319 // for other indications. Any counter decreasing is a hint.
1320 // None of them should ever decrease. So all three are checked.
1321 const uint8_t state = fHeader.fState & FTM::States::kFtmStates;
1322
1323 const bool first = state!=fPrevState ||
1324 fHeader.fTimeStamp<fTimeStamp ||
1325 fHeader.fTriggerCounter<fTriggerCounter ||
1326 fDynamicData.fOnTimeCounter<fTimeStampOn;
1327
1328 if (!first)
1329 {
1330 tdiff -= fTimeStamp;
1331 odiff -= fTimeStampOn;
1332 cdiff -= fTriggerCounter;
1333 }
1334
1335 // The observation time calculated in the first report is most likely
1336 // too large because the previous report is taken as reference,
1337 // but this is the best what could be done.
1338 const float rate = tdiff==0 ? 0 : 1e6*cdiff/tdiff;
1339
1340 fTimeStamp = fHeader.fTimeStamp;
1341 fTimeStampOn = fDynamicData.fOnTimeCounter;
1342 fTriggerCounter = fHeader.fTriggerCounter;
1343 fPrevState = state;
1344
1345 const FTM::DimTriggerRates rates(fHeader, fDynamicData, fStaticData,
1346 rate, tdiff*1e-6, odiff*1e-6);
1347
1348 fDimTriggerRates.setQuality(fHeader.fState);
1349 fDimTriggerRates.Update(rates);
1350 }
1351
1352 void UpdateError()
1353 {
1354 ConnectionFTM::UpdateError();
1355
1356 const FTM::DimError data(fHeader, fError);
1357 fDimError.Update(data);
1358 }
1359
1360 void UpdateCounter()
1361 {
1362 ConnectionFTM::UpdateCounter();
1363
1364 const uint32_t counter[6] =
1365 {
1366 fCounter[FTM::kHeader],
1367 fCounter[FTM::kStaticData],
1368 fCounter[FTM::kDynamicData],
1369 fCounter[FTM::kFtuList],
1370 fCounter[FTM::kErrorList],
1371 fCounter[FTM::kRegister],
1372 };
1373
1374 fDimCounter.setQuality(fHeader.fState);
1375 fDimCounter.Update(counter);
1376 }
1377
1378public:
1379 ConnectionDimFTM(ba::io_service& ioservice, MessageImp &imp) :
1380 ConnectionFTM(ioservice, imp),
1381 fDimPassport ("FTM_CONTROL/PASSPORT", "X:1;S:1",
1382 "Info about the FTM and FPGA version"
1383 "|BoardId[int]:BoardId, hexCode"
1384 "|DNA[int]:DNA of the FTM board"),
1385 fDimTriggerRates ("FTM_CONTROL/TRIGGER_RATES", "X:1;X:1;I:1;F:1;F:40;F:160;F:1;F:1",
1386 "Patch,Board,Camera trigger rates"
1387 "|FTMtimeStamp[us]:Time in microseconds, since trigger enabled or disabled"
1388 "|OnTimeCounter[us]:Effective on-time, ie. FTM triggers (eg. w/o busy)"
1389 "|TriggerCounter[int]:Counter of triggers since enabled or disabled"
1390 "|TriggerRate[Hz]:Trigger rate"
1391 "|BoardRate[Hz]:Trigger rate of individual FTUs"
1392 "|PatchRate[Hz]:Trigger rate of individual patches"
1393 "|ElapsedTime[s]:Time elapsed since previous report"
1394 "|OnTime[s]:OnTime elapsed since previous report"),
1395 fDimError ("FTM_CONTROL/ERROR", "X:1;S:1;S:28", ""),
1396 fDimFtuList ("FTM_CONTROL/FTU_LIST", "X:1;X:1;S:1;C:4;X:40;C:40;C:40",
1397 "Logs the changes of status of the FTUs"
1398 "|FTMtimeStamp[us]:Time in microseconds"
1399 "|ActiveFTU[bitpattern]:Description of enabled FTUs"
1400 "|NumBoards[int]:Total number of enabled FTUs"
1401 "|NumBoardsCrate[int]:Total number of enabled FTUs per crate"
1402 "|DNA[hexCode]:Hex code identifier of FTUs"
1403 "|Addr[bitpattern]:Crate address (hardware) of FTUs"
1404 "|Ping[int]:Number of pings until FTU response"),
1405 fDimStaticData ("FTM_CONTROL/STATIC_DATA", "X:1;S:1;S:1;X:1;S:1;S:3;C:4;S:1;S:1;S:1;S:1;S:1;S:1;I:1;I:8;S:90;S:160;S:40;S:40",
1406 "Configuration of FTM and FTUs"
1407 "|FTMtimeStamp[us]:Time in microseconds, since trigger enabled or disabled"
1408 "|GeneralSettings[bitpattern]:Status of the FTM settings (cf. FTM doc)"
1409 "|LEDStatus[bitpattern]:Not Used"
1410 "|ActiveFTU[bitpattern]:List of enabled FTUs"
1411 "|TriggerInterval[bitpattern]:Period of cal. and ped. events (cf. FTM doc)"
1412 "|TriggerSeq[int]:Sequence of calib. and pedestal events (LPint, LPext, Ped)"
1413 "|LPSettings[bitpattern]:Settings of LP, enabled int, ext, intensity int, ext"
1414 "|PhysTrigMult[int]:N for N out of 40 logic on FTM (Physics)"
1415 "|CalibTrigMult[int]: N for N out of 40 logic on FTM (Calib)"
1416 "|PhysTrigWindow[ns]:Coincidence window for N out of 40 (Physics)"
1417 "|CalibTrigWindow[ns]:Coincidence window for N out of 40 (Calib)"
1418 "|TrigDelay[ns]:Trigger delay applied on FTM"
1419 "|TMDelay[ns]:TM delay applied on FTM"
1420 "|DeadTime[ns]:Dead time applied after each event on the FTM"
1421 "|ClkCond[bitpattern]:Clock conditionner settings on the FTM (DRS sampling freq.)"
1422 "|PixEnabled[bitpattern]:Enabled pixels, pckd in 90 shorts (160*9bits=180bytes)"
1423 "|PatchThresh[DACcounts]:Threshold of the trigger patches"
1424 "|Multiplicity[DACcounts]:N out of 4 logic settings per FTU"
1425 "|Prescaling[500ms]:Update rate of the rate counter"),
1426 fDimDynamicData ("FTM_CONTROL/DYNAMIC_DATA", "X:1;X:1;F:4;I:160;I:40;S:40;S:40;S:40;S:1",
1427 "Regular reports sent by FTM"
1428 "|FTMtimeStamp[us]:Time in microseconds, since trigger enabled or disabled"
1429 "|OnTimeCounter[us]:Ontime, i.e. FTM processes triggers (e.g. No FAD busy)"
1430 "|Temperatures[Nan]:not yet defined nor used (wanna be FTM onboard temps)"
1431 "|TriggerPatchCounter[int]:counting since last update (prescaling)"
1432 "|BoardsCounter[int]:FTU board counting after N out of 4 and since last update"
1433 "|RateOverflow[bitpattern]:bits 0-4=patches overflow, 5=board overflow, 1 per board"
1434 "|Prescaling[500ms]:Update rate of the rate counter"
1435 "|CrcError[int]:Number of checksum error in RS485 communication"
1436 "|State[int]:State value of the FTM firmware (cf. FTM doc)"),
1437 fDimCounter ("FTM_CONTROL/COUNTER", "I:1;I:1;I:1;I:1;I:1;I:1",
1438 "Communication statistics to or from FTM control and FTM"
1439 "|NumHeaders[int]:Num. of headers (any header) received by ftm control"
1440 "|NumStaticData[int]:Num. of static data blocks (ftm and ftu settings)"
1441 "|NumDynamicData[int]:Num. of dynamic data blocks (e.g. rates)"
1442 "|NumFtuList[int]:Num. of FTU list (FTU identifiers, answer from ping)"
1443 "|NumErrors[int]:Num. of error messages"
1444 "|NumRegister[int]:Num. of answers from a single register accesess"),
1445 fTimeStamp(0), fTimeStampOn(0), fTriggerCounter(0), fPrevState(0)
1446 {
1447 }
1448
1449 // A B [C] [D] E [F] G H [I] J K [L] M N O P Q R [S] T U V W [X] Y Z
1450};
1451
1452// ------------------------------------------------------------------------
1453
1454template <class T, class S>
1455class StateMachineFTM : public StateMachineAsio<T>
1456{
1457 int Wrap(function<void()> f)
1458 {
1459 f();
1460 return T::GetCurrentState();
1461 }
1462
1463 function<int(const EventImp &)> Wrapper(function<void()> func)
1464 {
1465 return bind(&StateMachineFTM::Wrap, this, func);
1466 }
1467
1468private:
1469 S fFTM;
1470
1471 bool CheckEventSize(size_t has, const char *name, size_t size)
1472 {
1473 if (has==size)
1474 return true;
1475
1476 ostringstream msg;
1477 msg << name << " - Received event has " << has << " bytes, but expected " << size << ".";
1478 T::Fatal(msg);
1479 return false;
1480 }
1481
1482 int SetRegister(const EventImp &evt)
1483 {
1484 if (!CheckEventSize(evt.GetSize(), "SetRegister", 8))
1485 return T::kSM_FatalError;
1486
1487 const uint32_t *dat = evt.Ptr<uint32_t>();
1488
1489 if (dat[1]>uint16_t(-1))
1490 {
1491 ostringstream msg;
1492 msg << hex << "Value " << dat[1] << " out of range.";
1493 T::Error(msg);
1494 return T::GetCurrentState();
1495 }
1496
1497
1498 if (dat[0]>uint16_t(-1) || !fFTM.CmdSetRegister(dat[0], dat[1]))
1499 {
1500 ostringstream msg;
1501 msg << hex << "Address " << dat[0] << " out of range.";
1502 T::Error(msg);
1503 }
1504
1505 return T::GetCurrentState();
1506 }
1507
1508 int GetRegister(const EventImp &evt)
1509 {
1510 if (!CheckEventSize(evt.GetSize(), "GetRegister", 4))
1511 return T::kSM_FatalError;
1512
1513 const unsigned int addr = evt.GetInt();
1514 if (addr>uint16_t(-1) || !fFTM.CmdGetRegister(addr))
1515 {
1516 ostringstream msg;
1517 msg << hex << "Address " << addr << " out of range.";
1518 T::Error(msg);
1519 }
1520
1521 return T::GetCurrentState();
1522 }
1523
1524 int TakeNevents(const EventImp &evt)
1525 {
1526 if (!CheckEventSize(evt.GetSize(), "TakeNevents", 4))
1527 return T::kSM_FatalError;
1528
1529 const unsigned int dat = evt.GetUInt();
1530
1531 /*
1532 if (dat[1]>uint32_t(-1))
1533 {
1534 ostringstream msg;
1535 msg << hex << "Value " << dat[1] << " out of range.";
1536 T::Error(msg);
1537 return T::GetCurrentState();
1538 }*/
1539
1540 fFTM.CmdTakeNevents(dat);
1541
1542 return T::GetCurrentState();
1543 }
1544
1545 int DisableReports(const EventImp &evt)
1546 {
1547 if (!CheckEventSize(evt.GetSize(), "DisableReports", 1))
1548 return T::kSM_FatalError;
1549
1550 fFTM.CmdDisableReports(evt.GetBool());
1551
1552 return T::GetCurrentState();
1553 }
1554
1555 int SetVerbosity(const EventImp &evt)
1556 {
1557 if (!CheckEventSize(evt.GetSize(), "SetVerbosity", 1))
1558 return T::kSM_FatalError;
1559
1560 fFTM.SetVerbose(evt.GetBool());
1561
1562 return T::GetCurrentState();
1563 }
1564
1565 int SetHexOutput(const EventImp &evt)
1566 {
1567 if (!CheckEventSize(evt.GetSize(), "SetHexOutput", 1))
1568 return T::kSM_FatalError;
1569
1570 fFTM.SetHexOutput(evt.GetBool());
1571
1572 return T::GetCurrentState();
1573 }
1574
1575 int SetDynamicOut(const EventImp &evt)
1576 {
1577 if (!CheckEventSize(evt.GetSize(), "SetDynamicOut", 1))
1578 return T::kSM_FatalError;
1579
1580 fFTM.SetDynamicOut(evt.GetBool());
1581
1582 return T::GetCurrentState();
1583 }
1584
1585 int LoadStaticData(const EventImp &evt)
1586 {
1587 if (fFTM.LoadStaticData(evt.GetString()))
1588 return T::GetCurrentState();
1589
1590 ostringstream msg;
1591 msg << "Loading static data from file '" << evt.GetString() << "' failed ";
1592
1593 if (errno)
1594 msg << "(" << strerror(errno) << ")";
1595 else
1596 msg << "(wrong size, expected " << sizeof(FTM::StaticData) << " bytes)";
1597
1598 T::Warn(msg);
1599
1600 return T::GetCurrentState();
1601 }
1602
1603 int SaveStaticData(const EventImp &evt)
1604 {
1605 if (fFTM.SaveStaticData(evt.GetString()))
1606 return T::GetCurrentState();
1607
1608 ostringstream msg;
1609 msg << "Writing static data to file '" << evt.GetString() << "' failed ";
1610 msg << "(" << strerror(errno) << ")";
1611
1612 T::Warn(msg);
1613
1614 return T::GetCurrentState();
1615 }
1616
1617 int SetThreshold(const EventImp &evt)
1618 {
1619 if (!CheckEventSize(evt.GetSize(), "SetThreshold", 8))
1620 return T::kSM_FatalError;
1621
1622 const int32_t *data = evt.Ptr<int32_t>();
1623
1624 if (!fFTM.SetThreshold(data[0], data[1]))
1625 {
1626 ostringstream msg;
1627 msg << "SetThreshold - Maximum allowed patch number 159, valid value range 0-0xffff (got: " << data[0] << " " << data[1] << ")";
1628 T::Warn(msg);
1629 }
1630
1631 return T::GetCurrentState();
1632 }
1633
1634 int SetAllThresholds(const EventImp &evt)
1635 {
1636 if (!CheckEventSize(evt.GetSize(), "SetAllThresholds", 160*4))
1637 return T::kSM_FatalError;
1638
1639 const int32_t *data = evt.Ptr<int32_t>();
1640 if (!fFTM.SetAllThresholds(data))
1641 {
1642 ostringstream msg;
1643 msg << "SetAllThresholds - Value out of range [0; 0xffff]";
1644 T::Warn(msg);
1645 }
1646
1647 return T::GetCurrentState();
1648 }
1649
1650 int SetNoutof4(const EventImp &evt)
1651 {
1652 if (!CheckEventSize(evt.GetSize(), "SetNoutof4", 8))
1653 return T::kSM_FatalError;
1654
1655 const int32_t *data = evt.Ptr<int32_t>();
1656
1657 if (!fFTM.SetNoutof4(data[0], data[1]))
1658 T::Warn("SetNoutof4 - Maximum allowed board number 39, valid value range 0-0xffff");
1659
1660 return T::GetCurrentState();
1661 }
1662
1663 int EnableFTU(const EventImp &evt)
1664 {
1665 if (!CheckEventSize(evt.GetSize(), "EnableFTU", 5))
1666 return T::kSM_FatalError;
1667
1668 const int32_t &board = evt.Get<int32_t>();
1669 const int8_t &enable = evt.Get<int8_t>(4);
1670
1671 if (!fFTM.EnableFTU(board, enable))
1672 T::Warn("EnableFTU - Board number must be <40.");
1673
1674 return T::GetCurrentState();
1675 }
1676
1677 int ToggleFTU(const EventImp &evt)
1678 {
1679 if (!CheckEventSize(evt.GetSize(), "ToggleFTU", 4))
1680 return T::kSM_FatalError;
1681
1682 if (!fFTM.ToggleFTU(evt.GetInt()))
1683 T::Warn("ToggleFTU - Allowed range of boards 0-39.");
1684
1685 return T::GetCurrentState();
1686 }
1687
1688 int SetTriggerInterval(const EventImp &evt)
1689 {
1690 if (!CheckEventSize(evt.GetSize(), "SetTriggerInterval", 4))
1691 return T::kSM_FatalError;
1692
1693 if (!fFTM.SetTriggerInterval(evt.GetInt()))
1694 T::Warn("SetTriggerInterval - Value out of range.");
1695
1696 return T::GetCurrentState();
1697 }
1698
1699 int SetTriggerDelay(const EventImp &evt)
1700 {
1701 if (!CheckEventSize(evt.GetSize(), "SetTriggerDelay", 4))
1702 return T::kSM_FatalError;
1703
1704 if (!fFTM.SetTriggerDelay(evt.GetInt()))
1705 T::Warn("SetTriggerDealy - Value out of range.");
1706
1707 return T::GetCurrentState();
1708 }
1709
1710 int SetTimeMarkerDelay(const EventImp &evt)
1711 {
1712 if (!CheckEventSize(evt.GetSize(), "SetTimeMarkerDelay", 4))
1713 return T::kSM_FatalError;
1714
1715 if (!fFTM.SetTimeMarkerDelay(evt.GetInt()))
1716 T::Warn("SetTimeMarkerDelay - Value out of range.");
1717
1718 return T::GetCurrentState();
1719 }
1720
1721 int SetPrescaling(const EventImp &evt)
1722 {
1723 if (!CheckEventSize(evt.GetSize(), "SetPrescaling", 4))
1724 return T::kSM_FatalError;
1725
1726 if (!fFTM.SetPrescaling(evt.GetInt()-1))
1727 T::Warn("SetPrescaling - Value out of range.");
1728
1729 return T::GetCurrentState();
1730 }
1731
1732 int SetTriggerSeq(const EventImp &evt)
1733 {
1734 if (!CheckEventSize(evt.GetSize(), "SetTriggerSeq", 6))
1735 return T::kSM_FatalError;
1736
1737 const uint16_t *data = evt.Ptr<uint16_t>();
1738
1739 if (!fFTM.SetTriggerSeq(data))
1740 T::Warn("SetTriggerSeq - Value out of range.");
1741
1742 return T::GetCurrentState();
1743 }
1744
1745 int SetDeadTime(const EventImp &evt)
1746 {
1747 if (!CheckEventSize(evt.GetSize(), "SetDeadTime", 4))
1748 return T::kSM_FatalError;
1749
1750 if (!fFTM.SetDeadTime(evt.GetInt()))
1751 T::Warn("SetDeadTime - Value out of range.");
1752
1753 return T::GetCurrentState();
1754 }
1755
1756 int SetTriggerMultiplicity(const EventImp &evt)
1757 {
1758 if (!CheckEventSize(evt.GetSize(), "SetTriggerMultiplicity", 2))
1759 return T::kSM_FatalError;
1760
1761 if (!fFTM.SetTriggerMultiplicity(evt.GetUShort()))
1762 T::Warn("SetTriggerMultiplicity - Value out of range.");
1763
1764 return T::GetCurrentState();
1765 }
1766
1767 int SetCalibMultiplicity(const EventImp &evt)
1768 {
1769 if (!CheckEventSize(evt.GetSize(), "SetCalibMultiplicity", 2))
1770 return T::kSM_FatalError;
1771
1772 if (!fFTM.SetCalibMultiplicity(evt.GetUShort()))
1773 T::Warn("SetCalibMultiplicity - Value out of range.");
1774
1775 return T::GetCurrentState();
1776 }
1777
1778 int SetTriggerWindow(const EventImp &evt)
1779 {
1780 if (!CheckEventSize(evt.GetSize(), "SetTriggerWindow", 2))
1781 return T::kSM_FatalError;
1782
1783 if (!fFTM.SetTriggerWindow(evt.GetUShort()))
1784 T::Warn("SetTriggerWindow - Value out of range.");
1785
1786 return T::GetCurrentState();
1787 }
1788
1789 int SetCalibWindow(const EventImp &evt)
1790 {
1791 if (!CheckEventSize(evt.GetSize(), "SetCalibWindow", 2))
1792 return T::kSM_FatalError;
1793
1794 if (!fFTM.SetCalibWindow(evt.GetUShort()))
1795 T::Warn("SetCalibWindow - Value out of range.");
1796
1797 return T::GetCurrentState();
1798 }
1799
1800 int SetClockRegister(const EventImp &evt)
1801 {
1802 if (!CheckEventSize(evt.GetSize(), "SetClockRegister", 8*8))
1803 return T::kSM_FatalError;
1804
1805 const uint64_t *reg = evt.Ptr<uint64_t>();
1806
1807 if (!fFTM.SetClockRegister(reg))
1808 T::Warn("SetClockRegister - Value out of range.");
1809
1810 return T::GetCurrentState();
1811 }
1812
1813 int SetClockFrequency(const EventImp &evt)
1814 {
1815 if (!CheckEventSize(evt.GetSize(), "SetClockFrequency", 2))
1816 return T::kSM_FatalError;
1817
1818 const map<uint16_t,array<uint64_t, 8>>::const_iterator it =
1819 fClockCondSetup.find(evt.GetUShort());
1820
1821 if (it==fClockCondSetup.end())
1822 {
1823 T::Warn("SetClockFrequency - Frequency not supported.");
1824 return T::GetCurrentState();
1825 }
1826
1827 if (!fFTM.SetClockRegister(it->second.data()))
1828 T::Warn("SetClockFrequency - Register values out of range.");
1829
1830 return T::GetCurrentState();
1831 }
1832
1833 int EnableLP(const EventImp &evt, FTM::StaticData::GeneralSettings lp, FTM::StaticData::LightPulserEnable group)
1834 {
1835 if (!CheckEventSize(evt.GetSize(), "EnableLP", 1))
1836 return T::kSM_FatalError;
1837
1838 if (!fFTM.EnableLP(lp, group, evt.GetBool()))
1839 T::Warn("EnableLP - Invalid light pulser id.");
1840
1841 return T::GetCurrentState();
1842 }
1843
1844 int SetIntensity(const EventImp &evt, FTM::StaticData::GeneralSettings lp)
1845 {
1846 if (!CheckEventSize(evt.GetSize(), "SetIntensity", 2))
1847 return T::kSM_FatalError;
1848
1849 if (!fFTM.SetIntensity(lp, evt.GetShort()))
1850 T::Warn("SetIntensity - Value out of range.");
1851
1852 return T::GetCurrentState();
1853 }
1854
1855 int Enable(const EventImp &evt, FTM::StaticData::GeneralSettings type)
1856 {
1857 if (!CheckEventSize(evt.GetSize(), "Enable", 1))
1858 return T::kSM_FatalError;
1859
1860 fFTM.Enable(type, evt.GetBool());
1861
1862 return T::GetCurrentState();
1863 }
1864
1865 int EnablePixel(const EventImp &evt, bool b)
1866 {
1867 if (!CheckEventSize(evt.GetSize(), "EnablePixel", 2))
1868 return T::kSM_FatalError;
1869
1870 if (!fFTM.EnablePixel(evt.GetUShort(), b))
1871 T::Warn("EnablePixel - Value out of range.");
1872
1873 return T::GetCurrentState();
1874 }
1875
1876 int DisableAllPixelsExcept(const EventImp &evt)
1877 {
1878 if (!CheckEventSize(evt.GetSize(), "DisableAllPixelsExcept", 2))
1879 return T::kSM_FatalError;
1880
1881 if (!fFTM.DisableAllPixelsExcept(evt.GetUShort()))
1882 T::Warn("DisableAllPixelsExcept - Value out of range.");
1883
1884 return T::GetCurrentState();
1885 }
1886
1887 int DisableAllPatchesExcept(const EventImp &evt)
1888 {
1889 if (!CheckEventSize(evt.GetSize(), "DisableAllPatchesExcept", 2))
1890 return T::kSM_FatalError;
1891
1892 if (!fFTM.DisableAllPatchesExcept(evt.GetUShort()))
1893 T::Warn("DisableAllPatchesExcept - Value out of range.");
1894
1895 return T::GetCurrentState();
1896 }
1897
1898 int EnablePatch(const EventImp &evt, bool enable)
1899 {
1900 if (!CheckEventSize(evt.GetSize(), "EnablePatch", 2))
1901 return T::kSM_FatalError;
1902
1903 if (!fFTM.EnablePatch(evt.GetUShort(), enable))
1904 T::Warn("EnablePatch - Value out of range.");
1905
1906 return T::GetCurrentState();
1907 }
1908
1909 int TogglePixel(const EventImp &evt)
1910 {
1911 if (!CheckEventSize(evt.GetSize(), "TogglePixel", 2))
1912 return T::kSM_FatalError;
1913
1914 if (!fFTM.TogglePixel(evt.GetUShort()))
1915 T::Warn("TogglePixel - Value out of range.");
1916
1917 return T::GetCurrentState();
1918 }
1919
1920 int ResetCrate(const EventImp &evt)
1921 {
1922 if (!CheckEventSize(evt.GetSize(), "ResetCrate", 2))
1923 return T::kSM_FatalError;
1924
1925 fFTM.CmdResetCrate(evt.GetUShort());
1926
1927 return T::GetCurrentState();
1928 }
1929
1930 int Disconnect()
1931 {
1932 // Close all connections
1933 fFTM.PostClose(false);
1934
1935 /*
1936 // Now wait until all connection have been closed and
1937 // all pending handlers have been processed
1938 poll();
1939 */
1940
1941 return T::GetCurrentState();
1942 }
1943
1944 int Reconnect(const EventImp &evt)
1945 {
1946 // Close all connections to supress the warning in SetEndpoint
1947 fFTM.PostClose(false);
1948
1949 // Now wait until all connection have been closed and
1950 // all pending handlers have been processed
1951 ba::io_service::poll();
1952
1953 if (evt.GetBool())
1954 fFTM.SetEndpoint(evt.GetString());
1955
1956 // Now we can reopen the connection
1957 fFTM.PostClose(true);
1958
1959 return T::GetCurrentState();
1960 }
1961
1962 /*
1963 int Transition(const Event &evt)
1964 {
1965 switch (evt.GetTargetState())
1966 {
1967 case kDisconnected:
1968 case kConnected:
1969 }
1970
1971 return T::kSM_FatalError;
1972 }*/
1973
1974 uint32_t fCounterReg;
1975 uint32_t fCounterStat;
1976
1977 typedef map<string, FTM::StaticData> Configs;
1978 Configs fConfigs;
1979 Configs::const_iterator fTargetConfig;
1980
1981 int ConfigureFTM(const EventImp &evt)
1982 {
1983 const string name = evt.GetText();
1984
1985 fTargetConfig = fConfigs.find(name);
1986 if (fTargetConfig==fConfigs.end())
1987 {
1988 T::Error("ConfigureFTM - Run-type '"+name+"' not found.");
1989 return T::GetCurrentState();
1990 }
1991
1992 T::Message("Starting configuration for '"+name+"' ["+to_string(fFTM.IsTxQueueEmpty())+"]");
1993
1994 fCounterReg = fFTM.GetCounter(FTM::kRegister);
1995 fFTM.CmdStopRun();
1996
1997 return FTM::State::kConfiguring1;
1998 }
1999
2000 int ResetConfig()
2001 {
2002 return fFTM.GetState();
2003 }
2004
2005 int Execute()
2006 {
2007 // If FTM is neither in data taking nor idle,
2008 // leave configuration state
2009 switch (fFTM.GetState())
2010 {
2011 case ConnectionFTM::kDisconnected: return FTM::State::kDisconnected;
2012 case ConnectionFTM::kConnected: return FTM::State::kConnected;
2013 default:
2014 break;
2015 }
2016
2017 // FIXME: Add timeouts and go to error state
2018 // so that a configuration error can be handled
2019 switch (T::GetCurrentState())
2020 {
2021 case FTM::State::kConfiguring1:
2022 // If FTM has received an anwer to the stop_run command
2023 // the counter for the registers has been increased
2024 if (fFTM.GetCounter(FTM::kRegister)<=fCounterReg)
2025 break;
2026
2027 // If now the state is not idle as expected this means we had
2028 // an error (maybe old events waiting in the queue)
2029 if (fFTM.GetState()!=ConnectionFTM::kIdle &&
2030 fFTM.GetState()!=ConnectionFTM::kConfigured)
2031 return FTM::State::kConfigError1;
2032
2033 fCounterStat = fFTM.GetCounter(FTM::kStaticData);
2034
2035 fFTM.CmdSendStatDat(fTargetConfig->second);
2036
2037 T::Message("Trigger successfully disabled... sending new configuration.");
2038
2039 // Next state is: wait for the answer to our configuration
2040 return FTM::State::kConfiguring2;
2041
2042 case FTM::State::kConfiguring2:
2043 case FTM::State::kConfigured:
2044 // If FTM has received an anwer to the CmdSendStatDat
2045 // the counter for static data has been increased
2046 if (fFTM.GetCounter(FTM::kStaticData)<=fCounterStat)
2047 break;
2048
2049 // If now the configuration is not what we expected
2050 // we had an error (maybe old events waiting in the queue?)
2051 if (fFTM.GetState()!=ConnectionFTM::kConfigured)
2052 return FTM::State::kConfigError2;
2053
2054 // Check configuration again when a new static data block
2055 // will be received
2056 fCounterStat = fFTM.GetCounter(FTM::kStaticData);
2057
2058 // This is also displayed when the ratecontrol sends its configuration...
2059 if (T::GetCurrentState()==FTM::State::kConfiguring2)
2060 T::Message("Sending new configuration was successfull.");
2061 else
2062 T::Message("Configuration successfully updated.");
2063
2064 // Next state is: wait for the answer to our configuration
2065 return FTM::State::kConfigured;
2066
2067 case FTM::State::kConfigError1:
2068 case FTM::State::kConfigError2:
2069 //case FTM::State::kConfigError3:
2070 break;
2071
2072 default:
2073 switch (fFTM.GetState())
2074 {
2075 case ConnectionFTM::kIdle: return FTM::State::kIdle;
2076 case ConnectionFTM::kConfigured: return FTM::State::kValid;
2077 case ConnectionFTM::kTriggerOn: return FTM::State::kTriggerOn;
2078 default:
2079 throw runtime_error("StateMachineFTM - Execute() - Inavlid state.");
2080 }
2081 }
2082
2083 if (T::GetCurrentState()==FTM::State::kConfigured &&
2084 fFTM.GetState()==ConnectionFTM::kTriggerOn)
2085 {
2086// if (!fFTM.IsLocked())
2087// {
2088// T::Warn("Configuration successfull, but clock conditioner not locked.");
2089// return FTM::State::kConfigError3;
2090// }
2091
2092 return FTM::State::kTriggerOn;
2093 }
2094
2095 return T::GetCurrentState();
2096 }
2097
2098public:
2099 StateMachineFTM(ostream &out=cout) :
2100 StateMachineAsio<T>(out, "FTM_CONTROL"), fFTM(*this, *this)
2101 {
2102 // State names
2103 T::AddStateName(FTM::State::kDisconnected, "Disconnected",
2104 "FTM board not connected via ethernet.");
2105
2106 T::AddStateName(FTM::State::kConnected, "Connected",
2107 "Ethernet connection to FTM established (no state received yet).");
2108
2109 T::AddStateName(FTM::State::kIdle, "Idle",
2110 "Ethernet connection to FTM established, FTM in idle state.");
2111
2112 T::AddStateName(FTM::State::kValid, "Valid",
2113 "FTM in idle state and the last sent and received static data block are bitwise identical.");
2114
2115 T::AddStateName(FTM::State::kConfiguring1, "Configuring1",
2116 "Command to disable run sent... waiting for response.");
2117 T::AddStateName(FTM::State::kConfiguring2, "Configuring2",
2118 "New configuration sent... waiting for response.");
2119 T::AddStateName(FTM::State::kConfigured, "Configured",
2120 "Received answer identical with target configuration.");
2121
2122 T::AddStateName(FTM::State::kTriggerOn, "TriggerOn",
2123 "Ethernet connection to FTM established, FTM trigger output to FADs enabled.");
2124
2125 T::AddStateName(FTM::State::kConfigError1, "ErrorInConfig1", "Unexpected state received from FTM");
2126 T::AddStateName(FTM::State::kConfigError2, "ErrorInConfig2", "Unexpected state received from FTM");
2127 //T::AddStateName(FTM::State::kConfigError3, "ClockCondError", "Clock conditioner not locked");
2128
2129 // FTM Commands
2130 T::AddEvent("TOGGLE_LED", FTM::State::kIdle, FTM::State::kValid)
2131 (Wrapper(bind(&ConnectionFTM::CmdToggleLed, &fFTM)))
2132 ("toggle led");
2133
2134 T::AddEvent("PING", FTM::State::kIdle, FTM::State::kValid)
2135 (Wrapper(bind(&ConnectionFTM::CmdPing, &fFTM)))
2136 ("send ping");
2137
2138 T::AddEvent("REQUEST_DYNAMIC_DATA", FTM::State::kIdle, FTM::State::kValid)
2139 (Wrapper(bind(&ConnectionFTM::CmdReqDynDat, &fFTM)))
2140 ("request transmission of dynamic data block");
2141
2142 T::AddEvent("REQUEST_STATIC_DATA", FTM::State::kIdle, FTM::State::kValid)
2143 (Wrapper(bind(&ConnectionFTM::CmdReqStatDat, &fFTM)))
2144 ("request transmission of static data from FTM to memory");
2145
2146 T::AddEvent("GET_REGISTER", "I", FTM::State::kIdle, FTM::State::kValid)
2147 (bind(&StateMachineFTM::GetRegister, this, placeholders::_1))
2148 ("read register from address addr"
2149 "|addr[short]:Address of register");
2150
2151 T::AddEvent("SET_REGISTER", "I:2", FTM::State::kIdle, FTM::State::kValid)
2152 (bind(&StateMachineFTM::SetRegister, this, placeholders::_1))
2153 ("set register to value"
2154 "|addr[short]:Address of register"
2155 "|val[short]:Value to be set");
2156
2157 T::AddEvent("START_TRIGGER", FTM::State::kIdle, FTM::State::kValid, FTM::State::kConfigured)
2158 (Wrapper(bind(&ConnectionFTM::CmdStartRun, &fFTM)))
2159 ("start a run (start distributing triggers)");
2160
2161 T::AddEvent("STOP_TRIGGER", FTM::State::kTriggerOn)
2162 (Wrapper(bind(&ConnectionFTM::CmdStopRun, &fFTM)))
2163 ("stop a run (stop distributing triggers)");
2164
2165 T::AddEvent("TAKE_N_EVENTS", "I", FTM::State::kIdle, FTM::State::kValid)
2166 (bind(&StateMachineFTM::TakeNevents, this, placeholders::_1))
2167 ("take n events (distribute n triggers)|number[int]:Number of events to be taken");
2168
2169 T::AddEvent("DISABLE_REPORTS", "B", FTM::State::kIdle, FTM::State::kValid)
2170 (bind(&StateMachineFTM::DisableReports, this, placeholders::_1))
2171 ("disable sending rate reports"
2172 "|status[bool]:disable or enable that the FTM sends rate reports (yes/no)");
2173
2174 T::AddEvent("SET_THRESHOLD", "I:2", FTM::State::kIdle, FTM::State::kValid, FTM::State::kConfigured, FTM::State::kTriggerOn)
2175 (bind(&StateMachineFTM::SetThreshold, this, placeholders::_1))
2176 ("Set the comparator threshold"
2177 "|Patch[idx]:Index of the patch (0-159), -1 for all"
2178 "|Threshold[counts]:Threshold to be set in binary counts");
2179
2180 T::AddEvent("SET_ALL_THRESHOLDS", "I:160", FTM::State::kIdle, FTM::State::kValid, FTM::State::kConfigured, FTM::State::kTriggerOn)
2181 (bind(&StateMachineFTM::SetAllThresholds, this, placeholders::_1))
2182 ("Set the comparator thresholds"
2183 "|Thresholds[counts]:Threshold to be set in binary counts");
2184
2185 T::AddEvent("SET_N_OUT_OF_4", "I:2", FTM::State::kIdle, FTM::State::kValid, FTM::State::kTriggerOn)
2186 (bind(&StateMachineFTM::SetNoutof4, this, placeholders::_1))
2187 ("Set the comparator threshold"
2188 "|Board[idx]:Index of the board (0-39), -1 for all"
2189 "|Threshold[counts]:Threshold to be set in binary counts");
2190
2191 T::AddEvent("SET_PRESCALING", "I:1", FTM::State::kIdle, FTM::State::kValid)
2192 (bind(&StateMachineFTM::SetPrescaling, this, placeholders::_1))
2193 ("Sets the FTU readout time intervals"
2194 "|time[0.5s]:The interval is given in units of 0.5s, i.e. 1 means 0.5s, 2 means 1s, ...");
2195
2196 T::AddEvent("ENABLE_FTU", "I:1;B:1", FTM::State::kIdle, FTM::State::kValid, FTM::State::kConfigured)
2197 (bind(&StateMachineFTM::EnableFTU, this, placeholders::_1))
2198 ("Enable or disable FTU"
2199 "|Board[idx]:Index of the board (0-39), -1 for all"
2200 "|Enable[bool]:Whether FTU should be enabled or disabled (yes/no)");
2201
2202 T::AddEvent("DISABLE_PIXEL", "S:1", FTM::State::kIdle, FTM::State::kValid, FTM::State::kTriggerOn)
2203 (bind(&StateMachineFTM::EnablePixel, this, placeholders::_1, false))
2204 ("(-1 or all)");
2205
2206 T::AddEvent("ENABLE_PIXEL", "S:1", FTM::State::kIdle, FTM::State::kValid, FTM::State::kTriggerOn)
2207 (bind(&StateMachineFTM::EnablePixel, this, placeholders::_1, true))
2208 ("(-1 or all)");
2209
2210 T::AddEvent("DISABLE_ALL_PIXELS_EXCEPT", "S:1", FTM::State::kIdle, FTM::State::kValid)
2211 (bind(&StateMachineFTM::DisableAllPixelsExcept, this, placeholders::_1))
2212 ("");
2213
2214 T::AddEvent("DISABLE_ALL_PATCHES_EXCEPT", "S:1", FTM::State::kIdle, FTM::State::kValid)
2215 (bind(&StateMachineFTM::DisableAllPatchesExcept, this, placeholders::_1))
2216 ("");
2217
2218 T::AddEvent("ENABLE_PATCH", "S:1", FTM::State::kIdle, FTM::State::kValid)
2219 (bind(&StateMachineFTM::EnablePatch, this, placeholders::_1, true))
2220 ("");
2221
2222 T::AddEvent("DISABLE_PATCH", "S:1", FTM::State::kIdle, FTM::State::kValid)
2223 (bind(&StateMachineFTM::EnablePatch, this, placeholders::_1, false))
2224 ("");
2225
2226 T::AddEvent("TOGGLE_PIXEL", "S:1", FTM::State::kIdle, FTM::State::kValid)
2227 (bind(&StateMachineFTM::TogglePixel, this, placeholders::_1))
2228 ("");
2229
2230 T::AddEvent("TOGGLE_FTU", "I:1", FTM::State::kIdle, FTM::State::kValid)
2231 (bind(&StateMachineFTM::ToggleFTU, this, placeholders::_1))
2232 ("Toggle status of FTU (this is mainly meant to be used in the GUI)"
2233 "|Board[idx]:Index of the board (0-39)");
2234
2235 T::AddEvent("SET_TRIGGER_INTERVAL", "I:1", FTM::State::kIdle, FTM::State::kValid)
2236 (bind(&StateMachineFTM::SetTriggerInterval, this, placeholders::_1))
2237 ("Sets the trigger interval which is the distance between two consecutive artificial triggers."
2238 "|interval[ms]:The applied trigger interval in millisecond (min 1ms / 10bit)");
2239
2240 T::AddEvent("SET_TRIGGER_DELAY", "I:1", FTM::State::kIdle, FTM::State::kValid)
2241 (bind(&StateMachineFTM::SetTriggerDelay, this, placeholders::_1))
2242 (""
2243 "|delay[int]:The applied trigger delay is: delay*4ns+8ns");
2244
2245 T::AddEvent("SET_TIME_MARKER_DELAY", "I:1", FTM::State::kIdle, FTM::State::kValid)
2246 (bind(&StateMachineFTM::SetTimeMarkerDelay, this, placeholders::_1))
2247 (""
2248 "|delay[int]:The applied time marker delay is: delay*4ns+8ns");
2249
2250 T::AddEvent("SET_DEAD_TIME", "I:1", FTM::State::kIdle, FTM::State::kValid)
2251 (bind(&StateMachineFTM::SetDeadTime, this, placeholders::_1))
2252 (""
2253 "|dead_time[int]:The applied dead time is: dead_time*4ns+8ns");
2254
2255 T::AddEvent("ENABLE_TRIGGER", "B:1", FTM::State::kIdle, FTM::State::kValid)
2256 (bind(&StateMachineFTM::Enable, this, placeholders::_1, FTM::StaticData::kTrigger))
2257 ("Switch on the physics trigger"
2258 "|Enable[bool]:Enable physics trigger (yes/no)");
2259
2260 // FIXME: Switch on/off depending on sequence
2261 T::AddEvent("ENABLE_EXT1", "B:1", FTM::State::kIdle, FTM::State::kValid)
2262 (bind(&StateMachineFTM::Enable, this, placeholders::_1, FTM::StaticData::kExt1))
2263 ("Switch on the triggers through the first external line"
2264 "|Enable[bool]:Enable ext1 trigger (yes/no)");
2265
2266 // FIXME: Switch on/off depending on sequence
2267 T::AddEvent("ENABLE_EXT2", "B:1", FTM::State::kIdle, FTM::State::kValid)
2268 (bind(&StateMachineFTM::Enable, this, placeholders::_1, FTM::StaticData::kExt2))
2269 ("Switch on the triggers through the second external line"
2270 "|Enable[bool]:Enable ext2 trigger (yes/no)");
2271
2272 T::AddEvent("ENABLE_VETO", "B:1", FTM::State::kIdle, FTM::State::kValid)
2273 (bind(&StateMachineFTM::Enable, this, placeholders::_1, FTM::StaticData::kVeto))
2274 ("Enable veto line"
2275 "|Enable[bool]:Enable veto (yes/no)");
2276
2277 T::AddEvent("ENABLE_CLOCK_CONDITIONER", "B:1", FTM::State::kIdle, FTM::State::kValid)
2278 (bind(&StateMachineFTM::Enable, this, placeholders::_1, FTM::StaticData::kClockConditioner))
2279 ("Enable clock conidtioner output in favor of time marker output"
2280 "|Enable[bool]:Enable clock conditioner (yes/no)");
2281
2282 T::AddEvent("ENABLE_GROUP1_LPINT", "B:1", FTM::State::kIdle, FTM::State::kValid)
2283 (bind(&StateMachineFTM::EnableLP, this, placeholders::_1, FTM::StaticData::kLPint, FTM::StaticData::kGroup1))
2284 ("");
2285 T::AddEvent("ENABLE_GROUP1_LPEXT", "B:1", FTM::State::kIdle, FTM::State::kValid)
2286 (bind(&StateMachineFTM::EnableLP, this, placeholders::_1, FTM::StaticData::kLPext, FTM::StaticData::kGroup1))
2287 ("");
2288 T::AddEvent("ENABLE_GROUP2_LPINT", "B:1", FTM::State::kIdle, FTM::State::kValid)
2289 (bind(&StateMachineFTM::EnableLP, this, placeholders::_1, FTM::StaticData::kLPint, FTM::StaticData::kGroup2))
2290 ("");
2291 T::AddEvent("ENABLE_GROUP2_LPEXT", "B:1", FTM::State::kIdle, FTM::State::kValid)
2292 (bind(&StateMachineFTM::EnableLP, this, placeholders::_1, FTM::StaticData::kLPext, FTM::StaticData::kGroup2))
2293 ("");
2294 T::AddEvent("SET_INTENSITY_LPINT", "S:1", FTM::State::kIdle, FTM::State::kValid)
2295 (bind(&StateMachineFTM::SetIntensity, this, placeholders::_1, FTM::StaticData::kLPint))
2296 ("");
2297 T::AddEvent("SET_INTENSITY_LPEXT", "S:1", FTM::State::kIdle, FTM::State::kValid)
2298 (bind(&StateMachineFTM::SetIntensity, this, placeholders::_1, FTM::StaticData::kLPext))
2299 ("");
2300
2301
2302 T::AddEvent("SET_TRIGGER_SEQUENCE", "S:3", FTM::State::kIdle, FTM::State::kValid)
2303 (bind(&StateMachineFTM::SetTriggerSeq, this, placeholders::_1))
2304 ("Setup the sequence of artificial triggers produced by the FTM"
2305 "|Ped[short]:number of pedestal triggers in a row"
2306 "|LPext[short]:number of triggers of the external light pulser"
2307 "|LPint[short]:number of triggers of the internal light pulser");
2308
2309 T::AddEvent("SET_TRIGGER_MULTIPLICITY", "S:1", FTM::State::kIdle, FTM::State::kValid)
2310 (bind(&StateMachineFTM::SetTriggerMultiplicity, this, placeholders::_1))
2311 ("Setup the Multiplicity condition for physcis triggers"
2312 "|N[int]:Number of requirered coincident triggers from sum-patches (1-40)");
2313
2314 T::AddEvent("SET_TRIGGER_WINDOW", "S:1", FTM::State::kIdle, FTM::State::kValid)
2315 (bind(&StateMachineFTM::SetTriggerWindow, this, placeholders::_1))
2316 ("");
2317
2318 T::AddEvent("SET_CALIBRATION_MULTIPLICITY", "S:1", FTM::State::kIdle, FTM::State::kValid)
2319 (bind(&StateMachineFTM::SetCalibMultiplicity, this, placeholders::_1))
2320 ("Setup the Multiplicity condition for artificial (calibration) triggers"
2321 "|N[int]:Number of requirered coincident triggers from sum-patches (1-40)");
2322
2323 T::AddEvent("SET_CALIBRATION_WINDOW", "S:1", FTM::State::kIdle, FTM::State::kValid)
2324 (bind(&StateMachineFTM::SetCalibWindow, this, placeholders::_1))
2325 ("");
2326
2327 T::AddEvent("SET_CLOCK_FREQUENCY", "S:1", FTM::State::kIdle, FTM::State::kValid)
2328 (bind(&StateMachineFTM::SetClockFrequency, this, placeholders::_1))
2329 ("");
2330
2331 T::AddEvent("SET_CLOCK_REGISTER", "X:8", FTM::State::kIdle, FTM::State::kValid)
2332 (bind(&StateMachineFTM::SetClockRegister, this, placeholders::_1))
2333 ("");
2334
2335 // A new configure will first stop the FTM this means
2336 // we can allow it in idle _and_ taking data
2337 T::AddEvent("CONFIGURE", "C")(FTM::State::kIdle)(FTM::State::kValid)(FTM::State::kConfiguring1)(FTM::State::kConfiguring2)(FTM::State::kConfigured)(FTM::State::kTriggerOn)
2338 (bind(&StateMachineFTM::ConfigureFTM, this, placeholders::_1))
2339 ("");
2340
2341 T::AddEvent("RESET_CONFIGURE")(FTM::State::kConfiguring1)(FTM::State::kConfiguring2)(FTM::State::kConfigured)(FTM::State::kConfigError1)(FTM::State::kConfigError2)(FTM::State::kConfigError2)
2342 (bind(&StateMachineFTM::ResetConfig, this))
2343 ("Reset states during a configuration or in case of configuration error");
2344
2345
2346
2347 T::AddEvent("RESET_CRATE", "S:1", FTM::State::kIdle, FTM::State::kValid, FTM::State::kConfigured)
2348 (bind(&StateMachineFTM::ResetCrate, this, placeholders::_1))
2349 ("Reset one of the crates 0-3"
2350 "|crate[short]:Crate number to be reseted (0-3)");
2351
2352 T::AddEvent("RESET_CAMERA", FTM::State::kIdle, FTM::State::kValid)
2353 (Wrapper(bind(&ConnectionFTM::CmdResetCamera, &fFTM)))
2354 ("Reset all crates. The commands are sent in the order 0,1,2,3");
2355
2356
2357 // Load/save static data block
2358 T::AddEvent("SAVE", "C", FTM::State::kIdle, FTM::State::kValid)
2359 (bind(&StateMachineFTM::SaveStaticData, this, placeholders::_1))
2360 ("Saves the static data (FTM configuration) from memory to a file"
2361 "|filename[string]:Filename (can include a path), .bin is automatically added");
2362
2363 T::AddEvent("LOAD", "C", FTM::State::kIdle, FTM::State::kValid)
2364 (bind(&StateMachineFTM::LoadStaticData, this, placeholders::_1))
2365 ("Loads the static data (FTM configuration) from a file into memory and sends it to the FTM"
2366 "|filename[string]:Filename (can include a path), .bin is automatically added");
2367
2368
2369
2370 // Verbosity commands
2371 T::AddEvent("SET_VERBOSE", "B")
2372 (bind(&StateMachineFTM::SetVerbosity, this, placeholders::_1))
2373 ("set verbosity state"
2374 "|verbosity[bool]:disable or enable verbosity for received data (yes/no), except dynamic data");
2375
2376 T::AddEvent("SET_HEX_OUTPUT", "B")
2377 (bind(&StateMachineFTM::SetHexOutput, this, placeholders::_1))
2378 ("enable or disable hex output for received data"
2379 "|hexout[bool]:disable or enable hex output for received data (yes/no)");
2380
2381 T::AddEvent("SET_DYNAMIC_OUTPUT", "B")
2382 (bind(&StateMachineFTM::SetDynamicOut, this, placeholders::_1))
2383 ("enable or disable output for received dynamic data (data is still broadcasted via Dim)"
2384 "|dynout[bool]:disable or enable output for dynamic data (yes/no)");
2385
2386
2387 // Conenction commands
2388 T::AddEvent("DISCONNECT", FTM::State::kConnected, FTM::State::kIdle, FTM::State::kValid)
2389 (bind(&StateMachineFTM::Disconnect, this))
2390 ("disconnect from ethernet");
2391
2392 T::AddEvent("RECONNECT", "O", FTM::State::kDisconnected, FTM::State::kConnected, FTM::State::kIdle, FTM::State::kValid, FTM::State::kConfigured)
2393 (bind(&StateMachineFTM::Reconnect, this, placeholders::_1))
2394 ("(Re)connect ethernet connection to FTM, a new address can be given"
2395 "|[host][string]:new ethernet address in the form <host:port>");
2396
2397 fFTM.StartConnect();
2398 }
2399
2400 void SetEndpoint(const string &url)
2401 {
2402 fFTM.SetEndpoint(url);
2403 }
2404
2405 map<uint16_t, array<uint64_t, 8>> fClockCondSetup;
2406
2407 template<class V>
2408 bool CheckConfigVal(Configuration &conf, V max, const string &name, const string &sub)
2409 {
2410 if (!conf.HasDef(name, sub))
2411 {
2412 T::Error("Neither "+name+"default nor "+name+sub+" found.");
2413 return false;
2414 }
2415
2416 const V val = conf.GetDef<V>(name, sub);
2417
2418 if (val<=max)
2419 return true;
2420
2421 ostringstream str;
2422 str << name << sub << "=" << val << " exceeds allowed maximum of " << max << "!";
2423 T::Error(str);
2424
2425 return false;
2426 }
2427
2428 int EvalOptions(Configuration &conf)
2429 {
2430 // ---------- General setup ----------
2431 fFTM.SetVerbose(!conf.Get<bool>("quiet"));
2432 fFTM.SetHexOutput(conf.Get<bool>("hex-out"));
2433 fFTM.SetDynamicOut(conf.Get<bool>("dynamic-out"));
2434
2435 // ---------- Setup clock conditioner frequencies ----------
2436 const vector<uint16_t> freq = conf.Vec<uint16_t>("clock-conditioner.frequency");
2437 if (freq.empty())
2438 T::Warn("No frequencies for the clock-conditioner defined.");
2439 else
2440 T::Message("Defining clock conditioner frequencies");
2441 for (vector<uint16_t>::const_iterator it=freq.begin();
2442 it!=freq.end(); it++)
2443 {
2444 if (fClockCondSetup.count(*it)>0)
2445 {
2446 T::Error("clock-conditioner frequency defined twice.");
2447 return 1;
2448 }
2449
2450 if (!conf.HasDef("clock-conditioner.R0.", *it) ||
2451 !conf.HasDef("clock-conditioner.R1.", *it) ||
2452 !conf.HasDef("clock-conditioner.R8.", *it) ||
2453 !conf.HasDef("clock-conditioner.R9.", *it) ||
2454 !conf.HasDef("clock-conditioner.R11.", *it) ||
2455 !conf.HasDef("clock-conditioner.R13.", *it) ||
2456 !conf.HasDef("clock-conditioner.R14.", *it) ||
2457 !conf.HasDef("clock-conditioner.R15.", *it))
2458 {
2459 T::Error("clock-conditioner values incomplete.");
2460 return 1;
2461 }
2462
2463 array<uint64_t, 8> &arr = fClockCondSetup[*it];
2464
2465 arr[0] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R0.", *it);
2466 arr[1] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R1.", *it);
2467 arr[2] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R8.", *it);
2468 arr[3] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R9.", *it);
2469 arr[4] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R11.", *it);
2470 arr[5] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R13.", *it);
2471 arr[6] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R14.", *it);
2472 arr[7] = conf.GetDef<Hex<uint32_t>>("clock-conditioner.R15.", *it);
2473
2474 ostringstream out;
2475 out << " -> " << setw(4) << *it << "MHz:" << hex << setfill('0');
2476 for (int i=0; i<8; i++)
2477 out << " " << setw(8) << arr[i];
2478 T::Message(out.str());
2479 }
2480
2481 // ---------- Setup run types ---------
2482 const vector<string> types = conf.Vec<string>("run-type");
2483 if (types.empty())
2484 T::Warn("No run-types defined.");
2485 else
2486 T::Message("Defining run-types");
2487 for (vector<string>::const_iterator it=types.begin();
2488 it!=types.end(); it++)
2489 {
2490 T::Message(" -> "+ *it);
2491
2492 if (fConfigs.count(*it)>0)
2493 {
2494 T::Error("Run-type "+*it+" defined twice.");
2495 return 2;
2496 }
2497
2498 if (!conf.HasDef("sampling-frequency.", *it))
2499 {
2500 T::Error("Neither sampling-frequency."+*it+" nor sampling-frequency.default found.");
2501 return 2;
2502 }
2503
2504 const uint16_t frq = conf.GetDef<uint16_t>("sampling-frequency.", *it);
2505
2506 FTM::StaticData data;
2507 data.SetClockRegister(fClockCondSetup[frq].data());
2508
2509 // Trigger sequence ped:lp1:lp2
2510 // (data. is used here as an abbreviation for FTM::StaticData::
2511 if (!CheckConfigVal<bool> (conf, true, "trigger.enable-trigger.", *it) ||
2512 !CheckConfigVal<bool> (conf, true, "trigger.enable-external-1.", *it) ||
2513 !CheckConfigVal<bool> (conf, true, "trigger.enable-external-2.", *it) ||
2514 !CheckConfigVal<bool> (conf, true, "trigger.enable-veto.", *it) ||
2515 !CheckConfigVal<bool> (conf, true, "trigger.enable-clock-conditioner.", *it) ||
2516 !CheckConfigVal<bool> (conf, true, "light-pulser.external.enable-group1.", *it) ||
2517 !CheckConfigVal<bool> (conf, true, "light-pulser.external.enable-group2.", *it) ||
2518 !CheckConfigVal<bool> (conf, true, "light-pulser.internal.enable-group1.", *it) ||
2519 !CheckConfigVal<bool> (conf, true, "light-pulser.internal.enable-group2.", *it) ||
2520 !CheckConfigVal<uint16_t>(conf, data.kMaxSequence, "trigger.sequence.pedestal.", *it) ||
2521 !CheckConfigVal<uint16_t>(conf, data.kMaxSequence, "trigger.sequence.lp-ext.", *it) ||
2522 !CheckConfigVal<uint16_t>(conf, data.kMaxSequence, "trigger.sequence.lp-int.", *it) ||
2523 !CheckConfigVal<uint16_t>(conf, data.kMaxTriggerInterval, "trigger.sequence.interval.", *it) ||
2524 !CheckConfigVal<uint16_t>(conf, data.kMaxMultiplicity, "trigger.multiplicity-physics.", *it) ||
2525 !CheckConfigVal<uint16_t>(conf, data.kMaxMultiplicity, "trigger.multiplicity-calib.", *it) ||
2526 !CheckConfigVal<uint16_t>(conf, data.kMaxWindow, "trigger.coincidence-window-physics.", *it) ||
2527 !CheckConfigVal<uint16_t>(conf, data.kMaxWindow, "trigger.coincidence-window-calib.", *it) ||
2528 !CheckConfigVal<uint16_t>(conf, data.kMaxDeadTime, "trigger.dead-time.", *it) ||
2529 !CheckConfigVal<uint16_t>(conf, data.kMaxDelayTrigger, "trigger.delay.", *it) ||
2530 !CheckConfigVal<uint16_t>(conf, data.kMaxDelayTimeMarker, "trigger.time-marker-delay.", *it) ||
2531 !CheckConfigVal<uint16_t>(conf, 0xffff, "ftu-report-interval.", *it) ||
2532 !CheckConfigVal<uint16_t>(conf, data.kMaxIntensity, "light-pulser.external.intensity.", *it) ||
2533 !CheckConfigVal<uint16_t>(conf, data.kMaxIntensity, "light-pulser.internal.intensity.", *it) ||
2534 !CheckConfigVal<uint16_t>(conf, data.kMaxDAC, "trigger.threshold.patch.", *it) ||
2535 !CheckConfigVal<uint16_t>(conf, data.kMaxDAC, "trigger.threshold.logic.", *it) ||
2536 0)
2537 return 2;
2538
2539 data.Enable(data.kTrigger, conf.GetDef<bool>("trigger.enable-trigger.", *it));
2540 data.Enable(data.kExt1, conf.GetDef<bool>("trigger.enable-external-1.", *it));
2541 data.Enable(data.kExt2, conf.GetDef<bool>("trigger.enable-external-2.", *it));
2542 data.Enable(data.kVeto, conf.GetDef<bool>("trigger.enable-veto.", *it));
2543 data.Enable(data.kClockConditioner, conf.GetDef<bool>("trigger.enable-clock-conditioner.", *it));
2544
2545 data.EnableLPint(data.kGroup1, conf.GetDef<bool>("light-pulser.internal.enable-group1.", *it));
2546 data.EnableLPint(data.kGroup2, conf.GetDef<bool>("light-pulser.internal.enable-group2.", *it));
2547 data.EnableLPext(data.kGroup1, conf.GetDef<bool>("light-pulser.external.enable-group1.", *it));
2548 data.EnableLPext(data.kGroup2, conf.GetDef<bool>("light-pulser.external.enable-group2.", *it));
2549
2550 // [ms] Interval between two artificial triggers (no matter which type) minimum 1ms, 10 bit
2551 data.fIntensityLPint = conf.GetDef<uint16_t>("light-pulser.internal.intensity.", *it);
2552 data.fIntensityLPext = conf.GetDef<uint16_t>("light-pulser.external.intensity.", *it);
2553 data.fTriggerInterval = conf.GetDef<uint16_t>("trigger.sequence.interval.", *it);
2554 data.fMultiplicityPhysics = conf.GetDef<uint16_t>("trigger.multiplicity-physics.", *it);
2555 data.fMultiplicityCalib = conf.GetDef<uint16_t>("trigger.multiplicity-calib.", *it);
2556 data.fWindowPhysics = conf.GetDef<uint16_t>("trigger.coincidence-window-physics.", *it); /// (4ns * x + 8ns)
2557 data.fWindowCalib = conf.GetDef<uint16_t>("trigger.coincidence-window-calib.", *it); /// (4ns * x + 8ns)
2558 data.fDelayTrigger = conf.GetDef<uint16_t>("trigger.delay.", *it); /// (4ns * x + 8ns)
2559 data.fDelayTimeMarker = conf.GetDef<uint16_t>("trigger.time-marker-delay.", *it); /// (4ns * x + 8ns)
2560 data.fDeadTime = conf.GetDef<uint16_t>("trigger.dead-time.", *it); /// (4ns * x + 8ns)
2561
2562 data.SetPrescaling(conf.GetDef<uint16_t>("ftu-report-interval.", *it));
2563
2564 const uint16_t seqped = conf.GetDef<uint16_t>("trigger.sequence.pedestal.", *it);
2565 const uint16_t seqint = conf.GetDef<uint16_t>("trigger.sequence.lp-int.", *it);
2566 const uint16_t seqext = conf.GetDef<uint16_t>("trigger.sequence.lp-ext.", *it);
2567
2568 data.SetSequence(seqped, seqint, seqext);
2569
2570 data.EnableAllFTU();
2571 data.EnableAllPixel();
2572
2573 const vector<uint16_t> pat1 = conf.Vec<uint16_t>("trigger.disable-patch.default");
2574 const vector<uint16_t> pat2 = conf.Vec<uint16_t>("trigger.disable-patch."+*it);
2575
2576 const vector<uint16_t> pix1 = conf.Vec<uint16_t>("trigger.disable-pixel.default");
2577 const vector<uint16_t> pix2 = conf.Vec<uint16_t>("trigger.disable-pixel."+*it);
2578
2579 const vector<uint16_t> ftu1 = conf.Vec<uint16_t>("disable-ftu.default");
2580 const vector<uint16_t> ftu2 = conf.Vec<uint16_t>("disable-ftu."+*it);
2581
2582 vector<uint16_t> ftu, pat, pix;
2583 ftu.insert(ftu.end(), ftu1.begin(), ftu1.end());
2584 ftu.insert(ftu.end(), ftu2.begin(), ftu2.end());
2585 pat.insert(pat.end(), pat1.begin(), pat1.end());
2586 pat.insert(pat.end(), pat2.begin(), pat2.end());
2587 pix.insert(pix.end(), pix1.begin(), pix1.end());
2588 pix.insert(pix.end(), pix2.begin(), pix2.end());
2589
2590 for (vector<uint16_t>::const_iterator ip=ftu.begin(); ip!=ftu.end(); ip++)
2591 {
2592 if (*ip>FTM::StaticData::kMaxPatchIdx)
2593 {
2594 ostringstream str;
2595 str << "disable-ftu.*=" << *ip << " exceeds allowed maximum of " << FTM::StaticData::kMaxPatchIdx << "!";
2596 T::Error(str);
2597 return 2;
2598 }
2599 data.DisableFTU(*ip);
2600 }
2601 for (vector<uint16_t>::const_iterator ip=pat.begin(); ip!=pat.end(); ip++)
2602 {
2603 if (*ip>FTM::StaticData::kMaxPatchIdx)
2604 {
2605 ostringstream str;
2606 str << "trigger.disable-patch.*=" << *ip << " exceeds allowed maximum of " << FTM::StaticData::kMaxPatchIdx << "!";
2607 T::Error(str);
2608 return 2;
2609 }
2610 data.EnablePatch(*ip, false);
2611 }
2612 for (vector<uint16_t>::const_iterator ip=pix.begin(); ip!=pix.end(); ip++)
2613 {
2614 if (*ip>FTM::StaticData::kMaxPixelIdx)
2615 {
2616 ostringstream str;
2617 str << "trigger.disable-pixel.*=" << *ip << " exceeds allowed maximum of " << FTM::StaticData::kMaxPixelIdx << "!";
2618 T::Error(str);
2619 return 2;
2620 }
2621 data.EnablePixel(*ip, false);
2622 }
2623
2624 const uint16_t th0 = conf.GetDef<uint16_t>("trigger.threshold.patch.", *it);
2625 const uint16_t th1 = conf.GetDef<uint16_t>("trigger.threshold.logic.", *it);
2626
2627 for (int i=0; i<40; i++)
2628 {
2629 data[i].fDAC[0] = th0;
2630 data[i].fDAC[1] = th0;
2631 data[i].fDAC[2] = th0;
2632 data[i].fDAC[3] = th0;
2633 data[i].fDAC[4] = th1;
2634 }
2635
2636 fConfigs[*it] = data;
2637
2638 // trigger.threshold.dac-0:
2639
2640 /*
2641 threshold-A data[n].fDAC[0] = val
2642 threshold-B data[n].fDAC[1] = val
2643 threshold-C data[n].fDAC[2] = val
2644 threshold-D data[n].fDAC[3] = val
2645 threshold-H data[n].fDAC[4] = val
2646 */
2647
2648 // kMaxDAC = 0xfff,
2649 }
2650
2651 // FIXME: Add a check about unsused configurations
2652
2653 // ---------- FOR TESTING PURPOSE ---------
2654
2655 // fFTM.SetDefaultSetup(conf.Get<string>("default-setup"));
2656 fConfigs["test"] = FTM::StaticData();
2657
2658 // ---------- Setup connection endpoint ---------
2659 SetEndpoint(conf.Get<string>("addr"));
2660
2661 return -1;
2662 }
2663};
2664
2665// ------------------------------------------------------------------------
2666
2667#include "Main.h"
2668
2669template<class T, class S, class R>
2670int RunShell(Configuration &conf)
2671{
2672 return Main::execute<T, StateMachineFTM<S, R>>(conf);
2673}
2674
2675void SetupConfiguration(Configuration &conf)
2676{
2677 po::options_description control("Control options");
2678 control.add_options()
2679 ("no-dim", po_bool(), "Disable dim services")
2680 ("addr,a", var<string>("localhost:5000"), "Network address of FTM")
2681 ("quiet,q", po_bool(true), "Disable printing contents of all received messages (except dynamic data) in clear text.")
2682 ("hex-out", po_bool(), "Enable printing contents of all printed messages also as hex data.")
2683 ("dynamic-out", po_bool(), "Enable printing received dynamic data.")
2684// ("default-setup", var<string>(), "Binary file with static data loaded whenever a connection to the FTM was established.")
2685 ;
2686
2687 po::options_description freq("Sampling frequency setup");
2688 freq.add_options()
2689 ("clock-conditioner.frequency", vars<uint16_t>(), "Frequencies for which to setup the clock-conditioner (replace the * in the following options by this definition)")
2690 ("clock-conditioner.R0.*", var<Hex<uint32_t>>(), "Clock-conditioner R0")
2691 ("clock-conditioner.R1.*", var<Hex<uint32_t>>(), "Clock-conditioner R1")
2692 ("clock-conditioner.R8.*", var<Hex<uint32_t>>(), "Clock-conditioner R8")
2693 ("clock-conditioner.R9.*", var<Hex<uint32_t>>(), "Clock-conditioner R9")
2694 ("clock-conditioner.R11.*", var<Hex<uint32_t>>(), "Clock-conditioner R11")
2695 ("clock-conditioner.R13.*", var<Hex<uint32_t>>(), "Clock-conditioner R13")
2696 ("clock-conditioner.R14.*", var<Hex<uint32_t>>(), "Clock-conditioner R14")
2697 ("clock-conditioner.R15.*", var<Hex<uint32_t>>(), "Clock-conditioner R15");
2698
2699 po::options_description runtype("Run type configuration");
2700 runtype.add_options()
2701 ("run-type", vars<string>(), "Name of run-types (replace the * in the following configuration by the case-sensitive names defined here)")
2702 ("sampling-frequency.*", var<uint16_t>(), "Sampling frequency as defined in the clock-conditioner.frequency")
2703 ("trigger.enable-trigger.*", var<bool>(), "Enable trigger output of physics trigger")
2704 ("trigger.enable-external-1.*", var<bool>(), "Enable external trigger line 1")
2705 ("trigger.enable-external-2.*", var<bool>(), "Enable external trigger line 2")
2706 ("trigger.enable-veto.*", var<bool>(), "Enable veto line")
2707 ("trigger.enable-clock-conditioner.*", var<bool>(), "")
2708 ("trigger.sequence.interval.*", var<uint16_t>(), "Interval between two artifical triggers in units of n*4ns+8ns")
2709 ("trigger.sequence.pedestal.*", var<uint16_t>(), "Number of pedestal events in the sequence of artificial triggers")
2710 ("trigger.sequence.lp-int.*", var<uint16_t>(), "Number of LPint events in the sequence of artificial triggers")
2711 ("trigger.sequence.lp-ext.*", var<uint16_t>(), "Number of LPext events in the sequence of artificial triggers")
2712 ("trigger.multiplicity-physics.*", var<uint16_t>(), "Multiplicity for physics events (n out of 40)")
2713 ("trigger.multiplicity-calib.*", var<uint16_t>(), "Multiplicity for LPext events (n out of 40)")
2714 ("trigger.coincidence-window-physics.*", var<uint16_t>(), "Coincidence window for physics triggers in units of n*4ns+8ns")
2715 ("trigger.coincidence-window-calib.*", var<uint16_t>(), "Coincidence window for LPext triggers in units of n*4ns+8ns")
2716 ("trigger.dead-time.*", var<uint16_t>(), "Dead time after trigger in units of n*4ns+8ns")
2717 ("trigger.delay.*", var<uint16_t>(), "Delay of the trigger send to the FAD boards after a trigger in units of n*4ns+8ns")
2718 ("trigger.time-marker-delay.*", var<uint16_t>(), "Delay of the time-marker after a trigger in units of n*4ns+8ns")
2719 ("trigger.disable-pixel.*", vars<uint16_t>(), "")
2720 ("trigger.disable-patch.*", vars<uint16_t>(), "")
2721 ("trigger.threshold.patch.*", var<uint16_t>(), "")
2722 ("trigger.threshold.logic.*", var<uint16_t>(), "")
2723 ("ftu-report-interval.*", var<uint16_t>(), "")
2724 ("disable-ftu.*", var<uint16_t>(), "")
2725 ("light-pulser.external.enable-group1.*", var<bool>(), "Enable LED group 1 of external light pulser")
2726 ("light-pulser.external.enable-group2.*", var<bool>(), "Enable LED group 2 of external light pulser")
2727 ("light-pulser.internal.enable-group1.*", var<bool>(), "Enable LED group 1 of internal light pulser")
2728 ("light-pulser.internal.enable-group2.*", var<bool>(), "Enable LED group 2 of internal light pulser")
2729 ("light-pulser.external.intensity.*", var<uint16_t>(), "Intensity of external light pulser")
2730 ("light-pulser.internal.intensity.*", var<uint16_t>(), "Intensity of internal light pulser")
2731 ;
2732
2733 conf.AddOptions(control);
2734 conf.AddOptions(freq);
2735 conf.AddOptions(runtype);
2736}
2737
2738/*
2739 Extract usage clause(s) [if any] for SYNOPSIS.
2740 Translators: "Usage" and "or" here are patterns (regular expressions) which
2741 are used to match the usage synopsis in program output. An example from cp
2742 (GNU coreutils) which contains both strings:
2743 Usage: cp [OPTION]... [-T] SOURCE DEST
2744 or: cp [OPTION]... SOURCE... DIRECTORY
2745 or: cp [OPTION]... -t DIRECTORY SOURCE...
2746 */
2747void PrintUsage()
2748{
2749 cout <<
2750 "The ftmctrl controls the FTM (FACT Trigger Master) board.\n"
2751 "\n"
2752 "The default is that the program is started without user intercation. "
2753 "All actions are supposed to arrive as DimCommands. Using the -c "
2754 "option, a local shell can be initialized. With h or help a short "
2755 "help message about the usuage can be brought to the screen.\n"
2756 "\n"
2757 "Usage: ftmctrl [-c type] [OPTIONS]\n"
2758 " or: ftmctrl [OPTIONS]\n";
2759 cout << endl;
2760}
2761
2762void PrintHelp()
2763{
2764 Main::PrintHelp<StateMachineFTM<StateMachine, ConnectionFTM>>();
2765
2766 /* Additional help text which is printed after the configuration
2767 options goes here */
2768
2769 /*
2770 cout << "bla bla bla" << endl << endl;
2771 cout << endl;
2772 cout << "Environment:" << endl;
2773 cout << "environment" << endl;
2774 cout << endl;
2775 cout << "Examples:" << endl;
2776 cout << "test exam" << endl;
2777 cout << endl;
2778 cout << "Files:" << endl;
2779 cout << "files" << endl;
2780 cout << endl;
2781 */
2782}
2783
2784int main(int argc, const char* argv[])
2785{
2786 Configuration conf(argv[0]);
2787 conf.SetPrintUsage(PrintUsage);
2788 Main::SetupConfiguration(conf);
2789 SetupConfiguration(conf);
2790
2791 if (!conf.DoParse(argc, argv, PrintHelp))
2792 return 127;
2793
2794 //try
2795 {
2796 // No console access at all
2797 if (!conf.Has("console"))
2798 {
2799 if (conf.Get<bool>("no-dim"))
2800 return RunShell<LocalStream, StateMachine, ConnectionFTM>(conf);
2801 else
2802 return RunShell<LocalStream, StateMachineDim, ConnectionDimFTM>(conf);
2803 }
2804 // Cosole access w/ and w/o Dim
2805 if (conf.Get<bool>("no-dim"))
2806 {
2807 if (conf.Get<int>("console")==0)
2808 return RunShell<LocalShell, StateMachine, ConnectionFTM>(conf);
2809 else
2810 return RunShell<LocalConsole, StateMachine, ConnectionFTM>(conf);
2811 }
2812 else
2813 {
2814 if (conf.Get<int>("console")==0)
2815 return RunShell<LocalShell, StateMachineDim, ConnectionDimFTM>(conf);
2816 else
2817 return RunShell<LocalConsole, StateMachineDim, ConnectionDimFTM>(conf);
2818 }
2819 }
2820 /*catch (std::exception& e)
2821 {
2822 cerr << "Exception: " << e.what() << endl;
2823 return -1;
2824 }*/
2825
2826 return 0;
2827}
Note: See TracBrowser for help on using the repository browser.