当 P4 遇上 Mininet

前言

上一篇《P4,走进网络数据平面可编程》简单介绍了 P4 ,并通过命令行的方式搭建 P4 的网络拓扑,搭建过程需要敲很多命令,比较烦琐,这篇我们将通过 python 脚本搭建。
本文涉及的代码在:https://github.com/cykun/p4-mininet

相关介绍

Mininet

当 P4 遇上 Mininet
Mininet 是一个网络模拟器,或者更准确地说是一个网络模拟编排系统。它在单个 Linux 内核上运行一组终端主机、交换机、路由器和链接。它使用轻量级虚拟化使单个系统看起来像一个完整的网络,运行相同的内核、系统和用户代码。
通过 Mininet ,我们可以写 python 脚本来搭建网络拓扑,但 Mininet 本身是不支持 bmv2 交换机的,因此需要重写 Mininet下的 Switch 和 Host 类,好在 p4lang 已经提供了这些脚本,只要提取出来就能用了:https://github.com/p4lang/tutorials/tree/master/utils

当 P4 遇上 Mininet
p4_mininet.py:继承 Mininet 的 Switch 和 Host 类,使其支持 bmv2。
p4runtime_switch.py: 继承 P4Switch 类,在其基础上,支持 grpc 调用。

实验拓扑

当 P4 遇上 Mininet
拓扑比较简单,两个虚拟主机连接同一台 bmv2 交换机。

编写拓扑脚本,取名为 run_exercise.py

from mininet.net import Mininet
from mininet.topo import Topo
from mininet.log import setLogLevel, info
from mininet.cli import CLI

from p4_mininet import P4Switch, P4Host
from p4runtime_switch import P4RuntimeSwitch

import argparse
from time import sleep

parser = argparse.ArgumentParser(description='Mininet demo')
parser.add_argument('--behavioral-exe', help='Path to behavioral executable',
                    type=str, action="store", required=True)
parser.add_argument('--thrift-port', help='Thrift server port for table updates',
                    type=int, action="store", default=9090)
parser.add_argument('--num-hosts', help='Number of hosts to connect to switch',
                    type=int, action="store", default=2)
parser.add_argument('--mode', choices=['l2', 'l3'], type=str, default='l3')
parser.add_argument('--json', help='Path to JSON config file',
                    type=str, action="store", required=True)
parser.add_argument('--pcap-dump', help='Dump packets on interfaces to pcap files',
                    type=str, action="store", required=False, default=False)

args = parser.parse_args()

class SingleSwitchTopo(Topo):
    def __init__(self, sw_path, json_path, thrift_port, pcap_dump, n, **opts):
        # Initialize topology and default options
        Topo.__init__(self, **opts)
        switch = self.addSwitch('s1',
                                sw_path = sw_path,
                                json_path = json_path,
                                thrift_port = thrift_port,
                                pcap_dump = pcap_dump)
        for h in range(n):
            host = self.addHost('h%d' % (h + 1),
                                ip = "10.0.0.%d/24" % (h + 1),
                                mac = '00:04:00:00:00:%02x' %h)
            self.addLink(host, switch)
def main():
    num_hosts = args.num_hosts
    mode = args.mode
    topo = SingleSwitchTopo(args.behavioral_exe,
                            args.json,
                            args.thrift_port,
                            args.pcap_dump,
                            num_hosts)
    net = Mininet(topo = topo,
                  host = P4Host,
                  switch = P4RuntimeSwitch,
                  controller = None)
    net.start()
    sw_mac = ["00:aa:bb:00:00:%02x" % n for n in range(num_hosts)]
    sw_addr = ["10.0.%d.1" % n for n in range(num_hosts)]
    for n in range(num_hosts):
        h = net.get('h%d' % (n + 1))
        if mode == "l2":
            h.setDefaultRoute("dev eth0")
        else:
            h.setARP(sw_addr[n], sw_mac[n])
            h.setDefaultRoute("dev eth0 via %s" % sw_addr[n])
    for n in range(num_hosts):
        h = net.get('h%d' % (n + 1))
        h.describe()
    sleep(1)
    print("Ready !")
    CLI( net )
    net.stop()
if __name__ == '__main__':
    setLogLevel( 'info' )

Mininet 的 switch 改成 P4RuntimeSwitch,本质调用 bmv2 交换机。
SingleSwitchTopo 继承 Topo ,星型拓扑,内部添加一台 bmv2 交换机,主机数目自定,默认是2台,ip 前缀都为 10.0.0.0/24
写过 Mininet 脚本的应该对这个比较熟悉。

编写执行脚本,命名为 run.sh

BMV2_SWITCH_EXE="simple_switch_grpc"

p4c --target bmv2 --arch v1model --p4runtime-files demo.p4info.txt --std p4-16 -o build p4src/demo.p4
sudo ./run_exercise.py --behavioral-exe simple_switch_grpc --json build/demo.json

这里的 demo.p4 沿用上一篇 《P4,走进网络数据平面可编程》 的 P4代码。首先先用 p4c 编译 p4src 下的 demo.p4 源码,编译后的文件放在 build 文件夹下,接着运行 run_exercise.py 脚本。

执行运行脚本

cyquen@cyquen-virtual-machine:~/P4/p4_mininet$ ./run.sh 
[sudo] password for cyquen: 
*** Creating network
*** Adding hosts:
h1 h2 
*** Adding switches:
s1 
*** Adding links:
(h1, s1) (h2, s1) 
*** Configuring hosts
h1 h2 
*** Starting controller

*** Starting 1 switches
s1 Starting P4 switch s1.
simple_switch_grpc -i 1@s1-eth1 -i 2@s1-eth2 --nanolog ipc:///tmp/bm-0-log.ipc --device-id 0 build/demo.json --thrift-port 9090 -- --grpc-server-addr 0.0.0.0:50051
P4 switch s1 has been started.

**********
h1
default interface: h1-eth0	10.0.0.1	00:04:00:00:00:00
**********
**********
h2
default interface: h2-eth0	10.0.0.2	00:04:00:00:00:01
**********
Ready !
*** Starting CLI:
mininet> 

看到了熟悉的界面

上一篇:实验4:开源控制器实践——OpenDaylight


下一篇:实验2:Open vSwitch虚拟交换机实践