xref: /freebsd/tests/sys/netinet6/frag6/frag6_12.py (revision f74e6e494f48b76f357d43d3672cca3606ffac04)
1*f74e6e49SBjoern A. Zeeb#!/usr/bin/env python
2*f74e6e49SBjoern A. Zeeb#-
3*f74e6e49SBjoern A. Zeeb# SPDX-License-Identifier: BSD-2-Clause
4*f74e6e49SBjoern A. Zeeb#
5*f74e6e49SBjoern A. Zeeb# Copyright (c) 2019 Netflix, Inc.
6*f74e6e49SBjoern A. Zeeb#
7*f74e6e49SBjoern A. Zeeb# Redistribution and use in source and binary forms, with or without
8*f74e6e49SBjoern A. Zeeb# modification, are permitted provided that the following conditions
9*f74e6e49SBjoern A. Zeeb# are met:
10*f74e6e49SBjoern A. Zeeb# 1. Redistributions of source code must retain the above copyright
11*f74e6e49SBjoern A. Zeeb#    notice, this list of conditions and the following disclaimer.
12*f74e6e49SBjoern A. Zeeb# 2. Redistributions in binary form must reproduce the above copyright
13*f74e6e49SBjoern A. Zeeb#    notice, this list of conditions and the following disclaimer in the
14*f74e6e49SBjoern A. Zeeb#    documentation and/or other materials provided with the distribution.
15*f74e6e49SBjoern A. Zeeb#
16*f74e6e49SBjoern A. Zeeb# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17*f74e6e49SBjoern A. Zeeb# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*f74e6e49SBjoern A. Zeeb# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*f74e6e49SBjoern A. Zeeb# ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20*f74e6e49SBjoern A. Zeeb# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21*f74e6e49SBjoern A. Zeeb# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22*f74e6e49SBjoern A. Zeeb# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23*f74e6e49SBjoern A. Zeeb# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24*f74e6e49SBjoern A. Zeeb# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25*f74e6e49SBjoern A. Zeeb# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26*f74e6e49SBjoern A. Zeeb# SUCH DAMAGE.
27*f74e6e49SBjoern A. Zeeb#
28*f74e6e49SBjoern A. Zeeb# $FreeBSD$
29*f74e6e49SBjoern A. Zeeb#
30*f74e6e49SBjoern A. Zeeb
31*f74e6e49SBjoern A. Zeebimport argparse
32*f74e6e49SBjoern A. Zeebimport scapy.all as sp
33*f74e6e49SBjoern A. Zeebimport socket
34*f74e6e49SBjoern A. Zeebimport sys
35*f74e6e49SBjoern A. Zeebfrom sniffer import Sniffer
36*f74e6e49SBjoern A. Zeebfrom time import sleep
37*f74e6e49SBjoern A. Zeeb
38*f74e6e49SBjoern A. Zeebdef check_icmp6_error(args, packet):
39*f74e6e49SBjoern A. Zeeb	ip6 = packet.getlayer(sp.IPv6)
40*f74e6e49SBjoern A. Zeeb	if not ip6:
41*f74e6e49SBjoern A. Zeeb		return False
42*f74e6e49SBjoern A. Zeeb	oip6 = sp.IPv6(src=args.src[0], dst=args.to[0])
43*f74e6e49SBjoern A. Zeeb	if ip6.dst != oip6.src:
44*f74e6e49SBjoern A. Zeeb		return False
45*f74e6e49SBjoern A. Zeeb	icmp6 = packet.getlayer(sp.ICMPv6TimeExceeded)
46*f74e6e49SBjoern A. Zeeb	if not icmp6:
47*f74e6e49SBjoern A. Zeeb		return False
48*f74e6e49SBjoern A. Zeeb	# ICMP6_TIME_EXCEED_REASSEMBLY 1
49*f74e6e49SBjoern A. Zeeb	if icmp6.code != 1:
50*f74e6e49SBjoern A. Zeeb		return False
51*f74e6e49SBjoern A. Zeeb	# Should we check the payload as well?
52*f74e6e49SBjoern A. Zeeb	# We are running in a very isolated environment and nothing else
53*f74e6e49SBjoern A. Zeeb	# should trigger an ICMPv6 Time Exceeded / Frag reassembly so leave it.
54*f74e6e49SBjoern A. Zeeb	#icmp6.display()
55*f74e6e49SBjoern A. Zeeb	return True
56*f74e6e49SBjoern A. Zeeb
57*f74e6e49SBjoern A. Zeeb
58*f74e6e49SBjoern A. Zeebdef main():
59*f74e6e49SBjoern A. Zeeb	parser = argparse.ArgumentParser("frag6.py",
60*f74e6e49SBjoern A. Zeeb		description="IPv6 fragementation test tool")
61*f74e6e49SBjoern A. Zeeb	parser.add_argument('--sendif', nargs=1,
62*f74e6e49SBjoern A. Zeeb		required=True,
63*f74e6e49SBjoern A. Zeeb		help='The interface through which the packet will be sent')
64*f74e6e49SBjoern A. Zeeb	parser.add_argument('--recvif', nargs=1,
65*f74e6e49SBjoern A. Zeeb		required=True,
66*f74e6e49SBjoern A. Zeeb		help='The interface on which to check for the packet')
67*f74e6e49SBjoern A. Zeeb	parser.add_argument('--src', nargs=1,
68*f74e6e49SBjoern A. Zeeb		required=True,
69*f74e6e49SBjoern A. Zeeb		help='The source IP address')
70*f74e6e49SBjoern A. Zeeb	parser.add_argument('--to', nargs=1,
71*f74e6e49SBjoern A. Zeeb		required=True,
72*f74e6e49SBjoern A. Zeeb		help='The destination IP address')
73*f74e6e49SBjoern A. Zeeb	parser.add_argument('--debug',
74*f74e6e49SBjoern A. Zeeb		required=False, action='store_true',
75*f74e6e49SBjoern A. Zeeb		help='Enable test debugging')
76*f74e6e49SBjoern A. Zeeb
77*f74e6e49SBjoern A. Zeeb	args = parser.parse_args()
78*f74e6e49SBjoern A. Zeeb
79*f74e6e49SBjoern A. Zeeb
80*f74e6e49SBjoern A. Zeeb	# Start sniffing on recvif
81*f74e6e49SBjoern A. Zeeb	sniffer = Sniffer(args, check_icmp6_error)
82*f74e6e49SBjoern A. Zeeb
83*f74e6e49SBjoern A. Zeeb
84*f74e6e49SBjoern A. Zeeb	########################################################################
85*f74e6e49SBjoern A. Zeeb	#
86*f74e6e49SBjoern A. Zeeb	# A single start fragment with payload.
87*f74e6e49SBjoern A. Zeeb	#
88*f74e6e49SBjoern A. Zeeb	# A:  Waiting for more data.
89*f74e6e49SBjoern A. Zeeb	# R:  Timeout / Expiry.
90*f74e6e49SBjoern A. Zeeb	#
91*f74e6e49SBjoern A. Zeeb	data = "6" * 1280
92*f74e6e49SBjoern A. Zeeb	ip6f01 = sp.Ether() / \
93*f74e6e49SBjoern A. Zeeb		sp.IPv6(src=args.src[0], dst=args.to[0]) / \
94*f74e6e49SBjoern A. Zeeb		sp.IPv6ExtHdrFragment(offset=0, m=1, id=3) / \
95*f74e6e49SBjoern A. Zeeb		sp.UDP(dport=3456, sport=6543) / \
96*f74e6e49SBjoern A. Zeeb		data
97*f74e6e49SBjoern A. Zeeb	if args.debug :
98*f74e6e49SBjoern A. Zeeb		ip6f01.display()
99*f74e6e49SBjoern A. Zeeb	sp.sendp(ip6f01, iface=args.sendif[0], verbose=False)
100*f74e6e49SBjoern A. Zeeb
101*f74e6e49SBjoern A. Zeeb	# Wait for ICMPv6 error generation on timeout.
102*f74e6e49SBjoern A. Zeeb	sleep(75)
103*f74e6e49SBjoern A. Zeeb	sniffer.setEnd()
104*f74e6e49SBjoern A. Zeeb	sniffer.join()
105*f74e6e49SBjoern A. Zeeb	if not sniffer.foundCorrectPacket:
106*f74e6e49SBjoern A. Zeeb		sys.exit(1)
107*f74e6e49SBjoern A. Zeeb
108*f74e6e49SBjoern A. Zeeb	sys.exit(0)
109*f74e6e49SBjoern A. Zeeb
110*f74e6e49SBjoern A. Zeebif __name__ == '__main__':
111*f74e6e49SBjoern A. Zeeb	main()
112