inet_ntop.c
上传用户:wxp200602
上传日期:2007-10-30
资源大小:4028k
文件大小:5k
源码类别:

SNMP编程

开发平台:

Unix_Linux

  1. /* Id: inet_ntop.c,v 1.4 2001/04/17 07:53:47 lukem Exp  */
  2. /* $NetBSD: inet_ntop.c,v 1.9 2000/01/22 22:19:16 mycroft Exp $ */
  3. /* Copyright (c) 1996 by Internet Software Consortium.
  4.  *
  5.  * Permission to use, copy, modify, and distribute this software for any
  6.  * purpose with or without fee is hereby granted, provided that the above
  7.  * copyright notice and this permission notice appear in all copies.
  8.  *
  9.  * THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
  10.  * ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
  11.  * OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
  12.  * CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  13.  * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  14.  * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
  15.  * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
  16.  * SOFTWARE.
  17.  */
  18. #include <net-snmp/net-snmp-config.h>
  19. #ifndef HAVE_INET_NTOP
  20. #if HAVE_ARPA_NAMESER_H
  21. #include <arpa/nameser.h>
  22. #endif
  23.   /*
  24.    * Net-SNMP Win32 additions
  25.    */
  26. #if defined(HAVE_WINSOCK_H) || defined(cygwin)
  27. #include <winsock2.h>
  28. #include <ws2tcpip.h>
  29. #include <errno.h>
  30. #include <stdio.h>
  31. #endif
  32. #ifndef EAFNOSUPPORT
  33. #define EAFNOSUPPORT            WSAEAFNOSUPPORT
  34. #endif
  35.   /*
  36.    * End of Net-SNMP Win32 additions
  37.    */
  38. #ifndef IN6ADDRSZ
  39. #define IN6ADDRSZ 16
  40. #endif
  41. #ifndef INT16SZ
  42. #define INT16SZ 2
  43. #endif
  44. #ifdef SPRINTF_CHAR
  45. # define SPRINTF(x) strlen(sprintf/**/x)
  46. #else
  47. # define SPRINTF(x) ((size_t)sprintf x)
  48. #endif
  49. /*
  50.  * WARNING: Don't even consider trying to compile this on a system where
  51.  * sizeof(int) < 4.  sizeof(int) > 4 is fine; all the world's not a VAX.
  52.  */
  53. static const char *inet_ntop4(const u_char *src, char *dst, size_t size);
  54. #ifdef INET6
  55. static const char *inet_ntop6(const u_char *src, char *dst, size_t size);
  56. #endif /* INET6 */
  57. /* char *
  58.  * inet_ntop(af, src, dst, size)
  59.  * convert a network format address to presentation format.
  60.  * return:
  61.  * pointer to presentation format address (`dst'), or NULL (see errno).
  62.  * author:
  63.  * Paul Vixie, 1996.
  64.  */
  65. const char *
  66. inet_ntop(af, src, dst, size)
  67. int af;
  68. const void *src;
  69. char *dst;
  70. size_t size;
  71. {
  72. switch (af) {
  73. case AF_INET:
  74. return (inet_ntop4(src, dst, size));
  75. #ifdef INET6
  76. case AF_INET6:
  77. return (inet_ntop6(src, dst, size));
  78. #endif
  79. default:
  80. errno = EAFNOSUPPORT;
  81. return (NULL);
  82. }
  83. /* NOTREACHED */
  84. }
  85. /* const char *
  86.  * inet_ntop4(src, dst, size)
  87.  * format an IPv4 address, more or less like inet_ntoa()
  88.  * return:
  89.  * `dst' (as a const)
  90.  * notes:
  91.  * (1) uses no statics
  92.  * (2) takes a u_char* not an in_addr as input
  93.  * author:
  94.  * Paul Vixie, 1996.
  95.  */
  96. static const char *
  97. inet_ntop4(src, dst, size)
  98. const u_char *src;
  99. char *dst;
  100. size_t size;
  101. {
  102. static const char fmt[] = "%u.%u.%u.%u";
  103. char tmp[sizeof "255.255.255.255"];
  104. if (SPRINTF((tmp, fmt, src[0], src[1], src[2], src[3])) > size) {
  105. errno = ENOSPC;
  106. return (NULL);
  107. }
  108. strcpy(dst, tmp);
  109. return (dst);
  110. }
  111. #ifdef INET6
  112. /* const char *
  113.  * inet_ntop6(src, dst, size)
  114.  * convert IPv6 binary address into presentation (printable) format
  115.  * author:
  116.  * Paul Vixie, 1996.
  117.  */
  118. static const char *
  119. inet_ntop6(src, dst, size)
  120. const u_char *src;
  121. char *dst;
  122. size_t size;
  123. {
  124. /*
  125.  * Note that int32_t and int16_t need only be "at least" large enough
  126.  * to contain a value of the specified size.  On some systems, like
  127.  * Crays, there is no such thing as an integer variable with 16 bits.
  128.  * Keep this in mind if you think this function should have been coded
  129.  * to use pointer overlays.  All the world's not a VAX.
  130.  */
  131. char tmp[sizeof "ffff:ffff:ffff:ffff:ffff:ffff:255.255.255.255"], *tp;
  132. struct { int base, len; } best, cur;
  133. u_int words[IN6ADDRSZ / INT16SZ];
  134. int i;
  135. /*
  136.  * Preprocess:
  137.  * Copy the input (bytewise) array into a wordwise array.
  138.  * Find the longest run of 0x00's in src[] for :: shorthanding.
  139.  */
  140. memset(words, '', sizeof words);
  141. for (i = 0; i < IN6ADDRSZ; i++)
  142. words[i / 2] |= (src[i] << ((1 - (i % 2)) << 3));
  143. best.base = -1;
  144. cur.base = -1;
  145. for (i = 0; i < (IN6ADDRSZ / INT16SZ); i++) {
  146. if (words[i] == 0) {
  147. if (cur.base == -1)
  148. cur.base = i, cur.len = 1;
  149. else
  150. cur.len++;
  151. } else {
  152. if (cur.base != -1) {
  153. if (best.base == -1 || cur.len > best.len)
  154. best = cur;
  155. cur.base = -1;
  156. }
  157. }
  158. }
  159. if (cur.base != -1) {
  160. if (best.base == -1 || cur.len > best.len)
  161. best = cur;
  162. }
  163. if (best.base != -1 && best.len < 2)
  164. best.base = -1;
  165. /*
  166.  * Format the result.
  167.  */
  168. tp = tmp;
  169. for (i = 0; i < (IN6ADDRSZ / INT16SZ); i++) {
  170. /* Are we inside the best run of 0x00's? */
  171. if (best.base != -1 && i >= best.base &&
  172.     i < (best.base + best.len)) {
  173. if (i == best.base)
  174. *tp++ = ':';
  175. continue;
  176. }
  177. /* Are we following an initial run of 0x00s or any real hex? */
  178. if (i != 0)
  179. *tp++ = ':';
  180. /* Is this address an encapsulated IPv4? */
  181. if (i == 6 && best.base == 0 &&
  182.     (best.len == 6 || (best.len == 5 && words[5] == 0xffff))) {
  183. if (!inet_ntop4(src+12, tp, sizeof tmp - (tp - tmp)))
  184. return (NULL);
  185. tp += strlen(tp);
  186. break;
  187. }
  188. tp += SPRINTF((tp, "%x", words[i]));
  189. }
  190. /* Was it a trailing run of 0x00's? */
  191. if (best.base != -1 && (best.base + best.len) == (IN6ADDRSZ / INT16SZ))
  192. *tp++ = ':';
  193. *tp++ = '';
  194. /*
  195.  * Check for overflow, copy, and we're done.
  196.  */
  197. if ((size_t)(tp - tmp) > size) {
  198. errno = ENOSPC;
  199. return (NULL);
  200. }
  201. strcpy(dst, tmp);
  202. return (dst);
  203. }
  204. #endif
  205. #endif /* HAVE_INET_NTOP */