source: trunk/FACT++/src/mcp.cc@ 13576

Last change on this file since 13576 was 13571, checked in by tbretz, 13 years ago
Added description to the new service.
File size: 17.2 KB
Line 
1#include "Dim.h"
2#include "Event.h"
3#include "Shell.h"
4#include "StateMachineDim.h"
5#include "Connection.h"
6#include "Configuration.h"
7#include "Console.h"
8#include "Converter.h"
9#include "DimServiceInfoList.h"
10
11#include "tools.h"
12
13#include "LocalControl.h"
14
15#include "HeadersFTM.h"
16#include "HeadersFAD.h"
17
18
19namespace ba = boost::asio;
20namespace bs = boost::system;
21namespace dummy = ba::placeholders;
22
23using namespace std;
24
25// ------------------------------------------------------------------------
26
27#include "DimDescriptionService.h"
28
29// ------------------------------------------------------------------------
30
31class StateMachineMCP : public StateMachineDim, public DimInfoHandler
32{
33 /*
34 int Wrap(boost::function<void()> f)
35 {
36 f();
37 return T::GetCurrentState();
38 }
39
40 boost::function<int(const EventImp &)> Wrapper(boost::function<void()> func)
41 {
42 return bind(&StateMachineMCP::Wrap, this, func);
43 }*/
44
45private:
46 enum states_t
47 {
48 kStateDimNetworkNA = 1,
49 kStateDisconnected,
50 kStateConnecting,
51 kStateConnected,
52 kStateIdle,
53 kStateReadyForDataTaking,
54 kStateConfiguring1,
55 kStateConfiguring2,
56 kStateConfiguring3,
57 kStateConfigured,
58// kStateRunInProgress,
59 };
60
61 DimServiceInfoList fNetwork;
62
63 pair<Time, int> fStatusDim;
64 pair<Time, int> fStatusFTM;
65 pair<Time, int> fStatusFAD;
66 pair<Time, int> fStatusLog;
67 pair<Time, int> fStatusRC;
68
69 DimStampedInfo fDim;
70 DimStampedInfo fFTM;
71 DimStampedInfo fFAD;
72 DimStampedInfo fLog;
73 DimStampedInfo fRC;
74
75 DimDescribedService fService;
76
77 pair<Time, int> GetNewState(DimStampedInfo &info) const
78 {
79 const bool disconnected = info.getSize()==0;
80
81 // Make sure getTimestamp is called _before_ getTimestampMillisecs
82 const int tsec = info.getTimestamp();
83 const int tms = info.getTimestampMillisecs();
84
85 return make_pair(Time(tsec, tms*1000),
86 disconnected ? -2 : info.getQuality());
87 }
88
89 void infoHandler()
90 {
91 DimInfo *curr = getInfo(); // get current DimInfo address
92 if (!curr)
93 return;
94
95 if (curr==&fFTM)
96 {
97 fStatusFTM = GetNewState(fFTM);
98 return;
99 }
100
101 if (curr==&fFAD)
102 {
103 fStatusFAD = GetNewState(fFAD);
104 return;
105 }
106
107 if (curr==&fLog)
108 {
109 fStatusLog = GetNewState(fLog);
110 return;
111 }
112
113 if (curr==&fRC)
114 {
115 fStatusRC = GetNewState(fRC);
116 return;
117 }
118
119 if (curr==&fDim)
120 {
121 fStatusDim = GetNewState(fDim);
122 fStatusDim.second = curr->getSize()==4 ? curr->getInt() : 0;
123 return;
124 }
125 }
126
127 bool CheckEventSize(size_t has, const char *name, size_t size)
128 {
129 if (has==size)
130 return true;
131
132 ostringstream msg;
133 msg << name << " - Received event has " << has << " bytes, but expected " << size << ".";
134 Fatal(msg);
135 return false;
136 }
137
138 int SetVerbosity(const EventImp &)
139 {
140 /*
141 if (!CheckEventSize(evt.GetSize(), "SetVerbosity", 1))
142 return T::kSM_FatalError;
143
144 fFSC.SetVerbose(evt.GetBool());
145
146 */
147
148 return GetCurrentState();
149 }
150
151
152 void PrintState(const pair<Time,int> &state, const char *server)
153 {
154 const State rc = fNetwork.GetState(server, state.second);
155
156 Out() << state.first.GetAsStr("%H:%M:%S.%f").substr(0, 12) << " - ";
157 Out() << kBold << server << ": ";
158 Out() << rc.name << "[" << rc.index << "]";
159 Out() << kReset << " - " << kBlue << rc.comment << endl;
160 }
161
162 int Print()
163 {
164 Out() << fStatusDim.first.GetAsStr("%H:%M:%S.%f").substr(0, 12) << " - ";
165 Out() << kBold << "DIM_DNS: ";
166 if (fStatusDim.second==0)
167 Out() << "Offline" << endl;
168 else
169 Out() << "V" << fStatusDim.second/100 << 'r' << fStatusDim.second%100 << endl;
170
171 PrintState(fStatusFTM, "FTM_CONTROL");
172 PrintState(fStatusFAD, "FAD_CONTROL");
173 PrintState(fStatusLog, "DATA_LOGGER");
174 PrintState(fStatusRC, "RATE_CONTROL");
175
176 return GetCurrentState();
177 }
178
179 int GetReady()
180 {
181
182 return GetCurrentState();
183 }
184
185 int StopRun(const EventImp &)
186 {
187 if (fStatusFTM.second==FTM::kTriggerOn)
188 {
189 Message("Stopping FTM");
190 Dim::SendCommand("FTM_CONTROL/STOP_TRIGGER");
191 }
192
193 // FIXME: Do step 2 only when FTM is stopped
194 if (fStatusFAD.second==FAD::kConnected)
195 {
196 //Dim::SendCommand("FAD_CONTROL/ENABLE_TRIGGER_LINE", bool(false));
197 Message("Stopping FAD");
198 Dim::SendCommand("FAD_CONTROL/ENABLE_CONTINOUS_TRIGGER", bool(false));
199 }
200
201 return GetCurrentState();
202 }
203
204 int Reset(const EventImp &)
205 {
206 fRunType = "";
207 Message("Reseting configuration states of FAD and FTM");
208 Dim::SendCommand("FTM_CONTROL/RESET_CONFIGURE");
209 Dim::SendCommand("FAD_CONTROL/RESET_CONFIGURE");
210 return kStateIdle;
211 /*
212 // FIMXE: Handle error states!
213 if (fStatusLog.second>=20)//kSM_NightlyOpen
214 Dim::SendCommand("DATA_LOGGER/STOP");
215
216 if (fStatusLog.second==0)
217 Dim::SendCommand("DATA_LOGGER/WAIT_FOR_RUN_NUMBER");
218
219 if (fStatusFAD.second==FAD::kConnected)
220 {
221 Dim::SendCommand("FAD_CONTROL/ENABLE_TRIGGER_LINE", bool(false));
222 Dim::SendCommand("FAD_CONTROL/ENABLE_CONTINOUS_TRIGGER", bool(false));
223 }
224
225 if (fStatusFTM.second==FTM::kTakingData)
226 Dim::SendCommand("FTM_CONTROL/STOP");
227
228 return GetCurrentState(); */
229 }
230
231 int64_t fMaxTime;
232 int64_t fNumEvents;
233 string fRunType;
234
235 int StartRun(const EventImp &evt)
236 {
237 if (fStatusFTM.second==-2)
238 {
239 Error("No connection to ftmcontrol (see PRINT).");
240 return GetCurrentState();
241 }
242 if (fStatusFAD.second==-2)
243 {
244 Warn("No connection to fadcontrol (see PRINT).");
245 return GetCurrentState();
246 }
247 if (fStatusLog.second==-2)
248 {
249 Warn("No connection to datalogger (see PRINT).");
250 return GetCurrentState();
251 }
252 if (fStatusRC.second==-2)
253 {
254 Warn("No connection to ratecontrol (see PRINT).");
255 return GetCurrentState();
256 }
257
258 fMaxTime = evt.Get<int64_t>();
259 fNumEvents = evt.Get<int64_t>(8);
260 fRunType = evt.Ptr<char>(16);
261
262 ostringstream str;
263 str << "Starting configuration '" << fRunType << "' for new run";
264 if (fNumEvents>0 || fMaxTime>0)
265 str << " [";
266 if (fNumEvents>0)
267 str << fNumEvents << " events";
268 if (fNumEvents>0 && fMaxTime>0)
269 str << " / ";
270 if (fMaxTime>0)
271 str << fMaxTime << "s";
272 if (fNumEvents>0 || fMaxTime>0)
273 str << "]";
274 Message(str);
275
276 Update(kStateConfiguring1);
277
278 return kStateConfiguring1;
279 }
280
281 struct Value
282 {
283 uint64_t time;
284 uint64_t nevts;
285 char type[];
286 };
287
288 Value *GetBuffer()
289 {
290 const size_t len = sizeof(Value)+fRunType.length()+1;
291
292 char *buf = new char[len];
293
294 Value *val = reinterpret_cast<Value*>(buf);
295
296 val->time = fMaxTime;
297 val->nevts = fNumEvents;
298
299 strcpy(val->type, fRunType.c_str());
300
301 return val;
302 }
303
304 void Update(int newstate)
305 {
306 Value *buf = GetBuffer();
307 fService.setQuality(newstate);
308 fService.setData(buf, sizeof(Value)+fRunType.length()+1);
309 fService.Update();
310 delete buf;
311 }
312
313 void ConfigureFAD()
314 {
315 Value *buf = GetBuffer();
316
317 Message("Configuring FAD");
318 Dim::SendCommand("FAD_CONTROL/CONFIGURE", buf, sizeof(Value)+fRunType.length()+1);
319
320 delete buf;
321 }
322
323 int Execute()
324 {
325 // Dispatch (execute) at most one handler from the queue. In contrary
326 // to run_one(), it doesn't wait until a handler is available
327 // which can be dispatched, so poll_one() might return with 0
328 // handlers dispatched. The handlers are always dispatched/executed
329 // synchronously, i.e. within the call to poll_one()
330 //poll_one();
331
332 if (fStatusDim.second==0)
333 return kStateDimNetworkNA;
334
335 if (fStatusFTM.second >= FTM::kConnected &&
336 fStatusFAD.second >= FAD::kConnected &&
337 fStatusLog.second >= kSM_Ready)
338 {
339 if (GetCurrentState()==kStateConfiguring1)
340 {
341 if (fStatusLog.second<30/*kSM_WaitForRun*/)
342 {
343 Message("Starting datalogger");
344 Dim::SendCommand("DATA_LOGGER/START_RUN_LOGGING");
345 }
346 Message("Configuring Trigger (FTM)");
347 Dim::SendCommand("FTM_CONTROL/CONFIGURE", fRunType);
348 Update(kStateConfiguring2);
349 return kStateConfiguring2;
350 }
351
352 if (GetCurrentState()==kStateConfiguring2)
353 {
354 // FIMXE: Reset in case of error
355 if ((/*fStatusFTM.second != FTM::kConfiguring2 &&*/
356 fStatusFTM.second != FTM::kConfigured) ||
357 fStatusLog.second<30 || fStatusLog.second>0xff)
358 return GetCurrentState();
359
360 // FIMXE: This is to make sure that the rate control
361 // has received the correct trigger setup already...
362 //usleep(1000000);
363
364 Message("Starting Rate Control");
365 Dim::SendCommand("RATE_CONTROL/CALIBRATE");
366
367 ConfigureFAD();
368 Update(kStateConfiguring3);
369 return kStateConfiguring3;
370 }
371
372 if (GetCurrentState()==kStateConfiguring3)
373 {
374 if (fStatusFTM.second != FTM::kConfigured ||
375 fStatusFAD.second != FAD::kConfigured ||
376 fStatusRC.second < 6)
377 return GetCurrentState();
378
379 Message("Starting Trigger (FTM)");
380 Dim::SendCommand("FTM_CONTROL/START_TRIGGER");
381 Update(kStateConfigured);
382 return kStateConfigured;
383 }
384
385 if (GetCurrentState()==kStateConfigured)
386 {
387 if (fStatusFTM.second != FTM::kTriggerOn)
388 return GetCurrentState();
389
390 Update(kStateIdle);
391
392 // Now we are taking data... we could now wait for
393 // the fad to go back to -1
394 }
395
396 return kStateIdle;
397 }
398
399 /*
400 if (fStatusFTM.second >= FTM::kConnected &&
401 fStatusFAD.second >= FAD::kConnected &&
402 fStatusLog.second >= kSM_Ready)
403 return kStateIdle;
404 */
405 if (fStatusFTM.second >-2 &&
406 fStatusFAD.second >-2 &&
407 fStatusLog.second >-2 &&
408 fStatusRC.second >-2)
409 return kStateConnected;
410
411 if (fStatusFTM.second >-2 ||
412 fStatusFAD.second >-2 ||
413 fStatusLog.second >-2 ||
414 fStatusRC.second >-2)
415 return kStateConnecting;
416
417 return kStateDisconnected;
418 }
419
420public:
421 StateMachineMCP(ostream &out=cout) : StateMachineDim(out, "MCP"),
422 fStatusDim(make_pair(Time(), -2)),
423 fStatusFTM(make_pair(Time(), -2)),
424 fStatusFAD(make_pair(Time(), -2)),
425 fStatusLog(make_pair(Time(), -2)),
426 fDim("DIS_DNS/VERSION_NUMBER", (void*)NULL, 0, this),
427 fFTM("FTM_CONTROL/STATE", (void*)NULL, 0, this),
428 fFAD("FAD_CONTROL/STATE", (void*)NULL, 0, this),
429 fLog("DATA_LOGGER/STATE", (void*)NULL, 0, this),
430 fRC("RATE_CONTROL/STATE", (void*)NULL, 0, this),
431 fService("CONFIGURATION", "X:1;X:1;C", "Run configuration information"
432 "|MaxTime[s]:Maximum time before the run gets stopped"
433 "|MaxEvents[num]:Maximum number of events before the run gets stopped"
434 "|Name[text]:Name of the chosen configuration")
435 {
436 // ba::io_service::work is a kind of keep_alive for the loop.
437 // It prevents the io_service to go to stopped state, which
438 // would prevent any consecutive calls to run()
439 // or poll() to do nothing. reset() could also revoke to the
440 // previous state but this might introduce some overhead of
441 // deletion and creation of threads and more.
442
443 // State names
444 AddStateName(kStateDimNetworkNA, "DimNetworkNotAvailable",
445 "DIM dns server not available.");
446
447 AddStateName(kStateDisconnected, "Disconnected",
448 "Neither ftmctrl, fadctrl, datalogger nor rate control online.");
449
450 AddStateName(kStateConnecting, "Connecting",
451 "Either ftmctrl, fadctrl, datalogger or rate control not online.");
452
453 AddStateName(kStateConnected, "Connected",
454 "All needed subsystems online.");
455
456 AddStateName(kStateIdle, "Idle",
457 ".");
458
459 AddStateName(kStateReadyForDataTaking, "ReadyForDataTaking",
460 ".");
461
462 AddStateName(kStateConfiguring1, "Configuring1",
463 ".");
464
465 AddStateName(kStateConfiguring2, "Configuring2",
466 "Waiting for FTM and Datalogger to get ready");
467
468 AddStateName(kStateConfiguring3, "Configuring3",
469 "Waiting for FAD to get ready");
470
471 AddStateName(kStateConfigured, "Configured",
472 ".");
473
474
475 AddEvent("START", "X:2;C")//, kStateIdle)
476 (bind(&StateMachineMCP::StartRun, this, placeholders::_1))
477 ("");
478
479 AddEvent("STOP")
480 (bind(&StateMachineMCP::StopRun, this, placeholders::_1))
481 ("");
482
483 AddEvent("RESET", kStateConfiguring1, kStateConfiguring2, kStateConfiguring3, kStateConfigured)
484 (bind(&StateMachineMCP::Reset, this, placeholders::_1))
485 ("");
486
487 // Verbosity commands
488 AddEvent("SET_VERBOSE", "B:1")
489 (bind(&StateMachineMCP::SetVerbosity, this, placeholders::_1))
490 ("set verbosity state"
491 "|verbosity[bool]:disable or enable verbosity for received data (yes/no), except dynamic data");
492
493 AddEvent("PRINT")
494 (bind(&StateMachineMCP::Print, this))
495 ("");
496 }
497
498 int EvalOptions(Configuration &)
499 {
500 //SetEndpoint(conf.Get<string>("addr"));
501
502 //fFSC.SetVerbose(!conf.Get<bool>("quiet"));
503
504 return -1;
505 }
506};
507
508// ------------------------------------------------------------------------
509
510#include "Main.h"
511
512template<class T>
513int RunShell(Configuration &conf)
514{
515 return Main::execute<T, StateMachineMCP>(conf);
516}
517
518/*
519 Extract usage clause(s) [if any] for SYNOPSIS.
520 Translators: "Usage" and "or" here are patterns (regular expressions) which
521 are used to match the usage synopsis in program output. An example from cp
522 (GNU coreutils) which contains both strings:
523 Usage: cp [OPTION]... [-T] SOURCE DEST
524 or: cp [OPTION]... SOURCE... DIRECTORY
525 or: cp [OPTION]... -t DIRECTORY SOURCE...
526 */
527void PrintUsage()
528{
529 cout <<
530 "The ftmctrl controls the FSC (FACT Slow Control) board.\n"
531 "\n"
532 "The default is that the program is started without user intercation. "
533 "All actions are supposed to arrive as DimCommands. Using the -c "
534 "option, a local shell can be initialized. With h or help a short "
535 "help message about the usuage can be brought to the screen.\n"
536 "\n"
537 "Usage: fscctrl [-c type] [OPTIONS]\n"
538 " or: fscctrl [OPTIONS]\n";
539 cout << endl;
540}
541
542void PrintHelp()
543{
544 Main::PrintHelp<StateMachineMCP>();
545
546 /* Additional help text which is printed after the configuration
547 options goes here */
548
549 /*
550 cout << "bla bla bla" << endl << endl;
551 cout << endl;
552 cout << "Environment:" << endl;
553 cout << "environment" << endl;
554 cout << endl;
555 cout << "Examples:" << endl;
556 cout << "test exam" << endl;
557 cout << endl;
558 cout << "Files:" << endl;
559 cout << "files" << endl;
560 cout << endl;
561 */
562}
563
564int main(int argc, const char* argv[])
565{
566 Configuration conf(argv[0]);
567 conf.SetPrintUsage(PrintUsage);
568 Main::SetupConfiguration(conf);
569
570 if (!conf.DoParse(argc, argv, PrintHelp))
571 return -1;
572
573 //try
574 {
575 // No console access at all
576 if (!conf.Has("console"))
577 {
578// if (conf.Get<bool>("no-dim"))
579// return RunShell<LocalStream, StateMachine, ConnectionFSC>(conf);
580// else
581 return RunShell<LocalStream>(conf);
582 }
583 // Cosole access w/ and w/o Dim
584/* if (conf.Get<bool>("no-dim"))
585 {
586 if (conf.Get<int>("console")==0)
587 return RunShell<LocalShell, StateMachine, ConnectionFSC>(conf);
588 else
589 return RunShell<LocalConsole, StateMachine, ConnectionFSC>(conf);
590 }
591 else
592*/ {
593 if (conf.Get<int>("console")==0)
594 return RunShell<LocalShell>(conf);
595 else
596 return RunShell<LocalConsole>(conf);
597 }
598 }
599 /*catch (std::exception& e)
600 {
601 cerr << "Exception: " << e.what() << endl;
602 return -1;
603 }*/
604
605 return 0;
606}
Note: See TracBrowser for help on using the repository browser.