xref: /xnu-11215.81.4/tests/inet6_addr_mode.c (revision d4514f0bc1d3f944c22d92e68b646ac3fb40d452)
1*d4514f0bSApple OSS Distributions /*
2*d4514f0bSApple OSS Distributions  * Copyright (c) 2023-2024 Apple Inc. All rights reserved.
3*d4514f0bSApple OSS Distributions  *
4*d4514f0bSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*d4514f0bSApple OSS Distributions  *
6*d4514f0bSApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*d4514f0bSApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*d4514f0bSApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*d4514f0bSApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*d4514f0bSApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*d4514f0bSApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*d4514f0bSApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*d4514f0bSApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*d4514f0bSApple OSS Distributions  *
15*d4514f0bSApple OSS Distributions  * Please obtain a copy of the License at
16*d4514f0bSApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*d4514f0bSApple OSS Distributions  *
18*d4514f0bSApple OSS Distributions  * The Original Code and all software distributed under the License are
19*d4514f0bSApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*d4514f0bSApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*d4514f0bSApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*d4514f0bSApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*d4514f0bSApple OSS Distributions  * Please see the License for the specific language governing rights and
24*d4514f0bSApple OSS Distributions  * limitations under the License.
25*d4514f0bSApple OSS Distributions  *
26*d4514f0bSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*d4514f0bSApple OSS Distributions  */
28*d4514f0bSApple OSS Distributions 
29*d4514f0bSApple OSS Distributions #include <darwintest.h>
30*d4514f0bSApple OSS Distributions 
31*d4514f0bSApple OSS Distributions #include <sys/ioctl.h>
32*d4514f0bSApple OSS Distributions 
33*d4514f0bSApple OSS Distributions #include <stdlib.h>
34*d4514f0bSApple OSS Distributions #include <string.h>
35*d4514f0bSApple OSS Distributions #include <strings.h>
36*d4514f0bSApple OSS Distributions 
37*d4514f0bSApple OSS Distributions #include "net_test_lib.h"
38*d4514f0bSApple OSS Distributions 
39*d4514f0bSApple OSS Distributions T_GLOBAL_META(
40*d4514f0bSApple OSS Distributions 	T_META_NAMESPACE("xnu.net"),
41*d4514f0bSApple OSS Distributions 	T_META_ASROOT(true),
42*d4514f0bSApple OSS Distributions 	T_META_RADAR_COMPONENT_NAME("xnu"),
43*d4514f0bSApple OSS Distributions 	T_META_RADAR_COMPONENT_VERSION("networking"),
44*d4514f0bSApple OSS Distributions 	T_META_CHECK_LEAKS(false));
45*d4514f0bSApple OSS Distributions 
46*d4514f0bSApple OSS Distributions static char ifname1[IF_NAMESIZE];
47*d4514f0bSApple OSS Distributions 
48*d4514f0bSApple OSS Distributions /**
49*d4514f0bSApple OSS Distributions **  stolen from bootp/bootplib/util.c
50*d4514f0bSApple OSS Distributions **
51*d4514f0bSApple OSS Distributions **/
52*d4514f0bSApple OSS Distributions 
53*d4514f0bSApple OSS Distributions #define ROUNDUP(a) \
54*d4514f0bSApple OSS Distributions     ((a) > 0 ? (1 + (((a) - 1) | (sizeof(u_int32_t) - 1))) : sizeof(u_int32_t))
55*d4514f0bSApple OSS Distributions 
56*d4514f0bSApple OSS Distributions static int
rt_xaddrs(char * cp,const char * cplim,struct rt_addrinfo * rtinfo)57*d4514f0bSApple OSS Distributions rt_xaddrs(char * cp, const char * cplim, struct rt_addrinfo * rtinfo)
58*d4514f0bSApple OSS Distributions {
59*d4514f0bSApple OSS Distributions 	int         i;
60*d4514f0bSApple OSS Distributions 	struct sockaddr *   sa;
61*d4514f0bSApple OSS Distributions 
62*d4514f0bSApple OSS Distributions 	bzero(rtinfo->rti_info, sizeof(rtinfo->rti_info));
63*d4514f0bSApple OSS Distributions 	for (i = 0; (i < RTAX_MAX) && (cp < cplim); i++) {
64*d4514f0bSApple OSS Distributions 		if ((rtinfo->rti_addrs & (1 << i)) == 0) {
65*d4514f0bSApple OSS Distributions 			continue;
66*d4514f0bSApple OSS Distributions 		}
67*d4514f0bSApple OSS Distributions 		sa = (struct sockaddr *)cp;
68*d4514f0bSApple OSS Distributions 		if ((cp + sa->sa_len) > cplim) {
69*d4514f0bSApple OSS Distributions 			return EINVAL;
70*d4514f0bSApple OSS Distributions 		}
71*d4514f0bSApple OSS Distributions 		rtinfo->rti_info[i] = sa;
72*d4514f0bSApple OSS Distributions 		cp += ROUNDUP(sa->sa_len);
73*d4514f0bSApple OSS Distributions 	}
74*d4514f0bSApple OSS Distributions 	return 0;
75*d4514f0bSApple OSS Distributions }
76*d4514f0bSApple OSS Distributions 
77*d4514f0bSApple OSS Distributions /**
78*d4514f0bSApple OSS Distributions **  stolen from bootp/IPConfiguration.bproj/iputil.c
79*d4514f0bSApple OSS Distributions **
80*d4514f0bSApple OSS Distributions ** inet6_addrlist_*
81*d4514f0bSApple OSS Distributions **/
82*d4514f0bSApple OSS Distributions 
83*d4514f0bSApple OSS Distributions #define s6_addr16 __u6_addr.__u6_addr16
84*d4514f0bSApple OSS Distributions 
85*d4514f0bSApple OSS Distributions static char *
copy_if_info(unsigned int if_index,int af,int * ret_len_p)86*d4514f0bSApple OSS Distributions copy_if_info(unsigned int if_index, int af, int *ret_len_p)
87*d4514f0bSApple OSS Distributions {
88*d4514f0bSApple OSS Distributions 	char *          buf = NULL;
89*d4514f0bSApple OSS Distributions 	size_t          buf_len = 0;
90*d4514f0bSApple OSS Distributions 	int             mib[6];
91*d4514f0bSApple OSS Distributions 
92*d4514f0bSApple OSS Distributions 	mib[0] = CTL_NET;
93*d4514f0bSApple OSS Distributions 	mib[1] = PF_ROUTE;
94*d4514f0bSApple OSS Distributions 	mib[2] = 0;
95*d4514f0bSApple OSS Distributions 	mib[3] = af;
96*d4514f0bSApple OSS Distributions 	mib[4] = NET_RT_IFLIST;
97*d4514f0bSApple OSS Distributions 	mib[5] = (int)if_index;
98*d4514f0bSApple OSS Distributions 
99*d4514f0bSApple OSS Distributions 	*ret_len_p = 0;
100*d4514f0bSApple OSS Distributions 	if (sysctl(mib, 6, NULL, &buf_len, NULL, 0) < 0) {
101*d4514f0bSApple OSS Distributions 		fprintf(stderr, "sysctl() size failed: %s", strerror(errno));
102*d4514f0bSApple OSS Distributions 		goto failed;
103*d4514f0bSApple OSS Distributions 	}
104*d4514f0bSApple OSS Distributions 	buf_len *= 2; /* just in case something changes */
105*d4514f0bSApple OSS Distributions 	buf = malloc(buf_len);
106*d4514f0bSApple OSS Distributions 	if (sysctl(mib, 6, buf, &buf_len, NULL, 0) < 0) {
107*d4514f0bSApple OSS Distributions 		free(buf);
108*d4514f0bSApple OSS Distributions 		buf = NULL;
109*d4514f0bSApple OSS Distributions 		fprintf(stderr, "sysctl() failed: %s", strerror(errno));
110*d4514f0bSApple OSS Distributions 		goto failed;
111*d4514f0bSApple OSS Distributions 	}
112*d4514f0bSApple OSS Distributions 	*ret_len_p = (int)buf_len;
113*d4514f0bSApple OSS Distributions 
114*d4514f0bSApple OSS Distributions failed:
115*d4514f0bSApple OSS Distributions 	return buf;
116*d4514f0bSApple OSS Distributions }
117*d4514f0bSApple OSS Distributions 
118*d4514f0bSApple OSS Distributions static bool
inet6_get_linklocal_address(unsigned int if_index,struct in6_addr * ret_addr)119*d4514f0bSApple OSS Distributions inet6_get_linklocal_address(unsigned int if_index, struct in6_addr *ret_addr)
120*d4514f0bSApple OSS Distributions {
121*d4514f0bSApple OSS Distributions 	char *          buf = NULL;
122*d4514f0bSApple OSS Distributions 	char *          buf_end;
123*d4514f0bSApple OSS Distributions 	int             buf_len;
124*d4514f0bSApple OSS Distributions 	bool            found = FALSE;
125*d4514f0bSApple OSS Distributions 	char *scan;
126*d4514f0bSApple OSS Distributions 	struct rt_msghdr *rtm;
127*d4514f0bSApple OSS Distributions 
128*d4514f0bSApple OSS Distributions 	bzero(ret_addr, sizeof(*ret_addr));
129*d4514f0bSApple OSS Distributions 	buf = copy_if_info(if_index, AF_INET6, &buf_len);
130*d4514f0bSApple OSS Distributions 	if (buf == NULL) {
131*d4514f0bSApple OSS Distributions 		goto done;
132*d4514f0bSApple OSS Distributions 	}
133*d4514f0bSApple OSS Distributions 	buf_end = buf + buf_len;
134*d4514f0bSApple OSS Distributions 	for (scan = buf; scan < buf_end; scan += rtm->rtm_msglen) {
135*d4514f0bSApple OSS Distributions 		struct ifa_msghdr * ifam;
136*d4514f0bSApple OSS Distributions 		struct rt_addrinfo  info;
137*d4514f0bSApple OSS Distributions 
138*d4514f0bSApple OSS Distributions 		/* ALIGN: buf aligned (from calling copy_if_info), scan aligned,
139*d4514f0bSApple OSS Distributions 		 * cast ok. */
140*d4514f0bSApple OSS Distributions 		rtm = (struct rt_msghdr *)(void *)scan;
141*d4514f0bSApple OSS Distributions 		if (rtm->rtm_version != RTM_VERSION) {
142*d4514f0bSApple OSS Distributions 			continue;
143*d4514f0bSApple OSS Distributions 		}
144*d4514f0bSApple OSS Distributions 		if (rtm->rtm_type == RTM_NEWADDR) {
145*d4514f0bSApple OSS Distributions 			errno_t         error;
146*d4514f0bSApple OSS Distributions 			struct sockaddr_in6 *sin6_p;
147*d4514f0bSApple OSS Distributions 
148*d4514f0bSApple OSS Distributions 			ifam = (struct ifa_msghdr *)rtm;
149*d4514f0bSApple OSS Distributions 			info.rti_addrs = ifam->ifam_addrs;
150*d4514f0bSApple OSS Distributions 			error = rt_xaddrs((char *)(ifam + 1),
151*d4514f0bSApple OSS Distributions 			    ((char *)ifam) + ifam->ifam_msglen,
152*d4514f0bSApple OSS Distributions 			    &info);
153*d4514f0bSApple OSS Distributions 			if (error) {
154*d4514f0bSApple OSS Distributions 				fprintf(stderr, "couldn't extract rt_addrinfo %s (%d)\n",
155*d4514f0bSApple OSS Distributions 				    strerror(error), error);
156*d4514f0bSApple OSS Distributions 				goto done;
157*d4514f0bSApple OSS Distributions 			}
158*d4514f0bSApple OSS Distributions 			/* ALIGN: info.rti_info aligned (sockaddr), cast ok. */
159*d4514f0bSApple OSS Distributions 			sin6_p = (struct sockaddr_in6 *)(void *)info.rti_info[RTAX_IFA];
160*d4514f0bSApple OSS Distributions 			if (sin6_p == NULL
161*d4514f0bSApple OSS Distributions 			    || sin6_p->sin6_len < sizeof(struct sockaddr_in6)) {
162*d4514f0bSApple OSS Distributions 				continue;
163*d4514f0bSApple OSS Distributions 			}
164*d4514f0bSApple OSS Distributions 			if (IN6_IS_ADDR_LINKLOCAL(&sin6_p->sin6_addr)) {
165*d4514f0bSApple OSS Distributions 				*ret_addr = sin6_p->sin6_addr;
166*d4514f0bSApple OSS Distributions 				ret_addr->s6_addr16[1] = 0; /* mask scope id */
167*d4514f0bSApple OSS Distributions 				found = TRUE;
168*d4514f0bSApple OSS Distributions 				break;
169*d4514f0bSApple OSS Distributions 			}
170*d4514f0bSApple OSS Distributions 		}
171*d4514f0bSApple OSS Distributions 	}
172*d4514f0bSApple OSS Distributions 
173*d4514f0bSApple OSS Distributions done:
174*d4514f0bSApple OSS Distributions 	if (buf != NULL) {
175*d4514f0bSApple OSS Distributions 		free(buf);
176*d4514f0bSApple OSS Distributions 	}
177*d4514f0bSApple OSS Distributions 	return found;
178*d4514f0bSApple OSS Distributions }
179*d4514f0bSApple OSS Distributions 
180*d4514f0bSApple OSS Distributions static void
cleanup(void)181*d4514f0bSApple OSS Distributions cleanup(void)
182*d4514f0bSApple OSS Distributions {
183*d4514f0bSApple OSS Distributions 	if (ifname1[0] != '\0') {
184*d4514f0bSApple OSS Distributions 		T_LOG("ifnet_destroy %s", ifname1);
185*d4514f0bSApple OSS Distributions 		(void)ifnet_destroy(ifname1, false);
186*d4514f0bSApple OSS Distributions 	}
187*d4514f0bSApple OSS Distributions }
188*d4514f0bSApple OSS Distributions 
189*d4514f0bSApple OSS Distributions static void
set_sockaddr_in6(struct sockaddr_in6 * sin6_p,const struct in6_addr * addr)190*d4514f0bSApple OSS Distributions set_sockaddr_in6(struct sockaddr_in6 *sin6_p, const struct in6_addr *addr)
191*d4514f0bSApple OSS Distributions {
192*d4514f0bSApple OSS Distributions 	sin6_p->sin6_family = AF_INET6;
193*d4514f0bSApple OSS Distributions 	sin6_p->sin6_len = sizeof(struct sockaddr_in6);
194*d4514f0bSApple OSS Distributions 	sin6_p->sin6_addr = *addr;
195*d4514f0bSApple OSS Distributions 	return;
196*d4514f0bSApple OSS Distributions }
197*d4514f0bSApple OSS Distributions 
198*d4514f0bSApple OSS Distributions static int
inet6_difaddr(const char * name,const struct in6_addr * addr)199*d4514f0bSApple OSS Distributions inet6_difaddr(const char *name, const struct in6_addr *addr)
200*d4514f0bSApple OSS Distributions {
201*d4514f0bSApple OSS Distributions 	struct in6_ifreq    ifr;
202*d4514f0bSApple OSS Distributions 	int             s6 = inet6_dgram_socket_get();
203*d4514f0bSApple OSS Distributions 
204*d4514f0bSApple OSS Distributions 	bzero(&ifr, sizeof(ifr));
205*d4514f0bSApple OSS Distributions 	strncpy(ifr.ifr_name, name, sizeof(ifr.ifr_name));
206*d4514f0bSApple OSS Distributions 	if (addr != NULL) {
207*d4514f0bSApple OSS Distributions 		set_sockaddr_in6(&ifr.ifr_ifru.ifru_addr, addr);
208*d4514f0bSApple OSS Distributions 	}
209*d4514f0bSApple OSS Distributions 	return ioctl(s6, SIOCDIFADDR_IN6, &ifr);
210*d4514f0bSApple OSS Distributions }
211*d4514f0bSApple OSS Distributions 
212*d4514f0bSApple OSS Distributions static void
in6_len2mask(struct in6_addr * mask,int len)213*d4514f0bSApple OSS Distributions in6_len2mask(struct in6_addr *mask, int len)
214*d4514f0bSApple OSS Distributions {
215*d4514f0bSApple OSS Distributions 	int i;
216*d4514f0bSApple OSS Distributions 
217*d4514f0bSApple OSS Distributions 	bzero(mask, sizeof(*mask));
218*d4514f0bSApple OSS Distributions 	for (i = 0; i < len / 8; i++) {
219*d4514f0bSApple OSS Distributions 		mask->s6_addr[i] = 0xff;
220*d4514f0bSApple OSS Distributions 	}
221*d4514f0bSApple OSS Distributions 	if (len % 8) {
222*d4514f0bSApple OSS Distributions 		mask->s6_addr[i] = (0xff00 >> (len % 8)) & 0xff;
223*d4514f0bSApple OSS Distributions 	}
224*d4514f0bSApple OSS Distributions }
225*d4514f0bSApple OSS Distributions 
226*d4514f0bSApple OSS Distributions static int
inet6_aifaddr(const char * name,const struct in6_addr * addr,const struct in6_addr * dstaddr,int prefix_length,int flags,u_int32_t valid_lifetime,u_int32_t preferred_lifetime)227*d4514f0bSApple OSS Distributions inet6_aifaddr(const char *name, const struct in6_addr *addr,
228*d4514f0bSApple OSS Distributions     const struct in6_addr *dstaddr, int prefix_length,
229*d4514f0bSApple OSS Distributions     int flags,
230*d4514f0bSApple OSS Distributions     u_int32_t valid_lifetime,
231*d4514f0bSApple OSS Distributions     u_int32_t preferred_lifetime)
232*d4514f0bSApple OSS Distributions {
233*d4514f0bSApple OSS Distributions 	struct in6_aliasreq ifra_in6;
234*d4514f0bSApple OSS Distributions 	int             s6 = inet6_dgram_socket_get();
235*d4514f0bSApple OSS Distributions 
236*d4514f0bSApple OSS Distributions 	bzero(&ifra_in6, sizeof(ifra_in6));
237*d4514f0bSApple OSS Distributions 	strncpy(ifra_in6.ifra_name, name, sizeof(ifra_in6.ifra_name));
238*d4514f0bSApple OSS Distributions 	ifra_in6.ifra_lifetime.ia6t_vltime = valid_lifetime;
239*d4514f0bSApple OSS Distributions 	ifra_in6.ifra_lifetime.ia6t_pltime = preferred_lifetime;
240*d4514f0bSApple OSS Distributions 	ifra_in6.ifra_flags = flags;
241*d4514f0bSApple OSS Distributions 	if (addr != NULL) {
242*d4514f0bSApple OSS Distributions 		set_sockaddr_in6(&ifra_in6.ifra_addr, addr);
243*d4514f0bSApple OSS Distributions 	}
244*d4514f0bSApple OSS Distributions 
245*d4514f0bSApple OSS Distributions 	if (dstaddr != NULL) {
246*d4514f0bSApple OSS Distributions 		set_sockaddr_in6(&ifra_in6.ifra_dstaddr, dstaddr);
247*d4514f0bSApple OSS Distributions 	}
248*d4514f0bSApple OSS Distributions 
249*d4514f0bSApple OSS Distributions 	if (prefix_length != 0) {
250*d4514f0bSApple OSS Distributions 		struct in6_addr     prefixmask;
251*d4514f0bSApple OSS Distributions 
252*d4514f0bSApple OSS Distributions 		in6_len2mask(&prefixmask, prefix_length);
253*d4514f0bSApple OSS Distributions 		set_sockaddr_in6(&ifra_in6.ifra_prefixmask, &prefixmask);
254*d4514f0bSApple OSS Distributions 	}
255*d4514f0bSApple OSS Distributions 
256*d4514f0bSApple OSS Distributions 	return ioctl(s6, SIOCAIFADDR_IN6, &ifra_in6);
257*d4514f0bSApple OSS Distributions }
258*d4514f0bSApple OSS Distributions 
259*d4514f0bSApple OSS Distributions static void
create_fake_interface(void)260*d4514f0bSApple OSS Distributions create_fake_interface(void)
261*d4514f0bSApple OSS Distributions {
262*d4514f0bSApple OSS Distributions 	int     error;
263*d4514f0bSApple OSS Distributions 
264*d4514f0bSApple OSS Distributions 	strlcpy(ifname1, FETH_NAME, sizeof(ifname1));
265*d4514f0bSApple OSS Distributions 	error = ifnet_create_2(ifname1, sizeof(ifname1));
266*d4514f0bSApple OSS Distributions 	if (error != 0) {
267*d4514f0bSApple OSS Distributions 		ifname1[0] = '\0';
268*d4514f0bSApple OSS Distributions 		T_ASSERT_POSIX_SUCCESS(error, "ifnet_create_2");
269*d4514f0bSApple OSS Distributions 	}
270*d4514f0bSApple OSS Distributions 	T_LOG("created %s", ifname1);
271*d4514f0bSApple OSS Distributions }
272*d4514f0bSApple OSS Distributions 
273*d4514f0bSApple OSS Distributions T_DECL(inet6_addr_mode_auto_to_manual, "inet6 address mode-switching (auto -> manual)")
274*d4514f0bSApple OSS Distributions {
275*d4514f0bSApple OSS Distributions 	struct in6_addr lladdr;
276*d4514f0bSApple OSS Distributions 	struct in6_addr newaddr;
277*d4514f0bSApple OSS Distributions 	unsigned int    if_index;
278*d4514f0bSApple OSS Distributions 
279*d4514f0bSApple OSS Distributions 	T_ATEND(cleanup);
280*d4514f0bSApple OSS Distributions 
281*d4514f0bSApple OSS Distributions 	create_fake_interface();
282*d4514f0bSApple OSS Distributions 	ifnet_start_ipv6(ifname1);
283*d4514f0bSApple OSS Distributions 
284*d4514f0bSApple OSS Distributions 	if_index = if_nametoindex(ifname1);
285*d4514f0bSApple OSS Distributions 	T_EXPECT_GT(if_index, 0, NULL);
286*d4514f0bSApple OSS Distributions 	T_ASSERT_EQ(inet6_get_linklocal_address(if_index, &lladdr), 1, NULL);
287*d4514f0bSApple OSS Distributions 
288*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_aifaddr(ifname1, &lladdr, NULL, 64, 123, ND6_INFINITE_LIFETIME, ND6_INFINITE_LIFETIME), NULL);
289*d4514f0bSApple OSS Distributions 
290*d4514f0bSApple OSS Distributions 	/* Create address as an autoconfed address */
291*d4514f0bSApple OSS Distributions 	T_ASSERT_EQ(inet_pton(AF_INET6, "2001:db8::3", &newaddr), 1, NULL);
292*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_aifaddr(ifname1, &newaddr, NULL, 64, (IN6_IFF_AUTOCONF | IN6_IFF_TEMPORARY), ND6_INFINITE_LIFETIME, ND6_INFINITE_LIFETIME), NULL);
293*d4514f0bSApple OSS Distributions 
294*d4514f0bSApple OSS Distributions 	/* Now mark it as manual */
295*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_aifaddr(ifname1, &newaddr, NULL, 64, 0, ND6_INFINITE_LIFETIME, ND6_INFINITE_LIFETIME), NULL);
296*d4514f0bSApple OSS Distributions 
297*d4514f0bSApple OSS Distributions 	/* Deleting address should NOT result in panic */
298*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_difaddr(ifname1, &newaddr), NULL);
299*d4514f0bSApple OSS Distributions }
300*d4514f0bSApple OSS Distributions 
301*d4514f0bSApple OSS Distributions T_DECL(inet6_addr_mode_manual_to_auto, "inet6 address mode-switching (manual -> auto)")
302*d4514f0bSApple OSS Distributions {
303*d4514f0bSApple OSS Distributions 	struct in6_addr lladdr;
304*d4514f0bSApple OSS Distributions 	struct in6_addr newaddr;
305*d4514f0bSApple OSS Distributions 	unsigned int    if_index;
306*d4514f0bSApple OSS Distributions 
307*d4514f0bSApple OSS Distributions 	T_ATEND(cleanup);
308*d4514f0bSApple OSS Distributions 	create_fake_interface();
309*d4514f0bSApple OSS Distributions 
310*d4514f0bSApple OSS Distributions 	T_LOG("created %s", ifname1);
311*d4514f0bSApple OSS Distributions 
312*d4514f0bSApple OSS Distributions 	ifnet_start_ipv6(ifname1);
313*d4514f0bSApple OSS Distributions 
314*d4514f0bSApple OSS Distributions 	if_index = if_nametoindex(ifname1);
315*d4514f0bSApple OSS Distributions 	T_EXPECT_GT(if_index, 0, NULL);
316*d4514f0bSApple OSS Distributions 	T_ASSERT_EQ(inet6_get_linklocal_address(if_index, &lladdr), 1, NULL);
317*d4514f0bSApple OSS Distributions 
318*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_aifaddr(ifname1, &lladdr, NULL, 64, 123, ND6_INFINITE_LIFETIME, ND6_INFINITE_LIFETIME), NULL);
319*d4514f0bSApple OSS Distributions 
320*d4514f0bSApple OSS Distributions 	/* Create address as a manual address */
321*d4514f0bSApple OSS Distributions 	T_ASSERT_EQ(inet_pton(AF_INET6, "2001:db8::1", &newaddr), 1, NULL);
322*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_aifaddr(ifname1, &newaddr, NULL, 64, 0, ND6_INFINITE_LIFETIME, ND6_INFINITE_LIFETIME), NULL);
323*d4514f0bSApple OSS Distributions 
324*d4514f0bSApple OSS Distributions 	/* Now make it autoconfed */
325*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_aifaddr(ifname1, &newaddr, NULL, 64, (IN6_IFF_AUTOCONF | IN6_IFF_TEMPORARY), ND6_INFINITE_LIFETIME, ND6_INFINITE_LIFETIME), NULL);
326*d4514f0bSApple OSS Distributions 
327*d4514f0bSApple OSS Distributions 	/* Deleting address should NOT result in panic */
328*d4514f0bSApple OSS Distributions 	T_ASSERT_POSIX_SUCCESS(inet6_difaddr(ifname1, &newaddr), NULL);
329*d4514f0bSApple OSS Distributions }
330