blob: abc706c2e8a3c1a102d775c8eee2bf3a43987fe1 [file] [log] [blame]
Adrien Béraud612b55b2023-05-29 10:42:04 -04001/*
2 * Copyright (C) 2004-2023 Savoir-faire Linux Inc.
3 *
Adrien Béraudcb753622023-07-17 22:32:49 -04004 * This program is free software: you can redistribute it and/or modify
Adrien Béraud612b55b2023-05-29 10:42:04 -04005 * it under the terms of the GNU General Public License as published by
Adrien Béraudcb753622023-07-17 22:32:49 -04006 * the Free Software Foundation, either version 3 of the License, or
Adrien Béraud612b55b2023-05-29 10:42:04 -04007 * (at your option) any later version.
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Adrien Béraudcb753622023-07-17 22:32:49 -040011 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Adrien Béraud612b55b2023-05-29 10:42:04 -040012 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
Adrien Béraudcb753622023-07-17 22:32:49 -040015 * along with this program. If not, see <https://www.gnu.org/licenses/>.
Adrien Béraud612b55b2023-05-29 10:42:04 -040016 */
Adrien Béraud612b55b2023-05-29 10:42:04 -040017#include "ip_utils.h"
Adrien Béraud80c14e12023-07-18 16:13:15 -040018#include "sip_utils.h"
19#include "string_utils.h"
Adrien Béraud612b55b2023-05-29 10:42:04 -040020
Adrien Béraud80c14e12023-07-18 16:13:15 -040021#include <fmt/format.h>
Adrien Béraud612b55b2023-05-29 10:42:04 -040022
23#include <sys/types.h>
24#include <unistd.h>
25#include <limits.h>
26
27#ifdef _WIN32
28#define InetPtonA inet_pton
29WINSOCK_API_LINKAGE INT WSAAPI InetPtonA(INT Family, LPCSTR pStringBuf, PVOID pAddr);
30#else
31#include <arpa/inet.h>
32#include <arpa/nameser.h>
33#include <resolv.h>
34#include <netdb.h>
35#include <netinet/ip.h>
36#include <net/if.h>
37#include <ifaddrs.h>
38#include <sys/ioctl.h>
39#endif
40
41#ifndef HOST_NAME_MAX
42#ifdef MAX_COMPUTERNAME_LENGTH
43#define HOST_NAME_MAX MAX_COMPUTERNAME_LENGTH
44#else
45// Max 255 chars as per RFC 1035
46#define HOST_NAME_MAX 255
47#endif
48#endif
49
Adrien Béraud1ae60aa2023-07-07 09:55:09 -040050namespace dhtnet {
Adrien Béraud612b55b2023-05-29 10:42:04 -040051
Adrien Béraud9ed4dd92023-07-19 00:13:05 -040052namespace sip_utils {
Adrien Béraud612b55b2023-05-29 10:42:04 -040053std::string_view
54sip_strerror(pj_status_t code)
55{
56 thread_local char err_msg[PJ_ERR_MSG_SIZE];
Adrien Béraud80c14e12023-07-18 16:13:15 -040057 return sip_utils::as_view(pj_strerror(code, err_msg, sizeof err_msg));
Adrien Béraud612b55b2023-05-29 10:42:04 -040058}
Adrien Béraud9ed4dd92023-07-19 00:13:05 -040059}
Adrien Béraud612b55b2023-05-29 10:42:04 -040060
61std::string
62ip_utils::getHostname()
63{
64 char hostname[HOST_NAME_MAX];
65 if (gethostname(hostname, HOST_NAME_MAX))
66 return {};
67 return hostname;
68}
69
70int
71ip_utils::getHostName(char* out, size_t out_len)
72{
73 char tempstr[INET_ADDRSTRLEN];
74 const char* p = NULL;
75#ifdef _WIN32
76 struct hostent* h = NULL;
77 struct sockaddr_in localAddr;
78 memset(&localAddr, 0, sizeof(localAddr));
79 gethostname(out, out_len);
80 h = gethostbyname(out);
81 if (h != NULL) {
82 memcpy(&localAddr.sin_addr, h->h_addr_list[0], 4);
83 p = inet_ntop(AF_INET, &localAddr.sin_addr, tempstr, sizeof(tempstr));
84 if (p)
85 strncpy(out, p, out_len);
86 else
87 return -1;
88 } else {
89 return -1;
90 }
91#elif (defined(BSD) && BSD >= 199306) || defined(__FreeBSD_kernel__)
92 int retVal = 0;
93 struct ifaddrs* ifap;
94 struct ifaddrs* ifa;
95 if (getifaddrs(&ifap) != 0)
96 return -1;
97 // Cycle through available interfaces.
98 for (ifa = ifap; ifa != NULL; ifa = ifa->ifa_next) {
99 // Skip loopback, point-to-point and down interfaces.
100 // except don't skip down interfaces if we're trying to get
101 // a list of configurable interfaces.
102 if ((ifa->ifa_flags & IFF_LOOPBACK) || (!(ifa->ifa_flags & IFF_UP)))
103 continue;
104 if (ifa->ifa_addr->sa_family == AF_INET) {
105 if (((struct sockaddr_in*) (ifa->ifa_addr))->sin_addr.s_addr == htonl(INADDR_LOOPBACK)) {
106 // We don't want the loopback interface. Go to next one.
107 continue;
108 }
109 p = inet_ntop(AF_INET,
110 &((struct sockaddr_in*) (ifa->ifa_addr))->sin_addr,
111 tempstr,
112 sizeof(tempstr));
113 if (p)
114 strncpy(out, p, out_len);
115 else
116 retVal = -1;
117 break;
118 }
119 }
120 freeifaddrs(ifap);
121 retVal = ifa ? 0 : -1;
122 return retVal;
123#else
124 struct ifconf ifConf;
125 struct ifreq ifReq;
126 struct sockaddr_in localAddr;
127 char szBuffer[MAX_INTERFACE * sizeof(struct ifreq)];
128 int nResult;
129 int localSock;
130 memset(&ifConf, 0, sizeof(ifConf));
131 memset(&ifReq, 0, sizeof(ifReq));
132 memset(szBuffer, 0, sizeof(szBuffer));
133 memset(&localAddr, 0, sizeof(localAddr));
134 // Create an unbound datagram socket to do the SIOCGIFADDR ioctl on.
135 localSock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
136 if (localSock == INVALID_SOCKET)
137 return -1;
138 /* Get the interface configuration information... */
139 ifConf.ifc_len = (int) sizeof szBuffer;
140 ifConf.ifc_ifcu.ifcu_buf = (caddr_t) szBuffer;
141 nResult = ioctl(localSock, SIOCGIFCONF, &ifConf);
142 if (nResult < 0) {
143 close(localSock);
144 return -1;
145 }
146 unsigned int i;
147 unsigned int j = 0;
148 // Cycle through the list of interfaces looking for IP addresses.
149 for (i = 0u; i < (unsigned int) ifConf.ifc_len && j < MIN_INTERFACE;) {
150 struct ifreq* pifReq = (struct ifreq*) ((caddr_t) ifConf.ifc_req + i);
151 i += sizeof *pifReq;
152 // See if this is the sort of interface we want to deal with.
153 memset(ifReq.ifr_name, 0, sizeof(ifReq.ifr_name));
154 strncpy(ifReq.ifr_name, pifReq->ifr_name, sizeof(ifReq.ifr_name));
155 ioctl(localSock, SIOCGIFFLAGS, &ifReq);
156 // Skip loopback, point-to-point and down interfaces.
157 // except don't skip down interfaces if we're trying to get
158 // a list of configurable interfaces.
159 if ((ifReq.ifr_flags & IFF_LOOPBACK) || (!(ifReq.ifr_flags & IFF_UP)))
160 continue;
161 if (pifReq->ifr_addr.sa_family == AF_INET) {
162 memcpy(&localAddr, &pifReq->ifr_addr, sizeof pifReq->ifr_addr);
163 if (localAddr.sin_addr.s_addr == htonl(INADDR_LOOPBACK)) {
164 // We don't want the loopback interface. Go to the next one.
165 continue;
166 }
167 }
168 j++; // Increment j if we found an address which is not loopback and is up.
169 }
170 close(localSock);
171 p = inet_ntop(AF_INET, &localAddr.sin_addr, tempstr, sizeof(tempstr));
172 if (p)
173 strncpy(out, p, out_len);
174 else
175 return -1;
176#endif
177 return 0;
178}
179std::string
180ip_utils::getGateway(char* localHost, ip_utils::subnet_mask prefix)
181{
182 std::string_view localHostStr(localHost);
183 if (prefix == ip_utils::subnet_mask::prefix_32bit)
184 return std::string(localHostStr);
185 std::string defaultGw {};
186 // Make a vector of each individual number in the ip address.
187 std::vector<std::string_view> tokens = split_string(localHostStr, '.');
188 // Build a gateway address from the individual ip components.
189 for (unsigned i = 0; i <= (unsigned) prefix; i++)
Adrien Béraud80c14e12023-07-18 16:13:15 -0400190 defaultGw = fmt::format("{:s}{:s}.", defaultGw, tokens[i]);
Adrien Béraud612b55b2023-05-29 10:42:04 -0400191 for (unsigned i = (unsigned) ip_utils::subnet_mask::prefix_32bit;
192 i > (unsigned) prefix + 1;
193 i--)
194 defaultGw += "0.";
195 defaultGw += "1";
196 return defaultGw;
197}
198
199IpAddr
200ip_utils::getLocalGateway()
201{
202 char localHostBuf[INET_ADDRSTRLEN];
203 if (ip_utils::getHostName(localHostBuf, INET_ADDRSTRLEN) < 0) {
Morteza Namvar5f639522023-07-04 17:08:58 -0400204 // JAMI_WARN("Couldn't find local host");
Adrien Béraud612b55b2023-05-29 10:42:04 -0400205 return {};
206 } else {
207 return IpAddr(ip_utils::getGateway(localHostBuf, ip_utils::subnet_mask::prefix_24bit));
208 }
209}
210
211std::vector<IpAddr>
212ip_utils::getAddrList(std::string_view name, pj_uint16_t family)
213{
214 std::vector<IpAddr> ipList;
215 if (name.empty())
216 return ipList;
217 if (IpAddr::isValid(name, family)) {
218 ipList.emplace_back(name);
219 return ipList;
220 }
221
222 static constexpr unsigned MAX_ADDR_NUM = 128;
223 pj_addrinfo res[MAX_ADDR_NUM];
224 unsigned addr_num = MAX_ADDR_NUM;
225 const pj_str_t pjname(sip_utils::CONST_PJ_STR(name));
226 auto status = pj_getaddrinfo(family, &pjname, &addr_num, res);
227 if (status != PJ_SUCCESS) {
Morteza Namvar5f639522023-07-04 17:08:58 -0400228 // JAMI_ERR("Error resolving %.*s : %s",
229 // (int) name.size(),
230 // name.data(),
231 // sip_utils::sip_strerror(status).c_str());
Adrien Béraud612b55b2023-05-29 10:42:04 -0400232 return ipList;
233 }
234
235 for (unsigned i = 0; i < addr_num; i++) {
236 bool found = false;
237 for (const auto& ip : ipList)
238 if (!pj_sockaddr_cmp(&ip, &res[i].ai_addr)) {
239 found = true;
240 break;
241 }
242 if (!found)
243 ipList.emplace_back(res[i].ai_addr);
244 }
245
246 return ipList;
247}
248
249bool
250ip_utils::haveCommonAddr(const std::vector<IpAddr>& a, const std::vector<IpAddr>& b)
251{
252 for (const auto& i : a) {
253 for (const auto& j : b) {
254 if (i == j)
255 return true;
256 }
257 }
258 return false;
259}
260
261IpAddr
262ip_utils::getLocalAddr(pj_uint16_t family)
263{
264 IpAddr ip_addr {};
265 pj_status_t status = pj_gethostip(family, ip_addr.pjPtr());
266 if (status == PJ_SUCCESS) {
267 return ip_addr;
268 }
Morteza Namvar5f639522023-07-04 17:08:58 -0400269 // JAMI_WARN("Could not get preferred address familly (%s)",
270 // (family == pj_AF_INET6()) ? "IPv6" : "IPv4");
Adrien Béraud612b55b2023-05-29 10:42:04 -0400271 family = (family == pj_AF_INET()) ? pj_AF_INET6() : pj_AF_INET();
272 status = pj_gethostip(family, ip_addr.pjPtr());
273 if (status == PJ_SUCCESS) {
274 return ip_addr;
275 }
Morteza Namvar5f639522023-07-04 17:08:58 -0400276 // JAMI_ERR("Could not get local IP");
Adrien Béraud612b55b2023-05-29 10:42:04 -0400277 return ip_addr;
278}
279
280IpAddr
281ip_utils::getInterfaceAddr(const std::string& interface, pj_uint16_t family)
282{
283 if (interface == DEFAULT_INTERFACE)
284 return getLocalAddr(family);
285
286 IpAddr addr {};
287
288#ifndef _WIN32
289 const auto unix_family = family == pj_AF_INET() ? AF_INET : AF_INET6;
290
291 int fd = socket(unix_family, SOCK_DGRAM, 0);
292 if (fd < 0) {
Morteza Namvar5f639522023-07-04 17:08:58 -0400293 // JAMI_ERR("Could not open socket: %m");
Adrien Béraud612b55b2023-05-29 10:42:04 -0400294 return addr;
295 }
296
297 if (unix_family == AF_INET6) {
298 int val = family != pj_AF_UNSPEC();
299 if (setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, (void*) &val, sizeof(val)) < 0) {
Morteza Namvar5f639522023-07-04 17:08:58 -0400300 // JAMI_ERR("Could not setsockopt: %m");
Adrien Béraud612b55b2023-05-29 10:42:04 -0400301 close(fd);
302 return addr;
303 }
304 }
305
306 ifreq ifr;
307 strncpy(ifr.ifr_name, interface.c_str(), sizeof ifr.ifr_name);
308 // guarantee that ifr_name is NULL-terminated
309 ifr.ifr_name[sizeof(ifr.ifr_name) - 1] = '\0';
310
311 memset(&ifr.ifr_addr, 0, sizeof(ifr.ifr_addr));
312 ifr.ifr_addr.sa_family = unix_family;
313
314 ioctl(fd, SIOCGIFADDR, &ifr);
315 close(fd);
316
317 addr = ifr.ifr_addr;
318 if (addr.isUnspecified())
319 return getLocalAddr(addr.getFamily());
320#else // _WIN32
321 struct addrinfo hints;
322 struct addrinfo* result = NULL;
323 struct sockaddr_in* sockaddr_ipv4;
324 struct sockaddr_in6* sockaddr_ipv6;
325
326 ZeroMemory(&hints, sizeof(hints));
327
328 DWORD dwRetval = getaddrinfo(interface.c_str(), "0", &hints, &result);
329 if (dwRetval != 0) {
Morteza Namvar5f639522023-07-04 17:08:58 -0400330 // JAMI_ERR("getaddrinfo failed with error: %lu", dwRetval);
Adrien Béraud612b55b2023-05-29 10:42:04 -0400331 return addr;
332 }
333
334 switch (result->ai_family) {
335 sockaddr_ipv4 = (struct sockaddr_in*) result->ai_addr;
336 addr = sockaddr_ipv4->sin_addr;
337 break;
338 case AF_INET6:
339 sockaddr_ipv6 = (struct sockaddr_in6*) result->ai_addr;
340 addr = sockaddr_ipv6->sin6_addr;
341 break;
342 default:
343 break;
344 }
345
346 if (addr.isUnspecified())
347 return getLocalAddr(addr.getFamily());
348#endif // !_WIN32
349
350 return addr;
351}
352
353std::vector<std::string>
354ip_utils::getAllIpInterfaceByName()
355{
356 std::vector<std::string> ifaceList;
357 ifaceList.push_back("default");
358#ifndef _WIN32
359 static ifreq ifreqs[20];
360 ifconf ifconf;
361
362 ifconf.ifc_buf = (char*) (ifreqs);
363 ifconf.ifc_len = sizeof(ifreqs);
364
365 int sock = socket(AF_INET6, SOCK_STREAM, 0);
366
367 if (sock >= 0) {
368 if (ioctl(sock, SIOCGIFCONF, &ifconf) >= 0)
369 for (unsigned i = 0; i < ifconf.ifc_len / sizeof(ifreq); ++i)
370 ifaceList.push_back(std::string(ifreqs[i].ifr_name));
371
372 close(sock);
373 }
374
375#else
Morteza Namvar5f639522023-07-04 17:08:58 -0400376 // JAMI_ERR("Not implemented yet. (iphlpapi.h problem)");
Adrien Béraud612b55b2023-05-29 10:42:04 -0400377#endif
378 return ifaceList;
379}
380
381std::vector<std::string>
382ip_utils::getAllIpInterface()
383{
384 pj_sockaddr addrList[16];
385 unsigned addrCnt = PJ_ARRAY_SIZE(addrList);
386
387 std::vector<std::string> ifaceList;
388
389 if (pj_enum_ip_interface(pj_AF_UNSPEC(), &addrCnt, addrList) == PJ_SUCCESS) {
390 for (unsigned i = 0; i < addrCnt; i++) {
391 char addr[PJ_INET6_ADDRSTRLEN];
392 pj_sockaddr_print(&addrList[i], addr, sizeof(addr), 0);
393 ifaceList.push_back(std::string(addr));
394 }
395 }
396
397 return ifaceList;
398}
399
400std::vector<IpAddr>
401ip_utils::getLocalNameservers()
402{
403 std::vector<IpAddr> res;
404#if defined __ANDROID__ || defined _WIN32 || TARGET_OS_IPHONE
405#ifdef _MSC_VER
406#pragma message(__FILE__ "(" STR2(__LINE__) ") : -NOTE- " \
407 "Not implemented")
408#else
409#warning "Not implemented"
410#endif
411#else
412 if (not(_res.options & RES_INIT))
413 res_init();
414 res.insert(res.end(), _res.nsaddr_list, _res.nsaddr_list + _res.nscount);
415#endif
416 return res;
417}
418
419bool
420IpAddr::isValid(std::string_view address, pj_uint16_t family)
421{
422 const pj_str_t pjstring(sip_utils::CONST_PJ_STR(address));
423 pj_str_t ret_str;
424 pj_uint16_t ret_port;
425 int ret_family;
426 auto status = pj_sockaddr_parse2(pj_AF_UNSPEC(), 0, &pjstring, &ret_str, &ret_port, &ret_family);
427 if (status != PJ_SUCCESS || (family != pj_AF_UNSPEC() && ret_family != family))
428 return false;
429
430 char buf[PJ_INET6_ADDRSTRLEN];
431 pj_str_t addr_with_null = {buf, 0};
432 pj_strncpy_with_null(&addr_with_null, &ret_str, sizeof(buf));
433 struct sockaddr sa;
434 return inet_pton(ret_family == pj_AF_INET6() ? AF_INET6 : AF_INET, buf, &(sa.sa_data)) == 1;
435}
436
437bool
438IpAddr::isUnspecified() const
439{
440 switch (addr.addr.sa_family) {
441 case AF_INET:
442 return IN_IS_ADDR_UNSPECIFIED(&addr.ipv4.sin_addr);
443 case AF_INET6:
444 return IN6_IS_ADDR_UNSPECIFIED(reinterpret_cast<const in6_addr*>(&addr.ipv6.sin6_addr));
445 default:
446 return true;
447 }
448}
449
450bool
451IpAddr::isLoopback() const
452{
453 switch (addr.addr.sa_family) {
454 case AF_INET: {
455 auto addr_host = ntohl(addr.ipv4.sin_addr.s_addr);
456 uint8_t b1 = (uint8_t)(addr_host >> 24);
457 return b1 == 127;
458 }
459 case AF_INET6:
460 return IN6_IS_ADDR_LOOPBACK(reinterpret_cast<const in6_addr*>(&addr.ipv6.sin6_addr));
461 default:
462 return false;
463 }
464}
465
466bool
467IpAddr::isPrivate() const
468{
469 if (isLoopback()) {
470 return true;
471 }
472 switch (addr.addr.sa_family) {
473 case AF_INET: {
474 auto addr_host = ntohl(addr.ipv4.sin_addr.s_addr);
475 uint8_t b1, b2;
476 b1 = (uint8_t)(addr_host >> 24);
477 b2 = (uint8_t)((addr_host >> 16) & 0x0ff);
478 // 10.x.y.z
479 if (b1 == 10)
480 return true;
481 // 172.16.0.0 - 172.31.255.255
482 if ((b1 == 172) && (b2 >= 16) && (b2 <= 31))
483 return true;
484 // 192.168.0.0 - 192.168.255.255
485 if ((b1 == 192) && (b2 == 168))
486 return true;
487 return false;
488 }
489 case AF_INET6: {
490 const pj_uint8_t* addr6 = reinterpret_cast<const pj_uint8_t*>(&addr.ipv6.sin6_addr);
491 if (addr6[0] == 0xfc)
492 return true;
493 return false;
494 }
495 default:
496 return false;
497 }
498}
499} // namespace jami