* 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)
105 lines
4.9 KiB
Python
105 lines
4.9 KiB
Python
import subprocess
|
|
|
|
from shortest_path import ShortestPath
|
|
|
|
class AppController:
|
|
|
|
def __init__(self, manifest=None, target=None, topo=None, net=None, links=None):
|
|
self.manifest = manifest
|
|
self.target = target
|
|
self.conf = manifest['targets'][target]
|
|
self.topo = topo
|
|
self.net = net
|
|
self.links = links
|
|
|
|
def read_entries(self, filename):
|
|
entries = []
|
|
with open(filename, 'r') as f:
|
|
for line in f:
|
|
line = line.strip()
|
|
if line == '': continue
|
|
entries.append(line)
|
|
return entries
|
|
|
|
def add_entries(self, thrift_port=9090, sw=None, entries=None):
|
|
assert entries
|
|
if sw: thrift_port = sw.thrift_port
|
|
|
|
print '\n'.join(entries)
|
|
p = subprocess.Popen(['simple_switch_CLI', '--thrift-port', str(thrift_port)], stdin=subprocess.PIPE)
|
|
p.communicate(input='\n'.join(entries))
|
|
|
|
def read_register(self, register, idx, thrift_port=9090, sw=None):
|
|
if sw: thrift_port = sw.thrift_port
|
|
p = subprocess.Popen(['simple_switch_CLI', '--thrift-port', str(thrift_port)], stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
|
stdout, stderr = p.communicate(input="register_read %s %d" % (register, idx))
|
|
reg_val = filter(lambda l: ' %s[%d]' % (register, idx) in l, stdout.split('\n'))[0].split('= ', 1)[1]
|
|
return long(reg_val)
|
|
|
|
def start(self):
|
|
shortestpath = ShortestPath(self.links)
|
|
entries = {}
|
|
for sw in self.topo.switches():
|
|
entries[sw] = []
|
|
if 'switches' in self.conf and sw in self.conf['switches'] and 'entries' in self.conf['switches'][sw]:
|
|
extra_entries = self.conf['switches'][sw]['entries']
|
|
if type(extra_entries) == list: # array of entries
|
|
entries[sw] += extra_entries
|
|
else: # path to file that contains entries
|
|
entries[sw] += self.read_entries(extra_entries)
|
|
#entries[sw] += [
|
|
# 'table_set_default send_frame _drop',
|
|
# 'table_set_default forward _drop',
|
|
# 'table_set_default ipv4_lpm _drop']
|
|
|
|
for host_name in self.topo._host_links:
|
|
h = self.net.get(host_name)
|
|
for link in self.topo._host_links[host_name].values():
|
|
sw = link['sw']
|
|
#entries[sw].append('table_add send_frame rewrite_mac %d => %s' % (link['sw_port'], link['sw_mac']))
|
|
#entries[sw].append('table_add forward set_dmac %s => %s' % (link['host_ip'], link['host_mac']))
|
|
#entries[sw].append('table_add ipv4_lpm set_nhop %s/32 => %s %d' % (link['host_ip'], link['host_ip'], link['sw_port']))
|
|
iface = h.intfNames()[link['idx']]
|
|
# use mininet to set ip and mac to let it know the change
|
|
h.setIP(link['host_ip'], 24)
|
|
h.setMAC(link['host_mac'])
|
|
#h.cmd('ifconfig %s %s hw ether %s' % (iface, link['host_ip'], link['host_mac']))
|
|
h.cmd('arp -i %s -s %s %s' % (iface, link['sw_ip'], link['sw_mac']))
|
|
h.cmd('ethtool --offload %s rx off tx off' % iface)
|
|
h.cmd('ip route add %s dev %s' % (link['sw_ip'], iface))
|
|
h.setDefaultRoute("via %s" % link['sw_ip'])
|
|
|
|
for h in self.net.hosts:
|
|
h_link = self.topo._host_links[h.name].values()[0]
|
|
for sw in self.net.switches:
|
|
path = shortestpath.get(sw.name, h.name, exclude=lambda n: n[0]=='h')
|
|
if not path: continue
|
|
if not path[1][0] == 's': continue # next hop is a switch
|
|
sw_link = self.topo._sw_links[sw.name][path[1]]
|
|
#entries[sw.name].append('table_add send_frame rewrite_mac %d => %s' % (sw_link[0]['port'], sw_link[0]['mac']))
|
|
#entries[sw.name].append('table_add forward set_dmac %s => %s' % (h_link['host_ip'], sw_link[1]['mac']))
|
|
#entries[sw.name].append('table_add ipv4_lpm set_nhop %s/32 => %s %d' % (h_link['host_ip'], h_link['host_ip'], sw_link[0]['port']))
|
|
|
|
for h2 in self.net.hosts:
|
|
if h == h2: continue
|
|
path = shortestpath.get(h.name, h2.name, exclude=lambda n: n[0]=='h')
|
|
if not path: continue
|
|
h_link = self.topo._host_links[h.name][path[1]]
|
|
h2_link = self.topo._host_links[h2.name].values()[0]
|
|
h.cmd('ip route add %s via %s' % (h2_link['host_ip'], h_link['sw_ip']))
|
|
|
|
|
|
print "**********"
|
|
print "Configuring entries in p4 tables"
|
|
for sw_name in entries:
|
|
print
|
|
print "Configuring switch... %s" % sw_name
|
|
sw = self.net.get(sw_name)
|
|
if entries[sw_name]:
|
|
self.add_entries(sw=sw, entries=entries[sw_name])
|
|
print "Configuration complete."
|
|
print "**********"
|
|
|
|
def stop(self):
|
|
pass
|