1*640235e2SEnji Cooper /* $NetBSD: mcast.c,v 1.3 2015/05/28 10:19:17 ozaki-r Exp $ */
2*640235e2SEnji Cooper
3*640235e2SEnji Cooper /*-
4*640235e2SEnji Cooper * Copyright (c) 2014 The NetBSD Foundation, Inc.
5*640235e2SEnji Cooper * All rights reserved.
6*640235e2SEnji Cooper *
7*640235e2SEnji Cooper * This code is derived from software contributed to The NetBSD Foundation
8*640235e2SEnji Cooper * by Christos Zoulas.
9*640235e2SEnji Cooper *
10*640235e2SEnji Cooper * Redistribution and use in source and binary forms, with or without
11*640235e2SEnji Cooper * modification, are permitted provided that the following conditions
12*640235e2SEnji Cooper * are met:
13*640235e2SEnji Cooper * 1. Redistributions of source code must retain the above copyright
14*640235e2SEnji Cooper * notice, this list of conditions and the following disclaimer.
15*640235e2SEnji Cooper * 2. Redistributions in binary form must reproduce the above copyright
16*640235e2SEnji Cooper * notice, this list of conditions and the following disclaimer in the
17*640235e2SEnji Cooper * documentation and/or other materials provided with the distribution.
18*640235e2SEnji Cooper *
19*640235e2SEnji Cooper * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20*640235e2SEnji Cooper * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21*640235e2SEnji Cooper * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22*640235e2SEnji Cooper * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23*640235e2SEnji Cooper * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24*640235e2SEnji Cooper * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25*640235e2SEnji Cooper * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26*640235e2SEnji Cooper * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27*640235e2SEnji Cooper * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28*640235e2SEnji Cooper * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29*640235e2SEnji Cooper * POSSIBILITY OF SUCH DAMAGE.
30*640235e2SEnji Cooper */
31*640235e2SEnji Cooper #include <sys/cdefs.h>
32*640235e2SEnji Cooper #ifdef __RCSID
33*640235e2SEnji Cooper __RCSID("$NetBSD: mcast.c,v 1.3 2015/05/28 10:19:17 ozaki-r Exp $");
34*640235e2SEnji Cooper #else
35*640235e2SEnji Cooper extern const char *__progname;
36*640235e2SEnji Cooper #define getprogname() __progname
37*640235e2SEnji Cooper #endif
38*640235e2SEnji Cooper
39*640235e2SEnji Cooper #include <sys/types.h>
40*640235e2SEnji Cooper #include <sys/socket.h>
41*640235e2SEnji Cooper #include <sys/wait.h>
42*640235e2SEnji Cooper #include <sys/time.h>
43*640235e2SEnji Cooper #include <netinet/in.h>
44*640235e2SEnji Cooper
45*640235e2SEnji Cooper #include <assert.h>
46*640235e2SEnji Cooper #include <netdb.h>
47*640235e2SEnji Cooper #include <time.h>
48*640235e2SEnji Cooper #include <signal.h>
49*640235e2SEnji Cooper #include <stdio.h>
50*640235e2SEnji Cooper #include <string.h>
51*640235e2SEnji Cooper #include <stdlib.h>
52*640235e2SEnji Cooper #include <unistd.h>
53*640235e2SEnji Cooper #include <err.h>
54*640235e2SEnji Cooper #include <errno.h>
55*640235e2SEnji Cooper #include <poll.h>
56*640235e2SEnji Cooper #include <stdbool.h>
57*640235e2SEnji Cooper
58*640235e2SEnji Cooper #ifdef ATF
59*640235e2SEnji Cooper #include <atf-c.h>
60*640235e2SEnji Cooper
61*640235e2SEnji Cooper #define ERRX(ev, msg, ...) ATF_REQUIRE_MSG(0, msg, __VA_ARGS__)
62*640235e2SEnji Cooper #define ERRX0(ev, msg) ATF_REQUIRE_MSG(0, msg)
63*640235e2SEnji Cooper
64*640235e2SEnji Cooper #define SKIPX(ev, msg, ...) do { \
65*640235e2SEnji Cooper atf_tc_skip(msg, __VA_ARGS__); \
66*640235e2SEnji Cooper return; \
67*640235e2SEnji Cooper } while(/*CONSTCOND*/0)
68*640235e2SEnji Cooper
69*640235e2SEnji Cooper #else
70*640235e2SEnji Cooper #define ERRX(ev, msg, ...) errx(ev, msg, __VA_ARGS__)
71*640235e2SEnji Cooper #define ERRX0(ev, msg) errx(ev, msg)
72*640235e2SEnji Cooper #define SKIPX(ev, msg, ...) errx(ev, msg, __VA_ARGS__)
73*640235e2SEnji Cooper #endif
74*640235e2SEnji Cooper
75*640235e2SEnji Cooper static int debug;
76*640235e2SEnji Cooper
77*640235e2SEnji Cooper #define TOTAL 10
78*640235e2SEnji Cooper #define PORT_V4MAPPED "6666"
79*640235e2SEnji Cooper #define HOST_V4MAPPED "::FFFF:239.1.1.1"
80*640235e2SEnji Cooper #define PORT_V4 "6666"
81*640235e2SEnji Cooper #define HOST_V4 "239.1.1.1"
82*640235e2SEnji Cooper #define PORT_V6 "6666"
83*640235e2SEnji Cooper #define HOST_V6 "FF05:1:0:0:0:0:0:1"
84*640235e2SEnji Cooper
85*640235e2SEnji Cooper struct message {
86*640235e2SEnji Cooper size_t seq;
87*640235e2SEnji Cooper struct timespec ts;
88*640235e2SEnji Cooper };
89*640235e2SEnji Cooper
90*640235e2SEnji Cooper static int
addmc(int s,struct addrinfo * ai,bool bug)91*640235e2SEnji Cooper addmc(int s, struct addrinfo *ai, bool bug)
92*640235e2SEnji Cooper {
93*640235e2SEnji Cooper struct ip_mreq m4;
94*640235e2SEnji Cooper struct ipv6_mreq m6;
95*640235e2SEnji Cooper struct sockaddr_in *s4;
96*640235e2SEnji Cooper struct sockaddr_in6 *s6;
97*640235e2SEnji Cooper unsigned int ifc;
98*640235e2SEnji Cooper
99*640235e2SEnji Cooper switch (ai->ai_family) {
100*640235e2SEnji Cooper case AF_INET:
101*640235e2SEnji Cooper s4 = (void *)ai->ai_addr;
102*640235e2SEnji Cooper assert(sizeof(*s4) == ai->ai_addrlen);
103*640235e2SEnji Cooper m4.imr_multiaddr = s4->sin_addr;
104*640235e2SEnji Cooper m4.imr_interface.s_addr = htonl(INADDR_ANY);
105*640235e2SEnji Cooper return setsockopt(s, IPPROTO_IP, IP_ADD_MEMBERSHIP,
106*640235e2SEnji Cooper &m4, sizeof(m4));
107*640235e2SEnji Cooper case AF_INET6:
108*640235e2SEnji Cooper s6 = (void *)ai->ai_addr;
109*640235e2SEnji Cooper /*
110*640235e2SEnji Cooper * Linux: Does not support the v6 ioctls on v4 mapped
111*640235e2SEnji Cooper * sockets but it does support the v4 ones and
112*640235e2SEnji Cooper * it works.
113*640235e2SEnji Cooper * MacOS/X: Supports the v6 ioctls on v4 mapped sockets,
114*640235e2SEnji Cooper * but does not work and also does not support
115*640235e2SEnji Cooper * the v4 ioctls. So no way to make multicasting
116*640235e2SEnji Cooper * work with mapped addresses.
117*640235e2SEnji Cooper * NetBSD: Supports both and works for both.
118*640235e2SEnji Cooper */
119*640235e2SEnji Cooper if (bug && IN6_IS_ADDR_V4MAPPED(&s6->sin6_addr)) {
120*640235e2SEnji Cooper memcpy(&m4.imr_multiaddr, &s6->sin6_addr.s6_addr[12],
121*640235e2SEnji Cooper sizeof(m4.imr_multiaddr));
122*640235e2SEnji Cooper m4.imr_interface.s_addr = htonl(INADDR_ANY);
123*640235e2SEnji Cooper return setsockopt(s, IPPROTO_IP, IP_ADD_MEMBERSHIP,
124*640235e2SEnji Cooper &m4, sizeof(m4));
125*640235e2SEnji Cooper }
126*640235e2SEnji Cooper assert(sizeof(*s6) == ai->ai_addrlen);
127*640235e2SEnji Cooper memset(&m6, 0, sizeof(m6));
128*640235e2SEnji Cooper #if 0
129*640235e2SEnji Cooper ifc = 1;
130*640235e2SEnji Cooper if (setsockopt(s, IPPROTO_IPV6, IPV6_MULTICAST_LOOP,
131*640235e2SEnji Cooper &ifc, sizeof(ifc)) == -1)
132*640235e2SEnji Cooper return -1;
133*640235e2SEnji Cooper ifc = 224;
134*640235e2SEnji Cooper if (setsockopt(s, IPPROTO_IPV6, IPV6_MULTICAST_HOPS,
135*640235e2SEnji Cooper &ifc, sizeof(ifc)) == -1)
136*640235e2SEnji Cooper return -1;
137*640235e2SEnji Cooper ifc = 1; /* XXX should pick a proper interface */
138*640235e2SEnji Cooper if (setsockopt(s, IPPROTO_IPV6, IPV6_MULTICAST_IF, &ifc,
139*640235e2SEnji Cooper sizeof(ifc)) == -1)
140*640235e2SEnji Cooper return -1;
141*640235e2SEnji Cooper #else
142*640235e2SEnji Cooper ifc = 0; /* Let pick an appropriate interface */
143*640235e2SEnji Cooper #endif
144*640235e2SEnji Cooper m6.ipv6mr_interface = ifc;
145*640235e2SEnji Cooper m6.ipv6mr_multiaddr = s6->sin6_addr;
146*640235e2SEnji Cooper return setsockopt(s, IPPROTO_IPV6, IPV6_JOIN_GROUP,
147*640235e2SEnji Cooper &m6, sizeof(m6));
148*640235e2SEnji Cooper default:
149*640235e2SEnji Cooper errno = EOPNOTSUPP;
150*640235e2SEnji Cooper return -1;
151*640235e2SEnji Cooper }
152*640235e2SEnji Cooper }
153*640235e2SEnji Cooper
154*640235e2SEnji Cooper static int
allowv4mapped(int s,struct addrinfo * ai)155*640235e2SEnji Cooper allowv4mapped(int s, struct addrinfo *ai)
156*640235e2SEnji Cooper {
157*640235e2SEnji Cooper struct sockaddr_in6 *s6;
158*640235e2SEnji Cooper int zero = 0;
159*640235e2SEnji Cooper
160*640235e2SEnji Cooper if (ai->ai_family != AF_INET6)
161*640235e2SEnji Cooper return 0;
162*640235e2SEnji Cooper
163*640235e2SEnji Cooper s6 = (void *)ai->ai_addr;
164*640235e2SEnji Cooper
165*640235e2SEnji Cooper if (!IN6_IS_ADDR_V4MAPPED(&s6->sin6_addr))
166*640235e2SEnji Cooper return 0;
167*640235e2SEnji Cooper return setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &zero, sizeof(zero));
168*640235e2SEnji Cooper }
169*640235e2SEnji Cooper
170*640235e2SEnji Cooper static struct sockaddr_storage ss;
171*640235e2SEnji Cooper static int
connector(int fd,const struct sockaddr * sa,socklen_t slen)172*640235e2SEnji Cooper connector(int fd, const struct sockaddr *sa, socklen_t slen)
173*640235e2SEnji Cooper {
174*640235e2SEnji Cooper assert(sizeof(ss) > slen);
175*640235e2SEnji Cooper memcpy(&ss, sa, slen);
176*640235e2SEnji Cooper return 0;
177*640235e2SEnji Cooper }
178*640235e2SEnji Cooper
179*640235e2SEnji Cooper static void
show(const char * prefix,const struct message * msg)180*640235e2SEnji Cooper show(const char *prefix, const struct message *msg)
181*640235e2SEnji Cooper {
182*640235e2SEnji Cooper printf("%10.10s: %zu [%jd.%ld]\n", prefix, msg->seq, (intmax_t)
183*640235e2SEnji Cooper msg->ts.tv_sec, msg->ts.tv_nsec);
184*640235e2SEnji Cooper }
185*640235e2SEnji Cooper
186*640235e2SEnji Cooper static int
getsocket(const char * host,const char * port,int (* f)(int,const struct sockaddr *,socklen_t),socklen_t * slen,bool bug)187*640235e2SEnji Cooper getsocket(const char *host, const char *port,
188*640235e2SEnji Cooper int (*f)(int, const struct sockaddr *, socklen_t), socklen_t *slen,
189*640235e2SEnji Cooper bool bug)
190*640235e2SEnji Cooper {
191*640235e2SEnji Cooper int e, s, lasterrno = 0;
192*640235e2SEnji Cooper struct addrinfo hints, *ai0, *ai;
193*640235e2SEnji Cooper const char *cause = "?";
194*640235e2SEnji Cooper
195*640235e2SEnji Cooper memset(&hints, 0, sizeof(hints));
196*640235e2SEnji Cooper hints.ai_family = AF_UNSPEC;
197*640235e2SEnji Cooper hints.ai_socktype = SOCK_DGRAM;
198*640235e2SEnji Cooper e = getaddrinfo(host, port, &hints, &ai0);
199*640235e2SEnji Cooper if (e)
200*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "Can't resolve %s:%s (%s)", host, port,
201*640235e2SEnji Cooper gai_strerror(e));
202*640235e2SEnji Cooper
203*640235e2SEnji Cooper s = -1;
204*640235e2SEnji Cooper for (ai = ai0; ai; ai = ai->ai_next) {
205*640235e2SEnji Cooper s = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol);
206*640235e2SEnji Cooper if (s == -1) {
207*640235e2SEnji Cooper lasterrno = errno;
208*640235e2SEnji Cooper cause = "socket";
209*640235e2SEnji Cooper continue;
210*640235e2SEnji Cooper }
211*640235e2SEnji Cooper if (allowv4mapped(s, ai) == -1) {
212*640235e2SEnji Cooper cause = "allow v4 mapped";
213*640235e2SEnji Cooper goto out;
214*640235e2SEnji Cooper }
215*640235e2SEnji Cooper if ((*f)(s, ai->ai_addr, ai->ai_addrlen) == -1) {
216*640235e2SEnji Cooper cause = f == bind ? "bind" : "connect";
217*640235e2SEnji Cooper goto out;
218*640235e2SEnji Cooper }
219*640235e2SEnji Cooper if ((f == bind || f == connector) && addmc(s, ai, bug) == -1) {
220*640235e2SEnji Cooper cause = "join group";
221*640235e2SEnji Cooper goto out;
222*640235e2SEnji Cooper }
223*640235e2SEnji Cooper *slen = ai->ai_addrlen;
224*640235e2SEnji Cooper break;
225*640235e2SEnji Cooper out:
226*640235e2SEnji Cooper lasterrno = errno;
227*640235e2SEnji Cooper close(s);
228*640235e2SEnji Cooper s = -1;
229*640235e2SEnji Cooper continue;
230*640235e2SEnji Cooper }
231*640235e2SEnji Cooper freeaddrinfo(ai0);
232*640235e2SEnji Cooper if (s == -1)
233*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "%s (%s)", cause, strerror(lasterrno));
234*640235e2SEnji Cooper return s;
235*640235e2SEnji Cooper }
236*640235e2SEnji Cooper
237*640235e2SEnji Cooper static int
synchronize(const int fd,bool waiter)238*640235e2SEnji Cooper synchronize(const int fd, bool waiter)
239*640235e2SEnji Cooper {
240*640235e2SEnji Cooper int syncmsg = 0;
241*640235e2SEnji Cooper int r;
242*640235e2SEnji Cooper struct pollfd pfd;
243*640235e2SEnji Cooper
244*640235e2SEnji Cooper if (waiter) {
245*640235e2SEnji Cooper pfd.fd = fd;
246*640235e2SEnji Cooper pfd.events = POLLIN;
247*640235e2SEnji Cooper
248*640235e2SEnji Cooper /* We use poll to avoid lock up when the peer died unexpectedly */
249*640235e2SEnji Cooper r = poll(&pfd, 1, 10000);
250*640235e2SEnji Cooper if (r == -1)
251*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "poll (%s)", strerror(errno));
252*640235e2SEnji Cooper if (r == 0)
253*640235e2SEnji Cooper /* Timed out */
254*640235e2SEnji Cooper return -1;
255*640235e2SEnji Cooper
256*640235e2SEnji Cooper if (read(fd, &syncmsg, sizeof(syncmsg)) == -1)
257*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "read (%s)", strerror(errno));
258*640235e2SEnji Cooper } else {
259*640235e2SEnji Cooper if (write(fd, &syncmsg, sizeof(syncmsg)) == -1)
260*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "write (%s)", strerror(errno));
261*640235e2SEnji Cooper }
262*640235e2SEnji Cooper
263*640235e2SEnji Cooper return 0;
264*640235e2SEnji Cooper }
265*640235e2SEnji Cooper
266*640235e2SEnji Cooper static int
sender(const int fd,const char * host,const char * port,size_t n,bool conn,bool bug)267*640235e2SEnji Cooper sender(const int fd, const char *host, const char *port, size_t n, bool conn,
268*640235e2SEnji Cooper bool bug)
269*640235e2SEnji Cooper {
270*640235e2SEnji Cooper int s;
271*640235e2SEnji Cooper ssize_t l;
272*640235e2SEnji Cooper struct message msg;
273*640235e2SEnji Cooper
274*640235e2SEnji Cooper socklen_t slen;
275*640235e2SEnji Cooper
276*640235e2SEnji Cooper s = getsocket(host, port, conn ? connect : connector, &slen, bug);
277*640235e2SEnji Cooper
278*640235e2SEnji Cooper /* Wait until receiver gets ready. */
279*640235e2SEnji Cooper if (synchronize(fd, true) == -1)
280*640235e2SEnji Cooper return -1;
281*640235e2SEnji Cooper
282*640235e2SEnji Cooper for (msg.seq = 0; msg.seq < n; msg.seq++) {
283*640235e2SEnji Cooper #ifdef CLOCK_MONOTONIC
284*640235e2SEnji Cooper if (clock_gettime(CLOCK_MONOTONIC, &msg.ts) == -1)
285*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "clock (%s)", strerror(errno));
286*640235e2SEnji Cooper #else
287*640235e2SEnji Cooper struct timeval tv;
288*640235e2SEnji Cooper if (gettimeofday(&tv, NULL) == -1)
289*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "clock (%s)", strerror(errno));
290*640235e2SEnji Cooper msg.ts.tv_sec = tv.tv_sec;
291*640235e2SEnji Cooper msg.ts.tv_nsec = tv.tv_usec * 1000;
292*640235e2SEnji Cooper #endif
293*640235e2SEnji Cooper if (debug)
294*640235e2SEnji Cooper show("sending", &msg);
295*640235e2SEnji Cooper l = conn ? send(s, &msg, sizeof(msg), 0) :
296*640235e2SEnji Cooper sendto(s, &msg, sizeof(msg), 0, (void *)&ss, slen);
297*640235e2SEnji Cooper if (l == -1)
298*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "send (%s)", strerror(errno));
299*640235e2SEnji Cooper usleep(100);
300*640235e2SEnji Cooper }
301*640235e2SEnji Cooper
302*640235e2SEnji Cooper /* Wait until receiver finishes its work. */
303*640235e2SEnji Cooper if (synchronize(fd, true) == -1)
304*640235e2SEnji Cooper return -1;
305*640235e2SEnji Cooper
306*640235e2SEnji Cooper return 0;
307*640235e2SEnji Cooper }
308*640235e2SEnji Cooper
309*640235e2SEnji Cooper static void
receiver(const int fd,const char * host,const char * port,size_t n,bool conn,bool bug)310*640235e2SEnji Cooper receiver(const int fd, const char *host, const char *port, size_t n, bool conn,
311*640235e2SEnji Cooper bool bug)
312*640235e2SEnji Cooper {
313*640235e2SEnji Cooper int s;
314*640235e2SEnji Cooper ssize_t l;
315*640235e2SEnji Cooper size_t seq;
316*640235e2SEnji Cooper struct message msg;
317*640235e2SEnji Cooper struct pollfd pfd;
318*640235e2SEnji Cooper socklen_t slen;
319*640235e2SEnji Cooper
320*640235e2SEnji Cooper s = getsocket(host, port, bind, &slen, bug);
321*640235e2SEnji Cooper pfd.fd = s;
322*640235e2SEnji Cooper pfd.events = POLLIN;
323*640235e2SEnji Cooper
324*640235e2SEnji Cooper /* Tell I'm ready */
325*640235e2SEnji Cooper synchronize(fd, false);
326*640235e2SEnji Cooper
327*640235e2SEnji Cooper for (seq = 0; seq < n; seq++) {
328*640235e2SEnji Cooper if (poll(&pfd, 1, 10000) == -1)
329*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "poll (%s)", strerror(errno));
330*640235e2SEnji Cooper l = conn ? recv(s, &msg, sizeof(msg), 0) :
331*640235e2SEnji Cooper recvfrom(s, &msg, sizeof(msg), 0, (void *)&ss, &slen);
332*640235e2SEnji Cooper if (l == -1)
333*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "recv (%s)", strerror(errno));
334*640235e2SEnji Cooper if (debug)
335*640235e2SEnji Cooper show("got", &msg);
336*640235e2SEnji Cooper if (seq != msg.seq)
337*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "seq: expect=%zu actual=%zu",
338*640235e2SEnji Cooper seq, msg.seq);
339*640235e2SEnji Cooper }
340*640235e2SEnji Cooper
341*640235e2SEnji Cooper /* Tell I'm finished */
342*640235e2SEnji Cooper synchronize(fd, false);
343*640235e2SEnji Cooper }
344*640235e2SEnji Cooper
345*640235e2SEnji Cooper static void
run(const char * host,const char * port,size_t n,bool conn,bool bug)346*640235e2SEnji Cooper run(const char *host, const char *port, size_t n, bool conn, bool bug)
347*640235e2SEnji Cooper {
348*640235e2SEnji Cooper pid_t pid;
349*640235e2SEnji Cooper int status;
350*640235e2SEnji Cooper int syncfds[2];
351*640235e2SEnji Cooper int error;
352*640235e2SEnji Cooper
353*640235e2SEnji Cooper if (socketpair(AF_UNIX, SOCK_STREAM, 0, syncfds) == -1)
354*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "socketpair (%s)", strerror(errno));
355*640235e2SEnji Cooper
356*640235e2SEnji Cooper switch ((pid = fork())) {
357*640235e2SEnji Cooper case 0:
358*640235e2SEnji Cooper receiver(syncfds[0], host, port, n, conn, bug);
359*640235e2SEnji Cooper return;
360*640235e2SEnji Cooper case -1:
361*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "fork (%s)", strerror(errno));
362*640235e2SEnji Cooper default:
363*640235e2SEnji Cooper error = sender(syncfds[1], host, port, n, conn, bug);
364*640235e2SEnji Cooper again:
365*640235e2SEnji Cooper switch (waitpid(pid, &status, WNOHANG)) {
366*640235e2SEnji Cooper case -1:
367*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "wait (%s)", strerror(errno));
368*640235e2SEnji Cooper case 0:
369*640235e2SEnji Cooper if (error == 0)
370*640235e2SEnji Cooper /*
371*640235e2SEnji Cooper * Receiver is still alive, but we know
372*640235e2SEnji Cooper * it will exit soon.
373*640235e2SEnji Cooper */
374*640235e2SEnji Cooper goto again;
375*640235e2SEnji Cooper
376*640235e2SEnji Cooper if (kill(pid, SIGTERM) == -1)
377*640235e2SEnji Cooper ERRX(EXIT_FAILURE, "kill (%s)",
378*640235e2SEnji Cooper strerror(errno));
379*640235e2SEnji Cooper goto again;
380*640235e2SEnji Cooper default:
381*640235e2SEnji Cooper if (WIFSIGNALED(status)) {
382*640235e2SEnji Cooper if (WTERMSIG(status) == SIGTERM)
383*640235e2SEnji Cooper ERRX0(EXIT_FAILURE,
384*640235e2SEnji Cooper "receiver failed and was killed" \
385*640235e2SEnji Cooper "by sender");
386*640235e2SEnji Cooper else
387*640235e2SEnji Cooper ERRX(EXIT_FAILURE,
388*640235e2SEnji Cooper "receiver got signaled (%s)",
389*640235e2SEnji Cooper strsignal(WTERMSIG(status)));
390*640235e2SEnji Cooper } else if (WIFEXITED(status)) {
391*640235e2SEnji Cooper if (WEXITSTATUS(status) != 0)
392*640235e2SEnji Cooper ERRX(EXIT_FAILURE,
393*640235e2SEnji Cooper "receiver exited with status %d",
394*640235e2SEnji Cooper WEXITSTATUS(status));
395*640235e2SEnji Cooper } else {
396*640235e2SEnji Cooper ERRX(EXIT_FAILURE,
397*640235e2SEnji Cooper "receiver exited with unexpected status %d",
398*640235e2SEnji Cooper status);
399*640235e2SEnji Cooper }
400*640235e2SEnji Cooper break;
401*640235e2SEnji Cooper }
402*640235e2SEnji Cooper return;
403*640235e2SEnji Cooper }
404*640235e2SEnji Cooper }
405*640235e2SEnji Cooper
406*640235e2SEnji Cooper #ifndef ATF
407*640235e2SEnji Cooper int
main(int argc,char * argv[])408*640235e2SEnji Cooper main(int argc, char *argv[])
409*640235e2SEnji Cooper {
410*640235e2SEnji Cooper const char *host, *port;
411*640235e2SEnji Cooper int c;
412*640235e2SEnji Cooper size_t n;
413*640235e2SEnji Cooper bool conn, bug;
414*640235e2SEnji Cooper
415*640235e2SEnji Cooper host = HOST_V4;
416*640235e2SEnji Cooper port = PORT_V4;
417*640235e2SEnji Cooper n = TOTAL;
418*640235e2SEnji Cooper bug = conn = false;
419*640235e2SEnji Cooper
420*640235e2SEnji Cooper while ((c = getopt(argc, argv, "46bcdmn:")) != -1)
421*640235e2SEnji Cooper switch (c) {
422*640235e2SEnji Cooper case '4':
423*640235e2SEnji Cooper host = HOST_V4;
424*640235e2SEnji Cooper port = PORT_V4;
425*640235e2SEnji Cooper break;
426*640235e2SEnji Cooper case '6':
427*640235e2SEnji Cooper host = HOST_V6;
428*640235e2SEnji Cooper port = PORT_V6;
429*640235e2SEnji Cooper break;
430*640235e2SEnji Cooper case 'b':
431*640235e2SEnji Cooper bug = true;
432*640235e2SEnji Cooper break;
433*640235e2SEnji Cooper case 'c':
434*640235e2SEnji Cooper conn = true;
435*640235e2SEnji Cooper break;
436*640235e2SEnji Cooper case 'd':
437*640235e2SEnji Cooper debug++;
438*640235e2SEnji Cooper break;
439*640235e2SEnji Cooper case 'm':
440*640235e2SEnji Cooper host = HOST_V4MAPPED;
441*640235e2SEnji Cooper port = PORT_V4MAPPED;
442*640235e2SEnji Cooper break;
443*640235e2SEnji Cooper case 'n':
444*640235e2SEnji Cooper n = atoi(optarg);
445*640235e2SEnji Cooper break;
446*640235e2SEnji Cooper default:
447*640235e2SEnji Cooper fprintf(stderr, "Usage: %s [-cdm46] [-n <tot>]",
448*640235e2SEnji Cooper getprogname());
449*640235e2SEnji Cooper return 1;
450*640235e2SEnji Cooper }
451*640235e2SEnji Cooper
452*640235e2SEnji Cooper run(host, port, n, conn, bug);
453*640235e2SEnji Cooper return 0;
454*640235e2SEnji Cooper }
455*640235e2SEnji Cooper #else
456*640235e2SEnji Cooper
457*640235e2SEnji Cooper ATF_TC(conninet4);
ATF_TC_HEAD(conninet4,tc)458*640235e2SEnji Cooper ATF_TC_HEAD(conninet4, tc)
459*640235e2SEnji Cooper {
460*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks connected multicast for ipv4");
461*640235e2SEnji Cooper }
462*640235e2SEnji Cooper
ATF_TC_BODY(conninet4,tc)463*640235e2SEnji Cooper ATF_TC_BODY(conninet4, tc)
464*640235e2SEnji Cooper {
465*640235e2SEnji Cooper run(HOST_V4, PORT_V4, TOTAL, true, false);
466*640235e2SEnji Cooper }
467*640235e2SEnji Cooper
468*640235e2SEnji Cooper ATF_TC(connmappedinet4);
ATF_TC_HEAD(connmappedinet4,tc)469*640235e2SEnji Cooper ATF_TC_HEAD(connmappedinet4, tc)
470*640235e2SEnji Cooper {
471*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks connected multicast for mapped ipv4");
472*640235e2SEnji Cooper }
473*640235e2SEnji Cooper
ATF_TC_BODY(connmappedinet4,tc)474*640235e2SEnji Cooper ATF_TC_BODY(connmappedinet4, tc)
475*640235e2SEnji Cooper {
476*640235e2SEnji Cooper run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, true, false);
477*640235e2SEnji Cooper }
478*640235e2SEnji Cooper
479*640235e2SEnji Cooper ATF_TC(connmappedbuginet4);
ATF_TC_HEAD(connmappedbuginet4,tc)480*640235e2SEnji Cooper ATF_TC_HEAD(connmappedbuginet4, tc)
481*640235e2SEnji Cooper {
482*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks connected multicast for mapped ipv4 using the v4 ioctls");
483*640235e2SEnji Cooper }
484*640235e2SEnji Cooper
ATF_TC_BODY(connmappedbuginet4,tc)485*640235e2SEnji Cooper ATF_TC_BODY(connmappedbuginet4, tc)
486*640235e2SEnji Cooper {
487*640235e2SEnji Cooper run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, true, true);
488*640235e2SEnji Cooper }
489*640235e2SEnji Cooper
490*640235e2SEnji Cooper ATF_TC(conninet6);
ATF_TC_HEAD(conninet6,tc)491*640235e2SEnji Cooper ATF_TC_HEAD(conninet6, tc)
492*640235e2SEnji Cooper {
493*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks connected multicast for ipv6");
494*640235e2SEnji Cooper }
495*640235e2SEnji Cooper
ATF_TC_BODY(conninet6,tc)496*640235e2SEnji Cooper ATF_TC_BODY(conninet6, tc)
497*640235e2SEnji Cooper {
498*640235e2SEnji Cooper run(HOST_V6, PORT_V6, TOTAL, true, false);
499*640235e2SEnji Cooper }
500*640235e2SEnji Cooper
501*640235e2SEnji Cooper ATF_TC(unconninet4);
ATF_TC_HEAD(unconninet4,tc)502*640235e2SEnji Cooper ATF_TC_HEAD(unconninet4, tc)
503*640235e2SEnji Cooper {
504*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for ipv4");
505*640235e2SEnji Cooper }
506*640235e2SEnji Cooper
ATF_TC_BODY(unconninet4,tc)507*640235e2SEnji Cooper ATF_TC_BODY(unconninet4, tc)
508*640235e2SEnji Cooper {
509*640235e2SEnji Cooper run(HOST_V4, PORT_V4, TOTAL, false, false);
510*640235e2SEnji Cooper }
511*640235e2SEnji Cooper
512*640235e2SEnji Cooper ATF_TC(unconnmappedinet4);
ATF_TC_HEAD(unconnmappedinet4,tc)513*640235e2SEnji Cooper ATF_TC_HEAD(unconnmappedinet4, tc)
514*640235e2SEnji Cooper {
515*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for mapped ipv4");
516*640235e2SEnji Cooper }
517*640235e2SEnji Cooper
ATF_TC_BODY(unconnmappedinet4,tc)518*640235e2SEnji Cooper ATF_TC_BODY(unconnmappedinet4, tc)
519*640235e2SEnji Cooper {
520*640235e2SEnji Cooper run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, false, false);
521*640235e2SEnji Cooper }
522*640235e2SEnji Cooper
523*640235e2SEnji Cooper ATF_TC(unconnmappedbuginet4);
ATF_TC_HEAD(unconnmappedbuginet4,tc)524*640235e2SEnji Cooper ATF_TC_HEAD(unconnmappedbuginet4, tc)
525*640235e2SEnji Cooper {
526*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for mapped ipv4 using the v4 ioctls");
527*640235e2SEnji Cooper }
528*640235e2SEnji Cooper
ATF_TC_BODY(unconnmappedbuginet4,tc)529*640235e2SEnji Cooper ATF_TC_BODY(unconnmappedbuginet4, tc)
530*640235e2SEnji Cooper {
531*640235e2SEnji Cooper run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, false, true);
532*640235e2SEnji Cooper }
533*640235e2SEnji Cooper
534*640235e2SEnji Cooper ATF_TC(unconninet6);
ATF_TC_HEAD(unconninet6,tc)535*640235e2SEnji Cooper ATF_TC_HEAD(unconninet6, tc)
536*640235e2SEnji Cooper {
537*640235e2SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for ipv6");
538*640235e2SEnji Cooper }
539*640235e2SEnji Cooper
ATF_TC_BODY(unconninet6,tc)540*640235e2SEnji Cooper ATF_TC_BODY(unconninet6, tc)
541*640235e2SEnji Cooper {
542*640235e2SEnji Cooper run(HOST_V6, PORT_V6, TOTAL, false, false);
543*640235e2SEnji Cooper }
544*640235e2SEnji Cooper
ATF_TP_ADD_TCS(tp)545*640235e2SEnji Cooper ATF_TP_ADD_TCS(tp)
546*640235e2SEnji Cooper {
547*640235e2SEnji Cooper debug++;
548*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, conninet4);
549*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, connmappedinet4);
550*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, connmappedbuginet4);
551*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, conninet6);
552*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, unconninet4);
553*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, unconnmappedinet4);
554*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, unconnmappedbuginet4);
555*640235e2SEnji Cooper ATF_TP_ADD_TC(tp, unconninet6);
556*640235e2SEnji Cooper
557*640235e2SEnji Cooper return atf_no_error();
558*640235e2SEnji Cooper }
559*640235e2SEnji Cooper #endif
560