Home | History | Annotate | Line # | Download | only in mcast
mcast.c revision 1.3
      1  1.3  ozaki /*	$NetBSD: mcast.c,v 1.3 2015/05/28 10:19:17 ozaki-r 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  ozaki __RCSID("$NetBSD: mcast.c,v 1.3 2015/05/28 10:19:17 ozaki-r 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.1  ozaki 
     77  1.1  ozaki #define TOTAL 10
     78  1.1  ozaki #define PORT_V4MAPPED "6666"
     79  1.1  ozaki #define HOST_V4MAPPED "::FFFF:239.1.1.1"
     80  1.1  ozaki #define PORT_V4 "6666"
     81  1.1  ozaki #define HOST_V4 "239.1.1.1"
     82  1.1  ozaki #define PORT_V6 "6666"
     83  1.1  ozaki #define HOST_V6 "FF05:1:0:0:0:0:0:1"
     84  1.1  ozaki 
     85  1.1  ozaki struct message {
     86  1.1  ozaki 	size_t seq;
     87  1.1  ozaki 	struct timespec ts;
     88  1.1  ozaki };
     89  1.1  ozaki 
     90  1.1  ozaki static int
     91  1.1  ozaki addmc(int s, struct addrinfo *ai, bool bug)
     92  1.1  ozaki {
     93  1.1  ozaki 	struct ip_mreq m4;
     94  1.1  ozaki 	struct ipv6_mreq m6;
     95  1.1  ozaki 	struct sockaddr_in *s4;
     96  1.1  ozaki 	struct sockaddr_in6 *s6;
     97  1.1  ozaki 	unsigned int ifc;
     98  1.1  ozaki 
     99  1.1  ozaki 	switch (ai->ai_family) {
    100  1.1  ozaki 	case AF_INET:
    101  1.1  ozaki 		s4 = (void *)ai->ai_addr;
    102  1.1  ozaki 		assert(sizeof(*s4) == ai->ai_addrlen);
    103  1.1  ozaki 		m4.imr_multiaddr = s4->sin_addr;
    104  1.1  ozaki 		m4.imr_interface.s_addr = htonl(INADDR_ANY);
    105  1.1  ozaki 		return setsockopt(s, IPPROTO_IP, IP_ADD_MEMBERSHIP,
    106  1.1  ozaki 		    &m4, sizeof(m4));
    107  1.1  ozaki 	case AF_INET6:
    108  1.1  ozaki 		s6 = (void *)ai->ai_addr;
    109  1.1  ozaki 		/*
    110  1.1  ozaki 		 * Linux:	Does not support the v6 ioctls on v4 mapped
    111  1.1  ozaki 		 *		sockets but it does support the v4 ones and
    112  1.1  ozaki 		 *		it works.
    113  1.1  ozaki 		 * MacOS/X:	Supports the v6 ioctls on v4 mapped sockets,
    114  1.1  ozaki 		 *		but does not work and also does not support
    115  1.1  ozaki 		 *		the v4 ioctls. So no way to make multicasting
    116  1.1  ozaki 		 *		work with mapped addresses.
    117  1.1  ozaki 		 * NetBSD:	Supports both and works for both.
    118  1.1  ozaki 		 */
    119  1.1  ozaki 		if (bug && IN6_IS_ADDR_V4MAPPED(&s6->sin6_addr)) {
    120  1.1  ozaki 			memcpy(&m4.imr_multiaddr, &s6->sin6_addr.s6_addr[12],
    121  1.1  ozaki 			    sizeof(m4.imr_multiaddr));
    122  1.1  ozaki 			m4.imr_interface.s_addr = htonl(INADDR_ANY);
    123  1.1  ozaki 			return setsockopt(s, IPPROTO_IP, IP_ADD_MEMBERSHIP,
    124  1.1  ozaki 			    &m4, sizeof(m4));
    125  1.1  ozaki 		}
    126  1.1  ozaki 		assert(sizeof(*s6) == ai->ai_addrlen);
    127  1.1  ozaki 		memset(&m6, 0, sizeof(m6));
    128  1.1  ozaki #if 0
    129  1.1  ozaki 		ifc = 1;
    130  1.1  ozaki 		if (setsockopt(s, IPPROTO_IPV6, IPV6_MULTICAST_LOOP,
    131  1.1  ozaki 		    &ifc, sizeof(ifc)) == -1)
    132  1.1  ozaki 			return -1;
    133  1.1  ozaki 		ifc = 224;
    134  1.1  ozaki 		if (setsockopt(s, IPPROTO_IPV6, IPV6_MULTICAST_HOPS,
    135  1.1  ozaki 		    &ifc, sizeof(ifc)) == -1)
    136  1.1  ozaki 			return -1;
    137  1.1  ozaki 		ifc = 1; /* XXX should pick a proper interface */
    138  1.1  ozaki 		if (setsockopt(s, IPPROTO_IPV6, IPV6_MULTICAST_IF, &ifc,
    139  1.1  ozaki 		    sizeof(ifc)) == -1)
    140  1.1  ozaki 			return -1;
    141  1.1  ozaki #else
    142  1.1  ozaki 		ifc = 0; /* Let pick an appropriate interface */
    143  1.1  ozaki #endif
    144  1.1  ozaki 		m6.ipv6mr_interface = ifc;
    145  1.1  ozaki 		m6.ipv6mr_multiaddr = s6->sin6_addr;
    146  1.1  ozaki 		return setsockopt(s, IPPROTO_IPV6, IPV6_JOIN_GROUP,
    147  1.1  ozaki 		    &m6, sizeof(m6));
    148  1.1  ozaki 	default:
    149  1.1  ozaki 		errno = EOPNOTSUPP;
    150  1.1  ozaki 		return -1;
    151  1.1  ozaki 	}
    152  1.1  ozaki }
    153  1.1  ozaki 
    154  1.1  ozaki static int
    155  1.1  ozaki allowv4mapped(int s, struct addrinfo *ai)
    156  1.1  ozaki {
    157  1.1  ozaki 	struct sockaddr_in6 *s6;
    158  1.1  ozaki 	int zero = 0;
    159  1.1  ozaki 
    160  1.1  ozaki 	if (ai->ai_family != AF_INET6)
    161  1.1  ozaki 		return 0;
    162  1.1  ozaki 
    163  1.1  ozaki 	s6 = (void *)ai->ai_addr;
    164  1.1  ozaki 
    165  1.1  ozaki 	if (!IN6_IS_ADDR_V4MAPPED(&s6->sin6_addr))
    166  1.1  ozaki 		return 0;
    167  1.1  ozaki 	return setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &zero, sizeof(zero));
    168  1.1  ozaki }
    169  1.1  ozaki 
    170  1.1  ozaki static struct sockaddr_storage ss;
    171  1.1  ozaki static int
    172  1.1  ozaki connector(int fd, const struct sockaddr *sa, socklen_t slen)
    173  1.1  ozaki {
    174  1.1  ozaki 	assert(sizeof(ss) > slen);
    175  1.1  ozaki 	memcpy(&ss, sa, slen);
    176  1.1  ozaki 	return 0;
    177  1.1  ozaki }
    178  1.1  ozaki 
    179  1.1  ozaki static void
    180  1.1  ozaki show(const char *prefix, const struct message *msg)
    181  1.1  ozaki {
    182  1.1  ozaki 	printf("%10.10s: %zu [%jd.%ld]\n", prefix, msg->seq, (intmax_t)
    183  1.1  ozaki 	    msg->ts.tv_sec, msg->ts.tv_nsec);
    184  1.1  ozaki }
    185  1.1  ozaki 
    186  1.1  ozaki static int
    187  1.1  ozaki getsocket(const char *host, const char *port,
    188  1.1  ozaki     int (*f)(int, const struct sockaddr *, socklen_t), socklen_t *slen,
    189  1.1  ozaki     bool bug)
    190  1.1  ozaki {
    191  1.1  ozaki 	int e, s, lasterrno = 0;
    192  1.1  ozaki 	struct addrinfo hints, *ai0, *ai;
    193  1.1  ozaki 	const char *cause = "?";
    194  1.1  ozaki 
    195  1.1  ozaki 	memset(&hints, 0, sizeof(hints));
    196  1.1  ozaki 	hints.ai_family = AF_UNSPEC;
    197  1.1  ozaki 	hints.ai_socktype = SOCK_DGRAM;
    198  1.1  ozaki 	e = getaddrinfo(host, port, &hints, &ai0);
    199  1.1  ozaki 	if (e)
    200  1.1  ozaki 		ERRX(EXIT_FAILURE, "Can't resolve %s:%s (%s)", host, port,
    201  1.1  ozaki 		    gai_strerror(e));
    202  1.1  ozaki 
    203  1.1  ozaki 	s = -1;
    204  1.1  ozaki 	for (ai = ai0; ai; ai = ai->ai_next) {
    205  1.1  ozaki 		s = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol);
    206  1.1  ozaki 		if (s == -1) {
    207  1.1  ozaki 			lasterrno = errno;
    208  1.1  ozaki 			cause = "socket";
    209  1.1  ozaki 			continue;
    210  1.1  ozaki 		}
    211  1.1  ozaki 		if (allowv4mapped(s, ai) == -1) {
    212  1.1  ozaki 			cause = "allow v4 mapped";
    213  1.1  ozaki 			goto out;
    214  1.1  ozaki 		}
    215  1.1  ozaki 		if ((*f)(s, ai->ai_addr, ai->ai_addrlen) == -1) {
    216  1.1  ozaki 			cause = f == bind ? "bind" : "connect";
    217  1.1  ozaki 			goto out;
    218  1.1  ozaki 		}
    219  1.1  ozaki 		if ((f == bind || f == connector) && addmc(s, ai, bug) == -1) {
    220  1.1  ozaki 			cause = "join group";
    221  1.1  ozaki 			goto out;
    222  1.1  ozaki 		}
    223  1.1  ozaki 		*slen = ai->ai_addrlen;
    224  1.1  ozaki 		break;
    225  1.1  ozaki out:
    226  1.1  ozaki 		lasterrno = errno;
    227  1.1  ozaki 		close(s);
    228  1.1  ozaki 		s = -1;
    229  1.1  ozaki 		continue;
    230  1.1  ozaki 	}
    231  1.1  ozaki 	freeaddrinfo(ai0);
    232  1.1  ozaki 	if (s == -1)
    233  1.1  ozaki 		ERRX(EXIT_FAILURE, "%s (%s)", cause, strerror(lasterrno));
    234  1.1  ozaki 	return s;
    235  1.1  ozaki }
    236  1.1  ozaki 
    237  1.3  ozaki static int
    238  1.3  ozaki synchronize(const int fd, bool waiter)
    239  1.3  ozaki {
    240  1.3  ozaki 	int syncmsg = 0;
    241  1.3  ozaki 	int r;
    242  1.3  ozaki 	struct pollfd pfd;
    243  1.3  ozaki 
    244  1.3  ozaki 	if (waiter) {
    245  1.3  ozaki 		pfd.fd = fd;
    246  1.3  ozaki 		pfd.events = POLLIN;
    247  1.3  ozaki 
    248  1.3  ozaki 		/* We use poll to avoid lock up when the peer died unexpectedly */
    249  1.3  ozaki 		r = poll(&pfd, 1, 10000);
    250  1.3  ozaki 		if (r == -1)
    251  1.3  ozaki 			ERRX(EXIT_FAILURE, "poll (%s)", strerror(errno));
    252  1.3  ozaki 		if (r == 0)
    253  1.3  ozaki 			/* Timed out */
    254  1.3  ozaki 			return -1;
    255  1.3  ozaki 
    256  1.3  ozaki 		if (read(fd, &syncmsg, sizeof(syncmsg)) == -1)
    257  1.3  ozaki 			ERRX(EXIT_FAILURE, "read (%s)", strerror(errno));
    258  1.3  ozaki 	} else {
    259  1.3  ozaki 		if (write(fd, &syncmsg, sizeof(syncmsg)) == -1)
    260  1.3  ozaki 			ERRX(EXIT_FAILURE, "write (%s)", strerror(errno));
    261  1.3  ozaki 	}
    262  1.3  ozaki 
    263  1.3  ozaki 	return 0;
    264  1.3  ozaki }
    265  1.3  ozaki 
    266  1.3  ozaki static int
    267  1.3  ozaki sender(const int fd, const char *host, const char *port, size_t n, bool conn,
    268  1.3  ozaki     bool bug)
    269  1.1  ozaki {
    270  1.1  ozaki 	int s;
    271  1.1  ozaki 	ssize_t l;
    272  1.1  ozaki 	struct message msg;
    273  1.1  ozaki 
    274  1.1  ozaki 	socklen_t slen;
    275  1.1  ozaki 
    276  1.1  ozaki 	s = getsocket(host, port, conn ? connect : connector, &slen, bug);
    277  1.3  ozaki 
    278  1.3  ozaki 	/* Wait until receiver gets ready. */
    279  1.3  ozaki 	if (synchronize(fd, true) == -1)
    280  1.3  ozaki 		return -1;
    281  1.3  ozaki 
    282  1.1  ozaki 	for (msg.seq = 0; msg.seq < n; msg.seq++) {
    283  1.1  ozaki #ifdef CLOCK_MONOTONIC
    284  1.1  ozaki 		if (clock_gettime(CLOCK_MONOTONIC, &msg.ts) == -1)
    285  1.1  ozaki 			ERRX(EXIT_FAILURE, "clock (%s)", strerror(errno));
    286  1.1  ozaki #else
    287  1.1  ozaki 		struct timeval tv;
    288  1.1  ozaki 		if (gettimeofday(&tv, NULL) == -1)
    289  1.1  ozaki 			ERRX(EXIT_FAILURE, "clock (%s)", strerror(errno));
    290  1.1  ozaki 		msg.ts.tv_sec = tv.tv_sec;
    291  1.1  ozaki 		msg.ts.tv_nsec = tv.tv_usec * 1000;
    292  1.1  ozaki #endif
    293  1.1  ozaki 		if (debug)
    294  1.1  ozaki 			show("sending", &msg);
    295  1.1  ozaki 		l = conn ? send(s, &msg, sizeof(msg), 0) :
    296  1.1  ozaki 		    sendto(s, &msg, sizeof(msg), 0, (void *)&ss, slen);
    297  1.1  ozaki 		if (l == -1)
    298  1.1  ozaki 			ERRX(EXIT_FAILURE, "send (%s)", strerror(errno));
    299  1.1  ozaki 		usleep(100);
    300  1.1  ozaki 	}
    301  1.3  ozaki 
    302  1.3  ozaki 	/* Wait until receiver finishes its work. */
    303  1.3  ozaki 	if (synchronize(fd, true) == -1)
    304  1.3  ozaki 		return -1;
    305  1.3  ozaki 
    306  1.3  ozaki 	return 0;
    307  1.1  ozaki }
    308  1.1  ozaki 
    309  1.1  ozaki static void
    310  1.3  ozaki receiver(const int fd, const char *host, const char *port, size_t n, bool conn,
    311  1.3  ozaki     bool bug)
    312  1.1  ozaki {
    313  1.1  ozaki 	int s;
    314  1.1  ozaki 	ssize_t l;
    315  1.1  ozaki 	size_t seq;
    316  1.1  ozaki 	struct message msg;
    317  1.1  ozaki 	struct pollfd pfd;
    318  1.1  ozaki 	socklen_t slen;
    319  1.1  ozaki 
    320  1.1  ozaki 	s = getsocket(host, port, bind, &slen, bug);
    321  1.1  ozaki 	pfd.fd = s;
    322  1.1  ozaki 	pfd.events = POLLIN;
    323  1.3  ozaki 
    324  1.3  ozaki 	/* Tell I'm ready */
    325  1.3  ozaki 	synchronize(fd, false);
    326  1.3  ozaki 
    327  1.1  ozaki 	for (seq = 0; seq < n; seq++) {
    328  1.1  ozaki 		if (poll(&pfd, 1, 10000) == -1)
    329  1.1  ozaki 			ERRX(EXIT_FAILURE, "poll (%s)", strerror(errno));
    330  1.1  ozaki 		l = conn ? recv(s, &msg, sizeof(msg), 0) :
    331  1.1  ozaki 		    recvfrom(s, &msg, sizeof(msg), 0, (void *)&ss, &slen);
    332  1.1  ozaki 		if (l == -1)
    333  1.1  ozaki 			ERRX(EXIT_FAILURE, "recv (%s)", strerror(errno));
    334  1.1  ozaki 		if (debug)
    335  1.1  ozaki 			show("got", &msg);
    336  1.1  ozaki 		if (seq != msg.seq)
    337  1.2  ozaki 			ERRX(EXIT_FAILURE, "seq: expect=%zu actual=%zu",
    338  1.2  ozaki 			    seq, msg.seq);
    339  1.1  ozaki 	}
    340  1.3  ozaki 
    341  1.3  ozaki 	/* Tell I'm finished */
    342  1.3  ozaki 	synchronize(fd, false);
    343  1.1  ozaki }
    344  1.1  ozaki 
    345  1.1  ozaki static void
    346  1.1  ozaki run(const char *host, const char *port, size_t n, bool conn, bool bug)
    347  1.1  ozaki {
    348  1.1  ozaki 	pid_t pid;
    349  1.1  ozaki 	int status;
    350  1.3  ozaki 	int syncfds[2];
    351  1.3  ozaki 	int error;
    352  1.3  ozaki 
    353  1.3  ozaki 	if (socketpair(AF_UNIX, SOCK_STREAM, 0, syncfds) == -1)
    354  1.3  ozaki 		ERRX(EXIT_FAILURE, "socketpair (%s)", strerror(errno));
    355  1.1  ozaki 
    356  1.1  ozaki 	switch ((pid = fork())) {
    357  1.1  ozaki 	case 0:
    358  1.3  ozaki 		receiver(syncfds[0], host, port, n, conn, bug);
    359  1.1  ozaki 		return;
    360  1.1  ozaki 	case -1:
    361  1.1  ozaki 		ERRX(EXIT_FAILURE, "fork (%s)", strerror(errno));
    362  1.1  ozaki 	default:
    363  1.3  ozaki 		error = sender(syncfds[1], host, port, n, conn, bug);
    364  1.1  ozaki 	again:
    365  1.1  ozaki 		switch (waitpid(pid, &status, WNOHANG)) {
    366  1.1  ozaki 		case -1:
    367  1.1  ozaki 			ERRX(EXIT_FAILURE, "wait (%s)", strerror(errno));
    368  1.1  ozaki 		case 0:
    369  1.3  ozaki 			if (error == 0)
    370  1.3  ozaki 				/*
    371  1.3  ozaki 				 * Receiver is still alive, but we know
    372  1.3  ozaki 				 * it will exit soon.
    373  1.3  ozaki 				 */
    374  1.3  ozaki 				goto again;
    375  1.3  ozaki 
    376  1.1  ozaki 			if (kill(pid, SIGTERM) == -1)
    377  1.1  ozaki 				ERRX(EXIT_FAILURE, "kill (%s)",
    378  1.1  ozaki 				    strerror(errno));
    379  1.1  ozaki 			goto again;
    380  1.1  ozaki 		default:
    381  1.1  ozaki 			if (WIFSIGNALED(status)) {
    382  1.1  ozaki 				if (WTERMSIG(status) == SIGTERM)
    383  1.3  ozaki 					ERRX0(EXIT_FAILURE,
    384  1.3  ozaki 					    "receiver failed and was killed" \
    385  1.3  ozaki 					    "by sender");
    386  1.1  ozaki 				else
    387  1.1  ozaki 					ERRX(EXIT_FAILURE,
    388  1.1  ozaki 					    "receiver got signaled (%s)",
    389  1.1  ozaki 					    strsignal(WTERMSIG(status)));
    390  1.1  ozaki 			} else if (WIFEXITED(status)) {
    391  1.1  ozaki 				if (WEXITSTATUS(status) != 0)
    392  1.1  ozaki 					ERRX(EXIT_FAILURE,
    393  1.1  ozaki 					    "receiver exited with status %d",
    394  1.1  ozaki 					    WEXITSTATUS(status));
    395  1.1  ozaki 			} else {
    396  1.1  ozaki 				ERRX(EXIT_FAILURE,
    397  1.1  ozaki 				    "receiver exited with unexpected status %d",
    398  1.1  ozaki 				    status);
    399  1.1  ozaki 			}
    400  1.1  ozaki 			break;
    401  1.1  ozaki 		}
    402  1.1  ozaki 		return;
    403  1.1  ozaki 	}
    404  1.1  ozaki }
    405  1.1  ozaki 
    406  1.1  ozaki #ifndef ATF
    407  1.1  ozaki int
    408  1.1  ozaki main(int argc, char *argv[])
    409  1.1  ozaki {
    410  1.1  ozaki 	const char *host, *port;
    411  1.1  ozaki 	int c;
    412  1.1  ozaki 	size_t n;
    413  1.1  ozaki 	bool conn, bug;
    414  1.1  ozaki 
    415  1.1  ozaki 	host = HOST_V4;
    416  1.1  ozaki 	port = PORT_V4;
    417  1.1  ozaki 	n = TOTAL;
    418  1.1  ozaki 	bug = conn = false;
    419  1.1  ozaki 
    420  1.1  ozaki 	while ((c = getopt(argc, argv, "46bcdmn:")) != -1)
    421  1.1  ozaki 		switch (c) {
    422  1.1  ozaki 		case '4':
    423  1.1  ozaki 			host = HOST_V4;
    424  1.1  ozaki 			port = PORT_V4;
    425  1.1  ozaki 			break;
    426  1.1  ozaki 		case '6':
    427  1.1  ozaki 			host = HOST_V6;
    428  1.1  ozaki 			port = PORT_V6;
    429  1.1  ozaki 			break;
    430  1.1  ozaki 		case 'b':
    431  1.1  ozaki 			bug = true;
    432  1.1  ozaki 			break;
    433  1.1  ozaki 		case 'c':
    434  1.1  ozaki 			conn = true;
    435  1.1  ozaki 			break;
    436  1.1  ozaki 		case 'd':
    437  1.1  ozaki 			debug++;
    438  1.1  ozaki 			break;
    439  1.1  ozaki 		case 'm':
    440  1.1  ozaki 			host = HOST_V4MAPPED;
    441  1.1  ozaki 			port = PORT_V4MAPPED;
    442  1.1  ozaki 			break;
    443  1.1  ozaki 		case 'n':
    444  1.1  ozaki 			n = atoi(optarg);
    445  1.1  ozaki 			break;
    446  1.1  ozaki 		default:
    447  1.1  ozaki 			fprintf(stderr, "Usage: %s [-cdm46] [-n <tot>]",
    448  1.1  ozaki 			    getprogname());
    449  1.1  ozaki 			return 1;
    450  1.1  ozaki 		}
    451  1.1  ozaki 
    452  1.1  ozaki 	run(host, port, n, conn, bug);
    453  1.1  ozaki 	return 0;
    454  1.1  ozaki }
    455  1.1  ozaki #else
    456  1.1  ozaki 
    457  1.1  ozaki ATF_TC(conninet4);
    458  1.1  ozaki ATF_TC_HEAD(conninet4, tc)
    459  1.1  ozaki {
    460  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks connected multicast for ipv4");
    461  1.1  ozaki }
    462  1.1  ozaki 
    463  1.1  ozaki ATF_TC_BODY(conninet4, tc)
    464  1.1  ozaki {
    465  1.1  ozaki 	run(HOST_V4, PORT_V4, TOTAL, true, false);
    466  1.1  ozaki }
    467  1.1  ozaki 
    468  1.1  ozaki ATF_TC(connmappedinet4);
    469  1.1  ozaki ATF_TC_HEAD(connmappedinet4, tc)
    470  1.1  ozaki {
    471  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks connected multicast for mapped ipv4");
    472  1.1  ozaki }
    473  1.1  ozaki 
    474  1.1  ozaki ATF_TC_BODY(connmappedinet4, tc)
    475  1.1  ozaki {
    476  1.1  ozaki 	run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, true, false);
    477  1.1  ozaki }
    478  1.1  ozaki 
    479  1.1  ozaki ATF_TC(connmappedbuginet4);
    480  1.1  ozaki ATF_TC_HEAD(connmappedbuginet4, tc)
    481  1.1  ozaki {
    482  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks connected multicast for mapped ipv4 using the v4 ioctls");
    483  1.1  ozaki }
    484  1.1  ozaki 
    485  1.1  ozaki ATF_TC_BODY(connmappedbuginet4, tc)
    486  1.1  ozaki {
    487  1.1  ozaki 	run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, true, true);
    488  1.1  ozaki }
    489  1.1  ozaki 
    490  1.1  ozaki ATF_TC(conninet6);
    491  1.1  ozaki ATF_TC_HEAD(conninet6, tc)
    492  1.1  ozaki {
    493  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks connected multicast for ipv6");
    494  1.1  ozaki }
    495  1.1  ozaki 
    496  1.1  ozaki ATF_TC_BODY(conninet6, tc)
    497  1.1  ozaki {
    498  1.1  ozaki 	run(HOST_V6, PORT_V6, TOTAL, true, false);
    499  1.1  ozaki }
    500  1.1  ozaki 
    501  1.1  ozaki ATF_TC(unconninet4);
    502  1.1  ozaki ATF_TC_HEAD(unconninet4, tc)
    503  1.1  ozaki {
    504  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for ipv4");
    505  1.1  ozaki }
    506  1.1  ozaki 
    507  1.1  ozaki ATF_TC_BODY(unconninet4, tc)
    508  1.1  ozaki {
    509  1.1  ozaki 	run(HOST_V4, PORT_V4, TOTAL, false, false);
    510  1.1  ozaki }
    511  1.1  ozaki 
    512  1.1  ozaki ATF_TC(unconnmappedinet4);
    513  1.1  ozaki ATF_TC_HEAD(unconnmappedinet4, tc)
    514  1.1  ozaki {
    515  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for mapped ipv4");
    516  1.1  ozaki }
    517  1.1  ozaki 
    518  1.1  ozaki ATF_TC_BODY(unconnmappedinet4, tc)
    519  1.1  ozaki {
    520  1.1  ozaki 	run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, false, false);
    521  1.1  ozaki }
    522  1.1  ozaki 
    523  1.1  ozaki ATF_TC(unconnmappedbuginet4);
    524  1.1  ozaki ATF_TC_HEAD(unconnmappedbuginet4, tc)
    525  1.1  ozaki {
    526  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for mapped ipv4 using the v4 ioctls");
    527  1.1  ozaki }
    528  1.1  ozaki 
    529  1.1  ozaki ATF_TC_BODY(unconnmappedbuginet4, tc)
    530  1.1  ozaki {
    531  1.1  ozaki 	run(HOST_V4MAPPED, PORT_V4MAPPED, TOTAL, false, true);
    532  1.1  ozaki }
    533  1.1  ozaki 
    534  1.1  ozaki ATF_TC(unconninet6);
    535  1.1  ozaki ATF_TC_HEAD(unconninet6, tc)
    536  1.1  ozaki {
    537  1.1  ozaki 	atf_tc_set_md_var(tc, "descr", "Checks unconnected multicast for ipv6");
    538  1.1  ozaki }
    539  1.1  ozaki 
    540  1.1  ozaki ATF_TC_BODY(unconninet6, tc)
    541  1.1  ozaki {
    542  1.1  ozaki 	run(HOST_V6, PORT_V6, TOTAL, false, false);
    543  1.1  ozaki }
    544  1.1  ozaki 
    545  1.1  ozaki ATF_TP_ADD_TCS(tp)
    546  1.1  ozaki {
    547  1.1  ozaki 	debug++;
    548  1.1  ozaki 	ATF_TP_ADD_TC(tp, conninet4);
    549  1.1  ozaki 	ATF_TP_ADD_TC(tp, connmappedinet4);
    550  1.1  ozaki 	ATF_TP_ADD_TC(tp, connmappedbuginet4);
    551  1.1  ozaki 	ATF_TP_ADD_TC(tp, conninet6);
    552  1.1  ozaki 	ATF_TP_ADD_TC(tp, unconninet4);
    553  1.1  ozaki 	ATF_TP_ADD_TC(tp, unconnmappedinet4);
    554  1.1  ozaki 	ATF_TP_ADD_TC(tp, unconnmappedbuginet4);
    555  1.1  ozaki 	ATF_TP_ADD_TC(tp, unconninet6);
    556  1.1  ozaki 
    557  1.1  ozaki 	return atf_no_error();
    558  1.1  ozaki }
    559  1.1  ozaki #endif
    560