2 * Copyright (c) 2014 Spectra Logic Corporation
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions, and the following disclaimer,
10 * without modification.
11 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
12 * substantially similar to the "NO WARRANTY" disclaimer below
13 * ("Disclaimer") and any redistribution must be conditioned upon
14 * including a substantially similar Disclaimer requirement for further
15 * binary redistribution.
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
21 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
26 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
27 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28 * POSSIBILITY OF SUCH DAMAGES.
34 #include <sys/types.h>
35 #include <sys/socket.h>
38 #include <netinet/in.h>
39 #include <arpa/inet.h>
48 #define MAX_IFADDRS 16
50 int debugging = false;
52 /* Data for mocking getifaddrs */
53 struct ifaddr_storage {
54 struct ifaddrs ifaddr;
55 struct sockaddr_storage addr;
56 struct sockaddr_storage mask;
57 struct sockaddr_storage bcast;
58 } mock_ifaddr_storage[MAX_IFADDRS];
59 struct ifaddrs *mock_ifaddrs = NULL;
62 /* Data for mocking listen_addr */
63 int bind_address_count = 0;
64 struct sockaddr* bind_addresses[MAX_IFADDRS];
66 /* Stub library functions */
68 freeifaddrs(struct ifaddrs *ifp __unused)
74 getifaddrs(struct ifaddrs **ifap)
81 mock_ifaddr4(const char* name, const char* addr, const char* mask,
82 const char* bcast, unsigned int flags, bool bind)
84 struct ifaddrs *ifaddr = &mock_ifaddr_storage[ifaddr_count].ifaddr;
85 struct sockaddr_in *in = (struct sockaddr_in*)
86 &mock_ifaddr_storage[ifaddr_count].addr;
87 struct sockaddr_in *mask_in = (struct sockaddr_in*)
88 &mock_ifaddr_storage[ifaddr_count].mask;
89 struct sockaddr_in *bcast_in = (struct sockaddr_in*)
90 &mock_ifaddr_storage[ifaddr_count].bcast;
92 in->sin_family = AF_INET;
94 in->sin_len = sizeof(*in);
95 in->sin_addr.s_addr = inet_addr(addr);
96 mask_in->sin_family = AF_INET;
97 mask_in->sin_port = 0;
98 mask_in->sin_len = sizeof(*mask_in);
99 mask_in->sin_addr.s_addr = inet_addr(mask);
100 bcast_in->sin_family = AF_INET;
101 bcast_in->sin_port = 0;
102 bcast_in->sin_len = sizeof(*bcast_in);
103 bcast_in->sin_addr.s_addr = inet_addr(bcast);
104 *ifaddr = (struct ifaddrs) {
106 .ifa_name = (char*) name,
108 .ifa_addr = (struct sockaddr*) in,
109 .ifa_netmask = (struct sockaddr*) mask_in,
110 .ifa_broadaddr = (struct sockaddr*) bcast_in,
111 .ifa_data = NULL, /* addrmerge doesn't care*/
114 if (ifaddr_count > 0)
115 mock_ifaddr_storage[ifaddr_count - 1].ifaddr.ifa_next = ifaddr;
117 mock_ifaddrs = &mock_ifaddr_storage[0].ifaddr;
119 /* Optionally simulate binding an ip ala "rpcbind -h foo" */
121 bind_addresses[bind_address_count] = (struct sockaddr*)in;
122 bind_address_count++;
128 mock_ifaddr6(const char* name, const char* addr, const char* mask,
129 const char* bcast, unsigned int flags, uint32_t scope_id, bool bind)
131 struct ifaddrs *ifaddr = &mock_ifaddr_storage[ifaddr_count].ifaddr;
132 struct sockaddr_in6 *in6 = (struct sockaddr_in6*)
133 &mock_ifaddr_storage[ifaddr_count].addr;
134 struct sockaddr_in6 *mask_in6 = (struct sockaddr_in6*)
135 &mock_ifaddr_storage[ifaddr_count].mask;
136 struct sockaddr_in6 *bcast_in6 = (struct sockaddr_in6*)
137 &mock_ifaddr_storage[ifaddr_count].bcast;
139 in6->sin6_family = AF_INET6;
141 in6->sin6_len = sizeof(*in6);
142 in6->sin6_scope_id = scope_id;
143 ATF_REQUIRE_EQ(1, inet_pton(AF_INET6, addr, (void*)&in6->sin6_addr));
144 mask_in6->sin6_family = AF_INET6;
145 mask_in6->sin6_port = 0;
146 mask_in6->sin6_len = sizeof(*mask_in6);
147 mask_in6->sin6_scope_id = scope_id;
148 ATF_REQUIRE_EQ(1, inet_pton(AF_INET6, mask,
149 (void*)&mask_in6->sin6_addr));
150 bcast_in6->sin6_family = AF_INET6;
151 bcast_in6->sin6_port = 0;
152 bcast_in6->sin6_len = sizeof(*bcast_in6);
153 bcast_in6->sin6_scope_id = scope_id;
154 ATF_REQUIRE_EQ(1, inet_pton(AF_INET6, bcast,
155 (void*)&bcast_in6->sin6_addr));
156 *ifaddr = (struct ifaddrs) {
158 .ifa_name = (char*) name,
160 .ifa_addr = (struct sockaddr*) in6,
161 .ifa_netmask = (struct sockaddr*) mask_in6,
162 .ifa_broadaddr = (struct sockaddr*) bcast_in6,
163 .ifa_data = NULL, /* addrmerge doesn't care*/
166 if (ifaddr_count > 0)
167 mock_ifaddr_storage[ifaddr_count - 1].ifaddr.ifa_next = ifaddr;
169 mock_ifaddrs = &mock_ifaddr_storage[0].ifaddr;
171 /* Optionally simulate binding an ip ala "rpcbind -h foo" */
173 bind_addresses[bind_address_count] = (struct sockaddr*)in6;
174 bind_address_count++;
179 mock_ifaddr6(const char* name __unused, const char* addr __unused,
180 const char* mask __unused, const char* bcast __unused,
181 unsigned int flags __unused, uint32_t scope_id __unused, bool bind __unused)
190 * This broadcast address looks wrong, but it's what getifaddrs(2)
191 * actually returns. It's invalid because IFF_BROADCAST is not set
193 mock_ifaddr4("lo0", "127.0.0.1", "255.0.0.0", "127.0.0.1",
194 IFF_LOOPBACK | IFF_UP | IFF_RUNNING | IFF_MULTICAST, false);
195 mock_ifaddr6("lo0", "::1", "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff",
197 IFF_LOOPBACK | IFF_UP | IFF_RUNNING | IFF_MULTICAST, 0, false);
203 mock_ifaddr4("igb0", "192.0.2.2", "255.255.255.128", "192.0.2.127",
204 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
206 mock_ifaddr6("igb0", "2001:db8::2", "ffff:ffff:ffff:ffff::",
207 "2001:db8::ffff:ffff:ffff:ffff",
208 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
210 /* Link local address */
211 mock_ifaddr6("igb0", "fe80::2", "ffff:ffff:ffff:ffff::",
212 "fe80::ffff:ffff:ffff:ffff",
213 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
217 /* On the same subnet as igb0 */
221 mock_ifaddr4("igb1", "192.0.2.3", "255.255.255.128", "192.0.2.127",
222 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
224 mock_ifaddr6("igb1", "2001:db8::3", "ffff:ffff:ffff:ffff::",
225 "2001:db8::ffff:ffff:ffff:ffff",
226 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
228 /* Link local address */
229 mock_ifaddr6("igb1", "fe80::3", "ffff:ffff:ffff:ffff::",
230 "fe80::ffff:ffff:ffff:ffff",
231 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
235 /* igb2 is on a different subnet than igb0 */
239 mock_ifaddr4("igb2", "192.0.2.130", "255.255.255.128", "192.0.2.255",
240 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
242 mock_ifaddr6("igb2", "2001:db8:1::2", "ffff:ffff:ffff:ffff::",
243 "2001:db8:1:0:ffff:ffff:ffff:ffff",
244 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
248 /* tun0 is a P2P interface */
252 mock_ifaddr4("tun0", "192.0.2.5", "255.255.255.255", "192.0.2.6",
253 IFF_UP | IFF_RUNNING | IFF_POINTOPOINT | IFF_MULTICAST, false);
254 mock_ifaddr6("tun0", "2001:db8::5",
255 "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff",
257 IFF_UP | IFF_RUNNING | IFF_POINTOPOINT | IFF_MULTICAST, 0, false);
263 mock_ifaddr4("mlxen0", "192.0.3.1", "255.255.255.128", "192.0.3.127",
264 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
266 /* Setting link local address before ipv6 address*/
267 mock_ifaddr6("mlxen0", "fe80::4", "ffff:ffff:ffff:ffff::",
268 "fe80::ffff:ffff:ffff:ffff",
269 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
271 mock_ifaddr6("mlxen0", "2001:db8::7", "ffff:ffff:ffff:ffff::",
272 "2001:db8::ffff:ffff:ffff:ffff",
273 IFF_UP | IFF_BROADCAST | IFF_RUNNING | IFF_SIMPLEX | IFF_MULTICAST,
277 /* Stub rpcbind functions */
279 listen_addr(const struct sockaddr *sa)
283 if (bind_address_count == 0)
286 for (i = 0; i < bind_address_count; i++) {
287 if (bind_addresses[i]->sa_family != sa->sa_family)
290 if (0 == memcmp(bind_addresses[i]->sa_data, sa->sa_data,
298 rpcbind_get_conf(const char* netid __unused)
300 /* Use static variables so we can return pointers to them */
301 static char* lookups = NULL;
302 static struct netconfig nconf_udp;
304 static struct netconfig nconf_udp6;
307 nconf_udp.nc_netid = "udp"; //netid_storage;
308 nconf_udp.nc_semantics = NC_TPI_CLTS;
309 nconf_udp.nc_flag = NC_VISIBLE;
310 nconf_udp.nc_protofmly = (char*)"inet";
311 nconf_udp.nc_proto = (char*)"udp";
312 nconf_udp.nc_device = (char*)"-";
313 nconf_udp.nc_nlookups = 0;
314 nconf_udp.nc_lookups = &lookups;
317 nconf_udp6.nc_netid = "udp6"; //netid_storage;
318 nconf_udp6.nc_semantics = NC_TPI_CLTS;
319 nconf_udp6.nc_flag = NC_VISIBLE;
320 nconf_udp6.nc_protofmly = (char*)"inet6";
321 nconf_udp6.nc_proto = (char*)"udp6";
322 nconf_udp6.nc_device = (char*)"-";
323 nconf_udp6.nc_nlookups = 0;
324 nconf_udp6.nc_lookups = &lookups;
327 if (0 == strncmp("udp", netid, sizeof("udp")))
330 else if (0 == strncmp("udp6", netid, sizeof("udp6")))
331 return (&nconf_udp6);
338 * Helper function used by most test cases
339 * param recvdstaddr If non-null, the uaddr on which the request was received
342 do_addrmerge4(const char* recvdstaddr)
344 struct netbuf caller;
345 struct sockaddr_in caller_in;
346 const char *serv_uaddr, *clnt_uaddr, *netid;
348 /* caller contains the client's IP address */
349 caller.maxlen = sizeof(struct sockaddr_storage);
350 caller.len = sizeof(caller_in);
351 caller_in.sin_family = AF_INET;
352 caller_in.sin_len = sizeof(caller_in);
353 caller_in.sin_port = 1234;
354 caller_in.sin_addr.s_addr = inet_addr("192.0.2.1");
355 caller.buf = (void*)&caller_in;
356 if (recvdstaddr != NULL)
357 clnt_uaddr = recvdstaddr;
359 clnt_uaddr = "192.0.2.1.3.46";
361 /* assume server is bound in INADDR_ANY port 814 */
362 serv_uaddr = "0.0.0.0.3.46";
365 return (addrmerge(&caller, serv_uaddr, clnt_uaddr, netid));
370 * Variant of do_addrmerge4 where the caller has an IPv6 address
371 * param recvdstaddr If non-null, the uaddr on which the request was received
374 do_addrmerge6(const char* recvdstaddr)
376 struct netbuf caller;
377 struct sockaddr_in6 caller_in6;
378 const char *serv_uaddr, *clnt_uaddr, *netid;
380 /* caller contains the client's IP address */
381 caller.maxlen = sizeof(struct sockaddr_storage);
382 caller.len = sizeof(caller_in6);
383 caller_in6.sin6_family = AF_INET6;
384 caller_in6.sin6_len = sizeof(caller_in6);
385 caller_in6.sin6_port = 1234;
386 ATF_REQUIRE_EQ(1, inet_pton(AF_INET6, "2001:db8::1",
387 (void*)&caller_in6.sin6_addr));
388 caller.buf = (void*)&caller_in6;
389 if (recvdstaddr != NULL)
390 clnt_uaddr = recvdstaddr;
392 clnt_uaddr = "2001:db8::1.3.46";
394 /* assume server is bound in INADDR_ANY port 814 */
395 serv_uaddr = "::1.3.46";
398 return (addrmerge(&caller, serv_uaddr, clnt_uaddr, netid));
401 /* Variant of do_addrmerge6 where the caller uses a link local address */
403 do_addrmerge6_ll(void)
405 struct netbuf caller;
406 struct sockaddr_in6 caller_in6;
407 const char *serv_uaddr, *clnt_uaddr, *netid;
409 /* caller contains the client's IP address */
410 caller.maxlen = sizeof(struct sockaddr_storage);
411 caller.len = sizeof(caller_in6);
412 caller_in6.sin6_family = AF_INET6;
413 caller_in6.sin6_len = sizeof(caller_in6);
414 caller_in6.sin6_port = 1234;
415 caller_in6.sin6_scope_id = 2; /* same as igb0 */
416 ATF_REQUIRE_EQ(1, inet_pton(AF_INET6, "fe80::beef",
417 (void*)&caller_in6.sin6_addr));
418 caller.buf = (void*)&caller_in6;
419 clnt_uaddr = "fe80::beef.3.46";
421 /* assume server is bound in INADDR_ANY port 814 */
422 serv_uaddr = "::1.3.46";
425 return (addrmerge(&caller, serv_uaddr, clnt_uaddr, netid));
429 ATF_TC_WITHOUT_HEAD(addrmerge_noifaddrs);
430 ATF_TC_BODY(addrmerge_noifaddrs, tc)
434 maddr = do_addrmerge4(NULL);
436 /* Since getifaddrs returns null, addrmerge must too */
437 ATF_CHECK_EQ(NULL, maddr);
440 ATF_TC_WITHOUT_HEAD(addrmerge_localhost_only);
441 ATF_TC_BODY(addrmerge_localhost_only, tc)
445 /* getifaddrs will return localhost only */
448 maddr = do_addrmerge4(NULL);
450 /* We must return localhost if there is nothing better */
451 ATF_REQUIRE(maddr != NULL);
452 ATF_CHECK_STREQ("127.0.0.1.3.46", maddr);
456 ATF_TC_WITHOUT_HEAD(addrmerge_singlehomed);
457 ATF_TC_BODY(addrmerge_singlehomed, tc)
461 /* getifaddrs will return one public address */
465 maddr = do_addrmerge4(NULL);
467 ATF_REQUIRE(maddr != NULL);
468 ATF_CHECK_STREQ("192.0.2.2.3.46", maddr);
472 ATF_TC_WITHOUT_HEAD(addrmerge_one_addr_on_each_subnet);
473 ATF_TC_BODY(addrmerge_one_addr_on_each_subnet, tc)
481 maddr = do_addrmerge4(NULL);
483 /* We must return the address on the caller's subnet */
484 ATF_REQUIRE(maddr != NULL);
485 ATF_CHECK_STREQ("192.0.2.2.3.46", maddr);
491 * Like addrmerge_one_addr_on_each_subnet, but getifaddrs returns a different
494 ATF_TC_WITHOUT_HEAD(addrmerge_one_addr_on_each_subnet_rev);
495 ATF_TC_BODY(addrmerge_one_addr_on_each_subnet_rev, tc)
499 /* getifaddrs will return one public address on each of two subnets */
504 maddr = do_addrmerge4(NULL);
506 /* We must return the address on the caller's subnet */
507 ATF_REQUIRE(maddr != NULL);
508 ATF_CHECK_STREQ("192.0.2.2.3.46", maddr);
512 ATF_TC_WITHOUT_HEAD(addrmerge_point2point);
513 ATF_TC_BODY(addrmerge_point2point, tc)
517 /* getifaddrs will return one normal and one p2p address */
522 maddr = do_addrmerge4(NULL);
524 /* addrmerge should disprefer P2P interfaces */
525 ATF_REQUIRE(maddr != NULL);
526 ATF_CHECK_STREQ("192.0.2.130.3.46", maddr);
530 /* Like addrerge_point2point, but getifaddrs returns a different order */
531 ATF_TC_WITHOUT_HEAD(addrmerge_point2point_rev);
532 ATF_TC_BODY(addrmerge_point2point_rev, tc)
536 /* getifaddrs will return one normal and one p2p address */
541 maddr = do_addrmerge4(NULL);
543 /* addrmerge should disprefer P2P interfaces */
544 ATF_REQUIRE(maddr != NULL);
545 ATF_CHECK_STREQ("192.0.2.130.3.46", maddr);
550 * Simulate using rpcbind -h to select just one ip when the subnet has
553 ATF_TC_WITHOUT_HEAD(addrmerge_bindip);
554 ATF_TC_BODY(addrmerge_bindip, tc)
558 /* getifaddrs will return one public address on each of two subnets */
563 maddr = do_addrmerge4(NULL);
565 /* We must return the address to which we are bound */
566 ATF_REQUIRE(maddr != NULL);
567 ATF_CHECK_STREQ("192.0.2.3.3.46", maddr);
571 /* Like addrmerge_bindip, but getifaddrs returns a different order */
572 ATF_TC_WITHOUT_HEAD(addrmerge_bindip_rev);
573 ATF_TC_BODY(addrmerge_bindip_rev, tc)
577 /* getifaddrs will return one public address on each of two subnets */
582 maddr = do_addrmerge4(NULL);
584 /* We must return the address to which we are bound */
585 ATF_REQUIRE(maddr != NULL);
586 ATF_CHECK_STREQ("192.0.2.3.3.46", maddr);
591 * The address on which the request was received is known, and is provided as
594 ATF_TC_WITHOUT_HEAD(addrmerge_recvdstaddr);
595 ATF_TC_BODY(addrmerge_recvdstaddr, tc)
603 maddr = do_addrmerge4("192.0.2.2.3.46");
605 /* We must return the address on which the request was received */
606 ATF_REQUIRE(maddr != NULL);
607 ATF_CHECK_STREQ("192.0.2.2.3.46", maddr);
611 ATF_TC_WITHOUT_HEAD(addrmerge_recvdstaddr_rev);
612 ATF_TC_BODY(addrmerge_recvdstaddr_rev, tc)
620 maddr = do_addrmerge4("192.0.2.2.3.46");
622 /* We must return the address on which the request was received */
623 ATF_REQUIRE(maddr != NULL);
624 ATF_CHECK_STREQ("192.0.2.2.3.46", maddr);
629 ATF_TC_WITHOUT_HEAD(addrmerge_localhost_only6);
630 ATF_TC_BODY(addrmerge_localhost_only6, tc)
634 /* getifaddrs will return localhost only */
637 maddr = do_addrmerge6(NULL);
639 /* We must return localhost if there is nothing better */
640 ATF_REQUIRE(maddr != NULL);
641 ATF_CHECK_STREQ("::1.3.46", maddr);
645 ATF_TC_WITHOUT_HEAD(addrmerge_singlehomed6);
646 ATF_TC_BODY(addrmerge_singlehomed6, tc)
650 /* getifaddrs will return one public address */
654 maddr = do_addrmerge6(NULL);
656 ATF_REQUIRE(maddr != NULL);
657 ATF_CHECK_STREQ("2001:db8::2.3.46", maddr);
661 ATF_TC_WITHOUT_HEAD(addrmerge_one_addr_on_each_subnet6);
662 ATF_TC_BODY(addrmerge_one_addr_on_each_subnet6, tc)
670 maddr = do_addrmerge6(NULL);
672 /* We must return the address on the caller's subnet */
673 ATF_REQUIRE(maddr != NULL);
674 ATF_CHECK_STREQ("2001:db8::2.3.46", maddr);
680 * Like addrmerge_one_addr_on_each_subnet6, but getifaddrs returns a different
683 ATF_TC_WITHOUT_HEAD(addrmerge_one_addr_on_each_subnet6_rev);
684 ATF_TC_BODY(addrmerge_one_addr_on_each_subnet6_rev, tc)
688 /* getifaddrs will return one public address on each of two subnets */
693 maddr = do_addrmerge6(NULL);
695 /* We must return the address on the caller's subnet */
696 ATF_REQUIRE(maddr != NULL);
697 ATF_CHECK_STREQ("2001:db8::2.3.46", maddr);
701 ATF_TC_WITHOUT_HEAD(addrmerge_point2point6);
702 ATF_TC_BODY(addrmerge_point2point6, tc)
706 /* getifaddrs will return one normal and one p2p address */
711 maddr = do_addrmerge6(NULL);
713 /* addrmerge should disprefer P2P interfaces */
714 ATF_REQUIRE(maddr != NULL);
715 ATF_CHECK_STREQ("2001:db8:1::2.3.46", maddr);
719 /* Like addrerge_point2point, but getifaddrs returns a different order */
720 ATF_TC_WITHOUT_HEAD(addrmerge_point2point6_rev);
721 ATF_TC_BODY(addrmerge_point2point6_rev, tc)
725 /* getifaddrs will return one normal and one p2p address */
730 maddr = do_addrmerge6(NULL);
732 /* addrmerge should disprefer P2P interfaces */
733 ATF_REQUIRE(maddr != NULL);
734 ATF_CHECK_STREQ("2001:db8:1::2.3.46", maddr);
738 ATF_TC_WITHOUT_HEAD(addrmerge_bindip6);
739 ATF_TC_BODY(addrmerge_bindip6, tc)
743 /* getifaddrs will return one public address on each of two subnets */
748 maddr = do_addrmerge6(NULL);
750 /* We must return the address to which we are bound */
751 ATF_REQUIRE(maddr != NULL);
752 ATF_CHECK_STREQ("2001:db8::3.3.46", maddr);
756 /* Like addrerge_bindip, but getifaddrs returns a different order */
757 ATF_TC_WITHOUT_HEAD(addrmerge_bindip6_rev);
758 ATF_TC_BODY(addrmerge_bindip6_rev, tc)
762 /* getifaddrs will return one public address on each of two subnets */
767 maddr = do_addrmerge6(NULL);
769 /* We must return the address to which we are bound */
770 ATF_REQUIRE(maddr != NULL);
771 ATF_CHECK_STREQ("2001:db8::3.3.46", maddr);
776 * IPv6 Link Local addresses with the same scope id as the caller, if the caller
777 * is also a link local address, should be preferred
779 ATF_TC_WITHOUT_HEAD(addrmerge_ipv6_linklocal);
780 ATF_TC_BODY(addrmerge_ipv6_linklocal, tc)
785 * getifaddrs will return two link local addresses with the same netmask
786 * and prefix but different scope IDs
792 maddr = do_addrmerge6_ll();
794 /* We must return the address to which we are bound */
795 ATF_REQUIRE(maddr != NULL);
796 ATF_CHECK_STREQ("fe80::2.3.46", maddr);
800 ATF_TC_WITHOUT_HEAD(addrmerge_ipv6_linklocal_rev);
801 ATF_TC_BODY(addrmerge_ipv6_linklocal_rev, tc)
806 * getifaddrs will return two link local addresses with the same netmask
807 * and prefix but different scope IDs
813 maddr = do_addrmerge6_ll();
815 /* We must return the address to which we are bound */
816 ATF_REQUIRE(maddr != NULL);
817 ATF_CHECK_STREQ("fe80::2.3.46", maddr);
821 ATF_TC_WITHOUT_HEAD(addrmerge_recvdstaddr6);
822 ATF_TC_BODY(addrmerge_recvdstaddr6, tc)
830 maddr = do_addrmerge6("2001:db8::2.3.46");
832 /* We must return the address on which the request was received */
833 ATF_REQUIRE(maddr != NULL);
834 ATF_CHECK_STREQ("2001:db8::2.3.46", maddr);
838 ATF_TC_WITHOUT_HEAD(addrmerge_recvdstaddr6_rev);
839 ATF_TC_BODY(addrmerge_recvdstaddr6_rev, tc)
847 maddr = do_addrmerge6("2001:db8::2.3.46");
849 /* We must return the address on which the request was received */
850 ATF_REQUIRE(maddr != NULL);
851 ATF_CHECK_STREQ("2001:db8::2.3.46", maddr);
855 ATF_TC_WITHOUT_HEAD(addrmerge_ipv6_other_subnet);
856 ATF_TC_BODY(addrmerge_ipv6_other_subnet, tc)
860 /* getifaddrs will return link local before normal ipv6 */
864 maddr = do_addrmerge6("2001:db8:1::1.3.46");
866 /* We must return the closest ipv6 address*/
867 ATF_REQUIRE(maddr != NULL);
868 ATF_CHECK_STREQ("2001:db8::7.3.46", maddr);
876 ATF_TP_ADD_TC(tp, addrmerge_noifaddrs);
877 ATF_TP_ADD_TC(tp, addrmerge_localhost_only);
878 ATF_TP_ADD_TC(tp, addrmerge_singlehomed);
879 ATF_TP_ADD_TC(tp, addrmerge_one_addr_on_each_subnet);
880 ATF_TP_ADD_TC(tp, addrmerge_one_addr_on_each_subnet_rev);
881 ATF_TP_ADD_TC(tp, addrmerge_point2point);
882 ATF_TP_ADD_TC(tp, addrmerge_point2point_rev);
883 ATF_TP_ADD_TC(tp, addrmerge_bindip);
884 ATF_TP_ADD_TC(tp, addrmerge_bindip_rev);
885 ATF_TP_ADD_TC(tp, addrmerge_recvdstaddr);
886 ATF_TP_ADD_TC(tp, addrmerge_recvdstaddr_rev);
888 ATF_TP_ADD_TC(tp, addrmerge_localhost_only6);
889 ATF_TP_ADD_TC(tp, addrmerge_singlehomed6);
890 ATF_TP_ADD_TC(tp, addrmerge_one_addr_on_each_subnet6);
891 ATF_TP_ADD_TC(tp, addrmerge_one_addr_on_each_subnet6_rev);
892 ATF_TP_ADD_TC(tp, addrmerge_point2point6);
893 ATF_TP_ADD_TC(tp, addrmerge_point2point6_rev);
894 ATF_TP_ADD_TC(tp, addrmerge_bindip6);
895 ATF_TP_ADD_TC(tp, addrmerge_bindip6_rev);
896 ATF_TP_ADD_TC(tp, addrmerge_ipv6_linklocal);
897 ATF_TP_ADD_TC(tp, addrmerge_ipv6_linklocal_rev);
898 ATF_TP_ADD_TC(tp, addrmerge_recvdstaddr6);
899 ATF_TP_ADD_TC(tp, addrmerge_recvdstaddr6_rev);
900 ATF_TP_ADD_TC(tp, addrmerge_ipv6_other_subnet);
903 return (atf_no_error());