blob: 6e6f9692e647e3050314d594713fa61dd142088d [file] [log] [blame]
Kristian Monsen5ab50182010-05-14 18:53:44 +01001/***************************************************************************
2 * _ _ ____ _
3 * Project ___| | | | _ \| |
4 * / __| | | | |_) | |
5 * | (__| |_| | _ <| |___
6 * \___|\___/|_| \_\_____|
7 *
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -07008 * Copyright (C) 1998 - 2015, Daniel Stenberg, <daniel@haxx.se>, et al.
Kristian Monsen5ab50182010-05-14 18:53:44 +01009 *
10 * This software is licensed as described in the file COPYING, which
11 * you should have received as part of this distribution. The terms
12 * are also available at http://curl.haxx.se/docs/copyright.html.
13 *
14 * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15 * copies of the Software, and permit persons to whom the Software is
16 * furnished to do so, under the terms of the COPYING file.
17 *
18 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19 * KIND, either express or implied.
20 *
21 ***************************************************************************/
22
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -070023#include "curl_setup.h"
Kristian Monsen5ab50182010-05-14 18:53:44 +010024
Kristian Monsen5ab50182010-05-14 18:53:44 +010025#ifdef HAVE_NETINET_IN_H
26# include <netinet/in.h>
27#endif
28#ifdef HAVE_ARPA_INET_H
29# include <arpa/inet.h>
30#endif
31#ifdef HAVE_NET_IF_H
32# include <net/if.h>
33#endif
34#ifdef HAVE_SYS_IOCTL_H
35# include <sys/ioctl.h>
36#endif
37#ifdef HAVE_NETDB_H
38# include <netdb.h>
39#endif
40#ifdef HAVE_SYS_SOCKIO_H
41# include <sys/sockio.h>
42#endif
43#ifdef HAVE_IFADDRS_H
44# include <ifaddrs.h>
45#endif
46#ifdef HAVE_STROPTS_H
47# include <stropts.h>
48#endif
49#ifdef __VMS
50# include <inet.h>
51#endif
52
53#include "inet_ntop.h"
54#include "strequal.h"
55#include "if2ip.h"
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -070056#include "curl_printf.h"
Kristian Monsen5ab50182010-05-14 18:53:44 +010057
58#include "curl_memory.h"
59/* The last #include file should be: */
60#include "memdebug.h"
61
62/* ------------------------------------------------------------------ */
63
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -070064/* Return the scope of the given address. */
65unsigned int Curl_ipv6_scope(const struct sockaddr *sa)
66{
67#ifndef ENABLE_IPV6
68 (void) sa;
69#else
70 if(sa->sa_family == AF_INET6) {
71 const struct sockaddr_in6 * sa6 = (const struct sockaddr_in6 *) sa;
72 const unsigned char * b = sa6->sin6_addr.s6_addr;
73 unsigned short w = (unsigned short) ((b[0] << 8) | b[1]);
74
75 switch(w & 0xFFC0) {
76 case 0xFE80:
77 return IPV6_SCOPE_LINKLOCAL;
78 case 0xFEC0:
79 return IPV6_SCOPE_SITELOCAL;
80 case 0x0000:
81 w = b[1] | b[2] | b[3] | b[4] | b[5] | b[6] | b[7] | b[8] | b[9] |
82 b[10] | b[11] | b[12] | b[13] | b[14];
83 if(w || b[15] != 0x01)
84 break;
85 return IPV6_SCOPE_NODELOCAL;
86 default:
87 break;
88 }
89 }
90#endif
91
92 return IPV6_SCOPE_GLOBAL;
93}
94
95
Kristian Monsen5ab50182010-05-14 18:53:44 +010096#if defined(HAVE_GETIFADDRS)
97
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -070098bool Curl_if_is_interface_name(const char *interf)
Kristian Monsen5ab50182010-05-14 18:53:44 +010099{
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700100 bool result = FALSE;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100101
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700102 struct ifaddrs *iface, *head;
103
104 if(getifaddrs(&head) >= 0) {
105 for(iface=head; iface != NULL; iface=iface->ifa_next) {
106 if(curl_strequal(iface->ifa_name, interf)) {
107 result = TRUE;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100108 break;
109 }
110 }
111 freeifaddrs(head);
112 }
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700113 return result;
114}
115
116if2ip_result_t Curl_if2ip(int af, unsigned int remote_scope,
117 unsigned int remote_scope_id, const char *interf,
118 char *buf, int buf_size)
119{
120 struct ifaddrs *iface, *head;
121 if2ip_result_t res = IF2IP_NOT_FOUND;
122
123#ifndef ENABLE_IPV6
124 (void) remote_scope;
125
126#ifndef HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID
127 (void) remote_scope_id;
128#endif
129
130#endif
131
132 if(getifaddrs(&head) >= 0) {
133 for(iface = head; iface != NULL; iface=iface->ifa_next) {
134 if(iface->ifa_addr != NULL) {
135 if(iface->ifa_addr->sa_family == af) {
136 if(curl_strequal(iface->ifa_name, interf)) {
137 void *addr;
138 char *ip;
139 char scope[12] = "";
140 char ipstr[64];
141#ifdef ENABLE_IPV6
142 if(af == AF_INET6) {
143 unsigned int scopeid = 0;
144 unsigned int ifscope = Curl_ipv6_scope(iface->ifa_addr);
145
146 if(ifscope != remote_scope) {
147 /* We are interested only in interface addresses whose
148 scope matches the remote address we want to
149 connect to: global for global, link-local for
150 link-local, etc... */
151 if(res == IF2IP_NOT_FOUND) res = IF2IP_AF_NOT_SUPPORTED;
152 continue;
153 }
154
155 addr = &((struct sockaddr_in6 *)iface->ifa_addr)->sin6_addr;
156#ifdef HAVE_SOCKADDR_IN6_SIN6_SCOPE_ID
157 /* Include the scope of this interface as part of the address */
158 scopeid =
159 ((struct sockaddr_in6 *)iface->ifa_addr)->sin6_scope_id;
160
161 /* If given, scope id should match. */
162 if(remote_scope_id && scopeid != remote_scope_id) {
163 if(res == IF2IP_NOT_FOUND)
164 res = IF2IP_AF_NOT_SUPPORTED;
165
166 continue;
167 }
168#endif
169 if(scopeid)
170 snprintf(scope, sizeof(scope), "%%%u", scopeid);
171 }
172 else
173#endif
174 addr = &((struct sockaddr_in *)iface->ifa_addr)->sin_addr;
175 res = IF2IP_FOUND;
176 ip = (char *) Curl_inet_ntop(af, addr, ipstr, sizeof(ipstr));
177 snprintf(buf, buf_size, "%s%s", ip, scope);
178 break;
179 }
180 }
181 else if((res == IF2IP_NOT_FOUND) &&
182 curl_strequal(iface->ifa_name, interf)) {
183 res = IF2IP_AF_NOT_SUPPORTED;
184 }
185 }
186 }
187
188 freeifaddrs(head);
189 }
190
191 return res;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100192}
193
194#elif defined(HAVE_IOCTL_SIOCGIFADDR)
195
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700196bool Curl_if_is_interface_name(const char *interf)
197{
198 /* This is here just to support the old interfaces */
199 char buf[256];
200
201 return (Curl_if2ip(AF_INET, 0 /* unused */, 0, interf, buf, sizeof(buf)) ==
202 IF2IP_NOT_FOUND) ? FALSE : TRUE;
203}
204
205if2ip_result_t Curl_if2ip(int af, unsigned int remote_scope,
206 unsigned int remote_scope_id, const char *interf,
207 char *buf, int buf_size)
Kristian Monsen5ab50182010-05-14 18:53:44 +0100208{
209 struct ifreq req;
210 struct in_addr in;
211 struct sockaddr_in *s;
212 curl_socket_t dummy;
213 size_t len;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100214
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700215 (void)remote_scope;
216 (void)remote_scope_id;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100217
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700218 if(!interf || (af != AF_INET))
219 return IF2IP_NOT_FOUND;
220
221 len = strlen(interf);
Kristian Monsen5ab50182010-05-14 18:53:44 +0100222 if(len >= sizeof(req.ifr_name))
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700223 return IF2IP_NOT_FOUND;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100224
225 dummy = socket(AF_INET, SOCK_STREAM, 0);
226 if(CURL_SOCKET_BAD == dummy)
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700227 return IF2IP_NOT_FOUND;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100228
229 memset(&req, 0, sizeof(req));
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700230 memcpy(req.ifr_name, interf, len+1);
Kristian Monsen5ab50182010-05-14 18:53:44 +0100231 req.ifr_addr.sa_family = AF_INET;
232
233 if(ioctl(dummy, SIOCGIFADDR, &req) < 0) {
234 sclose(dummy);
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700235 /* With SIOCGIFADDR, we cannot tell the difference between an interface
236 that does not exist and an interface that has no address of the
237 correct family. Assume the interface does not exist */
238 return IF2IP_NOT_FOUND;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100239 }
240
241 s = (struct sockaddr_in *)&req.ifr_addr;
242 memcpy(&in, &s->sin_addr, sizeof(in));
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700243 Curl_inet_ntop(s->sin_family, &in, buf, buf_size);
Kristian Monsen5ab50182010-05-14 18:53:44 +0100244
245 sclose(dummy);
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700246 return IF2IP_FOUND;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100247}
248
249#else
250
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700251bool Curl_if_is_interface_name(const char *interf)
252{
253 (void) interf;
254
255 return FALSE;
256}
257
258if2ip_result_t Curl_if2ip(int af, unsigned int remote_scope,
259 unsigned int remote_scope_id, const char *interf,
260 char *buf, int buf_size)
Kristian Monsen5ab50182010-05-14 18:53:44 +0100261{
262 (void) af;
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700263 (void) remote_scope;
264 (void) remote_scope_id;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100265 (void) interf;
266 (void) buf;
267 (void) buf_size;
Bertrand SIMONNETe6cd7382015-07-01 15:39:44 -0700268 return IF2IP_NOT_FOUND;
Kristian Monsen5ab50182010-05-14 18:53:44 +0100269}
270
271#endif