sockunion.c
上传用户:xiaozhuqw
上传日期:2009-11-15
资源大小:1338k
文件大小:16k
源码类别:

网络

开发平台:

Unix_Linux

  1. /* 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. #include <zebra.h>
  22. #include "prefix.h"
  23. #include "vty.h"
  24. #include "sockunion.h"
  25. #include "memory.h"
  26. #include "str.h"
  27. #include "log.h"
  28. #ifndef HAVE_INET_ATON
  29. int
  30. inet_aton (const char *cp, struct in_addr *inaddr)
  31. {
  32.   int dots = 0;
  33.   register u_long addr = 0;
  34.   register u_long val = 0, base = 10;
  35.   do
  36.     {
  37.       register char c = *cp;
  38.       switch (c)
  39. {
  40. case '0': case '1': case '2': case '3': case '4': case '5':
  41. case '6': case '7': case '8': case '9':
  42.   val = (val * base) + (c - '0');
  43.   break;
  44. case '.':
  45.   if (++dots > 3)
  46.     return 0;
  47. case '':
  48.   if (val > 255)
  49.     return 0;
  50.   addr = addr << 8 | val;
  51.   val = 0;
  52.   break;
  53. default:
  54.   return 0;
  55. }
  56.     } while (*cp++) ;
  57.   if (dots < 3)
  58.     addr <<= 8 * (3 - dots);
  59.   if (inaddr)
  60.     inaddr->s_addr = htonl (addr);
  61.   return 1;
  62. }
  63. #endif /* ! HAVE_INET_ATON */
  64. #ifndef HAVE_INET_PTON
  65. int
  66. inet_pton (int family, const char *strptr, void *addrptr)
  67. {
  68.   if (family == AF_INET)
  69.     {
  70.       struct in_addr in_val;
  71.       if (inet_aton (strptr, &in_val))
  72. {
  73.   memcpy (addrptr, &in_val, sizeof (struct in_addr));
  74.   return 1;
  75. }
  76.       return 0;
  77.     }
  78.   errno = EAFNOSUPPORT;
  79.   return -1;
  80. }
  81. #endif /* ! HAVE_INET_PTON */
  82. #ifndef HAVE_INET_NTOP
  83. const char *
  84. inet_ntop (int family, const void *addrptr, char *strptr, size_t len)
  85. {
  86.   unsigned char *p = (unsigned char *) addrptr;
  87.   if (family == AF_INET) 
  88.     {
  89.       char temp[INET_ADDRSTRLEN];
  90.       snprintf(temp, sizeof(temp), "%d.%d.%d.%d", p[0], p[1], p[2], p[3]);
  91.       if (strlen(temp) >= len) 
  92. {
  93.   errno = ENOSPC;
  94.   return NULL;
  95. }
  96.       strcpy(strptr, temp);
  97.       return strptr;
  98.     }
  99.   errno = EAFNOSUPPORT;
  100.   return NULL;
  101. }
  102. #endif /* ! HAVE_INET_NTOP */
  103. const char *
  104. inet_sutop (union sockunion *su, char *str)
  105. {
  106.   switch (su->sa.sa_family)
  107.     {
  108.     case AF_INET:
  109.       inet_ntop (AF_INET, &su->sin.sin_addr, str, INET_ADDRSTRLEN);
  110.       break;
  111. #ifdef HAVE_IPV6
  112.     case AF_INET6:
  113.       inet_ntop (AF_INET6, &su->sin6.sin6_addr, str, INET6_ADDRSTRLEN);
  114.       break;
  115. #endif /* HAVE_IPV6 */
  116.     }
  117.   return str;
  118. }
  119. int
  120. str2sockunion (char *str, union sockunion *su)
  121. {
  122.   int ret;
  123.   memset (su, 0, sizeof (union sockunion));
  124.   ret = inet_pton (AF_INET, str, &su->sin.sin_addr);
  125.   if (ret > 0) /* Valid IPv4 address format. */
  126.     {
  127.       su->sin.sin_family = AF_INET;
  128. #ifdef HAVE_SIN_LEN
  129.       su->sin.sin_len = sizeof(struct sockaddr_in);
  130. #endif /* HAVE_SIN_LEN */
  131.       return 0;
  132.     }
  133. #ifdef HAVE_IPV6
  134.   ret = inet_pton (AF_INET6, str, &su->sin6.sin6_addr);
  135.   if (ret > 0) /* Valid IPv6 address format. */
  136.     {
  137.       su->sin6.sin6_family = AF_INET6;
  138. #ifdef SIN6_LEN
  139.       su->sin6.sin6_len = sizeof(struct sockaddr_in6);
  140. #endif /* SIN6_LEN */
  141.       return 0;
  142.     }
  143. #endif /* HAVE_IPV6 */
  144.   return -1;
  145. }
  146. const char *
  147. sockunion2str (union sockunion *su, char *buf, size_t len)
  148. {
  149.   if  (su->sa.sa_family == AF_INET)
  150.     return inet_ntop (AF_INET, &su->sin.sin_addr, buf, len);
  151. #ifdef HAVE_IPV6
  152.   else if (su->sa.sa_family == AF_INET6)
  153.     return inet_ntop (AF_INET6, &su->sin6.sin6_addr, buf, len);
  154. #endif /* HAVE_IPV6 */
  155.   return NULL;
  156. }
  157. union sockunion *
  158. sockunion_str2su (char *str)
  159. {
  160.   int ret;
  161.   union sockunion *su;
  162.   su = XMALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
  163.   memset (su, 0, sizeof (union sockunion));
  164.   ret = inet_pton (AF_INET, str, &su->sin.sin_addr);
  165.   if (ret > 0) /* Valid IPv4 address format. */
  166.     {
  167.       su->sin.sin_family = AF_INET;
  168. #ifdef HAVE_SIN_LEN
  169.       su->sin.sin_len = sizeof(struct sockaddr_in);
  170. #endif /* HAVE_SIN_LEN */
  171.       return su;
  172.     }
  173. #ifdef HAVE_IPV6
  174.   ret = inet_pton (AF_INET6, str, &su->sin6.sin6_addr);
  175.   if (ret > 0) /* Valid IPv6 address format. */
  176.     {
  177.       su->sin6.sin6_family = AF_INET6;
  178. #ifdef SIN6_LEN
  179.       su->sin6.sin6_len = sizeof(struct sockaddr_in6);
  180. #endif /* SIN6_LEN */
  181.       return su;
  182.     }
  183. #endif /* HAVE_IPV6 */
  184.   XFREE (MTYPE_SOCKUNION, su);
  185.   return NULL;
  186. }
  187. char *
  188. sockunion_su2str (union sockunion *su)
  189. {
  190.   char str[INET6_ADDRSTRLEN];
  191.   switch (su->sa.sa_family)
  192.     {
  193.     case AF_INET:
  194.       inet_ntop (AF_INET, &su->sin.sin_addr, str, sizeof (str));
  195.       break;
  196. #ifdef HAVE_IPV6
  197.     case AF_INET6:
  198.       inet_ntop (AF_INET6, &su->sin6.sin6_addr, str, sizeof (str));
  199.       break;
  200. #endif /* HAVE_IPV6 */
  201.     }
  202.   return strdup (str);
  203. }
  204. /* Return socket of sockunion. */
  205. int
  206. sockunion_socket (union sockunion *su)
  207. {
  208.   int sock;
  209.   sock = socket (su->sa.sa_family, SOCK_STREAM, 0);
  210.   if (sock < 0)
  211.     {
  212.       zlog (NULL, LOG_WARNING, "Can't make socket : %s", strerror (errno));
  213.       return -1;
  214.     }
  215.   return sock;
  216. }
  217. /* Return accepted new socket file descriptor. */
  218. int
  219. sockunion_accept (int sock, union sockunion *su)
  220. {
  221.   socklen_t len;
  222.   int client_sock;
  223.   len = sizeof (union sockunion);
  224.   client_sock = accept (sock, (struct sockaddr *) su, &len);
  225.   
  226.   /* Convert IPv4 compatible IPv6 address to IPv4 address. */
  227. #ifdef HAVE_IPV6
  228.   if (su->sa.sa_family == AF_INET6)
  229.     {
  230.       if (IN6_IS_ADDR_V4MAPPED (&su->sin6.sin6_addr))
  231. {
  232.   struct sockaddr_in sin;
  233.   memset (&sin, 0, sizeof (struct sockaddr_in));
  234.   sin.sin_family = AF_INET;
  235.   memcpy (&sin.sin_addr, ((char *)&su->sin6.sin6_addr) + 12, 4);
  236.   memcpy (su, &sin, sizeof (struct sockaddr_in));
  237. }
  238.     }
  239. #endif /* HAVE_IPV6 */
  240.   return client_sock;
  241. }
  242. /* Return sizeof union sockunion.  */
  243. int
  244. sockunion_sizeof (union sockunion *su)
  245. {
  246.   int ret;
  247.   ret = 0;
  248.   switch (su->sa.sa_family)
  249.     {
  250.     case AF_INET:
  251.       ret = sizeof (struct sockaddr_in);
  252.       break;
  253. #ifdef HAVE_IPV6
  254.     case AF_INET6:
  255.       ret = sizeof (struct sockaddr_in6);
  256.       break;
  257. #endif /* AF_INET6 */
  258.     }
  259.   return ret;
  260. }
  261. /* return sockunion structure : this function should be revised. */
  262. char *
  263. sockunion_log (union sockunion *su)
  264. {
  265.   static char buf[SU_ADDRSTRLEN];
  266.   switch (su->sa.sa_family) 
  267.     {
  268.     case AF_INET:
  269.       snprintf (buf, BUFSIZ, "%s", inet_ntoa (su->sin.sin_addr));
  270.       break;
  271. #ifdef HAVE_IPV6
  272.     case AF_INET6:
  273.       snprintf (buf, BUFSIZ, "%s",
  274. inet_ntop (AF_INET6, &(su->sin6.sin6_addr), buf, BUFSIZ));
  275.       break;
  276. #endif /* HAVE_IPV6 */
  277.     default:
  278.       snprintf (buf, BUFSIZ, "af_unknown %d ", su->sa.sa_family);
  279.       break;
  280.     }
  281.   return buf;
  282. }
  283. /* sockunion_connect returns
  284.    -1 : error occured
  285.    0 : connect success
  286.    1 : connect is in progress */
  287. enum connect_result
  288. sockunion_connect (int fd, union sockunion *peersu, unsigned short port,
  289.    unsigned int ifindex)
  290. {
  291.   int ret;
  292.   int val;
  293.   union sockunion su;
  294.   memcpy (&su, peersu, sizeof (union sockunion));
  295.   switch (su.sa.sa_family)
  296.     {
  297.     case AF_INET:
  298.       su.sin.sin_port = port;
  299.       break;
  300. #ifdef HAVE_IPV6
  301.     case AF_INET6:
  302.       su.sin6.sin6_port  = port;
  303. #ifdef KAME
  304.       if (IN6_IS_ADDR_LINKLOCAL(&su.sin6.sin6_addr) && ifindex)
  305. {
  306. #ifdef HAVE_SIN6_SCOPE_ID
  307.   /* su.sin6.sin6_scope_id = ifindex; */
  308. #endif /* HAVE_SIN6_SCOPE_ID */
  309.   SET_IN6_LINKLOCAL_IFINDEX (su.sin6.sin6_addr, ifindex);
  310. }
  311. #endif /* KAME */
  312.       break;
  313. #endif /* HAVE_IPV6 */
  314.     }      
  315.   /* Make socket non-block. */
  316.   val = fcntl (fd, F_GETFL, 0);
  317.   fcntl (fd, F_SETFL, val|O_NONBLOCK);
  318.   /* Call connect function. */
  319.   ret = connect (fd, (struct sockaddr *) &su, sockunion_sizeof (&su));
  320.   /* Immediate success */
  321.   if (ret == 0)
  322.     {
  323.       fcntl (fd, F_SETFL, val);
  324.       return connect_success;
  325.     }
  326.   /* If connect is in progress then return 1 else it's real error. */
  327.   if (ret < 0)
  328.     {
  329.       if (errno != EINPROGRESS)
  330. {
  331.   zlog_info ("can't connect to %s fd %d : %s",
  332.      sockunion_log (&su), fd, strerror (errno));
  333.   return connect_error;
  334. }
  335.     }
  336.   fcntl (fd, F_SETFL, val);
  337.   return connect_in_progress;
  338. }
  339. /* Make socket from sockunion union. */
  340. int
  341. sockunion_stream_socket (union sockunion *su)
  342. {
  343.   int sock;
  344.   if (su->sa.sa_family == 0)
  345.     su->sa.sa_family = AF_INET_UNION;
  346.   sock = socket (su->sa.sa_family, SOCK_STREAM, 0);
  347.   if (sock < 0)
  348.     zlog (NULL, LOG_WARNING, "can't make socket sockunion_stream_socket");
  349.   return sock;
  350. }
  351. /* Bind socket to specified address. */
  352. int
  353. sockunion_bind (int sock, union sockunion *su, unsigned short port, 
  354. union sockunion *su_addr)
  355. {
  356.   int size = 0;
  357.   int ret;
  358.   if (su->sa.sa_family == AF_INET)
  359.     {
  360.       size = sizeof (struct sockaddr_in);
  361.       su->sin.sin_port = htons (port);
  362. #ifdef HAVE_SIN_LEN
  363.       su->sin.sin_len = size;
  364. #endif /* HAVE_SIN_LEN */
  365.       if (su_addr == NULL)
  366. su->sin.sin_addr.s_addr = htonl (INADDR_ANY);
  367.     }
  368. #ifdef HAVE_IPV6
  369.   else if (su->sa.sa_family == AF_INET6)
  370.     {
  371.       size = sizeof (struct sockaddr_in6);
  372.       su->sin6.sin6_port = htons (port);
  373. #ifdef SIN6_LEN
  374.       su->sin6.sin6_len = size;
  375. #endif /* SIN6_LEN */
  376.       if (su_addr == NULL)
  377. {
  378. #if defined(LINUX_IPV6) || defined(NRL)
  379.   memset (&su->sin6.sin6_addr, 0, sizeof (struct in6_addr));
  380. #else
  381.   su->sin6.sin6_addr = in6addr_any;
  382. #endif /* LINUX_IPV6 */
  383. }
  384.     }
  385. #endif /* HAVE_IPV6 */
  386.   
  387.   ret = bind (sock, (struct sockaddr *)su, size);
  388.   if (ret < 0)
  389.     zlog (NULL, LOG_WARNING, "can't bind socket : %s", strerror (errno));
  390.   return ret;
  391. }
  392. int
  393. sockopt_reuseaddr (int sock)
  394. {
  395.   int ret;
  396.   int on = 1;
  397.   ret = setsockopt (sock, SOL_SOCKET, SO_REUSEADDR, 
  398.     (void *) &on, sizeof (on));
  399.   if (ret < 0)
  400.     {
  401.       zlog (NULL, LOG_WARNING, "can't set sockopt SO_REUSEADDR to socket %d", sock);
  402.       return -1;
  403.     }
  404.   return 0;
  405. }
  406. #ifdef SO_REUSEPORT
  407. int
  408. sockopt_reuseport (int sock)
  409. {
  410.   int ret;
  411.   int on = 1;
  412.   ret = setsockopt (sock, SOL_SOCKET, SO_REUSEPORT, 
  413.     (void *) &on, sizeof (on));
  414.   if (ret < 0)
  415.     {
  416.       zlog (NULL, LOG_WARNING, "can't set sockopt SO_REUSEADDR to socket %d", sock);
  417.       return -1;
  418.     }
  419.   return 0;
  420. }
  421. #else
  422. int
  423. sockopt_reuseport (int sock)
  424. {
  425.   return 0;
  426. }
  427. #endif /* 0 */
  428. int
  429. sockopt_ttl (int family, int sock, int ttl)
  430. {
  431.   int ret;
  432. #ifdef IP_TTL
  433.   if (family == AF_INET)
  434.     {
  435.       ret = setsockopt (sock, IPPROTO_IP, IP_TTL, 
  436. (void *) &ttl, sizeof (int));
  437.       if (ret < 0)
  438. {
  439.   zlog (NULL, LOG_WARNING, "can't set sockopt IP_TTL %d to socket %d", ttl, sock);
  440.   return -1;
  441. }
  442.       return 0;
  443.     }
  444. #endif /* IP_TTL */
  445. #ifdef HAVE_IPV6
  446.   if (family == AF_INET6)
  447.     {
  448.       ret = setsockopt (sock, IPPROTO_IPV6, IPV6_UNICAST_HOPS, 
  449. (void *) &ttl, sizeof (int));
  450.       if (ret < 0)
  451. {
  452.   zlog (NULL, LOG_WARNING, "can't set sockopt IPV6_UNICAST_HOPS %d to socket %d",
  453.     ttl, sock);
  454.   return -1;
  455. }
  456.       return 0;
  457.     }
  458. #endif /* HAVE_IPV6 */
  459.   return 0;
  460. }
  461. /* If same family and same prefix return 1. */
  462. int
  463. sockunion_same (union sockunion *su1, union sockunion *su2)
  464. {
  465.   int ret = 0;
  466.   if (su1->sa.sa_family != su2->sa.sa_family)
  467.     return 0;
  468.   switch (su1->sa.sa_family)
  469.     {
  470.     case AF_INET:
  471.       ret = memcmp (&su1->sin.sin_addr, &su2->sin.sin_addr,
  472.     sizeof (struct in_addr));
  473.       break;
  474. #ifdef HAVE_IPV6
  475.     case AF_INET6:
  476.       ret = memcmp (&su1->sin6.sin6_addr, &su2->sin6.sin6_addr,
  477.     sizeof (struct in6_addr));
  478.       break;
  479. #endif /* HAVE_IPV6 */
  480.     }
  481.   if (ret == 0)
  482.     return 1;
  483.   else
  484.     return 0;
  485. }
  486. /* After TCP connection is established.  Get local address and port. */
  487. union sockunion *
  488. sockunion_getsockname (int fd)
  489. {
  490.   int ret;
  491.   int len;
  492.   union
  493.   {
  494.     struct sockaddr sa;
  495.     struct sockaddr_in sin;
  496. #ifdef HAVE_IPV6
  497.     struct sockaddr_in6 sin6;
  498. #endif /* HAVE_IPV6 */
  499.     char tmp_buffer[128];
  500.   } name;
  501.   union sockunion *su;
  502.   memset (&name, 0, sizeof name);
  503.   len = sizeof name;
  504.   ret = getsockname (fd, (struct sockaddr *)&name, &len);
  505.   if (ret < 0)
  506.     {
  507.       zlog_warn ("Can't get local address and port by getsockname: %s",
  508.  strerror (errno));
  509.       return NULL;
  510.     }
  511.   if (name.sa.sa_family == AF_INET)
  512.     {
  513.       su = XCALLOC (MTYPE_TMP, sizeof (union sockunion));
  514.       memcpy (su, &name, sizeof (struct sockaddr_in));
  515.       return su;
  516.     }
  517. #ifdef HAVE_IPV6
  518.   if (name.sa.sa_family == AF_INET6)
  519.     {
  520.       su = XCALLOC (MTYPE_TMP, sizeof (union sockunion));
  521.       memcpy (su, &name, sizeof (struct sockaddr_in6));
  522.       if (IN6_IS_ADDR_V4MAPPED (&su->sin6.sin6_addr))
  523. {
  524.   struct sockaddr_in sin;
  525.   sin.sin_family = AF_INET;
  526.   memcpy (&sin.sin_addr, ((char *)&su->sin6.sin6_addr) + 12, 4);
  527.   sin.sin_port = su->sin6.sin6_port;
  528.   memcpy (su, &sin, sizeof (struct sockaddr_in));
  529. }
  530.       return su;
  531.     }
  532. #endif /* HAVE_IPV6 */
  533.   return NULL;
  534. }
  535. /* After TCP connection is established.  Get remote address and port. */
  536. union sockunion *
  537. sockunion_getpeername (int fd)
  538. {
  539.   int ret;
  540.   int len;
  541.   union
  542.   {
  543.     struct sockaddr sa;
  544.     struct sockaddr_in sin;
  545. #ifdef HAVE_IPV6
  546.     struct sockaddr_in6 sin6;
  547. #endif /* HAVE_IPV6 */
  548.     char tmp_buffer[128];
  549.   } name;
  550.   union sockunion *su;
  551.   memset (&name, 0, sizeof name);
  552.   len = sizeof name;
  553.   ret = getpeername (fd, (struct sockaddr *)&name, &len);
  554.   if (ret < 0)
  555.     {
  556.       zlog (NULL, LOG_WARNING, "Can't get remote address and port: %s",
  557.     strerror (errno));
  558.       return NULL;
  559.     }
  560.   if (name.sa.sa_family == AF_INET)
  561.     {
  562.       su = XCALLOC (MTYPE_TMP, sizeof (union sockunion));
  563.       memcpy (su, &name, sizeof (struct sockaddr_in));
  564.       return su;
  565.     }
  566. #ifdef HAVE_IPV6
  567.   if (name.sa.sa_family == AF_INET6)
  568.     {
  569.       su = XCALLOC (MTYPE_TMP, sizeof (union sockunion));
  570.       memcpy (su, &name, sizeof (struct sockaddr_in6));
  571.       if (IN6_IS_ADDR_V4MAPPED (&su->sin6.sin6_addr))
  572. {
  573.   struct sockaddr_in sin;
  574.   sin.sin_family = AF_INET;
  575.   memcpy (&sin.sin_addr, ((char *)&su->sin6.sin6_addr) + 12, 4);
  576.   sin.sin_port = su->sin6.sin6_port;
  577.   memcpy (su, &sin, sizeof (struct sockaddr_in));
  578. }
  579.       return su;
  580.     }
  581. #endif /* HAVE_IPV6 */
  582.   return NULL;
  583. }
  584. /* Print sockunion structure */
  585. void
  586. sockunion_print (union sockunion *su)
  587. {
  588.   if (su == NULL)
  589.     return;
  590.   switch (su->sa.sa_family) 
  591.     {
  592.     case AF_INET:
  593.       printf ("%sn", inet_ntoa (su->sin.sin_addr));
  594.       break;
  595. #ifdef HAVE_IPV6
  596.     case AF_INET6:
  597.       {
  598. char buf [64];
  599. printf ("%sn", inet_ntop (AF_INET6, &(su->sin6.sin6_addr),
  600.  buf, sizeof (buf)));
  601.       }
  602.       break;
  603. #endif /* HAVE_IPV6 */
  604. #ifdef AF_LINK
  605.     case AF_LINK:
  606.       {
  607. struct sockaddr_dl *sdl;
  608. sdl = (struct sockaddr_dl *)&(su->sa);
  609. printf ("link#%dn", sdl->sdl_index);
  610.       }
  611.       break;
  612. #endif /* AF_LINK */
  613.     default:
  614.       printf ("af_unknown %dn", su->sa.sa_family);
  615.       break;
  616.     }
  617. }
  618. #ifdef HAVE_IPV6
  619. int
  620. in6addr_cmp (struct in6_addr *addr1, struct in6_addr *addr2)
  621. {
  622.   int i;
  623.   u_char *p1, *p2;
  624.   p1 = (u_char *)addr1;
  625.   p2 = (u_char *)addr2;
  626.   for (i = 0; i < sizeof (struct in6_addr); i++)
  627.     {
  628.       if (p1[i] > p2[i])
  629. return 1;
  630.       else if (p1[i] < p2[i])
  631. return -1;
  632.     }
  633.   return 0;
  634. }
  635. #endif /* HAVE_IPV6 */
  636. int
  637. sockunion_cmp (union sockunion *su1, union sockunion *su2)
  638. {
  639.   if (su1->sa.sa_family > su2->sa.sa_family)
  640.     return 1;
  641.   if (su1->sa.sa_family < su2->sa.sa_family)
  642.     return -1;
  643.   if (su1->sa.sa_family == AF_INET)
  644.     {
  645.       if (ntohl (su1->sin.sin_addr.s_addr) == ntohl (su2->sin.sin_addr.s_addr))
  646. return 0;
  647.       if (ntohl (su1->sin.sin_addr.s_addr) > ntohl (su2->sin.sin_addr.s_addr))
  648. return 1;
  649.       else
  650. return -1;
  651.     }
  652. #ifdef HAVE_IPV6
  653.   if (su1->sa.sa_family == AF_INET6)
  654.     return in6addr_cmp (&su1->sin6.sin6_addr, &su2->sin6.sin6_addr);
  655. #endif /* HAVE_IPV6 */
  656.   return 0;
  657. }
  658. /* Duplicate sockunion. */
  659. union sockunion *
  660. sockunion_dup (union sockunion *su)
  661. {
  662.   union sockunion *dup = XCALLOC (MTYPE_SOCKUNION, sizeof (union sockunion));
  663.   memcpy (dup, su, sizeof (union sockunion));
  664.   return dup;
  665. }
  666. void
  667. sockunion_free (union sockunion *su)
  668. {
  669.   XFREE (MTYPE_SOCKUNION, su);
  670. }