1*5e7646d2SAndroid Build Coastguard Worker /*
2*5e7646d2SAndroid Build Coastguard Worker * Network interface functions for the CUPS scheduler.
3*5e7646d2SAndroid Build Coastguard Worker *
4*5e7646d2SAndroid Build Coastguard Worker * Copyright © 2007-2018 by Apple Inc.
5*5e7646d2SAndroid Build Coastguard Worker * Copyright © 1997-2006 by Easy Software Products, all rights reserved.
6*5e7646d2SAndroid Build Coastguard Worker *
7*5e7646d2SAndroid Build Coastguard Worker * Licensed under Apache License v2.0. See the file "LICENSE" for more
8*5e7646d2SAndroid Build Coastguard Worker * information.
9*5e7646d2SAndroid Build Coastguard Worker */
10*5e7646d2SAndroid Build Coastguard Worker
11*5e7646d2SAndroid Build Coastguard Worker /*
12*5e7646d2SAndroid Build Coastguard Worker * Include necessary headers.
13*5e7646d2SAndroid Build Coastguard Worker */
14*5e7646d2SAndroid Build Coastguard Worker
15*5e7646d2SAndroid Build Coastguard Worker #include <cups/http-private.h>
16*5e7646d2SAndroid Build Coastguard Worker #include "cupsd.h"
17*5e7646d2SAndroid Build Coastguard Worker #include <cups/getifaddrs-internal.h>
18*5e7646d2SAndroid Build Coastguard Worker
19*5e7646d2SAndroid Build Coastguard Worker
20*5e7646d2SAndroid Build Coastguard Worker /*
21*5e7646d2SAndroid Build Coastguard Worker * Local functions...
22*5e7646d2SAndroid Build Coastguard Worker */
23*5e7646d2SAndroid Build Coastguard Worker
24*5e7646d2SAndroid Build Coastguard Worker static void cupsdNetIFFree(void);
25*5e7646d2SAndroid Build Coastguard Worker static int compare_netif(cupsd_netif_t *a, cupsd_netif_t *b);
26*5e7646d2SAndroid Build Coastguard Worker
27*5e7646d2SAndroid Build Coastguard Worker
28*5e7646d2SAndroid Build Coastguard Worker /*
29*5e7646d2SAndroid Build Coastguard Worker * 'cupsdNetIFFind()' - Find a network interface.
30*5e7646d2SAndroid Build Coastguard Worker */
31*5e7646d2SAndroid Build Coastguard Worker
32*5e7646d2SAndroid Build Coastguard Worker cupsd_netif_t * /* O - Network interface data */
cupsdNetIFFind(const char * name)33*5e7646d2SAndroid Build Coastguard Worker cupsdNetIFFind(const char *name) /* I - Name of interface */
34*5e7646d2SAndroid Build Coastguard Worker {
35*5e7646d2SAndroid Build Coastguard Worker cupsd_netif_t key; /* Search key */
36*5e7646d2SAndroid Build Coastguard Worker
37*5e7646d2SAndroid Build Coastguard Worker
38*5e7646d2SAndroid Build Coastguard Worker /*
39*5e7646d2SAndroid Build Coastguard Worker * Update the interface list as needed...
40*5e7646d2SAndroid Build Coastguard Worker */
41*5e7646d2SAndroid Build Coastguard Worker
42*5e7646d2SAndroid Build Coastguard Worker if (NetIFUpdate)
43*5e7646d2SAndroid Build Coastguard Worker cupsdNetIFUpdate();
44*5e7646d2SAndroid Build Coastguard Worker
45*5e7646d2SAndroid Build Coastguard Worker /*
46*5e7646d2SAndroid Build Coastguard Worker * Search for the named interface...
47*5e7646d2SAndroid Build Coastguard Worker */
48*5e7646d2SAndroid Build Coastguard Worker
49*5e7646d2SAndroid Build Coastguard Worker strlcpy(key.name, name, sizeof(key.name));
50*5e7646d2SAndroid Build Coastguard Worker
51*5e7646d2SAndroid Build Coastguard Worker return ((cupsd_netif_t *)cupsArrayFind(NetIFList, &key));
52*5e7646d2SAndroid Build Coastguard Worker }
53*5e7646d2SAndroid Build Coastguard Worker
54*5e7646d2SAndroid Build Coastguard Worker
55*5e7646d2SAndroid Build Coastguard Worker /*
56*5e7646d2SAndroid Build Coastguard Worker * 'cupsdNetIFFree()' - Free the current network interface list.
57*5e7646d2SAndroid Build Coastguard Worker */
58*5e7646d2SAndroid Build Coastguard Worker
59*5e7646d2SAndroid Build Coastguard Worker static void
cupsdNetIFFree(void)60*5e7646d2SAndroid Build Coastguard Worker cupsdNetIFFree(void)
61*5e7646d2SAndroid Build Coastguard Worker {
62*5e7646d2SAndroid Build Coastguard Worker cupsd_netif_t *current; /* Current interface in array */
63*5e7646d2SAndroid Build Coastguard Worker
64*5e7646d2SAndroid Build Coastguard Worker
65*5e7646d2SAndroid Build Coastguard Worker /*
66*5e7646d2SAndroid Build Coastguard Worker * Loop through the interface list and free all the records...
67*5e7646d2SAndroid Build Coastguard Worker */
68*5e7646d2SAndroid Build Coastguard Worker
69*5e7646d2SAndroid Build Coastguard Worker for (current = (cupsd_netif_t *)cupsArrayFirst(NetIFList);
70*5e7646d2SAndroid Build Coastguard Worker current;
71*5e7646d2SAndroid Build Coastguard Worker current = (cupsd_netif_t *)cupsArrayNext(NetIFList))
72*5e7646d2SAndroid Build Coastguard Worker {
73*5e7646d2SAndroid Build Coastguard Worker cupsArrayRemove(NetIFList, current);
74*5e7646d2SAndroid Build Coastguard Worker free(current);
75*5e7646d2SAndroid Build Coastguard Worker }
76*5e7646d2SAndroid Build Coastguard Worker }
77*5e7646d2SAndroid Build Coastguard Worker
78*5e7646d2SAndroid Build Coastguard Worker
79*5e7646d2SAndroid Build Coastguard Worker /*
80*5e7646d2SAndroid Build Coastguard Worker * 'cupsdNetIFUpdate()' - Update the network interface list as needed...
81*5e7646d2SAndroid Build Coastguard Worker */
82*5e7646d2SAndroid Build Coastguard Worker
83*5e7646d2SAndroid Build Coastguard Worker void
cupsdNetIFUpdate(void)84*5e7646d2SAndroid Build Coastguard Worker cupsdNetIFUpdate(void)
85*5e7646d2SAndroid Build Coastguard Worker {
86*5e7646d2SAndroid Build Coastguard Worker int match; /* Matching address? */
87*5e7646d2SAndroid Build Coastguard Worker cupsd_listener_t *lis; /* Listen address */
88*5e7646d2SAndroid Build Coastguard Worker cupsd_netif_t *temp; /* New interface */
89*5e7646d2SAndroid Build Coastguard Worker struct ifaddrs *addrs, /* Interface address list */
90*5e7646d2SAndroid Build Coastguard Worker *addr; /* Current interface address */
91*5e7646d2SAndroid Build Coastguard Worker char hostname[1024]; /* Hostname for address */
92*5e7646d2SAndroid Build Coastguard Worker size_t hostlen; /* Length of hostname */
93*5e7646d2SAndroid Build Coastguard Worker
94*5e7646d2SAndroid Build Coastguard Worker
95*5e7646d2SAndroid Build Coastguard Worker /*
96*5e7646d2SAndroid Build Coastguard Worker * Only update the list if we need to...
97*5e7646d2SAndroid Build Coastguard Worker */
98*5e7646d2SAndroid Build Coastguard Worker
99*5e7646d2SAndroid Build Coastguard Worker if (!NetIFUpdate)
100*5e7646d2SAndroid Build Coastguard Worker return;
101*5e7646d2SAndroid Build Coastguard Worker
102*5e7646d2SAndroid Build Coastguard Worker NetIFUpdate = 0;
103*5e7646d2SAndroid Build Coastguard Worker
104*5e7646d2SAndroid Build Coastguard Worker /*
105*5e7646d2SAndroid Build Coastguard Worker * Free the old interfaces...
106*5e7646d2SAndroid Build Coastguard Worker */
107*5e7646d2SAndroid Build Coastguard Worker
108*5e7646d2SAndroid Build Coastguard Worker cupsdNetIFFree();
109*5e7646d2SAndroid Build Coastguard Worker
110*5e7646d2SAndroid Build Coastguard Worker /*
111*5e7646d2SAndroid Build Coastguard Worker * Make sure we have an array...
112*5e7646d2SAndroid Build Coastguard Worker */
113*5e7646d2SAndroid Build Coastguard Worker
114*5e7646d2SAndroid Build Coastguard Worker if (!NetIFList)
115*5e7646d2SAndroid Build Coastguard Worker NetIFList = cupsArrayNew((cups_array_func_t)compare_netif, NULL);
116*5e7646d2SAndroid Build Coastguard Worker
117*5e7646d2SAndroid Build Coastguard Worker if (!NetIFList)
118*5e7646d2SAndroid Build Coastguard Worker return;
119*5e7646d2SAndroid Build Coastguard Worker
120*5e7646d2SAndroid Build Coastguard Worker /*
121*5e7646d2SAndroid Build Coastguard Worker * Grab a new list of interfaces...
122*5e7646d2SAndroid Build Coastguard Worker */
123*5e7646d2SAndroid Build Coastguard Worker
124*5e7646d2SAndroid Build Coastguard Worker if (getifaddrs(&addrs) < 0)
125*5e7646d2SAndroid Build Coastguard Worker {
126*5e7646d2SAndroid Build Coastguard Worker cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdNetIFUpdate: Unable to get interface list - %s", strerror(errno));
127*5e7646d2SAndroid Build Coastguard Worker return;
128*5e7646d2SAndroid Build Coastguard Worker }
129*5e7646d2SAndroid Build Coastguard Worker
130*5e7646d2SAndroid Build Coastguard Worker for (addr = addrs; addr != NULL; addr = addr->ifa_next)
131*5e7646d2SAndroid Build Coastguard Worker {
132*5e7646d2SAndroid Build Coastguard Worker /*
133*5e7646d2SAndroid Build Coastguard Worker * See if this interface address is IPv4 or IPv6...
134*5e7646d2SAndroid Build Coastguard Worker */
135*5e7646d2SAndroid Build Coastguard Worker
136*5e7646d2SAndroid Build Coastguard Worker if (addr->ifa_addr == NULL ||
137*5e7646d2SAndroid Build Coastguard Worker (addr->ifa_addr->sa_family != AF_INET
138*5e7646d2SAndroid Build Coastguard Worker #ifdef AF_INET6
139*5e7646d2SAndroid Build Coastguard Worker && addr->ifa_addr->sa_family != AF_INET6
140*5e7646d2SAndroid Build Coastguard Worker #endif
141*5e7646d2SAndroid Build Coastguard Worker ) ||
142*5e7646d2SAndroid Build Coastguard Worker addr->ifa_netmask == NULL || addr->ifa_name == NULL)
143*5e7646d2SAndroid Build Coastguard Worker {
144*5e7646d2SAndroid Build Coastguard Worker cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdNetIFUpdate: Ignoring \"%s\".", addr->ifa_name);
145*5e7646d2SAndroid Build Coastguard Worker continue;
146*5e7646d2SAndroid Build Coastguard Worker }
147*5e7646d2SAndroid Build Coastguard Worker
148*5e7646d2SAndroid Build Coastguard Worker /*
149*5e7646d2SAndroid Build Coastguard Worker * Try looking up the hostname for the address as needed...
150*5e7646d2SAndroid Build Coastguard Worker */
151*5e7646d2SAndroid Build Coastguard Worker
152*5e7646d2SAndroid Build Coastguard Worker if (HostNameLookups)
153*5e7646d2SAndroid Build Coastguard Worker httpAddrLookup((http_addr_t *)(addr->ifa_addr), hostname,
154*5e7646d2SAndroid Build Coastguard Worker sizeof(hostname));
155*5e7646d2SAndroid Build Coastguard Worker else
156*5e7646d2SAndroid Build Coastguard Worker {
157*5e7646d2SAndroid Build Coastguard Worker /*
158*5e7646d2SAndroid Build Coastguard Worker * Map the default server address and localhost to the server name
159*5e7646d2SAndroid Build Coastguard Worker * and localhost, respectively; for all other addresses, use the
160*5e7646d2SAndroid Build Coastguard Worker * numeric address...
161*5e7646d2SAndroid Build Coastguard Worker */
162*5e7646d2SAndroid Build Coastguard Worker
163*5e7646d2SAndroid Build Coastguard Worker if (httpAddrLocalhost((http_addr_t *)(addr->ifa_addr)))
164*5e7646d2SAndroid Build Coastguard Worker strlcpy(hostname, "localhost", sizeof(hostname));
165*5e7646d2SAndroid Build Coastguard Worker else
166*5e7646d2SAndroid Build Coastguard Worker httpAddrString((http_addr_t *)(addr->ifa_addr), hostname,
167*5e7646d2SAndroid Build Coastguard Worker sizeof(hostname));
168*5e7646d2SAndroid Build Coastguard Worker }
169*5e7646d2SAndroid Build Coastguard Worker
170*5e7646d2SAndroid Build Coastguard Worker /*
171*5e7646d2SAndroid Build Coastguard Worker * Create a new address element...
172*5e7646d2SAndroid Build Coastguard Worker */
173*5e7646d2SAndroid Build Coastguard Worker
174*5e7646d2SAndroid Build Coastguard Worker hostlen = strlen(hostname);
175*5e7646d2SAndroid Build Coastguard Worker if ((temp = calloc(1, sizeof(cupsd_netif_t) + hostlen)) == NULL)
176*5e7646d2SAndroid Build Coastguard Worker {
177*5e7646d2SAndroid Build Coastguard Worker cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdNetIFUpdate: Unable to allocate memory for interface.");
178*5e7646d2SAndroid Build Coastguard Worker break;
179*5e7646d2SAndroid Build Coastguard Worker }
180*5e7646d2SAndroid Build Coastguard Worker
181*5e7646d2SAndroid Build Coastguard Worker /*
182*5e7646d2SAndroid Build Coastguard Worker * Copy all of the information...
183*5e7646d2SAndroid Build Coastguard Worker */
184*5e7646d2SAndroid Build Coastguard Worker
185*5e7646d2SAndroid Build Coastguard Worker strlcpy(temp->name, addr->ifa_name, sizeof(temp->name));
186*5e7646d2SAndroid Build Coastguard Worker temp->hostlen = hostlen;
187*5e7646d2SAndroid Build Coastguard Worker memcpy(temp->hostname, hostname, hostlen + 1);
188*5e7646d2SAndroid Build Coastguard Worker
189*5e7646d2SAndroid Build Coastguard Worker if (addr->ifa_addr->sa_family == AF_INET)
190*5e7646d2SAndroid Build Coastguard Worker {
191*5e7646d2SAndroid Build Coastguard Worker /*
192*5e7646d2SAndroid Build Coastguard Worker * Copy IPv4 addresses...
193*5e7646d2SAndroid Build Coastguard Worker */
194*5e7646d2SAndroid Build Coastguard Worker
195*5e7646d2SAndroid Build Coastguard Worker memcpy(&(temp->address), addr->ifa_addr, sizeof(struct sockaddr_in));
196*5e7646d2SAndroid Build Coastguard Worker memcpy(&(temp->mask), addr->ifa_netmask, sizeof(struct sockaddr_in));
197*5e7646d2SAndroid Build Coastguard Worker
198*5e7646d2SAndroid Build Coastguard Worker if (addr->ifa_dstaddr)
199*5e7646d2SAndroid Build Coastguard Worker memcpy(&(temp->broadcast), addr->ifa_dstaddr,
200*5e7646d2SAndroid Build Coastguard Worker sizeof(struct sockaddr_in));
201*5e7646d2SAndroid Build Coastguard Worker }
202*5e7646d2SAndroid Build Coastguard Worker #ifdef AF_INET6
203*5e7646d2SAndroid Build Coastguard Worker else
204*5e7646d2SAndroid Build Coastguard Worker {
205*5e7646d2SAndroid Build Coastguard Worker /*
206*5e7646d2SAndroid Build Coastguard Worker * Copy IPv6 addresses...
207*5e7646d2SAndroid Build Coastguard Worker */
208*5e7646d2SAndroid Build Coastguard Worker
209*5e7646d2SAndroid Build Coastguard Worker memcpy(&(temp->address), addr->ifa_addr, sizeof(struct sockaddr_in6));
210*5e7646d2SAndroid Build Coastguard Worker memcpy(&(temp->mask), addr->ifa_netmask, sizeof(struct sockaddr_in6));
211*5e7646d2SAndroid Build Coastguard Worker
212*5e7646d2SAndroid Build Coastguard Worker if (addr->ifa_dstaddr)
213*5e7646d2SAndroid Build Coastguard Worker memcpy(&(temp->broadcast), addr->ifa_dstaddr,
214*5e7646d2SAndroid Build Coastguard Worker sizeof(struct sockaddr_in6));
215*5e7646d2SAndroid Build Coastguard Worker }
216*5e7646d2SAndroid Build Coastguard Worker #endif /* AF_INET6 */
217*5e7646d2SAndroid Build Coastguard Worker
218*5e7646d2SAndroid Build Coastguard Worker if (!(addr->ifa_flags & IFF_POINTOPOINT) &&
219*5e7646d2SAndroid Build Coastguard Worker !httpAddrLocalhost(&(temp->address)))
220*5e7646d2SAndroid Build Coastguard Worker temp->is_local = 1;
221*5e7646d2SAndroid Build Coastguard Worker
222*5e7646d2SAndroid Build Coastguard Worker /*
223*5e7646d2SAndroid Build Coastguard Worker * Determine which port to use when advertising printers...
224*5e7646d2SAndroid Build Coastguard Worker */
225*5e7646d2SAndroid Build Coastguard Worker
226*5e7646d2SAndroid Build Coastguard Worker for (lis = (cupsd_listener_t *)cupsArrayFirst(Listeners);
227*5e7646d2SAndroid Build Coastguard Worker lis;
228*5e7646d2SAndroid Build Coastguard Worker lis = (cupsd_listener_t *)cupsArrayNext(Listeners))
229*5e7646d2SAndroid Build Coastguard Worker {
230*5e7646d2SAndroid Build Coastguard Worker match = 0;
231*5e7646d2SAndroid Build Coastguard Worker
232*5e7646d2SAndroid Build Coastguard Worker if (httpAddrAny(&(lis->address)))
233*5e7646d2SAndroid Build Coastguard Worker match = 1;
234*5e7646d2SAndroid Build Coastguard Worker else if (addr->ifa_addr->sa_family == AF_INET &&
235*5e7646d2SAndroid Build Coastguard Worker lis->address.addr.sa_family == AF_INET &&
236*5e7646d2SAndroid Build Coastguard Worker (lis->address.ipv4.sin_addr.s_addr &
237*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv4.sin_addr.s_addr) ==
238*5e7646d2SAndroid Build Coastguard Worker (temp->address.ipv4.sin_addr.s_addr &
239*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv4.sin_addr.s_addr))
240*5e7646d2SAndroid Build Coastguard Worker match = 1;
241*5e7646d2SAndroid Build Coastguard Worker #ifdef AF_INET6
242*5e7646d2SAndroid Build Coastguard Worker else if (addr->ifa_addr->sa_family == AF_INET6 &&
243*5e7646d2SAndroid Build Coastguard Worker lis->address.addr.sa_family == AF_INET6 &&
244*5e7646d2SAndroid Build Coastguard Worker (lis->address.ipv6.sin6_addr.s6_addr[0] &
245*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[0]) ==
246*5e7646d2SAndroid Build Coastguard Worker (temp->address.ipv6.sin6_addr.s6_addr[0] &
247*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[0]) &&
248*5e7646d2SAndroid Build Coastguard Worker (lis->address.ipv6.sin6_addr.s6_addr[1] &
249*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[1]) ==
250*5e7646d2SAndroid Build Coastguard Worker (temp->address.ipv6.sin6_addr.s6_addr[1] &
251*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[1]) &&
252*5e7646d2SAndroid Build Coastguard Worker (lis->address.ipv6.sin6_addr.s6_addr[2] &
253*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[2]) ==
254*5e7646d2SAndroid Build Coastguard Worker (temp->address.ipv6.sin6_addr.s6_addr[2] &
255*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[2]) &&
256*5e7646d2SAndroid Build Coastguard Worker (lis->address.ipv6.sin6_addr.s6_addr[3] &
257*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[3]) ==
258*5e7646d2SAndroid Build Coastguard Worker (temp->address.ipv6.sin6_addr.s6_addr[3] &
259*5e7646d2SAndroid Build Coastguard Worker temp->mask.ipv6.sin6_addr.s6_addr[3]))
260*5e7646d2SAndroid Build Coastguard Worker match = 1;
261*5e7646d2SAndroid Build Coastguard Worker #endif /* AF_INET6 */
262*5e7646d2SAndroid Build Coastguard Worker
263*5e7646d2SAndroid Build Coastguard Worker if (match)
264*5e7646d2SAndroid Build Coastguard Worker {
265*5e7646d2SAndroid Build Coastguard Worker temp->port = httpAddrPort(&(lis->address));
266*5e7646d2SAndroid Build Coastguard Worker break;
267*5e7646d2SAndroid Build Coastguard Worker }
268*5e7646d2SAndroid Build Coastguard Worker }
269*5e7646d2SAndroid Build Coastguard Worker
270*5e7646d2SAndroid Build Coastguard Worker /*
271*5e7646d2SAndroid Build Coastguard Worker * Add it to the array...
272*5e7646d2SAndroid Build Coastguard Worker */
273*5e7646d2SAndroid Build Coastguard Worker
274*5e7646d2SAndroid Build Coastguard Worker cupsArrayAdd(NetIFList, temp);
275*5e7646d2SAndroid Build Coastguard Worker
276*5e7646d2SAndroid Build Coastguard Worker cupsdLogMessage(CUPSD_LOG_DEBUG, "cupsdNetIFUpdate: \"%s\" = %s:%d",
277*5e7646d2SAndroid Build Coastguard Worker temp->name, temp->hostname, temp->port);
278*5e7646d2SAndroid Build Coastguard Worker }
279*5e7646d2SAndroid Build Coastguard Worker
280*5e7646d2SAndroid Build Coastguard Worker freeifaddrs(addrs);
281*5e7646d2SAndroid Build Coastguard Worker }
282*5e7646d2SAndroid Build Coastguard Worker
283*5e7646d2SAndroid Build Coastguard Worker
284*5e7646d2SAndroid Build Coastguard Worker /*
285*5e7646d2SAndroid Build Coastguard Worker * 'compare_netif()' - Compare two network interfaces.
286*5e7646d2SAndroid Build Coastguard Worker */
287*5e7646d2SAndroid Build Coastguard Worker
288*5e7646d2SAndroid Build Coastguard Worker static int /* O - Result of comparison */
compare_netif(cupsd_netif_t * a,cupsd_netif_t * b)289*5e7646d2SAndroid Build Coastguard Worker compare_netif(cupsd_netif_t *a, /* I - First network interface */
290*5e7646d2SAndroid Build Coastguard Worker cupsd_netif_t *b) /* I - Second network interface */
291*5e7646d2SAndroid Build Coastguard Worker {
292*5e7646d2SAndroid Build Coastguard Worker return (strcmp(a->name, b->name));
293*5e7646d2SAndroid Build Coastguard Worker }
294