A reference implementation of SIGCOMM'22 Paper FlyMon.
Key Features • Get Started • Simulations • License • Links
- P416-based hardware implementation.
- Jinja2 templates used to generate P4 codes according to variable configurations (e.g., CMU-Groups, Memory Size, Candidate Key Set).
- Several built-in algorithms used to measure various flow attributes.
- A reference control plane framework realizing task reconfiguration, resource management, data collection, and task query.
🔔 Our slides can be downloaded here.
The figure below shows that the FlyMon hardware implementation is based on the Tofino hardware platform, including the SDE, runtime interfaces, etc.
To better test and use FlyMon, we made some engineering efforts to implement the data plane and the control plane.
For the data plane, our P4 codes are generated using Jinja2 templates (inspired from the implementation of BeauCoup). The Jinja2 templates (located in p4_templates) allow us to flexibly and quickly extend our data plane codes to avoid various bugs. For example, when deploying 9 CMU-Groups, the P4 codes reach over 5000 LOC, which is difficult to maintain manually. More importantly, we can easily modify the configuration of the data plane, such as the number of CMU-Groups, the set of candidate keys, the size of the static memory of a single CMU, etc.
For the control plane, we provide user-friendly, interactive interfaces to dynamically configure tasks (i.e., add and delete tasks). We perform layers of abstraction, gradually shielding the underlying hardware details. When compiling the data plane code, the FlyMon compiler also generates a configuration file for the control plane, which is used to adjust the control plane's interfaces to adapt to user-customized underlayer data plane configurations.
Below we show how to use these codes.
This repository has strict hardware and software requirements.
Hardware Requirements
- A tofino-based hardware switch (e.g., Wdege-100BF-XX) or a tofino model.
- At least one server with QSFP28 connectors and wires if you use a hardware switch.
Software Requirements
- Switch OS: OpenNetworkLinux 4.14.151
- Python: 3.8.10
- SDE: Version 9.7.0+ (the same is best)
🔔 In this document, all 'python' and 'pip' refer to the python version of 3.8.10.
There are some dependencies for control plane functions. To install them.
git clone "https://github.com/NASA-NJU/FlyMon.git"
cd FlyMon
pip install -r ./requirements.txt
To generate your custom data plane code, use the Jinja2 code generator we provide.
python flymon_compiler.py -n 9 -m memory_level_mini
The above command will generate 9 CMU-Groups in the data plane (see p4src), and each CMU has a static (maximum) memory type of 'memory_level_mini' (32 counters in each register).
🔔 For easy viewing of memory status, we generate mini-level CMUs (i.e., only 32 16-bit counters in each CMU) here. You can choose a larger level of memory (e.g., memory_level_8) for more practical purposes. The available memory levels are list in
flymon_compiler.py
.
Once the data plane codes are generated, you can build the p4 codes with bf-p4c. Here we give a setup script if you don't known how to compile the codes.
# If you are working on SDE 9.7.0+
# Make sure you are in the directory of FlyMon, currently.
export FLYMON_DIR=`pwd`
./setup.sh
🔔 You also need to check if SDE environment variables (e.g.,
SDE
andSDE_INSTALL
) are set correctly.
🔔 The compilation process usually takes 20~60 minutes. Yes, it compiles so slowly QAQ. We expect some deep optimizations for the compiler. Note taht although the compilation is slow, this does not affect the forwarding performance of the switch after the compilation is complete.
Launching FlyMon requires starting the data plane and the control plane separately.
Firstly, load the program for the data plane.
$SDE/run_switchd.sh -p flymon
If you are working on a Tofino Model, we need to run the model before running the switchd. You also need to setup several virtual ports for the model.
$SDE/run_tofino_model.sh -p flymon
$SDE/run_switchd.sh -p flymon
Secondly, start the FlyMon interactive control plane in another terminal.
cd $FLYMON_DIR/control_plane/
./run_controller.sh
If all goes well, you will be taken to the command line interface of FlyMon.
----------------------------------------------------
______ __ __ ___
/ ____/ / / __ __ / |/ / ____ ____
/ /_ / / / / / / / /|_/ / / __ \ / __ \
/ __/ / / / /_/ / / / / / / /_/ / / / / /
/_/ /_/ \__, / /_/ /_/ \____/ /_/ /_/
/____/
----------------------------------------------------
An on-the-fly network measurement system.
**NOTE**: FlyMon's controller will clear all previous data plane
tasks/datas when setup.
flymon>
You can use `tab`, 'help', and '-h' to get the relevant commands and their prompts.
flymon> <tab><tab>
EOF add_port del_forward help read_cmug reset_all shell show_task
add_forward add_task del_task query_task read_task send_packets show_cmug
flymon> add_task -h
usage: controller_main.py [-h] -f FILTER -k KEY -a ATTRIBUTE -m MEM_SIZE [-q]
optional arguments:
-h, --help show this help message and exit
-f FILTER, --filter FILTER
e.g., 10.0.0.0/8,20.0.0.0/16 or 10.0.0.0/8,* or *,* Default: *,*
-k KEY, --key KEY e.g., hdr.ipv4.src_addr/24, hdr.ipv4.dst_addr/32
-a ATTRIBUTE, --attribute ATTRIBUTE
e.g., the attribute, e.g., frequency(1, cms)
-m MEM_SIZE, --mem_size MEM_SIZE
e.g., the number of buckets, e.g., 32768
-q, --quiet, --flag do not need the log?
flymon> help add_task
Add a task to CMU-Group.
Args:
"-f", "--filter" : what traffic does this task focus on
"-k", "--key" : flow key of the task
"-a", "--attribute" : flow attribute of the task
"-m", "--mem_size" : memory size in number of buckets
**Examples** :
a) [Per-flow Size]
add_task -f 10.0.0.0/8,* -k hdr.ipv4.src_addr -a frequency(1,cms) -m 48
b) [Single-key Distinct]
add_task -f 10.0.0.0/8,* -k None -a distinct(hdr.ipv4.src_addr,hll) -m 32
c) [Multi-key Distinct]
add_task -f 10.0.0.0/8,* -k hdr.ipv4.dst_addr -a distinct(hdr.ipv4.src_addr,beaucoup) -m 96
d) [Bloom Filter]
add_task -f 10.0.0.0/8,* -k None -a existence(hdr.ipv4.src_addr,bloomfilter) -m 32
e) [Max Packet Size]
add_task -f 20.0.0.0/8,* -k hdr.ipv4.dst_addr -a max(pkt_size,sumax) -m 48
Returns:
Added task id with task info.
Exceptions:
parser error of the key, the attribute, the memory.
Currently, we implement the functions of task deployment, deletion, status show, data reading, etc. We also implement other features, such as the simple configuration of ports and installation forwarding rules. But these functions can also be completed from the original SDE interfaces.
We demonstrate the dynamic features of FlyMon through several typical use cases. The flexibility of FlyMon lies in the ability to arbitrarily adjust the flow key, flow attribute, and memory size. The tasks that FlyMon can perform are not limited to the below use cases. We will add more use cases in the future.
Frequency Estimation
Suppose we want to measure the frequency (i.e., number of packets) of each SrcIP for the traffic with SrcIP in 10.0.0.0/8.
We can define the key as hdr.ipv4.src_addr
and attribute as frequency(1). We can deploy this measurement task with the add_task
command.
flymon> add_task -f 10.0.0.0/8,* -k hdr.ipv4.src_addr -a frequency(1,cms) -m 48
The above command will deploy a Count-Min Sketch (d=3, w=16) in the data plane for this task.
If there are enough resources in the data plane to deploy this measurement task, you will get the following output.
Required resources:
key=hdr.ipv4.src_addr/32, param1=1, param2=65535, operation=CondADD
key=hdr.ipv4.src_addr/32, param1=1, param2=65535, operation=CondADD
key=hdr.ipv4.src_addr/32, param1=1, param2=65535, operation=CondADD
group_id=1, hkey=1, cmu_id=1, memory_type:HALF offset:0
group_id=1, hkey=1, cmu_id=2, memory_type:HALF offset:0
group_id=1, hkey=1, cmu_id=3, memory_type:HALF offset:0
----------------------------------------------------
[Active Task]
Filter= [('10.0.0.0', '255.0.0.0'), ('0.0.0.0', '0.0.0.0')]
ID = 1
Key = hdr.ipv4.src_addr/32
Attribute = Frequency(param=1, algorithm=CMS)
Memory = 48(3*16)
Locations:
- loc0 = group_id=1, hkey=1, cmu_id=1, memory_type:HALF offset:0
- loc1 = group_id=1, hkey=1, cmu_id=2, memory_type:HALF offset:0
- loc2 = group_id=1, hkey=1, cmu_id=3, memory_type:HALF offset:0
[Success] Allocate TaskID: 1
The above output tells us that the task is successfully deployed to CMU-Group 5 and the task identifier in FlyMon is 1. The controller deploys the task with a 3 rows CM-Sketch and divides the 48 counters evenly over the 3 rows. Since we set the memory of each CMU to 32 (i.e., memory_level_mini
), the memory type of each row is HALF.
We can check the status of CMU-Group and information about the task by show_cmug
command and show_task
command, respectively.
flymon> show_task -t 1
----------------------------------------------------
[Active Task]
Filter= [('10.0.0.0', '255.0.0.0'), ('0.0.0.0', '0.0.0.0')]
ID = 1
Key = hdr.ipv4.src_addr/32
Attribute = Frequency(param=1, algorithm=CMS)
Memory = 48(3*16)
Locations:
- loc0 = group_id=1, hkey=1, cmu_id=1, memory_type:HALF offset:0
- loc1 = group_id=1, hkey=1, cmu_id=2, memory_type:HALF offset:0
- loc2 = group_id=1, hkey=1, cmu_id=3, memory_type:HALF offset:0
flymon> show_cmug -g 1
------------------------------------------------------------
Status of CMU-Group 1
------------------------------------------------------------
Compressed Key 1 (32b): hdr.ipv4.src_addr/32
Compressed Key 2 (16b): Empty
------------------------------------------------------------
CMU-1 Rest Memory: 16
CMU-2 Rest Memory: 16
CMU-3 Rest Memory: 16
------------------------------------------------------------
We can read the full memory of the task with the read_task
command.
flymon> read_task -t 1
Read all data for task: 1
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
Now, we inject some traffic into the switch to verify the measurement task.
🔔 You can inject traffic into the physical switch through an additional server. You can also inject traffic into the Tofino Model's virtual interface via software (e.g., scapy). Both require the FlyMon data plane's simple_fwd table to be configured in advance. We offer
add_port
andadd_forward
commands to fast configure you switch/model.
If you are using a Tofino Model, we provide some commands to help you perform the tests.
flymon> add_forward -s 0 -d 1
flymon> send_packets -l 64 -n 20 -p 0 -s 10.0.0.0/8
Send a packet with src_ip=10.0.0.1, dst_ip=30.60.90.1, pktlen=64.
Send a packet with src_ip=10.0.0.2, dst_ip=30.60.90.1, pktlen=64.
Send a packet with src_ip=10.0.0.3, dst_ip=30.60.90.1, pktlen=64.
...
The add_forward
command inserts a forwarding rule in simple_fwd
table of the data plane. It will forward the packets from Port-0 to Port-1 (i.e., DP Port). The send_packets
command sends 20 packets with length 64 and SrcIP in network range 10.0.0.0/8.
After generating the traffic, we can check the memory of the task again.
flymon> read_task -t 1
Read all data for task: 1
[2, 1, 1, 1, 0, 1, 2, 1, 1, 1, 2, 2, 1, 2, 1, 1]
[0, 0, 0, 0, 3, 0, 4, 3, 0, 0, 0, 0, 2, 0, 4, 4]
[2, 1, 1, 2, 2, 1, 1, 2, 1, 1, 1, 1, 0, 1, 2, 1]
The memory above is a visual representation of the Count-min sketch. To query a specific Key (e.g., SrcIP=10.0.0.1), we can use the query_task
command.
flymon> query_task -t 1 -k 10.0.0.1,*,*,*,*
1
Note that the queried key must be a complete 5-tuple.
Set arbitrary flow key from a candidate key set
With the dynamic feature of Tofino's hash computation unit, FlyMon can support setting any subpart (bit level) of the candidate key set as flow key. In this implementation, the candidate key set is the 5-tuple (i.e., SrcIP, DstIP, SrcPort, DstPort, Protocol). The source-destination port can be either for the UDP protocol or for the TCP protocol. To demonstrate this, we can test it with several tasks.Task 1: Set the Key as hdr.ipv4.src_addr/24
flymon> reset_all
flymon> add_task -f *,* -k hdr.ipv4.src_addr/24 -a frequency(1,cms) -m 48
🔔 The
reset_all
command is optional. It will clear all data plane tasks and controller plane status. The purpose of executing this command here is to make it easier for users to follow this manual. In other words, when this command is executed, all subsequent tasks will be allocated from TaskID=1, which is convenient for later manuals with the fixed task_id and CMU-Group allocations. If you do not execute this command. You need to correctly select the TaskID assigned to you by the task manager when you read/query the task.
We then generate traffic with SrcIP in 192.168.0.0/24. As the task only care about the first 24 bits of SrcIP, all the packets will be treated as a single flow. Count-Min Sketch should have only one non-zero counter in each row.
flymon> add_forward -s 0 -d 1
flymon> send_packets -p 0 -s 192.168.0.0/24 -n 10
Send a packet with src_ip=192.168.0.1, dst_ip=30.60.90.1, pktlen=64.
Send a packet with src_ip=192.168.0.2, dst_ip=30.60.90.1, pktlen=64.
Send a packet with src_ip=192.168.0.3, dst_ip=30.60.90.1, pktlen=64.
...
flymon> read_task -t 1
Read all data for task: 1
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 10, 0]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 10, 0, 0, 0, 0, 0]
[0, 10, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
The experimental result is consistent with our expectation.
Task 2: Set the Key as hdr.ipv4.src_addr, hdr.ipv4.dst_addr
We use following command to deploy this task. The controller will assign this task as Task 2.
add_task -f *,* -k hdr.ipv4.src_addr,hdr.ipv4.dst_addr -a frequency(1,cms) -m 48
...
...
[Success] Allocate TaskID: 2
To test this, we send two packets. The two packets have the same src IP address, but have different Dst IP addresses. The measurement task should treat the two packets as two distinct flows. In other words, each row of the task should have two non-zero counters, unless there is a hash conflict.
flymon> send_packets -p 0 -s 10.0.0.1 -d 20.0.0.0/8 -n 2
Send a packet with src_ip=10.0.0.1, dst_ip=20.0.0.1, pktlen=64.
Send a packet with src_ip=10.0.0.1, dst_ip=20.0.0.2, pktlen=64.
flymon> read_task -t 2
Read all data for task: 2
[1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 1, 0, 0]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0]
The experimental result is consistent with our expectation.
Task 3: Set the Key as hdr.ipv4.src_addr, hdr.ipv4.dst_addr/24
Follow the Task 2, if we add a prefix length of 24 to the DstIP in the key. There should be only one non-zero counter per row in the measurement data.
flymon> add_task -f *,* -k hdr.ipv4.src_addr,hdr.ipv4.dst_addr/24 -a frequency(1,cms) -m 48
...
[Success] Allocate TaskID: 3
flymon> send_packets -p 0 -s 10.0.0.1 -d 20.0.0.0/8 -n 2
Send a packet with src_ip=10.0.0.1, dst_ip=20.0.0.1, pktlen=64.
Send a packet with src_ip=10.0.0.1, dst_ip=20.0.0.2, pktlen=64.
flymon> read_task -t 3
Read all data for task: 3
[0, 0, 0, 0, 0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 0]
[2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
The experimental result is consistent with our expectation.
Through the above experiments, we basically verified that flymon supports bit-level arbitrary key settings. If you are interested, you can set other keys for testing (such as ports, etc.).
Dynamic Memory Allocation
Below we show FlyMon's dynamic memory allocation. First we issue two tasks with different memory sizes under the same CMU-Group.flymon> reset_all
flymon> add_task -f 10.0.0.0/8,* -k hdr.ipv4.src_addr -a frequency(1,cms) -m 48
flymon> add_task -f 20.0.0.0/8,* -k hdr.ipv4.src_addr -a frequency(1,cms) -m 24
These two tasks have the same key and attribute, but focus on different sets of traffic. The resource manager will assign them to the same CMU-Group.
🔔 The
reset_all
command is optional. It will clear all data plane tasks and controller plane status. The purpose of executing this command here is to make it easier for users to follow this manual. In other words, when this command is executed, all subsequent tasks will be allocated from TaskID=1, which is convenient for later manuals with the fixed task_id and CMU-Group allocations. If you do not execute this command. You need to correctly select the TaskID assigned to you by the task manager when you read/query the task.
The first task will be allocated 3x16 buckets, while the second task will be allocated 3x8 buckets. To verify this, we inject some packets into the switch.
🔔 You can inject traffic into the physical switch through an additional server. You can also inject traffic into the Tofino Model's virtual interface via software (e.g., scapy). Both require the FlyMon data plane's simple_fwd table to be configured in advance. We offer
add_port
andadd_forward
commands to fast configure you switch/model.
If you are using a Tofino Model, we provide some commands to help you perform the tests.
flymon> add_forward -s 0 -d 1
flymon> send_packets -l 64 -n 10 -p 0 -s 10.0.0.0/8
Send a packet with src_ip=10.0.0.1, pktlen=64.
Send a packet with src_ip=10.0.0.2, pktlen=64
...
flymon> send_packets -l 64 -n 10 -p 0 -s 20.0.0.0/8
Send a packet with src_ip=20.0.0.1, pktlen=64.
Send a packet with src_ip=20.0.0.2, pktlen=64.
...
After injecting the traffic of 10.0.0.0/8 and 20.0.0.0/8, we observe the memory status of whole CMU-Group, Task 1, and Task 2 respectively.
flymon> read_cmug -g 1
Read all data for CMU-Group 1
[0, 0, 1, 1, 0, 1, 1, 0, 1, 1, 0, 1, 0, 1, 1, 1, 1, 1, 2, 1, 2, 0, 1, 2, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 0, 0, 0, 0, 0, 1, 3, 0, 0, 0, 0, 0, 0, 2, 4, 0, 0, 3, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 1, 1, 0, 0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 0, 1, 1, 1, 0, 2, 2, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 0]
----------------------------------------------------
flymon> read_task -t 1
Read all data for task: 1
[0, 0, 1, 1, 0, 1, 1, 0, 1, 1, 0, 1, 0, 1, 1, 1]
[0, 0, 0, 0, 0, 0, 1, 3, 0, 0, 0, 0, 0, 0, 2, 4]
[0, 1, 1, 0, 0, 1, 1, 0, 1, 1, 1, 1, 0, 1, 0, 1]
flymon> read_task -t 2
Read all data for task: 2
[1, 1, 2, 1, 2, 0, 1, 2]
[0, 0, 3, 7, 0, 0, 0, 0]
[1, 1, 0, 2, 2, 1, 1, 2]
We can find that the data of both Task 1 and Task 2 are on CMU-Group 1. The memory range of task 1 is [0,16), while the memory range of task 2 is [16,24).
Next, we delete task 1 and its data.
flymon> del_task -t 1 -c True # The '-c' option will clear the data of the task when deleting.
flymon> read_cmug -g 1
Read all data for task: 1
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 2, 1, 2, 0, 1, 2, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 0, 2, 2, 1, 1, 2, 0, 0, 0, 0, 0, 0, 0, 0]
Finally, we find that all the data of task 1 is cleared. This use case demonstrates that FlyMon can dynamically distribute measurement tasks to different size and location memory ranges, and achieve isolation between tasks.
Single-key Distinct Counting (Flow Cardinality)
FlyMon uses the HyperLogLog algorithm to implement single-key distinct Count statistics. If we want to count the different number of SrcIP in the network, we can deploy this task with the following command.
flymon> reset_all
flymon> add_task -f *,* -k None -a distinct(hdr.ipv4.src_addr,hll) -m 32
🔔 The
reset_all
command is optional. It will clear all data plane tasks and controller plane status. The purpose of executing this command here is to make it easier for users to follow this manual. In other words, when this command is executed, all subsequent tasks will be allocated from TaskID=1, which is convenient for later manuals with the fixed task_id and CMU-Group allocations. If you do not execute this command. You need to correctly select the TaskID assigned to you by the task manager when you read/query the task.
We can inject some packets into the switch like this. For example, using scapy to generate 100 packets with different IP addresses.
🔔 Since the memory in our demo scenario is relatively small (i.e., only 32 16-bits counters in each CMU), we choose a very small size of traffic to be measured. HyperLogLog usually yields more reliable measurements (i.e., smaller variance) in larger traffic scenarios.
If you are using a Tofino Model, we provide some commands to help you perform the tests.
flymon> add_forward -s 0 -d 1
flymon> send_packets -l 64 -n 100 -p 0 -s 10.0.0.0/8
Send a packet with src_ip=10.0.0.1, pktlen=64.
Send a packet with src_ip=10.0.0.2, pktlen=64.
...
Send a packet with src_ip=10.0.0.98, pktlen=64.
Send a packet with src_ip=10.0.0.99, pktlen=64.
Send a packet with src_ip=10.0.0.100, pktlen=64.
After inserting the traffic, we can query the measurement data of the single-key distinct counting task.
flymon> read_task -t 1
Read all data for task: 1
[56233, 59082, 38840, 43739, 64347, 50744, 46922, 35369, 39501, 42798, 54876, 60223, 47807, 34780, 63150, 52173, 59120, 56211, 43745, 38786, 50690, 64353, 35347, 46960, 42772, 39543, 60165, 54886, 34790, 47749, 52215, 63124]
As you can see, the output of the original measurement data is not intuitive. We implement the HLL algorithm data parsing in the query_test
command.
flymon> query_task -t 1
173
The results seem not accurate enough because we used a minimal number of packets and a tiny amount of memory. HyperLogLog usually yields more reliable measurements (i.e., more minor variance) in more significant traffic scenarios. More importantly, adjusting the data plane hash function parameters is also a technical task. We will further optimize the configuration of the FlyMon data plane's hash functions in the future.
Capture Maximum Packet Size
Here, we show how to use FlyMon to measure the max attribute and how to set standard metadata as an attribute parameter. We do this by measuring the maximum packet size for each flow. We can deploy such a task with the following command.
flymon> reset_all
flymon> add_task -f 10.0.0.0/8,* -k hdr.ipv4.src_addr -a max(pkt_size,sumax) -m 48
🔔 The
reset_all
command is optional. It will clear all data plane tasks and controller plane status. The purpose of executing this command here is to make it easier for users to follow this manual. In other words, when this command is executed, all subsequent tasks will be allocated from TaskID=1, which is convenient for later manuals with the fixed task_id and CMU-Group allocations. If you do not execute this command. You need to correctly select the TaskID assigned to you by the task manager when you read/query the task.
Then we generate several packets of different sizes. If you are using a Tofino Model, we provide some commands to help you perform the tests.
flymon> add_forward -s 0 -d 1
flymon> send_packets -l 64 -n 10 -p 0 -s 10.0.0.0/8
...
flymon> send_packets -l 80 -n 10 -p 0 -s 10.0.0.0/8
...
Finally, we can get the following measurement results.
flymon> read_task -t 1
Read all data for task: 1
[0, 0, 80, 80, 0, 80, 80, 0, 80, 80, 0, 80, 0, 80, 80, 80]
[0, 0, 0, 0, 0, 0, 80, 80, 0, 0, 0, 0, 0, 0, 80, 80]
[0, 80, 80, 0, 0, 80, 80, 0, 80, 80, 80, 80, 0, 80, 0, 80]
flymon> query_task -t 1 -k 10.0.0.1,*,*,*,*
80
The underlying logic of the Max property is the SuMax algorithm, which obtains the closest estimate to the actual value by minimizing the value of multiple rows. In addition to packet size, FlyMon currently supports standard metadata, including queue length, timestamp, etc.
Flow Existence Check
Flow existence check is a measurement task to collect or inspect a set of flows present in the network. FlyMon uses a bloom filter to implement a large flow set's collection (bitwise-or) or inspection (bitwise-and). The flow inspection requires other functional modules depending on different scenarios, such as forwarding, dropping, sending to the control plane, etc. Here we use a bloom filter to demonstrate the collection of flows in the network.We can issue an existence collection task with the following command.
flymon> reset_all # optional
flymon> add_task -f *,* -k None -a existence(hdr.ipv4.src_addr,bloomfilter) -m 32
Then we inject some traffic. If you are using a Tofino Model, we provide some commands to help you perform the tests.
flymon> add_forward -s 0 -d 1
flymon> send_packets -l 64 -n 10 -p 0 -s 10.0.0.0/8
Then, we can query the measurement data of the data plane.
flymon> read_task -t 1
Read all data for task: 1
[0, 0, 0, 43739, 0, 0, 46922, 0, 39501, 0, 0, 0, 0, 34780, 0, 4078, 0, 0, 28354, 0, 0, 32768, 0, 0, 0, 24148, 0, 54886, 0, 0, 52215, 0]
As shown above, the measurement result of the bloom filter in FlyMon is not intuitive because we use all the 16 bits of each counter by encoding the param in the pre-processing stage. Despite this, it is easy to expand this bloom filter into its original form.
Multi-key Distinct Counting
Distinct counting for multiple keys means that measurement are performed for a set of different keys on a single CMU at the same time. FlyMon uses a variant of BeauCoup algorithm to perform this measurement task. To deploy the task in FlyMon, we can :
flymon> reset_all # optional
flymon> add_task -f 10.0.0.0/8,* -k hdr.ipv4.dst_addr -a distinct(hdr.ipv4.src_addr,beaucoup) -m 96
The above command deploys a measurement task that counts different DstIPs for each SrcIP (i.e., super spreader detection). The processing of flows that meet the threshold needs to be combined with other network components depending on user's scenarios (e.g., combined with a reporting module like SketchLib). Here we only demonstrate how the measurement is performed.
After the task is deployed, we inject traffic with different source and destination IP addresses into the network. If you are using a Tofino Model, we provide some commands to help you perform the tests.
flymon> add_forward -s 0 -d 1
flymon> send_packets -l 64 -n 20 -p 0 -s 10.0.0.0/8 -d 20.0.0.0/8
Send a packet with src_ip=10.0.0.1, dst_ip=20.0.0.1, pktlen=64.
Send a packet with src_ip=10.0.0.2, dst_ip=20.0.0.2, pktlen=64.
Send a packet with src_ip=10.0.0.3, dst_ip=20.0.0.3, pktlen=64.
Send a packet with src_ip=10.0.0.4, dst_ip=20.0.0.4, pktlen=64.
....
When the traffic generation is finished, we can read the memory of this task.
flymon> read_task -t 1
[8937, 8074, 43739, 0, 0, 0, 0, 46922, 0, 39501, 12060, 4735, 34780, 0, 4078, 0, 0, 59120, 0, 28354, 32768, 0, 29523, 20016, 24148, 0, 54886, 60165, 32422, 17349, 0, 52215]
[0, 0, 0, 26386, 0, 17996, 58504, 16867, 34868, 11615, 0, 10992, 44741, 2990, 43370, 0, 30653, 53974, 28690, 0, 20812, 32768, 0, 0, 0, 0, 15856, 39067, 7342, 0, 6913, 48746]
[0, 0, 0, 52209, 47894, 43831, 39764, 35701, 0, 0, 0, 0, 0, 0, 0, 0, 32768, 59811, 55744, 51681, 47366, 43303, 0, 35173, 30858, 26795, 22728, 18665, 14350, 10287, 6220, 2157]
There are two different ways to deploy for this task. One with a threshold-triggered approach similar to BeauCoup's. The other is a maximum likelihood estimation (MLE) approach similar to Linear Couting.
Other thoughts
The above examples cover most of the feature of FlyMon: the dynamic key, the dynamic attribute, and the dynamic memory.
There are some other features in FlyMon:
-
Cooperation of multiple CMU Groups. The upstream CMU can output its result to a downstream CMU to implement some multi-level algorithms or combinational tasks. The feature is not included in current implementation because it introduces additional design tricks, which are relatively difficult to understand the intents behind. Maybe I will try to provide the refactored code in an isolated branch.
-
Potential Extensions with Tofino2. Tofino2 introduces many new features, which can enable a more powerful FlyMon implementation. For example, SALUs can accommodate more "if" condition judgments, which can make the Reduced Operation Set aggregation step more efficient. In other words, we can use less stateful actions to enable the built-in sketches.
For the convenience of accuracy estimation, we implemented a simulated version of FlyMon in C++ to test the algorithms' accuracy. In this simulation, we implement various measurement algorithms based on CMU, including BeauCoup, SuMax, CM-Sketch, BloomFilter, MRAC, and HyperLogLog.
😔 However, the current simulations of FlyMon is messy, inefficient or even ugly. Maybe I will find a time to refactor it step by step.
The simulation codes are located in the simulations directory.
The project is released under the GNU Affero General Public License v3.
If you find the code useful, please consider citing our paper. Below are some links that may also be helpful to you: