-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathlevbdim.html
1012 lines (972 loc) · 176 KB
/
levbdim.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!DOCTYPE html>
<html lang="en-US">
<head>
<meta http-equiv="content-type" content="text/html; charset=UTF-8" />
<meta name="generator" content="Madoko, version 1.0.2" />
<meta name="viewport" content="initial-scale=1.0" />
<meta name="author" content="Laurent Mirabito" />
<title>LEVBDIM: A Light EVent builder based on DIM</title>
<style type="text/css" class="link">
/*# sourceURL=madoko.css */
.madoko .toc>.tocblock .tocblock .tocblock {
margin-left: 2.25em;
}
.madoko .toc>.tocblock .tocblock {
margin-left: 1.5em;
}
.madoko .toc-contents>.tocblock>.tocitem {
font-weight: bold;
}
.madoko .toc {
margin-top: 1em;
}
.madoko p.para-continue {
margin-bottom: 0pt;
}
.madoko .para-block+p {
margin-top: 0pt;
}
.madoko ul.para-block, .madoko ol.para-block {
margin-top: 0pt;
margin-bottom: 0pt;
}
.madoko ul.para-end, .madoko ol.para-end {
margin-bottom: 1em;
}
.madoko dl {
margin-left: 0em;
}
.madoko blockquote {
font-style: italic;
}
.madoko a.localref {
text-decoration: none;
}
.madoko a.localref:hover {
text-decoration: underline;
}
.madoko .footnotes {
font-size: smaller;
margin-top: 2em;
}
.madoko .footnotes hr {
width: 50%;
text-align: left;
}
.madoko .footnote {
margin-left: 1em;
}
.madoko .footnote-before {
margin-left: -1em;
width: 1em;
display: inline-block;
}
.madoko .align-center, .madoko .align-center>p {
text-align: center !important;
}
.madoko .align-center pre {
text-align: left;
}
.madoko .align-center>* {
margin-left: auto !important;
margin-right: auto !important;
}
.madoko .align-left, .madoko .align-left>p {
text-align: left !important;
}
.madoko .align-left>* {
margin-left: 0pt !important;
margin-right: auto !important;
}
.madoko .align-right, .madoko .align-right>p {
text-align: right !important;
}
.madoko .align-right>* {
margin-left: auto !important;
margin-right: 0pt !important;
}
.madoko .align-center>table,
.madoko .align-left>table,
.madoko .align-right>table {
text-align: left !important;
}
.madoko .equation-before {
float: right;
}
.madoko .bibitem {
font-size: smaller;
}
.madoko .bibsearch {
font-size: x-small;
text-decoration:none;
color: black;
font-family: "Segoe UI Symbol", Symbola, serif;
}
.madoko .block, .madoko .figure, .madoko .bibitem, .madoko .equation, .madoko div.math {
margin-top: 1ex;
margin-bottom: 1ex;
}
.madoko .figure {
padding: 0.5em;
margin-left: 0pt;
margin-right: 0pt;
}
.madoko .hidden {
display: none;
}
.madoko .invisible {
visibility: hidden;
}
.madoko.preview .invisible {
visibility: visible;
opacity: 0.5;
}
.madoko code.code, .madoko span.code {
white-space: pre-wrap;
}
.madoko hr, hr.madoko {
border: none;
border-bottom: black solid 1px;
margin-bottom: 0.5ex;
}
.madoko .framed>*:first-child {
margin-top: 0pt;
}
.madoko .framed>*:last-child {
margin-bottom: 0pt;
}
.madoko ul.list-style-type-dash {
list-style-type: none !important;
}
.madoko ul.list-style-type-dash > li:before {
content: "\2013";
position: absolute;
margin-left: -1em;
}
.madoko table.madoko {
border-collapse: collapse;
}
.madoko td, .madoko th {
padding: 0ex 0.5ex;
margin: 0pt;
vertical-align: top;
}
.madoko .cell-border-left {
border-left: 1px solid black;
}
.madoko .cell-border-right {
border-right: 1px solid black;
}
.madoko thead>tr:first-child>.cell-line,
.madoko tbody:first-child>tr:first-child>.cell-line {
border-top: 1px solid black;
border-bottom: none;
}
.madoko .cell-line, .madoko .cell-double-line {
border-bottom: 1px solid black;
border-top: none;
}
.madoko .cell-double-line {
border-top: 1px solid black;
padding-top: 1.5px !important;
}
.madoko .input-mathpre .MathJax_Display {
text-align: left !important;
}
.madoko div.input-mathpre {
text-align: left;
margin-top: 1.5ex;
margin-bottom: 1ex;
}
.madoko .math-rendering {
color: gray;
}
.madoko .mathdisplay {
text-align: center;
}
.madoko .pretty table {
border-collapse: collapse;
}
.madoko .pretty td {
padding: 0em;
}
.madoko .pretty td.empty {
min-width: 1.5ex;
}
.madoko .pretty td.expander {
width: 100em;
}
body.madoko, .madoko .serif {
font-family: Cambria,"Times New Roman","Liberation Serif","Times",serif;
}
.madoko .sans-serif {
font-family: "Calibri", "Optima", sans-serif;
}
.madoko .symbol {
font-family: "Segoe UI Symbol", Symbola, serif;
}
body.madoko {
-webkit-text-size-adjust: 100%;
text-rendering: optimizeLegibility;
}
body.madoko {
max-width: 88ex;
margin: 1em auto;
padding: 0em 2em;
}
body.preview.madoko {
padding: 0em 1em;
}
.madoko p {
text-align: justify;
}
.madoko h1, .madoko h2, .madoko h3, .madoko h4 {
margin-top: 1.22em;
margin-bottom: 1ex;
}
.madoko h1+p, .madoko h2+p, .madoko h3+p, .madoko h4+p, .madoko h5+p {
margin-top: 1ex;
}
.madoko h5, .madoko h6 {
margin-top: 1ex;
font-size: 1em;
}
.madoko h5 {
margin-bottom: 0.5ex;
}
.madoko h5 + p {
margin-top: 0.5ex;
}
.madoko h6 {
margin-bottom: 0pt;
}
.madoko h6 + p {
margin-top: 0pt;
}
.madoko pre, .madoko code, .madoko kbd, .madoko samp, .madoko tt,
.madoko .monospace, .madoko .token-indent, .madoko .reveal pre, .madoko .reveal code, .madoko .email {
font-family: Consolas,"Andale Mono WT","Andale Mono",Lucida Console,Monaco,monospace,monospace;
font-size: 0.85em;
}
.madoko pre code, .madoko .token-indent {
font-size: 0.95em;
}
.madoko pre code {
font-family: inherit !important;
}
.madoko ol.linenums li {
background-color: white;
list-style-type: decimal;
}
.madoko .remote {
background-color: #F0FFF0;
}
.madoko .remote + * {
margin-top: 0pt;
}
@media print {
body.madoko {
font-size: 10pt;
}
@page {
margin: 1in 1.5in;
}
}
@media only screen and (max-device-width:1024px) {
body.madoko {
padding: 0em 0.5em;
}
.madoko li {
text-align: left;
}
}
</style>
</head>
<body class="madoko">
<div class="body madoko" style="line-adjust:0">
<div class="titleblock align-center para-block" style="text-align:center;line-adjust:0">
<div class="titleheader align-center" style="text-align:center;line-adjust:0">
<div class="title para-block" style="font-size:xx-large;font-weight:bold;margin-bottom:0.5ex;line-adjust:0">LEVBDIM: A Light EVent builder based on DIM</div>
<div class="titlenote para-block" style="line-adjust:0">2016-08-22</div></div>
<div class="authors align-center" style="text-align:center;width:80%;line-adjust:0"><table class="authorrow columns block" style="margin-top:2ex;width:100%;line-adjust:0">
<tbody><tr><td class="author column" style="text-align:center;line-adjust:0">
<div class="authorname" style="font-size:large;line-adjust:0">Laurent Mirabito</div></td></tr></tbody></table></div></div>
<div class="abstract" style="margin-left:3em;margin-right:3em;font-size:small">
<p class="p noindent"><strong class="strong-star2">Abstract.</strong> We developed a light data acquisition system, based on
DIM and mongoose-cpp frameworks. Providing binary data collection, events building, web accessible finite state machine and process control, it is well suit to manage distribute data source of laboratory or beam test. It supports only simple event building (one unique process) </p></div><h3 id="sec-introduction" class="h1" data-heading-depth="1" style="display:block"><span class="heading-before"><span class="heading-label">1</span>. </span>Introduction</h3>
<p class="p noindent">DIM <span class="citations" style="target-element:bibitem">[<a href="#dim" title="C Gaspar, DIM," class="bibref localref" style="target-element:bibitem"><span class="cite-number">1</span></a>]</span> is an HEP acquisition framework developed in the DELPHI experiment.
It embeds binary packet exchange in messages published by server and subscribed by client.
It is light and TCP/IP based and can be installed on nearly all currently used OS nowdays.
Mongoose-cpp is a tiny web server that is used to bind CGI commands to the DAQ application,
either in the finite state machine of the process or in standalone mode.
</p>
<p class="p indent">LEVBDIM is a simple acquisition framework where DIM is used to exchange structured binary buffer.
It provides several functionalities often used in modern data acquisition:
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">Event Building
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">Predefine structure of buffer
</li>
<li class="li ul-li list-star-li compact-li">Publication of data
</li>
<li class="li ul-li list-star-li compact-li">Collection of data from numerous publisher
</li>
<li class="li ul-li list-star-li compact-li">data writing
</li></ul></li>
<li class="li ul-li list-star-li compact-li">Run control
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">Generic <strong class="strong-star2">F</strong>inite <strong class="strong-star2">S</strong>tate <strong class="strong-star2">M</strong>achine
</li>
<li class="li ul-li list-star-li compact-li">Web access
</li></ul></li>
<li class="li ul-li list-star-li compact-li">Process control
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">Dedicated FSM used to create process on remote computers with a JSON<span class="citations" style="target-element:bibitem">[<a href="#json" title="JSON reference" class="bibref localref" style="target-element:bibitem"><span class="cite-number">5</span></a>]</span> description
of the environment.
</li></ul></li></ul>
<p class="p noindent">All those functionalities are mainly independent and can be used on their own.
The last section of this documentation details a full example using all the capabilities of LEVBDIM.
</p><h3 id="sec-installation" class="h1" data-heading-depth="1" style="display:block"><span class="heading-before"><span class="heading-label">2</span>. </span>Installation</h3><h4 id="sec-additional-packages" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">2.1</span>. </span>Additional packages</h4>
<p class="p noindent">Few standard packages are needed and can be found on any Linux distributions: boost, jsoncpp, curl, git and scons
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>sudo apt-get -y install libboost-dev libboost-system-dev
libboost-filesystem-dev libboost-thread-dev libjsoncpp-dev
libcurl4-gnutls-dev git scons
</code></pre>
<p class="p noindent para-continued">Two network libraries should be installed. The first one is DIM that can be download and compile from https://dim.web.cern.ch/dim/.
The second one is used to give web access to the application, it's Mongoose-cpp<span class="citations" style="target-element:bibitem">[<a href="#mongoose-cpp" title="https://github.com/Gregwar/mongoose-cpp" class="bibref localref" style="target-element:bibitem"><span class="cite-number">2</span></a>]</span> based on Mongoose <span class="citations" style="target-element:bibitem">[<a href="#mongoose" title="https://github.com/cesanta/mongoose" class="bibref localref" style="target-element:bibitem"><span class="cite-number">3</span></a>]</span>.
One version is distributed with <em class="em-low1">levbdim</em> and can be compiled before the installation.
</p><h4 id="sec-levbdim-installation" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">2.2</span>. </span>LEVBDIM installation</h4><h5 id="sec-getting-the-levbdim-software" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">2.2.1</span>. </span>Getting the levbdim software</h5>
<p class="p noindent">The software is distributed on GitHub. The master version is download with
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>git clone http://github.com/mirabitl/levbdim.git </code></pre><h5 id="sec-installing-the-software" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">2.2.2</span>. </span>Installing the software</h5><h6 id="sec-mongoose-cpp" class="h4" data-heading-depth="4" style="display:block">Mongoose-cpp</h6>
<p class="p noindent">A snapshot of mongoose is copied in the web directory
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>cd levbdim
source web/mongoose.install </code></pre>
<p class="p noindent para-continued">The default installation is in /opt/dhcal/levbdim directory.
</p><h6 id="sec-compiling-levbdim" class="h4" data-heading-depth="4" style="display:block">Compiling LEVBDIM</h6>
<p class="p noindent">The compilation is using <strong class="strong-star2">Scons</strong>
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>cd levbdim
scons </code></pre>
<p class="p noindent para-continued">The library <em class="em-low1">liblevbdim.so</em> is installed in the <em class="em-low1">lib</em> subdirectory.
</p><h3 id="sec-library-functionalities" class="h1" data-heading-depth="1" style="display:block"><span class="heading-before"><span class="heading-label">3</span>. </span>Library functionalities</h3><h4 id="sec-the-event-builder" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">3.1</span>. </span>The Event builder</h4>
<p class="p noindent">Event building consists of merging various data source that collect event fragment at the same time. Each data source should consequently have a localization tag and a time tag for each data fragment it provides. This fragment are published by a DimService and is centrally collected to build an event, i.e a collection of data fragment with an identical time tag.
</p>
<p class="p indent">The event builder proposed is not itself distributed, it is unique and so no geographical or time dependent partitioning is available. All data sources are collected by a unique process. It subscribes to all available data source and writes them on receipt in a shared memory. A separate thread scans the memory and build the event, i.e a collection of data buffer from each registered data source with the same time tag. A last thread process completed event and call registered <em class="em-low1">event processors</em> that can write event to disk in user defined format. The next sections detail the software tools provided to achieve those tasks.
</p><h5 id="sec-the-buffer-structure" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">3.1.1</span>. </span>The buffer structure</h5>
<p class="p noindent">The <em class="em-low1">levbdim::buffer</em> class is a simple data structure that provides space to store the read data (the <em class="em-low1">payload</em>) and four tags:
</p>
<ul class="ul list-star loose">
<li class="li ul-li list-star-li loose-li">
<p>The <em class="em-low1">Detector ID</em>: A four bytes id used to tag different geographical partition (ex. tracker barrel, ECAL end cap…)
</p></li>
<li class="li ul-li list-star-li loose-li">
<p>The <em class="em-low1">source ID</em>: A four bytes id characterizing the data source (ex. ADC or TDC number)
</p></li>
<li class="li ul-li list-star-li loose-li">
<p>The <em class="em-low1">event number</em>: A four bytes number of readouts
</p></li>
<li class="li ul-li list-star-li loose-li">
<p>The <em class="em-low1">bunch crossing number</em>: an eight bytes number characterizing the time of the event in clocks count. The clock is the typical clock of the experiment. It may allows a finer events building if needed. It si currently optional
</p></li>
<li class="li ul-li list-star-li loose-li">
<p>The <em class="em-low1">payload</em>: a bytes array containing detector data for the given <em class="em-low1">event number</em> readout.
</p></li>
<li class="li ul-li list-star-li loose-li">
<p>The <em class="em-low1">payload size</em>: the size of the <em class="em-low1">payload</em> array.
</p></li></ul>
<p class="p noindent">The payload can be compressed and inflate on the fly using the gzip<span class="citations" style="target-element:bibitem">[<a href="#gzip" title="gzip reference" class="bibref localref" style="target-element:bibitem"><span class="cite-number">4</span></a>]</span> library calls
(<em class="em-low1">compress,uncompress</em> methods).
</p>
<p class="p indent">Buffer do not need to be allocated by data producers since they are embedded in the <em class="em-low1">levbdim::datasource</em> class
described bellow.
</p><h5 id="sec-server-side--the-datasource-class" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">3.1.2</span>. </span>Server side: The datasource class</h5>
<p class="p noindent">A <em class="em-low1">levbdim::datasource</em> object instantiates a <em class="em-low1">levbdim::buffer</em> with a maximal size specified. It creates also
the associated DimService with the name <strong class="strong-star2">/FSM/LEVBDIM/DS-X-Y/DATA</strong> where X is the detector id and Y the data source one.
The buffer has method to access the payload in order to allow the user process to fill it.
</p>
<p class="p indent">The server process can instantiate any number of <em class="em-low1">levbdim::datasource</em> fill the associated buffers and publish them every time a new event is collected. The only requirement is that a DimServer is started at the initialization phase. The following pseudo code gives an example of usage.
</p><h6 id="sec-initialization" class="clearnum h4" data-heading-depth="4" style="display:block">Initialization</h6>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code> <span style="color:darkgreen">// Header</span>
<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasource*> _sources;
<span style="color:darkgreen">// Initialisation</span>
<span style="color:darkgreen">// Starting the DIM server</span>
<span style="color:navy">std::</span>stringstream s0;
s0.str(<span style="color:navy">std::</span>string());
s0<<<span style="color:maroon">"</span><span style="color:maroon">dummyServer-</span><span style="color:maroon">"</span><<name;
<span style="color:navy">DimServer::</span>start(s0.str().c_str());
<span style="color:darkgreen">// Configuration</span>
<span style="color:darkgreen">// Loop on data sources</span>
<span style="color:navy">std::</span>cout <<<span style="color:maroon">"</span><span style="color:maroon">Creating datasource Detector=</span><span style="color:maroon">"</span><<det<<<span style="color:maroon">"</span><span style="color:maroon"> SourceId=</span><span style="color:maroon">"</span><<sid<<<span style="color:navy">std::</span>endl;
<span style="color:darkgreen">// Maximum buffer size is set to 128 kBytes</span>
<span style="color:navy">levbdim::</span>datasource* ds= <span style="color:blue">new</span> <span style="color:navy">levbdim::</span>datasource(det,sid,<span class="constant" style="color:purple">0x20000</span>);
_sources.push_back(ds);</code></pre><h6 id="sec-readout-loop" class="clearnum h4" data-heading-depth="4" style="display:block">Readout loop</h6>
<p class="p noindent">This is an example with one thread per data source but all data source can be filled and publish sequentially if needed.
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code> <span style="color:darkgreen">// Create a readout thread per data source</span>
<span style="color:darkgreen">// calling the readdata method </span>
<span style="color:teal">void</span> <span style="color:navy">dummyServer::</span>start()
{
_running=<span style="color:blue">true</span>;
<span style="color:blue">for</span> (<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasource*>::iterator ids=_sources.begin();ids!=_sources.end();ids++)
{
_gthr.create_thread(<span style="color:navy">boost::</span>bind(&<span style="color:navy">dummyServer::</span>readdata, <span style="color:blue">this</span>,(*ids)));
::usleep(<span class="constant" style="color:purple">500000</span>);
}
}
<span style="color:darkgreen">// Loop on Events and publish data</span>
<span style="color:teal">void</span> <span style="color:navy">dummyServer::</span>readdata(<span style="color:navy">levbdim::</span>datasource *ds)
{
<span style="color:darkgreen">// Filling the buffer with random data</span>
<span style="color:darkgreen">// evt and bx are global data</span>
<span style="color:navy">std::</span>srand(<span style="color:navy">std::</span>time(<span class="constant" style="color:purple">0</span>));
<span style="color:blue">while</span> (_running)
{
::usleep(<span class="constant" style="color:purple">10000</span>);
<span style="color:blue">if</span> (!_running) <span style="color:blue">break</span>;
<span style="color:blue">if</span> (evt%<span class="constant" style="color:purple">100</span>==<span class="constant" style="color:purple">0</span>)
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Thread of </span><span style="color:maroon">"</span><<ds->buffer()->dataSourceId()<<<span style="color:maroon">"</span><span style="color:maroon"> is running </span><span style="color:maroon">"</span><<evt<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<_running<<<span style="color:navy">std::</span>endl;
<span style="color:darkgreen">// Just fun , ds is publishing a buffer containing sourceid X int of value sourceid</span>
uint32_t psi=ds->buffer()->dataSourceId();
uint32_t* pld=(uint32_t*) ds->payload();
<span style="color:blue">for</span> (<span style="color:teal">int</span> i=<span class="constant" style="color:purple">0</span>;i<psi;i++) pld[i]= <span style="color:navy">std::</span>rand();
pld[<span class="constant" style="color:purple">0</span>]=evt;
pld[psi-<span class="constant" style="color:purple">1</span>]=evt;
<span style="color:darkgreen">// publishing data of current event number evt and bunch crossing number bx</span>
ds->publish(evt,bx,psi*<span style="color:blue">sizeof</span>(uint32_t));
}
<span style="color:darkgreen">// Running is set false by the stop method , thread exits</span>
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Thread of </span><span style="color:maroon">"</span><<ds->buffer()->dataSourceId()<<<span style="color:maroon">"</span><span style="color:maroon"> is exiting</span><span style="color:maroon">"</span><<<span style="color:navy">std::</span>endl;
} </code></pre>
<p class="p noindent para-continued">As one can see the <em class="em-low1">levbdim::datasource</em> provides an easy way to store event data and publish them on the fly.
</p><h5 id="sec-client-side--the-datasocket-class" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">3.1.3</span>. </span>Client side: The datasocket class</h5>
<p class="p noindent">On client side the process should subscribe to the datasource published it wants to collect. It is achieved by using <em class="em-low1">levbdim::datasocket</em> class. It instantiates a <em class="em-low1">DimInfo</em> object that subscribe to the <em class="em-low1">DimService</em> named <strong class="strong-star2">/FSM/LEVBDIM/DS-X-Y/DATA</strong> published by the <em class="em-low1">levbdim::datasource</em>. It's a <em class="em-low1">DimClient</em> and it has an
<em class="em-low1">infoHandler</em> method that will collect the publish data. The data are then accessible in the <em class="em-low1">levbdim::buffer</em>. This method can also be overwritten to implement user treatment but the preferred usage is to save received data to <em class="em-low1">/dev/shm</em> so it can be used by the event builder class. This is achieved at initialization in an analog way to data sources
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code> <span style="color:navy">std::</span>cout <<<span style="color:maroon">"</span><span style="color:maroon">Creating datasocket </span><span style="color:maroon">"</span><<det<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<sid<<<span style="color:navy">std::</span>endl;
<span style="color:darkgreen">// data socket storing data in buffer of same size </span>
<span style="color:navy">levbdim::</span>datasocket* ds= <span style="color:blue">new</span> <span style="color:navy">levbdim::</span>datasocket(det,sid,<span class="constant" style="color:purple">0x20000</span>);
<span style="color:darkgreen">// Save data in share memory directory</span>
ds->save2disk(<span style="color:maroon">"</span><span style="color:maroon">/dev/shm/levbdim/</span><span style="color:maroon">"</span>);
_sources.push_back(ds);</code></pre>
<p class="p noindent para-continued">The needed directory <em class="em-low1">/dev/shm/levbdim</em> and <em class="em-low1">/dev/shm/levbim/closed</em> needed by the method are not created by the code and MUST be created by the user. Real disks directory can obviously be used but performances may be
deeply affected since <em class="em-low1">/dev/shm</em> is memory disk and can provide much faster access.
</p>
<p class="p indent">In this example, each <em class="em-low1">datasocket</em> object will write on reception a binary file named
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code> Event_#det_#source_#event_#bx</code></pre>
<p class="p noindent para-continued">in <em class="em-low1">/dev/shm/levbdim</em> containing the buffer. Once this one-event,one-source data are written, it creates an empty file with the same name in the <em class="em-low1">/dev/shm/levbdim/closed</em> directory. By scanning this later directory, the event builder will be able to read back the completed data in the memory files.
</p>
<p class="p indent">This mechanism decoupled the data reception from the event building.
</p><h5 id="sec-event-building--the-shmdriver-class" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">3.1.4</span>. </span>Event building: The shmdriver class</h5>
<p class="p noindent">Finally the event collection is done asynchronously by the <em class="em-low1">levbdim::shmdriver</em> class.
The class registers data source identified by their <em class="em-low1">detectorId</em> and their <em class="em-low1">dataSourceId</em> at initialization. It also registers <em class="em-low1">levbdim::shmprocessors</em> that will treat completed events. Two threads are then created:
</p>
<ol class="ol compact">
<li class="li ol-li compact-li">The first one is doing the following task:
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">listing the files in <em class="em-low1">/dev/shm/levbdim/closed</em> directory
</li>
<li class="li ul-li list-star-li compact-li">reading the corresponding files in <em class="em-low1">/dev/shm/levbdim/</em> directory. The data are stored in a map with the key being the event number, an <em class="em-low1">levbdim::buffer</em> is pushed in an associated vector for each data source read.
</li>
<li class="li ul-li list-star-li compact-li">erasing the read files in <em class="em-low1">/dev/shm/levbdim</em> and <em class="em-low1">closed</em> directory
</li></ul></li>
<li class="li ol-li compact-li">The second one is looping on the map entries, If one entry collected the exact number of data sources registered, the entry is processed and then removed from the map.
</li></ol>
<p class="p noindent">The <em class="em-low1">shmprocessor</em> class is a pure virtual class implementing only three methods:
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li"><em class="em-low1">void start(uint32_t run)</em> called at each new run start.
</li>
<li class="li ul-li list-star-li compact-li"><em class="em-low1">void stop()</em> called at end of run
</li>
<li class="li ul-li list-star-li compact-li"><em class="em-low1">void processEvent(uint32_t key,std::vector<levbdim::buffer*> dss)</em> called for each completed pair of event number and buffer list.
</li></ul>
<p class="p noindent">An example of processor is given in <em class="em-low1">levbdim::basicwriter</em> class which is writing data in a binary format. Many processors can be registered and will be called sequentially (data writing, monitoring, visualization…).
</p>
<p class="p indent">The main limitation of LEVBDIM is that this event building is unique and the computing not distributed. It is nevertheless possible to duplicate the event building process on a different computer (for cpu consuming task) but data from data sources are then sent twice and the bandwidth might be affected if the event sizes are large.
</p>
<p class="p indent">The following pseudo code implements an example of event builder:
</p>
<ul class="ul list-star loose">
<li class="li ul-li list-star-li loose-li">
<p><strong class="strong-star2">configure</strong>
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code> <span style="color:darkgreen">// Delete existing datasockets</span>
<span style="color:blue">for</span> (<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasocket*>::iterator it=_sources.begin();it!=_sources.end();it++)
<span style="color:blue">delete</span> (*it);
_sources.clear();
<span style="color:darkgreen">// Now create the builder</span>
_evb= <span style="color:blue">new</span> <span style="color:navy">levbdim::</span>shmdriver(<span style="color:maroon">"</span><span style="color:maroon">/dev/shm/levbdim</span><span style="color:maroon">"</span>);
_evb->createDirectories();
_evb->cleanShm();
<span style="color:darkgreen">// register a processor</span>
_writer= <span style="color:blue">new</span> <span style="color:navy">levbdim::</span>basicwriter(<span style="color:maroon">"</span><span style="color:maroon">/tmp</span><span style="color:maroon">"</span>);
_evb->registerProcessor(_writer);
<span style="color:darkgreen">// register datasockets</span>
<span style="color:darkgreen">// Loop on data source ids</span>
<span style="color:darkgreen">// ...</span>
<span style="color:navy">std::</span>cout <<<span style="color:maroon">"</span><span style="color:maroon">Creating datasocket </span><span style="color:maroon">"</span><<det<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<sid<<<span style="color:navy">std::</span>endl;
<span style="color:navy">levbdim::</span>datasocket* ds= <span style="color:blue">new</span> <span style="color:navy">levbdim::</span>datasocket(det,sid,<span class="constant" style="color:purple">0x20000</span>);
ds->save2disk(<span style="color:maroon">"</span><span style="color:maroon">/dev/shm/levbdim/</span><span style="color:maroon">"</span>);
_sources.push_back(ds);
_evb->registerDataSource(det,sid);</code></pre></li>
<li class="li ul-li list-star-li loose-li">
<p><strong class="strong-star2">start</strong>
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:darkgreen">// Starts the two shmdriver threads and pass the run number to the processors</span>
_evb->start(run);</code></pre></li>
<li class="li ul-li list-star-li loose-li">
<p><strong class="strong-star2">stop</strong>
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:darkgreen">// stop the threads</span>
_evb->stop();</code></pre><h4 id="sec-the-finite-state-machine" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">3.2</span>. </span>The Finite State machine</h4></li></ul>
<h5 id="sec-structure-of-the-fsm-mechanism" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">3.2.1</span>. </span>Structure of the FSM mechanism</h5>
<p class="p noindent">All data acquisition processes ( hardware readout, event builder, DB interfaces, Slow control…) may have different states corresponding to different phases of the read out process (OFF, READY, CONFIGURED, RUNNING,STOPPED…). It is therefore compulsory to have a software mechanism able to trigger the transition between the different states. A DIM-based implementation is done with the <em class="em-low1">levbdim::fsm</em> class that contains:
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">A vector of states names
</li>
<li class="li ul-li list-star-li compact-li">A map of command names associated to a vector of <em class="em-low1">levbdim::fsmTransition</em>
</li>
<li class="li ul-li list-star-li compact-li">A <em class="em-low1">DimRpc</em> object (<em class="em-low1">levbdim::rpcFsmMessage</em> class) handling the command reception and its processing
</li></ul>
<p class="p noindent">An <em class="em-low1">fsmTransition</em> is a class associating an initial state, a final state and a command handler. The command handler is a boost functor with a <em class="em-low1">levbdim::fsmmessage</em> parameter. As an example one can add a transition to the fsm in a <em class="em-low1">wummyServer</em> class with the following code
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code> fsm->addTransition(<span style="color:maroon">"</span><span style="color:maroon">START</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">RUNNING</span><span style="color:maroon">"</span>,<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>start,
<span style="color:blue">this</span>,_1));</code></pre>
<p class="p noindent para-continued">where
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">START is the command
</li>
<li class="li ul-li list-star-li compact-li">CONFIGURED and RUNNING are the initial and final state
</li>
<li class="li ul-li list-star-li compact-li"><em class="em-low1">start(levbdim::fsmmessage* m)</em> is a method of the <em class="em-low1">wummyServer</em> class
</li></ul>
<p class="p noindent">The <em class="em-low1">levbdim::fsmmessage</em> is a class handling a <em class="em-low1">Json::Value</em> with two attributes:
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li"><em class="em-low1">command</em> : the command name
</li>
<li class="li ul-li list-star-li compact-li"><em class="em-low1">content</em> : A possibly empty set of parameters
</li></ul>
<p class="p noindent">On return of the fsm handler one may add to the <em class="em-low1">content</em> set a Json::Value named <em class="em-low1">answer</em> with the
<em class="em-low1">setAnswer()</em> method.
</p>
<p class="p indent">On the client side an <em class="em-low1">levbdim::fsmClient</em> class is provided allowing the controlling program to call
the <em class="em-low1">execute(levbdim::fsmmessage* m)</em> method with the message containing the required command and
parameters.
</p>
<p class="p indent">This structure is performing but required that the run control is written using DIM Rpc implementation.
In order to permit much lighter clients we implements a web based version of the state machine.
</p><h5 id="sec-the-fsm-web" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">3.2.2</span>. </span>The FSM web</h5>
<p class="p noindent">The <em class="em-low1">levbdim::fsmweb</em> class is an overloading of the <em class="em-low1">fsm</em> class with a mongoose-cpp <span class="citations" style="target-element:bibitem">[<a href="#mongoose-cpp" title="https://github.com/Gregwar/mongoose-cpp" class="bibref localref" style="target-element:bibitem"><span class="cite-number">2</span></a>]</span> webserver on a port XXXX with three services:
</p>
<ol class="ol compact">
<li class="li ol-li compact-li"><strong class="strong-star2">htpp://mypc:XXXX/</strong> returns the list of possible Finite State Machine commands, the list of standalone commands and the <em class="em-low1">PREFIX</em> used, in a JSON file format.
</li>
<li class="li ol-li compact-li"><strong class="strong-star2">htpp://mypc:XXXX/PREFIX/FSM?command=NAME&content={…}</strong> calls the FSM transition NAME with the <em class="em-low1">content</em> parameters set. The answered <em class="em-low1">levbdim::fsmmessage</em> is published in JSON format
</li>
<li class="li ol-li compact-li">
<p><strong class="strong-star2">htpp://mypc:XXXX/PREFIX/CMD?name=NAME&toto={…}</strong> calls the boost callback registered in the <em class="em-low1">fsmweb</em> for the command named NAME.
</p>
<p>In the last case additional commands to the FSM ones can be registered with <em class="em-low1">addCommand</em> method:
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code>_fsm->addCommand(<span style="color:maroon">"</span><span style="color:maroon">DOWNLOAD</span><span style="color:maroon">"</span>,<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>download, <span style="color:blue">this</span>,_1,_2));</code></pre>
<p>will register the command named DOWNLOAD and link it to the boost functor. The method <em class="em-low1">download(Mongoose::Request &request, Mongoose::JsonResponse &response)</em> is called. The <em class="em-low1">request</em> parameter is an object embedding the CGI parameters, the <em class="em-low1">response</em> is a pure Json::value object that will be published as a JSON file on return.
</p>
<p>This approach allows to add commands not linked to state transition (monitoring,debug,..)
</p><h3 id="sec-full-example-of-an-fsm-based-process" class="h1" data-heading-depth="1" style="display:block"><span class="heading-before"><span class="heading-label">4</span>. </span>Full example of an FSM based process</h3><h4 id="sec-the-wummyserver-class" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">4.1</span>. </span>The wummyServer class</h4>The class <em class="em-low1">wummyServer</em> is a dummy data server that will publish periodically random data. Its prototype is the following
</li></ol>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:blue">class</span> wummyServer {
<span style="color:blue">public</span>:
wummyServer(<span style="color:navy">std::</span>string name,uint32_t port);
<span style="color:teal">void</span> configure(<span style="color:navy">levbdim::</span>fsmmessage* m);
<span style="color:teal">void</span> start(<span style="color:navy">levbdim::</span>fsmmessage* m);
<span style="color:teal">void</span> stop(<span style="color:navy">levbdim::</span>fsmmessage* m);
<span style="color:teal">void</span> halt(<span style="color:navy">levbdim::</span>fsmmessage* m);
<span style="color:teal">void</span> readdata(<span style="color:navy">levbdim::</span>datasource *ds);
<span style="color:teal">void</span> download(<span style="color:navy">Mongoose::</span><span style="color:teal">Request</span> &request, <span style="color:navy">Mongoose::</span><span style="color:teal">JsonResponse</span> &response);
<span style="color:teal">void</span> list(<span style="color:navy">Mongoose::</span><span style="color:teal">Request</span> &request, <span style="color:navy">Mongoose::</span><span style="color:teal">JsonResponse</span> &response);
<span style="color:teal">void</span> setEvent(uint32_t e) {_event=e;}
<span style="color:teal">void</span> setBx(uint64_t b) {_bx=b;}
uint32_t event(){<span style="color:blue">return</span> _event;}
uint64_t bx(){<span style="color:blue">return</span> _bx;}
<span style="color:blue">private</span>:
fsmweb* _fsm;
<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasource*> _sources;
<span style="color:teal">bool</span> _running,_readout;
<span style="color:navy">boost::</span>thread_group _gthr;
uint32_t _event;
uint64_t _bx;
}</code></pre><h6 id="sec-initialization" class="h4" data-heading-depth="4" style="display:block">Initialization</h6>
<p class="p noindent">The constructor instantiates the state machine, registers the states and start the Web and DIM services.
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:navy">wummyServer::</span>wummyServer(<span style="color:navy">std::</span>string name,uint32_t port) : _running(<span style="color:blue">false</span>)
{
_fsm=<span style="color:blue">new</span> fsmweb(name);
<span style="color:darkgreen">// Register state</span>
_fsm->addState(<span style="color:maroon">"</span><span style="color:maroon">CREATED</span><span style="color:maroon">"</span>);
_fsm->addState(<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>);
_fsm->addState(<span style="color:maroon">"</span><span style="color:maroon">RUNNING</span><span style="color:maroon">"</span>);
<span style="color:darkgreen">// Register allowed transition</span>
_fsm->addTransition(<span style="color:maroon">"</span><span style="color:maroon">CONFIGURE</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CREATED</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>,
<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>configure, <span style="color:blue">this</span>,_1));
_fsm->addTransition(<span style="color:maroon">"</span><span style="color:maroon">CONFIGURE</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>,
<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>configure, <span style="color:blue">this</span>,_1));
_fsm->addTransition(<span style="color:maroon">"</span><span style="color:maroon">START</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">RUNNING</span><span style="color:maroon">"</span>,
<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>start, <span style="color:blue">this</span>,_1));
_fsm->addTransition(<span style="color:maroon">"</span><span style="color:maroon">STOP</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">RUNNING</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>,
<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>stop, <span style="color:blue">this</span>,_1));
_fsm->addTransition(<span style="color:maroon">"</span><span style="color:maroon">HALT</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">RUNNING</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CREATED</span><span style="color:maroon">"</span>,
<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>halt, <span style="color:blue">this</span>,_1));
_fsm->addTransition(<span style="color:maroon">"</span><span style="color:maroon">HALT</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CONFIGURED</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">CREATED</span><span style="color:maroon">"</span>,
<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>halt, <span style="color:blue">this</span>,_1));
<span style="color:darkgreen">// Register additional commands</span>
_fsm->addCommand(<span style="color:maroon">"</span><span style="color:maroon">DOWNLOAD</span><span style="color:maroon">"</span>,<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>download, <span style="color:blue">this</span>,_1,_2));
_fsm->addCommand(<span style="color:maroon">"</span><span style="color:maroon">LIST</span><span style="color:maroon">"</span>,<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>list, <span style="color:blue">this</span>,_1,_2));
<span style="color:darkgreen">//Start Dim server</span>
<span style="color:navy">std::</span>stringstream s0;
s0.str(<span style="color:navy">std::</span>string());
s0<<<span style="color:maroon">"</span><span style="color:maroon">wummyServer-</span><span style="color:maroon">"</span><<name;
<span style="color:navy">DimServer::</span>start(s0.str().c_str());
<span style="color:darkgreen">// Start web services </span>
_fsm->start(port);
}</code></pre><h5 id="sec-handlers" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">4.1.1</span>. </span>Handlers</h5>
<p class="p noindent">There is then handlers declared for each FSM or standalone commands
</p><h6 id="sec-configure-transition" class="h4" data-heading-depth="4" style="display:block">Configure transition</h6>
<p class="p noindent">In this configure transition there is only the registration of the data sources. In real world it will also include the hardware configuration.
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">void</span> <span style="color:navy">wummyServer::</span>configure(<span style="color:navy">levbdim::</span>fsmmessage* m)
{
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Received command </span><span style="color:maroon">"</span><<m->command()<<
<span style="color:maroon">"</span><span style="color:maroon"> with parameters </span><span style="color:maroon">"</span><<m->value()<<<span style="color:navy">std::</span>endl;
<span style="color:darkgreen">// Delete existing datasources if any</span>
<span style="color:blue">for</span> (<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasource*>::iterator it=_sources.begin();it!=_sources.end();it++)
<span style="color:blue">delete</span> (*it);
_sources.clear();
<span style="color:darkgreen">// Add the data source</span>
<span style="color:darkgreen">// Parse the json message, the format used in the example is</span>
<span style="color:darkgreen">// {"command": "CONFIGURE", "content": {"detid": 100, "sourceid": [23, 24, 26]}}</span>
<span style="color:navy">Json::</span>Value jc=m->content();
int32_t det=jc[<span style="color:maroon">"</span><span style="color:maroon">detid</span><span style="color:maroon">"</span>].asInt();
<span style="color:blue">const</span> <span style="color:navy">Json::</span><span style="color:teal">Value</span>& dss = jc[<span style="color:maroon">"</span><span style="color:maroon">sourceid</span><span style="color:maroon">"</span>];
<span style="color:darkgreen">// Book a Json array to return the list of source booked</span>
<span style="color:navy">Json::</span>Value array_keys;
<span style="color:darkgreen">// Loop on sources</span>
<span style="color:blue">for</span> (<span style="color:navy">Json::</span>ValueConstIterator it = dss.begin(); it != dss.end(); ++it)
{
<span style="color:blue">const</span> <span style="color:navy">Json::</span><span style="color:teal">Value</span>& book = *it;
int32_t sid=(*it).asInt();
<span style="color:navy">std::</span>cout <<<span style="color:maroon">"</span><span style="color:maroon">Creating datasource </span><span style="color:maroon">"</span><<det<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<sid<<<span style="color:navy">std::</span>endl;
array_keys.append((det<<<span class="constant" style="color:purple">16</span>)|sid);
<span style="color:navy">levbdim::</span>datasource* ds= <span style="color:blue">new</span> <span style="color:navy">levbdim::</span>datasource(det,sid,<span class="constant" style="color:purple">0x20000</span>);
_sources.push_back(ds);
}
<span style="color:darkgreen">// Overwrite msg</span>
<span style="color:darkgreen">//Prepare complex answer</span>
m->setAnswer(array_keys);
}</code></pre><h6 id="sec-start-transition" class="h4" data-heading-depth="4" style="display:block">Start transition</h6>
<p class="p noindent">The handler is just starting a boost thread of data “reading” for each data source
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">void</span> <span style="color:navy">wummyServer::</span>start(<span style="color:navy">levbdim::</span>fsmmessage* m)
{
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Received </span><span style="color:maroon">"</span><<m->command()<<<span style="color:navy">std::</span>endl;
_running=<span style="color:blue">true</span>;
<span style="color:blue">for</span> (<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasource*>::iterator ids=_sources.begin();ids!=_sources.end();ids++)
{
_gthr.create_thread(<span style="color:navy">boost::</span>bind(&<span style="color:navy">wummyServer::</span>readdata, <span style="color:blue">this</span>,(*ids)));
::usleep(<span class="constant" style="color:purple">500000</span>);
}
}</code></pre>
<p class="p noindent para-continued">The <em class="em-low1">readdata</em> method is defined with the datasource argument. It's typically the method that is interfacing
the hardware readout of the event data.
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">void</span> <span style="color:navy">wummyServer::</span>readdata(<span style="color:navy">levbdim::</span>datasource *ds)
{
uint32_t last_evt=<span class="constant" style="color:purple">0</span>;
<span style="color:navy">std::</span>srand(<span style="color:navy">std::</span>time(<span class="constant" style="color:purple">0</span>));
<span style="color:blue">while</span> (_running)
{
::usleep(<span class="constant" style="color:purple">10000</span>);
<span style="color:darkgreen">// ::sleep(1);</span>
<span style="color:blue">if</span> (!_running) <span style="color:blue">break</span>;
<span style="color:darkgreen">// Update data only when _event number is modified</span>
<span style="color:blue">if</span> (_event == last_evt) <span style="color:blue">continue</span>;
<span style="color:blue">if</span> (evt%<span class="constant" style="color:purple">100</span>==<span class="constant" style="color:purple">0</span>)
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Thread of </span><span style="color:maroon">"</span><<ds->buffer()->dataSourceId()<<<span style="color:maroon">"</span><span style="color:maroon"> is running </span><span style="color:maroon">"</span><<
last_evt<<<span style="color:maroon">"</span><span style="color:maroon"> events, running status : </span><span style="color:maroon">"</span><<_running<<<span style="color:navy">std::</span>endl;
<span style="color:darkgreen">// Just fun , ds is publishing a buffer containing sourceid X int of value sourceid</span>
uint32_t psi=ds->buffer()->dataSourceId();
<span style="color:darkgreen">// update the payload</span>
uint32_t* pld=(uint32_t*) ds->payload();
<span style="color:blue">for</span> (<span style="color:teal">int</span> i=<span class="constant" style="color:purple">0</span>;i<psi;i++) pld[i]= <span style="color:navy">std::</span>rand();
pld[<span class="constant" style="color:purple">0</span>]=evt;
pld[psi-<span class="constant" style="color:purple">1</span>]=evt;
<span style="color:darkgreen">// publish data</span>
ds->publish(_event,_bx,psi*<span style="color:blue">sizeof</span>(uint32_t));
last_evt= _event;
}
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Thread of </span><span style="color:maroon">"</span><<ds->buffer()->dataSourceId()<<<span style="color:maroon">"</span><span style="color:maroon"> is exiting</span><span style="color:maroon">"</span><<<span style="color:navy">std::</span>endl;
}</code></pre><h6 id="sec-stop-transition" class="h4" data-heading-depth="4" style="display:block">Stop transition</h6>
<p class="p noindent">The stop transition is just stopping the reading threads without reconfiguring the process.
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">void</span> <span style="color:navy">wummyServer::</span>stop(<span style="color:navy">levbdim::</span>fsmmessage* m)
{
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Received </span><span style="color:maroon">"</span><<m->command()<<<span style="color:navy">std::</span>endl;
<span style="color:darkgreen">// Stop running</span>
_running=<span style="color:blue">false</span>;
::sleep(<span class="constant" style="color:purple">1</span>);
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">joining the threads</span><span style="color:maroon">"</span><<<span style="color:navy">std::</span>endl;
_gthr.join_all();
}</code></pre><h6 id="sec-halt-transition" class="h4" data-heading-depth="4" style="display:block">Halt transition</h6>
<p class="p noindent">In the halt transition, the datasources are removed and a new configuration is needed
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">void</span> <span style="color:navy">wummyServer::</span>halt(<span style="color:navy">levbdim::</span>fsmmessage* m)
{
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Received </span><span style="color:maroon">"</span><<m->command()<<<span style="color:navy">std::</span>endl;
<span style="color:blue">if</span> (_running)
<span style="color:blue">this</span>->stop(m);
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">Destroying data sources</span><span style="color:maroon">"</span><<<span style="color:navy">std::</span>endl;
<span style="color:blue">for</span> (<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasource*>::iterator it=_sources.begin();it!=_sources.end();it++)
<span style="color:blue">delete</span> (*it);
_sources.clear();
}</code></pre><h6 id="sec-standalone-commands" class="h4" data-heading-depth="4" style="display:block">Standalone commands</h6>
<p class="p noindent">They are dummy commands:
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">The DOWNLOAD can implement the download of database data for future configuration
</li></ul>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">void</span> <span style="color:navy">wummyServer::</span>download(<span style="color:navy">Mongoose::</span><span style="color:teal">Request</span> &request, <span style="color:navy">Mongoose::</span><span style="color:teal">JsonResponse</span> &response)
{
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">download </span><span style="color:maroon">"</span><<request.getUrl()<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<request.getMethod()
<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<request.getData()<<<span style="color:navy">std::</span>endl;
<span style="color:darkgreen">// Getting the data base name</span>
<span style="color:navy">std::</span>string state= request.get(<span style="color:maroon">"</span><span style="color:maroon">DBSTATE</span><span style="color:maroon">"</span>,<span style="color:maroon">"</span><span style="color:maroon">NONE</span><span style="color:maroon">"</span>)
<span style="color:darkgreen">// if (state.compare("NONE")!=0) do the download</span>
<span style="color:navy">std::</span>stringstream os;
os<<state<<<span style="color:maroon">"</span><span style="color:maroon"> has been download</span><span style="color:maroon">"</span>
response[<span style="color:maroon">"</span><span style="color:maroon">answer</span><span style="color:maroon">"</span>]=os.str();
}</code></pre>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">The LIST command can return the list of data sources
</li></ul>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">void</span> <span style="color:navy">wummyServer::</span>list(<span style="color:navy">Mongoose::</span><span style="color:teal">Request</span> &request, <span style="color:navy">Mongoose::</span><span style="color:teal">JsonResponse</span> &response)
{
<span style="color:navy">std::</span>cout<<<span style="color:maroon">"</span><span style="color:maroon">list </span><span style="color:maroon">"</span><<request.getUrl()<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<request.getMethod()
<<<span style="color:maroon">"</span><span style="color:maroon"> </span><span style="color:maroon">"</span><<request.getData()<<<span style="color:navy">std::</span>endl;
<span style="color:navy">Json::</span>Value array_keys;
<span style="color:blue">for</span> (<span style="color:navy">std::</span>vector<<span style="color:navy">levbdim::</span>datasource*>::iterator ids=_sources.begin();ids!=_sources.end();ids++)
array_keys.append(((*ids)->detectorId()<<<span class="constant" style="color:purple">16</span>)|(*ids)->dataSourceId());
response[<span style="color:maroon">"</span><span style="color:maroon">answer</span><span style="color:maroon">"</span>]=array_keys;
}</code></pre><h4 id="sec-main-program" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">4.2</span>. </span>Main program</h4>
<p class="p noindent">The main program just instantiate the wummyServer object. In order to emulate new event arrival the event number is updated every second
</p>
<pre class="para-block pre-fenced pre-fenced3 language-cpp lang-cpp cpp colorized" style="display:block"><code><span style="color:teal">int</span> main()
{
wummyServer s(<span style="color:maroon">"</span><span style="color:maroon">myfirsttry</span><span style="color:maroon">"</span>,<span class="constant" style="color:purple">45000</span>);
uint32_t evt=<span class="constant" style="color:purple">0</span>;
<span style="color:darkgreen">// Just update the event number to trigger data publication</span>
<span style="color:blue">while</span> (<span class="constant" style="color:purple">1</span>)
{
::sleep(<span class="constant" style="color:purple">1</span>);
s.setEvent(evt++);
}
} </code></pre><h3 id="sec-the-run-and-process-control" class="h1" data-heading-depth="1" style="display:block"><span class="heading-before"><span class="heading-label">5</span>. </span>The Run and process Control</h3><h4 id="sec-run-control" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">5.1</span>. </span>Run Control</h4>
<p class="p noindent">With the web implementation of the finite state machine, the run control is really eased and can be
written with very light tools like the <em class="em-low1">curl</em><span class="citations" style="target-element:bibitem">[<a href="#curl" title="curl ref" class="bibref localref" style="target-element:bibitem"><span class="cite-number">7</span></a>]</span> library or python. One can ofcourse use more user friendly interfaces like <em class="em-low1">Qt</em> <span class="citations" style="target-element:bibitem">[<a href="#qt" title="QT ref" class="bibref localref" style="target-element:bibitem"><span class="cite-number">8</span></a>]</span> or web based graphical interfaces like <em class="em-low1">Wt</em><span class="citations" style="target-element:bibitem">[<a href="#wt" title="Wt ref" class="bibref localref" style="target-element:bibitem"><span class="cite-number">9</span></a>]</span> or javascript pages, since all these frameworks have http request capabilities. We will give two examples in curl and python of the control of the <em class="em-low1">wummyServer</em> process.
</p><h5 id="sec-curl-example" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">5.1.1</span>. </span>curl example</h5>
<p class="p noindent">For example using <em class="em-low1">curl</em>, one can <strong class="strong-star2">configure</strong> the <em class="em-low1">wummyServer</em> process with:
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>curl http://mypc:45000/unessai/FSM?command=CONFIGURE\&content=%7B\"detid\":100,\"sourceid\":%5
B12,13,14%5D%7D</code></pre>
<p class="p noindent para-continued">and <strong class="strong-star2">start</strong> a run with
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>curl http://mypc:45000/unessai/FSM?command=START\&content=%7B%7D</code></pre>
<p class="p noindent para-continued">Access to the <strong class="strong-star2">download</strong> standalone command is also easy
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>curl http://mypc:45000/unessai/CMD?name=DOWNLOAD\&state=MyDbStateForMyDetetector</code></pre>
<p class="p noindent para-continued">The configuration and the control of all daq processes is then a simple bash script that can be run on any
computer where <em class="em-low1">curl</em> is installed.
</p><h5 id="sec-python-example" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">5.1.2</span>. </span>python example</h5>
<p class="p noindent">We used <em class="em-low1">urllib</em> and <em class="em-low1">urllib2</em> and <em class="em-low1">socket</em> package (SOCKS support) and define two methods:
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">FSM access
</li></ul>
<pre class="para-block pre-fenced pre-fenced3 language-python lang-python python colorized" style="display:block"><code><span style="color:blue">def</span> executeFSM(host,port,prefix,cmd,params)<span style="color:blue">:</span>
<span style="color:blue">if</span> (params!=<span style="color:navy">None</span>)<span style="color:blue">:</span>
<span style="color:darkgreen"># build the URL</span>
myurl <span style="color:blue">=</span> <span style="color:maroon">"</span><span style="color:maroon">http://</span><span style="color:maroon">"</span>+host+ <span style="color:maroon">"</span><span style="color:maroon">:%d</span><span style="color:maroon">"</span> % (port)
<span style="color:darkgreen"># parameter list</span>
lq<span style="color:blue">=</span>{}
<span style="color:darkgreen"># content take values from the params python list</span>
lq[<span style="color:maroon">"</span><span style="color:maroon">content</span><span style="color:maroon">"</span>]<span style="color:blue">=</span>json.dumps(params,sort_keys<span style="color:blue">=</span><span style="color:navy">True</span>)
lq[<span style="color:maroon">"</span><span style="color:maroon">command</span><span style="color:maroon">"</span>]<span style="color:blue">=</span>cmd
<span style="color:darkgreen"># encode it </span>
lqs<span style="color:blue">=</span>urllib.urlencode(lq)
<span style="color:darkgreen"># Build the final url</span>
saction <span style="color:blue">=</span> <span style="color:maroon">'</span><span style="color:maroon">/%s/FSM?%s</span><span style="color:maroon">'</span> % (prefix,lqs)
myurl<span style="color:blue">=</span>myurl+saction
<span style="color:darkgreen"># send the command</span>
req<span style="color:blue">=</span>urllib2.<span style="color:navy">Request</span>(myurl)
r1<span style="color:blue">=</span>urllib2.urlopen(req)
<span style="color:blue">return</span> r1.read()</code></pre>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">Command access
</li></ul>
<pre class="para-block pre-fenced pre-fenced3 language-python lang-python python colorized" style="display:block"><code><span style="color:blue">def</span> executeCMD(host,port,prefix,cmd,params)<span style="color:blue">:</span>
<span style="color:blue">if</span> (params!=<span style="color:navy">None</span> <span style="color:blue">and</span> cmd!=<span style="color:navy">None</span>)<span style="color:blue">:</span>
myurl <span style="color:blue">=</span> <span style="color:maroon">"</span><span style="color:maroon">http://</span><span style="color:maroon">"</span>+host+ <span style="color:maroon">"</span><span style="color:maroon">:%d</span><span style="color:maroon">"</span> % (port)
<span style="color:darkgreen"># CGI parameter list</span>
lq<span style="color:blue">=</span>{}
lq[<span style="color:maroon">"</span><span style="color:maroon">name</span><span style="color:maroon">"</span>]<span style="color:blue">=</span>cmd
<span style="color:blue">for</span> x,y <span style="color:blue">in</span> params.iteritems()<span style="color:blue">:</span>
lq[x]<span style="color:blue">=</span>y
<span style="color:darkgreen"># build the list</span>
lqs<span style="color:blue">=</span>urllib.urlencode(lq)
saction <span style="color:blue">=</span> <span style="color:maroon">'</span><span style="color:maroon">/%s/CMD?%s</span><span style="color:maroon">'</span> % (prefix,lqs)
myurl<span style="color:blue">=</span>myurl+saction
req<span style="color:blue">=</span>urllib2.<span style="color:navy">Request</span>(myurl)
<span style="color:darkgreen"># Check the server is alived</span>
<span style="color:blue">try</span><span style="color:blue">:</span>
r1<span style="color:blue">=</span>urllib2.urlopen(req)
<span style="color:blue">except</span> <span style="color:navy">URLError</span>, e<span style="color:blue">:</span>
p_rep<span style="color:blue">=</span>{}
p_rep[<span style="color:maroon">"</span><span style="color:maroon">STATE</span><span style="color:maroon">"</span>]<span style="color:blue">=</span><span style="color:maroon">"</span><span style="color:maroon">DEAD</span><span style="color:maroon">"</span>
<span style="color:blue">return</span> json.dumps(p_rep,sort_keys<span style="color:blue">=</span><span style="color:navy">True</span>)
<span style="color:blue">else</span><span style="color:blue">:</span>
<span style="color:blue">return</span> r1.read()
<span style="color:blue">else</span><span style="color:blue">:</span>
<span style="color:darkgreen"># no additional parameters build the query </span>
myurl <span style="color:blue">=</span> <span style="color:maroon">"</span><span style="color:maroon">http://</span><span style="color:maroon">"</span>+host+ <span style="color:maroon">"</span><span style="color:maroon">:%d/%s/</span><span style="color:maroon">"</span> % (port,prefix)
req<span style="color:blue">=</span>urllib2.<span style="color:navy">Request</span>(myurl)
<span style="color:darkgreen"># Check the server is alived</span>
<span style="color:blue">try</span><span style="color:blue">:</span>
r1<span style="color:blue">=</span>urllib2.urlopen(req)
<span style="color:blue">except</span> <span style="color:navy">URLError</span>, e<span style="color:blue">:</span>
p_rep<span style="color:blue">=</span>{}
p_rep[<span style="color:maroon">"</span><span style="color:maroon">STATE</span><span style="color:maroon">"</span>]<span style="color:blue">=</span><span style="color:maroon">"</span><span style="color:maroon">DEAD</span><span style="color:maroon">"</span>
<span style="color:blue">return</span> json.dumps(p_rep,sort_keys<span style="color:blue">=</span><span style="color:navy">True</span>)
<span style="color:blue">else</span><span style="color:blue">:</span>
<span style="color:blue">return</span> r1.read()</code></pre>
<p class="p noindent para-continued">Using those two methods one can build any request to a <em class="em-low1">fsmweb</em> based process. The usage of python allows the developers to build a much more complex code architecture and benefit from a rich library resources (XML,JSON configuration for example).
</p><h4 id="sec-large-architecture-approach" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">5.2</span>. </span>Large architecture approach</h4>
<p class="p noindent">The LEVBDIM package was developed to read the Semi Digital HCAL of the Calice collaboration<span class="citations" style="target-element:bibitem">[<a href="#calice" title="CALICE ref" class="bibref localref" style="target-element:bibitem"><span class="cite-number">6</span></a>]</span>. It handles a large number of processes for readout (14), database access (1), trigger control (2), low voltage control (1) and event building(1). The configuration sequence is complex and can be coded in python but the control is then bind to a single python process and a single user. We preferred to have a hierarchical approach and created an upper <em class="em-low1">levbdim::fsmweb</em> process (<em class="em-low1">WDaqServer</em>) with a relatively simple finite state machine (INITIALISE,CONFIGURE,START,STOP,DESTROY). This process implements hierarchical (or parallel for readout) lower level FSM transitions via web or DIM message calls. It's a intrinsically a web process and can be accessed remotely by various client.
</p>
<p class="p indent">The most user-friendly client is a <em class="em-low1">Wt</em> based web service that is accessing the <em class="em-low1">WDaq</em> server. Several clients can be connected and the processing is driven by the <em class="em-low1">WDaq</em> FSM.
</p><h4 id="sec-the-process-management" class="h2" data-heading-depth="2" style="display:block"><span class="heading-before"><span class="heading-label">5.3</span>. </span>The process management</h4>
<p class="p noindent">One last requirement of a distributed acquisition system is the management of the different processes (start,kill,restart). Heavily inspired by the <em class="em-low1">XDAQ</em> <span class="citations" style="target-element:bibitem">[<a href="#xdaq" title="XDAQ ref" class="bibref localref" style="target-element:bibitem"><span class="cite-number">10</span></a>]</span> framework jobcontrol, we developed one application <em class="em-low1">levbdim::fsmjob</em> that is started on every computer used in the Daq. On request to this application a list of processes are forked and the specified programs are created (or killed/ restarted). The child processes are writing in PID identified log that can be requested to the application. The management of the <em class="em-low1">fsmjob</em> itself can be done using Linux daemon mechanism that will ensure that it is started during the computer boot sequence.
</p><h5 id="sec-process-description" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">5.3.1</span>. </span>Process description</h5>
<p class="p noindent">The process are described with the following structure:
</p>
<pre class="para-block pre-fenced pre-fenced3 language-javascript lang-javascript javascript colorized" style="display:block"><code> {
<span style="color:maroon">"</span><span style="color:maroon">NAME</span><span style="color:maroon">"</span>:<span style="color:maroon">"</span><span style="color:maroon">WRITER</span><span style="color:maroon">"</span>,
<span style="color:maroon">"</span><span style="color:maroon">ARGS</span><span style="color:maroon">"</span> : [<span style="color:maroon">"</span><span style="color:maroon">-d /data/NAS/oyonax</span><span style="color:maroon">"</span>],
<span style="color:maroon">"</span><span style="color:maroon">ENV</span><span style="color:maroon">"</span> : [
<span style="color:maroon">"</span><span style="color:maroon">DIM_DNS_NODE=lyosdhcal9</span><span style="color:maroon">"</span>,<span style="color:red">"LD_LIBRARY_PATH=/usr/lib:/usr/local/lib:</span>
<span style="color:maroon">/</span><span style="color:maroon">o</span><span style="color:maroon">p</span><span style="color:maroon">t</span><span style="color:maroon">/</span>dhcal<span style="color:maroon">/</span><span style="color:maroon">l</span><span style="color:maroon">e</span><span style="color:maroon">v</span><span style="color:maroon">b</span><span style="color:maroon">d</span><span style="color:maroon">i</span><span style="color:maroon">m</span><span style="color:maroon">/</span>lib:<span style="color:maroon">/</span><span style="color:maroon">o</span><span style="color:maroon">p</span><span style="color:maroon">t</span><span style="color:maroon">/</span>dhcal<span style="color:maroon">/</span><span style="color:maroon">l</span><span style="color:maroon">i</span><span style="color:maroon">b</span><span style="color:maroon">:</span><span style="color:maroon">/</span>opt<span style="color:maroon">/</span><span style="color:maroon">d</span><span style="color:maroon">h</span><span style="color:maroon">c</span><span style="color:maroon">a</span><span style="color:maroon">l</span><span style="color:maroon">/</span>root/lib:
<span style="color:maroon">/</span><span style="color:maroon">o</span><span style="color:maroon">p</span><span style="color:maroon">t</span><span style="color:maroon">/</span>dhcal<span style="color:maroon">/</span><span style="color:maroon">d</span><span style="color:maroon">i</span><span style="color:maroon">m</span><span style="color:maroon">/</span>linux:<span style="color:maroon">/</span><span style="color:maroon">o</span><span style="color:maroon">p</span><span style="color:maroon">t</span><span style="color:maroon">/</span>dhcal<span style="color:maroon">/</span><span style="color:maroon">l</span><span style="color:maroon">c</span><span style="color:maroon">i</span><span style="color:maroon">o</span><span style="color:maroon">/</span>v02-<span class="constant" style="color:purple">00</span><span style="color:maroon">/</span><span style="color:maroon">l</span><span style="color:maroon">i</span><span style="color:maroon">b</span><span style="color:maroon">/</span>:$LD_LIBRARY_PATH<span style="color:red">"</span>
],
<span style="color:maroon">"</span><span style="color:maroon">PROGRAM</span><span style="color:maroon">"</span> : <span style="color:maroon">"</span><span style="color:maroon">/opt/dhcal/bin/levbdim_build</span><span style="color:maroon">"</span>
}</code></pre>
<p class="p noindent para-continued">where
</p>
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li">NAME is a user given name of the process. Since sevral jobs can be started one the same computer the name has to be unique on agiven computer
</li>
<li class="li ul-li list-star-li compact-li">PROGRAM is the executable
</li>
<li class="li ul-li list-star-li compact-li">ARGS is an array of argument to the executable
</li>
<li class="li ul-li list-star-li compact-li">ENV is an array of environment variables to be set when the program runs.
</li></ul>
<p class="p noindent">The state machine of the <em class="em-low1">fsmjob</em> is described in figure <a href="#fig-fsmjob" title="Finite State machine of the levbdim::fsmjob" class="localref" style="target-element:figure"><span class="figure-label">1</span></a>. All jobs on a givern PC must be registered. The first possibility is to send an INITIALISE command with a local <em class="em-low1">file</em> name or an <em class="em-low1">url</em> name in the <em class="em-low1">content</em> argument. The file is read or load and parsed. The jobs corresponding to the computer name are registered. An example of the file structure is given here:
</p>
<pre class="para-block pre-fenced pre-fenced3" style="display:block"><code>{
"HOSTS" :
{
"lyosdhcal9":[
{
"NAME":"WRITER",
"ARGS" : ["-d /data/NAS/oyonax"],
"ENV" : [
"DIM_DNS_NODE=lyosdhcal9",
"LD_LIBRARY_PATH=/usr/lib:/usr/local/lib:
/opt/dhcal/levbdim/lib:/opt/dhcal/lib:
/opt/dhcal/root/lib:$LD_LIBRARY_PATH"
],
"PROGRAM" : "/opt/dhcal/bin/levbdim_build"
},
{
"NAME":"WIENER",...
},
{
"NAME":"DBSERVER",
"ARGS" : [],...
}
],
"lyoilcrpi17":[
{ "NAME":"MDCSERVER",
"ARGS" : [],...
}
]
}</code></pre>
<p class="p noindent para-continued">It's a map of HOSTS (computer's names) containing a list of processes with the previously described process's structure.
In order to use this initialisation, the file should be copied locally on each computer or put in an accessible web page.
</p>
<p class="p indent">Another possibility is to use the REGISTERJOB transition and to register each job independantly:
</p>
<ol class="ol compact">
<li class="li ol-li compact-li">Send a REGISTRATION command. It clears previous job list and prepare a map of jobs
</li>
<li class="li ol-li compact-li">Send any number of REGISTERJOB commands with parameters of <em class="em-low1">content</em> being JSON strings:
<ul class="ul list-star compact">
<li class="li ul-li list-star-li compact-li"><em class="em-low1">processname</em> for NAME tag
</li>
<li class="li ul-li list-star-li compact-li"><em class="em-low1">processargs</em> for ARGS tag
</li>
<li class="li ul-li list-star-li compact-li"><em class="em-low1">processenv</em> for ENV tag
</li>
<li class="li ul-li list-star-li compact-li"><em class="em-low1">processbin</em> for PROGRAM tag
</li></ul></li>
<li class="li ol-li compact-li">Send a ENDREGISTRATION to terminate the initialisation process
</li></ol>
<p class="p noindent">Once the process is INITIALISED one can send a START command to start all process on the PC and a KILL command to kill them. A DESTROY command put the daemon back in CREATED state.
</p>
<p class="p indent">Additionnal standalone commands are available:
</p><table class="madoko block">
<thead><tr><th class="thead tr-even col cell-line col-odd col-first" data-row="0" data-col="1"></th><th class="thead tr-even col cell-line col-even" data-row="0" data-col="2"></th><th class="thead tr-even col cell-line col-odd" data-row="0" data-col="3"></th><th class="col thead tr-even col-even col-last" data-row="0" data-col="4" style="font-weight:bold"> </th></tr>
<tr><th class="col thead tr-odd tr-last tr-first col-odd col-first" data-row="1" data-col="1" style="font-weight:bold">Command </th><th class="col thead tr-odd tr-last tr-first col-even" data-row="1" data-col="2" style="font-weight:bold"> Parameters </th><th class="col thead tr-odd tr-last tr-first col-odd" data-row="1" data-col="3" style="font-weight:bold"> Action </th><th class="col thead tr-odd tr-last tr-first col-even col-last" data-row="1" data-col="4" style="font-weight:bold"> </th></tr></thead>
<tbody><tr><td class="tbody tr-even col cell-line col-odd col-first" data-row="0" data-col="1"></td><td class="tbody tr-even col cell-line col-even" data-row="0" data-col="2"></td><td class="tbody tr-even col cell-line col-odd" data-row="0" data-col="3"></td><td class="tbody tr-even col col-even col-last" data-row="0" data-col="4"></td></tr>
<tr><td class="tbody tr-odd tr-first col col-odd col-first" data-row="1" data-col="1">STATUS </td><td class="tbody tr-odd tr-first col col-even" data-row="1" data-col="2"> No </td><td class="tbody tr-odd tr-first col col-odd" data-row="1" data-col="3"> Returns the list and status of all processes </td><td class="tbody tr-odd tr-first col col-even col-last" data-row="1" data-col="4"> </td></tr>
<tr><td class="tbody tr-even col col-odd col-first" data-row="2" data-col="1"> </td><td class="tbody tr-even col col-even" data-row="2" data-col="2"> </td><td class="tbody tr-even col col-odd" data-row="2" data-col="3"> with per process the HOST,NAME,PID and STATUS </td><td class="tbody tr-even col col-even col-last" data-row="2" data-col="4"> </td></tr>
<tr><td class="tbody tr-odd col col-odd col-first" data-row="3" data-col="1"> </td><td class="tbody tr-odd col col-even" data-row="3" data-col="2"> </td><td class="tbody tr-odd col col-odd" data-row="3" data-col="3"> (Running/DEAD) </td><td class="tbody tr-odd col col-even col-last" data-row="3" data-col="4"> </td></tr>
<tr><td class="tbody tr-even col col-odd col-first" data-row="4" data-col="1"> KILLJOB </td><td class="tbody tr-even col col-even" data-row="4" data-col="2"> <em class="em-low1">pid</em>, <em class="em-low1">processname</em>,<em class="em-low1">signal</em>(9) </td><td class="tbody tr-even col col-odd" data-row="4" data-col="3"> kill the specified process </td><td class="tbody tr-even col col-even col-last" data-row="4" data-col="4"> </td></tr>
<tr><td class="tbody tr-odd col col-odd col-first" data-row="5" data-col="1"> RESTART </td><td class="tbody tr-odd col col-even" data-row="5" data-col="2"> <em class="em-low1">pid</em>, <em class="em-low1">processname</em>,<em class="em-low1">signal</em>(9) </td><td class="tbody tr-odd col col-odd" data-row="5" data-col="3"> restart the specified process </td><td class="tbody tr-odd col col-even col-last" data-row="5" data-col="4"> </td></tr>
<tr><td class="tbody tr-even tr-last col col-odd col-first" data-row="6" data-col="1">JOBLOG </td><td class="tbody tr-even tr-last col col-even" data-row="6" data-col="2"> <em class="em-low1">pid</em>, <em class="em-low1">processname</em>,<em class="em-low1">lines</em>(100) </td><td class="tbody tr-even tr-last col col-odd col-last" data-row="6" data-col="3"> return the last <em class="em-low1">lines</em> of the log of the specified job </td></tr></tbody></table>
<figure id="fig-fsmjob" class="figure floating align-center float" style="text-align:center;float-env:figure;float-name:Figure">
<p class="p noindent"><img src="" title="fsmjob_fsm" alt="fsmjob_fsm" data-path="images/fsmjob_fsm.png" data-linkid="fsmjob_fsm" style="width:auto;max-width:90%">
</p>
<hr class="figureline madoko" style="display:block">
<div class="p noindent"><fig-caption class="figure-caption"><span class="caption-before"><strong class="strong-star2">Figure <span class="figure-label">1</span>.</strong> </span><span class="caption-text">Finite State machine of the <em class="em-low1">levbdim::fsmjob</em></span></fig-caption></div></figure><h5 id="sec-process-control-client" class="h3" data-heading-depth="3" style="display:block"><span class="heading-before"><span class="heading-label">5.3.2</span>. </span>Process control client</h5>
<p class="p noindent">Again it is free to the developper to write its own client using any web library. We personally used python since the JSON format is easily interfaced to the language structure and the network library simple. C++ (with jsoncpp and curl) is also used. A python example is given in the example directory.
</p><h3 id="sec-conclusion" class="h1" data-heading-depth="1" style="display:block"><span class="heading-before"><span class="heading-label">6</span>. </span>Conclusion</h3>
<div class="bibliography bib-auto" data-hanging-indent="0" style="bibstyle:madoko-numeric">
<div id="dim" class="bibitem" data-cite-label="1" style="bibitem-label:[1];text-indent:-2rem;margin-left:2rem;searchterm:+Gaspar+_DIM_++"><span class="bibitem-before" style="padding-right:0.5em;display: inline-block; width:calc(2rem - 0.5em);display: inline-block; text-align:right;font-size:90%">[1]</span> C Gaspar,
<em class="em-low1">DIM</em>, <a href="http://www.bing.com/search?q=+Gaspar+_DIM_++" class="bibsearch">🔎</a></div>