In today's exercise we will implement a new probabilistic data structure. The Count-Min Sketch. As you have seen during the lecture the Count-min sketch is a probabilistic algorithm that can be used to estimate occurrences of distinct elements (similar to the counting bloom filter we saw in last week lecture and previous exercise).
To implement the count-min sketch we will use registers
and hash
functions. This time though, we will
have to define several registers and use a different hash function for each of them. To count packets belonging to a flow using the count-min
sketch we have to hash every packet's 5-tuple (src,dst, sport, dport, proto) using N different hash functions. Each hash output is used to index a different register
and increase its value by 1 (as you can see in the picture below). In order to read how many entries were counted for a given flow, we have to hash the 5-tuple, read
the value at every register and pick the minimum.
In this exercise we will implement a count-min sketch with P4, a controller that will pull the counters from the switch and decode them, and finally play with the dimensions of the sketch.
To test the performance of our sketch we will need to send hundreds of thousands of packets through the switch. As you have already seen in the
previous exercises, if you do an iperf
between to directly connected hosts you get roughly a bandwidth of 15~20mbps
. In order to be able to send
packets fast to the switch we can clone the repository again with a different name and compile it with different flags. Since this process can take up to
10 minutes you can just leave it running in the background.
IMPORTANT: It is recommended that you do not run the sudo make install
command until you have a working solution. When using this optimized compilation the
switch will not generate log files, and thus it will be hard for you to properly debug your program.
cd ~/p4-tools/
git clone https://github.com/p4lang/behavioral-model.git bmv2-opt
cd bmv2-opt
git checkout 62a013a15ed2c42b1063c26331d73c2560d1e4d0
./autogen.sh
./configure --without-nanomsg --disable-elogger --disable-logging-macros 'CFLAGS=-g -O2' 'CXXFLAGS=-g -O2'
make -j 2
sudo make install
sudo ldconfig
Since we keep the two compiled versions of bmv2
in different folders, you can enable the one with the debugging
enabled by just running the make install
command
again:
cd ~/p4-tools/bmv2
sudo make install
sudo ldconfig
Thus by running sudo make install
in ~/p4-tools/bmv2
or ~/p4-tools/bmv2-opt
you can easily enable each compiled version.
For this exercise we provide you with the following files:
p4app.json
: describes the topology we want to create with the help of mininet and p4-utils package.network.py
: a Python scripts that initializes the topology using Mininet and P4-Utils. One can use indifferentlynetwork.py
orp4app.json
to start the network.p4src/cm-sketch.p4
: p4 program skeleton to use as a starting point. Furthermore, we provide you theheaders.p4
andparsers.p4
file such that you do not have to write this again.send.py
: python program that uses raw sockets to send packets (scapy
was not fast enough for what we needed in this exercise).cm-sketch-controller.py
: controller that uses thethrift_API
to communicate with the switch. You will find some util functions that will help with the sketch decoding. Here you will also have to implement part of the algorithm.crc.py
a pythoncrc32
implementation that will be used by the controller to read the counters.
For this exercise we will use the mixed
strategy, however this is not very important since we will just use the switch to send packets
through it and run the count-min sketch. In this small topology, where h1
and h2
are connected to the same switch s1
the ip assignment is 10.0.1.1
and 10.0.1.2
respectively.
You can find all the documentation about p4app.json
in the p4-utils
documentation. Also, you can find information about assignment strategies here.
To solve this exercise we have to do two things: i) implement the count-min sketch in P4, ii) decode the values using a the controller and compare them with the ground truth.
The program running in the switch needs to be able to do two things:
- Forward packets from port 1 to 2 and vice-versa.
- Execute the Sketch for every
tcp
packet observed.
To successfully complete the p4 program you have to:
-
Define a forwarding table that forwards packets from port 1 to port 2 and vice-versa. This is exactly what we did in the
repeater
exercise, thus you can copy your solution from there. The table should match theingress_port
and in case of match call aset_egress_port
action. -
Define the
set_egress_port
action. The action gets anbit<9> egress_port
number and sets it asegress_spec
. -
Define the
s1-commands.txt
file to fill the table. You just need two entries. -
Define
N
registers. To set the width of each register cell use the constantSKETCH_CELL_BIT_WIDTH
. To set the register size use theSKETCH_BUCKET_LENGTH
constant (which you have to define depending on your needs). In order to easily read the registers from the controller name them as follows:sketch<num>
for example:sketch0
,sketch1
,..,sketchN
. -
Define the
sketch_count
action. In this action you should do the following for each register:- Define N metadata field of width 32 in which to store the output of each hash computation.
- Define N metadata fields of width
SKETCH_CELL_BIT_WIDTH
to store the counter read from each register. - Hash the packets 5-tuple in this order: srcAddr, dstAddr, srcPort, dstPort, protocol. Send the output to the metadata field you just
defined. As a max (or module) value for the hash use the constant
SKETCH_BUCKET_LENGTH
. You must castSKETCH_BUCKET_LENGTH
to 32 bits while using it as the max argument of thehash
function, as the argument accepts fixed-width parameters. Important: use theHashAlgorithm.crc32_custom
: since we want to potentially define N hash functions and only few algorithms are available, we will use thecrc32_custom
algorithm that the simple switch provides. The polynomial of this algorithm can be modified at runtime by the controller, in our case (this is provided) the controller will automatically configure all the hash functions with different polynomials such that they return different hash outputs. You can read more about thecrc32
configuration here. - For each hash output value (index to register) read the count value of its respective register and save it into the variables defined in 2. Add one to
the variable and write that to the register. Make sure that you use the output of your
hash0
withsketch0
.
-
Define the ingress block logic. Apply the
forwarding
table. Furthermore, if the packet isipv4
andtcp
apply thesketch_count
action.
In order to read the count-min sketch counter you will have to implement a small algorithm at the controller. We provide you with a partial controller implementation that already does several things for you. Amongst others, the controller does:
-
Configures the
crc32_custom
hash algorithms that you defined in the switch (at the moment it can do up to 16 hashes). This is done by the functionset_crc_custom_hashes
. This function uses theruntimeAPI
(which uses thethrift
API) to set the polynomials of eachcrc
. You can see the list of polynomials at the beginning of the file. -
It will create some
crc32
hash objects and append them toself.hashes
. Note that this hashes (implemented in python) are the same than the ones you will be running in the switch. Thusself.hashes[0]
== the first hash you defined in your P4 program. To get the output of the hash you have to call the methodbit_by_bit_fast(5-tuple)
.
To successfully implement the controller you have to:
-
Define a function to read the N registers and save them in order in
self.registers
list. To read a register use the controller API:self.controller.register_read(<name>)
, this will return a list with all the register fields. The amount of registers available in the switch (which should be the same than hashes) can be found inself.register_num
. Remember that if you named your registers with the following patternsketch<num>
you can easily read them with a simple iteration. -
Define a function to get the count of a given flow. As an input the function needs the flow's 5-tuple and the length of the registers (to use it to modulo the output of the hash). To successfully implement this function you should hash the flow 5-tuple with all the different hash functions
self.hashes[i]
(remember to apply the modulo to the output to get the right index). For each hashi
read theself.registers[i][hash_output[i]]
and return the minimum of them. The hash function takes a bytestream as input, to generate the bytestream out of the 5-tuple use the functionflow_to_bytestream(flow)
, the input parameter has to be a list or tuple of the form ("10.0.1.1", "10.0.1.2", 5000, 7000). Note that the protocol is not needed and will be automatically added. -
Define the main (call it
decode_registers
) function to decode all the registers and compare them with the ground truth. This function has as input: epsilon, number of packets used in the experiment, registers length, and a file with the flows sent during the experiment (to know more about this check out the testing the solution section). In more details you have to:- Read the registers (call the function you defined in 1).
- Load the
sent_flows.pickle
file. This file contains a python dictionary with the ground truth. The dictionary is of the form{(flow): packet_count}
. To load a pickle file you need to use:pickle.load(open(file_name, "r"))
- For each flow read the count value out of the registers using the function defined in 2.
- For each flow you can check if you are inside the bound: counter <= (
real_counter
+ ϵ*η). - You can also determine if 4 holds with probability 1 - δ.
- You are free to compute any statistics out of your computations, such as total error, relative errors to single flows, etc.
Note: that all the variables needed to solve the above items can be passed through command line parameters. See the cm-sketch-controller.py
file at the bottom.
In today's lecture you have learned that a count-min sketch holds the following properties:
- The estimated count value will be the same as the real or bigger.
- The error will be not bigger than
real_count
+ ϵ*η with probability 1 - δ. Where ϵ is the error factor, η is the number of packets hashed and δ is the error probability.
After some analysis we can get some formulas for determining the dimensions of the array and the number of them as:
Number of registers:
ceil(ln(1/δ))Register length:
ceil(e/ϵ)
For example, if we want a ϵ=0.1 and δ=0.05. Registers = 3, length=28.
To test our solution we will use the send.py
script you can find in the provided files. This script uses raw sockets
to generate random flows. The number of packets, flow size, and flow distribution can be configured by tuning the command line
parameters:
parser.add_argument('--n-pkt', help='number of packets', type=int, required=False, default=5000)
parser.add_argument('--n-hh', help='number of heavy hitters', type=int, required=False, default=10)
parser.add_argument('--n-sfw', help='number of small flows', type=int, required=False, default=990)
parser.add_argument('--p-hh', help='percentage of packets sent by heavy hitters',type=float, required=False, default=0.95)
On top of sending the random flows, the script will save in a file a dictionary with {(flows): packets_sent}
. This dictionary
can be used as a ground truth by the controller to compare the number of packets sent and the number counted by the count-min sketch.
We will also use the controller we provided you and that you had to modify and enhance. The controller can be called using the following parameters:
parser.add_argument('--sw', help="switch name to configure" , type=str, required=False, default="s1")
parser.add_argument('--eps', help="epsilon to use when checking bound", type=float, required=False, default=0.01)
parser.add_argument('--n', help="number of packets sent by the send.py app", type=int, required=False, default=1000)
parser.add_argument('--mod', help="number of cells in each register", type=int, required=False, default=4096)
parser.add_argument('--flow-file', help="name of the file generated by send.py", type=str, required=False, default="sent_flows.pickle")
parser.add_argument('--option', help="controller option can be either set_hashes, decode or reset registers", type=str, required=False, default="set_hashes")
Dimension your sketch using ϵ=0.1 and δ=0.05 (as shown above). Then start the topology, configure the hash functions
with the controller and send traffic from h1
using send.py to send few flows (just to test).
-
Start the topology with
sudo p4run
orsudo python network.py
. -
Set the hash functions polynomials by running the default controller with
set_hashes
as option parameter. This will set the right polynomials to the running switch so when we decode we are use the same exact hash.
python cm-sketch-controller.py --option "set_hashes"
- Get a terminal in
h1
and send a bit of traffic (only 2 flows) to test if your decoder works.
mx h1
python send.py --n-pkt 1000 --n-hh 2 --n-sfw 0 --p-hh 1
This script will create two flows of size approximately 500 packets. It will also save in sent_flows.pickle
a dictionary with the exact number of packets sent by each flow.
The controller decoding function should read this file and try to extract the counters for those flows and check if the count is inside the boundary.
- Finally, run the controller with the
decode
option enabled and the right parameters for the decoding.
python cm-sketch-controller.py --eps 0.1 --n 1000 --mod 28 --option decode
Before starting more tests make sure you install the optimized version of bmv2
. Furthermore, open the send.py
file
go to the send_n
function and replace time.sleep(0.001)
by time.sleep(0.0002)
. With the optimized version we can sent packets
5 times faster.
Now instead of just sending 2 flows, we will mix big and small flows. Run send.py
with the following parameters:
mx h1
python send.py --n-pkt 100000 --n-hh 10 --n-sfw 990 --p-hh 0.95
You should see that big flows have a good size estimation, however small flows will all be far from the real value. This due to the size of the register (28 cells). Small flows send between 0 and 10 packets, however the expected value of each cell is around 150.
If you increase the cell size to something much bigger, for example 4096, and repeat the test (make sure you reboot the switch and the controller). You will see that the estimation count and real values will be very close (if not exactly the same).
At this point you should be able to reason about how to dimension your count-min sketch depending on what your traffic mix looks like, and what do you want to achieve. Modify the number of registers and size and play with different traffic distributions to check if the theory you learned during the lecture holds.
We have added a small guideline in the documentation section. Use it as a reference when things do not work as expected.