2019-02-21 22:38:09 +00:00
|
|
|
import nnpy
|
|
|
|
import struct
|
|
|
|
|
|
|
|
from p4utils.utils.topology import Topology
|
|
|
|
from p4utils.utils.sswitch_API import SimpleSwitchAPI
|
|
|
|
|
|
|
|
from scapy.all import sniff, get_if_list, Ether, get_if_hwaddr, sendp
|
|
|
|
from scapy.all import IP, Raw, IPv6, TCP, TCP_client
|
|
|
|
from scapy.all import Ether, sniff, Packet, BitField
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import re
|
|
|
|
|
|
|
|
import logging
|
2019-02-23 13:22:46 +00:00
|
|
|
import argparse
|
2019-02-23 14:13:47 +00:00
|
|
|
import subprocess
|
|
|
|
|
2019-02-23 17:14:34 +00:00
|
|
|
# Broken in python2
|
|
|
|
#import ipaddress
|
|
|
|
|
2019-02-21 22:38:09 +00:00
|
|
|
logging.basicConfig()
|
|
|
|
log = logging.getLogger("main")
|
|
|
|
|
|
|
|
class L2Controller(object):
|
|
|
|
def __init__(self, sw_name):
|
2019-02-23 13:36:19 +00:00
|
|
|
# Command line mapping
|
2019-02-23 17:58:04 +00:00
|
|
|
self.modes = ['base', 'router']
|
|
|
|
|
|
|
|
|
|
|
|
self.address_suffix = 42
|
2019-02-23 13:22:46 +00:00
|
|
|
|
|
|
|
# Network / egress
|
2019-02-23 17:20:07 +00:00
|
|
|
self.v6_mask = "64"
|
2019-02-23 17:14:34 +00:00
|
|
|
|
2019-02-23 13:22:46 +00:00
|
|
|
self.v6_routes = {}
|
2019-02-23 14:32:03 +00:00
|
|
|
self.v6_routes[None] = []
|
2019-02-23 13:22:46 +00:00
|
|
|
self.v6_routes['base'] = []
|
2019-02-23 17:14:34 +00:00
|
|
|
self.v6_routes['base'].append({ "net": "2001:db8:61::", "port": "1"})
|
|
|
|
self.v6_routes['base'].append({ "net": "2001:db8:62::", "port": "2"})
|
2019-02-23 17:58:04 +00:00
|
|
|
self.v6_routes['router'] = self.v6_routes['base']
|
2019-02-23 13:36:19 +00:00
|
|
|
|
2019-02-23 17:20:07 +00:00
|
|
|
self.v4_mask = "24"
|
2019-02-23 13:22:46 +00:00
|
|
|
self.v4_routes = {}
|
2019-02-23 14:32:03 +00:00
|
|
|
self.v4_routes[None] = []
|
2019-02-23 13:22:46 +00:00
|
|
|
self.v4_routes['base'] = []
|
2019-02-23 17:14:34 +00:00
|
|
|
self.v4_routes['base'].append({ "net": "10.0.41.", "port": "3"})
|
|
|
|
self.v4_routes['base'].append({ "net": "10.0.42.", "port": "4"})
|
2019-02-23 17:58:04 +00:00
|
|
|
self.v4_routes['router'] = self.v4_routes['base']
|
|
|
|
|
|
|
|
self.v6_addresses = {}
|
|
|
|
self.v6_addresses[None] = []
|
|
|
|
self.v6_addresses['base'] = []
|
|
|
|
self.v6_addresses['router'] = [ { "addr": "{}{}".format(n['net'], self.address_suffix), "port": n['port'] } for n in self.v6_routes['router'] ]
|
|
|
|
|
|
|
|
self.v4_addresses = {}
|
|
|
|
self.v4_addresses[None] = []
|
|
|
|
self.v4_addresses['base'] = []
|
|
|
|
self.v4_addresses['router'] = [ { "addr": "{}{}".format(n['net'], self.address_suffix), "port": n['port'] } for n in self.v4_routes['router'] ]
|
2019-02-23 13:22:46 +00:00
|
|
|
|
2019-02-23 13:36:19 +00:00
|
|
|
self.init_boilerplate(sw_name)
|
|
|
|
|
2019-02-23 17:14:34 +00:00
|
|
|
@staticmethod
|
|
|
|
def prefix_to_net(net, mask):
|
|
|
|
""" work around ipaddress bug """
|
2019-02-23 17:25:53 +00:00
|
|
|
net = "{}0/{}".format(net, mask)
|
|
|
|
log.debug("net: {}".format(net))
|
|
|
|
return net
|
2019-02-23 17:14:34 +00:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def prefix_to_host(net, mask):
|
|
|
|
""" work around ipaddress bug """
|
2019-02-23 17:25:53 +00:00
|
|
|
host = "{}1/{}".format(net, mask)
|
|
|
|
log.debug("host: {}".format(host))
|
2019-02-23 17:14:34 +00:00
|
|
|
return "{}1/{}".format(net, mask)
|
|
|
|
|
2019-02-23 17:17:00 +00:00
|
|
|
@staticmethod
|
2019-02-23 17:14:34 +00:00
|
|
|
def add_host_ips(host, net, ipaddr, dev):
|
2019-02-23 17:19:00 +00:00
|
|
|
subprocess.call(["mx", host, "ip", "addr", "flush", "dev", dev])
|
|
|
|
subprocess.call(["mx", host, "sysctl", "net.ipv6.conf.lo.disable_ipv6=0"])
|
|
|
|
subprocess.call(["mx", host, "sysctl", "net.ipv6.conf.{}.disable_ipv6=0".format(dev)])
|
|
|
|
subprocess.call(["mx", host, "ip", "addr", "add", ipaddr, "dev", dev])
|
2019-02-23 17:14:34 +00:00
|
|
|
|
2019-02-21 22:38:09 +00:00
|
|
|
def init_boilerplate(self, sw_name):
|
|
|
|
self.topo = Topology(db="topology.db")
|
|
|
|
self.sw_name = sw_name
|
|
|
|
self.thrift_port = self.topo.get_thrift_port(sw_name)
|
|
|
|
self.cpu_port = self.topo.get_cpu_port_index(self.sw_name)
|
|
|
|
self.controller = SimpleSwitchAPI(self.thrift_port)
|
|
|
|
self.intf = str(self.topo.get_cpu_port_intf(self.sw_name).replace("eth0", "eth1"))
|
|
|
|
self.controller.reset_state()
|
|
|
|
|
|
|
|
if self.cpu_port:
|
|
|
|
self.controller.mirroring_add(100, self.cpu_port)
|
|
|
|
|
2019-02-23 13:39:26 +00:00
|
|
|
def config(self):
|
|
|
|
self.fill_tables()
|
|
|
|
self.config_hosts()
|
|
|
|
|
2019-02-23 17:58:04 +00:00
|
|
|
|
2019-02-21 22:38:09 +00:00
|
|
|
def fill_tables(self):
|
2019-02-23 13:49:18 +00:00
|
|
|
self.controller.table_clear("v6_routing")
|
2019-02-23 13:22:46 +00:00
|
|
|
for v6route in self.v6_routes[self.mode]:
|
2019-02-23 17:14:34 +00:00
|
|
|
net = self.prefix_to_net(v6route['net'], self.v6_mask)
|
|
|
|
self.controller.table_add("v6_routing", "set_egress_port", [net], [v6route['port']])
|
2019-02-23 13:22:46 +00:00
|
|
|
|
2019-02-23 13:49:18 +00:00
|
|
|
self.controller.table_clear("v4_routing")
|
2019-02-23 13:22:46 +00:00
|
|
|
for v4route in self.v4_routes[self.mode]:
|
2019-02-23 17:14:34 +00:00
|
|
|
net = self.prefix_to_net(v4route['net'], self.v4_mask)
|
2019-02-23 17:28:47 +00:00
|
|
|
self.controller.table_add("v4_routing", "set_egress_port", [net], [v4route['port']])
|
2019-02-23 13:22:46 +00:00
|
|
|
|
2019-02-23 17:58:04 +00:00
|
|
|
self.controller.table_clear("v6_addresses")
|
|
|
|
for v6addr in self.v6_addresses[self.mode]:
|
2019-02-23 18:52:01 +00:00
|
|
|
self.controller.table_add("v6_addresses", "icmp6_answer", [v6addr['addr']])
|
2019-02-23 17:58:04 +00:00
|
|
|
|
|
|
|
|
2019-02-23 13:22:46 +00:00
|
|
|
def config_hosts(self):
|
|
|
|
""" Assumptions:
|
|
|
|
- all routes are networks (no /128 v6 or /32 v4
|
|
|
|
- hosts get the first ip address in the network
|
|
|
|
"""
|
2019-02-23 14:13:47 +00:00
|
|
|
|
2019-02-23 13:49:18 +00:00
|
|
|
for v6route in self.v6_routes[self.mode]:
|
2019-02-23 14:13:47 +00:00
|
|
|
host = "h{}".format(v6route['port'])
|
|
|
|
dev = "{}-eth0".format(host)
|
2019-02-23 17:14:34 +00:00
|
|
|
#net = ipaddress.ip_network(v6route['net'])
|
|
|
|
#ipaddr = "{}/{}".format(net[1],net.prefix_length)
|
|
|
|
net = self.prefix_to_net(v6route['net'], self.v6_mask)
|
|
|
|
ipaddr = self.prefix_to_host(v6route['net'], self.v6_mask)
|
2019-02-23 13:49:18 +00:00
|
|
|
|
2019-02-23 17:14:34 +00:00
|
|
|
self.add_host_ips(host, net, ipaddr, dev)
|
2019-02-23 14:13:47 +00:00
|
|
|
|
|
|
|
# mx h$i "ip -6 route add default via 2001:db8:6::42"
|
|
|
|
|
|
|
|
#for v4route in self.v4_routes[self.mode]:
|
2019-02-23 18:52:01 +00:00
|
|
|
# Todo: add v4 routes / ip addresses
|
|
|
|
|
|
|
|
# Fake mac addresses - this gets REALLY messy now
|
|
|
|
# for host in range(4):
|
|
|
|
# macrange = [1, 2, 3, 4, 42]
|
|
|
|
|
|
|
|
# for mac in macrange:
|
|
|
|
# subprocess.call(["mx", host, "ip", "addr", "add", ipaddr, "dev", dev])
|
2019-02-23 13:49:18 +00:00
|
|
|
|
2019-02-21 22:38:09 +00:00
|
|
|
|
|
|
|
def debug_print_pkg(self, pkg, msg="INCOMING"):
|
|
|
|
log.info("{}: {}".format(msg, self.debug_format_pkg(pkg)))
|
|
|
|
|
|
|
|
def debug_format_pkg(self, pkg):
|
|
|
|
packet = Ether(str(pkg))
|
|
|
|
|
|
|
|
if packet.type == 0x800:
|
|
|
|
ip = pkg.getlayer(IP)
|
|
|
|
elif packet.type == 0x86dd:
|
|
|
|
ip = pkg.getlayer(IPv6)
|
|
|
|
|
|
|
|
# tcp = pkg.getlayer(TCP)
|
|
|
|
|
|
|
|
# raw = pkg.getlayer(Raw)
|
|
|
|
|
|
|
|
# return "{}:{} => {}:{}: flags={} seq={} ack={} raw={}".format(
|
|
|
|
# ip.src, tcp.sport,
|
|
|
|
# ip.dst, tcp.dport,
|
|
|
|
# tcp.flags,
|
|
|
|
# tcp.seq,
|
|
|
|
# tcp.ack,
|
|
|
|
# raw)
|
|
|
|
|
|
|
|
|
|
|
|
def recv_msg_cpu(self, pkg):
|
|
|
|
packet = Ether(str(pkg))
|
|
|
|
|
|
|
|
if packet.type == 0x800:
|
|
|
|
pass
|
|
|
|
elif packet.type == 0x86dd:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
print("Broken pkg: {}".format(pkg))
|
|
|
|
return
|
|
|
|
|
|
|
|
def run_cpu_port_loop(self):
|
|
|
|
sniff(iface=self.intf, prn=self.recv_msg_cpu)
|
|
|
|
|
2019-02-23 13:22:46 +00:00
|
|
|
def commandline(self):
|
|
|
|
parser = argparse.ArgumentParser(description='controller++')
|
|
|
|
parser.add_argument('--mode', help='Select mode / settings to use', choices=self.modes)
|
2019-02-23 17:25:53 +00:00
|
|
|
parser.add_argument('--debug', help='Enable debug logging', action='store_true')
|
2019-02-23 17:26:34 +00:00
|
|
|
|
|
|
|
self.args = parser.parse_args()
|
2019-02-23 17:27:09 +00:00
|
|
|
self.mode = self.args.mode
|
2019-02-23 13:22:46 +00:00
|
|
|
|
|
|
|
|
2019-02-21 22:38:09 +00:00
|
|
|
if __name__ == "__main__":
|
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
|
|
|
|
if "DEBUG" in os.environ:
|
|
|
|
log.setLevel(logging.DEBUG)
|
|
|
|
else:
|
|
|
|
log.setLevel(logging.INFO)
|
|
|
|
|
|
|
|
log.info("Booting...")
|
|
|
|
log.debug("Debug enabled.")
|
|
|
|
|
|
|
|
sw_name = "s1"
|
2019-02-23 13:22:46 +00:00
|
|
|
controller = L2Controller(sw_name)
|
|
|
|
|
|
|
|
controller.commandline()
|
2019-02-23 17:25:53 +00:00
|
|
|
if controller.args.debug:
|
|
|
|
log.setLevel(logging.DEBUG)
|
2019-02-23 13:39:26 +00:00
|
|
|
controller.config()
|
2019-02-23 13:22:46 +00:00
|
|
|
controller.run_cpu_port_loop()
|