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

Last change on this file since 13838 was 13838, checked in by tbretz, 13 years ago
Adapted to the changes in the StateMachineDim. Now (almost) all received services are processed synchronously with the commands and (more important) with the Execute function.
File size: 16.1 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#include "DimState.h"
29
30// ------------------------------------------------------------------------
31
32class StateMachineMCP : public StateMachineDim
33{
34private:
35 enum states_t
36 {
37 kStateDimNetworkNA = 1,
38 kStateDisconnected,
39 kStateConnecting,
40 kStateConnected,
41 kStateIdle,
42 kStateDummy, // Doesn't exist, kept to keep the numbers
43 kStateConfiguring1,
44 kStateConfiguring2,
45 kStateConfiguring3,
46 kStateConfigured,
47 kStateTriggerOn,
48 kStateTakingData,
49// kStateRunInProgress,
50 };
51
52 DimServiceInfoListImp fNetwork;
53
54 DimVersion fDim;
55 DimState fDimFTM;
56 DimState fDimFAD;
57 DimState fDimLog;
58 DimState fDimRC;
59
60 DimDescribedService fService;
61
62 const State GetState(const DimState &s) const
63 {
64 return fNetwork.GetState(s.name(), s.state());
65 }
66
67 void PrintState(const DimState &state) const
68 {
69 const State rc = GetState(state);
70
71 Out() << state.time().GetAsStr("%H:%M:%S.%f").substr(0, 12) << " - ";
72 Out() << kBold << state.name() << ": ";
73 if (rc.index==-3)
74 {
75 Out() << kReset << "Offline" << endl;
76 return;
77 }
78 if (rc.index==-2)
79 Out() << state.state();
80 else
81 Out() << rc.name << "[" << rc.index << "]";
82 Out() << kReset << " - " << kBlue << rc.comment << endl;
83 }
84
85 int Print() const
86 {
87 Out() << fDim.time().GetAsStr("%H:%M:%S.%f").substr(0, 12) << " - ";
88 Out() << kBold << "DIM_DNS: " << fDim.version() << endl;
89
90 PrintState(fDim);
91 PrintState(fDimFTM);
92 PrintState(fDimFAD);
93 PrintState(fDimLog);
94 PrintState(fDimRC);
95
96 return GetCurrentState();
97 }
98
99 int GetReady()
100 {
101 return GetCurrentState();
102 }
103
104 int StopRun(const EventImp &)
105 {
106 if (fDimFTM.state()==FTM::kTriggerOn)
107 {
108 Message("Stopping FTM");
109 Dim::SendCommand("FTM_CONTROL/STOP_TRIGGER");
110 }
111
112 // FIXME: Do step 2 only when FTM is stopped
113 if (fDimFAD.state()==FAD::kConnected)
114 {
115 //Dim::SendCommand("FAD_CONTROL/ENABLE_TRIGGER_LINE", bool(false));
116 Message("Stopping FAD");
117 Dim::SendCommand("FAD_CONTROL/ENABLE_CONTINOUS_TRIGGER", bool(false));
118 }
119
120 return GetCurrentState();
121 }
122
123 int Reset(const EventImp &)
124 {
125 fRunType = "";
126 Message("Reseting configuration states of FAD and FTM");
127 Dim::SendCommand("FTM_CONTROL/RESET_CONFIGURE");
128 Dim::SendCommand("FAD_CONTROL/RESET_CONFIGURE");
129 Update(kStateIdle);
130 return kStateIdle;
131 /*
132 // FIMXE: Handle error states!
133 if (fDimLog.state()>=20)//kSM_NightlyOpen
134 Dim::SendCommand("DATA_LOGGER/STOP");
135
136 if (fDimLog.state()==0)
137 Dim::SendCommand("DATA_LOGGER/WAIT_FOR_RUN_NUMBER");
138
139 if (fDimFAD.state()==FAD::kConnected)
140 {
141 Dim::SendCommand("FAD_CONTROL/ENABLE_TRIGGER_LINE", bool(false));
142 Dim::SendCommand("FAD_CONTROL/ENABLE_CONTINOUS_TRIGGER", bool(false));
143 }
144
145 if (fDimFTM.state()==FTM::kTakingData)
146 Dim::SendCommand("FTM_CONTROL/STOP");
147
148 return GetCurrentState(); */
149 }
150
151 int64_t fMaxTime;
152 int64_t fNumEvents;
153 string fRunType;
154
155 int StartRun(const EventImp &evt)
156 {
157 if (fDimFTM.state()==-2)
158 {
159 Error("No connection to ftmcontrol (see PRINT).");
160 return GetCurrentState();
161 }
162 if (fDimFAD.state()==-2)
163 {
164 Warn("No connection to fadcontrol (see PRINT).");
165 return GetCurrentState();
166 }
167 if (fDimLog.state()==-2)
168 {
169 Warn("No connection to datalogger (see PRINT).");
170 return GetCurrentState();
171 }
172 if (fDimRC.state()==-2)
173 {
174 Warn("No connection to ratecontrol (see PRINT).");
175 return GetCurrentState();
176 }
177
178 fMaxTime = evt.Get<int64_t>();
179 fNumEvents = evt.Get<int64_t>(8);
180 fRunType = evt.Ptr<char>(16);
181
182 ostringstream str;
183 str << "Starting configuration '" << fRunType << "' for new run";
184 if (fNumEvents>0 || fMaxTime>0)
185 str << " [";
186 if (fNumEvents>0)
187 str << fNumEvents << " events";
188 if (fNumEvents>0 && fMaxTime>0)
189 str << " / ";
190 if (fMaxTime>0)
191 str << fMaxTime << "s";
192 if (fNumEvents>0 || fMaxTime>0)
193 str << "]";
194 Message(str);
195
196 Update(kStateConfiguring1);
197
198 return kStateConfiguring1;
199 }
200
201 struct Value
202 {
203 uint64_t time;
204 uint64_t nevts;
205 char type[];
206 };
207
208 Value *GetBuffer()
209 {
210 const size_t len = sizeof(Value)+fRunType.length()+1;
211
212 char *buf = new char[len];
213
214 Value *val = reinterpret_cast<Value*>(buf);
215
216 val->time = fMaxTime;
217 val->nevts = fNumEvents;
218
219 strcpy(val->type, fRunType.c_str());
220
221 return val;
222 }
223
224 void Update(int newstate)
225 {
226 Value *buf = GetBuffer();
227 fService.setQuality(newstate);
228 fService.setData(buf, sizeof(Value)+fRunType.length()+1);
229 fService.Update();
230 delete buf;
231 }
232
233 void ConfigureFAD()
234 {
235 Value *buf = GetBuffer();
236
237 Message("Configuring FAD");
238 Dim::SendCommand("FAD_CONTROL/CONFIGURE", buf, sizeof(Value)+fRunType.length()+1);
239
240 delete buf;
241 }
242
243 int Execute()
244 {
245 // Dispatch (execute) at most one handler from the queue. In contrary
246 // to run_one(), it doesn't wait until a handler is available
247 // which can be dispatched, so poll_one() might return with 0
248 // handlers dispatched. The handlers are always dispatched/executed
249 // synchronously, i.e. within the call to poll_one()
250 //poll_one();
251
252 if (!fDim.online())
253 return kStateDimNetworkNA;
254
255 if (fDimFTM.state() >= FTM::kConnected &&
256 fDimFAD.state() >= FAD::kConnected &&
257 fDimLog.state() >= kSM_Ready)
258 {
259 if (GetCurrentState()==kStateConfiguring1)
260 {
261 if (fDimLog.state()<30/*kSM_WaitForRun*/)
262 {
263 Message("Starting datalogger");
264 Dim::SendCommand("DATA_LOGGER/START_RUN_LOGGING");
265 }
266 Message("Configuring Trigger (FTM)");
267 Dim::SendCommand("FTM_CONTROL/CONFIGURE", fRunType);
268 Update(kStateConfiguring2);
269 return kStateConfiguring2;
270 }
271
272 if (GetCurrentState()==kStateConfiguring2)
273 {
274 // FIMXE: Reset in case of error
275 if ((/*fDimFTM.state() != FTM::kConfiguring2 &&*/
276 fDimFTM.state() != FTM::kConfigured) ||
277 fDimLog.state()<30 || fDimLog.state()>0xff)
278 return GetCurrentState();
279
280 // FIMXE: This is to make sure that the rate control
281 // has received the correct trigger setup already...
282 //usleep(1000000);
283
284 Message("Starting Rate Control");
285 Dim::SendCommand("RATE_CONTROL/CALIBRATE");
286
287 ConfigureFAD();
288 Update(kStateConfiguring3);
289 return kStateConfiguring3;
290 }
291
292 if (GetCurrentState()==kStateConfiguring3)
293 {
294 if (fDimFTM.state() != FTM::kConfigured ||
295 fDimFAD.state() != FAD::kConfigured ||
296 fDimRC.state() < 6)
297 return GetCurrentState();
298
299 Message("Starting Trigger (FTM)");
300 Dim::SendCommand("FTM_CONTROL/START_TRIGGER");
301 Update(kStateConfigured);
302 return kStateConfigured;
303 }
304
305 if (GetCurrentState()==kStateConfigured)
306 {
307 if (fDimFTM.state() != FTM::kTriggerOn)
308 return GetCurrentState();
309
310 Update(kStateTriggerOn);
311
312 return kStateTriggerOn;
313 }
314
315 if (GetCurrentState()==kStateTriggerOn)
316 {
317 if (fDimFAD.state() != FAD::kWritingData)
318 return GetCurrentState();
319
320 Update(kStateTakingData);
321
322 return kStateTakingData;
323 }
324
325 if (GetCurrentState()==kStateTakingData)
326 {
327 if (fDimFTM.state()==FTM::kTriggerOn &&
328 fDimFAD.state()==FAD::kWritingData)
329 return kStateTakingData;
330
331 Update(kStateIdle);
332 }
333
334 return kStateIdle;
335 }
336
337 /*
338 if (fDimFTM.state() >= FTM::kConnected &&
339 fDimFAD.state() >= FAD::kConnected &&
340 fDimLog.state() >= kSM_Ready)
341 return kStateIdle;
342 */
343 if (fDimFTM.state() >-2 &&
344 fDimFAD.state() >-2 &&
345 fDimLog.state() >-2 &&
346 fDimRC.state() >-2)
347 return kStateConnected;
348
349 if (fDimFTM.state() >-2 ||
350 fDimFAD.state() >-2 ||
351 fDimLog.state() >-2 ||
352 fDimRC.state() >-2)
353 return kStateConnecting;
354
355 return kStateDisconnected;
356 }
357
358public:
359 StateMachineMCP(ostream &out=cout) : StateMachineDim(out, "MCP"),
360 fDimFTM("FTM_CONTROL"),
361 fDimFAD("FAD_CONTROL"),
362 fDimLog("DATA_LOGGER"),
363 fDimRC("RATE_CONTROL"),
364 fService("MCP/CONFIGURATION", "X:1;X:1;C", "Run configuration information"
365 "|MaxTime[s]:Maximum time before the run gets stopped"
366 "|MaxEvents[num]:Maximum number of events before the run gets stopped"
367 "|Name[text]:Name of the chosen configuration")
368 {
369 // ba::io_service::work is a kind of keep_alive for the loop.
370 // It prevents the io_service to go to stopped state, which
371 // would prevent any consecutive calls to run()
372 // or poll() to do nothing. reset() could also revoke to the
373 // previous state but this might introduce some overhead of
374 // deletion and creation of threads and more.
375
376 fDim.Subscribe(*this);
377 fDimFTM.Subscribe(*this);
378 fDimFAD.Subscribe(*this);
379 fDimLog.Subscribe(*this);
380 fDimRC.Subscribe(*this);
381
382 // State names
383 AddStateName(kStateDimNetworkNA, "DimNetworkNotAvailable",
384 "DIM dns server not available.");
385
386 AddStateName(kStateDisconnected, "Disconnected",
387 "Neither ftmctrl, fadctrl, datalogger nor rate control online.");
388
389 AddStateName(kStateConnecting, "Connecting",
390 "Either ftmctrl, fadctrl, datalogger or rate control not online.");
391
392 AddStateName(kStateConnected, "Connected",
393 "All needed subsystems online.");
394
395 AddStateName(kStateIdle, "Idle",
396 "Waiting for next configuration command");
397
398 AddStateName(kStateConfiguring1, "Configuring1",
399 "Starting configuration procedure, checking Datalogger state");
400
401 AddStateName(kStateConfiguring2, "Configuring2",
402 "Waiting for FTM and Datalogger to get ready");
403
404 AddStateName(kStateConfiguring3, "Configuring3",
405 "Waiting for FADs and rate control to get ready");
406
407 AddStateName(kStateConfigured, "Configured",
408 "Everything is configured, trigger will be switched on now");
409
410 AddStateName(kStateTriggerOn, "TriggerOn",
411 "The trigger is switched on, waiting for FAD to receive data");
412
413 AddStateName(kStateTakingData, "TakingData",
414 "The trigger is switched on, FADs are sending data");
415
416
417 AddEvent("START", "X:2;C")//, kStateIdle)
418 (bind(&StateMachineMCP::StartRun, this, placeholders::_1))
419 ("Start the configuration and data taking for a run-type of a pre-defined setup"
420 "|TimeMax[s]:Maximum number of seconds before the run will be closed automatically"
421 "|NumMax[count]:Maximum number events before the run will be closed automatically"
422 "|Name[text]:Name of the configuration to be used for taking data");
423
424 AddEvent("STOP")
425 (bind(&StateMachineMCP::StopRun, this, placeholders::_1))
426 ("Stops the trigger (either disables the FTM trigger or the internal DRS trigger)");
427
428 AddEvent("RESET", kStateConfiguring1, kStateConfiguring2, kStateConfiguring3, kStateConfigured)
429 (bind(&StateMachineMCP::Reset, this, placeholders::_1))
430 ("If a configuration blockes because a system cannot configure itself properly, "
431 "this command can be called to leave the configuration procedure. The command "
432 "is also propagated to FTM and FAD");
433
434 AddEvent("PRINT")
435 (bind(&StateMachineMCP::Print, this))
436 ("Print the states and connection status of all systems connected to the MCP.");
437 }
438
439 int EvalOptions(Configuration &)
440 {
441 return -1;
442 }
443};
444
445// ------------------------------------------------------------------------
446
447#include "Main.h"
448
449template<class T>
450int RunShell(Configuration &conf)
451{
452 return Main::execute<T, StateMachineMCP>(conf);
453}
454
455/*
456 Extract usage clause(s) [if any] for SYNOPSIS.
457 Translators: "Usage" and "or" here are patterns (regular expressions) which
458 are used to match the usage synopsis in program output. An example from cp
459 (GNU coreutils) which contains both strings:
460 Usage: cp [OPTION]... [-T] SOURCE DEST
461 or: cp [OPTION]... SOURCE... DIRECTORY
462 or: cp [OPTION]... -t DIRECTORY SOURCE...
463 */
464void PrintUsage()
465{
466 cout <<
467 "The ftmctrl controls the FSC (FACT Slow Control) board.\n"
468 "\n"
469 "The default is that the program is started without user intercation. "
470 "All actions are supposed to arrive as DimCommands. Using the -c "
471 "option, a local shell can be initialized. With h or help a short "
472 "help message about the usuage can be brought to the screen.\n"
473 "\n"
474 "Usage: fscctrl [-c type] [OPTIONS]\n"
475 " or: fscctrl [OPTIONS]\n";
476 cout << endl;
477}
478
479void PrintHelp()
480{
481 Main::PrintHelp<StateMachineMCP>();
482
483 /* Additional help text which is printed after the configuration
484 options goes here */
485
486 /*
487 cout << "bla bla bla" << endl << endl;
488 cout << endl;
489 cout << "Environment:" << endl;
490 cout << "environment" << endl;
491 cout << endl;
492 cout << "Examples:" << endl;
493 cout << "test exam" << endl;
494 cout << endl;
495 cout << "Files:" << endl;
496 cout << "files" << endl;
497 cout << endl;
498 */
499}
500
501int main(int argc, const char* argv[])
502{
503 Configuration conf(argv[0]);
504 conf.SetPrintUsage(PrintUsage);
505 Main::SetupConfiguration(conf);
506
507 if (!conf.DoParse(argc, argv, PrintHelp))
508 return -1;
509
510 //try
511 {
512 // No console access at all
513 if (!conf.Has("console"))
514 {
515// if (conf.Get<bool>("no-dim"))
516// return RunShell<LocalStream, StateMachine, ConnectionFSC>(conf);
517// else
518 return RunShell<LocalStream>(conf);
519 }
520 // Cosole access w/ and w/o Dim
521/* if (conf.Get<bool>("no-dim"))
522 {
523 if (conf.Get<int>("console")==0)
524 return RunShell<LocalShell, StateMachine, ConnectionFSC>(conf);
525 else
526 return RunShell<LocalConsole, StateMachine, ConnectionFSC>(conf);
527 }
528 else
529*/ {
530 if (conf.Get<int>("console")==0)
531 return RunShell<LocalShell>(conf);
532 else
533 return RunShell<LocalConsole>(conf);
534 }
535 }
536 /*catch (std::exception& e)
537 {
538 cerr << "Exception: " << e.what() << endl;
539 return -1;
540 }*/
541
542 return 0;
543}
Note: See TracBrowser for help on using the repository browser.