* Repository reorganization for 2018 Spring P4 Developer Day. * Port tutorial exercises to P4Runtime with static controller (#156) * Switch VM to a minimal Ubuntu 16.04 desktop image * Add commands to install Protobuf Python bindings to user_bootstrap.sh * Implement P4Runtime static controller for use in exercises From the exercise perspective, the main difference is that control plane rules are now specified using JSON files instead of CLI commands. Such JSON files define rules that use the same name for tables, keys, etc. as in the P4Info file. All P4Runtime requests generated as part of the make run process are logged in the exercise's “logs” directory, making it easier for students to see the actual P4Runtime messages sent to the switch. Only the "basic" exercise has been ported to use P4Runtime. The "p4runtime" exercise has been updated to work with P4Runtime protocol changes. Known issues: - make run hangs in case of errors when running the P4Runtime controller (probably due to gRPC stream channel threads not terminated properly) - missing support for inserting table entries with default action (can specify in P4 program as a workaround) * Force install protobuf python module * Fixing Ctrl-C hang by shutdown switches * Moving gRPC error print to function for readability Unforuntately, if this gets moved out of the file, the process hangs. We'll need to figure out how why later. * Renaming ShutdownAllSwitches -> ShutdownAllSwitchConnections * Reverting counter index change * Porting the ECN exercise to use P4 Runtime Static Controller * updating the README in the ecn exercise to reflect the change in rule files * Allow set table default action in P4Runtime static controller * Fixed undefined match string when printing P4Runtime table entry * Updated basic_tunnel exercise to use P4Runtime controller. * Changed default action in the basic exercise's ipv4_lpm table to drop * Porting the MRI exercise to use P4runtime with static controller * Updating readme to reflect the change of controller for mri * Update calc exercise for P4Runtime static controller * Port source_routing to P4 Runtime static controller (#157) * Port Load Balance to P4 Runtime Static Controller (#158)
282 lines
7.4 KiB
Plaintext
282 lines
7.4 KiB
Plaintext
/* -*- P4_16 -*- */
|
|
#include <core.p4>
|
|
#include <v1model.p4>
|
|
|
|
const bit<8> UDP_PROTOCOL = 0x11;
|
|
const bit<16> TYPE_IPV4 = 0x800;
|
|
const bit<5> IPV4_OPTION_MRI = 31;
|
|
|
|
#define MAX_HOPS 9
|
|
|
|
/*************************************************************************
|
|
*********************** H E A D E R S ***********************************
|
|
*************************************************************************/
|
|
|
|
typedef bit<9> egressSpec_t;
|
|
typedef bit<48> macAddr_t;
|
|
typedef bit<32> ip4Addr_t;
|
|
typedef bit<32> switchID_t;
|
|
typedef bit<32> qdepth_t;
|
|
|
|
header ethernet_t {
|
|
macAddr_t dstAddr;
|
|
macAddr_t srcAddr;
|
|
bit<16> etherType;
|
|
}
|
|
|
|
header ipv4_t {
|
|
bit<4> version;
|
|
bit<4> ihl;
|
|
bit<8> diffserv;
|
|
bit<16> totalLen;
|
|
bit<16> identification;
|
|
bit<3> flags;
|
|
bit<13> fragOffset;
|
|
bit<8> ttl;
|
|
bit<8> protocol;
|
|
bit<16> hdrChecksum;
|
|
ip4Addr_t srcAddr;
|
|
ip4Addr_t dstAddr;
|
|
}
|
|
|
|
header ipv4_option_t {
|
|
bit<1> copyFlag;
|
|
bit<2> optClass;
|
|
bit<5> option;
|
|
bit<8> optionLength;
|
|
}
|
|
|
|
header mri_t {
|
|
bit<16> count;
|
|
}
|
|
|
|
header switch_t {
|
|
switchID_t swid;
|
|
qdepth_t qdepth;
|
|
}
|
|
|
|
struct ingress_metadata_t {
|
|
bit<16> count;
|
|
}
|
|
|
|
struct parser_metadata_t {
|
|
bit<16> remaining;
|
|
}
|
|
|
|
struct metadata {
|
|
ingress_metadata_t ingress_metadata;
|
|
parser_metadata_t parser_metadata;
|
|
}
|
|
|
|
struct headers {
|
|
ethernet_t ethernet;
|
|
ipv4_t ipv4;
|
|
ipv4_option_t ipv4_option;
|
|
mri_t mri;
|
|
switch_t[MAX_HOPS] swtraces;
|
|
}
|
|
|
|
error { IPHeaderTooShort }
|
|
|
|
/*************************************************************************
|
|
*********************** P A R S E R ***********************************
|
|
*************************************************************************/
|
|
|
|
parser MyParser(packet_in packet,
|
|
out headers hdr,
|
|
inout metadata meta,
|
|
inout standard_metadata_t standard_metadata) {
|
|
|
|
state start {
|
|
transition parse_ethernet;
|
|
}
|
|
|
|
state parse_ethernet {
|
|
packet.extract(hdr.ethernet);
|
|
transition select(hdr.ethernet.etherType) {
|
|
TYPE_IPV4: parse_ipv4;
|
|
default: accept;
|
|
}
|
|
}
|
|
|
|
state parse_ipv4 {
|
|
packet.extract(hdr.ipv4);
|
|
verify(hdr.ipv4.ihl >= 5, error.IPHeaderTooShort);
|
|
transition select(hdr.ipv4.ihl) {
|
|
5 : accept;
|
|
default : parse_ipv4_option;
|
|
}
|
|
}
|
|
|
|
state parse_ipv4_option {
|
|
/*
|
|
* TODO: Add logic to:
|
|
* - Extract the ipv4_option header.
|
|
* - If value is equal to IPV4_OPTION_MRI, transition to parse_mri.
|
|
* - Otherwise, accept.
|
|
*/
|
|
transition accept;
|
|
}
|
|
|
|
state parse_mri {
|
|
/*
|
|
* TODO: Add logic to:
|
|
* - Extract hdr.mri.
|
|
* - Set meta.parser_metadata.remaining to hdr.mri.count
|
|
* - Select on the value of meta.parser_metadata.remaining
|
|
* - If the value is equal to 0, accept.
|
|
* - Otherwise, transition to parse_swtrace.
|
|
*/
|
|
transition accept;
|
|
}
|
|
|
|
state parse_swtrace {
|
|
/*
|
|
* TODO: Add logic to:
|
|
* - Extract hdr.swtraces.next.
|
|
* - Decrement meta.parser_metadata.remaining by 1
|
|
* - Select on the value of meta.parser_metadata.remaining
|
|
* - If the value is equal to 0, accept.
|
|
* - Otherwise, transition to parse_swtrace.
|
|
*/
|
|
transition accept;
|
|
}
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
************ C H E C K S U M V E R I F I C A T I O N *************
|
|
*************************************************************************/
|
|
|
|
control MyVerifyChecksum(inout headers hdr, inout metadata meta) {
|
|
apply { }
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
************** I N G R E S S P R O C E S S I N G *******************
|
|
*************************************************************************/
|
|
|
|
control MyIngress(inout headers hdr,
|
|
inout metadata meta,
|
|
inout standard_metadata_t standard_metadata) {
|
|
action drop() {
|
|
mark_to_drop();
|
|
}
|
|
|
|
action ipv4_forward(macAddr_t dstAddr, egressSpec_t port) {
|
|
standard_metadata.egress_spec = port;
|
|
hdr.ethernet.srcAddr = hdr.ethernet.dstAddr;
|
|
hdr.ethernet.dstAddr = dstAddr;
|
|
hdr.ipv4.ttl = hdr.ipv4.ttl - 1;
|
|
}
|
|
|
|
table ipv4_lpm {
|
|
key = {
|
|
hdr.ipv4.dstAddr: lpm;
|
|
}
|
|
actions = {
|
|
ipv4_forward;
|
|
drop;
|
|
NoAction;
|
|
}
|
|
size = 1024;
|
|
default_action = NoAction();
|
|
}
|
|
|
|
apply {
|
|
if (hdr.ipv4.isValid()) {
|
|
ipv4_lpm.apply();
|
|
}
|
|
}
|
|
}
|
|
|
|
/*************************************************************************
|
|
**************** E G R E S S P R O C E S S I N G *******************
|
|
*************************************************************************/
|
|
|
|
control MyEgress(inout headers hdr,
|
|
inout metadata meta,
|
|
inout standard_metadata_t standard_metadata) {
|
|
action add_swtrace(switchID_t swid) {
|
|
/*
|
|
* TODO: add logic to:
|
|
- Increment hdr.mri.count by 1
|
|
- Add a new swtrace header by calling push_front(1) on hdr.swtraces.
|
|
- Set hdr.swtraces[0].swid to the id parameter
|
|
- Set hdr.swtraces[0].qdepth to (qdepth_t)standard_metadata.deq_qdepth
|
|
- Increment hdr.ipv4.ihl by 2
|
|
- Increment hdr.ipv4.totalLen by 8
|
|
- Increment hdr.ipv4_option.optionLength by 8
|
|
*/
|
|
}
|
|
|
|
table swtrace {
|
|
actions = {
|
|
add_swtrace;
|
|
NoAction;
|
|
}
|
|
|
|
default_action = NoAction();
|
|
}
|
|
|
|
apply {
|
|
/*
|
|
* TODO: add logic to:
|
|
* - If hdr.mri is valid:
|
|
* - Apply table swtrace
|
|
*/
|
|
swtrace.apply();
|
|
}
|
|
}
|
|
|
|
/*************************************************************************
|
|
************* C H E C K S U M C O M P U T A T I O N **************
|
|
*************************************************************************/
|
|
|
|
control MyComputeChecksum(inout headers hdr, inout metadata meta) {
|
|
apply {
|
|
update_checksum(
|
|
hdr.ipv4.isValid(),
|
|
{ hdr.ipv4.version,
|
|
hdr.ipv4.ihl,
|
|
hdr.ipv4.diffserv,
|
|
hdr.ipv4.totalLen,
|
|
hdr.ipv4.identification,
|
|
hdr.ipv4.flags,
|
|
hdr.ipv4.fragOffset,
|
|
hdr.ipv4.ttl,
|
|
hdr.ipv4.protocol,
|
|
hdr.ipv4.srcAddr,
|
|
hdr.ipv4.dstAddr },
|
|
hdr.ipv4.hdrChecksum,
|
|
HashAlgorithm.csum16);
|
|
}
|
|
}
|
|
|
|
/*************************************************************************
|
|
*********************** D E P A R S E R *******************************
|
|
*************************************************************************/
|
|
|
|
control MyDeparser(packet_out packet, in headers hdr) {
|
|
apply {
|
|
packet.emit(hdr.ethernet);
|
|
packet.emit(hdr.ipv4);
|
|
|
|
/* TODO: emit ipv4_option, mri and swtraces headers */
|
|
}
|
|
}
|
|
|
|
/*************************************************************************
|
|
*********************** S W I T C H *******************************
|
|
*************************************************************************/
|
|
|
|
V1Switch(
|
|
MyParser(),
|
|
MyVerifyChecksum(),
|
|
MyIngress(),
|
|
MyEgress(),
|
|
MyComputeChecksum(),
|
|
MyDeparser()
|
|
) main;
|