-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.html
3132 lines (3103 loc) · 564 KB
/
index.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 PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<meta http-equiv="Content-Style-Type" content="text/css" />
<meta name="generator" content="pandoc" />
<meta name="author" content="Mike Pall" />
<meta name="author" content="Luke Gorrie" />
<meta name="author" content="Andy Wingo" />
<meta name="author" content="Max Rottenkolber" />
<meta name="author" content="Diego Pino Garcia" />
<meta name="author" content="Asumu Takikawa" />
<meta name="author" content="Jessica Tallon" />
<meta name="author" content="Diego Pino" />
<meta name="author" content="Alexander Gall" />
<meta name="author" content="Cosmin Apreutesei" />
<meta name="author" content="Nikolay Nikolaev" />
<meta name="author" content="Katerina Barone-Adesi" />
<meta name="author" content="Hans Huebner" />
<meta name="author" content="Javier Guerra" />
<meta name="author" content="Marcel Wiget" />
<meta name="author" content="Nicola Larosa" />
<meta name="author" content="Pete Bristow" />
<meta name="author" content="Adrian Perez de Castro" />
<meta name="author" content="Adrián Pérez de Castro" />
<meta name="author" content="Nicola tekNico Larosa" />
<meta name="author" content="Antonio Nikishaev" />
<meta name="author" content="Domen Kožar" />
<meta name="author" content="Peter Cawley" />
<meta name="author" content="Alexander Altshuler" />
<meta name="author" content="Rolf Sommerhalder" />
<meta name="author" content="Timo Buhrmester" />
<meta name="author" content="Pete Kazmier" />
<meta name="author" content="Dibyendu Majumdar" />
<meta name="author" content="Lesley De Cruz" />
<meta name="author" content="Mikhail Nazarov" />
<meta name="author" content="Kristian Larsson" />
<meta name="author" content="Justin Cormack" />
<meta name="author" content="Ben Agricola" />
<meta name="author" content="Felix Geißler" />
<meta name="author" content="R. Matthew Emerson" />
<meta name="author" content="Michael G" />
<meta name="author" content="Christian Graf" />
<meta name="author" content="Carlos Alberto Lopez Perez" />
<meta name="author" content="Jeff Loughridge" />
<meta name="author" content="Kacper Wysocki" />
<meta name="author" content="Adrian Perez" />
<meta name="author" content="William Adams" />
<meta name="author" content="Vladimir Fedin" />
<meta name="author" content="Vincenzo Maffione" />
<meta name="author" content="Tomas Korcak" />
<meta name="author" content="Tim Upthegrove" />
<meta name="author" content="Tim LaBerge" />
<meta name="author" content="Stevan Markovic" />
<meta name="author" content="Simon Leinen" />
<meta name="author" content="Ryan Hartlage" />
<meta name="author" content="Kristian Kielhofner" />
<meta name="author" content="Jon Olsson" />
<meta name="author" content="Jianbo Liu" />
<meta name="author" content="Jay Fenton" />
<meta name="author" content="James Cunningham" />
<meta name="author" content="Hui Xiang" />
<meta name="author" content="Gernot Nusshall" />
<meta name="author" content="Fabian Bonk" />
<meta name="author" content="Edward Hope-Morley" />
<meta name="author" content="Darius Bacon" />
<meta name="author" content="Anshul Makkar" />
<meta name="author" content="Andy Chong" />
<meta name="author" content="Alex Kordic" />
<meta name="author" content="Alexandr Kostrikov" />
<meta name="author" content="Alexander Spyridakis" />
<title>Snabb Reference Manual</title>
<style type="text/css">code{white-space: pre;}</style>
<style type="text/css">
div.sourceCode { overflow-x: auto; }
table.sourceCode, tr.sourceCode, td.lineNumbers, td.sourceCode {
margin: 0; padding: 0; vertical-align: baseline; border: none; }
table.sourceCode { width: 100%; line-height: 100%; }
td.lineNumbers { text-align: right; padding-right: 4px; padding-left: 4px; color: #aaaaaa; border-right: 1px solid #aaaaaa; }
td.sourceCode { padding-left: 5px; }
code > span.kw { color: #007020; font-weight: bold; } /* Keyword */
code > span.dt { color: #902000; } /* DataType */
code > span.dv { color: #40a070; } /* DecVal */
code > span.bn { color: #40a070; } /* BaseN */
code > span.fl { color: #40a070; } /* Float */
code > span.ch { color: #4070a0; } /* Char */
code > span.st { color: #4070a0; } /* String */
code > span.co { color: #60a0b0; font-style: italic; } /* Comment */
code > span.ot { color: #007020; } /* Other */
code > span.al { color: #ff0000; font-weight: bold; } /* Alert */
code > span.fu { color: #06287e; } /* Function */
code > span.er { color: #ff0000; font-weight: bold; } /* Error */
code > span.wa { color: #60a0b0; font-weight: bold; font-style: italic; } /* Warning */
code > span.cn { color: #880000; } /* Constant */
code > span.sc { color: #4070a0; } /* SpecialChar */
code > span.vs { color: #4070a0; } /* VerbatimString */
code > span.ss { color: #bb6688; } /* SpecialString */
code > span.im { } /* Import */
code > span.va { color: #19177c; } /* Variable */
code > span.cf { color: #007020; font-weight: bold; } /* ControlFlow */
code > span.op { color: #666666; } /* Operator */
code > span.bu { } /* BuiltIn */
code > span.ex { } /* Extension */
code > span.pp { color: #bc7a00; } /* Preprocessor */
code > span.at { color: #7d9029; } /* Attribute */
code > span.do { color: #ba2121; font-style: italic; } /* Documentation */
code > span.an { color: #60a0b0; font-weight: bold; font-style: italic; } /* Annotation */
code > span.cv { color: #60a0b0; font-weight: bold; font-style: italic; } /* CommentVar */
code > span.in { color: #60a0b0; font-weight: bold; font-style: italic; } /* Information */
</style>
<link href="data:text/css;charset=utf-8,%0A%23TOC%20%7B%0Aposition%3A%20fixed%3B%0Aoverflow%3A%20scroll%3B%0Atop%3A%200%3B%0Aleft%3A%200%3B%0Awidth%3A%2019%2E6em%3B%0Aheight%3A%20100%25%3B%0Amargin%3A%200%3B%0Apadding%3A%200%3B%0A%7D%0A%23TOC%20%3E%20header%20%7B%0Amargin%2Dtop%3A%201%2E4em%3B%0A%7D%0A%23TOC%20ul%2C%20%23TOC%20ol%20%7B%0Apadding%2Dleft%3A%201%2E4em%3B%0Amargin%3A%200%3B%0A%7D%0A%23TOC%20%3E%20ul%2C%20%23TOC%20%3E%20ol%20%7B%0Apadding%2Dtop%3A%200%2E7em%3B%0Awidth%3A%2025%2E2em%3B%0A%7D%0Abody%20%7B%0Amargin%3A%200%200%200%2021em%3B%0A%7D%0Abody%20%7B%0Amax%2Dwidth%3A%2040em%3B%0Afont%2Dfamily%3A%20sans%2Dserif%3B%0A%7D%0Acode%2C%20pre%20%7B%0Acolor%3A%20%23333%3B%0Abackground%3A%20%23f7f7f7%3B%0A%7D%0Apre%20%7B%0Apadding%3A%200%2E6em%3B%0A%7D%0Apre%20%3E%20code%20%7B%0Apadding%3A%200%3B%0Abackground%3A%20none%3B%0A%7D%0Acode%20%7B%0Apadding%3A%200%200%2E2em%200%2E1em%200%2E2em%3B%0A%7D%0Ap%2C%20li%20%7B%0Aline%2Dheight%3A%201%2E5%3B%0A%7D%0Ap%2Ecaption%20%7B%0Adisplay%3A%20none%3B%0A%7D%0Ah1%2Etitle%20%7B%0Afont%2Dsize%3A%202em%3B%0A%7D%0Ah2%2Eauthor%20%7B%0Adisplay%3A%20inline%3B%0Afont%2Dsize%3A%20small%3B%0Afont%2Dweight%3A%20normal%3B%0Awhite%2Dspace%3A%20nowrap%3B%0A%7D%0Ah2%2Eauthor%3Aafter%20%7B%0Acontent%3A%20%27%2C%20%27%3B%0Awhite%2Dspace%3A%20normal%3B%0A%7D%0Ah2%2Eauthor%3Anth%2Dlast%2Dchild%28%2Dn%2B2%29%3Aafter%20%7B%0Acontent%3A%20%27%27%3B%0A%7D%0Ah3%2Edate%20%7B%0Afont%2Dfamily%3A%20monospace%3B%0Afont%2Dsize%3A%20medium%3B%0Afont%2Dweight%3A%20normal%3B%0A%7D%0Adiv%23TOC%20a%20%7B%0Atext%2Ddecoration%3A%20none%3B%0A%7D%0Ah1%20a%2C%20h2%20a%2C%20h3%20a%2C%20h4%20a%2C%20h5%20a%2C%20h6%20a%20%7B%0Atext%2Ddecoration%3A%20none%3B%0Acolor%3A%20black%3B%0A%7D%0Ah1%20%7B%0Afont%2Dsize%3A%201%2E8em%3B%0A%7D%0Ah2%20%7B%0Afont%2Dsize%3A%201%2E6em%3B%0A%7D%0Ah3%20%7B%0Afont%2Dsize%3A%201%2E4em%3B%0A%7D%0Ah4%20%7B%0Afont%2Dsize%3A%201%2E2em%3B%0A%7D%0Ah5%20%7B%0Afont%2Dsize%3A%201%2E0em%3B%0A%7D%0Ah6%20%7B%0Afont%2Dweight%3A%20normal%3B%0Afont%2Dsize%3A%201%2E0em%3B%0Atext%2Ddecoration%3A%20underline%3B%0A%7D%0Adiv%2Efigure%20%3E%20img%20%7B%0Adisplay%3A%20block%3B%0Amargin%3A%20auto%3B%0A%7D%0A" rel="stylesheet" type="text/css" />
</head>
<body>
<div id="header">
<h1 class="title">Snabb Reference Manual</h1>
<h2 class="author">Mike Pall</h2>
<h2 class="author">Luke Gorrie</h2>
<h2 class="author">Andy Wingo</h2>
<h2 class="author">Max Rottenkolber</h2>
<h2 class="author">Diego Pino Garcia</h2>
<h2 class="author">Asumu Takikawa</h2>
<h2 class="author">Jessica Tallon</h2>
<h2 class="author">Diego Pino</h2>
<h2 class="author">Alexander Gall</h2>
<h2 class="author">Cosmin Apreutesei</h2>
<h2 class="author">Nikolay Nikolaev</h2>
<h2 class="author">Katerina Barone-Adesi</h2>
<h2 class="author">Hans Huebner</h2>
<h2 class="author">Javier Guerra</h2>
<h2 class="author">Marcel Wiget</h2>
<h2 class="author">Nicola Larosa</h2>
<h2 class="author">Pete Bristow</h2>
<h2 class="author">Adrian Perez de Castro</h2>
<h2 class="author">Adrián Pérez de Castro</h2>
<h2 class="author">Nicola ‘tekNico’ Larosa</h2>
<h2 class="author">Antonio Nikishaev</h2>
<h2 class="author">Domen Kožar</h2>
<h2 class="author">Peter Cawley</h2>
<h2 class="author">Alexander Altshuler</h2>
<h2 class="author">Rolf Sommerhalder</h2>
<h2 class="author">Timo Buhrmester</h2>
<h2 class="author">Pete Kazmier</h2>
<h2 class="author">Dibyendu Majumdar</h2>
<h2 class="author">Lesley De Cruz</h2>
<h2 class="author">Mikhail Nazarov</h2>
<h2 class="author">Kristian Larsson</h2>
<h2 class="author">Justin Cormack</h2>
<h2 class="author">Ben Agricola</h2>
<h2 class="author">Felix Geißler</h2>
<h2 class="author">R. Matthew Emerson</h2>
<h2 class="author">Michael G</h2>
<h2 class="author">Christian Graf</h2>
<h2 class="author">Carlos Alberto Lopez Perez</h2>
<h2 class="author">Jeff Loughridge</h2>
<h2 class="author">Kacper Wysocki</h2>
<h2 class="author">Adrian Perez</h2>
<h2 class="author">William Adams</h2>
<h2 class="author">Vladimir Fedin</h2>
<h2 class="author">Vincenzo Maffione</h2>
<h2 class="author">Tomas Korcak</h2>
<h2 class="author">Tim Upthegrove</h2>
<h2 class="author">Tim LaBerge</h2>
<h2 class="author">Stevan Markovic</h2>
<h2 class="author">Simon Leinen</h2>
<h2 class="author">Ryan Hartlage</h2>
<h2 class="author">Kristian Kielhofner</h2>
<h2 class="author">Jon Olsson</h2>
<h2 class="author">Jianbo Liu</h2>
<h2 class="author">Jay Fenton</h2>
<h2 class="author">James Cunningham</h2>
<h2 class="author">Hui Xiang</h2>
<h2 class="author">Gernot Nusshall</h2>
<h2 class="author">Fabian Bonk</h2>
<h2 class="author">Edward Hope-Morley</h2>
<h2 class="author">Darius Bacon</h2>
<h2 class="author">Anshul Makkar</h2>
<h2 class="author">Andy Chong</h2>
<h2 class="author">Alex Kordic</h2>
<h2 class="author">Alexandr Kostrikov</h2>
<h2 class="author">Alexander Spyridakis</h2>
<h3 class="date">Version 4b0c18b, Fri Nov 8 08:53:57 2019 +0100</h3>
</div>
<div id="TOC">
<ul>
<li><a href="#introduction">Introduction</a></li>
<li><a href="#snabb-api">Snabb API</a><ul>
<li><a href="#app">App</a></li>
<li><a href="#config-core.config">Config (core.config)</a></li>
<li><a href="#engine-core.app">Engine (core.app)</a></li>
<li><a href="#link-core.link">Link (core.link)</a></li>
<li><a href="#packet-core.packet">Packet (core.packet)</a></li>
<li><a href="#memory-core.memory">Memory (core.memory)</a></li>
<li><a href="#shared-memory-core.shm">Shared Memory (core.shm)</a><ul>
<li><a href="#counter-core.counter">Counter (core.counter)</a></li>
<li><a href="#histogram-core.histogram">Histogram (core.histogram)</a></li>
</ul></li>
<li><a href="#lib-core.lib">Lib (core.lib)</a></li>
<li><a href="#multiprocess-operation-core.worker">Multiprocess operation (core.worker)</a></li>
<li><a href="#main">Main</a></li>
</ul></li>
<li><a href="#basic-apps-apps.basic.basic_apps">Basic Apps (apps.basic.basic_apps)</a><ul>
<li><a href="#source">Source</a></li>
<li><a href="#join">Join</a></li>
<li><a href="#split">Split</a></li>
<li><a href="#sink">Sink</a></li>
<li><a href="#tee">Tee</a></li>
<li><a href="#repeater">Repeater</a></li>
<li><a href="#truncate">Truncate</a></li>
<li><a href="#sample">Sample</a></li>
</ul></li>
<li><a href="#intel-82599-ethernet-controller-apps">Intel 82599 Ethernet Controller Apps</a><ul>
<li><a href="#intel82599-apps.intel.intel_app">Intel82599 (apps.intel.intel_app)</a><ul>
<li><a href="#configuration">Configuration</a></li>
<li><a href="#performance">Performance</a></li>
<li><a href="#hardware-limits">Hardware limits</a></li>
</ul></li>
<li><a href="#loadgen-apps.intel.loadgen">LoadGen (apps.intel.loadgen)</a><ul>
<li><a href="#configuration-1">Configuration</a></li>
<li><a href="#performance-1">Performance</a></li>
</ul></li>
</ul></li>
<li><a href="#intel-i210-i350-82599-ethernet-controller-apps-apps.intel_mp.intel_mp">Intel i210 / i350 / 82599 Ethernet Controller apps (apps.intel_mp.intel_mp)</a><ul>
<li><a href="#caveats">Caveats</a></li>
<li><a href="#configuration-2">Configuration</a><ul>
<li><a href="#rss-hashing-methods">RSS hashing methods</a></li>
<li><a href="#default-rss-queue">Default RSS Queue</a></li>
<li><a href="#hardware-limits-1">Hardware limits</a></li>
</ul></li>
</ul></li>
<li><a href="#solarflare-ethernet-controller-apps">Solarflare Ethernet Controller Apps</a><ul>
<li><a href="#solarflare-apps.solarflare.solarflare">Solarflare (apps.solarflare.solarflare)</a><ul>
<li><a href="#configuration-3">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#ratelimiter-app-apps.rate_limiter.rate_limiter">RateLimiter App (apps.rate_limiter.rate_limiter)</a><ul>
<li><a href="#configuration-4">Configuration</a></li>
<li><a href="#performance-2">Performance</a></li>
</ul></li>
<li><a href="#pcapfilter-app-apps.packet_filter.pcap_filter">PcapFilter App (apps.packet_filter.pcap_filter)</a><ul>
<li><a href="#configuration-5">Configuration</a></li>
<li><a href="#special-counters">Special Counters</a></li>
</ul></li>
<li><a href="#ipv4-apps">IPv4 Apps</a><ul>
<li><a href="#arp-apps.ipv4.arp">ARP (apps.ipv4.arp)</a><ul>
<li><a href="#configuration-6">Configuration</a></li>
</ul></li>
<li><a href="#reassembler-apps.ipv4.reassemble">Reassembler (apps.ipv4.reassemble)</a><ul>
<li><a href="#configuration-7">Configuration</a></li>
</ul></li>
<li><a href="#fragmenter-apps.ipv4.fragment">Fragmenter (apps.ipv4.fragment)</a><ul>
<li><a href="#configuration-8">Configuration</a></li>
</ul></li>
<li><a href="#icmp-echo-responder-apps.ipv4.echo">ICMP Echo responder (apps.ipv4.echo)</a><ul>
<li><a href="#configuration-9">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#ipv6-apps">IPv6 Apps</a><ul>
<li><a href="#nd_light-apps.ipv6.nd_light">Nd_light (apps.ipv6.nd_light)</a><ul>
<li><a href="#configuration-10">Configuration</a></li>
<li><a href="#special-counters-1">Special Counters</a></li>
</ul></li>
<li><a href="#simplekeyedtunnel-apps.keyed_ipv6_tunnel.tunnel">SimpleKeyedTunnel (apps.keyed_ipv6_tunnel.tunnel)</a><ul>
<li><a href="#configuration-11">Configuration</a></li>
<li><a href="#special-counters-2">Special Counters</a></li>
</ul></li>
<li><a href="#fragmenter-apps.ipv6.fragment">Fragmenter (apps.ipv6.fragment)</a><ul>
<li><a href="#configuration-12">Configuration</a></li>
</ul></li>
<li><a href="#icmp-echo-responder-apps.ipv6.echo">ICMP Echo responder (apps.ipv6.echo)</a><ul>
<li><a href="#configuration-13">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#vhostuser-app-apps.vhost.vhost_user">VhostUser App (apps.vhost.vhost_user)</a><ul>
<li><a href="#configuration-14">Configuration</a></li>
</ul></li>
<li><a href="#virtionet-app-apps.virtio_net.virtio_net">VirtioNet App (apps.virtio_net.virtio_net)</a><ul>
<li><a href="#configuration-15">Configuration</a></li>
</ul></li>
<li><a href="#pcap-savefile-apps">Pcap Savefile Apps</a><ul>
<li><a href="#pcapreader-and-pcapwriter-apps-apps.pcap.pcap">PcapReader and PcapWriter Apps (apps.pcap.pcap)</a><ul>
<li><a href="#configuration-16">Configuration</a></li>
</ul></li>
<li><a href="#tap-apps.pcap.tap">Tap (apps.pcap.tap)</a><ul>
<li><a href="#configuration-17">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#rawsocket-app-apps.socket.raw">RawSocket App (apps.socket.raw)</a><ul>
<li><a href="#configuration-18">Configuration</a></li>
</ul></li>
<li><a href="#unixsocket-app-apps.socket.unix">UnixSocket App (apps.socket.unix)</a><ul>
<li><a href="#configuration-19">Configuration</a></li>
</ul></li>
<li><a href="#tap-app-apps.tap.tap">Tap app (apps.tap.tap)</a><ul>
<li><a href="#configuration-20">Configuration</a></li>
</ul></li>
<li><a href="#vlan-apps">VLAN Apps</a><ul>
<li><a href="#tagger-apps.vlan.vlan">Tagger (apps.vlan.vlan)</a><ul>
<li><a href="#configuration-21">Configuration</a></li>
</ul></li>
<li><a href="#untagger-apps.vlan.vlan">Untagger (apps.vlan.vlan)</a><ul>
<li><a href="#configuration-22">Configuration</a></li>
</ul></li>
<li><a href="#vlanmux-apps.vlan.vlan">VlanMux (apps.vlan.vlan)</a><ul>
<li><a href="#configuration-23">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#bridge-apps">Bridge Apps</a><ul>
<li><a href="#configuration-24">Configuration</a></li>
<li><a href="#flooding-bridge-apps.bridge.flooding">Flooding bridge (apps.bridge.flooding)</a><ul>
<li><a href="#configuration-25">Configuration</a></li>
</ul></li>
<li><a href="#learning-bridge-apps.bridge.learning">Learning bridge (apps.bridge.learning)</a><ul>
<li><a href="#configuration-26">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#ipfix-and-netflow-apps">IPFIX and NetFlow apps</a><ul>
<li><a href="#ipfix-apps.ipfix.ipfix">IPFIX (apps.ipfix.ipfix)</a><ul>
<li><a href="#configuration-27">Configuration</a></li>
<li><a href="#to-do-list">To-do list</a></li>
</ul></li>
</ul></li>
<li><a href="#ipsec-apps">IPsec Apps</a><ul>
<li><a href="#esp-transport6-and-tunnel6-apps.ipsec.esp">ESP Transport6 and Tunnel6 (apps.ipsec.esp)</a><ul>
<li><a href="#configuration-28">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#test-apps">Test Apps</a><ul>
<li><a href="#match-apps.test.match">Match (apps.test.match)</a><ul>
<li><a href="#configuration-29">Configuration</a></li>
</ul></li>
<li><a href="#synth-apps.test.synth">Synth (apps.test.synth)</a><ul>
<li><a href="#configuration-30">Configuration</a></li>
</ul></li>
<li><a href="#npackets-apps.test.npackets">Npackets (apps.test.npackets)</a><ul>
<li><a href="#configuration-31">Configuration</a></li>
</ul></li>
</ul></li>
<li><a href="#snabbwall-apps">SnabbWall Apps</a><ul>
<li><a href="#l7spy-apps.wall.l7spy">L7Spy (apps.wall.l7spy)</a></li>
<li><a href="#filter-apps.wall.filter">Filter (apps.wall.filter)</a></li>
<li><a href="#scanner-apps.wall.scanner">Scanner (apps.wall.scanner)</a><ul>
<li><a href="#subclassing">Subclassing</a></li>
<li><a href="#ndpiscanner-apps.wall.scanner.ndpi">NdpiScanner (apps.wall.scanner.ndpi)</a></li>
</ul></li>
<li><a href="#utilities">Utilities</a><ul>
<li><a href="#southandnorth-apps.wall.util">SouthAndNorth (apps.wall.util)</a></li>
</ul></li>
</ul></li>
<li><a href="#rss-app-apps.rss.rss">RSS app (apps.rss.rss)</a><ul>
<li><a href="#flow-director">Flow-director</a></li>
<li><a href="#packet-replication">Packet replication</a></li>
<li><a href="#weighted-links">Weighted links</a></li>
<li><a href="#packet-meta-data">Packet meta-data</a></li>
<li><a href="#ipv6-extension-header-elimination">IPv6 extension header elimination</a></li>
<li><a href="#vlan-pseudo-tagging">VLAN pseudo-tagging</a></li>
<li><a href="#configuration-32">Configuration</a></li>
<li><a href="#meta-data-api">Meta-data API</a></li>
</ul></li>
<li><a href="#inter-process-links-apps.interlink.">Inter-process links (apps.interlink.*)</a><ul>
<li><a href="#configuration-33">Configuration</a></li>
</ul></li>
<li><a href="#libraries">Libraries</a><ul>
<li><a href="#ip-checksum-lib.checksum">IP checksum (lib.checksum)</a></li>
<li><a href="#ctable-lib.ctable">Ctable (lib.ctable)</a></li>
<li><a href="#poptrie-lib.poptrie">Poptrie (lib.poptrie)</a></li>
<li><a href="#pmu-lib.pmu">PMU (lib.pmu)</a></li>
<li><a href="#snabb-program-configuration-with-yang-lib.yang">Snabb program configuration with YANG (<code>lib.yang</code>)</a></li>
<li><a href="#hardware">Hardware</a><ul>
<li><a href="#pci-lib.hardware.pci">PCI (lib.hardware.pci)</a></li>
<li><a href="#register-lib.hardware.register">Register (lib.hardware.register)</a></li>
</ul></li>
<li><a href="#protocols">Protocols</a><ul>
<li><a href="#protocol-header-lib.protocol.header">Protocol Header (lib.protocol.header)</a></li>
<li><a href="#ethernet-lib.protocol.ethernet">Ethernet (lib.protocol.ethernet)</a></li>
<li><a href="#ipv4-lib.protocol.ipv4">IPv4 (lib.protocol.ipv4)</a></li>
<li><a href="#ipv6-lib.protocol.ipv6">IPv6 (lib.protocol.ipv6)</a></li>
<li><a href="#tcp-lib.protocol.tcp">TCP (lib.protocol.tcp)</a></li>
<li><a href="#udp-lib.protocol.udp">UDP (lib.protocol.udp)</a></li>
<li><a href="#gre-lib.protocol.gre">GRE (lib.protocol.gre)</a></li>
<li><a href="#icmp-lib.protocol.icmp.header">ICMP (lib.protocol.icmp.header)</a></li>
<li><a href="#datagram-lib.protocol.datagram">Datagram (lib.protocol.datagram)</a></li>
</ul></li>
<li><a href="#ipsec">IPsec</a><ul>
<li><a href="#encapsulating-security-payload-lib.ipsec.esp">Encapsulating Security Payload (lib.ipsec.esp)</a></li>
</ul></li>
<li><a href="#snabb-nfv">Snabb NFV</a><ul>
<li><a href="#nfv-config-program.snabbnfv.nfvconfig">NFV config (program.snabbnfv.nfvconfig)</a></li>
<li><a href="#snabbnfv-traffic">snabbnfv traffic</a></li>
<li><a href="#snabbnfv-neutron2snabb">snabbnfv neutron2snabb</a></li>
</ul></li>
<li><a href="#lisper">LISPER</a><ul>
<li><a href="#lisper-program.lisper">LISPER (program.lisper)</a></li>
</ul></li>
<li><a href="#ptree">Ptree</a><ul>
<li><a href="#ptree-program.ptree">Ptree (program.ptree)</a></li>
</ul></li>
<li><a href="#watchdog-lib.watchdog.watchdog">Watchdog (lib.watchdog.watchdog)</a></li>
</ul></li>
<li><a href="#snabblab">Snabblab</a><ul>
<li><a href="#guidelines">Guidelines</a></li>
<li><a href="#servers">Servers</a></li>
<li><a href="#get-started">Get started</a></li>
<li><a href="#using-the-lab">Using the lab</a></li>
<li><a href="#questions">Questions</a></li>
<li><a href="#thanks">Thanks</a></li>
</ul></li>
</ul>
</div>
<p><em><strong>Note:</strong> This reference manual is a draft. The API defined in this document is not guaranteed to be stable or complete and future versions of Snabb will introduce backwards incompatible changes. With that being said, discrepancies between this document and the actual Snabb Switch implementation are considered to be bugs. Please report them in order to help improve this document.</em></p>
<h1 id="introduction">Introduction</h1>
<p><em>Snabb</em> is an extensible, virtualized, Ethernet networking toolkit. With Snabb you can implement networking applications using the <em>Lua language</em>. Snabb includes all the tools you need to quickly realize your network designs and its really fast too! Furthermore, Snabb is extensible and encourages you to grow the ecosystem to match your requirements.</p>
<div class="figure">
<img src="" alt="Architecture" />
<p class="caption">Architecture</p>
</div>
<p>The Snabb Core forms a runtime environment (<em>engine</em>) which executes your <em>design</em>. A design is simply a Lua script used to drive the Snabb stack, you can think of it as your top-level “main” routine.</p>
<p>In order to add functionality to the Snabb stack you can load modules into the Snabb engine. These can be Lua modules as well as native code objects. We differentiate between two classes of modules, namely libraries and <em>Apps</em>. Libraries are simple collections of program utilities to be used in your designs, apps or other libraries, just as you might expect. Apps, on the other hand, are code objects that implement a specific interface, which is used by the Snabb engine to organize an <em>App Network</em>.</p>
<div class="figure">
<img src="" alt="Network" />
<p class="caption">Network</p>
</div>
<p>Usually, a Snabb design will create a series of apps, interconnect these in a desired way using <em>links</em> and finally pass the resulting app network on to the Snabb engine. The engine’s job is to:</p>
<ul>
<li>Pump traffic through the app network</li>
<li>Keep the app network running (e.g. restart failed apps)</li>
<li>Report on the network status</li>
</ul>
<h1 id="snabb-api">Snabb API</h1>
<p>The core modules defined below can be loaded using Lua’s <code>require</code>. For example:</p>
<pre><code>local config = require("core.config")
local c = config.new()
...</code></pre>
<h2 id="app">App</h2>
<p>An <em>app</em> is an isolated implementation of a specific networking function. For example, a switch, a router, or a packet filter.</p>
<p>Apps receive packets on <em>input ports</em>, perform some processing, and transmit packets on <em>output ports</em>. Each app has zero or more input and output ports. For example, a packet filter may have one input and one output port, while a packet recorder may have only an input port. Every app must implement the interface below. Methods which may be left unimplemented are marked as “optional”.</p>
<p>— Method <strong>myapp:new</strong> <em>arg</em></p>
<p><em>Required</em>. Create an instance of the app with a given argument <em>arg</em>. <code>Myapp:new</code> must return an instance of the app. The handling of <em>arg</em> is up to the app but it is encouraged to use <code>core.config</code>’s <code>parse_app_arg</code> to parse <em>arg</em>.</p>
<p>— Field <strong>myapp.input</strong></p>
<p>— Field <strong>myapp.output</strong></p>
<p>Tables of named input and output links. These tables are initialized by the engine for use in processing and are <em>read-only</em>.</p>
<p>— Field <strong>myapp.appname</strong></p>
<p>Name of the app. <em>Read-only</em>.</p>
<p>— Field <strong>myapp.shm</strong></p>
<p>Can be set to a specification for <code>core.shm.create_frame</code>. When set, this field will be initialized to a frame of shared memory objects by the engine.</p>
<p>— Field <strong>myapp.config</strong></p>
<p>Can be set to a specification for <code>core.lib.parse</code>. When set, the specification will be used to validate the app’s arg when it is configured using <code>config.app</code>.</p>
<p>— Method <strong>myapp:link</strong></p>
<p><em>Optional</em>. Called any time the app’s links may have been changed (including on start-up). Guaranteed to be called before <code>pull</code> and <code>push</code> are called with new links.</p>
<p>— Method <strong>myapp:pull</strong></p>
<p><em>Optional</em>. Pull packets into the network.</p>
<p>For example: Pull packets from a network adapter into the app network by transmitting them to output ports.</p>
<p>— Method <strong>myapp:push</strong></p>
<p><em>Optional</em>. Push packets through the system.</p>
<p>For example: Move packets from input ports to output ports or to a network adapter.</p>
<p>— Method <strong>myapp:reconfig</strong> <em>arg</em></p>
<p><em>Optional</em>. Reconfigure the app with a new <em>arg</em>. If this method is not implemented the app instance is discarded and a new instance is created.</p>
<p>— Method <strong>myapp:report</strong></p>
<p><em>Optional</em>. Print a report of the current app status.</p>
<p>— Method <strong>myapp:stop</strong></p>
<p><em>Optional</em>. Stop the app and release associated external resources.</p>
<p>— Field <strong>myapp.zone</strong></p>
<p><em>Optional</em>. Name of the LuaJIT <em>profiling zone</em> used for this app (descriptive string). The default is the module name.</p>
<h2 id="config-core.config">Config (core.config)</h2>
<p>A <em>config</em> is a description of a packet-processing network. The network is a directed graph. Nodes in the graph are <em>apps</em> that each process packets in a specific way. Each app has a set of named input and output <em>ports</em>—often called <em>rx</em> and <em>tx</em>. Edges of the graph are unidirectional <em>links</em> that carry packets from an output port to an input port.</p>
<p>The config is a purely passive data structure. Creating and manipulating a config object does not immediately affect operation. The config has to be activated using <code>engine.configure</code>.</p>
<p>— Function <strong>config.new</strong></p>
<p>Creates and returns a new empty configuration.</p>
<p>— Function <strong>config.app</strong> <em>config</em>, <em>name</em>, <em>class</em>, <em>arg</em></p>
<p>Adds an app of <em>class</em> with <em>arg</em> to the <em>config</em> where it will be assigned to <em>name</em>.</p>
<p>Example:</p>
<pre><code>config.app(c, "nic", Intel82599, {pciaddr = "0000:00:00.0"})</code></pre>
<p>— Function <strong>config.link</strong> <em>config</em>, <em>linkspec</em></p>
<p>Add a link defined by <em>linkspec</em> to the config <em>config</em>. <em>Linkspec</em> must be a string of the format</p>
<pre><code>app_name1.output_port->app_name2.input_port</code></pre>
<p>where <code>app_name1</code> and <code>app_name2</code> are names of apps in <em>config</em> and <code>output_port</code> and <code>input_port</code> are valid output and input ports of the referenced apps respectively.</p>
<p>Example:</p>
<pre><code>config.link(c, "nic1.tx->nic2.rx")</code></pre>
<h2 id="engine-core.app">Engine (core.app)</h2>
<p>The <em>engine</em> executes a config by initializing apps, creating links, and driving the flow of execution. The engine also performs profiling and reporting functions. It can be reconfigured during runtime. Within Snabb Switch scripts the <code>core.app</code> module is bound to the global <code>engine</code> variable.</p>
<p>— Function <strong>engine.configure</strong> <em>config</em></p>
<p>Configure the engine to use a new config <em>config</em>. You can safely call this method many times to incrementally update the running app network. The engine updates the app network as follows:</p>
<ul>
<li>Apps that did not exist in the old configuration are started.</li>
<li>Apps that do not exist in the new configuration are stopped. (The app <code>stop()</code> method is called if defined.)</li>
<li>Apps with unchanged configurations are preserved.</li>
<li>Apps with changed configurations are updated by calling their <code>reconfig()</code> method. If the <code>reconfig()</code> method is not implemented then the old instance is stopped a new one started.</li>
<li>Links with unchanged endpoints are preserved.</li>
</ul>
<p>— Function <strong>engine.main</strong> <em>options</em></p>
<p>Run the Snabb engine. <em>Options</em> is a table of key/value pairs. The following keys are recognized:</p>
<ul>
<li><code>duration</code> - Duration in seconds to run the engine for (as a floating point number). If this is set you cannot supply <code>done</code>.</li>
<li><code>done</code> - A function to be called repeatedly by <code>engine.main</code> until it returns <code>true</code>. Once it returns <code>true</code> the engine will be stopped and <code>engine.main</code> will return. If this is set you cannot supply <code>duration</code>.</li>
<li><code>report</code> - A table which configures the report printed before <code>engine.main()</code> returns. The keys <code>showlinks</code> and <code>showapps</code> can be set to boolean values to force or suppress link and app reporting individually. By default `engine.main()’ will report on links but not on apps.</li>
<li><code>measure_latency</code> - By default, the <code>breathe()</code> loop is instrumented to record the latency distribution of running the app graph. This information can be processed by the <code>snabb top</code> program. Passing <code>measure_latency=false</code> in the <em>options</em> will disable this instrumentation.</li>
<li><code>no_report</code> - A boolean value. If <code>true</code> no final report will be printed.</li>
</ul>
<p>— Function <strong>engine.stop</strong></p>
<p>Stop all apps in the engine by loading an empty configuration.</p>
<p>— Function <strong>engine.now</strong></p>
<p>Returns monotonic time in seconds as a floating point number. Suitable for timers.</p>
<p>— Variable <strong>engine.busywait</strong></p>
<p>If set to true then the engine polls continuously for new packets to process. This consumes 100% CPU and makes processing latency less vulnerable to kernel scheduling behavior which can cause pauses of more than one millisecond.</p>
<p>Default: false</p>
<p>— Variable <strong>engine.Hz</strong></p>
<p>Frequency at which to poll for new input packets. The default value is ‘false’ which means to adjust dynamically up to 100us during low traffic. The value can be overridden with a constant integer saying how many times per second to poll.</p>
<p>This setting is not used when engine.busywait is true.</p>
<h2 id="link-core.link">Link (core.link)</h2>
<p>A <em>link</em> is a <a href="http://en.wikipedia.org/wiki/Circular_buffer">ring buffer</a> used to store packets between apps. Links can be treated either like arrays—accessing their internal structure directly—or as streams of packets by using their API functions.</p>
<p>— Function <strong>link.empty</strong> <em>link</em></p>
<p>Predicate used to test if a link is empty. Returns true if <em>link</em> is empty and false otherwise.</p>
<p>— Function <strong>link.full</strong> <em>link</em></p>
<p>Predicate used to test if a link is full. Returns true if <em>link</em> is full and false otherwise.</p>
<p>— Function <strong>link.nreadable</strong> <em>link</em></p>
<p>Returns the number of packets on <em>link</em>.</p>
<p>— Function <strong>link.nwriteable</strong> <em>link</em></p>
<p>Returns the remaining number of packets that fit onto <em>link</em>.</p>
<p>— Function <strong>link.receive</strong> <em>link</em></p>
<p>Returns the next available packet (and advances the read cursor) on <em>link</em>. If the link is empty an error is signaled.</p>
<p>— Function <strong>link.front</strong> <em>link</em></p>
<p>Return the next available packet without advancing the read cursor on <em>link</em>. If the link is empty, <code>nil</code> is returned.</p>
<p>— Function <strong>link.transmit</strong> <em>link</em>, <em>packet</em></p>
<p>Transmits <em>packet</em> onto <em>link</em>. If the link is full <em>packet</em> is dropped (and the drop counter increased).</p>
<p>— Function <strong>link.stats</strong> <em>link</em></p>
<p>Returns a structure holding ring statistics for the <em>link</em>:</p>
<ul>
<li><code>txbytes</code>, <code>rxbytes</code>: Counts of transferred bytes.</li>
<li><code>txpackets</code>, <code>rxpackets</code>: Counts of transferred packets.</li>
<li><code>txdrop</code>: Count of packets dropped due to ring overflow.</li>
</ul>
<h2 id="packet-core.packet">Packet (core.packet)</h2>
<p>A <em>packet</em> is an FFI object of type <code>struct packet</code> representing a network packet that is currently being processed. The packet is used to explicitly manage the life cycle of the packet. Packets are explicitly allocated and freed by using <code>packet.allocate</code> and <code>packet.free</code>. When a packet is received using <code>link.receive</code> its ownership is acquired by the calling app. The app must then ensure to either transfer the packet ownership to another app by calling <code>link.transmit</code> on the packet or free the packet using <code>packet.free</code>. Apps may only use packets they own, e.g. packets that have not been transmitted or freed. The number of allocatable packets is limited by the size of the underlying “freelist”, e.g. a pool of unused packet objects from and to which packets are allocated and freed.</p>
<p>— Type <strong>struct packet</strong></p>
<pre><code>struct packet {
uint16_t length;
uint8_t data[packet.max_payload];
};</code></pre>
<p>— Constant <strong>packet.max_payload</strong></p>
<p>The maximum payload length of a packet.</p>
<p>— Function <strong>packet.allocate</strong></p>
<p>Returns a new empty packet. An an error is raised if there are no packets left on the freelist. Initially the <code>length</code> of the allocated is 0, and its <code>data</code> is uninitialized garbage.</p>
<p>— Function <strong>packet.free</strong> <em>packet</em></p>
<p>Frees <em>packet</em> and puts in back onto the freelist.</p>
<p>— Function <strong>packet.clone</strong> <em>packet</em></p>
<p>Returns an exact copy of <em>packet</em>.</p>
<p>— Function <strong>packet.resize</strong> <em>packet</em>, <em>length</em></p>
<p>Sets the payload length of <em>packet</em>, truncating or extending its payload. In the latter case the contents of the extended area at the end of the payload are filled with zeros.</p>
<p>— Function <strong>packet.append</strong> <em>packet</em>, <em>pointer</em>, <em>length</em></p>
<p>Appends <em>length</em> bytes starting at <em>pointer</em> to the end of <em>packet</em>. An error is raised if there is not enough space in <em>packet</em> to accomodate <em>length</em> additional bytes.</p>
<p>— Function <strong>packet.prepend</strong> <em>packet</em>, <em>pointer</em>, <em>length</em></p>
<p>Prepends <em>length</em> bytes starting at <em>pointer</em> to the front of <em>packet</em>, taking ownership of the packet and returning a new packet. An error is raised if there is not enough space in <em>packet</em> to accomodate <em>length</em> additional bytes.</p>
<p>— Function <strong>packet.shiftleft</strong> <em>packet</em>, <em>length</em></p>
<p>Take ownership of <em>packet</em>, truncate it by <em>length</em> bytes from the front, and return a new packet. <em>Length</em> must be less than or equal to <code>length</code> of <em>packet</em>.</p>
<p>— Function <strong>packet.shiftright</strong> <em>packet</em>, <em>length</em></p>
<p>Take ownership of <em>packet</em>, moves <em>packet</em> payload to the right by <em>length</em> bytes, growing <em>packet</em> by <em>length</em>. Returns a new packet. The sum of <em>length</em> and <code>length</code> of <em>packet</em> must be less than or equal to <code>packet.max_payload</code>.</p>
<p>— Function <strong>packet.from_pointer</strong> <em>pointer</em>, <em>length</em></p>
<p>Allocate packet and fill it with <em>length</em> bytes from <em>pointer</em>.</p>
<p>— Function <strong>packet.from_string</strong> <em>string</em></p>
<p>Allocate packet and fill it with the contents of <em>string</em>.</p>
<p>— Function **packet.clone_to_memory* <em>pointer</em> <em>packet</em></p>
<p>Creates an exact copy of at memory pointed to by <em>pointer</em>. <em>Pointer</em> must point to a <code>packet.packet_t</code>.</p>
<h2 id="memory-core.memory">Memory (core.memory)</h2>
<p>Snabb allocates special <a href="https://en.wikipedia.org/wiki/Direct_memory_access">DMA</a> memory that can be accessed directly by network cards. The important characteristic of DMA memory is being located in contiguous physical memory at a stable address.</p>
<p>— Function <strong>memory.dma_alloc</strong> <em>bytes</em>, [<em>alignment</em>]</p>
<p>Returns a pointer to <em>bytes</em> of new DMA memory.</p>
<p>Optionally a specific <em>alignment</em> requirement can be provided (in bytes). The default alignment is 128.</p>
<p>— Function <strong>memory.virtual_to_physical</strong> <em>pointer</em></p>
<p>Returns the physical address (<code>uint64_t</code>) the DMA memory at <em>pointer</em>.</p>
<p>— Variable <strong>memory.huge_page_size</strong></p>
<p>Size of a single huge page in bytes. Read-only.</p>
<h2 id="shared-memory-core.shm">Shared Memory (core.shm)</h2>
<p>This module facilitates creation and management of named shared memory objects. Objects can be created using <code>shm.create</code> similar to <code>ffi.new</code>, except that separate calls to <code>shm.open</code> for the same name will each return a new mapping of the same shared memory. Different processes can share memory by mapping an object with the same name (and type). Each process can map any object any number of times.</p>
<p>Mappings are deleted on process termination or with an explicit <code>shm.unmap</code>. Names are unlinked from objects that are no longer needed using <code>shm.unlink</code>. Object memory is freed when the name is unlinked and all mappings have been deleted.</p>
<p>Names can be fully qualified or abbreviated to be within the current process. Here are examples of names and how they are resolved where <code><pid></code> is the PID of this process:</p>
<ul>
<li>Local: <code>foo/bar</code> ⇒ <code>/var/run/snabb/<pid>/foo/bar</code></li>
<li>Fully qualified: <code>/1234/foo/bar</code> ⇒ <code>/var/run/snabb/1234/foo/bar</code></li>
</ul>
<p>Behind the scenes the objects are backed by files on ram disk (<code>/var/run/snabb/<pid></code>) and accessed with the equivalent of POSIX shared memory (<code>shm_overview(7)</code>). The files are automatically removed on shutdown unless the environment <code>SNABB_SHM_KEEP</code> is set. The location <code>/var/run/snabb</code> can be overridden by the environment variable <code>SNABB_SHM_ROOT</code>.</p>
<p>Shared memory objects are created world-readable for convenient access by diagnostic tools. You can lock this down by setting <code>SNABB_SHM_ROOT</code> to a path under a directory with appropriate permissions.</p>
<p>The practical limit on the number of objects that can be mapped will depend on the operating system limit for memory mappings. On Linux the default limit is 65,530 mappings:</p>
<pre><code>$ sysctl vm.max_map_count vm.max_map_count = 65530</code></pre>
<p>— Function <strong>shm.create</strong> <em>name</em>, <em>type</em></p>
<p>Creates and maps a shared object of <em>type</em> into memory via a hierarchical <em>name</em>. Returns a pointer to the mapped object.</p>
<p>— Function <strong>shm.open</strong> <em>name</em>, <em>type</em>, [<em>readonly</em>]</p>
<p>Maps an existing shared object of <em>type</em> into memory via a hierarchical <em>name</em>. If <em>readonly</em> is non-nil the shared object is mapped in read-only mode. <em>Readonly</em> defaults to nil. Fails if the shared object does not already exist. Returns a pointer to the mapped object.</p>
<p>— Function <strong>shm.alias</strong> <em>new-path</em> <em>existing-path</em></p>
<p>Create an alias (symbolic link) for an object.</p>
<p>— Function <strong>shm.path</strong> <em>name</em></p>
<p>Returns the fully-qualified path for an object called <em>name</em>.</p>
<p>— Function <strong>shm.exists</strong> <em>name</em></p>
<p>Returns a true value if shared object by <em>name</em> exists.</p>
<p>— Function <strong>shm.unmap</strong> <em>pointer</em></p>
<p>Deletes the memory mapping for <em>pointer</em>.</p>
<p>— Function <strong>shm.unlink</strong> <em>path</em></p>
<p>Unlinks the subtree of objects designated by <em>path</em> from the filesystem.</p>
<p>— Function <strong>shm.children</strong> <em>path</em></p>
<p>Returns an array of objects in the directory designated by <em>path</em>.</p>
<p>— Function <strong>shm.register</strong> <em>type</em>, <em>module</em></p>
<p>Registers an abstract shared memory object <em>type</em> implemented by <em>module</em> in <code>shm.types</code>. <em>Module</em> must provide the following functions:</p>
<ul>
<li><strong>create</strong> <em>name</em>, …</li>
<li><strong>open</strong>, <em>name</em></li>
</ul>
<p>and can optionally provide the function:</p>
<ul>
<li><strong>delete</strong>, <em>name</em></li>
</ul>
<p>The <em>module</em>’s <code>type</code> variable must be bound to <em>type</em>. To register a new type a module might invoke <code>shm.register</code> like so:</p>
<pre><code>type = shm.register('mytype', getfenv())
-- Now the following holds true:
-- shm.types[type] == getfenv()</code></pre>
<p>— Variable <strong>shm.types</strong></p>
<p>A table that maps types to modules. See <code>shm.register</code>.</p>
<p>— Function <strong>shm.create_frame</strong> <em>path</em>, <em>specification</em></p>
<p>Creates and returns a shared memory frame by <em>specification</em> under <em>path</em>. A frame is a table of mapped—possibly abstract‑shared memory objects. <em>Specification</em> must be of the form:</p>
<pre><code>{ <name> = {<module>, ...},
... }</code></pre>
<p><em>Module</em> must implement an abstract type registered with <code>shm.register</code>, and is followed by additional initialization arguments to its <code>create</code> function. Example usage:</p>
<pre><code>local counter = require("core.counter")
-- Create counters foo/bar/{dtime,rxpackets,txpackets}.counter
local f = shm.create_frame(
"foo/bar",
{dtime = {counter, C.get_unix_time()},
rxpackets = {counter},
txpackets = {counter}})
counter.add(f.rxpackets)
counter.read(f.dtime)</code></pre>
<p>— Function <strong>shm.open_frame</strong> <em>path</em></p>
<p>Opens and returns the shared memory frame under <em>path</em> for reading.</p>
<p>— Function <strong>shm.delete_frame</strong> <em>frame</em></p>
<p>Deletes/unmaps a shared memory <em>frame</em>. The <em>frame</em> directory is unlinked if <em>frame</em> was created by <code>shm.create_frame</code>.</p>
<h3 id="counter-core.counter">Counter (core.counter)</h3>
<p>Double-buffered shared memory counters. Counters are 64-bit unsigned values. Registered with <code>core.shm</code> as type <code>counter</code>.</p>
<p>— Function <strong>counter.create</strong> <em>name</em>, [<em>initval</em>]</p>
<p>Creates and returns a <code>counter</code> by <em>name</em>, initialized to <em>initval</em>. <em>Initval</em> defaults to 0.</p>
<p>— Function <strong>counter.open</strong> <em>name</em></p>
<p>Opens and returns the counter by <em>name</em> for reading.</p>
<p>— Function <strong>counter.delete</strong> <em>name</em></p>
<p>Deletes and unmaps the counter by <em>name</em>.</p>
<p>— Function <strong>counter.commit</strong></p>
<p>Commits buffered counter values to public shared memory.</p>
<p>— Function <strong>counter.set</strong> <em>counter</em>, <em>value</em></p>
<p>Sets <em>counter</em> to <em>value</em>.</p>
<p>— Function <strong>counter.add</strong> <em>counter</em>, [<em>value</em>]</p>
<p>Increments <em>counter</em> by <em>value</em>. <em>Value</em> defaults to 1.</p>
<p>— Function <strong>counter.read</strong> <em>counter</em></p>
<p>Returns the value of <em>counter</em>.</p>
<h3 id="histogram-core.histogram">Histogram (core.histogram)</h3>
<p>Shared memory histogram with logarithmic buckets. Registered with <code>core.shm</code> as type <code>histogram</code>.</p>
<p>— Function <strong>histogram.new</strong> <em>min</em>, <em>max</em></p>
<p>Returns a new <code>histogram</code>, with buckets covering the range from <em>min</em> to <em>max</em>. The range between <em>min</em> and <em>max</em> will be divided logarithmically.</p>
<p>— Function <strong>histogram.create</strong> <em>name</em>, <em>min</em>, <em>max</em></p>
<p>Creates and returns a <code>histogram</code> as in <code>histogram.new</code> by <em>name</em>. If the file exists already, it will be cleared.</p>
<p>— Function <strong>histogram.open</strong> <em>name</em></p>
<p>Opens and returns <code>histogram</code> by <em>name</em> for reading.</p>
<p>— Method <strong>histogram:add</strong> <em>measurement</em></p>
<p>Adds <em>measurement</em> to <em>histogram</em>.</p>
<p>— Method <strong>histogram:iterate</strong> <em>prev</em></p>
<p>When used as <code>for count, lo, hi in histogram:iterate()</code>, visits all buckets in <em>histogram</em> in order from lowest to highest. <em>Count</em> is the number of samples recorded in that bucket, and <em>lo</em> and <em>hi</em> are the lower and upper bounds of the bucket. Note that <em>count</em> is an unsigned 64-bit integer; to get it as a Lua number, use <code>tonumber</code>.</p>
<p>If <em>prev</em> is given, it should be a snapshot of the previous version of the histogram. In that case, the <em>count</em> values will be returned as a difference between their values in <em>histogram</em> and their values in <em>prev</em>.</p>
<p>— Method <strong>histogram:snapshot</strong> [<em>dest</em>]</p>
<p>Copies out the contents of <em>histogram</em> into the <code>histogram</code> <em>dest</em> and returns <em>dest</em>. If <em>dest</em> is not given, the result will be a fresh <code>histogram</code>.</p>
<p>— Method <strong>histogram:clear</strong></p>
<p>Clears the buckets of <em>histogram</em>.</p>
<p>— Method **histogram:wrap_thunk* <em>thunk</em>, <em>now</em></p>
<p>Returns a closure that wraps <em>thunk</em>, measuring and recording the difference between calls to <em>now</em> before and after <em>thunk</em> into <em>histogram</em>.</p>
<p>— Method **histogram:summarize* <em>prev</em></p>
<p>Returns the approximate minimum, average, and maximum values recorded in <em>histogram</em>.</p>
<p>If <em>prev</em> is given, it should be a snapshot of a previous version of the histogram. In that case, this method returns the approximate minimum, average and maximum values for the difference between <em>histogram</em> and <em>prev</em>.</p>
<h2 id="lib-core.lib">Lib (core.lib)</h2>
<p>The <code>core.lib</code> module contains miscellaneous utilities.</p>
<p>— Function <strong>lib.equal</strong> <em>x</em>, <em>y</em></p>
<p>Predicate to test if <em>x</em> and <em>y</em> are structurally similar (isomorphic).</p>
<p>— Function <strong>lib.can_open</strong> <em>filename</em>, <em>mode</em></p>
<p>Predicate to test if file at <em>filename</em> can be successfully opened with <em>mode</em>.</p>
<p>— Function <strong>lib.can_read</strong> <em>filename</em></p>
<p>Predicate to test if file at <em>filename</em> can be successfully opened for reading.</p>
<p>— Function <strong>lib.can_write</strong> <em>filename</em></p>
<p>Predicate to test if file at <em>filename</em> can be successfully opened for writing.</p>
<p>— Function <strong>lib.readcmd</strong> <em>command</em>, <em>what</em></p>
<p>Runs Unix shell <em>command</em> and returns <em>what</em> of its output. <em>What</em> must be a valid argument to <code>file:read</code>.</p>
<p>— Function <strong>lib.readfile</strong> <em>filename</em>, <em>what</em></p>
<p>Reads and returns <em>what</em> from file at <em>filename</em>. <em>What</em> must be a valid argument to <code>file:read</code>.</p>
<p>— Function <strong>lib.writefile</strong> <em>filename</em>, <em>value</em></p>
<p>Writes <em>value</em> to file at <em>filename</em> using <code>file:write</code>. Returns the value returned by <code>file:write</code>.</p>
<p>— Function <strong>lib.readlink</strong> <em>filename</em></p>
<p>Returns the true name of symbolic link at <em>filename</em>.</p>
<p>— Function <strong>lib.dirname</strong> <em>filename</em></p>
<p>Returns the <code>dirname(3)</code> of <em>filename</em>.</p>
<p>— Function <strong>lib.basename</strong> <em>filename</em></p>
<p>Returns the <code>basename(3)</code> of <em>filename</em>.</p>
<p>— Function <strong>lib.firstfile</strong> <em>directory</em></p>
<p>Returns the filename of the first file in <em>directory</em>.</p>
<p>— Function <strong>lib.firstline</strong> <em>filename</em></p>
<p>Returns the first line of file at <em>filename</em> as a string.</p>
<p>— Function <strong>lib.load_string</strong> <em>string</em></p>
<p>Evaluates and returns the value of the Lua expression in <em>string</em>.</p>
<p>— Function <strong>lib.load_conf</strong> <em>filename</em></p>
<p>Evaluates and returns the value of the Lua expression in file at <em>filename</em>.</p>
<p>— Function <strong>lib.store_conf</strong> <em>filename</em>, <em>value</em></p>
<p>Writes <em>value</em> to file at <em>filename</em> as a Lua expression. Supports tables, strings and everything that can be readably printed using <code>print</code>.</p>
<p>— Function <strong>lib.bits</strong> <em>bitset</em>, <em>basevalue</em></p>
<p>Returns a bitmask using the values of <em>bitset</em> as indexes. The keys of <em>bitset</em> are ignored (and can be used as comments).</p>
<p>Example:</p>
<pre><code>bits({RESET=0,ENABLE=4}, 123) => 1<<0 | 1<<4 | 123</code></pre>
<p>— Function <strong>lib.bitset</strong> <em>value</em>, <em>n</em></p>
<p>Predicate to test if bit number <em>n</em> of <em>value</em> is set.</p>
<p>— Function <strong>lib.bitfield</strong> <em>size</em>, <em>struct</em>, <em>member</em>, <em>offset</em>, <em>nbits</em>, <em>value</em></p>
<p>Combined accesor and setter function for bit ranges of integers in cdata structs. Sets <em>nbits</em> (number of bits) starting from <em>offset</em> to <em>value</em>. If <em>value</em> is not given the current value is returned.</p>
<p><em>Size</em> may be one of 8, 16 or 32 depending on the bit size of the integer being set or read.</p>
<p><em>Struct</em> must be a pointer to a cdata object and <em>member</em> must be the literal name of a member of <em>struct</em>.</p>
<p>Example:</p>
<pre><code>local struct_t = ffi.typeof[[struct { uint16_t flags; }]]
-- Assuming `s' is an instance of `struct_t', set bits 4-7 to 0xF:
lib.bitfield(16, s, 'flags', 4, 4, 0xf)
-- Get the value:
lib.bitfield(16, s, 'flags', 4, 4) -- => 0xF</code></pre>
<p>— Function <strong>string:split</strong> <em>pattern</em></p>
<p>Returns an iterator over the string split by <em>pattern</em>. <em>Pattern</em> must be a valid argument to <code>string:gmatch</code>.</p>
<p>Example:</p>
<pre><code>for word, sep in ("foo!bar!baz"):split("(!)") do
print(word, sep)
end
> foo !
> bar !
> baz nil</code></pre>
<p>— Function <strong>lib.hexdump</strong> <em>string</em></p>
<p>Returns hexadecimal string for bytes in <em>string</em>.</p>
<p>— Function <strong>lib.hexundump</strong> <em>hexstring</em>, <em>n</em>, <em>error</em></p>
<p>Returns string of <em>n</em> bytes for <em>hexstring</em>. Throws an error if less than <em>n</em> hex-encoded bytes could be parsed unless <em>error</em> is <code>false</code>.</p>
<p><em>Error</em> is optional and can be the error message to throw.</p>
<p>— Function <strong>lib.comma_value</strong> <em>n</em></p>
<p>Returns a string for decimal number <em>n</em> with magnitudes separated by commas. Example:</p>
<pre><code>comma_value(1000000) => "1,000,000"</code></pre>
<p>— Function <strong>lib.random_bytes_from_dev_urandom</strong> <em>length</em></p>
<p>Return <em>length</em> bytes of random data, as a byte array, taken from <code>/dev/urandom</code>. Suitable for cryptographic usage.</p>
<p>— Function <strong>lib.random_bytes_from_math_random</strong> <em>length</em></p>
<p>Return <em>length</em> bytes of random data, as a byte array, where each byte was taken from <code>math.random(0, 255)</code>. <em>Not</em> suitable for cryptographic usage.</p>
<p>— Function <strong>lib.random_bytes</strong> <em>length</em> — Function <strong>lib.randomseed</strong> <em>seed</em></p>
<p>Initialize Snabb’s random number generation facility. If <em>seed</em> is nil, then the Lua <code>math.random()</code> function will be seeded from <code>/dev/urandom</code>, and <code>lib.random_bytes</code> will be initialized to <code>lib.random_bytes_from_dev_urandom</code>. This is Snabb’s default mode of operation.</p>
<p>Sometimes it’s useful to make Snabb use deterministic random numbers. In that case, pass a seed to <strong>lib.randomseed</strong>; Snabb will set <code>lib.random_bytes</code> to <code>lib.random_bytes_from_math_random</code>, and also print out a message to stderr indicating that we are using lower-quality deterministic random numbers.</p>
<p>As part of its initialization process, Snabb will call <code>lib.randomseed</code> with the value of the <code>SNABB_RANDOM_SEED</code> environment variable (if any). Set this environment variable to enable deterministic random numbers.</p>
<p>— Function <strong>lib.bounds_checked</strong> <em>type</em>, <em>base</em>, <em>offset</em>, <em>size</em></p>
<p>Returns a table that acts as a bounds checked wrapper around a C array of <em>type</em> and <em>size</em> starting at <em>base</em> plus <em>offset</em>. <em>Type</em> must be a ctype and the caller must ensure that the allocated memory region at <em>base</em>/<em>offset</em> is at least <code>sizeof(type)*size</code> bytes long.</p>
<p>— Function <strong>lib.throttle</strong> <em>seconds</em></p>
<p>Return a closure that returns <code>true</code> at most once during any <em>seconds</em> (a floating point value) time interval, otherwise false.</p>
<p>— Function <strong>lib.timeout</strong> <em>seconds</em></p>
<p>Returns a closure that returns <code>true</code> if <em>seconds</em> (a floating point value) have elapsed since it was created, otherwise false.</p>
<p>— Function <strong>lib.waitfor</strong> <em>condition</em></p>
<p>Blocks until the function <em>condition</em> returns a true value.</p>
<p>— Function <strong>lib.waitfor2</strong> <em>name</em>, <em>condition</em>, <em>attempts</em>, <em>interval</em></p>
<p>Repeatedly calls the function <em>condition</em> in <em>interval</em> (milliseconds). If <em>condition</em> returns a true value <code>waitfor2</code> returns. If <em>condition</em> does not return a true value after <em>attempts</em> <code>waitfor2</code> raises an error identified by <em>name</em>.</p>
<p>— Function <strong>lib.yesno</strong> <em>flag</em></p>
<p>Returns the string <code>"yes"</code> if <em>flag</em> is a true value and <code>"no"</code> otherwise.</p>
<p>— Function <strong>lib.align</strong> <em>value</em>, <em>size</em></p>
<p>Return the next integer that is a multiple of <em>size</em> starting from <em>value</em>.</p>
<p>— Function <strong>lib.csum</strong> <em>pointer</em>, <em>length</em></p>
<p>Computes and returns the “IP checksum” <em>length</em> bytes starting at <em>pointer</em>.</p>
<p>— Function <strong>lib.update_csum</strong> <em>pointer</em>, <em>length</em>, <em>checksum</em></p>
<p>Returns <em>checksum</em> updated by <em>length</em> bytes starting at <em>pointer</em>. The default of <em>checksum</em> is <code>0LL</code>.</p>
<p>— Function <strong>lib.finish_csum</strong> <em>checksum</em></p>
<p>Returns the finalized <em>checksum</em>.</p>
<p>— Function <strong>lib.malloc</strong> <em>etype</em></p>
<p>Returns a pointer to newly allocated DMA memory for <em>etype</em>.</p>
<p>— Function <strong>lib.deepcopy</strong> <em>object</em></p>
<p>Returns a copy of <em>object</em>. Supports tables as well as ctypes.</p>
<p>— Function <strong>lib.array_copy</strong> <em>array</em></p>
<p>Returns a copy of <em>array</em>. <em>Array</em> must not be a “sparse array”.</p>
<p>— Function <strong>lib.htonl</strong> <em>n</em></p>
<p>— Function <strong>lib.htons</strong> <em>n</em></p>
<p>Host to network byte order conversion functions for 32 and 16 bit integers <em>n</em> respectively. Unsigned.</p>
<p>— Function <strong>lib.ntohl</strong> <em>n</em></p>
<p>— Function <strong>lib.ntohs</strong> <em>n</em></p>
<p>Network to host byte order conversion functions for 32 and 16 bit integers <em>n</em> respectively. Unsigned.</p>
<p>— Function <strong>lib.random_bytes</strong> <em>count</em></p>
<p>Return a fresh array of <em>count</em> random bytes. Suitable for cryptographic usage.</p>
<p>— Function <strong>lib.parse</strong> <em>arg</em>, <em>config</em></p>
<p>Validates <em>arg</em> against the specification in <em>config</em>, and returns a fresh table containing the parameters in <em>arg</em> and any omitted optional parameters with their default values. Given <em>arg</em>, a table of parameters or <code>nil</code>, assert that from <em>config</em> all of the required keys are present, fill in any missing values for optional keys, and error if any unknown keys are found. <em>Config</em> has the following format:</p>
<pre><code>config := { key = {[required=boolean], [default=value]}, ... }</code></pre>
<p>Each key is optional unless <code>required</code> is set to a true value, and its default value defaults to <code>nil</code>.</p>
<p>Example:</p>
<pre><code>lib.parse({foo=42, bar=43}, {foo={required=true}, bar={}, baz={default=44}})
=> {foo=42, bar=43, baz=44}</code></pre>
<ul>
<li>Function <strong>lib.set</strong> <em>vargs</em></li>
</ul>
<p>Reads a variable number of arguments and returns a table representing a set. The returned value can be used to query whether an element belongs or not to the set.</p>
<p>Example:</p>
<pre><code>local t = set('foo', 'bar')
t['foo'] -- yields true.
t['quax'] -- yields false.</code></pre>
<h2 id="multiprocess-operation-core.worker">Multiprocess operation (core.worker)</h2>
<p>Snabb can operate as a <em>group</em> of cooperating processes. The <em>main</em> process is the initial one that you start directly. The optional <em>worker</em> processes are children spawned when the main process calls the <code>core.worker</code> module.</p>
<div class="figure">
<img src="" alt="Multiprocessing" />
<p class="caption">Multiprocessing</p>
</div>
<p>Each worker is a complete Snabb process. They can define app networks, run the engine, and do everything else that ordinary Snabb processes do. The exact behavior of each worker is determined by a Lua expression provided upon creation.</p>
<p>Groups of Snabb processes each have the following special properties:</p>
<ul>
<li><strong>Group termination</strong>: Terminating the main process automatically terminates all of the workers. This works for all process termination scenarios including <code>kill -9</code>.</li>
<li><strong>Shared DMA memory</strong>: DMA memory pointers obtained with <code>memory.dma_alloc()</code> are usable by all processes in the group. This means that you can share DMA memory pointers between processes, for example via <code>shm</code> shared memory objects, and reference them from any process. (The memory is automatically mapped at the expected address via a <code>SEGV</code> signal handler.)</li>
<li><strong>PCI device shutdown</strong>: For each PCI device opened by a process within the group, bus mastering (DMA) is disabled upon termination before any DMA memory is returned to the kernel. This prevents “dangling” DMA requests from corrupting memory that has been freed and reused. See <a href="#pci-lib.hardware.pci">lib.hardware.pci</a> for details.</li>
</ul>
<p>The <code>core.worker</code> API functions are available in the main process only:</p>
<p>— Function <strong>worker.start</strong> <em>name</em> <em>luacode</em></p>
<p>Start a named worker process. The worker starts with a completely fresh Snabb process image (<code>fork()+execve()</code>) and then executes the string <em>luacode</em> as a Lua source code expression.</p>
<p>Example:</p>
<pre><code>worker.start("myworker", [[
print("hello world, from a Snabb worker process!")
print("could configure and run the engine now...")
]])</code></pre>
<p>— Function <strong>worker.stop</strong> <em>name</em></p>
<p>Stop a named worker process. The worker is abruptly killed.</p>
<p>Example:</p>
<pre><code>worker.stop("myworker")</code></pre>
<p>— Function <strong>worker.status</strong></p>
<p>Return a table summarizing the status of all workers. The table key is the worker name and the value is a table with <code>pid</code> and <code>alive</code> attributes.</p>
<p>Example:</p>
<pre><code>for w, s in pairs(worker.status()) do
print((" worker %s: pid=%s alive=%s"):format(
w, s.pid, s.alive))
end</code></pre>
<p>Output:</p>
<pre><code>worker w3: pid=21949 alive=true
worker w1: pid=21947 alive=true
worker w2: pid=21948 alive=true</code></pre>
<h2 id="main">Main</h2>
<p>Snabb designs can be run either with:</p>
<pre><code>snabb <snabb-arg>* <design> <design-arg>*</code></pre>
<p>or</p>
<pre><code>#!/usr/bin/env snabb <snabb-arg>*
...</code></pre>
<p>The <em>main</em> module provides an interface for running Snabb scripts. It exposes various operating system functions to scripts.</p>
<p>— Field <strong>main.parameters</strong></p>
<p>A list of command-line arguments to the running script. Read-only.</p>
<p>— Function <strong>main.exit</strong> <em>status</em></p>
<p>Cleanly exits the process with <em>status</em>.</p>
<h1 id="basic-apps-apps.basic.basic_apps">Basic Apps (apps.basic.basic_apps)</h1>
<p>The module <em>apps.basic.basic_apps</em> provides apps with general functionality for use in you app networks.</p>
<h2 id="source">Source</h2>
<p>The <code>Source</code> app is a synthetic packet generator. On each breath it fills each attached output link with new packets. It accepts a number as its configuration argument which is the byte size of the generated packets. By default, each packet is 60 bytes long. The packet data is initialized with zero bytes.</p>
<div class="figure">
<img src="" alt="Source" />
<p class="caption">Source</p>
</div>
<h2 id="join">Join</h2>
<p>The <code>Join</code> app joins together packets from N input links onto one output link. On each breath it outputs as many packets as possible from the inputs onto the output.</p>
<div class="figure">
<img src="" alt="Join" />
<p class="caption">Join</p>
</div>
<h2 id="split">Split</h2>
<p>The <code>Split</code> app splits packets from multiple inputs across multiple outputs. On each breath it transfers as many packets as possible from the input links to the output links.</p>
<div class="figure">
<img src="" alt="Split" />
<p class="caption">Split</p>
</div>
<h2 id="sink">Sink</h2>
<p>The <code>Sink</code> app receives all packets from any number of input links and discards them. This can be handy in combination with a <code>Source</code>.</p>
<div class="figure">
<img src="" alt="Sink" />
<p class="caption">Sink</p>
</div>
<h2 id="tee">Tee</h2>
<p>The <code>Tee</code> app receives all packets from any number of input links and transfers each received packet to all output links. It can be used to merge and/or duplicate packet streams</p>
<div class="figure">
<img src="" alt="Tee" />
<p class="caption">Tee</p>
</div>
<h2 id="repeater">Repeater</h2>
<p>The <code>Repeater</code> app collects all packets received from the <code>input</code> link and repeatedly transfers the accumulated packets to the <code>output</code> link. The packets are transmitted in the order they were received.</p>
<div class="figure">
<img src="" alt="Repeater" />
<p class="caption">Repeater</p>
</div>
<h2 id="truncate">Truncate</h2>
<p>The <code>Truncate</code> app sends all packets received from the <code>input</code> to the <code>output</code> link and truncates or zero pads each packet to a given length. It accepts a number as its configuration argument which is the length of the truncated or padded packets.</p>
<div class="figure">
<img src="" alt="Truncate" />
<p class="caption">Truncate</p>
</div>
<h2 id="sample">Sample</h2>
<p>The <code>Sample</code> app forwards packets every <em>n</em>th packet from the <code>input</code> link to the <code>output</code> link, and drops all others packets. It accepts a number as its configuration argument which is <em>n</em>.</p>
<div class="figure">
<img src="" alt="Sample" />
<p class="caption">Sample</p>
</div>
<h1 id="intel-82599-ethernet-controller-apps">Intel 82599 Ethernet Controller Apps</h1>
<h2 id="intel82599-apps.intel.intel_app">Intel82599 (apps.intel.intel_app)</h2>
<p>The <code>Intel82599</code> drives one port of an Intel 82599 Ethernet controller. Packets taken from the <code>rx</code> port are transmitted onto the network. Packets received from the network are put on the <code>tx</code> port.</p>
<div class="figure">
<img src="" alt="Intel82599" />
<p class="caption">Intel82599</p>
</div>
<p>— Method <strong>Intel82599.dev:get_rxstats</strong></p>
<p>Returns a table with the following keys:</p>
<ul>
<li><code>counter_id</code> - Counter id</li>
<li><code>packets</code> - Number of packets received</li>
<li><code>dropped</code> - Number of packets dropped</li>
<li><code>bytes</code> - Total bytes received</li>
</ul>
<p>— Method <strong>Intel82599.dev:get_txstats</strong></p>
<p>Returns a table with the following keys:</p>
<ul>
<li><code>counter_id</code> - Counter id</li>
<li><code>packets</code> - Number of packets sent</li>
<li><code>bytes</code> - Total bytes sent</li>
</ul>
<h3 id="configuration">Configuration</h3>
<p>The <code>Intel82599</code> app accepts a table as its configuration argument. The following keys are defined:</p>
<p>— Key <strong>pciaddr</strong></p>
<p><em>Required</em>. The PCI address of the NIC as a string.</p>
<p>— Key <strong>macaddr</strong></p>
<p><em>Optional</em>. The MAC address to use as a string. The default is a wild-card (e.g. accept all packets).</p>
<p>— Key <strong>vlan</strong></p>
<p><em>Optional</em>. A twelve bit integer (0-4095). If set, incoming packets from other VLANs are dropped and outgoing packets are tagged with a VLAN header.</p>
<p>— Key <strong>vmdq</strong></p>
<p><em>Optional</em>. Boolean, defaults to false. Enables interface virtualization. Allows to have multiple <code>Intel82599</code> apps per port. If enabled, <em>macaddr</em> must be specified.</p>
<p>— Key <strong>mirror</strong></p>
<p><em>Optional</em>. A table. If set, this app will receive copies of all selected packets on the physical port. The selection is configured by setting keys of the <em>mirror</em> table. Either <em>mirror.pool</em> or <em>mirror.port</em> may be set.</p>
<p>If <em>mirror.pool</em> is <code>true</code> all pools defined on this physical port are mirrored. If <em>mirror.pool</em> is an array of pool numbers then the specified pools are mirrored.</p>
<p>If <em>mirror.port</em> is one of “in”, “out” or “inout” all incoming and/or outgoing packets on the port are mirrored respectively. Note that this does not include internal traffic which does not enter or exit through the physical port.</p>
<p>— Key <strong>rxcounter</strong></p>
<p>— Key <strong>txcounter</strong></p>
<p><em>Optional</em>. Four bit integers (0-15). If set, incoming/outgoing packets will be counted in the selected statistics counter respectively. Multiple apps can share a counter. To retrieve counter statistics use <code>Intel82599.dev:get_rxstats()</code> and <code>Intel82599.dev:get_txstats()</code>.</p>
<p>— Key <strong>rate_limit</strong></p>
<p><em>Optional</em>. Number. Limits the maximum Mbit/s to transmit. Default is 0 which means no limit. Only applies to outgoing traffic.</p>
<p>— Key <strong>priority</strong></p>
<p><em>Optional</em>. Floating point number. Weight for the <em>round-robin</em> algorithm used to arbitrate transmission when <em>rate_limit</em> is not set or adds up to more than the line rate of the physical port. Default is 1.0 (scaled to the geometric middle of the scale which goes from 1/128 to 128). The absolute value is not relevant, instead only the ratio between competing apps controls their respective bandwidths. Only applies to outgoing traffic.</p>
<p>For example, if two apps without <em>rate_limit</em> set have the same <em>priority</em>, both get the same output bandwidth. If the priorities are 3.0/1.0, the output bandwidth is split 75%/25%. Likewise, 1.0/0.333 or 1.5/0.5 yield the same result.</p>
<p>Note that even a low-priority app can use the whole line rate unless other (higher priority) apps are using up the available bandwidth.</p>
<h3 id="performance">Performance</h3>
<p>The <code>Intel82599</code> app can transmit and receive at approximately 10 Mpps per processor core.</p>
<h3 id="hardware-limits">Hardware limits</h3>
<p>Each physical Intel 82599 port supports the use of up to:</p>
<ul>
<li>64 <em>pools</em> (virtualized <code>Intel82599</code> app instances)</li>
<li>127 MAC addresses (see the <code>macaddr</code> configuration option)</li>
<li>64 VLANs (see the <code>vlan</code> configuration option)</li>
<li>4 <em>mirror pools</em> (see the <code>mirror</code> configuration option)</li>
</ul>
<h2 id="loadgen-apps.intel.loadgen">LoadGen (apps.intel.loadgen)</h2>
<p><code>LoadGen</code> is a <em>load generator</em> app based on the Intel 82599 Ethernet controller. It reads up to 32,000 packets from the <code>input</code> port and transmits them repeatedly onto the network. All incoming packets are dropped.</p>
<div class="figure">
<img src="" alt="LoadGen" />
<p class="caption">LoadGen</p>
</div>
<h3 id="configuration-1">Configuration</h3>
<p>The <code>LoadGen</code> app accepts a string as its configuration argument. The given string denotes the PCI address of the NIC to use.</p>
<h3 id="performance-1">Performance</h3>
<p>The <code>LoadGen</code> app can transmit at line-rate (14 Mpps) without significant CPU usage.</p>
<h1 id="intel-i210-i350-82599-ethernet-controller-apps-apps.intel_mp.intel_mp">Intel i210 / i350 / 82599 Ethernet Controller apps (apps.intel_mp.intel_mp)</h1>
<p>The <code>intel_mp.Intel</code> app provides drivers for Intel i210/i250/82599 based network cards. The driver exposes multiple receive and transmit queues that can be attached to separate instances of the app on different processes.</p>
<p>The links are named <code>input</code> and <code>output</code>.</p>
<div class="figure">
<img src="" alt="Intel" />
<p class="caption">Intel</p>
</div>
<h2 id="caveats">Caveats</h2>
<p>If attaching multiple processes to a single NIC, performance appears better with <code>engine.busywait = false</code>.</p>
<p>The <code>intel_mp.Intel</code> app can drive an Intel 82599 NIC at 14 million pps.</p>
<p>— Method <strong>Intel:get_rxstats</strong></p>
<p>Returns a table with the following keys:</p>
<ul>
<li><code>counter_id</code> - Counter id</li>
<li><code>packets</code> - Number of packets received</li>
<li><code>dropped</code> - Number of packets dropped</li>
<li><code>bytes</code> - Total bytes received</li>
</ul>
<p>— Method <strong>Intel:get_txstats</strong></p>
<p>Returns a table with the following keys:</p>
<ul>
<li><code>counter_id</code> - Counter id</li>
<li><code>packets</code> - Number of packets sent</li>
<li><code>bytes</code> - Total bytes sent</li>
</ul>
<h2 id="configuration-2">Configuration</h2>
<p>— Key <strong>pciaddr</strong></p>
<p><em>Required</em>. The PCI address of the NIC as a string.</p>
<p>— Key <strong>ndesc</strong></p>
<p><em>Optional</em>. Number of DMA descriptors to use i.e. size of the DMA transmit and receive queues. Must be a multiple of 128. Default is not specified but assumed to be broadly applicable.</p>
<p>— Key <strong>rxq</strong></p>
<p><em>Optional</em>. The receive queue to attach to, numbered from 0. The default is 0. When VMDq is enabled, this number is used to index a queue (0 or 1) for the selected pool. Passing <code>false</code> will disable the receive queue.</p>
<p>— Key <strong>txq</strong></p>
<p><em>Optional</em>. The transmit queue to attach to, numbered from 0. The default is 0. Passing <code>false</code> will disable the transmit queue.</p>
<p>— Key <strong>vmdq</strong></p>
<p><em>Optional</em>. A boolean parameter that specifies whether VMDq (Virtual Machine Device Queues) is enabled. When VMDq is enabled, each instance of the driver is associated with a <em>pool</em> that can be assigned a MAC address or VLAN tag. Packets are delivered to pools that match the corresponding MACs or VLAN tags. Each pool may be associated with several receive and transmit queues.</p>
<p>For a given NIC, all driver instances should have this parameter either enabled or disabled uniformly. If this is enabled, <em>macaddr</em> must be specified.</p>
<p>— Key <strong>vmdq_queueing_mode</strong></p>
<p><em>Optional</em>. Sets the queueing mode to use in VMDq mode. Has no effect when VMDq is disabled. The available queueing modes for the 82599 are <code>"rss-64-2"</code> (the default with 64 pools, 2 queues each) and <code>"rss-32-4"</code> (32 pools, 4 queues each). The i350 provides only a single mode (8 pools, 2 queues each) and hence ignores this option.</p>
<p>— Key <strong>poolnum</strong></p>
<p><em>Optional</em>. The VMDq pool to associate with, numbered from 0. The default is to select a pool number automatically. The maximum pool number depends on the queueing mode.</p>
<p>— Key <strong>macaddr</strong></p>
<p><em>Optional</em>. The MAC address to use as a string. The default is a wild-card (i.e., accept all packets).</p>
<p>— Key <strong>vlan</strong> <em>Optional</em>. A twelve-bit integer (0-4095). If set, incoming packets from other VLANs are dropped and outgoing packets are tagged with a VLAN header.</p>