| |
| # Copyright 2017-present Open Networking Foundation |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| |
| """ |
| Check README file |
| """ |
| import Queue |
| |
| from oftest import config |
| import inspect |
| import logging |
| import oftest.base_tests as base_tests |
| import ofp |
| import time |
| from oftest.oft12.testutils import delete_all_flows_one_table |
| from oftest.testutils import * |
| from accton_util import * |
| from utils import * |
| |
| class PacketInUDP( base_tests.SimpleDataPlane ): |
| """ |
| Verify ACL rule for IP_PROTO=2 wont match a UDP packet and a rule for IP_PROTO=17 WILL match a UDP packet. |
| """ |
| |
| def runTest( self ): |
| try: |
| parsed_vlan_pkt = simple_udp_packet( pktlen=104, vlan_vid=0x1001, dl_vlan_enable=True ) |
| vlan_pkt = str( parsed_vlan_pkt ) |
| # create match |
| match = ofp.match( ) |
| match.oxm_list.append( ofp.oxm.eth_type( 0x0800 ) ) |
| match.oxm_list.append( ofp.oxm.ip_proto( 2 ) ) |
| request = ofp.message.flow_add( table_id=60, cookie=42, match=match, instructions=[ |
| ofp.instruction.apply_actions( actions=[ |
| ofp.action.output( port=ofp.OFPP_CONTROLLER, max_len=ofp.OFPCML_NO_BUFFER ) ] ), ], |
| buffer_id=ofp.OFP_NO_BUFFER, priority=1 ) |
| logging.info( "Inserting packet in flow to controller" ) |
| self.controller.message_send( request ) |
| |
| for of_port in config[ "port_map" ].keys( ): |
| logging.info( "PacketInMiss test, port %d", of_port ) |
| self.dataplane.send( of_port, vlan_pkt ) |
| |
| verify_no_packet_in( self, vlan_pkt, of_port ) |
| delete_all_flows( self.controller ) |
| do_barrier( self.controller ) |
| |
| match = ofp.match( ) |
| match.oxm_list.append( ofp.oxm.eth_type( 0x0800 ) ) |
| match.oxm_list.append( ofp.oxm.ip_proto( 17 ) ) |
| request = ofp.message.flow_add( table_id=60, cookie=42, match=match, instructions=[ |
| ofp.instruction.apply_actions( actions=[ |
| ofp.action.output( port=ofp.OFPP_CONTROLLER, max_len=ofp.OFPCML_NO_BUFFER ) ] ), ], |
| buffer_id=ofp.OFP_NO_BUFFER, priority=1 ) |
| logging.info( "Inserting packet in flow to controller" ) |
| self.controller.message_send( request ) |
| do_barrier( self.controller ) |
| |
| for of_port in config[ "port_map" ].keys( ): |
| logging.info( "PacketInMiss test, port %d", of_port ) |
| self.dataplane.send( of_port, vlan_pkt ) |
| |
| verify_packet_in( self, vlan_pkt, of_port, ofp.OFPR_ACTION ) |
| |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| |
| @disabled |
| class ArpNL2( base_tests.SimpleDataPlane ): |
| """ |
| Needs a description, disabled for now. Also needs try/finally |
| """ |
| def runTest( self ): |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| match = ofp.match( ) |
| match.oxm_list.append( ofp.oxm.eth_type( 0x0806 ) ) |
| request = ofp.message.flow_add( table_id=60, cookie=42, match=match, instructions=[ |
| ofp.instruction.apply_actions( actions=[ |
| ofp.action.output( port=ofp.OFPP_CONTROLLER, max_len=ofp.OFPCML_NO_BUFFER ) ] ), ], |
| buffer_id=ofp.OFP_NO_BUFFER, priority=40000 ) |
| self.controller.message_send( request ) |
| for port in ports: |
| add_one_l2_interface_group( self.controller, port, 1, False, False ) |
| add_one_vlan_table_flow( self.controller, port, 1, flag=VLAN_TABLE_FLAG_ONLY_BOTH ) |
| group_id = encode_l2_interface_group_id( 1, port ) |
| add_bridge_flow( self.controller, [ 0x00, 0x12, 0x34, 0x56, 0x78, port ], 1, group_id, True ) |
| do_barrier( self.controller ) |
| parsed_arp_pkt = simple_arp_packet( ) |
| arp_pkt = str( parsed_arp_pkt ) |
| |
| for out_port in ports: |
| self.dataplane.send( out_port, arp_pkt ) |
| verify_packet_in( self, arp_pkt, out_port, ofp.OFPR_ACTION ) |
| # change dest based on port number |
| mac_dst = '00:12:34:56:78:%02X' % out_port |
| for in_port in ports: |
| if in_port == out_port: |
| continue |
| # change source based on port number to avoid packet-ins from learning |
| mac_src = '00:12:34:56:78:%02X' % in_port |
| parsed_pkt = simple_tcp_packet( eth_dst=mac_dst, eth_src=mac_src ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for ofport in ports: |
| if ofport in [ out_port ]: |
| verify_packet( self, pkt, ofport ) |
| else: |
| verify_no_packet( self, pkt, ofport ) |
| |
| verify_no_other_packets( self ) |
| |
| class PacketInArp( base_tests.SimpleDataPlane ): |
| """ |
| Verify Packet-in message from eth_type 0x806 on ACL table |
| """ |
| |
| def runTest( self ): |
| try: |
| parsed_arp_pkt = simple_arp_packet( ) |
| arp_pkt = str( parsed_arp_pkt ) |
| # create match |
| match = ofp.match( ) |
| match.oxm_list.append( ofp.oxm.eth_type( 0x0806 ) ) |
| request = ofp.message.flow_add( table_id=60, cookie=42, match=match, instructions=[ |
| ofp.instruction.apply_actions( actions=[ |
| ofp.action.output( port=ofp.OFPP_CONTROLLER, max_len=ofp.OFPCML_NO_BUFFER ) ] ), ], |
| buffer_id=ofp.OFP_NO_BUFFER, priority=1 ) |
| |
| logging.info( "Inserting arp flow " ) |
| self.controller.message_send( request ) |
| do_barrier( self.controller ) |
| |
| for of_port in config[ "port_map" ].keys( ): |
| logging.info( "PacketInArp test, sending arp packet to port %d", of_port ) |
| self.dataplane.send( of_port, arp_pkt ) |
| |
| verify_packet_in( self, arp_pkt, of_port, ofp.OFPR_ACTION ) |
| |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class PacketInIPTable( base_tests.SimpleDataPlane ): |
| """ |
| Verify Packet-in message from IP table when controller action is used |
| Send a packet to each dataplane port and verify that a packet |
| in message is received from the controller for each |
| #todo verify you stop receiving after adding rule |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| |
| for port in ports: |
| # add l2 interface group |
| vlan_id = port |
| add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id, is_tagged=True, |
| send_barrier=False ) |
| dst_mac[ 5 ] = vlan_id |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=vlan_id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, vlan_id=vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffff00, l3_msg.group_id, |
| send_ctrl=True ) |
| Groups.put( l3_msg.group_id ) |
| |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % in_port |
| ip_src = '192.168.%02d.1' % in_port |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % out_port |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=in_port, |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| verify_packet_in( self, pkt, in_port, ofp.OFPR_ACTION ) |
| # verify_no_other_packets(self) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class L2FloodQinQ( base_tests.SimpleDataPlane ): |
| """ |
| Verify Vlan based flooding of QinQ based on its outer vlan |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| vlan_id = 100 |
| |
| for port in ports: |
| L2gid, l2msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| add_one_vlan_table_flow( self.controller, port, vlan_id=vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| Groups.put( L2gid ) |
| |
| msg = add_l2_flood_group( self.controller, ports, vlan_id, vlan_id ) |
| Groups.put( msg.group_id ) |
| add_bridge_flow( self.controller, None, vlan_id, msg.group_id, True ) |
| do_barrier( self.controller ) |
| |
| # verify flood |
| for ofport in ports: |
| # change dest based on port number |
| mac_src = '00:12:34:56:78:%02X' % ofport |
| parsed_pkt = simple_tcp_packet_two_vlan( pktlen=108, out_dl_vlan_enable=True, |
| out_vlan_vid=vlan_id, in_dl_vlan_enable=True, in_vlan_vid=10, |
| eth_dst='00:12:34:56:78:9a', eth_src=mac_src ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( ofport, pkt ) |
| # self won't rx packet |
| verify_no_packet( self, pkt, ofport ) |
| # others will rx packet |
| tmp_ports = list( ports ) |
| tmp_ports.remove( ofport ) |
| verify_packets( self, pkt, tmp_ports ) |
| |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| |
| @disabled |
| class L2FloodTagged( base_tests.SimpleDataPlane ): |
| """ |
| currently disabled; fix with try/finally |
| Test L2 flood to a vlan |
| Send a packet with unknown dst_mac and check if the packet is flooded to all ports except inport |
| """ |
| |
| def runTest( self ): |
| # Hashes Test Name and uses it as id for installing unique groups |
| vlan_id = abs( hash( inspect.stack( )[ 0 ][ 3 ] ) ) % (256) |
| print vlan_id |
| |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| # Installing flows to avoid packet-in |
| for port in ports: |
| add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| add_one_vlan_table_flow( self.controller, port, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| msg = add_l2_flood_group( self.controller, ports, vlan_id, vlan_id ) |
| add_bridge_flow( self.controller, None, vlan_id, msg.group_id, True ) |
| do_barrier( self.controller ) |
| |
| # verify flood |
| for ofport in ports: |
| # change dest based on port number |
| pkt = str( |
| simple_tcp_packet( dl_vlan_enable=True, vlan_vid=vlan_id, eth_dst='00:12:34:56:78:9a' ) ) |
| self.dataplane.send( ofport, pkt ) |
| # self won't rx packet |
| verify_no_packet( self, pkt, ofport ) |
| # others will rx packet |
| tmp_ports = list( ports ) |
| tmp_ports.remove( ofport ) |
| verify_packets( self, pkt, tmp_ports ) |
| verify_no_other_packets( self ) |
| |
| |
| class L2UnicastTagged( base_tests.SimpleDataPlane ): |
| """ Verify Bridging works: match(VID, DST_MAC)> fwd(port) """ |
| |
| def runTest( self ): |
| |
| Groups = Queue.LifoQueue( ) |
| try: |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| vlan_id = 1; |
| for port in ports: |
| L2gid, l2msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| add_one_vlan_table_flow( self.controller, port, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| Groups.put( L2gid ) |
| add_bridge_flow( self.controller, [ 0x00, 0x12, 0x34, 0x56, 0x78, port ], vlan_id, L2gid, |
| True ) |
| do_barrier( self.controller ) |
| |
| for out_port in ports: |
| # change dest based on port number |
| mac_dst = '00:12:34:56:78:%02X' % out_port |
| for in_port in ports: |
| if in_port == out_port: |
| continue |
| pkt = str( simple_tcp_packet( dl_vlan_enable=True, vlan_vid=vlan_id, eth_dst=mac_dst ) ) |
| self.dataplane.send( in_port, pkt ) |
| for ofport in ports: |
| if ofport in [ out_port ]: |
| verify_packet( self, pkt, ofport ) |
| else: |
| verify_no_packet( self, pkt, ofport ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class Bridging( base_tests.SimpleDataPlane ): |
| """ |
| Verify bridging works including flooding with different vlans |
| ports[0] has vlan 31 untagged |
| ports[1] has vlan 31 untagged (native) and vlan 41 tagged |
| ARP request should be flooded |
| ARP reply should be forwarded by bridging rule |
| Both arp messages should also be copied to controller |
| """ |
| |
| def runTest( self ): |
| Groupd = Queue.LifoQueue() |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| ports = sorted( config[ "port_map" ].keys() ) |
| vlan_p0_untagged = 31 |
| vlan_p1_tagged = 31 |
| vlan_p1_native = 41 |
| |
| #l2 interface groups and table 10 flows |
| L2p0gid, l2msg0 = add_one_l2_interface_group( self.controller, ports[0], vlan_p0_untagged, |
| is_tagged=False, send_barrier=False ) |
| add_one_vlan_table_flow( self.controller, ports[0], vlan_id=vlan_p0_untagged, flag=VLAN_TABLE_FLAG_ONLY_BOTH, |
| send_barrier=True) |
| L2p1gid, l2msg1 = add_one_l2_interface_group( self.controller, ports[1], vlan_p1_tagged, |
| is_tagged=True, send_barrier=False ) |
| add_one_vlan_table_flow( self.controller, ports[1], vlan_id=vlan_p1_tagged, flag=VLAN_TABLE_FLAG_ONLY_TAG, |
| send_barrier=True) |
| L2p1gid2, l2msg3 = add_one_l2_interface_group( self.controller, ports[1], vlan_p1_native, |
| is_tagged=False, send_barrier=False ) |
| add_one_vlan_table_flow( self.controller, ports[1], vlan_id=vlan_p1_native, flag=VLAN_TABLE_FLAG_ONLY_BOTH, |
| send_barrier=True) |
| #flooding groups |
| Floodmsg31 = add_l2_flood_group( self.controller, ports, vlan_p0_untagged, id=0 ) |
| Floodmsg41 = add_l2_flood_group( self.controller, [ ports[1] ], vlan_p1_native, id=0 ) |
| |
| #add bridging flows for flooding groups |
| add_bridge_flow( self.controller, dst_mac=None, vlanid=vlan_p0_untagged, group_id=Floodmsg31.group_id ) |
| add_bridge_flow( self.controller, dst_mac=None, vlanid=vlan_p1_native, group_id=Floodmsg41.group_id ) |
| do_barrier( self.controller ) |
| |
| # add bridging flows for dstMac+vlan |
| add_bridge_flow( self.controller, [ 0x00, 0x11, 0x22, 0x33, 0x44, 0x55 ], vlan_p0_untagged, L2p0gid, True ) |
| add_bridge_flow( self.controller, [ 0x00, 0x66, 0x77, 0x88, 0x99, 0xaa ], vlan_p1_tagged, L2p1gid, True ) |
| add_bridge_flow( self.controller, [ 0x00, 0x66, 0x77, 0x88, 0x99, 0xaa ], vlan_p1_native, L2p1gid2, True ) |
| |
| # add terminationMac flow |
| router_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, router_mac, vlan_p0_untagged ) |
| add_termination_flow( self.controller, 0, 0x0800, router_mac, vlan_p1_native ) |
| else: |
| add_termination_flow( self.controller, ports[0], 0x0800, router_mac, vlan_p0_untagged ) |
| add_termination_flow( self.controller, ports[1], 0x0800, router_mac, vlan_p1_tagged ) |
| add_termination_flow( self.controller, ports[1], 0x0800, router_mac, vlan_p1_native ) |
| |
| # add acl rule for arp |
| match = ofp.match( ) |
| match.oxm_list.append( ofp.oxm.eth_type( 0x0806 ) ) |
| request = ofp.message.flow_add( table_id=60, cookie=42, match=match, instructions=[ |
| ofp.instruction.apply_actions( actions=[ |
| ofp.action.output( port=ofp.OFPP_CONTROLLER, max_len=ofp.OFPCML_NO_BUFFER ) ] ), ], |
| buffer_id=ofp.OFP_NO_BUFFER, priority=1 ) |
| self.controller.message_send( request ) |
| do_barrier( self.controller ) |
| |
| #acl rule for gateway ip |
| match = ofp.match( ) |
| match.oxm_list.append( ofp.oxm.eth_type( 0x0800 ) ) |
| match.oxm_list.append( ofp.oxm.ipv4_dst( 0xc0a80003 ) ) |
| request = ofp.message.flow_add( table_id=60, cookie=42, match=match, instructions=[ |
| ofp.instruction.apply_actions( actions=[ |
| ofp.action.output( port=ofp.OFPP_CONTROLLER, max_len=ofp.OFPCML_NO_BUFFER ) ] ), |
| ofp.instruction.clear_actions() ], |
| buffer_id=ofp.OFP_NO_BUFFER, priority=1 ) |
| self.controller.message_send( request ) |
| do_barrier( self.controller ) |
| |
| # send ARP request |
| parsed_arp_pkt = simple_arp_packet(pktlen=80, |
| eth_dst='ff:ff:ff:ff:ff:ff', |
| eth_src='00:66:77:88:99:aa', |
| vlan_vid=vlan_p1_tagged, |
| vlan_pcp=0, |
| arp_op=1, |
| ip_snd='192.168.0.2', |
| ip_tgt='192.168.0.1', |
| hw_snd='00:66:77:88:99:aa', |
| hw_tgt='00:00:00:00:00:00') |
| arp_pkt_to_send = str( parsed_arp_pkt ) |
| logging.info( "sending arp request to port %d", ports[1] ) |
| self.dataplane.send( ports[1], arp_pkt_to_send ) |
| verify_packet_in( self, arp_pkt_to_send, ports[1], ofp.OFPR_ACTION ) |
| parsed_arp_pkt_untagged = simple_arp_packet(pktlen=76, |
| eth_dst='ff:ff:ff:ff:ff:ff', |
| eth_src='00:66:77:88:99:aa', |
| vlan_vid=0, |
| vlan_pcp=0, |
| arp_op=1, |
| ip_snd='192.168.0.2', |
| ip_tgt='192.168.0.1', |
| hw_snd='00:66:77:88:99:aa', |
| hw_tgt='00:00:00:00:00:00') |
| arp_pkt_dest = str( parsed_arp_pkt_untagged ) |
| verify_packet( self, arp_pkt_dest, ports[0] ) |
| #verify_no_other_packets( self ) |
| |
| # send ARP reply |
| parsed_arp_pkt = simple_arp_packet(pktlen=76, |
| eth_dst='00:66:77:88:99:aa', |
| eth_src='00:11:22:33:44:55', |
| vlan_vid=0, |
| vlan_pcp=0, |
| arp_op=2, |
| ip_snd='192.168.0.1', |
| ip_tgt='192.168.0.2', |
| hw_snd='00:11:22:33:44:55', |
| hw_tgt='00:66:77:88:99:aa') |
| arp_pkt_to_send = str( parsed_arp_pkt ) |
| logging.info( "sending arp reply to port %d", ports[0] ) |
| self.dataplane.send( ports[0], arp_pkt_to_send ) |
| verify_packet_in( self, arp_pkt_to_send, ports[0], ofp.OFPR_ACTION ) |
| parsed_arp_pkt_tagged = simple_arp_packet(pktlen=80, |
| eth_dst='00:66:77:88:99:aa', |
| eth_src='00:11:22:33:44:55', |
| vlan_vid=vlan_p1_tagged, |
| vlan_pcp=0, |
| arp_op=2, |
| ip_snd='192.168.0.1', |
| ip_tgt='192.168.0.2', |
| hw_snd='00:11:22:33:44:55', |
| hw_tgt='00:66:77:88:99:aa') |
| arp_pkt_dest = str( parsed_arp_pkt_tagged ) |
| verify_packet( self, arp_pkt_dest, ports[1] ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| #print("done") |
| |
| |
| |
| class Mtu1500( base_tests.SimpleDataPlane ): |
| """ |
| Verifies basic mtu limits |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| vlan_id = 18 |
| for port in ports: |
| L2gid, msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| Groups.put( L2gid ) |
| add_bridge_flow( self.controller, [ 0x00, 0x12, 0x34, 0x56, 0x78, port ], vlan_id, L2gid, |
| True ) |
| do_barrier( self.controller ) |
| |
| for out_port in ports: |
| # change dest based on port number |
| mac_dst = '00:12:34:56:78:%02X' % out_port |
| for in_port in ports: |
| if in_port == out_port: |
| continue |
| pkt = str( simple_tcp_packet( pktlen=1500, dl_vlan_enable=True, vlan_vid=vlan_id, |
| eth_dst=mac_dst ) ) |
| self.dataplane.send( in_port, pkt ) |
| for ofport in ports: |
| if ofport in [ out_port ]: |
| verify_packet( self, pkt, ofport ) |
| else: |
| verify_no_packet( self, pkt, ofport ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class _32UcastTagged( base_tests.SimpleDataPlane ): |
| """ Verify /32 IP forwarding to L3 Unicast-> L2Interface""" |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| test_id = 26 |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| vlan_id = port + test_id |
| # add l2 interface group and l3 unicast group |
| l2gid, msg = add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id, |
| is_tagged=True, send_barrier=False ) |
| dst_mac[ 5 ] = vlan_id |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=vlan_id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, l3_msg.group_id ) |
| Groups.put( l2gid ) |
| Groups.put( l3_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:32:%02X' % (test_id + in_port) |
| ip_src = '192.168.%02d.1' % (test_id + in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (test_id + out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, |
| vlan_vid=(test_id + in_port), eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, |
| ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % (test_id + out_port) |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, |
| vlan_vid=(test_id + out_port), eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class _32VPN( base_tests.SimpleDataPlane ): |
| """ |
| Verify /32 routing rule -> MPLS_VPN_Label -> MPLSInterface -> L2Interface |
| No ECMP group used |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| id = port |
| vlan_id = port |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, True ) |
| dst_mac[ 5 ] = vlan_id |
| # add MPLS interface group |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dst_mac, intf_src_mac, |
| vlan_id, id ) |
| # add MPLS L3 VPN group |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=True, set_mpls_label=port, set_bos=1, set_ttl=32 ) |
| do_barrier( self.controller ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, vrf=2, |
| flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, mpls_label_gid, vrf=2 ) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| ip_src = '192.168.%02d.1' % (in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(in_port), |
| eth_dst=switch_mac, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % (out_port) |
| label = (out_port, 0, 1, 32) |
| exp_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(out_port), ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=mac_dst, eth_src=switch_mac, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class _32EcmpVpn( base_tests.SimpleDataPlane ): |
| """ |
| Verify /32 routing rule -> L3 ECMP -> MPLS_VPN_Label -> MPLSInterface -> L2Interface |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| id = port |
| vlan_id = port |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, True ) |
| dst_mac[ 5 ] = vlan_id |
| # add MPLS interface group |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dst_mac, intf_src_mac, |
| vlan_id, id ) |
| # add MPLS L3 VPN group |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=True, set_mpls_label=port, set_bos=1, set_ttl=32 ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, vlan_id, [ mpls_label_gid ] ) |
| do_barrier( self.controller ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, vrf=0, |
| flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, ecmp_msg.group_id ) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| Groups._put( ecmp_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| ip_src = '192.168.%02d.1' % (in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(in_port), |
| eth_dst=switch_mac, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % (out_port) |
| label = (out_port, 0, 1, 32) |
| exp_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(out_port), ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=mac_dst, eth_src=switch_mac, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| @disabled |
| class One_32EcmpVpn( base_tests.SimpleDataPlane ): |
| """ |
| Verify /32 routing rule -> L3 ECMP -> MPLS_VPN_Label -> MPLSInterface -> L2Interface |
| in only one direction |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| # add l2 interface group |
| id = ports[1] |
| in_offset = 19 |
| out_offset = 20 |
| vlan_id = ports[1] + out_offset |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, ports[1], vlan_id, True, True ) |
| dst_mac[ 5 ] = ports[1] |
| # add MPLS interface group |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dst_mac, intf_src_mac, |
| vlan_id, id ) |
| # add MPLS L3 VPN group |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=True, set_mpls_label=ports[1] + out_offset, set_bos=1, set_ttl=32 ) |
| # add ECMP group |
| ecmp_msg = add_l3_ecmp_group( self.controller, vlan_id, [ mpls_label_gid ] ) |
| do_barrier( self.controller ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, ports[0], 1, vlan_id=ports[0] + in_offset, vrf=0, |
| flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlanid=ports[0] + in_offset ) |
| else: |
| add_termination_flow( self.controller, ports[0], 0x0800, intf_src_mac, vlanid=ports[0] + in_offset ) |
| # add routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, ecmp_msg.group_id, send_barrier=True ) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| Groups._put( ecmp_msg.group_id ) |
| |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| in_port = ports[0] |
| out_port = ports[1] |
| ip_src = '192.168.%02d.1' % (in_port) |
| ip_dst = '192.168.%02d.1' % (out_port+out_offset) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(in_port + in_offset), |
| eth_dst=switch_mac, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % (out_port) |
| label = (out_port+out_offset, 0, 1, 32) |
| exp_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(out_port + out_offset), ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=mac_dst, eth_src=switch_mac, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| #verify_no_other_packets( self ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_group(self.controller, ecmp_msg.group_id) |
| delete_group(self.controller, mpls_label_gid) |
| delete_group(self.controller, mpls_gid) |
| delete_group(self.controller, l2_gid) |
| |
| |
| class _32ECMPL3( base_tests.SimpleDataPlane ): |
| """ |
| Verifies /32 IP routing and ECMP with no label push |
| IP -> L3ECMP -> L3Unicast -> L2Interface |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| # Hashes Test Name and uses it as id for installing unique groups |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| vlan_id = port |
| id = port |
| # add l2 interface group |
| l2_gid, msg = add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id, |
| is_tagged=True, send_barrier=False ) |
| dst_mac[ 5 ] = vlan_id |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ l3_msg.group_id ] ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, ecmp_msg.group_id ) |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| Groups._put( ecmp_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % in_port |
| ip_src = '192.168.%02d.1' % in_port |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % out_port |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=in_port, |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % out_port |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=out_port, |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class One_32ECMPL3( base_tests.SimpleDataPlane ): |
| """ |
| Verifies /32 IP routing and ECMP with no label push |
| IP -> L3ECMP -> L3Unicast -> L2Interface |
| in only one direction |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| # Hashes Test Name and uses it as id for installing unique groups |
| ports = config[ "port_map" ].keys( ) |
| inport = ports[0] |
| outport = ports[1] |
| in_offset = 19 |
| out_offset = 20 |
| vlan_id = outport + out_offset |
| id = outport |
| # add l2 interface group, l3 unicast and ecmp group for outport |
| l2_gid, msg = add_one_l2_interface_group( self.controller, outport, vlan_id=vlan_id, |
| is_tagged=True, send_barrier=False ) |
| dst_mac[ 5 ] = outport |
| l3_msg = add_l3_unicast_group( self.controller, outport, vlanid=vlan_id, id=id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ l3_msg.group_id ] ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, of_port=inport, vlan_id=inport+in_offset, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlanid=inport+in_offset ) |
| else: |
| add_termination_flow( self.controller, in_port=inport, eth_type=0x0800, dst_mac=intf_src_mac, vlanid=inport+in_offset ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, ecmp_msg.group_id, send_barrier=True ) |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| Groups._put( ecmp_msg.group_id ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| mac_src = '00:00:00:22:22:%02X' % inport |
| ip_src = '192.168.%02d.1' % inport |
| ip_dst = '192.168.%02d.1' % (outport+out_offset) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=inport+in_offset, |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( inport, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % outport |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=outport+out_offset, |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, outport ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| |
| |
| @disabled |
| class _24VPN( base_tests.SimpleDataPlane ): |
| """ Verify MPLS IP VPN Initiation from /32 rule without using ECMP """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| id = port |
| vlan_id = port |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, True ) |
| dst_mac[ 5 ] = vlan_id |
| # add MPLS interface group |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dst_mac, intf_src_mac, |
| vlan_id, id ) |
| # add MPLS L3 VPN group |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=True, set_mpls_label=port, set_bos=1, set_ttl=32 ) |
| # ecmp_msg=add_l3_ecmp_group(self.controller, vlan_id, [mpls_label_gid]) |
| do_barrier( self.controller ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, vrf=0, |
| flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffff00, mpls_label_gid ) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| ip_src = '192.168.%02d.1' % (in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(in_port), |
| eth_dst=switch_mac, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % (out_port) |
| label = (out_port, 0, 1, 32) |
| exp_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(out_port), ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=mac_dst, eth_src=switch_mac, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class _24EcmpVpn( base_tests.SimpleDataPlane ): |
| """ Verify MPLS IP VPN Initiation from /24 rule using ECMP """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| id = port |
| vlan_id = id |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, True ) |
| dst_mac[ 5 ] = vlan_id |
| # add MPLS interface group |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dst_mac, intf_src_mac, |
| vlan_id, id ) |
| # add MPLS L3 VPN group |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=True, set_mpls_label=port, set_bos=1, set_ttl=32 ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ mpls_label_gid ] ) |
| do_barrier( self.controller ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, vrf=0, |
| flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add routing flow |
| dst_ip = dip + (vlan_id << 8) |
| # add_unicast_routing_flow(self.controller, 0x0800, dst_ip, 0, mpls_label_gid, vrf=2) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffff00, ecmp_msg.group_id, |
| vrf=0 ) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| Groups._put( ecmp_msg.group_id ) |
| |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % (in_port) |
| ip_src = '192.168.%02d.1' % (in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(in_port), |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % out_port |
| label = (out_port, 0, 1, 32) |
| exp_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(out_port), ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=mac_dst, eth_src=switch_mac, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class FloodGroupMod( base_tests.SimpleDataPlane ): |
| """ Modify a referenced flood group """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| vlan_id = 1 |
| |
| for port in ports: |
| L2gid, l2msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| add_one_vlan_table_flow( self.controller, port, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| Groups.put( L2gid ) |
| |
| msg = add_l2_flood_group( self.controller, ports, vlan_id, vlan_id ) |
| Groups.put( msg.group_id ) |
| add_bridge_flow( self.controller, None, vlan_id, msg.group_id, True ) |
| do_barrier( self.controller ) |
| # verify flood |
| for ofport in ports: |
| # change dest based on port number |
| mac_src = '00:12:34:56:78:%02X' % ofport |
| parsed_pkt = simple_tcp_packet_two_vlan( pktlen=108, out_dl_vlan_enable=True, |
| out_vlan_vid=vlan_id, in_dl_vlan_enable=True, in_vlan_vid=10, |
| eth_dst='00:12:34:56:78:9a', eth_src=mac_src ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( ofport, pkt ) |
| # self won't rx packet |
| verify_no_packet( self, pkt, ofport ) |
| # others will rx packet |
| tmp_ports = list( ports ) |
| tmp_ports.remove( ofport ) |
| verify_packets( self, pkt, tmp_ports ) |
| verify_no_other_packets( self ) |
| msg = mod_l2_flood_group( self.controller, [ ports[ 0 ] ], vlan_id, vlan_id ) |
| mac_src = '00:12:34:56:78:%02X' % ports[ 1 ] |
| parsed_pkt = simple_tcp_packet_two_vlan( pktlen=108, out_dl_vlan_enable=True, |
| out_vlan_vid=vlan_id, in_dl_vlan_enable=True, in_vlan_vid=10, eth_dst='00:12:34:56:78:9a', |
| eth_src=mac_src ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( ports[ 1 ], pkt ) |
| verify_packets( self, pkt, [ ports[ 0 ] ] ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class _24ECMPL3( base_tests.SimpleDataPlane ): |
| """ Verifies /24 IP routing using ECMP -> L3U -> L2I """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| # Hashes Test Name and uses it as id for installing unique groups |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| vlan_id = port |
| id = port |
| # add l2 interface group |
| l2_gid, msg = add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id, |
| is_tagged=True, send_barrier=False ) |
| dst_mac[ 5 ] = vlan_id |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ l3_msg.group_id ] ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffff00, ecmp_msg.group_id ) |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| Groups._put( ecmp_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % in_port |
| ip_src = '192.168.%02d.1' % in_port |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % out_port |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=in_port, |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % out_port |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=out_port, |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| @disabled |
| class MPLSBUG( base_tests.SimpleDataPlane ): |
| """ |
| Needs a description or needs to be removed |
| """ |
| def runTest( self ): |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| Groups = Queue.LifoQueue( ) |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| vlan_id = port |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| dst_mac[ 5 ] = vlan_id |
| # add L3 Unicast group |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=vlan_id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_BOTH ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x08847, intf_src_mac, vlan_id, goto_table=24 ) |
| else: |
| add_termination_flow( self.controller, port, 0x8847, intf_src_mac, vlan_id, goto_table=24 ) |
| # add mpls flow |
| add_mpls_flow( self.controller, l3_msg.group_id, port ) |
| # add termination flow |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, l3_msg.group_id ) |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % in_port |
| ip_src = '192.168.%02d.1' % in_port |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % out_port |
| switch_mac = "00:00:00:cc:cc:cc" |
| label = (out_port, 0, 1, 32) |
| parsed_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=in_port, ip_src=ip_src, |
| ip_dst=ip_dst, eth_dst=switch_mac, eth_src=mac_src, label=[ label ] ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % out_port |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=out_port, |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=31, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=in_port, |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % out_port |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=out_port, |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| |
| class L3McastToL3( base_tests.SimpleDataPlane ): |
| """ |
| Mcast routing, in this test case the traffic comes in tagged. |
| port+1 is used as ingress vlan_id. The packet goes out tagged on |
| different ports. 4094-port is used as egress vlan_id. |
| """ |
| def runTest( self ): |
| """ |
| port1 (vlan 300)-> All Ports (vlan 300) |
| """ |
| Groups = Queue.LifoQueue( ) |
| try: |
| # We can forward on the in_port but egress_vlan has to be different from ingress_vlan |
| if len( config[ "port_map" ] ) < 1: |
| logging.info( "Port count less than 1, can't run this case" ) |
| assert (False) |
| return |
| ports = config[ "port_map" ].keys( ) |
| dst_ip_str = "224.0.0.1" |
| ( |
| port_to_in_vlan, |
| port_to_out_vlan, |
| port_to_src_mac_str, |
| port_to_dst_mac_str, |
| port_to_src_ip, |
| port_to_src_ip_str, |
| port_to_intf_src_mac_str, |
| Groups) = fill_mcast_pipeline_L3toL3( |
| self.controller, |
| logging, |
| ports, |
| is_ingress_tagged = True, |
| is_egress_tagged = True, |
| is_vlan_translated = True, |
| is_max_vlan = False |
| ) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_in_vlan[in_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_out_vlan[out_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_intf_src_mac_str[out_port], |
| ip_ttl = 63, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| verify_packet( self, pkt, out_port ) |
| |
| verify_no_other_packets( self ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class L3McastToL2UntagToUntag( base_tests.SimpleDataPlane ): |
| """ |
| Fails on alternate runs |
| Mcast routing, in this test case the traffic is untagged. |
| 4094 is used as internal vlan_id. The packet goes out |
| untagged. |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| assert (False) |
| return |
| ports = config[ "port_map" ].keys( ) |
| dst_ip_str = "224.0.0.1" |
| ( |
| port_to_in_vlan, |
| port_to_out_vlan, |
| port_to_src_mac_str, |
| port_to_dst_mac_str, |
| port_to_src_ip, |
| port_to_src_ip_str, |
| Groups) = fill_mcast_pipeline_L3toL2( |
| self.controller, |
| logging, |
| ports, |
| is_ingress_tagged = False, |
| is_egress_tagged = False, |
| is_vlan_translated = False, |
| is_max_vlan = True |
| ) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| if out_port == in_port: |
| verify_no_packet( self, pkt, in_port ) |
| continue |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| class L3McastToL2UntagToTag( base_tests.SimpleDataPlane ): |
| """ |
| Mcast routing, in this test case the traffic is untagged. |
| 300 is used as vlan_id. The packet goes out |
| tagged. |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| assert (False) |
| return |
| ports = config[ "port_map" ].keys( ) |
| dst_ip_str = "224.0.0.1" |
| ( |
| port_to_in_vlan, |
| port_to_out_vlan, |
| port_to_src_mac_str, |
| port_to_dst_mac_str, |
| port_to_src_ip, |
| port_to_src_ip_str, |
| Groups) = fill_mcast_pipeline_L3toL2( |
| self.controller, |
| logging, |
| ports, |
| is_ingress_tagged = False, |
| is_egress_tagged = True, |
| is_vlan_translated = False, |
| is_max_vlan = False |
| ) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_out_vlan[in_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| if out_port == in_port: |
| verify_no_packet( self, pkt, in_port ) |
| continue |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class L3McastToL2TagToUntag( base_tests.SimpleDataPlane ): |
| """ |
| Mcast routing, in this test case the traffic is tagged. |
| 300 is used as vlan_id. The packet goes out |
| untagged. |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| assert (False) |
| return |
| ports = config[ "port_map" ].keys( ) |
| dst_ip_str = "224.0.0.1" |
| ( |
| port_to_in_vlan, |
| port_to_out_vlan, |
| port_to_src_mac_str, |
| port_to_dst_mac_str, |
| port_to_src_ip, |
| port_to_src_ip_str, |
| Groups) = fill_mcast_pipeline_L3toL2( |
| self.controller, |
| logging, |
| ports, |
| is_ingress_tagged = True, |
| is_egress_tagged = False, |
| is_vlan_translated = False, |
| is_max_vlan = False |
| ) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_in_vlan[in_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| if out_port == in_port: |
| verify_no_packet( self, pkt, in_port ) |
| continue |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| class L3McastToL2TagToTag( base_tests.SimpleDataPlane ): |
| """ |
| Mcast routing, in this test case the traffic is tagged. |
| 300 is used as vlan_id. The packet goes out tagged. |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| assert (False) |
| return |
| ports = config[ "port_map" ].keys( ) |
| dst_ip_str = "224.0.0.1" |
| ( |
| port_to_in_vlan, |
| port_to_out_vlan, |
| port_to_src_mac_str, |
| port_to_dst_mac_str, |
| port_to_src_ip, |
| port_to_src_ip_str, |
| Groups) = fill_mcast_pipeline_L3toL2( |
| self.controller, |
| logging, |
| ports, |
| is_ingress_tagged = True, |
| is_egress_tagged = True, |
| is_vlan_translated = False, |
| is_max_vlan = False |
| ) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_in_vlan[in_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_in_vlan[in_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| if out_port == in_port: |
| verify_no_packet( self, pkt, in_port ) |
| continue |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| class L3McastToL2TagToTagTranslated( base_tests.SimpleDataPlane ): |
| """ |
| Mcast routing, in this test case the traffic is tagged. |
| port+1 is used as ingress vlan_id. The packet goes out |
| tagged. 4094-port is used as egress vlan_id |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| assert (False) |
| return |
| ports = config[ "port_map" ].keys( ) |
| dst_ip_str = "224.0.0.1" |
| ( |
| port_to_in_vlan, |
| port_to_out_vlan, |
| port_to_src_mac_str, |
| port_to_dst_mac_str, |
| port_to_src_ip, |
| port_to_src_ip_str, |
| Groups) = fill_mcast_pipeline_L3toL2( |
| self.controller, |
| logging, |
| ports, |
| is_ingress_tagged = True, |
| is_egress_tagged = True, |
| is_vlan_translated = True, |
| is_max_vlan = False |
| ) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_in_vlan[in_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 100, |
| dl_vlan_enable = True, |
| vlan_vid = port_to_out_vlan[in_port], |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| if out_port == in_port: |
| verify_no_packet( self, pkt, in_port ) |
| continue |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class L3McastTrafficThenDrop( base_tests.SimpleDataPlane ): |
| # fails on alternate repeated runs |
| """ |
| Mcast routing, in this test case the traffic is untagged. |
| 4094 is used as internal vlan_id. We first install aa full pipeline, |
| test that it forwards properly then remove all rules on table 40 and |
| mcast groups and ensure traffic is dropped. Blackhole of mcast traffic |
| if no tree is programmed on the switch. |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| assert (False) |
| return |
| ports = config[ "port_map" ].keys( ) |
| dst_ip_str = "224.0.0.1" |
| ( |
| port_to_in_vlan, |
| port_to_out_vlan, |
| port_to_src_mac_str, |
| port_to_dst_mac_str, |
| port_to_src_ip, |
| port_to_src_ip_str, |
| Groups) = fill_mcast_pipeline_L3toL2( |
| self.controller, |
| logging, |
| ports, |
| is_ingress_tagged = False, |
| is_egress_tagged = False, |
| is_vlan_translated = False, |
| is_max_vlan = True |
| ) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| if out_port == in_port: |
| verify_no_packet( self, pkt, in_port ) |
| continue |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| delete_all_flows_one_table(ctrl=self.controller, logger=logging, table_id=40); |
| delete_all_groups(self.controller) |
| |
| for in_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| for out_port in ports: |
| |
| parsed_pkt = simple_udp_packet( |
| pktlen = 96, |
| eth_dst = port_to_dst_mac_str[in_port], |
| eth_src = port_to_src_mac_str[in_port], |
| ip_ttl = 64, |
| ip_src = port_to_src_ip_str[in_port], |
| ip_dst = dst_ip_str |
| ) |
| pkt = str( parsed_pkt ) |
| if out_port == in_port: |
| verify_no_packet( self, pkt, in_port ) |
| continue |
| verify_no_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class _MplsFwd( base_tests.SimpleDataPlane ): |
| """ Verify basic MPLS forwarding: Label switch router """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| dip = 0xc0a80001 |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| # Assigns unique hardcoded test_id to make sure tests don't overlap when writing rules |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # Shift MPLS label and VLAN ID by 16 to avoid reserved values |
| vlan_id = port + 16 |
| mpls_label = port + 16 |
| |
| # add l2 interface group |
| id = port |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| dst_mac[ 5 ] = port |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dst_mac, intf_src_mac, |
| vlan_id, id ) |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_SWAP_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=False, set_mpls_label=mpls_label, set_bos=1 ) |
| #ecmp_gid, ecmp_msg = add_mpls_forwarding_group( self.controller, |
| # subtype=OFDPA_MPLS_GROUP_SUBTYPE_ECMP, index=id, ref_gids=[mpls_label_gid] ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x8847, intf_src_mac, vlan_id, goto_table=24 ) |
| else: |
| add_termination_flow( self.controller, port, 0x8847, intf_src_mac, vlan_id, goto_table=24 ) |
| #add_mpls_flow( self.controller, ecmp_gid, port, goto_table=29 ) |
| if config["switch_type"] != 'xpliant': |
| add_mpls_flow( self.controller, mpls_label_gid, mpls_label, goto_table=29 ) |
| else: |
| xpliant_add_mpls_flow( self.controller, mpls_label_gid, mpls_label, goto_table=29 ) |
| dst_ip = dip + (vlan_id << 8) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| #Groups._put( ecmp_gid ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| ip_src = '192.168.%02d.1' % (in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| |
| # Shift MPLS label and VLAN ID by 16 to avoid reserved values |
| out_mpls_label = out_port + 16 |
| in_vlan_vid = in_port + 16 |
| out_vlan_vid = out_port + 16 |
| |
| ip_dst = '192.168.%02d.1' % (out_port) |
| label = (out_mpls_label, 0, 1, 32) |
| parsed_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(in_vlan_vid), |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=switch_mac, label=[ label ] ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % (out_port) |
| label = (out_mpls_label, 0, 1, 31) |
| exp_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(out_vlan_vid), |
| ip_src=ip_src, ip_dst=ip_dst, eth_src=switch_mac, eth_dst=mac_dst, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class _MplsTermination( base_tests.SimpleDataPlane ): |
| """ Verify MPLS VPN Termination at penultimate hop """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| dip = 0xc0a80001 |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| # Assigns unique hardcoded test_id to make sure tests don't overlap when writing rules |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # Shift MPLS label and VLAN ID by 16 to avoid reserved values |
| vlan_id = port + 16 |
| mpls_label = port + 16 |
| |
| # add l2 interface group |
| id, dst_mac[ 5 ] = port, port |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, port, vlan_id, True, False ) |
| # add L3 Unicast group |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| # add L3 ecmp group |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ l3_msg.group_id ] ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x8847, intf_src_mac, vlan_id, goto_table=24 ) |
| else: |
| add_termination_flow( self.controller, port, 0x8847, intf_src_mac, vlan_id, goto_table=24 ) |
| |
| if config["switch_type"] != 'xpliant': |
| add_mpls_flow( self.controller, ecmp_msg.group_id, mpls_label ) |
| else: |
| xpliant_add_mpls_flow( self.controller, ecmp_msg.group_id, mpls_label ) |
| |
| # add_mpls_flow(self.controller, label=port) |
| dst_ip = dip + (vlan_id << 8) |
| # add_unicast_routing_flow(self.controller, 0x0800, dst_ip, 0xffffff00, |
| # ecmp_msg.group_id, 1) |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| Groups._put( ecmp_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| ip_src = '192.168.%02d.1' % (in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| |
| # Shift MPLS label and VLAN ID by 16 to avoid reserved values |
| out_mpls_label = out_port + 16 |
| in_vlan_vid = in_port + 16 |
| out_vlan_vid = out_port + 16 |
| |
| ip_dst = '192.168.%02d.1' % (out_port) |
| label = (out_mpls_label, 0, 1, 32) |
| parsed_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(in_vlan_vid), |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=switch_mac, label=[ label ] ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % (out_port) |
| if config["switch_type"] != 'xpliant': |
| ip_ttl=31 |
| else: |
| ip_ttl=64 |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(out_vlan_vid), |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=ip_ttl, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| @disabled |
| class One_MplsTermination( base_tests.SimpleDataPlane ): |
| """ |
| Verify MPLS VPN Termination at penultimate hop in only one direction |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| dip = 0xc0a80001 |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| # Assigns unique hardcoded test_id to make sure tests don't overlap when writing rules |
| ports = config[ "port_map" ].keys( ) |
| inport = ports[0] |
| outport = ports[1] |
| |
| # Shift MPLS label and VLAN ID by 16 to avoid reserved values |
| invlan_id = inport + 16 |
| outvlan_id = outport + 16 |
| mpls_label = outport + 16 |
| |
| # add l2 interface group |
| id, dst_mac[ 5 ] = inport, outport |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, outport, outvlan_id, True, False ) |
| # add L3 Unicast group |
| l3_msg = add_l3_unicast_group( self.controller, outport, vlanid=outvlan_id, id=id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| # add L3 ecmp group |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ l3_msg.group_id ] ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, inport, 1, invlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add tmac flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x8847, intf_src_mac, invlan_id, goto_table=24 ) |
| else: |
| add_termination_flow( self.controller, inport, 0x8847, intf_src_mac, invlan_id, goto_table=24 ) |
| # add mpls termination flow |
| add_mpls_flow( self.controller, ecmp_msg.group_id, mpls_label, send_barrier=True ) |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| Groups._put( ecmp_msg.group_id ) |
| |
| time.sleep(0.1) |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| ip_src = '192.168.%02d.1' % (inport) |
| # Shift MPLS label and VLAN ID by 16 to avoid reserved values |
| out_mpls_label = outport + 16 |
| in_vlan_vid = inport + 16 |
| out_vlan_vid = outport + 16 |
| |
| ip_dst = '192.168.%02d.1' % (outport) |
| label = (out_mpls_label, 0, 1, 32) |
| parsed_pkt = mpls_packet( pktlen=104, dl_vlan_enable=True, vlan_vid=(in_vlan_vid), |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=switch_mac, label=[ label ] ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( inport, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % (outport) |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(out_vlan_vid), |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=31, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, outport ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class _24UcastTagged( base_tests.SimpleDataPlane ): |
| """ Verify /24 IP forwarding to L3 Interface """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| test_id = 26 |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| vlan_id = port + test_id |
| l2gid, msg = add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id, |
| is_tagged=True, send_barrier=False ) |
| dst_mac[ 5 ] = vlan_id |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=vlan_id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| # def add_unicast_routing_flow(ctrl, eth_type, dst_ip, mask, action_group_id, vrf=0, send_ctrl=False, send_barrier=False, priority = 1): |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffff00, l3_msg.group_id ) |
| Groups.put( l2gid ) |
| Groups.put( l3_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % (test_id + in_port) |
| ip_src = '192.168.%02d.1' % (test_id + in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (test_id + out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, |
| vlan_vid=(test_id + in_port), eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, |
| ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % (test_id + out_port) |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, |
| vlan_vid=(test_id + out_port), eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| class _24UcastRouteBlackHole( base_tests.SimpleDataPlane ): |
| """ Verify black-holing of unicast routes, feature present only from OFDPA Premium 1.1.3.0""" |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| test_id = 27 |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| vlan_id = port + test_id |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| #dst ip = 10.0.0.0/8 |
| dst_ip = 0x0a000000 |
| add_unicast_blackhole_flow(self.controller, 0x0800, dst_ip, 0xffffff00 ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % (test_id + in_port) |
| ip_src = '192.168.%02d.1' % (test_id + in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (test_id + out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, |
| vlan_vid=(test_id + in_port), eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, |
| ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % (test_id + out_port) |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, |
| vlan_vid=(test_id + out_port), eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_no_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class _0Ucast( base_tests.SimpleDataPlane ): |
| """ Verify default gateway IP forwarding to L3 Interface ( /0 rule ) """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| for port in ports: |
| # add l2 interface group |
| vlan_id = port |
| l2gid, msg = add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id + 1, |
| is_tagged=True, send_barrier=False ) |
| dst_mac[ 5 ] = vlan_id |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id + 1, id=vlan_id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffffff, l3_msg.group_id, priority=2 ) |
| Groups.put( l2gid ) |
| Groups.put( l3_msg.group_id ) |
| l3_gid = encode_l3_unicast_group_id( ports[ 0 ] ) |
| dst_ip = 0x0 |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0x0, l3_gid ) |
| do_barrier( self.controller ) |
| |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| for in_port in ports: |
| mac_src = '00:00:00:22:22:%02X' % (in_port) |
| ip_src = '192.168.%02d.1' % (in_port) |
| for out_port in ports: |
| if in_port == out_port: |
| continue |
| ip_dst = '192.168.%02d.1' % (out_port) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(in_port), |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % (out_port) |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=(out_port + 1), |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| ip_dst = '1.168.%02d.1' % ports[ 0 ] |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=in_port, |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| # build expect packet |
| mac_dst = '00:00:00:22:22:%02X' % ports[ 0 ] |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=ports[ 0 ] + 1, |
| ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst, eth_dst=mac_dst, eth_src=switch_mac ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, ports[ 0 ] ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class Unfiltered( base_tests.SimpleDataPlane ): |
| """ |
| Attempt to add an unfiltered group: [ATTENTION] this doesn't verify addition |
| """ |
| |
| def runTest( self ): |
| try: |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| vlan_id = 1; |
| for port in ports: |
| add_l2_unfiltered_group( self.controller, [ port ], False ) |
| do_barrier( self.controller ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class L3McastToVPN( base_tests.SimpleDataPlane ): |
| """ |
| Mcast routing and VPN initiation |
| """ |
| |
| def runTest( self ): |
| """ |
| port1 (vlan 1)-> port 2 (vlan 2) |
| """ |
| try: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| if len( config[ "port_map" ] ) < 3: |
| logging.info( "Port count less than 3, can't run this case" ) |
| assert (False) |
| return |
| |
| vlan_id = 1 |
| port2_out_vlan = 2 |
| port3_out_vlan = 3 |
| in_vlan = 1 # macast group vid shall use input vlan diffe from l3 interface use output vlan |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| intf_src_mac_str = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| dst_mac = [ 0x01, 0x00, 0x5e, 0x01, 0x01, 0x01 ] |
| dst_mac_str = ':'.join( [ '%02X' % x for x in dst_mac ] ) |
| port1_mac = [ 0x00, 0x11, 0x11, 0x11, 0x11, 0x11 ] |
| port1_mac_str = ':'.join( [ '%02X' % x for x in port1_mac ] ) |
| src_ip = 0xc0a80101 |
| src_ip_str = "192.168.1.1" |
| dst_ip = 0xe0010101 |
| dst_ip_str = "224.1.1.1" |
| |
| port1 = config[ "port_map" ].keys( )[ 0 ] |
| port2 = config[ "port_map" ].keys( )[ 1 ] |
| # port3=config["port_map"].keys()[2] |
| |
| # add l2 interface group |
| for port in config[ "port_map" ].keys( ): |
| add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id, is_tagged=False, |
| send_barrier=False ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| vlan_id += 1 |
| |
| # add termination flow |
| add_termination_flow( self.controller, port1, 0x0800, [ 0x01, 0x00, 0x5e, 0x00, 0x00, 0x00 ], |
| vlan_id ) |
| |
| # add MPLS interface group |
| l2_gid = encode_l2_interface_group_id( port2_out_vlan, port2 ) |
| mpls_gid2, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dst_mac, intf_src_mac, |
| port2_out_vlan, port2 ) |
| # l2_gid3 = encode_l2_interface_group_id(port3_out_vlan, port3) |
| # mpls_gid3, mpls_msg = add_mpls_intf_group(self.controller, l2_gid3, dst_mac, intf_src_mac, port3_out_vlan, port3) |
| # add L3VPN groups |
| mpls_label_gid2, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=(0x20000 + port2), ref_gid=mpls_gid2, |
| push_mpls_header=True, set_mpls_label=port2, set_bos=1, cpy_ttl_outward=True ) |
| # mpls_label_gid3, mpls_label_msg = add_mpls_label_group(self.controller, subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, |
| # index=(0x10000+port3), ref_gid= mpls_gid3, push_mpls_header=True, set_mpls_label=port3, set_bos=1, cpy_ttl_outward=True) |
| |
| mcat_group_msg = add_l3_mcast_group( self.controller, in_vlan, 2, [ mpls_label_gid2 ] ) |
| add_mcast4_routing_flow( self.controller, in_vlan, src_ip, 0, dst_ip, mcat_group_msg.group_id ) |
| |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=1, eth_dst=dst_mac_str, |
| eth_src=port1_mac_str, ip_ttl=64, ip_src=src_ip_str, ip_dst=dst_ip_str ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( port1, pkt ) |
| label = (12, 0, 1, 63) |
| exp_pkt = mpls_packet( pktlen=100, eth_dst=dst_mac_str, eth_src=intf_src_mac_str, ip_ttl=64, |
| ip_src=src_ip_str, label=[ label ], ip_dst=dst_ip_str ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, port2 ) |
| # verify_packet(self, pkt, port3) |
| verify_no_other_packets( self ) |
| delete_all_groups( self.controller ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class PacketInSrcMacMiss( base_tests.SimpleDataPlane ): |
| """ |
| Test packet in function on a src-mac miss |
| Send a packet to each dataplane port and verify that a packet |
| in message is received from the controller for each |
| #todo verify you stop receiving after adding rule |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| |
| Groups = Queue.LifoQueue( ) |
| for port in ports: |
| L2gid, l2msg = add_one_l2_interface_group( self.controller, port, 1, True, False ) |
| add_one_vlan_table_flow( self.controller, port, 1, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| Groups.put( L2gid ) |
| parsed_vlan_pkt = simple_tcp_packet( pktlen=104, vlan_vid=0x1001, dl_vlan_enable=True ) |
| vlan_pkt = str( parsed_vlan_pkt ) |
| for of_port in config[ "port_map" ].keys( ): |
| logging.info( "PacketInMiss test, port %d", of_port ) |
| self.dataplane.send( of_port, vlan_pkt ) |
| verify_packet_in( self, vlan_pkt, of_port, ofp.OFPR_NO_MATCH ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| |
| class EcmpGroupMod( base_tests.SimpleDataPlane ): |
| """ |
| Verify referenced group can be modified by adding or removing buckets |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| dip = 0xc0a80001 |
| # Hashes Test Name and uses it as id for installing unique groups |
| ports = config[ "port_map" ].keys( ) |
| ecmp = [ ] |
| dst_ips = [] |
| # add flows for all ports but include only the egress switchport (connected to ports[1]) |
| # in the ecmp group |
| for port in ports: |
| vlan_id = port |
| id = port |
| # add l2 interface group |
| l2_gid, msg = add_one_l2_interface_group( self.controller, port, vlan_id=vlan_id, |
| is_tagged=True, send_barrier=False ) |
| dst_mac[ 5 ] = vlan_id |
| l3_msg = add_l3_unicast_group( self.controller, port, vlanid=vlan_id, id=id, |
| src_mac=intf_src_mac, dst_mac=dst_mac ) |
| if port == ports[1]: |
| ecmp += [ l3_msg.group_id ] |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, ports[ 0 ], [ l3_msg.group_id ] ) |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, intf_src_mac, vlan_id ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, intf_src_mac, vlan_id ) |
| # add unicast routing flow |
| dst_ip = dip + (vlan_id << 8) |
| dst_ips += [dst_ip] |
| Groups._put( ecmp_msg.group_id ) |
| mod_l3_ecmp_group( self.controller, ports[ 0 ], ecmp ) |
| for dst_ip in dst_ips: |
| add_unicast_routing_flow( self.controller, 0x0800, dst_ip, 0xffffff00, ecmp_msg.group_id ) |
| do_barrier(self.controller) |
| time.sleep(0.1) |
| # first part of the test: send packet from ingress switchport and expect it at egress switchport |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| parsed_pkt = exp_pkt = 0 |
| in_port = ports[0] |
| out_port = ports[1] |
| logging.info("\nSending packet to port: " + str(in_port) + ", expected egress on port: " + str(out_port)) |
| mac_src = '00:00:00:22:22:%02X' % ports[ 0 ] |
| ip_src = '192.168.%02d.%02d' % (ports[ 0 ], 1) |
| ip_dst = '192.168.%02d.%02d' % (ports[ 1 ], 1) |
| tcp = out_port if out_port == 24 else 25 |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=ports[ 0 ], |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, |
| ip_dst=ip_dst, tcp_dport=tcp ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( ports[ 0 ], pkt ) |
| # build expected packet at egress switchport |
| mac_dst = '00:00:00:22:22:%02X' % out_port |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=out_port, |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, |
| ip_dst=ip_dst, tcp_dport=tcp ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| # second part of the test - edit the ecmp group to remove the orginal egress switchport |
| # and instead add the ingress switchport. Send packet from ingress switchport, and expect |
| # it back on the ingress switchport |
| l3_gid = encode_l3_unicast_group_id( ports[ 0 ] ) |
| mod_l3_ecmp_group( self.controller, ports[ 0 ], [ l3_gid ] ) |
| time.sleep(0.1) |
| logging.info("Sending packet to port: " + str(ports[0]) + ", expected egress on port: " + str(ports[0])) |
| mac_src = '00:00:00:22:22:%02X' % ports[ 0 ] |
| ip_src = '192.168.%02d.%02d' % (ports[ 0 ], 1) |
| ip_dst = '192.168.%02d.%02d' % (ports[ 1 ], 1) |
| tcp = port if port == 24 else 25 |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=ports[ 0 ], |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, |
| ip_dst=ip_dst,tcp_dport=tcp ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( ports[ 0 ], pkt ) |
| # build expected packet |
| mac_dst = '00:00:00:22:22:%02X' % ports[ 0 ] |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=ports[ 0 ], |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, |
| ip_dst=ip_dst,tcp_dport=tcp ) |
| # Expects packet on the input port |
| if config["switch_type"] != 'xpliant': |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, ports[ 0 ] ) |
| verify_no_other_packets( self ) |
| |
| # third part of the test - edit the group to completely remove bucket. Packet sent |
| # should be dropped by the switch |
| mod_l3_ecmp_group( self.controller, ports[ 0 ], [ ] ) |
| time.sleep(0.1) |
| logging.info("Sending packet to port: " + str(ports[0]) + ", expected drop") |
| mac_src = '00:00:00:22:22:%02X' % ports[ 0 ] |
| ip_src = '192.168.%02d.%02d' % (ports[ 0 ], 1) |
| ip_dst = '192.168.%02d.%02d' % (ports[ 1 ], 1) |
| tcp = port if port == 24 else 25 |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=ports[ 0 ], |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, |
| ip_dst=ip_dst,tcp_dport=tcp ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( ports[ 0 ], pkt ) |
| verify_no_other_packets( self ) |
| |
| # final part of the test - edit the empty group to add back the bucket for the |
| # original egress port, and verify packet is received on egress switch port |
| l3_gid = encode_l3_unicast_group_id( ports[ 1 ] ) |
| mod_l3_ecmp_group( self.controller, ports[ 0 ], [ l3_gid ] ) |
| do_barrier(self.controller) |
| in_port = ports[0] |
| out_port = ports[1] |
| logging.info("Sending packet to port: " + str(in_port) + ", expected egress on port: " + str(out_port)) |
| mac_src = '00:00:00:22:22:%02X' % ports[ 0 ] |
| ip_src = '192.168.%02d.%02d' % (ports[ 0 ], 1) |
| ip_dst = '192.168.%02d.%02d' % (ports[ 1 ], 1) |
| tcp = out_port if out_port == 24 else 25 |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=ports[ 0 ], |
| eth_dst=switch_mac, eth_src=mac_src, ip_ttl=64, ip_src=ip_src, |
| ip_dst=ip_dst, tcp_dport=tcp ) |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( ports[ 0 ], pkt ) |
| # build expected packet at egress switchport |
| mac_dst = '00:00:00:22:22:%02X' % out_port |
| exp_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=out_port, |
| eth_dst=mac_dst, eth_src=switch_mac, ip_ttl=63, ip_src=ip_src, |
| ip_dst=ip_dst, tcp_dport=tcp ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class Untagged( base_tests.SimpleDataPlane ): |
| """ |
| Verify VLAN filtering table does not require OFPVID_PRESENT bit to be 0. |
| This should be fixed in OFDPA 2.0 GA and above, the test fails with |
| previous versions of the OFDPA. |
| |
| Two rules are necessary in VLAN table (10): |
| 1) Assignment: match 0x0000/(no mask), set_vlan_vid 0x100A, goto 20 |
| 2) Filtering: match 0x100A/0x1FFF, goto 20 |
| |
| In this test case vlan_id = (MAX_INTERNAL_VLAN - port_no). |
| The remaining part of the test is based on the use of the bridging table |
| """ |
| |
| MAX_INTERNAL_VLAN = 4094 |
| |
| def runTest( self ): |
| groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| ports = sorted( config[ "port_map" ].keys( ) ) |
| for port in ports: |
| vlan_id = Untagged.MAX_INTERNAL_VLAN - port |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| add_one_vlan_table_flow( self.controller, port, 1, vlan_id, flag=VLAN_TABLE_FLAG_ONLY_UNTAG ) |
| for other_port in ports: |
| if other_port == port: |
| continue |
| L2gid, l2msg = add_one_l2_interface_group( self.controller, other_port, vlan_id, False, False ) |
| groups.put( L2gid ) |
| add_bridge_flow( self.controller, [ 0x00, 0x12, 0x34, 0x56, 0x78, other_port ], vlan_id, L2gid, True ) |
| |
| do_barrier( self.controller ) |
| |
| for out_port in ports: |
| # change dest based on port number |
| mac_dst = '00:12:34:56:78:%02X' % out_port |
| for in_port in ports: |
| if in_port == out_port: |
| continue |
| pkt = str( simple_tcp_packet( eth_dst=mac_dst ) ) |
| self.dataplane.send( in_port, pkt ) |
| for ofport in ports: |
| if ofport in [ out_port ]: |
| verify_packet( self, pkt, ofport ) |
| else: |
| verify_no_packet( self, pkt, ofport ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, groups ) |
| delete_all_groups( self.controller ) |
| |
| class MPLSSwapTest( base_tests.SimpleDataPlane ): |
| """ |
| MPLS switching with the same label used. |
| Used for interconnecting spines between different fabrics where |
| the label should not be popped, but swapepd with the same label. |
| """ |
| |
| def runTest( self ): |
| try: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 3, can't run this case" ) |
| assert (False) |
| return |
| |
| input_src_mac = [ 0x00, 0x00, 0x5e, 0x01, 0x01, 0x01 ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x5e, 0x01, 0x01, 0x02 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x5e, 0x01, 0x01, 0x03 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| mpls_label = 1000 |
| |
| src_ip = 0xc0a80101 |
| src_ip_str = "192.168.1.1" |
| dst_ip = 0xe0010101 |
| dst_ip_str = "224.1.1.1" |
| |
| src_port = config[ "port_map" ].keys( )[ 0 ] |
| dst_port = config[ "port_map" ].keys( )[ 1 ] |
| |
| out_vlan = 4094 |
| |
| add_one_l2_interface_group( self.controller, dst_port, vlan_id=out_vlan, is_tagged=False, |
| send_barrier=True ) |
| |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, src_port, out_vlan_id=out_vlan, vlan_id=out_vlan, flag=VLAN_TABLE_FLAG_ONLY_TAG ) |
| add_one_vlan_table_flow( self.controller, src_port, out_vlan_id=out_vlan, vlan_id=out_vlan, flag=VLAN_TABLE_FLAG_ONLY_UNTAG ) |
| |
| # add termination flow |
| |
| if config["switch_type"] == "qmx": |
| logging.debug("MPLSSwitching : Adding flow for qmx, without input port") |
| add_termination_flow( self.controller, 0, eth_type=0x08847, dst_mac=input_dst_mac, vlanid=out_vlan, goto_table=23) |
| else: |
| add_termination_flow( self.controller, in_port=src_port, |
| eth_type=0x8847, dst_mac=input_dst_mac, vlanid=out_vlan, goto_table=23) |
| |
| # add groups that will be used now |
| l2_gid = encode_l2_interface_group_id( out_vlan, dst_port) |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, |
| output_dst_mac, input_dst_mac, |
| out_vlan, dst_port, send_barrier=True) |
| index = 60 |
| mpls_swap_gid, mpls_swap_msg = add_mpls_swap_label_group( self.controller, mpls_gid, |
| 5, index, mpls_label) |
| |
| # add flow to mpls table |
| add_mpls_flow_swap( self.controller, mpls_swap_gid, mpls_label, 0x8847, 1, send_barrier=True) |
| |
| # we generate the packet which carries a single label |
| label = (mpls_label, 0, 1, 63) |
| parsed_pkt = mpls_packet( |
| pktlen=104, |
| label=[label], |
| eth_src=input_src_mac_str, |
| eth_dst=input_dst_mac_str, |
| ) |
| pkt = str( parsed_pkt ) |
| |
| self.dataplane.send( src_port, pkt ) |
| |
| label = (mpls_label, 0, 1, 62) |
| parsed_pkt = mpls_packet( |
| pktlen=104, |
| label=[label], |
| eth_src=input_dst_mac_str, |
| eth_dst=output_dst_mac_str, |
| ) |
| pkt = str( parsed_pkt ) |
| |
| verify_packet( self, pkt, dst_port ) |
| verify_no_packet( self, pkt, src_port ) |
| verify_no_other_packets( self ) |
| |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class DoubleToUntagged( base_tests.SimpleDataPlane ): |
| """ |
| Verify MPLS IP VPN Initiation from /24 rule using ECMP |
| where we receive double tagged packets and output untagged |
| packets. |
| |
| Each double tagged packet represents a subscriber where Outer tag is pon |
| and inner tag is the subrscriber tag. |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = 66 |
| outer_vlan = 77 |
| id = 10 |
| mpls_label = 152 |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| # add l2 interface group |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, out_port, inner_vlan, False, True ) |
| |
| # add MPLS interface group |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, output_dst_mac, input_dst_mac, |
| inner_vlan, id ) |
| |
| # add MPLS L3 VPN group |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=True, set_mpls_label=mpls_label, set_bos=1, set_ttl=32 ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ mpls_label_gid ] ) |
| |
| do_barrier( self.controller ) |
| |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, outer_vlan, vrf=0, |
| flag=VLAN_TABLE_FLAG_ONLY_STACKED ) |
| |
| add_one_vlan_1_table_flow( self.controller, port, outer_vlan_id=outer_vlan, inner_vlan_id=inner_vlan, |
| flag=VLAN_TABLE_FLAG_ONLY_UNTAG ) |
| |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, input_dst_mac, inner_vlan ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, input_dst_mac, inner_vlan ) |
| |
| # add_unicast_routing_flow(self.controller, 0x0800, dst_ip, 0, mpls_label_gid, vrf=2) |
| add_unicast_routing_flow( self.controller, 0x0800, dip, 0xffffff00, ecmp_msg.group_id, |
| vrf=0 ) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| Groups._put( ecmp_msg.group_id ) |
| |
| do_barrier( self.controller ) |
| |
| ip_src = '192.168.5.5' |
| ip_dst = '192.168.0.5' |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=inner_vlan, outer_vlan=outer_vlan, |
| eth_dst=input_dst_mac_str, eth_src=input_src_mac_str, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| |
| # print("Expected %s" % format_packet(pkt)) |
| |
| self.dataplane.send( port, pkt ) |
| |
| # build expect packet |
| label = (mpls_label, 0, 1, 32) |
| exp_pkt = mpls_packet( pktlen=96, dl_vlan_enable=False, ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=output_dst_mac_str, eth_src=input_dst_mac_str, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class DoubleToUntaggedMultipleSubscribers( base_tests.SimpleDataPlane ): |
| """ |
| Verify MPLS IP VPN Initiation from /24 rule using ECMP |
| where we receive double tagged packets and output untagged |
| packets. |
| |
| Each double tagged packet represents a subscriber where Outer tag is pon |
| and inner tag is the subrscriber tag. |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| # each entry represents a subscriber [id, ip in hex, inner_vlan, outer_vlan, ip in dot form] |
| subscriber_info = [ [10, 0xc0a80001, 10, 100, "192.168.0.1"], |
| [20, 0xc0a80002, 10, 101, "192.168.0.2"], |
| [30, 0xc0a80003, 11, 100, "192.168.0.3"], |
| [40, 0xc0a80004, 11, 101, "192.168.0.4"]] |
| |
| print("") |
| |
| for sub_info in subscriber_info: |
| |
| print("Initializing rules for subscriber with id {0}".format(sub_info[0])) |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, sub_info[0] ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = sub_info[1] |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = sub_info[2] |
| outer_vlan = sub_info[3] |
| id = 10 |
| mpls_label = 152 |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| # add l2 interface group |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, out_port, inner_vlan, False, True ) |
| |
| # add MPLS interface group |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, output_dst_mac, input_dst_mac, |
| inner_vlan, id ) |
| |
| # add MPLS L3 VPN group |
| mpls_label_gid, mpls_label_msg = add_mpls_label_group( self.controller, |
| subtype=OFDPA_MPLS_GROUP_SUBTYPE_L3_VPN_LABEL, index=id, ref_gid=mpls_gid, |
| push_mpls_header=True, set_mpls_label=mpls_label, set_bos=1, set_ttl=32 ) |
| ecmp_msg = add_l3_ecmp_group( self.controller, id, [ mpls_label_gid ] ) |
| |
| do_barrier( self.controller ) |
| |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, outer_vlan, vrf=0, |
| flag=VLAN_TABLE_FLAG_ONLY_STACKED ) |
| |
| add_one_vlan_1_table_flow( self.controller, port, outer_vlan_id=outer_vlan, inner_vlan_id=inner_vlan, |
| flag=VLAN_TABLE_FLAG_ONLY_UNTAG ) |
| |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, input_dst_mac, inner_vlan ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, input_dst_mac, inner_vlan ) |
| |
| # add_unicast_routing_flow(self.controller, 0x0800, dst_ip, 0, mpls_label_gid, vrf=2) |
| add_unicast_routing_flow( self.controller, 0x0800, dip, 0xffffff00, ecmp_msg.group_id, |
| vrf=0 ) |
| Groups._put( l2_gid ) |
| Groups._put( mpls_gid ) |
| Groups._put( mpls_label_gid ) |
| Groups._put( ecmp_msg.group_id ) |
| |
| do_barrier( self.controller ) |
| |
| for sub_info in subscriber_info: |
| |
| print("Sending packet for subscriber with id {0}".format(sub_info[0])) |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, sub_info[0] ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = sub_info[1] |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = sub_info[2] |
| outer_vlan = sub_info[3] |
| id = 10 |
| mpls_label = 152 |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| ip_src = sub_info[4] |
| ip_dst = '192.168.0.{}'.format(sub_info[0]) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=True, vlan_vid=inner_vlan, outer_vlan=outer_vlan, |
| eth_dst=input_dst_mac_str, eth_src=input_src_mac_str, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| |
| # print("Sent %s" % format_packet(pkt)) |
| |
| self.dataplane.send( port, pkt ) |
| |
| # build expect packet |
| label = (mpls_label, 0, 1, 32) |
| exp_pkt = mpls_packet( pktlen=96, dl_vlan_enable=False, ip_ttl=63, |
| ip_src=ip_src, ip_dst=ip_dst, eth_dst=output_dst_mac_str, eth_src=input_dst_mac_str, |
| label=[ label ] ) |
| pkt = str( exp_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| @disabled |
| class UntaggedToDouble ( base_tests.SimpleDataPlane ): |
| """ |
| Verify google senario where we need to go from |
| an untagged packet to a double tagged packet. |
| |
| This is used for a single subscriber. |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = 66 |
| outer_vlan = 77 |
| id = 10 |
| mpls_label = 152 |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| # add l2 unfiltered interface group |
| l2_gid, l2_msg = add_one_l2_unfiltered_group( self.controller, out_port, True) |
| |
| l3_msg = add_l3_unicast_group( self.controller, out_port, vlanid=4094, id=id, |
| src_mac=input_dst_mac, dst_mac=output_dst_mac, gid=l2_gid) |
| |
| do_barrier( self.controller ) |
| |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, 4094, |
| flag=VLAN_TABLE_FLAG_ONLY_BOTH ) |
| |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, input_dst_mac, 4094 ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, input_dst_mac, 4094 ) |
| |
| add_unicast_routing_flow( self.controller, 0x0800, dip, 0xffffffff, l3_msg.group_id, |
| vrf=0 ) |
| |
| add_one_egress_vlan_table_flow( self.controller, out_port, 4094 , inner_vlan, outer_vlan) |
| |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| |
| do_barrier( self.controller ) |
| |
| ip_src = '192.168.5.5' |
| ip_dst = '192.168.0.1' |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=False, |
| eth_dst=input_dst_mac_str, eth_src=input_src_mac_str, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| |
| # print("Input Packet %s" % format_packet(pkt)) |
| |
| self.dataplane.send( port, pkt ) |
| |
| # build expect packet |
| exp_pkt = simple_tcp_packet( pktlen=108, dl_vlan_enable=True, vlan_vid=inner_vlan, outer_vlan=outer_vlan, |
| eth_dst=output_dst_mac_str, eth_src=input_dst_mac_str, ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| |
| # print("Expected Packet %s" % format_packet(pkt)) |
| |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class UntaggedToDoubleMultipleSubscribers ( base_tests.SimpleDataPlane ): |
| """ |
| Verify google senario where we need to go from |
| an untagged packet to a double tagged packet. |
| |
| This is used for multiple subscribers. |
| |
| However, this solution does not scale, since we assign an internal vlan to each subscriber |
| used in L3 Unicast Group in order to differentiate between them in the Egress Vlan Table. |
| """ |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| # each entry represents a subscriber [id, ip in hex, inner_vlan, outer_vlan, ip in dot form, internal vlan] |
| subscriber_info = [[1, 0xc0a80001, 10, 100, "192.168.0.1", 4000], |
| [2, 0xc0a80002, 10, 101, "192.168.0.2", 4001], |
| [3, 0xc0a80003, 11, 100, "192.168.0.3", 4002], |
| [4, 0xc0a80004, 11, 101, "192.168.0.4", 4003]] |
| |
| print("") |
| |
| for sub_info in subscriber_info: |
| |
| print("Initializing rules for subscriber with id {0}".format(sub_info[0])) |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, sub_info[0] ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = sub_info[1] |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = sub_info[2] |
| outer_vlan = sub_info[3] |
| internal_vlan = sub_info[5] |
| id = sub_info[0] + 10 |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| # add l2 unfiltered interface group |
| l2_gid, l2_msg = add_one_l2_unfiltered_group( self.controller, out_port, True) |
| |
| l3_msg = add_l3_unicast_group( self.controller, out_port, vlanid=internal_vlan, id=id, |
| src_mac=input_dst_mac, dst_mac=output_dst_mac, gid=l2_gid) |
| |
| do_barrier( self.controller ) |
| |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, 4094, |
| flag=VLAN_TABLE_FLAG_ONLY_BOTH ) |
| |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, input_dst_mac, 4094 ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, input_dst_mac, 4094 ) |
| |
| add_unicast_routing_flow( self.controller, 0x0800, dip, 0xffffffff, l3_msg.group_id, |
| vrf=0 ) |
| |
| add_one_egress_vlan_table_flow( self.controller, out_port, internal_vlan, inner_vlan, outer_vlan) |
| |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| do_barrier( self.controller ) |
| |
| for sub_info in subscriber_info: |
| |
| print("Sending packet for subscriber with id {0}".format(sub_info[0])) |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, sub_info[0] ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = sub_info[1] |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = sub_info[2] |
| outer_vlan = sub_info[3] |
| internal_vlan = sub_info[5] |
| |
| id = sub_info[0] + 10 |
| ip_src = '192.168.5.5' |
| ip_dst = '192.168.0.{}'.format(sub_info[0]) |
| parsed_pkt = simple_tcp_packet( pktlen=100, dl_vlan_enable=False, |
| eth_dst=input_dst_mac_str, eth_src=input_src_mac_str, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| |
| # print("Input Packet %s" % format_packet(pkt)) |
| |
| self.dataplane.send( port, pkt ) |
| |
| # build expect packet |
| exp_pkt = simple_tcp_packet( pktlen=108, dl_vlan_enable=True, vlan_vid=inner_vlan, outer_vlan=outer_vlan, |
| eth_dst=output_dst_mac_str, eth_src=input_dst_mac_str, ip_ttl=63, ip_src=ip_src, ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| |
| # print("Expected Packet %s" % format_packet(pkt)) |
| |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class UntaggedToDoubleChangeEthertype ( base_tests.SimpleDataPlane ): |
| |
| def runTest( self ): |
| Groups = Queue.LifoQueue() |
| |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "port count less than 2, can't run this case" ) |
| return |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = 0xc0a80001 |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = 66 |
| outer_vlan = 77 |
| id = 10 |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| # add l2 unfiltered interface group |
| l2_gid, l2_msg = add_one_l2_unfiltered_group( self.controller, out_port, True) |
| |
| l3_msg = add_l3_unicast_group( self.controller, out_port, vlanid=4094, id=id, |
| src_mac=input_dst_mac, dst_mac=output_dst_mac, gid=l2_gid) |
| |
| do_barrier( self.controller ) |
| |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, 1, 4094, |
| flag=VLAN_TABLE_FLAG_ONLY_BOTH ) |
| |
| # add termination flow |
| if config["switch_type"] == "qmx": |
| add_termination_flow( self.controller, 0, 0x0800, input_dst_mac, 4094 ) |
| else: |
| add_termination_flow( self.controller, port, 0x0800, input_dst_mac, 4094 ) |
| |
| add_unicast_routing_flow( self.controller, 0x0800, dip, 0xffffffff, l3_msg.group_id, |
| vrf=0 ) |
| |
| add_one_egress_vlan_table_flow( self.controller, out_port, 4094 , inner_vlan, outer_vlan) |
| |
| Groups._put( l2_gid ) |
| Groups._put( l3_msg.group_id ) |
| |
| # add vlan flow table |
| add_one_egress_vlan_tpid_table_flow( self.controller, out_port, outer_vlan+0x1000 ) |
| do_barrier( self.controller ) |
| |
| ip_src = '192.168.5.5' |
| ip_dst = '192.168.0.1' |
| parsed_pkt = simple_tcp_packet( pktlen=100, |
| dl_vlan_enable=False, |
| eth_dst=input_dst_mac_str, |
| eth_src=input_src_mac_str, |
| ip_ttl=64, |
| ip_src=ip_src, |
| ip_dst=ip_dst ) |
| pkt = str( parsed_pkt ) |
| |
| print("Input Packet %s" % format_packet(pkt)) |
| |
| self.dataplane.send( port, pkt ) |
| |
| # build expect packet |
| exp_pkt = simple_tcp_packet_two_vlan( pktlen=108, |
| out_dl_vlan_enable=True, |
| out_vlan_vid=outer_vlan, |
| out_vlan_tpid=0x88a8, |
| in_dl_vlan_enable=True, |
| in_vlan_vid=inner_vlan, |
| eth_dst=output_dst_mac_str, |
| eth_src=input_dst_mac_str, |
| ip_ttl=63, |
| ip_src=ip_src, |
| ip_dst=ip_dst ) |
| pkt = str( exp_pkt ) |
| |
| print("Expected Packet %s" % format_packet(pkt)) |
| |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| @disabled |
| class _MplsFwdInterfaceProblem_PW( base_tests.SimpleDataPlane ): |
| """ |
| Reproduces the pseudowire bug with the wrongly set destination mac address. |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 1: |
| logging.info( "Port count less than 1, can't run this case" ) |
| assert (False) |
| return |
| |
| macs_to_test = [( [ 0x00, 0x00, 0x00, 0x11, 0x11, 0x00 ], [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ], '00:00:00:11:11:00', '00:00:00:22:22:00'), |
| ( [ 0x00, 0x00, 0x00, 0x11, 0x22, 0x00 ], [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ], '00:00:00:11:22:00', '00:00:00:33:33:00'), |
| ( [ 0x00, 0x00, 0x00, 0x11, 0x33, 0x00 ], [ 0x00, 0x00, 0x00, 0x44, 0x44, 0x00 ], '00:00:00:11:33:00', '00:00:00:44:44:00'), |
| ( [ 0x00, 0x00, 0x00, 0x11, 0x44, 0x00 ], [ 0x00, 0x00, 0x00, 0x55, 0x55, 0x00 ], '00:00:00:11:44:00', '00:00:00:55:55:00'), |
| ( [ 0x00, 0x00, 0x00, 0x11, 0x55, 0x00 ], [ 0x00, 0x00, 0x00, 0x66, 0x66, 0x00 ], '00:00:00:11:55:00', '00:00:00:66:66:00')] |
| |
| for dummy_dst_mac, dst_mac, mac_dst_dummy, mac_dst in macs_to_test: |
| |
| dip = 0xc0a80001 |
| intf_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc ] |
| |
| out_port = 12 |
| port = 24 |
| # Shift MPLS label and VLAN ID by 16 to avoid reserved values |
| vlan_id = port + 16 |
| mpls_label = port + 16 |
| |
| in_port = 24 |
| ip_src = '192.168.%02d.1' % (in_port) |
| |
| # create dummy groups |
| id = port |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, out_port, vlan_id, False, False) |
| mpls_gid, mpls_msg = add_mpls_intf_group( self.controller, l2_gid, dummy_dst_mac, intf_src_mac, |
| vlan_id, id, send_barrier=True) |
| do_barrier( self.controller ) |
| |
| # PW Case. |
| raw_input("Press anything to move on with pseudowire rules, after that you should see the updated groups in the switch.") |
| logging.info("Installing entries for pseudowire!") |
| switch_mac = ':'.join( [ '%02X' % x for x in intf_src_mac ] ) |
| mpls_label = 100 |
| mpls_label_SR2 = 100 + 10 |
| mpls_label_PW = 100 + 15 |
| |
| print("Install MPLS intf group for dst mac {0}", dst_mac) |
| # add MPLS interface group |
| mpls_intf_gid, mpls_intf_msg = add_mpls_intf_group( |
| ctrl=self.controller, |
| ref_gid=l2_gid, |
| dst_mac=dst_mac, |
| src_mac=intf_src_mac, |
| vid=vlan_id, |
| index=id, |
| add=False, |
| send_barrier=True |
| ) |
| |
| # add MPLS flow with BoS=0 |
| add_mpls_flow_pw( |
| ctrl=self.controller, |
| action_group_id=mpls_intf_gid, |
| label=mpls_label_SR2, |
| ethertype=0x8847, |
| tunnel_index=1, |
| bos=0 |
| ) |
| |
| # add Termination flow |
| add_termination_flow( |
| ctrl=self.controller, |
| in_port=in_port, |
| eth_type=0x8847, |
| dst_mac=intf_src_mac, |
| vlanid=vlan_id, |
| goto_table=23 |
| ) |
| |
| # add VLAN flows |
| add_one_vlan_table_flow( |
| ctrl=self.controller, |
| of_port=in_port, |
| vlan_id=vlan_id, |
| flag=VLAN_TABLE_FLAG_ONLY_TAG, |
| ) |
| add_one_vlan_table_flow( |
| ctrl=self.controller, |
| of_port=in_port, |
| vlan_id=vlan_id, |
| flag=VLAN_TABLE_FLAG_ONLY_UNTAG |
| ) |
| # Packet generation with sleep |
| time.sleep(2) |
| label_SR2 = (mpls_label_SR2, 0, 0, 32) |
| label_2 = (mpls_label_PW, 0, 1, 32) |
| |
| # set to false to test if routing traffic |
| # comes through |
| raw_input("Press enter to send the packet, inspect the groups in the switch!") |
| print("Sending packet with dst mac {0} and labels {1}".format(switch_mac, [label_SR2, label_2])) |
| parsed_pkt = mpls_packet( |
| pktlen=104, |
| ip_ttl=63, |
| label=[label_SR2, label_2], |
| encapsulated_ethernet = True, |
| eth_dst=switch_mac |
| ) |
| |
| pkt = str( parsed_pkt ) |
| self.dataplane.send( in_port, pkt ) |
| |
| expected_label = (mpls_label_PW, 0, 1, 31) |
| print("Expecting packet with dst mac {0} and labels {1}".format(mac_dst, [label_2])) |
| parsed_pkt = mpls_packet( |
| pktlen=100, |
| ip_ttl=63, |
| eth_dst=mac_dst, |
| eth_src=switch_mac, |
| label=[ expected_label ], |
| encapsulated_ethernet = True |
| ) |
| |
| pkt = str( parsed_pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_packet( self, pkt, in_port ) |
| |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |
| class VlanCrossConnect ( base_tests.SimpleDataPlane ): |
| """ |
| Tries the cross connect functionality of the ofdpa switches. |
| """ |
| def runTest( self ): |
| Groups = Queue.LifoQueue( ) |
| try: |
| if len( config[ "port_map" ] ) < 2: |
| logging.info( "Port count less than 2, can't run this case" ) |
| return |
| |
| # each entry represents a subscriber [id, ip in hex, inner_vlan, outer_vlan, ip in dot form] |
| subscriber_info = [ [10, 0xc0a80001, 12, 11, "192.168.0.1"] ] |
| |
| index = 5 |
| for sub_info in subscriber_info: |
| |
| #print("Initializing rules for subscriber with id {0}".format(sub_info[0])) |
| |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, sub_info[0] ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| output_dst_mac = [ 0x00, 0x00, 0x00, 0x33, 0x33, 0x00 ] |
| output_dst_mac_str = ':'.join( [ '%02X' % x for x in output_dst_mac ] ) |
| |
| dip = sub_info[1] |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = sub_info[2] |
| outer_vlan = sub_info[3] |
| |
| index = inner_vlan |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| # add l2 interface group, uncomment for unfiltered |
| l2_gid, l2_msg = add_one_l2_interface_group( self.controller, out_port, outer_vlan, True, True) |
| #i l2_gid, l2_msg = add_one_l2_unfiltered_group( self.controller, out_port, 1, True) |
| |
| # add vlan flow table |
| add_one_vlan_table_flow( self.controller, port, inner_vlan, outer_vlan, vrf=0, flag=VLAN_TABLE_FLAG_ONLY_STACKED ) |
| add_one_vlan_1_table_flow_pw( self.controller, port, index, new_outer_vlan_id=outer_vlan ,outer_vlan_id=outer_vlan, inner_vlan_id=inner_vlan, cross_connect=True, send_barrier=True) |
| add_mpls_l2_port_flow(ctrl=self.controller, of_port=port, mpls_l2_port=port, tunnel_index=index, ref_gid=l2_gid, qos_index=0, goto=ACL_FLOW_TABLE) |
| index += 1 |
| |
| Groups._put( l2_gid ) |
| do_barrier( self.controller ) |
| |
| for sub_info in subscriber_info: |
| #print("Sending packet for subscriber with id {0}".format(sub_info[0])) |
| input_src_mac = [ 0x00, 0x00, 0x00, 0xcc, 0xcc, sub_info[0] ] |
| input_src_mac_str = ':'.join( [ '%02X' % x for x in input_src_mac ] ) |
| |
| input_dst_mac = [ 0x00, 0x00, 0x00, 0x22, 0x22, 0x00 ] |
| input_dst_mac_str = ':'.join( [ '%02X' % x for x in input_dst_mac ] ) |
| |
| dip = sub_info[1] |
| ports = config[ "port_map" ].keys( ) |
| |
| inner_vlan = sub_info[2] |
| outer_vlan = sub_info[3] |
| |
| port = ports[0] |
| out_port = ports[1] |
| |
| ip_src = sub_info[4] |
| ip_dst = '192.168.0.{}'.format(sub_info[0]) |
| parsed_pkt = qinq_tcp_packet( pktlen=120, vlan_vid=inner_vlan, dl_vlan_outer=outer_vlan, |
| eth_dst=input_dst_mac_str, eth_src=input_src_mac_str, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| parsed_pkt = simple_tcp_packet_two_vlan( pktlen=120, out_dl_vlan_enable=True, in_dl_vlan_enable=True, in_vlan_vid=inner_vlan, out_vlan_vid=outer_vlan, |
| eth_dst=input_dst_mac_str, eth_src=input_src_mac_str, ip_ttl=64, ip_src=ip_src, ip_dst=ip_dst ) |
| |
| pkt = str( parsed_pkt ) |
| |
| self.dataplane.send( port, pkt ) |
| verify_packet( self, pkt, out_port ) |
| verify_no_other_packets( self ) |
| |
| finally: |
| delete_all_flows( self.controller ) |
| delete_groups( self.controller, Groups ) |
| delete_all_groups( self.controller ) |
| |
| |