The objective of this exercise is to write a P4 program that implements basic forwarding. To keep things simple, we will just implement forwarding for IPv4.
With IPv4 forwarding, the switch must perform the following actions for every packet: (i) update the source and destination MAC addresses, (ii) decrement the time-to-live (TTL) in the IP header, and (iii) forward the packet out the appropriate port.
Your switch will have a single table, which the control plane will populate with static rules. Each rule will map an IP address to the MAC address and output port for the next hop. We have already defined the control plane rules, so you only need to implement the data plane logic of your P4 program.
We will use the following topology for this exercise. It is a single pod of a fat-tree topology and henceforth referred to as pod-topo:
Our P4 program will be written for the V1Model architecture implemented on P4.org's bmv2 software switch. The architecture file for the V1Model can be found at: /usr/local/share/p4c/p4include/v1model.p4. This file desribes the interfaces of the P4 programmable elements in the architecture, the supported externs, as well as the architecture's standard metadata fields. We encourage you to take a look at it.
Spoiler alert: There is a reference solution in the
solution
sub-directory. Feel free to compare your implementation to the reference.
To ensure a smooth experience with the tutorials, it's essential to review and follow the Obtaining required software guidelines to install required development tools.
The directory with this README also contains a skeleton P4 program,
basic.p4
, which initially drops all packets. Your job will be to
extend this skeleton program to properly forward IPv4 packets.
Before that, let's compile the incomplete basic.p4
and bring
up a switch in Mininet to test its behavior.
-
In your shell, run:
make run
This will:
- compile
basic.p4
, and - start the pod-topo in Mininet and configure all switches with the appropriate P4 program + table entries, and
- configure all hosts with the commands listed in pod-topo/topology.json
- compile
-
You should now see a Mininet command prompt. Try to ping between hosts in the topology:
mininet> h1 ping h2 mininet> pingall
-
Type
exit
to leave each xterm and the Mininet command line. Then, to stop mininet:make stop
And to delete all pcaps, build files, and logs:
make clean
The ping failed because each switch is programmed
according to basic.p4
, which drops all packets on arrival.
Your job is to extend this file so it forwards packets.
A P4 program defines a packet-processing pipeline, but the rules within each table are inserted by the control plane. When a rule matches a packet, its action is invoked with parameters supplied by the control plane as part of the rule.
In this exercise, we have already implemented the control plane
logic for you. As part of bringing up the Mininet instance, the
make run
command will install packet-processing rules in the tables of
each switch. These are defined in the sX-runtime.json
files, where
X
corresponds to the switch number.
Important: We use P4Runtime to install the control plane rules. The
content of files sX-runtime.json
refer to specific names of tables, keys, and
actions, as defined in the P4Info file produced by the compiler (look for the
file build/basic.p4.p4info.txtpb
after executing make run
). Any changes in the P4
program that add or rename tables, keys, or actions will need to be reflected in
these sX-runtime.json
files.
The basic.p4
file contains a skeleton P4 program with key pieces of
logic replaced by TODO
comments. Your implementation should follow
the structure given in this file---replace each TODO
with logic
implementing the missing piece.
A complete basic.p4
will contain the following components:
- Header type definitions for Ethernet (
ethernet_t
) and IPv4 (ipv4_t
). - TODO: Parsers for Ethernet and IPv4 that populate
ethernet_t
andipv4_t
fields. - An action to drop a packet, using
mark_to_drop()
. - TODO: An action (called
ipv4_forward
) that:- Sets the egress port for the next hop.
- Updates the ethernet destination address with the address of the next hop.
- Updates the ethernet source address with the address of the switch.
- Decrements the TTL.
- TODO: A control that:
- Defines a table that will read an IPv4 destination address, and
invoke either
drop
oripv4_forward
. - An
apply
block that applies the table.
- Defines a table that will read an IPv4 destination address, and
invoke either
- TODO: A deparser that selects the order in which fields inserted into the outgoing packet.
- A
package
instantiation supplied with the parser, control, and deparser.In general, a package also requires instances of checksum verification and recomputation controls. These are not necessary for this tutorial and are replaced with instantiations of empty controls.
Follow the instructions from Step 1. This time, you should be able to sucessfully ping between any two hosts in the topology.
The "test suite" for your solution---sending pings between hosts in the topology---is not very robust. What else should you test to be confident that you implementation is correct?
Although the Python
scapy
library is outside the scope of this tutorial, it can be used to generate packets for testing. Thesend.py
file shows how to use it.
Other questions to consider:
- How would you enhance your program to respond to ARP requests?
- How would you enhance your program to support traceroute?
- How would you enhance your program to support next hops?
- Is this program enough to replace a router? What's missing?
There are several problems that might manifest as you develop your program:
-
basic.p4
might fail to compile. In this case,make run
will report the error emitted from the compiler and halt. -
basic.p4
might compile but fail to support the control plane rules in thes1-runtime.json
throughs3-runtime.json
files thatmake run
tries to install using P4Runtime. In this case,make run
will report errors if control plane rules cannot be installed. Use these error messages to fix yourbasic.p4
implementation. -
basic.p4
might compile, and the control plane rules might be installed, but the switch might not process packets in the desired way. Thelogs/sX.log
files contain detailed logs that describing how each switch processes each packet. The output is detailed and can help pinpoint logic errors in your implementation.
In the latter two cases above, make run
may leave a Mininet instance
running in the background. Use the following command to clean up
these instances:
make stop
- Gateway (gw) Command: The
route add default gw
command is used to set the default gateway for a host. This tells the host which IP address to send packets to if the destination IP is not on the same subnet. It's important for hosts to know their default gateway to communicate with devices outside their local network. - ARP Command: The
arp -i eth0 -s
command is used to add static ARP entries to the host's ARP cache. When you add static ARP entries, you're telling your computer something like i.e. "Hey, I already know who lives at this IP address (like 10.0.0.1), so you don't need to keep asking everyone on the network 'Who has this IP?' all the time." This is particularly important in this exercise (and most of the other exercises) since the switches do not respond to ARP requests. In real networks, production switches typically do respond to ARP requests, but in these exercises, static ARP entries are necessary for the gateway router due to this behavior.-i eth0
: This specifies the network interface (e.g. eth0) on which you want to perform the ARP operation.-s
: This flag is used to set a static ARP entry.
Note: If you remove the gateway and ARP commands, the hosts in your network might lose connectivity to each other and to devices outside their local subnet. This can result in 100% packet loss when running the pingall
command because the hosts don't have the necessary routing information and ARP entries to reach their destinations.
Congratulations, your implementation works! Move onto the next assignment Basic Tunneling
The documentation for P4_16 and P4Runtime is available here
All excercises in this repository use the v1model architecture, the documentation for which is available at: