blob: 92bb350d1b995a66aa726434c1de5347b48fedac [file] [log] [blame]
Damjan Marionf56b77a2016-10-03 19:44:57 +02001#!/usr/bin/env python
2
Damjan Marionf56b77a2016-10-03 19:44:57 +02003import unittest
Klement Sekeraf62ae122016-10-11 11:47:09 +02004import socket
5from logging import *
6
Damjan Marionf56b77a2016-10-03 19:44:57 +02007from framework import VppTestCase, VppTestRunner
Klement Sekeraf62ae122016-10-11 11:47:09 +02008from vpp_sub_interface import VppSubInterface, VppDot1QSubint, VppDot1ADSubint
Damjan Marionf56b77a2016-10-03 19:44:57 +02009
10from scapy.packet import Raw
11from scapy.layers.l2 import Ether, Dot1Q
Klement Sekeraf62ae122016-10-11 11:47:09 +020012from scapy.layers.inet6 import ICMPv6ND_NS, IPv6, UDP
Damjan Marionf56b77a2016-10-03 19:44:57 +020013
14
Klement Sekeraf62ae122016-10-11 11:47:09 +020015class TestIPv6(VppTestCase):
Damjan Marionf56b77a2016-10-03 19:44:57 +020016 """ IPv6 Test Case """
17
18 @classmethod
19 def setUpClass(cls):
20 super(TestIPv6, cls).setUpClass()
21
Klement Sekeraf62ae122016-10-11 11:47:09 +020022 def setUp(self):
23 super(TestIPv6, self).setUp()
Damjan Marionf56b77a2016-10-03 19:44:57 +020024
Klement Sekeraf62ae122016-10-11 11:47:09 +020025 # create 3 pg interfaces
26 self.create_pg_interfaces(range(3))
Damjan Marionf56b77a2016-10-03 19:44:57 +020027
Klement Sekeraf62ae122016-10-11 11:47:09 +020028 # create 2 subinterfaces for p1 and pg2
29 self.sub_interfaces = [
30 VppDot1QSubint(self, self.pg1, 100),
31 VppDot1QSubint(self, self.pg2, 200)]
32 # TODO: VppDot1ADSubint(self, self.pg2, 200, 300, 400)
Damjan Marionf56b77a2016-10-03 19:44:57 +020033
Klement Sekeraf62ae122016-10-11 11:47:09 +020034 # packet flows mapping pg0 -> pg1.sub, pg2.sub, etc.
35 self.flows = dict()
36 self.flows[self.pg0] = [self.pg1.sub_if, self.pg2.sub_if]
37 self.flows[self.pg1.sub_if] = [self.pg0, self.pg2.sub_if]
38 self.flows[self.pg2.sub_if] = [self.pg0, self.pg1.sub_if]
Damjan Marionf56b77a2016-10-03 19:44:57 +020039
Klement Sekeraf62ae122016-10-11 11:47:09 +020040 # packet sizes
41 self.pg_if_packet_sizes = [64, 512, 1518, 9018]
42 self.sub_if_packet_sizes = [64, 512, 1518 + 4, 9018 + 4]
Damjan Marionf56b77a2016-10-03 19:44:57 +020043
Klement Sekeraf62ae122016-10-11 11:47:09 +020044 self.interfaces = list(self.pg_interfaces)
45 self.interfaces.extend(self.sub_interfaces)
Damjan Marionf56b77a2016-10-03 19:44:57 +020046
Klement Sekeraf62ae122016-10-11 11:47:09 +020047 # setup all interfaces
48 for i in self.interfaces:
49 i.admin_up()
50 i.config_ip6()
51 i.resolve_ndp()
52
53 # config 2M FIB enries
54 self.config_fib_entries(200)
Damjan Marionf56b77a2016-10-03 19:44:57 +020055
56 def tearDown(self):
Klement Sekeraf62ae122016-10-11 11:47:09 +020057 super(TestIPv6, self).tearDown()
58 if not self.vpp_dead:
59 info(self.vapi.cli("show ip6 neighbors"))
60 # info(self.vapi.cli("show ip6 fib")) # many entries
Damjan Marionf56b77a2016-10-03 19:44:57 +020061
Klement Sekeraf62ae122016-10-11 11:47:09 +020062 def config_fib_entries(self, count):
63 n_int = len(self.interfaces)
64 percent = 0
65 counter = 0.0
66 dest_addr = socket.inet_pton(socket.AF_INET6, "fd02::1")
67 dest_addr_len = 128
68 for i in self.interfaces:
69 next_hop_address = i.local_ip6n
70 for j in range(count / n_int):
71 self.vapi.ip_add_del_route(
72 dest_addr, dest_addr_len, next_hop_address, is_ipv6=1)
73 counter = counter + 1
74 if counter / count * 100 > percent:
75 info("Configure %d FIB entries .. %d%% done" %
76 (count, percent))
77 percent = percent + 1
Damjan Marionf56b77a2016-10-03 19:44:57 +020078
Klement Sekeraf62ae122016-10-11 11:47:09 +020079 def create_stream(self, src_if, packet_sizes):
Damjan Marionf56b77a2016-10-03 19:44:57 +020080 pkts = []
81 for i in range(0, 257):
Klement Sekeraf62ae122016-10-11 11:47:09 +020082 dst_if = self.flows[src_if][i % 2]
83 info = self.create_packet_info(
84 src_if.sw_if_index, dst_if.sw_if_index)
Damjan Marionf56b77a2016-10-03 19:44:57 +020085 payload = self.info_to_payload(info)
Klement Sekeraf62ae122016-10-11 11:47:09 +020086 p = (Ether(dst=src_if.local_mac, src=src_if.remote_mac) /
87 IPv6(src=src_if.remote_ip6, dst=dst_if.remote_ip6) /
Damjan Marionf56b77a2016-10-03 19:44:57 +020088 UDP(sport=1234, dport=1234) /
89 Raw(payload))
90 info.data = p.copy()
Klement Sekeraf62ae122016-10-11 11:47:09 +020091 if isinstance(src_if, VppSubInterface):
92 p = src_if.add_dot1_layer(p)
93 size = packet_sizes[(i // 2) % len(packet_sizes)]
Damjan Marionf56b77a2016-10-03 19:44:57 +020094 self.extend_packet(p, size)
95 pkts.append(p)
96 return pkts
97
Klement Sekeraf62ae122016-10-11 11:47:09 +020098 def verify_capture(self, dst_if, capture):
99 info("Verifying capture on interface %s" % dst_if.name)
100 last_info = dict()
Damjan Marionf56b77a2016-10-03 19:44:57 +0200101 for i in self.interfaces:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200102 last_info[i.sw_if_index] = None
103 is_sub_if = False
104 dst_sw_if_index = dst_if.sw_if_index
105 if hasattr(dst_if, 'parent'):
106 is_sub_if = True
Damjan Marionf56b77a2016-10-03 19:44:57 +0200107 for packet in capture:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200108 if is_sub_if:
109 # Check VLAN tags and Ethernet header
110 packet = dst_if.remove_dot1_layer(packet)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200111 self.assertTrue(Dot1Q not in packet)
112 try:
113 ip = packet[IPv6]
114 udp = packet[UDP]
115 payload_info = self.payload_to_info(str(packet[Raw]))
116 packet_index = payload_info.index
Klement Sekeraf62ae122016-10-11 11:47:09 +0200117 self.assertEqual(payload_info.dst, dst_sw_if_index)
118 debug("Got packet on port %s: src=%u (id=%u)" %
119 (dst_if.name, payload_info.src, packet_index))
120 next_info = self.get_next_packet_info_for_interface2(
121 payload_info.src, dst_sw_if_index,
122 last_info[payload_info.src])
123 last_info[payload_info.src] = next_info
Damjan Marionf56b77a2016-10-03 19:44:57 +0200124 self.assertTrue(next_info is not None)
125 self.assertEqual(packet_index, next_info.index)
126 saved_packet = next_info.data
127 # Check standard fields
128 self.assertEqual(ip.src, saved_packet[IPv6].src)
129 self.assertEqual(ip.dst, saved_packet[IPv6].dst)
130 self.assertEqual(udp.sport, saved_packet[UDP].sport)
131 self.assertEqual(udp.dport, saved_packet[UDP].dport)
132 except:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200133 error("Unexpected or invalid packet:")
134 error(packet.show())
Damjan Marionf56b77a2016-10-03 19:44:57 +0200135 raise
136 for i in self.interfaces:
Klement Sekeraf62ae122016-10-11 11:47:09 +0200137 remaining_packet = self.get_next_packet_info_for_interface2(
138 i.sw_if_index, dst_sw_if_index, last_info[i.sw_if_index])
139 self.assertTrue(
140 remaining_packet is None,
141 "Interface %s: Packet expected from interface %s didn't arrive" %
142 (dst_if.name, i.name))
Damjan Marionf56b77a2016-10-03 19:44:57 +0200143
144 def test_fib(self):
145 """ IPv6 FIB test """
146
Klement Sekeraf62ae122016-10-11 11:47:09 +0200147 pkts = self.create_stream(self.pg0, self.pg_if_packet_sizes)
148 self.pg0.add_stream(pkts)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200149
Klement Sekeraf62ae122016-10-11 11:47:09 +0200150 for i in self.sub_interfaces:
151 pkts = self.create_stream(i, self.sub_if_packet_sizes)
152 i.parent.add_stream(pkts)
153
154 self.pg_enable_capture(self.pg_interfaces)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200155 self.pg_start()
156
Klement Sekeraf62ae122016-10-11 11:47:09 +0200157 pkts = self.pg0.get_capture()
158 self.verify_capture(self.pg0, pkts)
159
160 for i in self.sub_interfaces:
161 pkts = i.parent.get_capture()
162 self.verify_capture(i, pkts)
Damjan Marionf56b77a2016-10-03 19:44:57 +0200163
164
165if __name__ == '__main__':
Klement Sekeraf62ae122016-10-11 11:47:09 +0200166 unittest.main(testRunner=VppTestRunner)