tcp_wce.c
上传用户:ycwykj01
上传日期:2007-01-04
资源大小:1819k
文件大小:18k
源码类别:

网络编程

开发平台:

Unix_Linux

  1. /*
  2.  * Program: Winsock TCP/IP routines
  3.  *
  4.  * Author: Mark Crispin from Mike Seibel's Winsock code
  5.  * Networks and Distributed Computing
  6.  * Computing & Communications
  7.  * University of Washington
  8.  * Administration Building, AG-44
  9.  * Seattle, WA  98195
  10.  * Internet: MRC@CAC.Washington.EDU
  11.  *
  12.  * Date: 11 April 1989
  13.  * Last Edited: 2 November 1999
  14.  *
  15.  * Copyright 1999 by the University of Washington
  16.  *
  17.  *  Permission to use, copy, modify, and distribute this software and its
  18.  * documentation for any purpose and without fee is hereby granted, provided
  19.  * that the above copyright notice appears in all copies and that both the
  20.  * above copyright notice and this permission notice appear in supporting
  21.  * documentation, and that the name of the University of Washington not be
  22.  * used in advertising or publicity pertaining to distribution of the software
  23.  * without specific, written prior permission.  This software is made
  24.  * available "as is", and
  25.  * THE UNIVERSITY OF WASHINGTON DISCLAIMS ALL WARRANTIES, EXPRESS OR IMPLIED,
  26.  * WITH REGARD TO THIS SOFTWARE, INCLUDING WITHOUT LIMITATION ALL IMPLIED
  27.  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, AND IN
  28.  * NO EVENT SHALL THE UNIVERSITY OF WASHINGTON BE LIABLE FOR ANY SPECIAL,
  29.  * INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  30.  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, TORT
  31.  * (INCLUDING NEGLIGENCE) OR STRICT LIABILITY, ARISING OUT OF OR IN CONNECTION
  32.  * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  33.  *
  34.  */
  35. /* Private functions */
  36. long tcp_abort (SOCKET *sock);
  37. char *tcp_name (struct sockaddr_in *sin,long flag);
  38. /* Private data */
  39. int wsa_initted = 0; /* init ? */
  40. static int wsa_sock_open = 0; /* keep track of open sockets */
  41. static tcptimeout_t tmoh = NIL; /* TCP timeout handler routine */
  42. static long ttmo_read = 0; /* TCP timeouts, in seconds */
  43. static long ttmo_write = 0;
  44. static long allowreversedns = /* allow reverse DNS lookup */
  45. #ifdef DISABLE_REVERSE_DNS_LOOKUP
  46.   NIL /* Not recommended, especially if using Kerberos authentication */
  47. #else
  48.   T
  49. #endif
  50.   ;
  51. /* TCP/IP manipulate parameters
  52.  * Accepts: function code
  53.  *     function-dependent value
  54.  * Returns: function-dependent return value
  55.  */
  56. void *tcp_parameters (long function,void *value)
  57. {
  58.   switch ((int) function) {
  59.   case SET_TIMEOUT:
  60.     tmoh = (tcptimeout_t) value;
  61.     break;
  62.   case GET_TIMEOUT:
  63.     value = (void *) tmoh;
  64.     break;
  65.   case SET_READTIMEOUT:
  66.     ttmo_read = (long) value;
  67.     break;
  68.   case GET_READTIMEOUT:
  69.     value = (void *) ttmo_read;
  70.     break;
  71.   case SET_WRITETIMEOUT:
  72.     ttmo_write = (long) value;
  73.     break;
  74.   case GET_WRITETIMEOUT:
  75.     value = (void *) ttmo_write;
  76.     break;
  77.   default:
  78.     value = NIL; /* error case */
  79.     break;
  80.   }
  81.   return value;
  82. }
  83. /* TCP/IP open
  84.  * Accepts: host name
  85.  *     contact service name
  86.  *     contact port number and optional silent flag
  87.  * Returns: TCP/IP stream if success else NIL
  88.  */
  89. TCPSTREAM *tcp_open (char *host,char *service,unsigned long port)
  90. {
  91.   TCPSTREAM *stream = NIL;
  92.   SOCKET sock;
  93.   char *s;
  94.   struct sockaddr_in sin;
  95.   struct hostent *he;
  96.   char tmp[MAILTMPLEN];
  97.   char *hostname = NIL;
  98.   blocknotify_t bn = (blocknotify_t) mail_parameters (NIL,GET_BLOCKNOTIFY,NIL);
  99.   int silent = (port & 0x80000000) ? T : NIL;
  100.   port &= 0x7fffffff; /* erase silent flag */
  101.   if (!wsa_initted++) { /* init Windows Sockets */
  102.     WSADATA wsock;
  103.     int i = (int) WSAStartup (WSA_VERSION,&wsock);
  104.     if (i) { /* failed */
  105.       wsa_initted = 0; /* in case we try again */
  106.       sprintf (tmp,"Unable to start Windows Sockets (%d)",i);
  107.       mm_log (tmp,ERROR);
  108.       return NIL;
  109.     }
  110.   }
  111.   /* The domain literal form is used (rather than simply the dotted decimal
  112.      as with other Unix programs) because it has to be a valid "host name"
  113.      in mailsystem terminology. */
  114.   sin.sin_family = AF_INET; /* family is always Internet */
  115. /* look like domain literal? */
  116.   if (host[0] == '[' && host[(strlen (host))-1] == ']') {
  117.     strcpy (tmp,host+1); /* yes, copy number part */
  118.     tmp[strlen (tmp)-1] = '';
  119.     if ((sin.sin_addr.s_addr = inet_addr (tmp)) == INADDR_NONE) {
  120.       sprintf (tmp,"Bad format domain-literal: %.80s",host);
  121.       mm_log (tmp,ERROR);
  122.       return NIL;
  123.     }
  124.     else hostname = cpystr (host);
  125.   }
  126.   else { /* lookup host name */
  127.     if (bn) (*bn) (BLOCK_DNSLOOKUP,NIL);
  128.     if ((he = gethostbyname (lcase (strcpy (tmp,host))))) {
  129.       if (bn) (*bn) (BLOCK_NONE,NIL);
  130. /* copy host name */
  131.       hostname = cpystr (he->h_name);
  132. /* copy host addresses */
  133.       memcpy (&sin.sin_addr,he->h_addr,he->h_length);
  134.     }
  135.     else {
  136.       if (bn) (*bn) (BLOCK_NONE,NIL);
  137.       sprintf (tmp,"Host not found (#%d): %s",WSAGetLastError(),host);
  138.       mm_log (tmp,ERROR);
  139.       return NIL;
  140.     }
  141.   }
  142. /* copy port number in network format */
  143.   if (!(sin.sin_port = htons ((u_short) port)))
  144.     fatal ("Bad port argument to tcp_open");
  145.   if (bn) (*bn) (BLOCK_TCPOPEN,NIL);
  146. /* get a TCP stream */
  147.   if ((sock = socket (sin.sin_family,SOCK_STREAM,0)) == INVALID_SOCKET) {
  148.     sprintf (tmp,"Unable to create TCP socket (%d)",WSAGetLastError());
  149.     mm_log (tmp,ERROR);
  150.     fs_give ((void **) &hostname);
  151.     return NIL;
  152.   }
  153.   wsa_sock_open++; /* now have a socket open */
  154. /* open connection */
  155.   if (connect (sock,(struct sockaddr *) &sin,sizeof (sin)) == SOCKET_ERROR) {
  156.     switch (WSAGetLastError ()) { /* analyze error */
  157.     case WSAECONNREFUSED:
  158.       s = "Refused";
  159.       break;
  160.     case WSAENOBUFS:
  161.       s = "Insufficient system resources";
  162.       break;
  163.     case WSAETIMEDOUT:
  164.       s = "Timed out";
  165.       break;
  166.     default:
  167.       s = "Unknown error";
  168.       break;
  169.     }
  170.     sprintf (tmp,"Can't connect to %.80s,%ld: %s (%d)",hostname,port,s,
  171.      WSAGetLastError ());
  172.     if (!silent) mm_log (tmp,ERROR);
  173.     fs_give ((void **) &hostname);
  174.     return (TCPSTREAM *) tcp_abort (&sock);
  175.   }
  176.   if (bn) (*bn) (BLOCK_NONE,NIL);
  177. /* create TCP/IP stream */
  178.   stream = (TCPSTREAM *) memset (fs_get (sizeof (TCPSTREAM)),0,
  179.  sizeof (TCPSTREAM));
  180.   stream->host = hostname; /* official host name */
  181.   stream->port = port; /* port number */
  182.   stream->tcps = sock; /* init socket */
  183.   stream->ictr = 0; /* init input counter */
  184.   return stream; /* return success */
  185. }
  186.   
  187. /* TCP/IP authenticated open
  188.  * Accepts: NETMBX specifier
  189.  *     service name
  190.  *     returned user name buffer
  191.  * Returns: TCP/IP stream if success else NIL
  192.  */
  193. TCPSTREAM *tcp_aopen (NETMBX *mb,char *service,char *usrbuf)
  194. {
  195.   return NIL; /* always NIL on Windows */
  196. }
  197. /* TCP/IP receive line
  198.  * Accepts: TCP/IP stream
  199.  * Returns: text line string or NIL if failure
  200.  */
  201. char *tcp_getline (TCPSTREAM *stream)
  202. {
  203.   int n,m;
  204.   char *st,*ret,*stp;
  205.   char c = '';
  206.   char d;
  207. /* make sure have data */
  208.   if (!tcp_getdata (stream)) return NIL;
  209.   st = stream->iptr; /* save start of string */
  210.   n = 0; /* init string count */
  211.   while (stream->ictr--) { /* look for end of line */
  212.     d = *stream->iptr++; /* slurp another character */
  213.     if ((c == '15') && (d == '12')) {
  214.       ret = (char *) fs_get (n--);
  215.       memcpy (ret,st,n); /* copy into a free storage string */
  216.       ret[n] = ''; /* tie off string with null */
  217.       return ret;
  218.     }
  219.     n++; /* count another character searched */
  220.     c = d; /* remember previous character */
  221.   }
  222. /* copy partial string from buffer */
  223.   memcpy ((ret = stp = (char *) fs_get (n)),st,n);
  224. /* get more data from the net */
  225.   if (!tcp_getdata (stream)) fs_give ((void **) &ret);
  226. /* special case of newline broken by buffer */
  227.   else if ((c == '15') && (*stream->iptr == '12')) {
  228.     stream->iptr++; /* eat the line feed */
  229.     stream->ictr--;
  230.     ret[n - 1] = ''; /* tie off string with null */
  231.   }
  232. /* else recurse to get remainder */
  233.   else if (st = tcp_getline (stream)) {
  234.     ret = (char *) fs_get (n + 1 + (m = strlen (st)));
  235.     memcpy (ret,stp,n); /* copy first part */
  236.     memcpy (ret + n,st,m); /* and second part */
  237.     fs_give ((void **) &stp); /* flush first part */
  238.     fs_give ((void **) &st); /* flush second part */
  239.     ret[n + m] = ''; /* tie off string with null */
  240.   }
  241.   return ret;
  242. }
  243. /* TCP/IP receive buffer
  244.  * Accepts: TCP/IP stream
  245.  *     size in bytes
  246.  *     buffer to read into
  247.  * Returns: T if success, NIL otherwise
  248.  */
  249. long tcp_getbuffer (TCPSTREAM *stream,unsigned long size,char *buffer)
  250. {
  251.   unsigned long n;
  252.   char *bufptr = buffer;
  253.   while (size > 0) { /* until request satisfied */
  254.     if (!tcp_getdata (stream)) return NIL;
  255.     n = min (size,stream->ictr);/* number of bytes to transfer */
  256. /* do the copy */
  257.     memcpy (bufptr,stream->iptr,(size_t) n);
  258.     bufptr += n; /* update pointer */
  259.     stream->iptr +=n;
  260.     size -= n; /* update # of bytes to do */
  261.     stream->ictr -=n;
  262.   }
  263.   bufptr[0] = ''; /* tie off string */
  264.   return T;
  265. }
  266. /* TCP/IP receive data
  267.  * Accepts: TCP/IP stream
  268.  * Returns: T if success, NIL otherwise
  269.  */
  270. long tcp_getdata (TCPSTREAM *stream)
  271. {
  272.   struct timeval tmo;
  273.   int i;
  274.   fd_set fds;
  275.   time_t tc,t = time (0);
  276.   blocknotify_t bn = (blocknotify_t) mail_parameters (NIL,GET_BLOCKNOTIFY,NIL);
  277.   FD_ZERO (&fds); /* initialize selection vector */
  278.   if (stream->tcps == INVALID_SOCKET) return NIL;
  279.   if (bn) (*bn) (BLOCK_TCPREAD,NIL);
  280.   tmo.tv_sec = ttmo_read;
  281.   tmo.tv_usec = 0;
  282.   while (stream->ictr < 1) { /* if nothing in the buffer */
  283.     time_t tl = time (0);
  284.     FD_SET (stream->tcps,&fds); /* set bit in selection vector */
  285. /* block and read */
  286.     switch (select (stream->tcps+1,&fds,0,0,
  287.     ttmo_read ? &tmo : (struct timeval *) 0)) {
  288.     case SOCKET_ERROR: /* error */
  289.       if (WSAGetLastError () != WSAEINTR) return tcp_abort (&stream->tcps);
  290.       break;
  291.     case 0: /* timeout */
  292.       tc = time (0);
  293.       if (tmoh && ((*tmoh) (tc - t,tc - tl))) break;
  294.       return tcp_abort (&stream->tcps);
  295.     default:
  296.       while (((i = recv (stream->tcps,stream->ibuf,BUFLEN,0)) == SOCKET_ERROR)
  297.      && (WSAGetLastError () == WSAEINTR));
  298.       switch (i) {
  299.       case SOCKET_ERROR: /* error */
  300.       case 0: /* no data read */
  301. return tcp_abort (&stream->tcps);
  302.       default:
  303. stream->ictr = i; /* set new byte count */
  304. /* point at TCP buffer */
  305. stream->iptr = stream->ibuf;
  306.       }
  307.     }
  308.   }
  309.   if (bn) (*bn) (BLOCK_NONE,NIL);
  310.   return T;
  311. }
  312. /* TCP/IP send string as record
  313.  * Accepts: TCP/IP stream
  314.  *     string pointer
  315.  * Returns: T if success else NIL
  316.  */
  317. long tcp_soutr (TCPSTREAM *stream,char *string)
  318. {
  319.   return tcp_sout (stream,string,(unsigned long) strlen (string));
  320. }
  321. /* TCP/IP send string
  322.  * Accepts: TCP/IP stream
  323.  *     string pointer
  324.  *     byte count
  325.  * Returns: T if success else NIL
  326.  */
  327. long tcp_sout (TCPSTREAM *stream,char *string,unsigned long size)
  328. {
  329.   int i;
  330.   struct timeval tmo;
  331.   fd_set fds;
  332.   time_t tc,t = time (0);
  333.   blocknotify_t bn = (blocknotify_t) mail_parameters (NIL,GET_BLOCKNOTIFY,NIL);
  334.   tmo.tv_sec = ttmo_write;
  335.   tmo.tv_usec = 0;
  336.   FD_ZERO (&fds); /* initialize selection vector */
  337.   if (stream->tcps == INVALID_SOCKET) return NIL;
  338.   if (bn) (*bn) (BLOCK_TCPWRITE,NIL);
  339.   while (size > 0) { /* until request satisfied */
  340.     time_t tl = time (0);
  341.     FD_SET (stream->tcps,&fds); /* set bit in selection vector */
  342. /* block and write */
  343.     switch (select (stream->tcps+1,NULL,&fds,NULL,
  344.     tmo.tv_sec ? &tmo : (struct timeval *) 0)) {
  345.     case SOCKET_ERROR: /* error */
  346.       if (WSAGetLastError () != WSAEINTR) return tcp_abort (&stream->tcps);
  347.       break;
  348.     case 0: /* timeout */
  349.       tc = time (0);
  350.       if (tmoh && ((*tmoh) (tc - t,tc - tl))) break;
  351.       return tcp_abort (&stream->tcps);
  352.     default:
  353.       while (((i = send (stream->tcps,string,(int) size,0)) == SOCKET_ERROR) &&
  354.      (WSAGetLastError () == WSAEINTR));
  355.       if (i == SOCKET_ERROR) return tcp_abort (&stream->tcps);
  356.       size -= i; /* count this size */
  357.       string += i;
  358.     }
  359.   }
  360.   if (bn) (*bn) (BLOCK_NONE,NIL);
  361.   return T; /* all done */
  362. }
  363. /* TCP/IP close
  364.  * Accepts: TCP/IP stream
  365.  */
  366. void tcp_close (TCPSTREAM *stream)
  367. {
  368.   tcp_abort (&stream->tcps); /* nuke the socket */
  369. /* flush host names */
  370.   if (stream->host) fs_give ((void **) &stream->host);
  371.   if (stream->localhost) fs_give ((void **) &stream->localhost);
  372.   fs_give ((void **) &stream); /* flush the stream */
  373. }
  374. /* TCP/IP abort stream
  375.  * Accepts: WinSock socket
  376.  */
  377. long tcp_abort (SOCKET *sock)
  378. {
  379.   blocknotify_t bn = (blocknotify_t) mail_parameters (NIL,GET_BLOCKNOTIFY,NIL);
  380. /* something to close? */
  381.   if (sock && *sock != INVALID_SOCKET) {
  382.     if (bn) (*bn) (BLOCK_TCPCLOSE,NIL);
  383.     closesocket (*sock); /* WinSock socket close */
  384.     *sock = INVALID_SOCKET;
  385. /* no more open streams? */
  386.     if (wsa_initted && !--wsa_sock_open) {
  387.       mm_log ("Winsock cleanup",NIL);
  388.       wsa_initted = 0; /* no more sockets, so... */
  389.       WSACleanup (); /* free up resources until needed */
  390.     }
  391.   }
  392.   if (bn) (*bn) (BLOCK_NONE,NIL);
  393.   return NIL;
  394. }
  395. /* TCP/IP get host name
  396.  * Accepts: TCP/IP stream
  397.  * Returns: host name for this stream
  398.  */
  399. char *tcp_host (TCPSTREAM *stream)
  400. {
  401.   return stream->host; /* use tcp_remotehost() if want guarantees */
  402. }
  403. /* TCP/IP get remote host name
  404.  * Accepts: TCP/IP stream
  405.  * Returns: host name for this stream
  406.  */
  407. char *tcp_remotehost (TCPSTREAM *stream)
  408. {
  409.   if (!stream->remotehost) {
  410.     struct sockaddr_in sin;
  411.     int sinlen = sizeof (struct sockaddr_in);
  412.     stream->remotehost = /* get socket's peer name */
  413.       ((getpeername (stream->tcps,(struct sockaddr *) &sin,&sinlen) ==
  414. SOCKET_ERROR) || (sinlen <= 0)) ?
  415.   cpystr (stream->host) : tcp_name (&sin,NIL);
  416.   }
  417.   return stream->remotehost;
  418. }
  419. /* TCP/IP return port for this stream
  420.  * Accepts: TCP/IP stream
  421.  * Returns: port number for this stream
  422.  */
  423. unsigned long tcp_port (TCPSTREAM *stream)
  424. {
  425.   return stream->port; /* return port number */
  426. }
  427. /* TCP/IP get local host name
  428.  * Accepts: TCP/IP stream
  429.  * Returns: local host name
  430.  */
  431. char *tcp_localhost (TCPSTREAM *stream)
  432. {
  433.   if (!stream->localhost) {
  434.     struct sockaddr_in sin;
  435.     int sinlen = sizeof (struct sockaddr_in);
  436.     stream->localhost = /* get socket's name */
  437.       ((stream->port & 0xffff000) ||
  438.        ((getsockname (stream->tcps,(struct sockaddr *) &sin,&sinlen) ==
  439.  SOCKET_ERROR) || (sinlen <= 0))) ?
  440.    cpystr (mylocalhost ()) : tcp_name (&sin,NIL);
  441.   }
  442.   return stream->localhost; /* return local host name */
  443. }
  444. /* TCP/IP get client host name (server calls only)
  445.  * Returns: client host name
  446.  */
  447. char *tcp_clienthost ()
  448. {
  449.   if (!myClientHost) {
  450.     struct sockaddr_in sin;
  451.     int sinlen = sizeof (struct sockaddr_in);
  452.     myClientHost = /* get stdin's peer name */
  453.       ((getpeername (0,(struct sockaddr *) &sin,&sinlen) == SOCKET_ERROR) ||
  454.        (sinlen <= 0)) ? cpystr ("UNKNOWN") : tcp_name (&sin,T);
  455.   }
  456.   return myClientHost;
  457. }
  458. /* TCP/IP get server host name (server calls only)
  459.  * Returns: server host name
  460.  */
  461. static long myServerPort = -1;
  462. char *tcp_serverhost ()
  463. {
  464.   if (!myServerHost) {
  465.     struct sockaddr_in sin;
  466.     int sinlen = sizeof (struct sockaddr_in);
  467.     if (!wsa_initted++) { /* init Windows Sockets */
  468.       WSADATA wsock;
  469.       if (WSAStartup (WSA_VERSION,&wsock)) {
  470. wsa_initted = 0;
  471. return "random-pc"; /* try again later? */
  472.       }
  473.     }
  474. /* get stdin's name */
  475.     if ((getsockname (0,(struct sockaddr *) &sin,&sinlen) == SOCKET_ERROR) ||
  476. (sinlen <= 0)) myServerHost = cpystr (mylocalhost ());
  477.     else {
  478.       myServerHost = tcp_name (&sin,NIL);
  479.       myServerPort = ntohs (sin.sin_port);
  480.     }
  481.   }
  482.   return myServerHost;
  483. }
  484. /* TCP/IP get server port number (server calls only)
  485.  * Returns: server port number
  486.  */
  487. long tcp_serverport ()
  488. {
  489.   if (!myServerHost) tcp_serverhost ();
  490.   return myServerPort;
  491. }
  492. /* TCP/IP return canonical form of host name
  493.  * Accepts: host name
  494.  * Returns: canonical form of host name
  495.  */
  496. char *tcp_canonical (char *name)
  497. {
  498.   char *ret,host[MAILTMPLEN];
  499.   struct hostent *he;
  500.   blocknotify_t bn = (blocknotify_t) mail_parameters (NIL,GET_BLOCKNOTIFY,NIL);
  501. /* look like domain literal? */
  502.   if (name[0] == '[' && name[strlen (name) - 1] == ']') return name;
  503.   if (bn) (*bn) (BLOCK_DNSLOOKUP,NIL);
  504. /* note that NT requires lowercase! */
  505.   ret = (he = gethostbyname (lcase (strcpy (host,name)))) ? he->h_name : name;
  506.   if (bn) (*bn) (BLOCK_NONE,NIL);
  507.   return ret;
  508. }
  509. /* TCP/IP return name from socket
  510.  * Accepts: socket
  511.  *     verbose flag
  512.  * Returns: cpystr name
  513.  */
  514. char *tcp_name (struct sockaddr_in *sin,long flag)
  515. {
  516.   char *s,tmp[MAILTMPLEN];
  517.   if (allowreversedns) {
  518.     struct hostent *he;
  519.     blocknotify_t bn = (blocknotify_t)mail_parameters(NIL,GET_BLOCKNOTIFY,NIL);
  520.     if (bn) (*bn) (BLOCK_DNSLOOKUP,NIL);
  521. /* translate address to name */
  522.     if (!(he = gethostbyaddr ((char *) &sin->sin_addr,
  523.       sizeof (struct in_addr),sin->sin_family)))
  524.       sprintf (s = tmp,"[%s]",inet_ntoa (sin->sin_addr));
  525.     else if (flag) sprintf (s = tmp,"%s [%s]",he->h_name,
  526.     inet_ntoa (sin->sin_addr));
  527.     else s = he->h_name;
  528.     if (bn) (*bn) (BLOCK_NONE,NIL);
  529.   }
  530.   else sprintf (s = tmp,"[%s]",inet_ntoa (sin->sin_addr));
  531.   return cpystr (s);
  532. }
  533. /* Return my local host name
  534.  * Returns: my local host name
  535.  */
  536. char *mylocalhost (void)
  537. {
  538.   if (!myLocalHost) {
  539.     char *s,tmp[MAILTMPLEN];
  540.     struct hostent *he;
  541.     struct sockaddr_in sin, stmp;
  542.     int sinlen = sizeof (struct sockaddr_in);
  543.     SOCKET sock;
  544.     if (!wsa_initted++) { /* init Windows Sockets */
  545.       WSADATA wsock;
  546.       if (WSAStartup (WSA_VERSION,&wsock)) {
  547. wsa_initted = 0;
  548. return "random-pc"; /* try again later? */
  549.       }
  550.     }
  551.     sin.sin_family = AF_INET; /* family is always Internet */
  552.     sin.sin_addr.s_addr = inet_addr ("127.0.0.1");
  553.     sin.sin_port = htons ((u_short) 7);
  554.     if (allowreversedns &&
  555. ((sock = socket (sin.sin_family,SOCK_DGRAM,0)) != INVALID_SOCKET) &&
  556. (getsockname (sock,(struct sockaddr *) &stmp,&sinlen)!= SOCKET_ERROR)&&
  557. (sinlen > 0) &&
  558. (he = gethostbyaddr ((char *) &stmp.sin_addr,
  559.      sizeof (struct in_addr),stmp.sin_family)))
  560.       s = he->h_name;
  561.     else if (gethostname (tmp,MAILTMPLEN-1) == SOCKET_ERROR) s = "random-pc";
  562.     else s = (he = gethostbyname (tmp)) ? he->h_name : tmp;
  563.     myLocalHost = cpystr (s); /* canonicalize it */
  564. /* leave wsa_initted to save work later */
  565.     if (sock != INVALID_SOCKET) closesocket (sock);
  566.   }
  567.   return myLocalHost;
  568. }