xref: /freebsd-src/contrib/netbsd-tests/net/icmp/t_forward.c (revision 57718be8fa0bd5edc11ab9a72e68cc71982939a6)
1*57718be8SEnji Cooper /*	$NetBSD: t_forward.c,v 1.8 2012/03/18 09:46:50 jruoho Exp $	*/
2*57718be8SEnji Cooper 
3*57718be8SEnji Cooper /*-
4*57718be8SEnji Cooper  * Copyright (c) 2010 The NetBSD Foundation, Inc.
5*57718be8SEnji Cooper  * All rights reserved.
6*57718be8SEnji Cooper  *
7*57718be8SEnji Cooper  * Redistribution and use in source and binary forms, with or without
8*57718be8SEnji Cooper  * modification, are permitted provided that the following conditions
9*57718be8SEnji Cooper  * are met:
10*57718be8SEnji Cooper  * 1. Redistributions of source code must retain the above copyright
11*57718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer.
12*57718be8SEnji Cooper  * 2. Redistributions in binary form must reproduce the above copyright
13*57718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer in the
14*57718be8SEnji Cooper  *    documentation and/or other materials provided with the distribution.
15*57718be8SEnji Cooper  *
16*57718be8SEnji Cooper  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
17*57718be8SEnji Cooper  * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
18*57718be8SEnji Cooper  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19*57718be8SEnji Cooper  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20*57718be8SEnji Cooper  * IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY
21*57718be8SEnji Cooper  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*57718be8SEnji Cooper  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
23*57718be8SEnji Cooper  * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24*57718be8SEnji Cooper  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25*57718be8SEnji Cooper  * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
26*57718be8SEnji Cooper  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27*57718be8SEnji Cooper  * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*57718be8SEnji Cooper  */
29*57718be8SEnji Cooper 
30*57718be8SEnji Cooper #include <sys/cdefs.h>
31*57718be8SEnji Cooper #ifndef lint
32*57718be8SEnji Cooper __RCSID("$NetBSD: t_forward.c,v 1.8 2012/03/18 09:46:50 jruoho Exp $");
33*57718be8SEnji Cooper #endif /* not lint */
34*57718be8SEnji Cooper 
35*57718be8SEnji Cooper #include <sys/types.h>
36*57718be8SEnji Cooper #include <sys/socket.h>
37*57718be8SEnji Cooper #include <sys/time.h>
38*57718be8SEnji Cooper #include <sys/sysctl.h>
39*57718be8SEnji Cooper #include <sys/wait.h>
40*57718be8SEnji Cooper 
41*57718be8SEnji Cooper #include <arpa/inet.h>
42*57718be8SEnji Cooper 
43*57718be8SEnji Cooper #include <netinet/in.h>
44*57718be8SEnji Cooper #include <netinet/in_systm.h>
45*57718be8SEnji Cooper #include <netinet/ip.h>
46*57718be8SEnji Cooper #include <netinet/ip_icmp.h>
47*57718be8SEnji Cooper #include <netinet/icmp_var.h>
48*57718be8SEnji Cooper #include <net/route.h>
49*57718be8SEnji Cooper 
50*57718be8SEnji Cooper #include <rump/rump.h>
51*57718be8SEnji Cooper #include <rump/rump_syscalls.h>
52*57718be8SEnji Cooper 
53*57718be8SEnji Cooper #include <atf-c.h>
54*57718be8SEnji Cooper #include <errno.h>
55*57718be8SEnji Cooper #include <stdio.h>
56*57718be8SEnji Cooper #include <stdlib.h>
57*57718be8SEnji Cooper #include <string.h>
58*57718be8SEnji Cooper #include <unistd.h>
59*57718be8SEnji Cooper 
60*57718be8SEnji Cooper #include "../../h_macros.h"
61*57718be8SEnji Cooper #include "../config/netconfig.c"
62*57718be8SEnji Cooper 
63*57718be8SEnji Cooper /*
64*57718be8SEnji Cooper  * Since our maxttl is in our private namespace, we don't need raw packet
65*57718be8SEnji Cooper  * construction like traceroute(8) -- we can just use the global maxttl.
66*57718be8SEnji Cooper  */
67*57718be8SEnji Cooper static void
68*57718be8SEnji Cooper sendttl(void)
69*57718be8SEnji Cooper {
70*57718be8SEnji Cooper 	struct sockaddr_in sin;
71*57718be8SEnji Cooper 	char payload[1024];
72*57718be8SEnji Cooper 	char ifname[IFNAMSIZ];
73*57718be8SEnji Cooper 	int mib[4] = { CTL_NET, PF_INET, IPPROTO_IP, IPCTL_DEFTTL };
74*57718be8SEnji Cooper 	int nv;
75*57718be8SEnji Cooper 	int s;
76*57718be8SEnji Cooper 
77*57718be8SEnji Cooper 	netcfg_rump_makeshmif("bus1", ifname);
78*57718be8SEnji Cooper 	netcfg_rump_if(ifname, "1.0.0.1", "255.255.255.0");
79*57718be8SEnji Cooper 	netcfg_rump_route("0.0.0.0", "0.0.0.0", "1.0.0.2"); /* default router */
80*57718be8SEnji Cooper 
81*57718be8SEnji Cooper 	/* set global ttl to 1 */
82*57718be8SEnji Cooper 	nv = 1;
83*57718be8SEnji Cooper 	if (rump_sys___sysctl(mib, 4, NULL, NULL, &nv, sizeof(nv)) == -1)
84*57718be8SEnji Cooper 		atf_tc_fail_errno("set ttl");
85*57718be8SEnji Cooper 
86*57718be8SEnji Cooper 	s = rump_sys_socket(PF_INET, SOCK_DGRAM, 0);
87*57718be8SEnji Cooper 	if (s == -1)
88*57718be8SEnji Cooper 		atf_tc_fail_errno("create send socket");
89*57718be8SEnji Cooper 
90*57718be8SEnji Cooper 	memset(&sin, 0, sizeof(sin));
91*57718be8SEnji Cooper 	sin.sin_len = sizeof(sin);
92*57718be8SEnji Cooper 	sin.sin_family = AF_INET;
93*57718be8SEnji Cooper 	sin.sin_port = htons(33434);
94*57718be8SEnji Cooper 	sin.sin_addr.s_addr = inet_addr("9.9.9.9");
95*57718be8SEnji Cooper 
96*57718be8SEnji Cooper 	/* send udp datagram with ttl == 1 */
97*57718be8SEnji Cooper 	if (rump_sys_sendto(s, payload, sizeof(payload), 0,
98*57718be8SEnji Cooper 	    (struct sockaddr *)&sin, sizeof(sin)) == -1)
99*57718be8SEnji Cooper 		atf_tc_fail_errno("sendto");
100*57718be8SEnji Cooper }
101*57718be8SEnji Cooper 
102*57718be8SEnji Cooper static void
103*57718be8SEnji Cooper router(void)
104*57718be8SEnji Cooper {
105*57718be8SEnji Cooper 	int mib[4] = { CTL_NET, PF_INET, IPPROTO_ICMP,
106*57718be8SEnji Cooper 		    ICMPCTL_RETURNDATABYTES };
107*57718be8SEnji Cooper 	char ifname[IFNAMSIZ];
108*57718be8SEnji Cooper 	int nv;
109*57718be8SEnji Cooper 
110*57718be8SEnji Cooper 	/* set returndatabytes to 200 */
111*57718be8SEnji Cooper 	nv = 200;
112*57718be8SEnji Cooper 	if (rump_sys___sysctl(mib, 4, NULL, NULL, &nv, sizeof(nv)) == -1)
113*57718be8SEnji Cooper 		atf_tc_fail_errno("sysctl returndatabytes");
114*57718be8SEnji Cooper 
115*57718be8SEnji Cooper 	netcfg_rump_makeshmif("bus1", ifname);
116*57718be8SEnji Cooper 	netcfg_rump_if(ifname, "1.0.0.2", "255.255.255.0");
117*57718be8SEnji Cooper 
118*57718be8SEnji Cooper 	/*
119*57718be8SEnji Cooper 	 * Wait for parent to send us the data and for us to have
120*57718be8SEnji Cooper 	 * a chance to process it.
121*57718be8SEnji Cooper 	 */
122*57718be8SEnji Cooper 	sleep(1);
123*57718be8SEnji Cooper 	exit(0);
124*57718be8SEnji Cooper }
125*57718be8SEnji Cooper 
126*57718be8SEnji Cooper ATF_TC(returndatabytes);
127*57718be8SEnji Cooper ATF_TC_HEAD(returndatabytes, tc)
128*57718be8SEnji Cooper {
129*57718be8SEnji Cooper 
130*57718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "icmp.returndatabytes with certain "
131*57718be8SEnji Cooper 	    "packets can cause kernel panic (PR kern/43548)");
132*57718be8SEnji Cooper 	atf_tc_set_md_var(tc, "timeout", "4"); /* just in case */
133*57718be8SEnji Cooper }
134*57718be8SEnji Cooper 
135*57718be8SEnji Cooper ATF_TC_BODY(returndatabytes, tc)
136*57718be8SEnji Cooper {
137*57718be8SEnji Cooper 	pid_t cpid;
138*57718be8SEnji Cooper 	int status;
139*57718be8SEnji Cooper 
140*57718be8SEnji Cooper 	cpid = fork();
141*57718be8SEnji Cooper 	rump_init();
142*57718be8SEnji Cooper 
143*57718be8SEnji Cooper 	switch (cpid) {
144*57718be8SEnji Cooper 	case -1:
145*57718be8SEnji Cooper 		atf_tc_fail_errno("fork failed");
146*57718be8SEnji Cooper 	case 0:
147*57718be8SEnji Cooper 		router();
148*57718be8SEnji Cooper 		break;
149*57718be8SEnji Cooper 	default:
150*57718be8SEnji Cooper 		sendttl();
151*57718be8SEnji Cooper 		if (wait(&status) == -1)
152*57718be8SEnji Cooper 			atf_tc_fail_errno("wait");
153*57718be8SEnji Cooper 		if (WIFEXITED(status)) {
154*57718be8SEnji Cooper 			if (WEXITSTATUS(status))
155*57718be8SEnji Cooper 				atf_tc_fail("child exited with status %d",
156*57718be8SEnji Cooper 				    WEXITSTATUS(status));
157*57718be8SEnji Cooper 		} else {
158*57718be8SEnji Cooper 			atf_tc_fail("child died");
159*57718be8SEnji Cooper 		}
160*57718be8SEnji Cooper 	}
161*57718be8SEnji Cooper }
162*57718be8SEnji Cooper 
163*57718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
164*57718be8SEnji Cooper {
165*57718be8SEnji Cooper 
166*57718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, returndatabytes);
167*57718be8SEnji Cooper 
168*57718be8SEnji Cooper 	return atf_no_error();
169*57718be8SEnji Cooper }
170