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