blob: cfb6f9c5a6921042620c34ff7c11d1e3edfaca52 [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"
30
31#ifndef HAVE_INET_ATON
32int
33inet_aton (const char *cp, struct in_addr *inaddr)
34{
35 int dots = 0;
36 register u_long addr = 0;
37 register u_long val = 0, base = 10;
38
39 do
40 {
41 register char c = *cp;
42
43 switch (c)
44 {
45 case '0': case '1': case '2': case '3': case '4': case '5':
46 case '6': case '7': case '8': case '9':
47 val = (val * base) + (c - '0');
48 break;
49 case '.':
50 if (++dots > 3)
51 return 0;
52 case '\0':
53 if (val > 255)
54 return 0;
55 addr = addr << 8 | val;
56 val = 0;
57 break;
58 default:
59 return 0;
60 }
61 } while (*cp++) ;
62
63 if (dots < 3)
64 addr <<= 8 * (3 - dots);
65 if (inaddr)
66 inaddr->s_addr = htonl (addr);
67 return 1;
68}
69#endif /* ! HAVE_INET_ATON */
70
71
72#ifndef HAVE_INET_PTON
73int
74inet_pton (int family, const char *strptr, void *addrptr)
75{
76 if (family == AF_INET)
77 {
78 struct in_addr in_val;
79
80 if (inet_aton (strptr, &in_val))
81 {
82 memcpy (addrptr, &in_val, sizeof (struct in_addr));
83 return 1;
84 }
85 return 0;
86 }
87 errno = EAFNOSUPPORT;
88 return -1;
89}
90#endif /* ! HAVE_INET_PTON */
91
92#ifndef HAVE_INET_NTOP
93const char *
94inet_ntop (int family, const void *addrptr, char *strptr, size_t len)
95{
96 unsigned char *p = (unsigned char *) addrptr;
97
98 if (family == AF_INET)
99 {
100 char temp[INET_ADDRSTRLEN];
101
102 snprintf(temp, sizeof(temp), "%d.%d.%d.%d", p[0], p[1], p[2], p[3]);
103
104 if (strlen(temp) >= len)
105 {
106 errno = ENOSPC;
107 return NULL;
108 }
109 strcpy(strptr, temp);
110 return strptr;
111 }
112
113 errno = EAFNOSUPPORT;
114 return NULL;
115}
116#endif /* ! HAVE_INET_NTOP */
117
118const char *
119inet_sutop (union sockunion *su, char *str)
120{
121 switch (su->sa.sa_family)
122 {
123 case AF_INET:
124 inet_ntop (AF_INET, &su->sin.sin_addr, str, INET_ADDRSTRLEN);
125 break;
126#ifdef HAVE_IPV6
127 case AF_INET6:
128 inet_ntop (AF_INET6, &su->sin6.sin6_addr, str, INET6_ADDRSTRLEN);
129 break;
130#endif /* HAVE_IPV6 */
131 }
132 return str;
133}
134
135int
hassoa1494112004-10-11 12:53:17 +0000136str2sockunion (const char *str, union sockunion *su)
paul718e3742002-12-13 20:15:29 +0000137{
138 int ret;
139
140 memset (su, 0, sizeof (union sockunion));
141
142 ret = inet_pton (AF_INET, str, &su->sin.sin_addr);
143 if (ret > 0) /* Valid IPv4 address format. */
144 {
145 su->sin.sin_family = AF_INET;
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000146#ifdef HAVE_STRUCT_SOCKADDR_IN_SIN_LEN
paul718e3742002-12-13 20:15:29 +0000147 su->sin.sin_len = sizeof(struct sockaddr_in);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000148#endif /* HAVE_STRUCT_SOCKADDR_IN_SIN_LEN */
paul718e3742002-12-13 20:15:29 +0000149 return 0;
150 }
151#ifdef HAVE_IPV6
152 ret = inet_pton (AF_INET6, str, &su->sin6.sin6_addr);
153 if (ret > 0) /* Valid IPv6 address format. */
154 {
155 su->sin6.sin6_family = AF_INET6;
156#ifdef SIN6_LEN
157 su->sin6.sin6_len = sizeof(struct sockaddr_in6);
158#endif /* SIN6_LEN */
159 return 0;
160 }
161#endif /* HAVE_IPV6 */
162 return -1;
163}
164
165const char *
166sockunion2str (union sockunion *su, char *buf, size_t len)
167{
168 if (su->sa.sa_family == AF_INET)
169 return inet_ntop (AF_INET, &su->sin.sin_addr, buf, len);
170#ifdef HAVE_IPV6
171 else if (su->sa.sa_family == AF_INET6)
172 return inet_ntop (AF_INET6, &su->sin6.sin6_addr, buf, len);
173#endif /* HAVE_IPV6 */
174 return NULL;
175}
176
177union sockunion *
paul42d49862004-10-13 05:22:18 +0000178sockunion_str2su (const char *str)
paul718e3742002-12-13 20:15:29 +0000179{
180 int ret;
181 union sockunion *su;
182
Stephen Hemminger393deb92008-08-18 14:13:29 -0700183 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000184
185 ret = inet_pton (AF_INET, str, &su->sin.sin_addr);
186 if (ret > 0) /* Valid IPv4 address format. */
187 {
188 su->sin.sin_family = AF_INET;
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000189#ifdef HAVE_STRUCT_SOCKADDR_IN_SIN_LEN
paul718e3742002-12-13 20:15:29 +0000190 su->sin.sin_len = sizeof(struct sockaddr_in);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000191#endif /* HAVE_STRUCT_SOCKADDR_IN_SIN_LEN */
paul718e3742002-12-13 20:15:29 +0000192 return su;
193 }
194#ifdef HAVE_IPV6
195 ret = inet_pton (AF_INET6, str, &su->sin6.sin6_addr);
196 if (ret > 0) /* Valid IPv6 address format. */
197 {
198 su->sin6.sin6_family = AF_INET6;
199#ifdef SIN6_LEN
200 su->sin6.sin6_len = sizeof(struct sockaddr_in6);
201#endif /* SIN6_LEN */
202 return su;
203 }
204#endif /* HAVE_IPV6 */
205
206 XFREE (MTYPE_SOCKUNION, su);
207 return NULL;
208}
209
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100210/* Convert IPv4 compatible IPv6 address to IPv4 address. */
211static void
212sockunion_normalise_mapped (union sockunion *su)
213{
214 struct sockaddr_in sin;
215
216#ifdef HAVE_IPV6
217 if (su->sa.sa_family == AF_INET6
218 && IN6_IS_ADDR_V4MAPPED (&su->sin6.sin6_addr))
219 {
220 memset (&sin, 0, sizeof (struct sockaddr_in));
221 sin.sin_family = AF_INET;
Stephen Hemminger3fa3f952009-07-11 21:27:51 -0700222 sin.sin_port = su->sin6.sin6_port;
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100223 memcpy (&sin.sin_addr, ((char *)&su->sin6.sin6_addr) + 12, 4);
224 memcpy (su, &sin, sizeof (struct sockaddr_in));
225 }
226#endif /* HAVE_IPV6 */
227}
228
paul718e3742002-12-13 20:15:29 +0000229/* Return socket of sockunion. */
230int
231sockunion_socket (union sockunion *su)
232{
233 int sock;
234
235 sock = socket (su->sa.sa_family, SOCK_STREAM, 0);
236 if (sock < 0)
237 {
ajs6099b3b2004-11-20 02:06:59 +0000238 zlog (NULL, LOG_WARNING, "Can't make socket : %s", safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000239 return -1;
240 }
241
242 return sock;
243}
244
245/* Return accepted new socket file descriptor. */
246int
247sockunion_accept (int sock, union sockunion *su)
248{
249 socklen_t len;
250 int client_sock;
251
252 len = sizeof (union sockunion);
253 client_sock = accept (sock, (struct sockaddr *) su, &len);
254
Paul Jakma84152ee2008-11-24 22:25:16 +0000255 sockunion_normalise_mapped (su);
paul718e3742002-12-13 20:15:29 +0000256 return client_sock;
257}
258
259/* Return sizeof union sockunion. */
paul8cc41982005-05-06 21:25:49 +0000260static int
paul718e3742002-12-13 20:15:29 +0000261sockunion_sizeof (union sockunion *su)
262{
263 int ret;
264
265 ret = 0;
266 switch (su->sa.sa_family)
267 {
268 case AF_INET:
269 ret = sizeof (struct sockaddr_in);
270 break;
271#ifdef HAVE_IPV6
272 case AF_INET6:
273 ret = sizeof (struct sockaddr_in6);
274 break;
275#endif /* AF_INET6 */
276 }
277 return ret;
278}
279
280/* return sockunion structure : this function should be revised. */
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400281static const char *
282sockunion_log (union sockunion *su, char *buf, size_t len)
paul718e3742002-12-13 20:15:29 +0000283{
paul718e3742002-12-13 20:15:29 +0000284 switch (su->sa.sa_family)
285 {
286 case AF_INET:
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400287 return inet_ntop(AF_INET, &su->sin.sin_addr, buf, len);
288
paul718e3742002-12-13 20:15:29 +0000289#ifdef HAVE_IPV6
290 case AF_INET6:
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400291 return inet_ntop(AF_INET6, &(su->sin6.sin6_addr), buf, len);
paul718e3742002-12-13 20:15:29 +0000292 break;
293#endif /* HAVE_IPV6 */
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400294
paul718e3742002-12-13 20:15:29 +0000295 default:
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400296 snprintf (buf, len, "af_unknown %d ", su->sa.sa_family);
297 return buf;
paul718e3742002-12-13 20:15:29 +0000298 }
paul718e3742002-12-13 20:15:29 +0000299}
300
301/* sockunion_connect returns
302 -1 : error occured
303 0 : connect success
304 1 : connect is in progress */
305enum connect_result
306sockunion_connect (int fd, union sockunion *peersu, unsigned short port,
307 unsigned int ifindex)
308{
309 int ret;
310 int val;
311 union sockunion su;
312
313 memcpy (&su, peersu, sizeof (union sockunion));
314
315 switch (su.sa.sa_family)
316 {
317 case AF_INET:
318 su.sin.sin_port = port;
319 break;
320#ifdef HAVE_IPV6
321 case AF_INET6:
322 su.sin6.sin6_port = port;
323#ifdef KAME
324 if (IN6_IS_ADDR_LINKLOCAL(&su.sin6.sin6_addr) && ifindex)
325 {
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000326#ifdef HAVE_STRUCT_SOCKADDR_IN6_SIN6_SCOPE_ID
paul718e3742002-12-13 20:15:29 +0000327 /* su.sin6.sin6_scope_id = ifindex; */
hasso726f9b22003-05-25 21:04:54 +0000328#ifdef MUSICA
329 su.sin6.sin6_scope_id = ifindex;
330#endif
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000331#endif /* HAVE_STRUCT_SOCKADDR_IN6_SIN6_SCOPE_ID */
hasso726f9b22003-05-25 21:04:54 +0000332#ifndef MUSICA
paul718e3742002-12-13 20:15:29 +0000333 SET_IN6_LINKLOCAL_IFINDEX (su.sin6.sin6_addr, ifindex);
hasso726f9b22003-05-25 21:04:54 +0000334#endif
paul718e3742002-12-13 20:15:29 +0000335 }
336#endif /* KAME */
337 break;
338#endif /* HAVE_IPV6 */
339 }
340
341 /* Make socket non-block. */
342 val = fcntl (fd, F_GETFL, 0);
343 fcntl (fd, F_SETFL, val|O_NONBLOCK);
344
345 /* Call connect function. */
346 ret = connect (fd, (struct sockaddr *) &su, sockunion_sizeof (&su));
347
348 /* Immediate success */
349 if (ret == 0)
350 {
351 fcntl (fd, F_SETFL, val);
352 return connect_success;
353 }
354
355 /* If connect is in progress then return 1 else it's real error. */
356 if (ret < 0)
357 {
358 if (errno != EINPROGRESS)
359 {
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400360 char str[SU_ADDRSTRLEN];
paul718e3742002-12-13 20:15:29 +0000361 zlog_info ("can't connect to %s fd %d : %s",
Stephen Hemmingerb24b19f2011-12-06 14:09:18 +0400362 sockunion_log (&su, str, sizeof str),
363 fd, safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000364 return connect_error;
365 }
366 }
367
368 fcntl (fd, F_SETFL, val);
369
370 return connect_in_progress;
371}
372
373/* Make socket from sockunion union. */
374int
375sockunion_stream_socket (union sockunion *su)
376{
377 int sock;
378
379 if (su->sa.sa_family == 0)
380 su->sa.sa_family = AF_INET_UNION;
381
382 sock = socket (su->sa.sa_family, SOCK_STREAM, 0);
383
384 if (sock < 0)
385 zlog (NULL, LOG_WARNING, "can't make socket sockunion_stream_socket");
386
387 return sock;
388}
389
390/* Bind socket to specified address. */
391int
392sockunion_bind (int sock, union sockunion *su, unsigned short port,
393 union sockunion *su_addr)
394{
395 int size = 0;
396 int ret;
397
398 if (su->sa.sa_family == AF_INET)
399 {
400 size = sizeof (struct sockaddr_in);
401 su->sin.sin_port = htons (port);
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000402#ifdef HAVE_STRUCT_SOCKADDR_IN_SIN_LEN
paul718e3742002-12-13 20:15:29 +0000403 su->sin.sin_len = size;
Paul Jakma6f0e3f62007-05-10 02:38:51 +0000404#endif /* HAVE_STRUCT_SOCKADDR_IN_SIN_LEN */
paul718e3742002-12-13 20:15:29 +0000405 if (su_addr == NULL)
406 su->sin.sin_addr.s_addr = htonl (INADDR_ANY);
407 }
408#ifdef HAVE_IPV6
409 else if (su->sa.sa_family == AF_INET6)
410 {
411 size = sizeof (struct sockaddr_in6);
412 su->sin6.sin6_port = htons (port);
413#ifdef SIN6_LEN
414 su->sin6.sin6_len = size;
415#endif /* SIN6_LEN */
416 if (su_addr == NULL)
417 {
418#if defined(LINUX_IPV6) || defined(NRL)
419 memset (&su->sin6.sin6_addr, 0, sizeof (struct in6_addr));
420#else
421 su->sin6.sin6_addr = in6addr_any;
422#endif /* LINUX_IPV6 */
423 }
424 }
425#endif /* HAVE_IPV6 */
426
427
428 ret = bind (sock, (struct sockaddr *)su, size);
429 if (ret < 0)
ajs6099b3b2004-11-20 02:06:59 +0000430 zlog (NULL, LOG_WARNING, "can't bind socket : %s", safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000431
432 return ret;
433}
434
435int
436sockopt_reuseaddr (int sock)
437{
438 int ret;
439 int on = 1;
440
441 ret = setsockopt (sock, SOL_SOCKET, SO_REUSEADDR,
442 (void *) &on, sizeof (on));
443 if (ret < 0)
444 {
445 zlog (NULL, LOG_WARNING, "can't set sockopt SO_REUSEADDR to socket %d", sock);
446 return -1;
447 }
448 return 0;
449}
450
451#ifdef SO_REUSEPORT
452int
453sockopt_reuseport (int sock)
454{
455 int ret;
456 int on = 1;
457
458 ret = setsockopt (sock, SOL_SOCKET, SO_REUSEPORT,
459 (void *) &on, sizeof (on));
460 if (ret < 0)
461 {
hassoe7fe8c82005-05-06 19:33:35 +0000462 zlog (NULL, LOG_WARNING, "can't set sockopt SO_REUSEPORT to socket %d", sock);
paul718e3742002-12-13 20:15:29 +0000463 return -1;
464 }
465 return 0;
466}
467#else
468int
469sockopt_reuseport (int sock)
470{
471 return 0;
472}
473#endif /* 0 */
474
475int
476sockopt_ttl (int family, int sock, int ttl)
477{
478 int ret;
479
480#ifdef IP_TTL
481 if (family == AF_INET)
482 {
483 ret = setsockopt (sock, IPPROTO_IP, IP_TTL,
484 (void *) &ttl, sizeof (int));
485 if (ret < 0)
486 {
487 zlog (NULL, LOG_WARNING, "can't set sockopt IP_TTL %d to socket %d", ttl, sock);
488 return -1;
489 }
490 return 0;
491 }
492#endif /* IP_TTL */
493#ifdef HAVE_IPV6
494 if (family == AF_INET6)
495 {
496 ret = setsockopt (sock, IPPROTO_IPV6, IPV6_UNICAST_HOPS,
497 (void *) &ttl, sizeof (int));
498 if (ret < 0)
499 {
500 zlog (NULL, LOG_WARNING, "can't set sockopt IPV6_UNICAST_HOPS %d to socket %d",
501 ttl, sock);
502 return -1;
503 }
504 return 0;
505 }
506#endif /* HAVE_IPV6 */
507 return 0;
508}
509
Stephen Hemminger58192df2010-08-05 10:26:24 -0700510int
511sockopt_cork (int sock, int onoff)
512{
513#ifdef TCP_CORK
514 return setsockopt (sock, IPPROTO_TCP, TCP_CORK, &onoff, sizeof(onoff));
515#else
516 return 0;
517#endif
518}
519
Nick Hilliardfa411a22011-03-23 15:33:17 +0000520int
521sockopt_minttl (int family, int sock, int minttl)
522{
Stephen Hemminger89b6d1f2011-03-24 10:51:59 +0000523#ifdef IP_MINTTL
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700524 if (family == AF_INET)
Nick Hilliardfa411a22011-03-23 15:33:17 +0000525 {
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700526 int ret = setsockopt (sock, IPPROTO_IP, IP_MINTTL, &minttl, sizeof(minttl));
527 if (ret < 0)
528 zlog (NULL, LOG_WARNING,
529 "can't set sockopt IP_MINTTL to %d on socket %d: %s",
530 minttl, sock, safe_strerror (errno));
531 return ret;
Nick Hilliardfa411a22011-03-23 15:33:17 +0000532 }
Stephen Hemmingerd876bdf2010-08-05 10:26:27 -0700533#endif /* IP_MINTTL */
534#ifdef IPV6_MINHOPCNT
535 if (family == AF_INET6)
536 {
537 int ret = setsockopt (sock, IPPROTO_IPV6, IPV6_MINHOPCNT, &minttl, sizeof(minttl));
538 if (ret < 0)
539 zlog (NULL, LOG_WARNING,
540 "can't set sockopt IPV6_MINHOPCNT to %d on socket %d: %s",
541 minttl, sock, safe_strerror (errno));
542 return ret;
543 }
544#endif
Nick Hilliardfa411a22011-03-23 15:33:17 +0000545
Stephen Hemminger89b6d1f2011-03-24 10:51:59 +0000546 errno = EOPNOTSUPP;
547 return -1;
Nick Hilliardfa411a22011-03-23 15:33:17 +0000548}
549
David Lamparterca051262009-10-04 16:21:49 +0200550int
551sockopt_v6only (int family, int sock)
552{
553 int ret, on = 1;
554
555#ifdef HAVE_IPV6
556#ifdef IPV6_V6ONLY
557 if (family == AF_INET6)
558 {
559 ret = setsockopt (sock, IPPROTO_IPV6, IPV6_V6ONLY,
560 (void *) &on, sizeof (int));
561 if (ret < 0)
562 {
563 zlog (NULL, LOG_WARNING, "can't set sockopt IPV6_V6ONLY "
564 "to socket %d", sock);
565 return -1;
566 }
567 return 0;
568 }
569#endif /* IPV6_V6ONLY */
570#endif /* HAVE_IPV6 */
571 return 0;
572}
573
paul718e3742002-12-13 20:15:29 +0000574/* If same family and same prefix return 1. */
575int
576sockunion_same (union sockunion *su1, union sockunion *su2)
577{
578 int ret = 0;
579
580 if (su1->sa.sa_family != su2->sa.sa_family)
581 return 0;
582
583 switch (su1->sa.sa_family)
584 {
585 case AF_INET:
586 ret = memcmp (&su1->sin.sin_addr, &su2->sin.sin_addr,
587 sizeof (struct in_addr));
588 break;
589#ifdef HAVE_IPV6
590 case AF_INET6:
591 ret = memcmp (&su1->sin6.sin6_addr, &su2->sin6.sin6_addr,
592 sizeof (struct in6_addr));
593 break;
594#endif /* HAVE_IPV6 */
595 }
596 if (ret == 0)
597 return 1;
598 else
599 return 0;
600}
601
602/* After TCP connection is established. Get local address and port. */
603union sockunion *
604sockunion_getsockname (int fd)
605{
606 int ret;
paul22528292004-05-08 05:10:38 +0000607 socklen_t len;
paul718e3742002-12-13 20:15:29 +0000608 union
609 {
610 struct sockaddr sa;
611 struct sockaddr_in sin;
612#ifdef HAVE_IPV6
613 struct sockaddr_in6 sin6;
614#endif /* HAVE_IPV6 */
615 char tmp_buffer[128];
616 } name;
617 union sockunion *su;
618
619 memset (&name, 0, sizeof name);
620 len = sizeof name;
621
622 ret = getsockname (fd, (struct sockaddr *)&name, &len);
623 if (ret < 0)
624 {
625 zlog_warn ("Can't get local address and port by getsockname: %s",
ajs6099b3b2004-11-20 02:06:59 +0000626 safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000627 return NULL;
628 }
629
630 if (name.sa.sa_family == AF_INET)
631 {
paul2ba9a372005-05-19 01:37:50 +0000632 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000633 memcpy (su, &name, sizeof (struct sockaddr_in));
634 return su;
635 }
636#ifdef HAVE_IPV6
637 if (name.sa.sa_family == AF_INET6)
638 {
paul2ba9a372005-05-19 01:37:50 +0000639 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000640 memcpy (su, &name, sizeof (struct sockaddr_in6));
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100641 sockunion_normalise_mapped (su);
paul718e3742002-12-13 20:15:29 +0000642 return su;
643 }
644#endif /* HAVE_IPV6 */
645 return NULL;
646}
647
648/* After TCP connection is established. Get remote address and port. */
649union sockunion *
650sockunion_getpeername (int fd)
651{
652 int ret;
paul22528292004-05-08 05:10:38 +0000653 socklen_t len;
paul718e3742002-12-13 20:15:29 +0000654 union
655 {
656 struct sockaddr sa;
657 struct sockaddr_in sin;
658#ifdef HAVE_IPV6
659 struct sockaddr_in6 sin6;
660#endif /* HAVE_IPV6 */
661 char tmp_buffer[128];
662 } name;
663 union sockunion *su;
664
665 memset (&name, 0, sizeof name);
666 len = sizeof name;
667 ret = getpeername (fd, (struct sockaddr *)&name, &len);
668 if (ret < 0)
669 {
670 zlog (NULL, LOG_WARNING, "Can't get remote address and port: %s",
ajs6099b3b2004-11-20 02:06:59 +0000671 safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000672 return NULL;
673 }
674
675 if (name.sa.sa_family == AF_INET)
676 {
paul2ba9a372005-05-19 01:37:50 +0000677 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000678 memcpy (su, &name, sizeof (struct sockaddr_in));
679 return su;
680 }
681#ifdef HAVE_IPV6
682 if (name.sa.sa_family == AF_INET6)
683 {
paul2ba9a372005-05-19 01:37:50 +0000684 su = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
paul718e3742002-12-13 20:15:29 +0000685 memcpy (su, &name, sizeof (struct sockaddr_in6));
Paul Jakma1a7dcf42008-09-09 21:17:04 +0100686 sockunion_normalise_mapped (su);
paul718e3742002-12-13 20:15:29 +0000687 return su;
688 }
689#endif /* HAVE_IPV6 */
690 return NULL;
691}
692
693/* Print sockunion structure */
paul8cc41982005-05-06 21:25:49 +0000694static void __attribute__ ((unused))
paul718e3742002-12-13 20:15:29 +0000695sockunion_print (union sockunion *su)
696{
697 if (su == NULL)
698 return;
699
700 switch (su->sa.sa_family)
701 {
702 case AF_INET:
703 printf ("%s\n", inet_ntoa (su->sin.sin_addr));
704 break;
705#ifdef HAVE_IPV6
706 case AF_INET6:
707 {
paul42d49862004-10-13 05:22:18 +0000708 char buf [SU_ADDRSTRLEN];
paul718e3742002-12-13 20:15:29 +0000709
710 printf ("%s\n", inet_ntop (AF_INET6, &(su->sin6.sin6_addr),
711 buf, sizeof (buf)));
712 }
713 break;
714#endif /* HAVE_IPV6 */
715
716#ifdef AF_LINK
717 case AF_LINK:
718 {
719 struct sockaddr_dl *sdl;
720
721 sdl = (struct sockaddr_dl *)&(su->sa);
722 printf ("link#%d\n", sdl->sdl_index);
723 }
724 break;
725#endif /* AF_LINK */
726 default:
727 printf ("af_unknown %d\n", su->sa.sa_family);
728 break;
729 }
730}
731
732#ifdef HAVE_IPV6
paul8cc41982005-05-06 21:25:49 +0000733static int
paul718e3742002-12-13 20:15:29 +0000734in6addr_cmp (struct in6_addr *addr1, struct in6_addr *addr2)
735{
hasso8c328f12004-10-05 21:01:23 +0000736 unsigned int i;
paul718e3742002-12-13 20:15:29 +0000737 u_char *p1, *p2;
738
739 p1 = (u_char *)addr1;
740 p2 = (u_char *)addr2;
741
742 for (i = 0; i < sizeof (struct in6_addr); i++)
743 {
744 if (p1[i] > p2[i])
745 return 1;
746 else if (p1[i] < p2[i])
747 return -1;
748 }
749 return 0;
750}
751#endif /* HAVE_IPV6 */
752
753int
754sockunion_cmp (union sockunion *su1, union sockunion *su2)
755{
756 if (su1->sa.sa_family > su2->sa.sa_family)
757 return 1;
758 if (su1->sa.sa_family < su2->sa.sa_family)
759 return -1;
760
761 if (su1->sa.sa_family == AF_INET)
762 {
763 if (ntohl (su1->sin.sin_addr.s_addr) == ntohl (su2->sin.sin_addr.s_addr))
764 return 0;
765 if (ntohl (su1->sin.sin_addr.s_addr) > ntohl (su2->sin.sin_addr.s_addr))
766 return 1;
767 else
768 return -1;
769 }
770#ifdef HAVE_IPV6
771 if (su1->sa.sa_family == AF_INET6)
772 return in6addr_cmp (&su1->sin6.sin6_addr, &su2->sin6.sin6_addr);
773#endif /* HAVE_IPV6 */
774 return 0;
775}
776
777/* Duplicate sockunion. */
778union sockunion *
779sockunion_dup (union sockunion *su)
780{
781 union sockunion *dup = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
782 memcpy (dup, su, sizeof (union sockunion));
783 return dup;
784}
785
786void
787sockunion_free (union sockunion *su)
788{
789 XFREE (MTYPE_SOCKUNION, su);
790}