1*5e3eaea3SApple OSS Distributions /*
2*5e3eaea3SApple OSS Distributions * Copyright (c) 2020-2022 Apple Inc. All rights reserved.
3*5e3eaea3SApple OSS Distributions *
4*5e3eaea3SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*5e3eaea3SApple OSS Distributions *
6*5e3eaea3SApple OSS Distributions * This file contains Original Code and/or Modifications of Original Code
7*5e3eaea3SApple OSS Distributions * as defined in and that are subject to the Apple Public Source License
8*5e3eaea3SApple OSS Distributions * Version 2.0 (the 'License'). You may not use this file except in
9*5e3eaea3SApple OSS Distributions * compliance with the License. The rights granted to you under the License
10*5e3eaea3SApple OSS Distributions * may not be used to create, or enable the creation or redistribution of,
11*5e3eaea3SApple OSS Distributions * unlawful or unlicensed copies of an Apple operating system, or to
12*5e3eaea3SApple OSS Distributions * circumvent, violate, or enable the circumvention or violation of, any
13*5e3eaea3SApple OSS Distributions * terms of an Apple operating system software license agreement.
14*5e3eaea3SApple OSS Distributions *
15*5e3eaea3SApple OSS Distributions * Please obtain a copy of the License at
16*5e3eaea3SApple OSS Distributions * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*5e3eaea3SApple OSS Distributions *
18*5e3eaea3SApple OSS Distributions * The Original Code and all software distributed under the License are
19*5e3eaea3SApple OSS Distributions * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*5e3eaea3SApple OSS Distributions * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*5e3eaea3SApple OSS Distributions * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*5e3eaea3SApple OSS Distributions * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*5e3eaea3SApple OSS Distributions * Please see the License for the specific language governing rights and
24*5e3eaea3SApple OSS Distributions * limitations under the License.
25*5e3eaea3SApple OSS Distributions *
26*5e3eaea3SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*5e3eaea3SApple OSS Distributions */
28*5e3eaea3SApple OSS Distributions
29*5e3eaea3SApple OSS Distributions /* -*- compile-command: "xcrun --sdk iphoneos.internal make sioc-if-addr-bounds" -*- */
30*5e3eaea3SApple OSS Distributions
31*5e3eaea3SApple OSS Distributions #include <sys/ioctl.h>
32*5e3eaea3SApple OSS Distributions #include <sys/socket.h>
33*5e3eaea3SApple OSS Distributions #include <sys/sockio.h>
34*5e3eaea3SApple OSS Distributions
35*5e3eaea3SApple OSS Distributions #include <net/if.h>
36*5e3eaea3SApple OSS Distributions #include <net/if_dl.h>
37*5e3eaea3SApple OSS Distributions
38*5e3eaea3SApple OSS Distributions #include <netinet/in.h>
39*5e3eaea3SApple OSS Distributions #include <netinet/in_var.h>
40*5e3eaea3SApple OSS Distributions
41*5e3eaea3SApple OSS Distributions #include <string.h>
42*5e3eaea3SApple OSS Distributions #include <unistd.h>
43*5e3eaea3SApple OSS Distributions
44*5e3eaea3SApple OSS Distributions #include <arpa/inet.h>
45*5e3eaea3SApple OSS Distributions
46*5e3eaea3SApple OSS Distributions #include <darwintest.h>
47*5e3eaea3SApple OSS Distributions #include <darwintest_utils.h>
48*5e3eaea3SApple OSS Distributions
49*5e3eaea3SApple OSS Distributions #include <uuid/uuid.h>
50*5e3eaea3SApple OSS Distributions
51*5e3eaea3SApple OSS Distributions #include <ifaddrs.h>
52*5e3eaea3SApple OSS Distributions
53*5e3eaea3SApple OSS Distributions #include <arpa/inet.h>
54*5e3eaea3SApple OSS Distributions
55*5e3eaea3SApple OSS Distributions #include "ioc_str.h"
56*5e3eaea3SApple OSS Distributions
57*5e3eaea3SApple OSS Distributions T_GLOBAL_META(T_META_NAMESPACE("xnu.net"));
58*5e3eaea3SApple OSS Distributions
59*5e3eaea3SApple OSS Distributions #ifndef STRINGIFY
60*5e3eaea3SApple OSS Distributions #define __STR(x) #x /* just a helper macro */
61*5e3eaea3SApple OSS Distributions #define STRINGIFY(x) __STR(x)
62*5e3eaea3SApple OSS Distributions #endif /* STRINGIFY */
63*5e3eaea3SApple OSS Distributions
64*5e3eaea3SApple OSS Distributions #define IF_NAME "bridge"
65*5e3eaea3SApple OSS Distributions
66*5e3eaea3SApple OSS Distributions /* On some platforms with DEBUG kernel, we need to wait a while */
67*5e3eaea3SApple OSS Distributions #define SIFCREATE_RETRY 600
68*5e3eaea3SApple OSS Distributions
69*5e3eaea3SApple OSS Distributions #define PATTERN_SIZE 8
70*5e3eaea3SApple OSS Distributions
71*5e3eaea3SApple OSS Distributions static int
ifnet_destroy(int s,const char * ifname,bool fail_on_error)72*5e3eaea3SApple OSS Distributions ifnet_destroy(int s, const char * ifname, bool fail_on_error)
73*5e3eaea3SApple OSS Distributions {
74*5e3eaea3SApple OSS Distributions int err;
75*5e3eaea3SApple OSS Distributions struct ifreq ifr;
76*5e3eaea3SApple OSS Distributions
77*5e3eaea3SApple OSS Distributions bzero(&ifr, sizeof(ifr));
78*5e3eaea3SApple OSS Distributions strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
79*5e3eaea3SApple OSS Distributions err = ioctl(s, SIOCIFDESTROY, &ifr);
80*5e3eaea3SApple OSS Distributions if (fail_on_error) {
81*5e3eaea3SApple OSS Distributions T_QUIET;
82*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(err, "SIOCSIFDESTROY %s", ifr.ifr_name);
83*5e3eaea3SApple OSS Distributions }
84*5e3eaea3SApple OSS Distributions if (err < 0) {
85*5e3eaea3SApple OSS Distributions T_LOG("SIOCSIFDESTROY %s", ifr.ifr_name);
86*5e3eaea3SApple OSS Distributions }
87*5e3eaea3SApple OSS Distributions return err;
88*5e3eaea3SApple OSS Distributions }
89*5e3eaea3SApple OSS Distributions
90*5e3eaea3SApple OSS Distributions static int
ifnet_set_flags(int s,const char * ifname,uint16_t flags_set,uint16_t flags_clear)91*5e3eaea3SApple OSS Distributions ifnet_set_flags(int s, const char * ifname,
92*5e3eaea3SApple OSS Distributions uint16_t flags_set, uint16_t flags_clear)
93*5e3eaea3SApple OSS Distributions {
94*5e3eaea3SApple OSS Distributions uint16_t flags_after;
95*5e3eaea3SApple OSS Distributions uint16_t flags_before;
96*5e3eaea3SApple OSS Distributions struct ifreq ifr;
97*5e3eaea3SApple OSS Distributions int ret;
98*5e3eaea3SApple OSS Distributions
99*5e3eaea3SApple OSS Distributions bzero(&ifr, sizeof(ifr));
100*5e3eaea3SApple OSS Distributions strncpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
101*5e3eaea3SApple OSS Distributions ret = ioctl(s, SIOCGIFFLAGS, (caddr_t)&ifr);
102*5e3eaea3SApple OSS Distributions if (ret != 0) {
103*5e3eaea3SApple OSS Distributions T_LOG("SIOCGIFFLAGS %s", ifr.ifr_name);
104*5e3eaea3SApple OSS Distributions return ret;
105*5e3eaea3SApple OSS Distributions }
106*5e3eaea3SApple OSS Distributions flags_before = (uint16_t)ifr.ifr_flags;
107*5e3eaea3SApple OSS Distributions ifr.ifr_flags |= flags_set;
108*5e3eaea3SApple OSS Distributions ifr.ifr_flags &= ~(flags_clear);
109*5e3eaea3SApple OSS Distributions flags_after = (uint16_t)ifr.ifr_flags;
110*5e3eaea3SApple OSS Distributions if (flags_before == flags_after) {
111*5e3eaea3SApple OSS Distributions /* nothing to do */
112*5e3eaea3SApple OSS Distributions ret = 0;
113*5e3eaea3SApple OSS Distributions } else {
114*5e3eaea3SApple OSS Distributions /* issue the ioctl */
115*5e3eaea3SApple OSS Distributions T_QUIET;
116*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ioctl(s, SIOCSIFFLAGS, &ifr),
117*5e3eaea3SApple OSS Distributions "SIOCSIFFLAGS %s 0x%x",
118*5e3eaea3SApple OSS Distributions ifr.ifr_name, (uint16_t)ifr.ifr_flags);
119*5e3eaea3SApple OSS Distributions }
120*5e3eaea3SApple OSS Distributions return ret;
121*5e3eaea3SApple OSS Distributions }
122*5e3eaea3SApple OSS Distributions
123*5e3eaea3SApple OSS Distributions static int
ifnet_create(int s,char * ifname,size_t ifname_size)124*5e3eaea3SApple OSS Distributions ifnet_create(int s, char * ifname, size_t ifname_size)
125*5e3eaea3SApple OSS Distributions {
126*5e3eaea3SApple OSS Distributions int error = 0;
127*5e3eaea3SApple OSS Distributions struct ifreq ifr;
128*5e3eaea3SApple OSS Distributions
129*5e3eaea3SApple OSS Distributions bzero(&ifr, sizeof(ifr));
130*5e3eaea3SApple OSS Distributions strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
131*5e3eaea3SApple OSS Distributions
132*5e3eaea3SApple OSS Distributions for (int i = 0; i < SIFCREATE_RETRY; i++) {
133*5e3eaea3SApple OSS Distributions if (ioctl(s, SIOCIFCREATE, &ifr) < 0) {
134*5e3eaea3SApple OSS Distributions error = errno;
135*5e3eaea3SApple OSS Distributions T_LOG("SIOCSIFCREATE %s: %s", ifname,
136*5e3eaea3SApple OSS Distributions strerror(error));
137*5e3eaea3SApple OSS Distributions if (error == EBUSY) {
138*5e3eaea3SApple OSS Distributions /* interface is tearing down, try again */
139*5e3eaea3SApple OSS Distributions usleep(10000);
140*5e3eaea3SApple OSS Distributions } else if (error == EEXIST) {
141*5e3eaea3SApple OSS Distributions /* interface exists, try destroying it */
142*5e3eaea3SApple OSS Distributions (void)ifnet_destroy(s, ifname, false);
143*5e3eaea3SApple OSS Distributions } else {
144*5e3eaea3SApple OSS Distributions /* unexpected failure */
145*5e3eaea3SApple OSS Distributions break;
146*5e3eaea3SApple OSS Distributions }
147*5e3eaea3SApple OSS Distributions } else {
148*5e3eaea3SApple OSS Distributions error = 0;
149*5e3eaea3SApple OSS Distributions break;
150*5e3eaea3SApple OSS Distributions }
151*5e3eaea3SApple OSS Distributions }
152*5e3eaea3SApple OSS Distributions if (error == 0) {
153*5e3eaea3SApple OSS Distributions /* Copy back the interface name with unit number */
154*5e3eaea3SApple OSS Distributions strlcpy(ifname, ifr.ifr_name, ifname_size);
155*5e3eaea3SApple OSS Distributions error = ifnet_set_flags(s, ifname, IFF_UP, 0);
156*5e3eaea3SApple OSS Distributions }
157*5e3eaea3SApple OSS Distributions return error;
158*5e3eaea3SApple OSS Distributions }
159*5e3eaea3SApple OSS Distributions
160*5e3eaea3SApple OSS Distributions #define MAXBUF 32
161*5e3eaea3SApple OSS Distributions
162*5e3eaea3SApple OSS Distributions static void
HexDump(void * data,size_t len)163*5e3eaea3SApple OSS Distributions HexDump(void *data, size_t len)
164*5e3eaea3SApple OSS Distributions {
165*5e3eaea3SApple OSS Distributions size_t i, j, k;
166*5e3eaea3SApple OSS Distributions unsigned char *ptr = (unsigned char *)data;
167*5e3eaea3SApple OSS Distributions unsigned char buf[3 * MAXBUF + 1];
168*5e3eaea3SApple OSS Distributions
169*5e3eaea3SApple OSS Distributions for (i = 0; i < len; i += MAXBUF) {
170*5e3eaea3SApple OSS Distributions for (j = i, k = 0; j < i + MAXBUF && j < len; j++) {
171*5e3eaea3SApple OSS Distributions unsigned char msnbl = ptr[j] >> 4;
172*5e3eaea3SApple OSS Distributions unsigned char lsnbl = ptr[j] & 0x0f;
173*5e3eaea3SApple OSS Distributions
174*5e3eaea3SApple OSS Distributions buf[k++] = msnbl < 10 ? msnbl + '0' : msnbl + 'a' - 10;
175*5e3eaea3SApple OSS Distributions buf[k++] = lsnbl < 10 ? lsnbl + '0' : lsnbl + 'a' - 10;
176*5e3eaea3SApple OSS Distributions if ((j % 2) == 1) {
177*5e3eaea3SApple OSS Distributions buf[k++] = ' ';
178*5e3eaea3SApple OSS Distributions }
179*5e3eaea3SApple OSS Distributions if ((j % MAXBUF) == MAXBUF - 1) {
180*5e3eaea3SApple OSS Distributions buf[k++] = ' ';
181*5e3eaea3SApple OSS Distributions }
182*5e3eaea3SApple OSS Distributions }
183*5e3eaea3SApple OSS Distributions buf[k] = 0;
184*5e3eaea3SApple OSS Distributions T_LOG("%5zd: %s\n", i, buf);
185*5e3eaea3SApple OSS Distributions }
186*5e3eaea3SApple OSS Distributions }
187*5e3eaea3SApple OSS Distributions
188*5e3eaea3SApple OSS Distributions static size_t
snprint_dottedhex(char * str,size_t strsize,const void * data,const size_t datasize)189*5e3eaea3SApple OSS Distributions snprint_dottedhex(char *str, size_t strsize, const void *data, const size_t datasize)
190*5e3eaea3SApple OSS Distributions {
191*5e3eaea3SApple OSS Distributions size_t is = 0, ip = 0;
192*5e3eaea3SApple OSS Distributions const unsigned char *ptr = (const unsigned char *)data;
193*5e3eaea3SApple OSS Distributions
194*5e3eaea3SApple OSS Distributions for (is = 0, ip = 0; is + 3 < strsize - 1 && ip < datasize; ip++) {
195*5e3eaea3SApple OSS Distributions unsigned char msnbl = ptr[ip] >> 4;
196*5e3eaea3SApple OSS Distributions unsigned char lsnbl = ptr[ip] & 0x0f;
197*5e3eaea3SApple OSS Distributions
198*5e3eaea3SApple OSS Distributions if (ip > 0) {
199*5e3eaea3SApple OSS Distributions str[is++] = '.';
200*5e3eaea3SApple OSS Distributions }
201*5e3eaea3SApple OSS Distributions str[is++] = (char)(msnbl + (msnbl < 10 ? '0' : 'a' - 10));
202*5e3eaea3SApple OSS Distributions str[is++] = (char)(lsnbl + (lsnbl < 10 ? '0' : 'a' - 10));
203*5e3eaea3SApple OSS Distributions }
204*5e3eaea3SApple OSS Distributions str[is] = 0;
205*5e3eaea3SApple OSS Distributions return is;
206*5e3eaea3SApple OSS Distributions }
207*5e3eaea3SApple OSS Distributions
208*5e3eaea3SApple OSS Distributions static void
print_sockaddr_dl(const char * pre,const struct sockaddr * sa,const char * post)209*5e3eaea3SApple OSS Distributions print_sockaddr_dl(const char *pre, const struct sockaddr *sa, const char *post)
210*5e3eaea3SApple OSS Distributions {
211*5e3eaea3SApple OSS Distributions char nbuffer[256];
212*5e3eaea3SApple OSS Distributions char abuffer[256];
213*5e3eaea3SApple OSS Distributions char sbuffer[256];
214*5e3eaea3SApple OSS Distributions struct sockaddr_dl sdl = {};
215*5e3eaea3SApple OSS Distributions
216*5e3eaea3SApple OSS Distributions if (sa == NULL) {
217*5e3eaea3SApple OSS Distributions return;
218*5e3eaea3SApple OSS Distributions }
219*5e3eaea3SApple OSS Distributions memcpy(&sdl, sa, MIN(sizeof(sdl), sa->sa_len));
220*5e3eaea3SApple OSS Distributions strlcpy(nbuffer, sdl.sdl_data, sdl.sdl_nlen);
221*5e3eaea3SApple OSS Distributions snprint_dottedhex(abuffer, sizeof(abuffer), sdl.sdl_data + sdl.sdl_nlen, sdl.sdl_alen);
222*5e3eaea3SApple OSS Distributions snprint_dottedhex(sbuffer, sizeof(sbuffer), sdl.sdl_data + sdl.sdl_nlen + sdl.sdl_alen, sdl.sdl_slen);
223*5e3eaea3SApple OSS Distributions
224*5e3eaea3SApple OSS Distributions T_LOG("%ssdl_len %u sdl_family %u sdl_index %u sdl_type %u sdl_nlen %u (%s) sdl_alen %u (%s) sdl_slen %u (%s)%s",
225*5e3eaea3SApple OSS Distributions pre != NULL ? pre : "",
226*5e3eaea3SApple OSS Distributions sdl.sdl_len, sdl.sdl_family, sdl.sdl_index, sdl.sdl_type,
227*5e3eaea3SApple OSS Distributions sdl.sdl_nlen, nbuffer, sdl.sdl_alen, abuffer, sdl.sdl_slen, sbuffer,
228*5e3eaea3SApple OSS Distributions post != NULL ? post : "");
229*5e3eaea3SApple OSS Distributions }
230*5e3eaea3SApple OSS Distributions
231*5e3eaea3SApple OSS Distributions static void
print_sockaddr_in(const char * pre,const struct sockaddr * sa,const char * post)232*5e3eaea3SApple OSS Distributions print_sockaddr_in(const char *pre, const struct sockaddr *sa, const char *post)
233*5e3eaea3SApple OSS Distributions {
234*5e3eaea3SApple OSS Distributions char abuffer[256];
235*5e3eaea3SApple OSS Distributions char zbuffer[256];
236*5e3eaea3SApple OSS Distributions struct sockaddr_in sin = {};
237*5e3eaea3SApple OSS Distributions
238*5e3eaea3SApple OSS Distributions if (sa == NULL) {
239*5e3eaea3SApple OSS Distributions return;
240*5e3eaea3SApple OSS Distributions }
241*5e3eaea3SApple OSS Distributions
242*5e3eaea3SApple OSS Distributions memcpy(&sin, sa, MIN(sizeof(sin), sa->sa_len));
243*5e3eaea3SApple OSS Distributions inet_ntop(AF_INET, &sin.sin_addr, abuffer, sizeof(abuffer));
244*5e3eaea3SApple OSS Distributions snprint_dottedhex(zbuffer, sizeof(zbuffer), sin.sin_zero, sizeof(sin.sin_zero));
245*5e3eaea3SApple OSS Distributions
246*5e3eaea3SApple OSS Distributions T_LOG("%ssin_len %u sin_family %u sin_port %u sin_addr %s sin_zero %s%s",
247*5e3eaea3SApple OSS Distributions pre != NULL ? pre : "",
248*5e3eaea3SApple OSS Distributions sin.sin_len, sin.sin_family, htons(sin.sin_port), abuffer, zbuffer,
249*5e3eaea3SApple OSS Distributions post != NULL ? post : "");
250*5e3eaea3SApple OSS Distributions }
251*5e3eaea3SApple OSS Distributions
252*5e3eaea3SApple OSS Distributions static void
print_sockaddr_in6(const char * pre,const struct sockaddr * sa,const char * post)253*5e3eaea3SApple OSS Distributions print_sockaddr_in6(const char *pre, const struct sockaddr *sa, const char *post)
254*5e3eaea3SApple OSS Distributions {
255*5e3eaea3SApple OSS Distributions char abuffer[256];
256*5e3eaea3SApple OSS Distributions struct sockaddr_in6 sin6 = {};
257*5e3eaea3SApple OSS Distributions
258*5e3eaea3SApple OSS Distributions if (sa == NULL) {
259*5e3eaea3SApple OSS Distributions return;
260*5e3eaea3SApple OSS Distributions }
261*5e3eaea3SApple OSS Distributions
262*5e3eaea3SApple OSS Distributions memcpy(&sin6, sa, MIN(sizeof(sin6), sa->sa_len));
263*5e3eaea3SApple OSS Distributions inet_ntop(AF_INET6, &sin6.sin6_addr, abuffer, sizeof(abuffer));
264*5e3eaea3SApple OSS Distributions
265*5e3eaea3SApple OSS Distributions T_LOG("%ssin6_len %u sin6_family %u sin6_port %u sin6_flowinfo %u sin6_addr %s sin6_scope_id %u%s",
266*5e3eaea3SApple OSS Distributions pre != NULL ? pre : "",
267*5e3eaea3SApple OSS Distributions sin6.sin6_len, sin6.sin6_family, htons(sin6.sin6_port), sin6.sin6_flowinfo, abuffer, sin6.sin6_scope_id,
268*5e3eaea3SApple OSS Distributions post != NULL ? post : "");
269*5e3eaea3SApple OSS Distributions }
270*5e3eaea3SApple OSS Distributions
271*5e3eaea3SApple OSS Distributions static void
print_sockaddr(const char * pre,const struct sockaddr * sa,const char * post)272*5e3eaea3SApple OSS Distributions print_sockaddr(const char *pre, const struct sockaddr *sa, const char *post)
273*5e3eaea3SApple OSS Distributions {
274*5e3eaea3SApple OSS Distributions char buffer[256];
275*5e3eaea3SApple OSS Distributions
276*5e3eaea3SApple OSS Distributions if (sa == NULL) {
277*5e3eaea3SApple OSS Distributions return;
278*5e3eaea3SApple OSS Distributions }
279*5e3eaea3SApple OSS Distributions
280*5e3eaea3SApple OSS Distributions snprint_dottedhex(buffer, sizeof(buffer), sa->sa_data, sa->sa_len - 2);
281*5e3eaea3SApple OSS Distributions
282*5e3eaea3SApple OSS Distributions T_LOG("%ssa_len %u sa_family %u sa_data %s%s",
283*5e3eaea3SApple OSS Distributions pre != NULL ? pre : "",
284*5e3eaea3SApple OSS Distributions sa->sa_len, sa->sa_family, buffer,
285*5e3eaea3SApple OSS Distributions post != NULL ? post : "");
286*5e3eaea3SApple OSS Distributions }
287*5e3eaea3SApple OSS Distributions
288*5e3eaea3SApple OSS Distributions
289*5e3eaea3SApple OSS Distributions #define ROUNDUP(a, size) (((a) & ((size) - 1)) ? (1 + ((a)|(size - 1))) : (a))
290*5e3eaea3SApple OSS Distributions
291*5e3eaea3SApple OSS Distributions #define NEXT_SA(p) (struct sockaddr *) \
292*5e3eaea3SApple OSS Distributions ((caddr_t)p + (p->sa_len ? ROUNDUP(p->sa_len, sizeof(u_int32_t)) : \
293*5e3eaea3SApple OSS Distributions sizeof(u_long)))
294*5e3eaea3SApple OSS Distributions
295*5e3eaea3SApple OSS Distributions static size_t
get_rti_info(int addrs,struct sockaddr * sa,struct sockaddr ** rti_info)296*5e3eaea3SApple OSS Distributions get_rti_info(int addrs, struct sockaddr *sa, struct sockaddr **rti_info)
297*5e3eaea3SApple OSS Distributions {
298*5e3eaea3SApple OSS Distributions int i;
299*5e3eaea3SApple OSS Distributions size_t len = 0;
300*5e3eaea3SApple OSS Distributions
301*5e3eaea3SApple OSS Distributions for (i = 0; i < RTAX_MAX; i++) {
302*5e3eaea3SApple OSS Distributions if (addrs & (1 << i)) {
303*5e3eaea3SApple OSS Distributions rti_info[i] = sa;
304*5e3eaea3SApple OSS Distributions if (sa->sa_len < sizeof(struct sockaddr)) {
305*5e3eaea3SApple OSS Distributions len += sizeof(struct sockaddr);
306*5e3eaea3SApple OSS Distributions } else {
307*5e3eaea3SApple OSS Distributions len += sa->sa_len;
308*5e3eaea3SApple OSS Distributions }
309*5e3eaea3SApple OSS Distributions sa = NEXT_SA(sa);
310*5e3eaea3SApple OSS Distributions } else {
311*5e3eaea3SApple OSS Distributions rti_info[i] = NULL;
312*5e3eaea3SApple OSS Distributions }
313*5e3eaea3SApple OSS Distributions }
314*5e3eaea3SApple OSS Distributions return len;
315*5e3eaea3SApple OSS Distributions }
316*5e3eaea3SApple OSS Distributions
317*5e3eaea3SApple OSS Distributions static void
print_address(const char * pre,const struct sockaddr * sa,const char * post,u_char asFamily)318*5e3eaea3SApple OSS Distributions print_address(const char *pre, const struct sockaddr *sa, const char *post, u_char asFamily)
319*5e3eaea3SApple OSS Distributions {
320*5e3eaea3SApple OSS Distributions if (sa == NULL) {
321*5e3eaea3SApple OSS Distributions T_LOG("%s(NULL)%s",
322*5e3eaea3SApple OSS Distributions pre != NULL ? pre : "",
323*5e3eaea3SApple OSS Distributions post != NULL ? post : "");
324*5e3eaea3SApple OSS Distributions return;
325*5e3eaea3SApple OSS Distributions }
326*5e3eaea3SApple OSS Distributions if (sa->sa_len == 0) {
327*5e3eaea3SApple OSS Distributions T_LOG("%ssa_len 0%s",
328*5e3eaea3SApple OSS Distributions pre != NULL ? pre : "",
329*5e3eaea3SApple OSS Distributions post != NULL ? post : "");
330*5e3eaea3SApple OSS Distributions return;
331*5e3eaea3SApple OSS Distributions }
332*5e3eaea3SApple OSS Distributions if (sa->sa_len == 1) {
333*5e3eaea3SApple OSS Distributions T_LOG("%ssa_len 1%s",
334*5e3eaea3SApple OSS Distributions pre != NULL ? pre : "",
335*5e3eaea3SApple OSS Distributions post != NULL ? post : "");
336*5e3eaea3SApple OSS Distributions return;
337*5e3eaea3SApple OSS Distributions }
338*5e3eaea3SApple OSS Distributions
339*5e3eaea3SApple OSS Distributions // If not forced
340*5e3eaea3SApple OSS Distributions if (asFamily == AF_UNSPEC) {
341*5e3eaea3SApple OSS Distributions asFamily = sa->sa_family;
342*5e3eaea3SApple OSS Distributions }
343*5e3eaea3SApple OSS Distributions switch (asFamily) {
344*5e3eaea3SApple OSS Distributions case AF_INET: {
345*5e3eaea3SApple OSS Distributions print_sockaddr_in(pre, sa, post);
346*5e3eaea3SApple OSS Distributions break;
347*5e3eaea3SApple OSS Distributions }
348*5e3eaea3SApple OSS Distributions case AF_INET6: {
349*5e3eaea3SApple OSS Distributions print_sockaddr_in6(pre, sa, post);
350*5e3eaea3SApple OSS Distributions break;
351*5e3eaea3SApple OSS Distributions }
352*5e3eaea3SApple OSS Distributions case AF_LINK: {
353*5e3eaea3SApple OSS Distributions print_sockaddr_dl(pre, sa, post);
354*5e3eaea3SApple OSS Distributions break;
355*5e3eaea3SApple OSS Distributions }
356*5e3eaea3SApple OSS Distributions default:
357*5e3eaea3SApple OSS Distributions print_sockaddr(pre, sa, post);
358*5e3eaea3SApple OSS Distributions break;
359*5e3eaea3SApple OSS Distributions }
360*5e3eaea3SApple OSS Distributions }
361*5e3eaea3SApple OSS Distributions
362*5e3eaea3SApple OSS Distributions static void
print_rti_info(struct sockaddr * rti_info[])363*5e3eaea3SApple OSS Distributions print_rti_info(struct sockaddr *rti_info[])
364*5e3eaea3SApple OSS Distributions {
365*5e3eaea3SApple OSS Distributions struct sockaddr *sa;
366*5e3eaea3SApple OSS Distributions u_char asFamily = 0;
367*5e3eaea3SApple OSS Distributions
368*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_IFA])) {
369*5e3eaea3SApple OSS Distributions asFamily = sa->sa_family;
370*5e3eaea3SApple OSS Distributions print_address(" RTAX_IFA ", sa, "\n", 0);
371*5e3eaea3SApple OSS Distributions }
372*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_DST])) {
373*5e3eaea3SApple OSS Distributions asFamily = sa->sa_family;
374*5e3eaea3SApple OSS Distributions print_address(" RTAX_DST ", sa, "\n", 0);
375*5e3eaea3SApple OSS Distributions }
376*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_BRD])) {
377*5e3eaea3SApple OSS Distributions print_address(" RTAX_BRD ", sa, "\n", asFamily);
378*5e3eaea3SApple OSS Distributions }
379*5e3eaea3SApple OSS Distributions
380*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_NETMASK])) {
381*5e3eaea3SApple OSS Distributions print_address(" RTAX_NETMASK ", sa, "\n", asFamily);
382*5e3eaea3SApple OSS Distributions }
383*5e3eaea3SApple OSS Distributions
384*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_GATEWAY])) {
385*5e3eaea3SApple OSS Distributions print_address(" RTAX_GATEWAY ", sa, "\n", 0);
386*5e3eaea3SApple OSS Distributions }
387*5e3eaea3SApple OSS Distributions
388*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_GENMASK])) {
389*5e3eaea3SApple OSS Distributions print_address(" RTAX_GENMASK ", sa, "\n", asFamily);
390*5e3eaea3SApple OSS Distributions }
391*5e3eaea3SApple OSS Distributions
392*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_AUTHOR])) {
393*5e3eaea3SApple OSS Distributions print_address(" RTAX_AUTHOR ", sa, "\n", asFamily);
394*5e3eaea3SApple OSS Distributions }
395*5e3eaea3SApple OSS Distributions
396*5e3eaea3SApple OSS Distributions if ((sa = rti_info[RTAX_IFP])) {
397*5e3eaea3SApple OSS Distributions print_address(" RTAX_IFP ", sa, "\n", 0);
398*5e3eaea3SApple OSS Distributions }
399*5e3eaea3SApple OSS Distributions }
400*5e3eaea3SApple OSS Distributions
401*5e3eaea3SApple OSS Distributions static void
print_rt_iflist2(const char * label)402*5e3eaea3SApple OSS Distributions print_rt_iflist2(const char *label)
403*5e3eaea3SApple OSS Distributions {
404*5e3eaea3SApple OSS Distributions size_t len;
405*5e3eaea3SApple OSS Distributions int mib[6] = { CTL_NET, PF_ROUTE, 0, 0, NET_RT_IFLIST2, 0 };
406*5e3eaea3SApple OSS Distributions unsigned char *buf = NULL;
407*5e3eaea3SApple OSS Distributions unsigned char *lim, *next;
408*5e3eaea3SApple OSS Distributions struct if_msghdr *ifmsg;
409*5e3eaea3SApple OSS Distributions
410*5e3eaea3SApple OSS Distributions T_LOG("interface address list for %s", label);
411*5e3eaea3SApple OSS Distributions
412*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(sysctl(mib, 6, NULL, &len, NULL, 0), "sysctl NET_RT_IFLIST2");
413*5e3eaea3SApple OSS Distributions
414*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(buf = calloc(1, len), "rt_if_list_buf calloc(1, %zd)", len);
415*5e3eaea3SApple OSS Distributions
416*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(sysctl(mib, 6, buf, &len, NULL, 0), "sysctl NET_RT_IFLIST2");
417*5e3eaea3SApple OSS Distributions
418*5e3eaea3SApple OSS Distributions lim = buf + len;
419*5e3eaea3SApple OSS Distributions for (next = buf; next < lim; next += ifmsg->ifm_msglen) {
420*5e3eaea3SApple OSS Distributions ifmsg = (struct if_msghdr *)(void *)next;
421*5e3eaea3SApple OSS Distributions char ifname[IF_NAMESIZE + 1];
422*5e3eaea3SApple OSS Distributions
423*5e3eaea3SApple OSS Distributions if (ifmsg->ifm_type == RTM_IFINFO2) {
424*5e3eaea3SApple OSS Distributions struct if_msghdr2 *ifm = (struct if_msghdr2 *)ifmsg;
425*5e3eaea3SApple OSS Distributions struct sockaddr *sa = (struct sockaddr *)(ifm + 1);
426*5e3eaea3SApple OSS Distributions
427*5e3eaea3SApple OSS Distributions (void)if_indextoname(ifm->ifm_index, ifname);
428*5e3eaea3SApple OSS Distributions T_LOG("interface: %s", ifname);
429*5e3eaea3SApple OSS Distributions print_address(" PRIMARY ", sa, "", 0);
430*5e3eaea3SApple OSS Distributions } else if (ifmsg->ifm_type == RTM_NEWADDR) {
431*5e3eaea3SApple OSS Distributions struct sockaddr *rti_info[RTAX_MAX];
432*5e3eaea3SApple OSS Distributions struct ifa_msghdr *ifam = (struct ifa_msghdr *)ifmsg;
433*5e3eaea3SApple OSS Distributions
434*5e3eaea3SApple OSS Distributions (void) get_rti_info(ifam->ifam_addrs, (struct sockaddr *)(ifam + 1), rti_info);
435*5e3eaea3SApple OSS Distributions
436*5e3eaea3SApple OSS Distributions print_rti_info(rti_info);
437*5e3eaea3SApple OSS Distributions }
438*5e3eaea3SApple OSS Distributions }
439*5e3eaea3SApple OSS Distributions free(buf);
440*5e3eaea3SApple OSS Distributions }
441*5e3eaea3SApple OSS Distributions
442*5e3eaea3SApple OSS Distributions static int
check_rt_if_list_for_pattern(const char * label,unsigned char pattern,size_t pattern_size)443*5e3eaea3SApple OSS Distributions check_rt_if_list_for_pattern(const char *label, unsigned char pattern, size_t pattern_size)
444*5e3eaea3SApple OSS Distributions {
445*5e3eaea3SApple OSS Distributions size_t i;
446*5e3eaea3SApple OSS Distributions size_t len;
447*5e3eaea3SApple OSS Distributions int mib[6] = { CTL_NET, PF_ROUTE, 0, 0, NET_RT_IFLIST, 0 };
448*5e3eaea3SApple OSS Distributions unsigned char *rt_if_list_buf = NULL;
449*5e3eaea3SApple OSS Distributions int count = 0;
450*5e3eaea3SApple OSS Distributions unsigned char *pattern_buf = NULL;
451*5e3eaea3SApple OSS Distributions
452*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(pattern_buf = calloc(1, pattern_size), "pattern_buf calloc(1, %zd)", pattern_size);
453*5e3eaea3SApple OSS Distributions memset(pattern_buf, pattern, pattern_size);
454*5e3eaea3SApple OSS Distributions
455*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(sysctl(mib, 6, NULL, &len, NULL, 0), "sysctl NET_RT_IFLIST");
456*5e3eaea3SApple OSS Distributions
457*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(rt_if_list_buf = calloc(1, len), "rt_if_list_buf calloc(1, %zd)", len);
458*5e3eaea3SApple OSS Distributions
459*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(sysctl(mib, 6, rt_if_list_buf, &len, NULL, 0), "sysctl NET_RT_IFLIST");
460*5e3eaea3SApple OSS Distributions
461*5e3eaea3SApple OSS Distributions if (label != NULL) {
462*5e3eaea3SApple OSS Distributions T_LOG("%s sysctl NET_RT_IFLIST buffer length %zd\n", label, len);
463*5e3eaea3SApple OSS Distributions }
464*5e3eaea3SApple OSS Distributions
465*5e3eaea3SApple OSS Distributions count = 0;
466*5e3eaea3SApple OSS Distributions for (i = 0; i < len - pattern_size; i++) {
467*5e3eaea3SApple OSS Distributions if (memcmp(rt_if_list_buf + i, pattern_buf, pattern_size) == 0) {
468*5e3eaea3SApple OSS Distributions count++;
469*5e3eaea3SApple OSS Distributions i += pattern_size - 1;
470*5e3eaea3SApple OSS Distributions }
471*5e3eaea3SApple OSS Distributions }
472*5e3eaea3SApple OSS Distributions
473*5e3eaea3SApple OSS Distributions if (label != NULL) {
474*5e3eaea3SApple OSS Distributions if (label != NULL && count > 0) {
475*5e3eaea3SApple OSS Distributions T_LOG("%s found pattern at %zd count %d times\n", label, i, count);
476*5e3eaea3SApple OSS Distributions HexDump(rt_if_list_buf, len);
477*5e3eaea3SApple OSS Distributions }
478*5e3eaea3SApple OSS Distributions }
479*5e3eaea3SApple OSS Distributions
480*5e3eaea3SApple OSS Distributions free(rt_if_list_buf);
481*5e3eaea3SApple OSS Distributions free(pattern_buf);
482*5e3eaea3SApple OSS Distributions
483*5e3eaea3SApple OSS Distributions return count;
484*5e3eaea3SApple OSS Distributions }
485*5e3eaea3SApple OSS Distributions
486*5e3eaea3SApple OSS Distributions static bool
find_unused_pattern_in_rt_if_list(unsigned char * pattern,size_t pattern_size)487*5e3eaea3SApple OSS Distributions find_unused_pattern_in_rt_if_list(unsigned char *pattern, size_t pattern_size)
488*5e3eaea3SApple OSS Distributions {
489*5e3eaea3SApple OSS Distributions bool found_pattern = false;
490*5e3eaea3SApple OSS Distributions unsigned char i;
491*5e3eaea3SApple OSS Distributions unsigned char *pattern_buf = NULL;
492*5e3eaea3SApple OSS Distributions
493*5e3eaea3SApple OSS Distributions T_QUIET; T_ASSERT_NOTNULL(pattern_buf = calloc(1, pattern_size), "pattern_buf calloc(1, %zd)", pattern_size);
494*5e3eaea3SApple OSS Distributions
495*5e3eaea3SApple OSS Distributions /* Try 10 times to find an unused pattern */
496*5e3eaea3SApple OSS Distributions for (i = 1; i < 255; i++) {
497*5e3eaea3SApple OSS Distributions if (check_rt_if_list_for_pattern(NULL, i, pattern_size) == 0) {
498*5e3eaea3SApple OSS Distributions found_pattern = true;
499*5e3eaea3SApple OSS Distributions *pattern = i;
500*5e3eaea3SApple OSS Distributions memset(pattern_buf, i, pattern_size);
501*5e3eaea3SApple OSS Distributions T_LOG("PATTERN: ");
502*5e3eaea3SApple OSS Distributions HexDump(pattern_buf, pattern_size);
503*5e3eaea3SApple OSS Distributions break;
504*5e3eaea3SApple OSS Distributions }
505*5e3eaea3SApple OSS Distributions }
506*5e3eaea3SApple OSS Distributions free(pattern_buf);
507*5e3eaea3SApple OSS Distributions
508*5e3eaea3SApple OSS Distributions return found_pattern;
509*5e3eaea3SApple OSS Distributions }
510*5e3eaea3SApple OSS Distributions
511*5e3eaea3SApple OSS Distributions static const char *
ioc_str(unsigned long cmd)512*5e3eaea3SApple OSS Distributions ioc_str(unsigned long cmd)
513*5e3eaea3SApple OSS Distributions {
514*5e3eaea3SApple OSS Distributions #define X(a) case a: return #a;
515*5e3eaea3SApple OSS Distributions
516*5e3eaea3SApple OSS Distributions switch (cmd) {
517*5e3eaea3SApple OSS Distributions SIOC_LIST
518*5e3eaea3SApple OSS Distributions
519*5e3eaea3SApple OSS Distributions default:
520*5e3eaea3SApple OSS Distributions break;
521*5e3eaea3SApple OSS Distributions }
522*5e3eaea3SApple OSS Distributions return "";
523*5e3eaea3SApple OSS Distributions }
524*5e3eaea3SApple OSS Distributions
525*5e3eaea3SApple OSS Distributions struct ioc_ifreq {
526*5e3eaea3SApple OSS Distributions unsigned long ioc_cmd;
527*5e3eaea3SApple OSS Distributions uint8_t salen;
528*5e3eaea3SApple OSS Distributions uint8_t safamily;
529*5e3eaea3SApple OSS Distributions const char *sastr;
530*5e3eaea3SApple OSS Distributions int error; // 0 means no error, -1, end of list, otherwise expected errno
531*5e3eaea3SApple OSS Distributions };
532*5e3eaea3SApple OSS Distributions
533*5e3eaea3SApple OSS Distributions static struct ioc_ifreq ioc_list[] = {
534*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in), AF_INET6, "10.2.3.1", EINVAL },
535*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in), AF_INET6, "10.2.3.1", EADDRNOTAVAIL },
536*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in6), AF_INET, "10.2.3.1", EADDRNOTAVAIL },
537*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in), AF_INET, "10.2.3.1", EADDRNOTAVAIL },
538*5e3eaea3SApple OSS Distributions
539*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, 0xf0, AF_INET6, "10.2.3.1", EINVAL },
540*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, 0xf0, AF_INET6, "10.2.3.1", EADDRNOTAVAIL },
541*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, 0xf0, AF_INET, "10.2.3.1", EADDRNOTAVAIL },
542*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, 0xf0, AF_INET, "10.2.3.1", EADDRNOTAVAIL },
543*5e3eaea3SApple OSS Distributions
544*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, 0, AF_INET6, "10.2.3.1", EINVAL },
545*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, 0, AF_INET6, "10.2.3.1", EADDRNOTAVAIL },
546*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, 0, AF_INET, "10.2.3.1", EADDRNOTAVAIL },
547*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, 0, AF_INET, "10.2.3.1", EADDRNOTAVAIL },
548*5e3eaea3SApple OSS Distributions
549*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in6), AF_INET, "10.2.3.2", 0 },
550*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in), AF_INET6, "10.2.3.2", 0 },
551*5e3eaea3SApple OSS Distributions
552*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in), AF_INET, "10.2.3.3", 0 },
553*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in6), AF_INET, "10.2.3.3", 0 },
554*5e3eaea3SApple OSS Distributions
555*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in6), AF_INET6, "10.2.3.4", EINVAL },
556*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in6), AF_INET, "10.2.3.4", EADDRNOTAVAIL },
557*5e3eaea3SApple OSS Distributions
558*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in), AF_INET, "10.2.3.5", 0 },
559*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in), AF_INET, "0.0.0.0", EADDRNOTAVAIL },
560*5e3eaea3SApple OSS Distributions { SIOCDIFADDR, sizeof(struct sockaddr_in), AF_INET, "10.2.3.5", 0 },
561*5e3eaea3SApple OSS Distributions
562*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in), AF_INET, "10.2.3.6", 0 },
563*5e3eaea3SApple OSS Distributions
564*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, sizeof(struct sockaddr_in), 0, "ff.00.00.00", 0 },
565*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, sizeof(struct sockaddr_in), AF_INET, "ff.00.00.00", 0 },
566*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, sizeof(struct sockaddr_in), AF_INET6, "ff.f.00.00", 0 },
567*5e3eaea3SApple OSS Distributions
568*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, sizeof(struct sockaddr_in6), 0, "ff.ff.00.00", 0 },
569*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, sizeof(struct sockaddr_in6), AF_INET, "ff.ff.00.00", 0 },
570*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, sizeof(struct sockaddr_in6), AF_INET6, "ff.ff.f0.00", 0 },
571*5e3eaea3SApple OSS Distributions
572*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, 0, 0, "ff.ff.00.00", 0 },
573*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, 0, AF_INET, "ff.ff.00.00", 0 },
574*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, 0, AF_INET6, "ff.ff.f0.00", 0 },
575*5e3eaea3SApple OSS Distributions
576*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, 0xf0, 0, "ff.ff.00.00", 0 },
577*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, 0xf0, AF_INET, "ff.ff.00.00", 0 },
578*5e3eaea3SApple OSS Distributions { SIOCSIFNETMASK, 0xf0, AF_INET6, "ff.ff.f0.00", 0 },
579*5e3eaea3SApple OSS Distributions
580*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, sizeof(struct sockaddr_in), 0, "10.255.255.255", 0 },
581*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, sizeof(struct sockaddr_in), AF_INET, "10.255.255.255", 0 },
582*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, sizeof(struct sockaddr_in), AF_INET6, "10.255.255.255", 0 },
583*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, sizeof(struct sockaddr_in6), AF_INET, "10.255.255.255", 0 },
584*5e3eaea3SApple OSS Distributions
585*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0xf0, 0, "10.255.255.255", 0 },
586*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0xf0, AF_INET, "10.255.255.255", 0 },
587*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0xf0, AF_INET6, "10.255.255.255", 0 },
588*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0xf0, AF_INET, "10.255.255.255", 0 },
589*5e3eaea3SApple OSS Distributions
590*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0, 0, "10.255.255.255", 0 },
591*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0, AF_INET, "10.255.255.255", 0 },
592*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0, AF_INET6, "10.255.255.255", 0 },
593*5e3eaea3SApple OSS Distributions { SIOCSIFBRDADDR, 0, AF_INET, "10.255.255.255", 0 },
594*5e3eaea3SApple OSS Distributions
595*5e3eaea3SApple OSS Distributions { 0, 0, 0, "", -1 },
596*5e3eaea3SApple OSS Distributions };
597*5e3eaea3SApple OSS Distributions
598*5e3eaea3SApple OSS Distributions static void
test_sioc_ifr_bounds(struct ioc_ifreq * ioc_ifreq,int s,const char * ifname)599*5e3eaea3SApple OSS Distributions test_sioc_ifr_bounds(struct ioc_ifreq *ioc_ifreq, int s, const char *ifname)
600*5e3eaea3SApple OSS Distributions {
601*5e3eaea3SApple OSS Distributions struct ifreq ifr = {};
602*5e3eaea3SApple OSS Distributions unsigned char pattern;
603*5e3eaea3SApple OSS Distributions struct sockaddr_in *sin;
604*5e3eaea3SApple OSS Distributions
605*5e3eaea3SApple OSS Distributions T_LOG("");
606*5e3eaea3SApple OSS Distributions T_LOG("TEST CASE: %s ioctl(%s, sa_len %u, sa_family %u, %s) -> %d", __func__,
607*5e3eaea3SApple OSS Distributions ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->salen, ioc_ifreq->safamily, ioc_ifreq->sastr, ioc_ifreq->error);
608*5e3eaea3SApple OSS Distributions
609*5e3eaea3SApple OSS Distributions
610*5e3eaea3SApple OSS Distributions if (find_unused_pattern_in_rt_if_list(&pattern, PATTERN_SIZE) == false) {
611*5e3eaea3SApple OSS Distributions T_SKIP("Could not find unused pattern");
612*5e3eaea3SApple OSS Distributions }
613*5e3eaea3SApple OSS Distributions
614*5e3eaea3SApple OSS Distributions strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
615*5e3eaea3SApple OSS Distributions memset(&ifr.ifr_addr.sa_data, pattern, sizeof(ifr.ifr_dstaddr.sa_data));
616*5e3eaea3SApple OSS Distributions
617*5e3eaea3SApple OSS Distributions sin = (struct sockaddr_in *)(void *)&ifr.ifr_addr;
618*5e3eaea3SApple OSS Distributions sin->sin_len = ioc_ifreq->salen;
619*5e3eaea3SApple OSS Distributions sin->sin_family = ioc_ifreq->safamily;
620*5e3eaea3SApple OSS Distributions sin->sin_addr.s_addr = inet_addr(ioc_ifreq->sastr);
621*5e3eaea3SApple OSS Distributions
622*5e3eaea3SApple OSS Distributions int retval;
623*5e3eaea3SApple OSS Distributions if (ioc_ifreq->error == 0) {
624*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_SUCCESS(retval = ioctl(s, ioc_ifreq->ioc_cmd, &ifr),
625*5e3eaea3SApple OSS Distributions "%s, %s: retval %d", ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->sastr, retval);
626*5e3eaea3SApple OSS Distributions } else {
627*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_FAILURE(retval = ioctl(s, ioc_ifreq->ioc_cmd, &ifr), ioc_ifreq->error,
628*5e3eaea3SApple OSS Distributions "%s, %s: retval %d errno %s", ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->sastr, retval, strerror(errno));
629*5e3eaea3SApple OSS Distributions }
630*5e3eaea3SApple OSS Distributions
631*5e3eaea3SApple OSS Distributions T_EXPECT_EQ(check_rt_if_list_for_pattern("test_sioc_ifr_bounds", pattern, PATTERN_SIZE), 0, "pattern should not be found");
632*5e3eaea3SApple OSS Distributions
633*5e3eaea3SApple OSS Distributions
634*5e3eaea3SApple OSS Distributions fflush(stdout);
635*5e3eaea3SApple OSS Distributions fflush(stderr);
636*5e3eaea3SApple OSS Distributions }
637*5e3eaea3SApple OSS Distributions
638*5e3eaea3SApple OSS Distributions T_DECL(sioc_ifr_bounds, "test bound checks on struct ifreq addresses passed to interface ioctls",
639*5e3eaea3SApple OSS Distributions T_META_ASROOT(true))
640*5e3eaea3SApple OSS Distributions {
641*5e3eaea3SApple OSS Distributions int s = -1;
642*5e3eaea3SApple OSS Distributions char ifname[IFNAMSIZ];
643*5e3eaea3SApple OSS Distributions
644*5e3eaea3SApple OSS Distributions T_LOG("%s", __func__);
645*5e3eaea3SApple OSS Distributions
646*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(s = socket(AF_INET, SOCK_DGRAM, 0), "socket");
647*5e3eaea3SApple OSS Distributions
648*5e3eaea3SApple OSS Distributions strlcpy(ifname, IF_NAME, sizeof(ifname));
649*5e3eaea3SApple OSS Distributions
650*5e3eaea3SApple OSS Distributions int error = 0;
651*5e3eaea3SApple OSS Distributions if ((error = ifnet_create(s, ifname, sizeof(ifname))) != 0) {
652*5e3eaea3SApple OSS Distributions if (error == EINVAL) {
653*5e3eaea3SApple OSS Distributions T_SKIP("The system does not support the %s cloning interface", IF_NAME);
654*5e3eaea3SApple OSS Distributions }
655*5e3eaea3SApple OSS Distributions T_SKIP("This test failed creating a %s cloning interface", IF_NAME);
656*5e3eaea3SApple OSS Distributions }
657*5e3eaea3SApple OSS Distributions T_LOG("created clone interface '%s'", ifname);
658*5e3eaea3SApple OSS Distributions
659*5e3eaea3SApple OSS Distributions struct ioc_ifreq *ioc_ifreq;
660*5e3eaea3SApple OSS Distributions for (ioc_ifreq = ioc_list; ioc_ifreq->error != -1; ioc_ifreq++) {
661*5e3eaea3SApple OSS Distributions test_sioc_ifr_bounds(ioc_ifreq, s, ifname);
662*5e3eaea3SApple OSS Distributions }
663*5e3eaea3SApple OSS Distributions print_rt_iflist2(__func__);
664*5e3eaea3SApple OSS Distributions (void)ifnet_destroy(s, ifname, true);
665*5e3eaea3SApple OSS Distributions
666*5e3eaea3SApple OSS Distributions close(s);
667*5e3eaea3SApple OSS Distributions }
668*5e3eaea3SApple OSS Distributions
669*5e3eaea3SApple OSS Distributions struct ioc_ifra {
670*5e3eaea3SApple OSS Distributions const char *description;
671*5e3eaea3SApple OSS Distributions
672*5e3eaea3SApple OSS Distributions uint8_t addr_len;
673*5e3eaea3SApple OSS Distributions uint8_t addr_fam;
674*5e3eaea3SApple OSS Distributions const char *addr_str;
675*5e3eaea3SApple OSS Distributions
676*5e3eaea3SApple OSS Distributions uint8_t broad_len;
677*5e3eaea3SApple OSS Distributions uint8_t broad_fam;
678*5e3eaea3SApple OSS Distributions const char *broad_str;
679*5e3eaea3SApple OSS Distributions
680*5e3eaea3SApple OSS Distributions uint8_t mask_len;
681*5e3eaea3SApple OSS Distributions uint8_t mask_fam;
682*5e3eaea3SApple OSS Distributions const char *mask_str;
683*5e3eaea3SApple OSS Distributions
684*5e3eaea3SApple OSS Distributions int error; // 0 means no error, -1, end of list, otherwise expected errno
685*5e3eaea3SApple OSS Distributions };
686*5e3eaea3SApple OSS Distributions
687*5e3eaea3SApple OSS Distributions static struct ioc_ifra ioc_ifra_list[] = {
688*5e3eaea3SApple OSS Distributions {
689*5e3eaea3SApple OSS Distributions .description = "fully formed",
690*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.1.1.1",
691*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET, .broad_str = "1.1.1.255",
692*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET, .mask_str = "255.255.0.0",
693*5e3eaea3SApple OSS Distributions .error = 0
694*5e3eaea3SApple OSS Distributions },
695*5e3eaea3SApple OSS Distributions {
696*5e3eaea3SApple OSS Distributions .description = "addr_len 0",
697*5e3eaea3SApple OSS Distributions .addr_len = 0, .addr_fam = AF_INET, .addr_str = "10.2.2.0",
698*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET, .broad_str = "10.2.2.255",
699*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET, .mask_str = "255.0.0.0",
700*5e3eaea3SApple OSS Distributions .error = 0
701*5e3eaea3SApple OSS Distributions },
702*5e3eaea3SApple OSS Distributions {
703*5e3eaea3SApple OSS Distributions .description = "addr_len 1",
704*5e3eaea3SApple OSS Distributions .addr_len = 1, .addr_fam = AF_INET, .addr_str = "10.2.2.1",
705*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET, .broad_str = "10.2.2.255",
706*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET, .mask_str = "255.0.0.0",
707*5e3eaea3SApple OSS Distributions .error = 0
708*5e3eaea3SApple OSS Distributions },
709*5e3eaea3SApple OSS Distributions {
710*5e3eaea3SApple OSS Distributions .description = "addr_len 250",
711*5e3eaea3SApple OSS Distributions .addr_len = 250, .addr_fam = AF_INET, .addr_str = "10.2.2.250",
712*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET, .broad_str = "10.2.2.255",
713*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET, .mask_str = "255.0.0.0",
714*5e3eaea3SApple OSS Distributions .error = 0
715*5e3eaea3SApple OSS Distributions },
716*5e3eaea3SApple OSS Distributions {
717*5e3eaea3SApple OSS Distributions .description = "addr_family AF_INET6",
718*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET6, .addr_str = "10.3.3.3",
719*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET, .broad_str = "10.3.255.255",
720*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET, .mask_str = "255.255.255.0",
721*5e3eaea3SApple OSS Distributions .error = EINVAL
722*5e3eaea3SApple OSS Distributions },
723*5e3eaea3SApple OSS Distributions {
724*5e3eaea3SApple OSS Distributions .description = "broadcast_len 0xf0",
725*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.4.4.4",
726*5e3eaea3SApple OSS Distributions .broad_len = 0xf0, .broad_fam = AF_INET, .broad_str = "10.4.4.255",
727*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET, .mask_str = "255.255.255.0",
728*5e3eaea3SApple OSS Distributions .error = 0
729*5e3eaea3SApple OSS Distributions },
730*5e3eaea3SApple OSS Distributions {
731*5e3eaea3SApple OSS Distributions .description = "broadcast_family AF_INET6",
732*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.5.5.5",
733*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET6, .broad_str = "10.5.5.255",
734*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET, .mask_str = "255.255.0.0",
735*5e3eaea3SApple OSS Distributions .error = 0
736*5e3eaea3SApple OSS Distributions },
737*5e3eaea3SApple OSS Distributions {
738*5e3eaea3SApple OSS Distributions .description = "mask_len 0xf0",
739*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.6.6.6",
740*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET, .broad_str = "1.6.6.255",
741*5e3eaea3SApple OSS Distributions .mask_len = 0xf0, .mask_fam = AF_INET, .mask_str = "255.255.0.0",
742*5e3eaea3SApple OSS Distributions .error = 0
743*5e3eaea3SApple OSS Distributions },
744*5e3eaea3SApple OSS Distributions {
745*5e3eaea3SApple OSS Distributions .description = "mask_family AF_INET6",
746*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.7.7.7",
747*5e3eaea3SApple OSS Distributions .broad_len = sizeof(struct sockaddr_in), .broad_fam = AF_INET, .broad_str = "10.7.7.255",
748*5e3eaea3SApple OSS Distributions .mask_len = sizeof(struct sockaddr_in), .mask_fam = AF_INET6, .mask_str = "255.255.0.0",
749*5e3eaea3SApple OSS Distributions .error = 0
750*5e3eaea3SApple OSS Distributions },
751*5e3eaea3SApple OSS Distributions {
752*5e3eaea3SApple OSS Distributions .description = "ifra address only",
753*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.8.8.8",
754*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
755*5e3eaea3SApple OSS Distributions .mask_len = 0, .mask_str = NULL,
756*5e3eaea3SApple OSS Distributions .error = 0
757*5e3eaea3SApple OSS Distributions },
758*5e3eaea3SApple OSS Distributions {
759*5e3eaea3SApple OSS Distributions .description = "ifra mask len 1",
760*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.9.9.1",
761*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
762*5e3eaea3SApple OSS Distributions .mask_len = 1, .mask_fam = AF_INET, .mask_str = "255.255.255.0",
763*5e3eaea3SApple OSS Distributions .error = 0
764*5e3eaea3SApple OSS Distributions },
765*5e3eaea3SApple OSS Distributions {
766*5e3eaea3SApple OSS Distributions .description = "ifra mask len 3",
767*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.9.9.3",
768*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
769*5e3eaea3SApple OSS Distributions .mask_len = 1, .mask_fam = AF_INET, .mask_str = "255.255.255.0",
770*5e3eaea3SApple OSS Distributions .error = 0
771*5e3eaea3SApple OSS Distributions },
772*5e3eaea3SApple OSS Distributions {
773*5e3eaea3SApple OSS Distributions .description = "ifra mask len 5",
774*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.9.9.5",
775*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
776*5e3eaea3SApple OSS Distributions .mask_len = 1, .mask_fam = AF_INET, .mask_str = "255.255.255.0",
777*5e3eaea3SApple OSS Distributions .error = 0
778*5e3eaea3SApple OSS Distributions },
779*5e3eaea3SApple OSS Distributions {
780*5e3eaea3SApple OSS Distributions .description = "ifra mask len 7",
781*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.9.9.7",
782*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
783*5e3eaea3SApple OSS Distributions .mask_len = 1, .mask_fam = AF_INET, .mask_str = "255.255.255.0",
784*5e3eaea3SApple OSS Distributions .error = 0
785*5e3eaea3SApple OSS Distributions },
786*5e3eaea3SApple OSS Distributions {
787*5e3eaea3SApple OSS Distributions .description = "ifra mask len 9",
788*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.9.9.9",
789*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
790*5e3eaea3SApple OSS Distributions .mask_len = 1, .mask_fam = AF_INET, .mask_str = "255.255.255.0",
791*5e3eaea3SApple OSS Distributions .error = 0
792*5e3eaea3SApple OSS Distributions },
793*5e3eaea3SApple OSS Distributions {
794*5e3eaea3SApple OSS Distributions .description = "ifra mask len 11",
795*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.9.9.11",
796*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
797*5e3eaea3SApple OSS Distributions .mask_len = 1, .mask_fam = AF_INET, .mask_str = "255.255.255.0",
798*5e3eaea3SApple OSS Distributions .error = 0
799*5e3eaea3SApple OSS Distributions },
800*5e3eaea3SApple OSS Distributions {
801*5e3eaea3SApple OSS Distributions .description = "ifra mask len 13",
802*5e3eaea3SApple OSS Distributions .addr_len = sizeof(struct sockaddr_in), .addr_fam = AF_INET, .addr_str = "10.9.9.13",
803*5e3eaea3SApple OSS Distributions .broad_len = 0, .broad_str = NULL,
804*5e3eaea3SApple OSS Distributions .mask_len = 1, .mask_fam = AF_INET, .mask_str = "255.255.255.0",
805*5e3eaea3SApple OSS Distributions .error = 0
806*5e3eaea3SApple OSS Distributions },
807*5e3eaea3SApple OSS Distributions {
808*5e3eaea3SApple OSS Distributions .description = NULL,
809*5e3eaea3SApple OSS Distributions .error = -1
810*5e3eaea3SApple OSS Distributions }
811*5e3eaea3SApple OSS Distributions };
812*5e3eaea3SApple OSS Distributions
813*5e3eaea3SApple OSS Distributions T_DECL(sioc_ifra_addr_bounds, "test bound checks on socket address passed to interface ioctls",
814*5e3eaea3SApple OSS Distributions T_META_ASROOT(true))
815*5e3eaea3SApple OSS Distributions {
816*5e3eaea3SApple OSS Distributions int s = -1;
817*5e3eaea3SApple OSS Distributions
818*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(s = socket(AF_INET, SOCK_DGRAM, 0), "socket");
819*5e3eaea3SApple OSS Distributions
820*5e3eaea3SApple OSS Distributions char ifname[IFNAMSIZ];
821*5e3eaea3SApple OSS Distributions strlcpy(ifname, IF_NAME, sizeof(ifname));
822*5e3eaea3SApple OSS Distributions int error = 0;
823*5e3eaea3SApple OSS Distributions if ((error = ifnet_create(s, ifname, sizeof(ifname))) != 0) {
824*5e3eaea3SApple OSS Distributions if (error == EINVAL) {
825*5e3eaea3SApple OSS Distributions T_SKIP("The system does not support the %s cloning interface", IF_NAME);
826*5e3eaea3SApple OSS Distributions }
827*5e3eaea3SApple OSS Distributions T_SKIP("This test failed creating a %s cloning interface", IF_NAME);
828*5e3eaea3SApple OSS Distributions }
829*5e3eaea3SApple OSS Distributions T_LOG("created clone interface '%s'", ifname);
830*5e3eaea3SApple OSS Distributions
831*5e3eaea3SApple OSS Distributions struct ioc_ifra *ioc_ifra;
832*5e3eaea3SApple OSS Distributions
833*5e3eaea3SApple OSS Distributions for (ioc_ifra = ioc_ifra_list; ioc_ifra->error != -1; ioc_ifra++) {
834*5e3eaea3SApple OSS Distributions struct in_aliasreq ifra = {};
835*5e3eaea3SApple OSS Distributions unsigned char pattern;
836*5e3eaea3SApple OSS Distributions int retval;
837*5e3eaea3SApple OSS Distributions
838*5e3eaea3SApple OSS Distributions T_LOG("");
839*5e3eaea3SApple OSS Distributions T_LOG("TEST CASE: %s, ioctl(SIOCAIFADDR, %s)", ioc_ifra->description, ioc_ifra->addr_str != NULL ? ioc_ifra->addr_str : "");
840*5e3eaea3SApple OSS Distributions
841*5e3eaea3SApple OSS Distributions if (find_unused_pattern_in_rt_if_list(&pattern, PATTERN_SIZE) == false) {
842*5e3eaea3SApple OSS Distributions T_SKIP("Could not find unused pattern in rt_if_list");
843*5e3eaea3SApple OSS Distributions return;
844*5e3eaea3SApple OSS Distributions }
845*5e3eaea3SApple OSS Distributions
846*5e3eaea3SApple OSS Distributions memset(&ifra, pattern, sizeof(ifra));
847*5e3eaea3SApple OSS Distributions
848*5e3eaea3SApple OSS Distributions strlcpy(ifra.ifra_name, ifname, sizeof(ifra.ifra_name));
849*5e3eaea3SApple OSS Distributions
850*5e3eaea3SApple OSS Distributions ifra.ifra_addr.sin_len = ioc_ifra->addr_len;
851*5e3eaea3SApple OSS Distributions ifra.ifra_addr.sin_family = ioc_ifra->addr_fam;
852*5e3eaea3SApple OSS Distributions if (ioc_ifra->addr_str != NULL) {
853*5e3eaea3SApple OSS Distributions ifra.ifra_addr.sin_addr.s_addr = inet_addr(ioc_ifra->addr_str);
854*5e3eaea3SApple OSS Distributions }
855*5e3eaea3SApple OSS Distributions
856*5e3eaea3SApple OSS Distributions ifra.ifra_broadaddr.sin_len = ioc_ifra->broad_len;
857*5e3eaea3SApple OSS Distributions ifra.ifra_broadaddr.sin_family = ioc_ifra->broad_fam;
858*5e3eaea3SApple OSS Distributions if (ioc_ifra->broad_str != NULL) {
859*5e3eaea3SApple OSS Distributions ifra.ifra_broadaddr.sin_addr.s_addr = inet_addr(ioc_ifra->broad_str);
860*5e3eaea3SApple OSS Distributions }
861*5e3eaea3SApple OSS Distributions
862*5e3eaea3SApple OSS Distributions ifra.ifra_mask.sin_len = ioc_ifra->mask_len;
863*5e3eaea3SApple OSS Distributions ifra.ifra_mask.sin_family = ioc_ifra->mask_fam;
864*5e3eaea3SApple OSS Distributions if (ioc_ifra->mask_str != NULL) {
865*5e3eaea3SApple OSS Distributions ifra.ifra_mask.sin_addr.s_addr = inet_addr(ioc_ifra->mask_str);
866*5e3eaea3SApple OSS Distributions }
867*5e3eaea3SApple OSS Distributions
868*5e3eaea3SApple OSS Distributions if (ioc_ifra->error == 0) {
869*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_SUCCESS(retval = ioctl(s, SIOCAIFADDR, &ifra), "SIOCAIFADDR retval %d", retval);
870*5e3eaea3SApple OSS Distributions } else {
871*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_FAILURE(retval = ioctl(s, SIOCAIFADDR, &ifra), EINVAL, "SIOCAIFADDR retval %d, %s", retval, strerror(errno));
872*5e3eaea3SApple OSS Distributions }
873*5e3eaea3SApple OSS Distributions
874*5e3eaea3SApple OSS Distributions T_EXPECT_EQ(check_rt_if_list_for_pattern("after ioctl SIOCAIFADDR", pattern, PATTERN_SIZE), 0, "pattern should not be found");
875*5e3eaea3SApple OSS Distributions }
876*5e3eaea3SApple OSS Distributions
877*5e3eaea3SApple OSS Distributions print_rt_iflist2(__func__);
878*5e3eaea3SApple OSS Distributions
879*5e3eaea3SApple OSS Distributions (void)ifnet_destroy(s, ifname, true);
880*5e3eaea3SApple OSS Distributions
881*5e3eaea3SApple OSS Distributions close(s);
882*5e3eaea3SApple OSS Distributions }
883*5e3eaea3SApple OSS Distributions
884*5e3eaea3SApple OSS Distributions T_DECL(sioc_ifr_dstaddr_leak, "test bound checks on socket address passed to interface ioctls",
885*5e3eaea3SApple OSS Distributions T_META_ASROOT(true))
886*5e3eaea3SApple OSS Distributions {
887*5e3eaea3SApple OSS Distributions int s = -1;
888*5e3eaea3SApple OSS Distributions struct ifreq ifr = {};
889*5e3eaea3SApple OSS Distributions unsigned char pattern;
890*5e3eaea3SApple OSS Distributions struct ifaddrs *ifap = NULL, *ifa;
891*5e3eaea3SApple OSS Distributions bool found_gif0 = false;
892*5e3eaea3SApple OSS Distributions
893*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(getifaddrs(&ifap), "getifaddrs");
894*5e3eaea3SApple OSS Distributions for (ifa = ifap; ifa; ifa = ifa->ifa_next) {
895*5e3eaea3SApple OSS Distributions if (strcmp(ifa->ifa_name, "gif0") == 0) {
896*5e3eaea3SApple OSS Distributions found_gif0 = true;
897*5e3eaea3SApple OSS Distributions break;
898*5e3eaea3SApple OSS Distributions }
899*5e3eaea3SApple OSS Distributions }
900*5e3eaea3SApple OSS Distributions freeifaddrs(ifap);
901*5e3eaea3SApple OSS Distributions ifap = NULL;
902*5e3eaea3SApple OSS Distributions if (found_gif0 == false) {
903*5e3eaea3SApple OSS Distributions T_SKIP("gif0 does not exists");
904*5e3eaea3SApple OSS Distributions }
905*5e3eaea3SApple OSS Distributions
906*5e3eaea3SApple OSS Distributions if (find_unused_pattern_in_rt_if_list(&pattern, PATTERN_SIZE) == false) {
907*5e3eaea3SApple OSS Distributions T_SKIP("Could not find unused pattern");
908*5e3eaea3SApple OSS Distributions return;
909*5e3eaea3SApple OSS Distributions }
910*5e3eaea3SApple OSS Distributions
911*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(s = socket(AF_INET, SOCK_DGRAM, 0), "socket");
912*5e3eaea3SApple OSS Distributions
913*5e3eaea3SApple OSS Distributions strlcpy(ifr.ifr_name, "gif0", sizeof(ifr.ifr_name));
914*5e3eaea3SApple OSS Distributions ifr.ifr_dstaddr.sa_family = AF_INET6;
915*5e3eaea3SApple OSS Distributions ifr.ifr_dstaddr.sa_len = 0xf0;
916*5e3eaea3SApple OSS Distributions memset(&ifr.ifr_dstaddr.sa_data, pattern, PATTERN_SIZE);
917*5e3eaea3SApple OSS Distributions
918*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_SUCCESS(ioctl(s, SIOCSIFDSTADDR, &ifr), "ioctl(SIOCSIFDSTADDR)");
919*5e3eaea3SApple OSS Distributions
920*5e3eaea3SApple OSS Distributions print_rt_iflist2(__func__);
921*5e3eaea3SApple OSS Distributions
922*5e3eaea3SApple OSS Distributions close(s);
923*5e3eaea3SApple OSS Distributions
924*5e3eaea3SApple OSS Distributions T_EXPECT_EQ(check_rt_if_list_for_pattern("AFTER", pattern, PATTERN_SIZE), 0, "pattern should not be found");
925*5e3eaea3SApple OSS Distributions }
926*5e3eaea3SApple OSS Distributions
927*5e3eaea3SApple OSS Distributions static struct ioc_ifreq ioc_list_config[] = {
928*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in), AF_INET, "0.0.0.0", EINVAL },
929*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in), AF_INET, "255.255.255.255", EINVAL },
930*5e3eaea3SApple OSS Distributions { SIOCSIFADDR, sizeof(struct sockaddr_in), AF_INET, "238.207.49.91", EINVAL },
931*5e3eaea3SApple OSS Distributions
932*5e3eaea3SApple OSS Distributions { SIOCAIFADDR_IN6, sizeof(struct sockaddr_in6), AF_INET6, "::", EINVAL },
933*5e3eaea3SApple OSS Distributions { SIOCAIFADDR_IN6, sizeof(struct sockaddr_in6), AF_INET6, "ff33:40:fd25:549c:675b::1", EINVAL },
934*5e3eaea3SApple OSS Distributions { SIOCAIFADDR_IN6, sizeof(struct sockaddr_in6), AF_INET6, "::ffff:0a0a:0a0a", EINVAL },
935*5e3eaea3SApple OSS Distributions { SIOCAIFADDR_IN6, sizeof(struct sockaddr_in6), AF_INET6, "::0a0a:0a0a", EINVAL },
936*5e3eaea3SApple OSS Distributions
937*5e3eaea3SApple OSS Distributions { 0, 0, 0, "", -1 },
938*5e3eaea3SApple OSS Distributions };
939*5e3eaea3SApple OSS Distributions
940*5e3eaea3SApple OSS Distributions static void
test_sioc_ifr_addr_config_v4(struct ioc_ifreq * ioc_ifreq,int s,const char * ifname)941*5e3eaea3SApple OSS Distributions test_sioc_ifr_addr_config_v4(struct ioc_ifreq *ioc_ifreq, int s, const char *ifname)
942*5e3eaea3SApple OSS Distributions {
943*5e3eaea3SApple OSS Distributions struct ifreq ifr = {};
944*5e3eaea3SApple OSS Distributions struct sockaddr_in *sin;
945*5e3eaea3SApple OSS Distributions
946*5e3eaea3SApple OSS Distributions T_LOG("");
947*5e3eaea3SApple OSS Distributions T_LOG("TEST CASE: %s ioctl(%s, sa_len %u, sa_family %u, %s) -> %d", __func__,
948*5e3eaea3SApple OSS Distributions ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->salen, ioc_ifreq->safamily, ioc_ifreq->sastr, ioc_ifreq->error);
949*5e3eaea3SApple OSS Distributions
950*5e3eaea3SApple OSS Distributions
951*5e3eaea3SApple OSS Distributions strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
952*5e3eaea3SApple OSS Distributions
953*5e3eaea3SApple OSS Distributions sin = (struct sockaddr_in *)(void *)&ifr.ifr_addr;
954*5e3eaea3SApple OSS Distributions sin->sin_len = ioc_ifreq->salen;
955*5e3eaea3SApple OSS Distributions sin->sin_family = ioc_ifreq->safamily;
956*5e3eaea3SApple OSS Distributions sin->sin_addr.s_addr = inet_addr(ioc_ifreq->sastr);
957*5e3eaea3SApple OSS Distributions
958*5e3eaea3SApple OSS Distributions int retval;
959*5e3eaea3SApple OSS Distributions if (ioc_ifreq->error == 0) {
960*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_SUCCESS(retval = ioctl(s, ioc_ifreq->ioc_cmd, &ifr),
961*5e3eaea3SApple OSS Distributions "%s, %s: retval %d", ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->sastr, retval);
962*5e3eaea3SApple OSS Distributions } else {
963*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_FAILURE(retval = ioctl(s, ioc_ifreq->ioc_cmd, &ifr), ioc_ifreq->error,
964*5e3eaea3SApple OSS Distributions "%s, %s: retval %d errno %s", ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->sastr, retval, strerror(errno));
965*5e3eaea3SApple OSS Distributions }
966*5e3eaea3SApple OSS Distributions
967*5e3eaea3SApple OSS Distributions fflush(stdout);
968*5e3eaea3SApple OSS Distributions fflush(stderr);
969*5e3eaea3SApple OSS Distributions }
970*5e3eaea3SApple OSS Distributions
971*5e3eaea3SApple OSS Distributions static void
test_sioc_ifr_addr_config_v6(struct ioc_ifreq * ioc_ifreq,int s,const char * ifname)972*5e3eaea3SApple OSS Distributions test_sioc_ifr_addr_config_v6(struct ioc_ifreq *ioc_ifreq, int s, const char *ifname)
973*5e3eaea3SApple OSS Distributions {
974*5e3eaea3SApple OSS Distributions struct in6_aliasreq ifra = {{ 0 }, { 0 }, { 0 }, { 0 }, 0, { 0, 0, ND6_INFINITE_LIFETIME, ND6_INFINITE_LIFETIME }};
975*5e3eaea3SApple OSS Distributions struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)(void *)&ifra.ifra_addr;
976*5e3eaea3SApple OSS Distributions struct sockaddr_in6 *mask6 = (struct sockaddr_in6 *)(void *)&ifra.ifra_prefixmask;
977*5e3eaea3SApple OSS Distributions T_LOG("");
978*5e3eaea3SApple OSS Distributions T_LOG("TEST CASE: %s ioctl(%s, sa_len %u, sa_family %u, %s) -> %d", __func__,
979*5e3eaea3SApple OSS Distributions ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->salen, ioc_ifreq->safamily, ioc_ifreq->sastr, ioc_ifreq->error);
980*5e3eaea3SApple OSS Distributions
981*5e3eaea3SApple OSS Distributions strlcpy(ifra.ifra_name, ifname, sizeof(ifra.ifra_name));
982*5e3eaea3SApple OSS Distributions
983*5e3eaea3SApple OSS Distributions sin6->sin6_len = ioc_ifreq->salen;
984*5e3eaea3SApple OSS Distributions sin6->sin6_family = ioc_ifreq->safamily;
985*5e3eaea3SApple OSS Distributions T_ASSERT_EQ(inet_pton(AF_INET6, ioc_ifreq->sastr, &sin6->sin6_addr), 1, NULL);
986*5e3eaea3SApple OSS Distributions sin6->sin6_scope_id = if_nametoindex(ifname);
987*5e3eaea3SApple OSS Distributions
988*5e3eaea3SApple OSS Distributions mask6->sin6_len = ioc_ifreq->salen;
989*5e3eaea3SApple OSS Distributions mask6->sin6_family = ioc_ifreq->safamily;
990*5e3eaea3SApple OSS Distributions T_ASSERT_EQ(inet_pton(AF_INET6, "ffff:ffff:ffff:ffff::", &mask6->sin6_addr), 1, NULL);
991*5e3eaea3SApple OSS Distributions
992*5e3eaea3SApple OSS Distributions ifra.ifra_lifetime.ia6t_vltime = ND6_INFINITE_LIFETIME;
993*5e3eaea3SApple OSS Distributions ifra.ifra_lifetime.ia6t_pltime = ND6_INFINITE_LIFETIME;
994*5e3eaea3SApple OSS Distributions
995*5e3eaea3SApple OSS Distributions int retval;
996*5e3eaea3SApple OSS Distributions if (ioc_ifreq->error == 0) {
997*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_SUCCESS(retval = ioctl(s, ioc_ifreq->ioc_cmd, &ifra),
998*5e3eaea3SApple OSS Distributions "%s, %s: retval %d", ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->sastr, retval);
999*5e3eaea3SApple OSS Distributions } else {
1000*5e3eaea3SApple OSS Distributions T_EXPECT_POSIX_FAILURE(retval = ioctl(s, ioc_ifreq->ioc_cmd, &ifra), ioc_ifreq->error,
1001*5e3eaea3SApple OSS Distributions "%s, %s: retval %d errno %s", ioc_str(ioc_ifreq->ioc_cmd), ioc_ifreq->sastr, retval, strerror(errno));
1002*5e3eaea3SApple OSS Distributions }
1003*5e3eaea3SApple OSS Distributions
1004*5e3eaea3SApple OSS Distributions fflush(stdout);
1005*5e3eaea3SApple OSS Distributions fflush(stderr);
1006*5e3eaea3SApple OSS Distributions }
1007*5e3eaea3SApple OSS Distributions
1008*5e3eaea3SApple OSS Distributions
1009*5e3eaea3SApple OSS Distributions T_DECL(sioc_ifr_addr_config, "test failure cases for interface address configuration ioctls",
1010*5e3eaea3SApple OSS Distributions T_META_ASROOT(true))
1011*5e3eaea3SApple OSS Distributions {
1012*5e3eaea3SApple OSS Distributions int s = -1;
1013*5e3eaea3SApple OSS Distributions int s6 = -1;
1014*5e3eaea3SApple OSS Distributions char ifname[IFNAMSIZ];
1015*5e3eaea3SApple OSS Distributions
1016*5e3eaea3SApple OSS Distributions T_LOG("%s", __func__);
1017*5e3eaea3SApple OSS Distributions
1018*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(s = socket(AF_INET, SOCK_DGRAM, 0), "socket");
1019*5e3eaea3SApple OSS Distributions T_QUIET; T_EXPECT_POSIX_SUCCESS(s6 = socket(AF_INET6, SOCK_DGRAM, 0), "socket");
1020*5e3eaea3SApple OSS Distributions
1021*5e3eaea3SApple OSS Distributions strlcpy(ifname, IF_NAME, sizeof(ifname));
1022*5e3eaea3SApple OSS Distributions
1023*5e3eaea3SApple OSS Distributions int error = 0;
1024*5e3eaea3SApple OSS Distributions if ((error = ifnet_create(s, ifname, sizeof(ifname))) != 0) {
1025*5e3eaea3SApple OSS Distributions if (error == EINVAL) {
1026*5e3eaea3SApple OSS Distributions T_SKIP("The system does not support the %s cloning interface", IF_NAME);
1027*5e3eaea3SApple OSS Distributions }
1028*5e3eaea3SApple OSS Distributions T_SKIP("This test failed creating a %s cloning interface", IF_NAME);
1029*5e3eaea3SApple OSS Distributions }
1030*5e3eaea3SApple OSS Distributions T_LOG("created clone interface '%s'", ifname);
1031*5e3eaea3SApple OSS Distributions
1032*5e3eaea3SApple OSS Distributions struct ioc_ifreq *ioc_ifreq;
1033*5e3eaea3SApple OSS Distributions for (ioc_ifreq = ioc_list_config; ioc_ifreq->error != -1; ioc_ifreq++) {
1034*5e3eaea3SApple OSS Distributions if (ioc_ifreq->safamily == AF_INET) {
1035*5e3eaea3SApple OSS Distributions test_sioc_ifr_addr_config_v4(ioc_ifreq, s, ifname);
1036*5e3eaea3SApple OSS Distributions } else if (ioc_ifreq->safamily == AF_INET6) {
1037*5e3eaea3SApple OSS Distributions test_sioc_ifr_addr_config_v6(ioc_ifreq, s6, ifname);
1038*5e3eaea3SApple OSS Distributions }
1039*5e3eaea3SApple OSS Distributions }
1040*5e3eaea3SApple OSS Distributions print_rt_iflist2(__func__);
1041*5e3eaea3SApple OSS Distributions (void)ifnet_destroy(s, ifname, true);
1042*5e3eaea3SApple OSS Distributions
1043*5e3eaea3SApple OSS Distributions close(s);
1044*5e3eaea3SApple OSS Distributions close(s6);
1045*5e3eaea3SApple OSS Distributions }
1046*5e3eaea3SApple OSS Distributions
1047*5e3eaea3SApple OSS Distributions /**
1048*5e3eaea3SApple OSS Distributions ** Test SIOCPROTOATTACH, SIOCPROTOATTACH_IN6
1049*5e3eaea3SApple OSS Distributions **/
1050*5e3eaea3SApple OSS Distributions static int S_socket = -1;
1051*5e3eaea3SApple OSS Distributions
1052*5e3eaea3SApple OSS Distributions static int
get_inet_dgram_socket(void)1053*5e3eaea3SApple OSS Distributions get_inet_dgram_socket(void)
1054*5e3eaea3SApple OSS Distributions {
1055*5e3eaea3SApple OSS Distributions if (S_socket >= 0) {
1056*5e3eaea3SApple OSS Distributions return S_socket;
1057*5e3eaea3SApple OSS Distributions }
1058*5e3eaea3SApple OSS Distributions S_socket = socket(AF_INET, SOCK_DGRAM, 0);
1059*5e3eaea3SApple OSS Distributions T_QUIET;
1060*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(S_socket, "socket(AF_INET, SOCK_DGRAM, 0)");
1061*5e3eaea3SApple OSS Distributions return S_socket;
1062*5e3eaea3SApple OSS Distributions }
1063*5e3eaea3SApple OSS Distributions
1064*5e3eaea3SApple OSS Distributions static void
close_inet_dgram_socket(void)1065*5e3eaea3SApple OSS Distributions close_inet_dgram_socket(void)
1066*5e3eaea3SApple OSS Distributions {
1067*5e3eaea3SApple OSS Distributions if (S_socket >= 0) {
1068*5e3eaea3SApple OSS Distributions close(S_socket);
1069*5e3eaea3SApple OSS Distributions S_socket = -1;
1070*5e3eaea3SApple OSS Distributions }
1071*5e3eaea3SApple OSS Distributions }
1072*5e3eaea3SApple OSS Distributions
1073*5e3eaea3SApple OSS Distributions static int S_socket_6 = -1;
1074*5e3eaea3SApple OSS Distributions
1075*5e3eaea3SApple OSS Distributions static int
get_inet6_dgram_socket(void)1076*5e3eaea3SApple OSS Distributions get_inet6_dgram_socket(void)
1077*5e3eaea3SApple OSS Distributions {
1078*5e3eaea3SApple OSS Distributions if (S_socket_6 >= 0) {
1079*5e3eaea3SApple OSS Distributions return S_socket_6;
1080*5e3eaea3SApple OSS Distributions }
1081*5e3eaea3SApple OSS Distributions S_socket_6 = socket(AF_INET6, SOCK_DGRAM, 0);
1082*5e3eaea3SApple OSS Distributions T_QUIET;
1083*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(S_socket_6, "socket(AF_INET6, SOCK_DGRAM, 0)");
1084*5e3eaea3SApple OSS Distributions return S_socket_6;
1085*5e3eaea3SApple OSS Distributions }
1086*5e3eaea3SApple OSS Distributions
1087*5e3eaea3SApple OSS Distributions static void
close_inet6_dgram_socket(void)1088*5e3eaea3SApple OSS Distributions close_inet6_dgram_socket(void)
1089*5e3eaea3SApple OSS Distributions {
1090*5e3eaea3SApple OSS Distributions if (S_socket_6 >= 0) {
1091*5e3eaea3SApple OSS Distributions close(S_socket_6);
1092*5e3eaea3SApple OSS Distributions S_socket_6 = -1;
1093*5e3eaea3SApple OSS Distributions }
1094*5e3eaea3SApple OSS Distributions }
1095*5e3eaea3SApple OSS Distributions
1096*5e3eaea3SApple OSS Distributions static char ifname[IFNAMSIZ];
1097*5e3eaea3SApple OSS Distributions
1098*5e3eaea3SApple OSS Distributions static void
test_proto_attach_cleanup(void)1099*5e3eaea3SApple OSS Distributions test_proto_attach_cleanup(void)
1100*5e3eaea3SApple OSS Distributions {
1101*5e3eaea3SApple OSS Distributions if (ifname[0] != '\0') {
1102*5e3eaea3SApple OSS Distributions int s;
1103*5e3eaea3SApple OSS Distributions
1104*5e3eaea3SApple OSS Distributions s = get_inet_dgram_socket();
1105*5e3eaea3SApple OSS Distributions (void)ifnet_destroy(s, ifname, false);
1106*5e3eaea3SApple OSS Distributions T_LOG("Destroyed '%s'", ifname);
1107*5e3eaea3SApple OSS Distributions ifname[0] = '\0';
1108*5e3eaea3SApple OSS Distributions }
1109*5e3eaea3SApple OSS Distributions }
1110*5e3eaea3SApple OSS Distributions
1111*5e3eaea3SApple OSS Distributions static void
test_proto_attach_atend(void)1112*5e3eaea3SApple OSS Distributions test_proto_attach_atend(void)
1113*5e3eaea3SApple OSS Distributions {
1114*5e3eaea3SApple OSS Distributions test_proto_attach_cleanup();
1115*5e3eaea3SApple OSS Distributions close_inet_dgram_socket();
1116*5e3eaea3SApple OSS Distributions close_inet6_dgram_socket();
1117*5e3eaea3SApple OSS Distributions }
1118*5e3eaea3SApple OSS Distributions
1119*5e3eaea3SApple OSS Distributions static void
test_proto_attach(void)1120*5e3eaea3SApple OSS Distributions test_proto_attach(void)
1121*5e3eaea3SApple OSS Distributions {
1122*5e3eaea3SApple OSS Distributions int error;
1123*5e3eaea3SApple OSS Distributions struct ifreq ifr;
1124*5e3eaea3SApple OSS Distributions int s;
1125*5e3eaea3SApple OSS Distributions int s6;
1126*5e3eaea3SApple OSS Distributions
1127*5e3eaea3SApple OSS Distributions T_ATEND(test_proto_attach_atend);
1128*5e3eaea3SApple OSS Distributions s = get_inet_dgram_socket();
1129*5e3eaea3SApple OSS Distributions strlcpy(ifname, "feth", sizeof(ifname));
1130*5e3eaea3SApple OSS Distributions error = ifnet_create(s, ifname, sizeof(ifname));
1131*5e3eaea3SApple OSS Distributions if (error != 0) {
1132*5e3eaea3SApple OSS Distributions if (error == EINVAL) {
1133*5e3eaea3SApple OSS Distributions T_SKIP("%s cloning interface not supported", ifname);
1134*5e3eaea3SApple OSS Distributions }
1135*5e3eaea3SApple OSS Distributions T_SKIP("%s cloning interface failed", ifname);
1136*5e3eaea3SApple OSS Distributions }
1137*5e3eaea3SApple OSS Distributions T_LOG("Created '%s'", ifname);
1138*5e3eaea3SApple OSS Distributions
1139*5e3eaea3SApple OSS Distributions /* first proto attach should succeed */
1140*5e3eaea3SApple OSS Distributions bzero(&ifr, sizeof(ifr));
1141*5e3eaea3SApple OSS Distributions strncpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
1142*5e3eaea3SApple OSS Distributions error = ioctl(s, SIOCPROTOATTACH, &ifr);
1143*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(error, "SIOCPROTOATTACH %s succeeded",
1144*5e3eaea3SApple OSS Distributions ifr.ifr_name);
1145*5e3eaea3SApple OSS Distributions
1146*5e3eaea3SApple OSS Distributions /* second proto attach should fail with EEXIST */
1147*5e3eaea3SApple OSS Distributions error = ioctl(s, SIOCPROTOATTACH, &ifr);
1148*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_FAILURE(error, EEXIST,
1149*5e3eaea3SApple OSS Distributions "SIOCPROTOATTACH %s failed as expected",
1150*5e3eaea3SApple OSS Distributions ifr.ifr_name);
1151*5e3eaea3SApple OSS Distributions
1152*5e3eaea3SApple OSS Distributions /* first proto attach should succeed */
1153*5e3eaea3SApple OSS Distributions s6 = get_inet6_dgram_socket();
1154*5e3eaea3SApple OSS Distributions bzero(&ifr, sizeof(ifr));
1155*5e3eaea3SApple OSS Distributions strncpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
1156*5e3eaea3SApple OSS Distributions error = ioctl(s6, SIOCPROTOATTACH_IN6, &ifr);
1157*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_SUCCESS(error,
1158*5e3eaea3SApple OSS Distributions "SIOCPROTOATTACH_IN6 %s succeeded",
1159*5e3eaea3SApple OSS Distributions ifr.ifr_name);
1160*5e3eaea3SApple OSS Distributions
1161*5e3eaea3SApple OSS Distributions /* second proto attach should fail with EEXIST */
1162*5e3eaea3SApple OSS Distributions error = ioctl(s6, SIOCPROTOATTACH_IN6, &ifr);
1163*5e3eaea3SApple OSS Distributions T_ASSERT_POSIX_FAILURE(error, EEXIST,
1164*5e3eaea3SApple OSS Distributions "SIOCPROTOATTACH_IN6 %s failed as expected",
1165*5e3eaea3SApple OSS Distributions ifr.ifr_name);
1166*5e3eaea3SApple OSS Distributions test_proto_attach_cleanup();
1167*5e3eaea3SApple OSS Distributions }
1168*5e3eaea3SApple OSS Distributions
1169*5e3eaea3SApple OSS Distributions T_DECL(sioc_proto_attach, "test protocol attachment",
1170*5e3eaea3SApple OSS Distributions T_META_ASROOT(true))
1171*5e3eaea3SApple OSS Distributions {
1172*5e3eaea3SApple OSS Distributions test_proto_attach();
1173*5e3eaea3SApple OSS Distributions }
1174