blob: 8f3540d3f63742bf668044ad5d20f292b3bb0f9f [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/* Socket union related function.
2 * Copyright (c) 1997, 98 Kunihiro Ishiguro
3 *
4 * This file is part of GNU Zebra.
5 *
6 * GNU Zebra is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation; either version 2, or (at your option) any
9 * later version.
10 *
11 * GNU Zebra is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with GNU Zebra; see the file COPYING. If not, write to the Free
18 * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19 * 02111-1307, USA.
20 */
21
22#include <zebra.h>
23
24#include "prefix.h"
25#include "vty.h"
26#include "sockunion.h"
27#include "memory.h"
28#include "str.h"
29#include "log.h"
Timo Teräs9196caf2015-04-29 09:43:05 +030030#include "jhash.h"
paul718e3742002-12-13 20:15:29 +000031
Donald Sharp54b88ca2015-05-13 20:34:46 -040032#if defined(HAVE_LINUX_IN6_H)
33#include <linux/in6.h>
34#endif
35
paul718e3742002-12-13 20:15:29 +000036#ifndef HAVE_INET_ATON
37int
38inet_aton (const char *cp, struct in_addr *inaddr)
39{
40 int dots = 0;
41 register u_long addr = 0;
42 register u_long val = 0, base = 10;
43
44 do
45 {
46 register char c = *cp;
47
48 switch (c)
49 {
50 case '0': case '1': case '2': case '3': case '4': case '5':
51 case '6': case '7': case '8': case '9':
52 val = (val * base) + (c - '0');
53 break;
54 case '.':
55 if (++dots > 3)
56 return 0;
57 case '\0':
58 if (val > 255)
59 return 0;
60 addr = addr << 8 | val;
61 val = 0;
62 break;
63 default:
64 return 0;
65 }
66 } while (*cp++) ;
67
68 if (dots < 3)
69 addr <<= 8 * (3 - dots);
70 if (inaddr)
71 inaddr->s_addr = htonl (addr);
72 return 1;
73}
74#endif /* ! HAVE_INET_ATON */
75
76
77#ifndef HAVE_INET_PTON
78int
79inet_pton (int family, const char *strptr, void *addrptr)
80{
81 if (family == AF_INET)
82 {
83 struct in_addr in_val;
84
85 if (inet_aton (strptr, &in_val))
86 {
87 memcpy (addrptr, &in_val, sizeof (struct in_addr));
88 return 1;
89 }
90 return 0;
91 }
92 errno = EAFNOSUPPORT;
93 return -1;
94}
95#endif /* ! HAVE_INET_PTON */
96
97#ifndef HAVE_INET_NTOP
98const char *
99inet_ntop (int family, const void *addrptr, char *strptr, size_t len)
100{
101 unsigned char *p = (unsigned char *) addrptr;
102
103 if (family == AF_INET)
104 {
105 char temp[INET_ADDRSTRLEN];
106
107 snprintf(temp, sizeof(temp), "%d.%d.%d.%d", p[0], p[1], p[2], p[3]);
108
109 if (strlen(temp) >= len)
110 {
111 errno = ENOSPC;
112 return NULL;
113 }
114 strcpy(strptr, temp);
115 return strptr;
116 }
117
118 errno = EAFNOSUPPORT;
119 return NULL;
120}
121#endif /* ! HAVE_INET_NTOP */
122
123const char *
Timo Teräs81b139b2015-04-29 09:43:01 +0300124inet_sutop (const union sockunion *su, char *str)
paul718e3742002-12-13 20:15:29 +0000125{
126 switch (su->sa.sa_family)
127 {
128 case AF_INET:
129 inet_ntop (AF_INET, &su->sin.sin_addr, str, INET_ADDRSTRLEN);
130 break;
131#ifdef HAVE_IPV6
132 case AF_INET6:
133 inet_ntop (AF_INET6, &su->sin6.sin6_addr, str, INET6_ADDRSTRLEN);
134 break;
135#endif /* HAVE_IPV6 */
136 }
137 return str;
138}
139
140int
hassoa1494112004-10-11 12:53:17 +0000141str2sockunion (const char *str, union sockunion *su)
paul718e3742002-12-13 20:15:29 +0000142{
143 int ret;
144
145 memset (su, 0, sizeof (union sockunion));
146
147 ret = inet_pton (AF_INET, str, &su->sin.sin_addr);
148 if (ret > 0) /* Valid IPv4 address format. */
149 {
150 su->sin.sin_family = AF_INET;
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000151#ifdef HAVE_STRUCT_SOCKADDR_IN_SIN_LEN
paul718e3742002-12-13 20:15:29 +0000152 su->sin.sin_len = sizeof(struct sockaddr_in);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000153#endif /* HAVE_STRUCT_SOCKADDR_IN_SIN_LEN */
paul718e3742002-12-13 20:15:29 +0000154 return 0;
155 }
156#ifdef HAVE_IPV6
157 ret = inet_pton (AF_INET6, str, &su->sin6.sin6_addr);
158 if (ret > 0) /* Valid IPv6 address format. */
159 {
160 su->sin6.sin6_family = AF_INET6;
161#ifdef SIN6_LEN
162 su->sin6.sin6_len = sizeof(struct sockaddr_in6);
163#endif /* SIN6_LEN */
164 return 0;
165 }
166#endif /* HAVE_IPV6 */
167 return -1;
168}
169
170const char *
Timo Teräs81b139b2015-04-29 09:43:01 +0300171sockunion2str (const union sockunion *su, char *buf, size_t len)
paul718e3742002-12-13 20:15:29 +0000172{
173 if (su->sa.sa_family == AF_INET)
174 return inet_ntop (AF_INET, &su->sin.sin_addr, buf, len);
175#ifdef HAVE_IPV6
176 else if (su->sa.sa_family == AF_INET6)
177 return inet_ntop (AF_INET6, &su->sin6.sin6_addr, buf, len);
178#endif /* HAVE_IPV6 */
179 return NULL;
180}
181
Paul Jakma2fb2a452012-06-14 10:37:40 +0100182union sockunion *
183sockunion_str2su (const char *str)
184{
185 union sockunion *su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
186
187 if (!str2sockunion (str, su))
188 return su;
189
190 XFREE (MTYPE_SOCKUNION, su);
191 return NULL;
192}
193
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100194/* Convert IPv4 compatible IPv6 address to IPv4 address. */
195static void
196sockunion_normalise_mapped (union sockunion *su)
197{
198 struct sockaddr_in sin;
199
200#ifdef HAVE_IPV6
201 if (su->sa.sa_family == AF_INET6
202 && IN6_IS_ADDR_V4MAPPED (&su->sin6.sin6_addr))
203 {
204 memset (&sin, 0, sizeof (struct sockaddr_in));
205 sin.sin_family = AF_INET;
Stephen Hemminger3fa3f952009-07-11 21:27:51 -0700206 sin.sin_port = su->sin6.sin6_port;
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100207 memcpy (&sin.sin_addr, ((char *)&su->sin6.sin6_addr) + 12, 4);
208 memcpy (su, &sin, sizeof (struct sockaddr_in));
209 }
210#endif /* HAVE_IPV6 */
211}
212
paul718e3742002-12-13 20:15:29 +0000213/* Return socket of sockunion. */
214int
Timo Teräs81b139b2015-04-29 09:43:01 +0300215sockunion_socket (const union sockunion *su)
paul718e3742002-12-13 20:15:29 +0000216{
217 int sock;
218
219 sock = socket (su->sa.sa_family, SOCK_STREAM, 0);
220 if (sock < 0)
221 {
ajs6099b3b2004-11-20 02:06:59 +0000222 zlog (NULL, LOG_WARNING, "Can't make socket : %s", safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000223 return -1;
224 }
225
226 return sock;
227}
228
229/* Return accepted new socket file descriptor. */
230int
231sockunion_accept (int sock, union sockunion *su)
232{
233 socklen_t len;
234 int client_sock;
235
236 len = sizeof (union sockunion);
237 client_sock = accept (sock, (struct sockaddr *) su, &len);
238
Paul Jakma84152ee2008-11-24 22:25:16 +0000239 sockunion_normalise_mapped (su);
paul718e3742002-12-13 20:15:29 +0000240 return client_sock;
241}
242
243/* Return sizeof union sockunion. */
paul8cc41982005-05-06 21:25:49 +0000244static int
Timo Teräs81b139b2015-04-29 09:43:01 +0300245sockunion_sizeof (const union sockunion *su)
paul718e3742002-12-13 20:15:29 +0000246{
247 int ret;
248
249 ret = 0;
250 switch (su->sa.sa_family)
251 {
252 case AF_INET:
253 ret = sizeof (struct sockaddr_in);
254 break;
255#ifdef HAVE_IPV6
256 case AF_INET6:
257 ret = sizeof (struct sockaddr_in6);
258 break;
259#endif /* AF_INET6 */
260 }
261 return ret;
262}
263
264/* return sockunion structure : this function should be revised. */
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400265static const char *
Timo Teräs81b139b2015-04-29 09:43:01 +0300266sockunion_log (const union sockunion *su, char *buf, size_t len)
paul718e3742002-12-13 20:15:29 +0000267{
paul718e3742002-12-13 20:15:29 +0000268 switch (su->sa.sa_family)
269 {
270 case AF_INET:
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400271 return inet_ntop(AF_INET, &su->sin.sin_addr, buf, len);
272
paul718e3742002-12-13 20:15:29 +0000273#ifdef HAVE_IPV6
274 case AF_INET6:
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400275 return inet_ntop(AF_INET6, &(su->sin6.sin6_addr), buf, len);
paul718e3742002-12-13 20:15:29 +0000276 break;
277#endif /* HAVE_IPV6 */
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400278
paul718e3742002-12-13 20:15:29 +0000279 default:
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400280 snprintf (buf, len, "af_unknown %d ", su->sa.sa_family);
281 return buf;
paul718e3742002-12-13 20:15:29 +0000282 }
paul718e3742002-12-13 20:15:29 +0000283}
284
285/* sockunion_connect returns
286 -1 : error occured
287 0 : connect success
288 1 : connect is in progress */
289enum connect_result
Timo Teräs81b139b2015-04-29 09:43:01 +0300290sockunion_connect (int fd, const union sockunion *peersu, unsigned short port,
paul718e3742002-12-13 20:15:29 +0000291 unsigned int ifindex)
292{
293 int ret;
294 int val;
295 union sockunion su;
296
297 memcpy (&su, peersu, sizeof (union sockunion));
298
299 switch (su.sa.sa_family)
300 {
301 case AF_INET:
302 su.sin.sin_port = port;
303 break;
304#ifdef HAVE_IPV6
305 case AF_INET6:
306 su.sin6.sin6_port = port;
307#ifdef KAME
308 if (IN6_IS_ADDR_LINKLOCAL(&su.sin6.sin6_addr) && ifindex)
309 {
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000310#ifdef HAVE_STRUCT_SOCKADDR_IN6_SIN6_SCOPE_ID
paul718e3742002-12-13 20:15:29 +0000311 /* su.sin6.sin6_scope_id = ifindex; */
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000312#endif /* HAVE_STRUCT_SOCKADDR_IN6_SIN6_SCOPE_ID */
paul718e3742002-12-13 20:15:29 +0000313 SET_IN6_LINKLOCAL_IFINDEX (su.sin6.sin6_addr, ifindex);
paul718e3742002-12-13 20:15:29 +0000314 }
315#endif /* KAME */
316 break;
317#endif /* HAVE_IPV6 */
318 }
319
320 /* Make socket non-block. */
321 val = fcntl (fd, F_GETFL, 0);
322 fcntl (fd, F_SETFL, val|O_NONBLOCK);
323
324 /* Call connect function. */
325 ret = connect (fd, (struct sockaddr *) &su, sockunion_sizeof (&su));
326
327 /* Immediate success */
328 if (ret == 0)
329 {
330 fcntl (fd, F_SETFL, val);
331 return connect_success;
332 }
333
334 /* If connect is in progress then return 1 else it's real error. */
335 if (ret < 0)
336 {
337 if (errno != EINPROGRESS)
338 {
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400339 char str[SU_ADDRSTRLEN];
paul718e3742002-12-13 20:15:29 +0000340 zlog_info ("can't connect to %s fd %d : %s",
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400341 sockunion_log (&su, str, sizeof str),
342 fd, safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000343 return connect_error;
344 }
345 }
346
347 fcntl (fd, F_SETFL, val);
348
349 return connect_in_progress;
350}
351
352/* Make socket from sockunion union. */
353int
354sockunion_stream_socket (union sockunion *su)
355{
356 int sock;
357
358 if (su->sa.sa_family == 0)
359 su->sa.sa_family = AF_INET_UNION;
360
361 sock = socket (su->sa.sa_family, SOCK_STREAM, 0);
362
363 if (sock < 0)
364 zlog (NULL, LOG_WARNING, "can't make socket sockunion_stream_socket");
365
366 return sock;
367}
368
369/* Bind socket to specified address. */
370int
371sockunion_bind (int sock, union sockunion *su, unsigned short port,
372 union sockunion *su_addr)
373{
374 int size = 0;
375 int ret;
376
377 if (su->sa.sa_family == AF_INET)
378 {
379 size = sizeof (struct sockaddr_in);
380 su->sin.sin_port = htons (port);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000381#ifdef HAVE_STRUCT_SOCKADDR_IN_SIN_LEN
paul718e3742002-12-13 20:15:29 +0000382 su->sin.sin_len = size;
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000383#endif /* HAVE_STRUCT_SOCKADDR_IN_SIN_LEN */
paul718e3742002-12-13 20:15:29 +0000384 if (su_addr == NULL)
Jorge Boncompte [DTI2]12829322012-04-10 16:57:25 +0200385 sockunion2ip (su) = htonl (INADDR_ANY);
paul718e3742002-12-13 20:15:29 +0000386 }
387#ifdef HAVE_IPV6
388 else if (su->sa.sa_family == AF_INET6)
389 {
390 size = sizeof (struct sockaddr_in6);
391 su->sin6.sin6_port = htons (port);
392#ifdef SIN6_LEN
393 su->sin6.sin6_len = size;
394#endif /* SIN6_LEN */
395 if (su_addr == NULL)
396 {
David Lamparter6d6df302014-06-28 21:12:37 +0200397#ifdef LINUX_IPV6
paul718e3742002-12-13 20:15:29 +0000398 memset (&su->sin6.sin6_addr, 0, sizeof (struct in6_addr));
399#else
400 su->sin6.sin6_addr = in6addr_any;
401#endif /* LINUX_IPV6 */
402 }
403 }
404#endif /* HAVE_IPV6 */
405
406
407 ret = bind (sock, (struct sockaddr *)su, size);
408 if (ret < 0)
ajs6099b3b2004-11-20 02:06:59 +0000409 zlog (NULL, LOG_WARNING, "can't bind socket : %s", safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000410
411 return ret;
412}
413
414int
415sockopt_reuseaddr (int sock)
416{
417 int ret;
418 int on = 1;
419
420 ret = setsockopt (sock, SOL_SOCKET, SO_REUSEADDR,
421 (void *) &on, sizeof (on));
422 if (ret < 0)
423 {
424 zlog (NULL, LOG_WARNING, "can't set sockopt SO_REUSEADDR to socket %d", sock);
425 return -1;
426 }
427 return 0;
428}
429
430#ifdef SO_REUSEPORT
431int
432sockopt_reuseport (int sock)
433{
434 int ret;
435 int on = 1;
436
437 ret = setsockopt (sock, SOL_SOCKET, SO_REUSEPORT,
438 (void *) &on, sizeof (on));
439 if (ret < 0)
440 {
hassoe7fe8c82005-05-06 19:33:35 +0000441 zlog (NULL, LOG_WARNING, "can't set sockopt SO_REUSEPORT to socket %d", sock);
paul718e3742002-12-13 20:15:29 +0000442 return -1;
443 }
444 return 0;
445}
446#else
447int
448sockopt_reuseport (int sock)
449{
450 return 0;
451}
452#endif /* 0 */
453
454int
455sockopt_ttl (int family, int sock, int ttl)
456{
457 int ret;
458
459#ifdef IP_TTL
460 if (family == AF_INET)
461 {
462 ret = setsockopt (sock, IPPROTO_IP, IP_TTL,
463 (void *) &ttl, sizeof (int));
464 if (ret < 0)
465 {
466 zlog (NULL, LOG_WARNING, "can't set sockopt IP_TTL %d to socket %d", ttl, sock);
467 return -1;
468 }
469 return 0;
470 }
471#endif /* IP_TTL */
472#ifdef HAVE_IPV6
473 if (family == AF_INET6)
474 {
475 ret = setsockopt (sock, IPPROTO_IPV6, IPV6_UNICAST_HOPS,
476 (void *) &ttl, sizeof (int));
477 if (ret < 0)
478 {
479 zlog (NULL, LOG_WARNING, "can't set sockopt IPV6_UNICAST_HOPS %d to socket %d",
480 ttl, sock);
481 return -1;
482 }
483 return 0;
484 }
485#endif /* HAVE_IPV6 */
486 return 0;
487}
488
Stephen Hemminger58192df2010-08-05 10:26:24 -0700489int
490sockopt_cork (int sock, int onoff)
491{
492#ifdef TCP_CORK
493 return setsockopt (sock, IPPROTO_TCP, TCP_CORK, &onoff, sizeof(onoff));
494#else
495 return 0;
496#endif
497}
498
Nick Hilliardfa411a22011-03-23 15:33:17 +0000499int
500sockopt_minttl (int family, int sock, int minttl)
501{
Stephen Hemminger89b6d1f2011-03-24 10:51:59 +0000502#ifdef IP_MINTTL
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700503 if (family == AF_INET)
Nick Hilliardfa411a22011-03-23 15:33:17 +0000504 {
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700505 int ret = setsockopt (sock, IPPROTO_IP, IP_MINTTL, &minttl, sizeof(minttl));
506 if (ret < 0)
507 zlog (NULL, LOG_WARNING,
508 "can't set sockopt IP_MINTTL to %d on socket %d: %s",
509 minttl, sock, safe_strerror (errno));
510 return ret;
Nick Hilliardfa411a22011-03-23 15:33:17 +0000511 }
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700512#endif /* IP_MINTTL */
Donald Sharp54b88ca2015-05-13 20:34:46 -0400513#ifdef IPV6_MINHOPCOUNT
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700514 if (family == AF_INET6)
515 {
Donald Sharp54b88ca2015-05-13 20:34:46 -0400516 int ret = setsockopt (sock, IPPROTO_IPV6, IPV6_MINHOPCOUNT, &minttl, sizeof(minttl));
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700517 if (ret < 0)
518 zlog (NULL, LOG_WARNING,
Donald Sharp54b88ca2015-05-13 20:34:46 -0400519 "can't set sockopt IPV6_MINHOPCOUNT to %d on socket %d: %s",
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700520 minttl, sock, safe_strerror (errno));
521 return ret;
522 }
523#endif
Nick Hilliardfa411a22011-03-23 15:33:17 +0000524
Stephen Hemminger89b6d1f2011-03-24 10:51:59 +0000525 errno = EOPNOTSUPP;
526 return -1;
Nick Hilliardfa411a22011-03-23 15:33:17 +0000527}
528
David Lamparterca051262009-10-04 16:21:49 +0200529int
530sockopt_v6only (int family, int sock)
531{
532 int ret, on = 1;
533
534#ifdef HAVE_IPV6
535#ifdef IPV6_V6ONLY
536 if (family == AF_INET6)
537 {
538 ret = setsockopt (sock, IPPROTO_IPV6, IPV6_V6ONLY,
539 (void *) &on, sizeof (int));
540 if (ret < 0)
541 {
542 zlog (NULL, LOG_WARNING, "can't set sockopt IPV6_V6ONLY "
543 "to socket %d", sock);
544 return -1;
545 }
546 return 0;
547 }
548#endif /* IPV6_V6ONLY */
549#endif /* HAVE_IPV6 */
550 return 0;
551}
552
paul718e3742002-12-13 20:15:29 +0000553/* If same family and same prefix return 1. */
554int
Timo Teräs81b139b2015-04-29 09:43:01 +0300555sockunion_same (const union sockunion *su1, const union sockunion *su2)
paul718e3742002-12-13 20:15:29 +0000556{
557 int ret = 0;
558
559 if (su1->sa.sa_family != su2->sa.sa_family)
560 return 0;
561
562 switch (su1->sa.sa_family)
563 {
564 case AF_INET:
565 ret = memcmp (&su1->sin.sin_addr, &su2->sin.sin_addr,
566 sizeof (struct in_addr));
567 break;
568#ifdef HAVE_IPV6
569 case AF_INET6:
570 ret = memcmp (&su1->sin6.sin6_addr, &su2->sin6.sin6_addr,
571 sizeof (struct in6_addr));
572 break;
573#endif /* HAVE_IPV6 */
574 }
575 if (ret == 0)
576 return 1;
577 else
578 return 0;
579}
580
Timo Teräs9196caf2015-04-29 09:43:05 +0300581unsigned int
582sockunion_hash (const union sockunion *su)
583{
584 switch (sockunion_family(su))
585 {
586 case AF_INET:
587 return jhash_1word(su->sin.sin_addr.s_addr, 0);
588#ifdef HAVE_IPV6
589 case AF_INET6:
590 return jhash2(su->sin6.sin6_addr.s6_addr32, ZEBRA_NUM_OF(su->sin6.sin6_addr.s6_addr32), 0);
591#endif /* HAVE_IPV6 */
592 }
593 return 0;
594}
595
paul718e3742002-12-13 20:15:29 +0000596/* After TCP connection is established. Get local address and port. */
597union sockunion *
598sockunion_getsockname (int fd)
599{
600 int ret;
paul22528292004-05-08 05:10:38 +0000601 socklen_t len;
paul718e3742002-12-13 20:15:29 +0000602 union
603 {
604 struct sockaddr sa;
605 struct sockaddr_in sin;
606#ifdef HAVE_IPV6
607 struct sockaddr_in6 sin6;
608#endif /* HAVE_IPV6 */
609 char tmp_buffer[128];
610 } name;
611 union sockunion *su;
612
613 memset (&name, 0, sizeof name);
614 len = sizeof name;
615
616 ret = getsockname (fd, (struct sockaddr *)&name, &len);
617 if (ret < 0)
618 {
619 zlog_warn ("Can't get local address and port by getsockname: %s",
ajs6099b3b2004-11-20 02:06:59 +0000620 safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000621 return NULL;
622 }
623
624 if (name.sa.sa_family == AF_INET)
625 {
paul2ba9a372005-05-19 01:37:50 +0000626 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000627 memcpy (su, &name, sizeof (struct sockaddr_in));
628 return su;
629 }
630#ifdef HAVE_IPV6
631 if (name.sa.sa_family == AF_INET6)
632 {
paul2ba9a372005-05-19 01:37:50 +0000633 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000634 memcpy (su, &name, sizeof (struct sockaddr_in6));
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100635 sockunion_normalise_mapped (su);
paul718e3742002-12-13 20:15:29 +0000636 return su;
637 }
638#endif /* HAVE_IPV6 */
639 return NULL;
640}
641
642/* After TCP connection is established. Get remote address and port. */
643union sockunion *
644sockunion_getpeername (int fd)
645{
646 int ret;
paul22528292004-05-08 05:10:38 +0000647 socklen_t len;
paul718e3742002-12-13 20:15:29 +0000648 union
649 {
650 struct sockaddr sa;
651 struct sockaddr_in sin;
652#ifdef HAVE_IPV6
653 struct sockaddr_in6 sin6;
654#endif /* HAVE_IPV6 */
655 char tmp_buffer[128];
656 } name;
657 union sockunion *su;
658
659 memset (&name, 0, sizeof name);
660 len = sizeof name;
661 ret = getpeername (fd, (struct sockaddr *)&name, &len);
662 if (ret < 0)
663 {
664 zlog (NULL, LOG_WARNING, "Can't get remote address and port: %s",
ajs6099b3b2004-11-20 02:06:59 +0000665 safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000666 return NULL;
667 }
668
669 if (name.sa.sa_family == AF_INET)
670 {
paul2ba9a372005-05-19 01:37:50 +0000671 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000672 memcpy (su, &name, sizeof (struct sockaddr_in));
673 return su;
674 }
675#ifdef HAVE_IPV6
676 if (name.sa.sa_family == AF_INET6)
677 {
paul2ba9a372005-05-19 01:37:50 +0000678 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000679 memcpy (su, &name, sizeof (struct sockaddr_in6));
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100680 sockunion_normalise_mapped (su);
paul718e3742002-12-13 20:15:29 +0000681 return su;
682 }
683#endif /* HAVE_IPV6 */
684 return NULL;
685}
686
687/* Print sockunion structure */
paul8cc41982005-05-06 21:25:49 +0000688static void __attribute__ ((unused))
Timo Teräs81b139b2015-04-29 09:43:01 +0300689sockunion_print (const union sockunion *su)
paul718e3742002-12-13 20:15:29 +0000690{
691 if (su == NULL)
692 return;
693
694 switch (su->sa.sa_family)
695 {
696 case AF_INET:
697 printf ("%s\n", inet_ntoa (su->sin.sin_addr));
698 break;
699#ifdef HAVE_IPV6
700 case AF_INET6:
701 {
paul42d49862004-10-13 05:22:18 +0000702 char buf [SU_ADDRSTRLEN];
paul718e3742002-12-13 20:15:29 +0000703
704 printf ("%s\n", inet_ntop (AF_INET6, &(su->sin6.sin6_addr),
705 buf, sizeof (buf)));
706 }
707 break;
708#endif /* HAVE_IPV6 */
709
710#ifdef AF_LINK
711 case AF_LINK:
712 {
713 struct sockaddr_dl *sdl;
714
715 sdl = (struct sockaddr_dl *)&(su->sa);
716 printf ("link#%d\n", sdl->sdl_index);
717 }
718 break;
719#endif /* AF_LINK */
720 default:
721 printf ("af_unknown %d\n", su->sa.sa_family);
722 break;
723 }
724}
725
726#ifdef HAVE_IPV6
paul8cc41982005-05-06 21:25:49 +0000727static int
Timo Teräs81b139b2015-04-29 09:43:01 +0300728in6addr_cmp (const struct in6_addr *addr1, const struct in6_addr *addr2)
paul718e3742002-12-13 20:15:29 +0000729{
hasso8c328f12004-10-05 21:01:23 +0000730 unsigned int i;
paul718e3742002-12-13 20:15:29 +0000731 u_char *p1, *p2;
732
733 p1 = (u_char *)addr1;
734 p2 = (u_char *)addr2;
735
736 for (i = 0; i < sizeof (struct in6_addr); i++)
737 {
738 if (p1[i] > p2[i])
739 return 1;
740 else if (p1[i] < p2[i])
741 return -1;
742 }
743 return 0;
744}
745#endif /* HAVE_IPV6 */
746
747int
Timo Teräs81b139b2015-04-29 09:43:01 +0300748sockunion_cmp (const union sockunion *su1, const union sockunion *su2)
paul718e3742002-12-13 20:15:29 +0000749{
750 if (su1->sa.sa_family > su2->sa.sa_family)
751 return 1;
752 if (su1->sa.sa_family < su2->sa.sa_family)
753 return -1;
754
755 if (su1->sa.sa_family == AF_INET)
756 {
Jorge Boncompte [DTI2]12829322012-04-10 16:57:25 +0200757 if (ntohl (sockunion2ip (su1)) == ntohl (sockunion2ip (su2)))
paul718e3742002-12-13 20:15:29 +0000758 return 0;
Jorge Boncompte [DTI2]12829322012-04-10 16:57:25 +0200759 if (ntohl (sockunion2ip (su1)) > ntohl (sockunion2ip (su2)))
paul718e3742002-12-13 20:15:29 +0000760 return 1;
761 else
762 return -1;
763 }
764#ifdef HAVE_IPV6
765 if (su1->sa.sa_family == AF_INET6)
766 return in6addr_cmp (&su1->sin6.sin6_addr, &su2->sin6.sin6_addr);
767#endif /* HAVE_IPV6 */
768 return 0;
769}
770
771/* Duplicate sockunion. */
772union sockunion *
Timo Teräs81b139b2015-04-29 09:43:01 +0300773sockunion_dup (const union sockunion *su)
paul718e3742002-12-13 20:15:29 +0000774{
775 union sockunion *dup = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
776 memcpy (dup, su, sizeof (union sockunion));
777 return dup;
778}
779
780void
781sockunion_free (union sockunion *su)
782{
783 XFREE (MTYPE_SOCKUNION, su);
784}