# P4 IN NIX ### Gauvain Roussel-Tarbouriech "Programming Protocol-independent Packet Processors (P4) is a domain-specific language for network devices, specifying how data plane devices (switches, NICs, routers, filters, etc.) process packets. "Programming Protocol-independent Packet Processors (P4) is a domain-specific language for network devices, specifying how data plane devices (switches, NICs, routers, filters, etc.) process packets. Great, what does that mean? "Programming Protocol-independent Packet Processors (P4) is a domain-specific language for network devices, specifying how data plane devices (switches, NICs, routers, filters, etc.) process packets. Great, what does that mean? It's a language for hardware optimized network processing (think SIMD for network) # **WHAT IS P4???** # WHAT IS P4??? It roughly looks like C: ### It roughly looks like C: ## It roughly looks like C: ...With a few oddities:) Functions are replaced by parser, control, package. Functions are replaced by parser, control, package. • parser: Parses an incoming packet according to structs, typedefs, etc... ### Functions are replaced by parser, control, package. - parser: Parses an incoming packet according to structs, typedefs, etc... - control: Modify a parsed packet in order to resent ### Functions are replaced by parser, control, package. - parser: Parses an incoming packet according to structs, typedefs, etc... - **control**: Modify a parsed packet in order to resent - package: Defines the binding logic between the hardware and P4 (e.g. control and data plane) ### Functions are replaced by parser, control, package. - parser: Parses an incoming packet according to structs, typedefs, etc... - **control**: Modify a parsed packet in order to resent - package: Defines the binding logic between the hardware and P4 (e.g. control and data plane) Other interesting keywords such as state or tables exist but are out of scope for this talk. ### Functions are replaced by parser, control, package. - parser: Parses an incoming packet according to structs, typedefs, etc... - control: Modify a parsed packet in order to resent - package: Defines the binding logic between the hardware and P4 (e.g. control and data plane) Other interesting keywords such as state or tables exist but are out of scope for this talk. ## LET'S MAKE A TRANSPILER! ## WHAT IS A TRANSPILER? ## WHAT IS A TRANSPILER? • Nix -> P4 translator ## WHAT IS A TRANSPILER? - Nix -> P4 translator - P4 Compiler ## WHAT IS A TRANSPILER? - Nix -> P4 translator - P4 Compiler - Target compiler ## WHAT IS A TRANSPILER? - Nix -> P4 translator - P4 Compiler - Target compiler What does it look like? ``` 1 source = { include = [ "core.p4" "v1model.p4" ]; define = { "test" = "test2"; }; headers = { const = { "MAX_HOPS" = { type = "int"; value = "10"; }; "STANDARD" = { type = "int"; value = "0"; }; "HOPS" = { type = "int"; value = "1"; }; 10 }; 11 12 header = { "type_t".content = [ { "tag" = "bit<8>"; } ]; 13 "hop t".content = [ 14 { "port" = "bit<8>"; } 15 { "bos" = "bit<8>"; } 16 17 "standard_t".content = [ 18 { "src" = "bit<8>"; } 19 { "dst" = "bit<8>"; } 20 21 }; 22 [\ldots] 23 }; 24 in 25 p4Platform.mkProgram { name = "test"; 26 src = (p4Platform.runTranspiler 27 28 { p4Source = source; }); 29 ``` ## WHICH WE CAN SIMPLIFY! ``` 1 source = { include = [ "core.p4" "v1model.p4" ]; define = { "test" = "test2"; }; headers = { header = { inherit ethernet_h ipv4_no_options_h; }; typedef = { inherit macAddr ip4Addr; }; }; 10 [\ldots] 11 }; 12 in 13 p4Platform.mkProgram { name = "test"; 14 src = (p4Platform.runTranspiler 15 { p4Source = source; }); 16 17 ``` ## THANKS TO HELPERS! ## THANKS TO HELPERS! What does the end result looks like? ## WHICH IS PARSED BY THIS ``` 1 # transpiler: 2 mkHeader = header: concatStringsSep "\n\n" (mapAttrsToList (name: value: (if (value.union) then "header_union " else "header ") + name + " {\n " + (concatStringsSep "\n" (flatten (imap1 (_: v: (mapAttrsToList (name: value: " + value + " " + name + ";") v) ) value.content))) + "\n}" ) header); 10 [...] 11 # module: 12 header = mkOption { description = '' 13 14 The list of headers of the program. 15 16 default = { }; 17 type = types.attrsOf (types.submodule { options = { 18 union = mkOption { 19 20 type = types.bool; 21 default = false; 22 content = mkOption { 23 type = types.listOf (types.attrsOf types.str); 24 25 default = [ ]; }; 26 27 ``` ## WHICH IS PARSED BY THIS ``` 1 # transpiler: 2 mkHeader = header: concatStringsSep "\n\n" (mapAttrsToList (name: value: (if (value.union) then "header_union " else "header ") + name + " {\n " + (concatStringsSep "\n" (flatten (imap1 (_: v: (mapAttrsToList (name: value: " " + value + " " + name + ";") v) ) value.content))) + "\n}" ) header); 10 [...] 11 # module: 12 header = mkOption { description = '' 13 14 The list of headers of the program. 15 default = { }; 16 type = types.attrsOf (types.submodule { 17 options = { 18 union = mkOption { 19 20 type = types.bool; 21 default = false; 22 23 content = mkOption { type = types.listOf (types.attrsOf types.str); 24 25 default = [ ]; }; 26 27 ٦١. ``` What does the end result looks like? ## P4 CODE: ``` 1 /* This file has been auto-generated by Nix, do not edit it manually! */ 2 #include <core.p4> 3 #include <v1model.p4> 5 #define test test2 7 const int HOPS = 1; 8 const int MAX HOPS = 10; 9 const int STANDARD = 0; 10 11 typedef standard_metadata_t std_meta_t; 12 13 header standard_t { bit<8> src; 14 bit<8> dst; 15 16 } 17 18 19 struct headers_t { 20 type_t type; 21 hop_t[MAX_HOPS] hops; 22 standard_t standard; 23 } 24 25 parser MyParser(packet_in pkt, out headers_t hdr, inout meta_t meta, inout state start { 26 27 [\ldots] ``` #### The end result looks like this on BMV2: ``` 2 #include <core.p4> 3 #include <v1model.p4> 5 #define test test2 7 const int HOPS = 1; 8 const int MAX_HOPS = 10; 9 const int STANDARD = 0; 10 11 typedef standard_metadata_t std_meta_t; 12 13 header standard_t { bit<8> src; 14 bit<8> dst; 15 16 } 17 18 ``` #### The end result looks like this on BMV2: ``` 2 #include <core.p4> 3 #include <v1model.p4> 5 #define test test2 7 const int HOPS = 1; 8 const int MAX HOPS = 10; 9 const int STANDARD = 0; 10 11 typedef standard_metadata_t std_meta_t; 12 13 header standard t { bit<8> src; 14 bit<8> dst; 15 16 } 17 18 ``` But what is BMV2? # **GLAD YOU ASKED!** ## **GLAD YOU ASKED!** "The simple\_switch architecture is the de-facto architecture for most users, as it is roughly equivalent to the "abstract switch model" described in the P4\_14 spec. ## **GLAD YOU ASKED!** "The simple\_switch architecture is the de-facto architecture for most users, as it is roughly equivalent to the "abstract switch model" described in the P4\_14 spec. Basically an interface for hardware targeting the switch. ## setup To use P4 you need a target, the three most common targets are: ### setup To use P4 you need a target, the three most common targets are: • userland (eBPF, DPDK, etc) ### setup To use P4 you need a target, the three most common targets are: - userland (eBPF, DPDK, etc) - hardware (FPGAs, custom ASIC) To use P4 you need a target, the three most common targets are: - userland (eBPF, DPDK, etc) - hardware (FPGAs, custom ASIC) - emulated (BMV2) To use P4 you need a target, the three most common targets are: - userland (eBPF, DPDK, etc) - hardware (FPGAs, custom ASIC) - emulated (BMV2) Obviously, those need some kind of interface! To use P4 you need a target, the three most common targets are: - userland (eBPF, DPDK, etc) - hardware (FPGAs, custom ASIC) - emulated (BMV2) Obviously, those need some kind of interface! The Abstract Switch Interface is usually used, with a few per-device changes To use P4 you need a target, the three most common targets are: - userland (eBPF, DPDK, etc) - hardware (FPGAs, custom ASIC) - emulated (BMV2) Obviously, those need some kind of interface! The Abstract Switch Interface is usually used, with a few per-device changes This also needs changes to the transpiler! Introducing: FPGAs on Nix Introducing: FPGAs on Nix (Yes, really) Introducing: FPGAs on Nix (Yes, really) (I forgot to take the picture before going to FOSDEM so imagine an FPGA sitting on a computer, with USB and ethernet plugged in) The core idea is: ## The core idea is: 1. Add hardware definitions to Nix for FPGAs #### The core idea is: - 1. Add hardware definitions to Nix for FPGAs - 2. Add an auto-reload/deploy mechanism through USB or custom interfaces (e.g. JTAG) #### The core idea is: - 1. Add hardware definitions to Nix for FPGAs - 2. Add an auto-reload/deploy mechanism through USB or custom interfaces (e.g. JTAG) - 3. Add a data plane mechanism for feeding data from the host (further off). #### The core idea is: - 1. Add hardware definitions to Nix for FPGAs - 2. Add an auto-reload/deploy mechanism through USB or custom interfaces (e.g. JTAG) - 3. Add a data plane mechanism for feeding data from the host (further off). All of this is a work-in-progress for now. #### The core idea is: - 1. Add hardware definitions to Nix for FPGAs - 2. Add an auto-reload/deploy mechanism through USB or custom interfaces (e.g. JTAG) - 3. Add a data plane mechanism for feeding data from the host (further off). All of this is a work-in-progress for now. But software P4 works! konami-code # QUESTIONS? konami-code # THANK YOU! @GovanifY govanify.com(/resume.pdf) gauvain@govanify.com secure-boot # ONE LAST THING...