[inseparable changes from match from perl-5.003_97g to perl-5.003_97h]
[p5sagit/p5-mst-13.2.git] / win32 / win32sck.c
1 /* NTSock.C
2  *
3  * (c) 1995 Microsoft Corporation. All rights reserved. 
4  *              Developed by hip communications inc., http://info.hip.com/info/
5  * Portions (c) 1993 Intergraph Corporation. All rights reserved.
6  *
7  *    You may distribute under the terms of either the GNU General Public
8  *    License or the Artistic License, as specified in the README file.
9  */
10
11 #include <windows.h>
12 #define WIN32_LEAN_AND_MEAN
13 #include "EXTERN.h"
14 #include "perl.h"
15 #include <sys/socket.h>
16 #include <fcntl.h>
17 #include <sys/stat.h>
18 #include <assert.h>
19
20 #define CROAK croak
21
22 #ifdef USE_SOCKETS_AS_HANDLES
23 /* thanks to Beverly Brown      (beverly@datacube.com) */
24
25 #define OPEN_SOCKET(x)  _open_osfhandle(x,_O_RDWR|_O_BINARY)
26 #define TO_SOCKET(x)    _get_osfhandle(x)
27
28 #else
29
30 #       define OPEN_SOCKET(x)   (x)
31 #       define TO_SOCKET(x)     (x)
32
33 #endif  /* USE_SOCKETS_AS_HANDLES */
34
35 static struct servent* win32_savecopyservent(struct servent*d,
36                                              struct servent*s,
37                                              const char *proto);
38 #define SOCKETAPI PASCAL 
39
40 typedef SOCKET (SOCKETAPI *LPSOCKACCEPT)(SOCKET, struct sockaddr *, int *);
41 typedef int (SOCKETAPI *LPSOCKBIND)(SOCKET, const struct sockaddr *, int);
42 typedef int (SOCKETAPI *LPSOCKCLOSESOCKET)(SOCKET);
43 typedef int (SOCKETAPI *LPSOCKCONNECT)(SOCKET, const struct sockaddr *, int);
44 typedef int (SOCKETAPI *LPSOCKIOCTLSOCKET)(SOCKET, long, u_long *);
45 typedef int (SOCKETAPI *LPSOCKGETPEERNAME)(SOCKET, struct sockaddr *, int *);
46 typedef int (SOCKETAPI *LPSOCKGETSOCKNAME)(SOCKET, struct sockaddr *, int *);
47 typedef int (SOCKETAPI *LPSOCKGETSOCKOPT)(SOCKET, int, int, char *, int *);
48 typedef u_long (SOCKETAPI *LPSOCKHTONL)(u_long);
49 typedef u_short (SOCKETAPI *LPSOCKHTONS)(u_short);
50 typedef int (SOCKETAPI *LPSOCKLISTEN)(SOCKET, int);
51 typedef u_long (SOCKETAPI *LPSOCKNTOHL)(u_long);
52 typedef u_short (SOCKETAPI *LPSOCKNTOHS)(u_short);
53 typedef int (SOCKETAPI *LPSOCKRECV)(SOCKET, char *, int, int);
54 typedef int (SOCKETAPI *LPSOCKRECVFROM)(SOCKET, char *, int, int, struct sockaddr *, int *);
55 typedef int (SOCKETAPI *LPSOCKSELECT)(int, fd_set *, fd_set *, fd_set *, const struct timeval *);
56 typedef int (SOCKETAPI *LPSOCKSEND)(SOCKET, const char *, int, int);
57 typedef int (SOCKETAPI *LPSOCKSENDTO)(SOCKET, const char *, int, int, const struct sockaddr *, int);
58 typedef int (SOCKETAPI *LPSOCKSETSOCKOPT)(SOCKET, int, int, const char *, int);
59 typedef int (SOCKETAPI *LPSOCKSHUTDOWN)(SOCKET, int);
60 typedef SOCKET (SOCKETAPI *LPSOCKSOCKET)(int, int, int);
61 typedef char FAR *(SOCKETAPI *LPSOCKINETNTOA)(struct in_addr in);
62 typedef unsigned long (SOCKETAPI *LPSOCKINETADDR)(const char FAR * cp);
63
64
65 /* Database function prototypes */
66 typedef struct hostent *(SOCKETAPI *LPSOCKGETHOSTBYADDR)(const char *, int, int);
67 typedef struct hostent *(SOCKETAPI *LPSOCKGETHOSTBYNAME)(const char *);
68 typedef int (SOCKETAPI *LPSOCKGETHOSTNAME)(char *, int);
69 typedef struct servent *(SOCKETAPI *LPSOCKGETSERVBYPORT)(int, const char *);
70 typedef struct servent *(SOCKETAPI *LPSOCKGETSERVBYNAME)(const char *, const char *);
71 typedef struct protoent *(SOCKETAPI *LPSOCKGETPROTOBYNUMBER)(int);
72 typedef struct protoent *(SOCKETAPI *LPSOCKGETPROTOBYNAME)(const char *);
73
74 /* Microsoft Windows Extension function prototypes */
75 typedef int (SOCKETAPI *LPSOCKWSASTARTUP)(unsigned short, LPWSADATA);
76 typedef int (SOCKETAPI *LPSOCKWSACLEANUP)(void);
77 typedef int (SOCKETAPI *LPSOCKWSAGETLASTERROR)(void);
78 typedef int (SOCKETAPI *LPWSAFDIsSet)(SOCKET, fd_set *);
79
80 static HINSTANCE hWinSockDll = 0;
81 /* extern CRITICAL_SECTION csSock; */
82
83 static LPSOCKACCEPT paccept = 0;
84 static LPSOCKBIND pbind = 0;
85 static LPSOCKCLOSESOCKET pclosesocket = 0;
86 static LPSOCKCONNECT pconnect = 0;
87 static LPSOCKIOCTLSOCKET pioctlsocket = 0;
88 static LPSOCKGETPEERNAME pgetpeername = 0;
89 static LPSOCKGETSOCKNAME pgetsockname = 0;
90 static LPSOCKGETSOCKOPT pgetsockopt = 0;
91 static LPSOCKHTONL phtonl = 0;
92 static LPSOCKHTONS phtons = 0;
93 static LPSOCKLISTEN plisten = 0;
94 static LPSOCKNTOHL pntohl = 0;
95 static LPSOCKNTOHS pntohs = 0;
96 static LPSOCKRECV precv = 0;
97 static LPSOCKRECVFROM precvfrom = 0;
98 static LPSOCKSELECT pselect = 0;
99 static LPSOCKSEND psend = 0;
100 static LPSOCKSENDTO psendto = 0;
101 static LPSOCKSETSOCKOPT psetsockopt = 0;
102 static LPSOCKSHUTDOWN pshutdown = 0;
103 static LPSOCKSOCKET psocket = 0;
104 static LPSOCKGETHOSTBYADDR pgethostbyaddr = 0;
105 static LPSOCKGETHOSTBYNAME pgethostbyname = 0;
106 static LPSOCKGETHOSTNAME pgethostname = 0;
107 static LPSOCKGETSERVBYPORT pgetservbyport = 0;
108 static LPSOCKGETSERVBYNAME pgetservbyname = 0;
109 static LPSOCKGETPROTOBYNUMBER pgetprotobynumber = 0;
110 static LPSOCKGETPROTOBYNAME pgetprotobyname = 0;
111 static LPSOCKWSASTARTUP pWSAStartup = 0;
112 static LPSOCKWSACLEANUP pWSACleanup = 0;
113 static LPSOCKWSAGETLASTERROR pWSAGetLastError = 0;
114 static LPWSAFDIsSet pWSAFDIsSet = 0;
115 static LPSOCKINETNTOA pinet_ntoa = 0;
116 static LPSOCKINETADDR pinet_addr = 0;
117
118 __declspec(thread) struct servent myservent;
119
120
121 void *
122 GetAddress(HINSTANCE hInstance, char *lpFunctionName)
123 {
124     char buffer[512];
125     FARPROC proc = GetProcAddress(hInstance, lpFunctionName);
126     if(proc == 0) {
127         sprintf(buffer, "Unable to get address of %s in WSock32.dll", lpFunctionName);
128         CROAK(buffer);
129     }
130     return proc;
131 }
132
133 void
134 LoadWinSock(void)
135 {
136 /*  EnterCriticalSection(&csSock); */
137     if(hWinSockDll == NULL) {
138         HINSTANCE hLib = LoadLibrary("WSock32.DLL");
139         if(hLib == NULL)
140             CROAK("Could not load WSock32.dll\n");
141
142         paccept = (LPSOCKACCEPT)GetAddress(hLib, "accept");
143         pbind = (LPSOCKBIND)GetAddress(hLib, "bind");
144         pclosesocket = (LPSOCKCLOSESOCKET)GetAddress(hLib, "closesocket");
145         pconnect = (LPSOCKCONNECT)GetAddress(hLib, "connect");
146         pioctlsocket = (LPSOCKIOCTLSOCKET)GetAddress(hLib, "ioctlsocket");
147         pgetpeername = (LPSOCKGETPEERNAME)GetAddress(hLib, "getpeername");
148         pgetsockname = (LPSOCKGETSOCKNAME)GetAddress(hLib, "getsockname");
149         pgetsockopt = (LPSOCKGETSOCKOPT)GetAddress(hLib, "getsockopt");
150         phtonl = (LPSOCKHTONL)GetAddress(hLib, "htonl");
151         phtons = (LPSOCKHTONS)GetAddress(hLib, "htons");
152         plisten = (LPSOCKLISTEN)GetAddress(hLib, "listen");
153         pntohl = (LPSOCKNTOHL)GetAddress(hLib, "ntohl");
154         pntohs = (LPSOCKNTOHS)GetAddress(hLib, "ntohs");
155         precv = (LPSOCKRECV)GetAddress(hLib, "recv");
156         precvfrom = (LPSOCKRECVFROM)GetAddress(hLib, "recvfrom");
157         pselect = (LPSOCKSELECT)GetAddress(hLib, "select");
158         psend = (LPSOCKSEND)GetAddress(hLib, "send");
159         psendto = (LPSOCKSENDTO)GetAddress(hLib, "sendto");
160         psetsockopt = (LPSOCKSETSOCKOPT)GetAddress(hLib, "setsockopt");
161         pshutdown = (LPSOCKSHUTDOWN)GetAddress(hLib, "shutdown");
162         psocket = (LPSOCKSOCKET)GetAddress(hLib, "socket");
163         pgethostbyaddr = (LPSOCKGETHOSTBYADDR)GetAddress(hLib, "gethostbyaddr");
164         pgethostbyname = (LPSOCKGETHOSTBYNAME)GetAddress(hLib, "gethostbyname");
165         pgethostname = (LPSOCKGETHOSTNAME)GetAddress(hLib, "gethostname");
166         pgetservbyport = (LPSOCKGETSERVBYPORT)GetAddress(hLib, "getservbyport");
167         pgetservbyname = (LPSOCKGETSERVBYNAME)GetAddress(hLib, "getservbyname");
168         pgetprotobynumber = (LPSOCKGETPROTOBYNUMBER)GetAddress(hLib, "getprotobynumber");
169         pgetprotobyname = (LPSOCKGETPROTOBYNAME)GetAddress(hLib, "getprotobyname");
170         pWSAStartup = (LPSOCKWSASTARTUP)GetAddress(hLib, "WSAStartup");
171         pWSACleanup = (LPSOCKWSACLEANUP)GetAddress(hLib, "WSACleanup");
172         pWSAGetLastError = (LPSOCKWSAGETLASTERROR)GetAddress(hLib, "WSAGetLastError");
173         pWSAFDIsSet = (LPWSAFDIsSet)GetAddress(hLib, "__WSAFDIsSet");
174         pinet_addr = (LPSOCKINETADDR)GetAddress(hLib,"inet_addr");
175         pinet_ntoa = (LPSOCKINETNTOA)GetAddress(hLib,"inet_ntoa");
176
177         hWinSockDll = hLib;
178     }
179 /*  LeaveCriticalSection(&csSock); */
180 }
181
182 void
183 EndSockets(void)
184 {
185     if(hWinSockDll != NULL) {
186         pWSACleanup();
187         FreeLibrary(hWinSockDll);
188     }
189     hWinSockDll = NULL;
190 }
191
192 void
193 StartSockets(void) 
194 {
195     unsigned short version;
196     WSADATA retdata;
197     int ret;
198     int iSockOpt = SO_SYNCHRONOUS_NONALERT;
199
200     LoadWinSock();
201     /*
202      * initalize the winsock interface and insure that it is
203      * cleaned up at exit.
204      */
205     version = 0x101;
206     if(ret = pWSAStartup(version, &retdata))
207         CROAK("Unable to locate winsock library!\n");
208     if(retdata.wVersion != version)
209         CROAK("Could not find version 1.1 of winsock dll\n");
210
211     /* atexit((void (*)(void)) EndSockets); */
212
213 #ifdef USE_SOCKETS_AS_HANDLES
214     /*
215      * Enable the use of sockets as filehandles
216      */
217     psetsockopt(INVALID_SOCKET, SOL_SOCKET, SO_OPENTYPE,
218                 (char *)&iSockOpt, sizeof(iSockOpt));
219 #endif  /* USE_SOCKETS_AS_HANDLES */
220 }
221
222
223 #ifndef USE_SOCKETS_AS_HANDLES
224 FILE *
225 myfdopen(int fd, char *mode)
226 {
227     FILE *fp;
228     char sockbuf[256];
229     int optlen = sizeof(sockbuf);
230     int retval;
231
232     if (hWinSockDll == 0)
233         LoadWinSock();
234
235     retval = pgetsockopt((SOCKET)fd, SOL_SOCKET, SO_TYPE, sockbuf, &optlen);
236     if(retval == SOCKET_ERROR && pWSAGetLastError() == WSAENOTSOCK) {
237         return(_fdopen(fd, mode));
238     }
239
240     /*
241      * If we get here, then fd is actually a socket.
242      */
243     Newz(1601, fp, 1, FILE);
244     if(fp == NULL) {
245         errno = ENOMEM;
246         return NULL;
247     }
248
249     fp->_file = fd;
250     if(*mode == 'r')
251         fp->_flag = _IOREAD;
252     else
253         fp->_flag = _IOWRT;
254    
255     return fp;
256 }
257 #endif  /* USE_SOCKETS_AS_HANDLES */
258
259
260 u_long
261 win32_htonl(u_long hostlong)
262 {
263     if(hWinSockDll == 0)
264         LoadWinSock();
265
266     return phtonl(hostlong);
267 }
268
269 u_short
270 win32_htons(u_short hostshort)
271 {
272     if(hWinSockDll == 0)
273         LoadWinSock();
274
275     return phtons(hostshort);
276 }
277
278 u_long
279 win32_ntohl(u_long netlong)
280 {
281     if(hWinSockDll == 0)
282         LoadWinSock();
283
284     return pntohl(netlong);
285 }
286
287 u_short
288 win32_ntohs(u_short netshort)
289 {
290     if(hWinSockDll == 0)
291         LoadWinSock();
292
293     return pntohs(netshort);
294 }
295
296
297 #define SOCKET_TEST(x, y)       if(hWinSockDll == 0) StartSockets();\
298                                 if((x) == (y)) errno = pWSAGetLastError()
299
300 #define SOCKET_TEST_ERROR(x) SOCKET_TEST(x, SOCKET_ERROR)
301
302 SOCKET
303 win32_accept(SOCKET s, struct sockaddr *addr, int *addrlen)
304 {
305     SOCKET r;
306
307     SOCKET_TEST((r = paccept(TO_SOCKET(s), addr, addrlen)), INVALID_SOCKET);
308     return OPEN_SOCKET(r);
309 }
310
311 int
312 win32_bind(SOCKET s, const struct sockaddr *addr, int addrlen)
313 {
314     int r;
315
316     SOCKET_TEST_ERROR(r = pbind(TO_SOCKET(s), addr, addrlen));
317     return r;
318 }
319
320 int
321 win32_connect(SOCKET s, const struct sockaddr *addr, int addrlen)
322 {
323     int r;
324
325     SOCKET_TEST_ERROR(r = pconnect(TO_SOCKET(s), addr, addrlen));
326     return r;
327 }
328
329
330 int
331 win32_getpeername(SOCKET s, struct sockaddr *addr, int *addrlen)
332 {
333     int r;
334
335     SOCKET_TEST_ERROR(r = pgetpeername(TO_SOCKET(s), addr, addrlen));
336     return r;
337 }
338
339 int
340 win32_getsockname(SOCKET s, struct sockaddr *addr, int *addrlen)
341 {
342     int r;
343
344     SOCKET_TEST_ERROR(r = pgetsockname(TO_SOCKET(s), addr, addrlen));
345     return r;
346 }
347
348 int
349 win32_getsockopt(SOCKET s, int level, int optname, char *optval, int *optlen)
350 {
351     int r;
352
353     SOCKET_TEST_ERROR(r = pgetsockopt(TO_SOCKET(s), level, optname, optval, optlen));
354     return r;
355 }
356
357 int
358 win32_ioctlsocket(SOCKET s, long cmd, u_long *argp)
359 {
360     int r;
361
362     SOCKET_TEST_ERROR(r = pioctlsocket(TO_SOCKET(s), cmd, argp));
363     return r;
364 }
365
366 int
367 win32_listen(SOCKET s, int backlog)
368 {
369     int r;
370
371     SOCKET_TEST_ERROR(r = plisten(TO_SOCKET(s), backlog));
372     return r;
373 }
374
375 int
376 win32_recv(SOCKET s, char *buf, int len, int flags)
377 {
378     int r;
379
380     SOCKET_TEST_ERROR(r = precv(TO_SOCKET(s), buf, len, flags));
381     return r;
382 }
383
384 int
385 win32_recvfrom(SOCKET s, char *buf, int len, int flags, struct sockaddr *from, int *fromlen)
386 {
387     int r;
388
389     SOCKET_TEST_ERROR(r = precvfrom(TO_SOCKET(s), buf, len, flags, from, fromlen));
390     return r;
391 }
392
393 /* select contributed by Vincent R. Slyngstad (vrs@ibeam.intel.com) */
394 int
395 win32_select(int nfds, int* rd, int* wr, int* ex, const struct timeval* timeout)
396 {
397     long r;
398     int dummy = 0;
399     int i, fd, bit, offset;
400     FD_SET nrd, nwr, nex,*prd,*pwr,*pex;
401
402     if (!rd)
403         rd = &dummy, prd = NULL;
404     else
405         prd = &nrd;
406     if (!wr)
407         wr = &dummy, pwr = NULL;
408     else
409         pwr = &nwr;
410     if (!ex)
411         ex = &dummy, pex = NULL;
412     else
413         pex = &nex;
414
415     FD_ZERO(&nrd);
416     FD_ZERO(&nwr);
417     FD_ZERO(&nex);
418     for (i = 0; i < nfds; i++) {
419         fd = TO_SOCKET(i);
420         bit = 1L<<(i % (sizeof(int)*8));
421         offset = i / (sizeof(int)*8);
422         if (rd[offset] & bit)
423             FD_SET(fd, &nrd);
424         if (wr[offset] & bit)
425             FD_SET(fd, &nwr);
426         if (ex[offset] & bit)
427             FD_SET(fd, &nex);
428     }
429
430     SOCKET_TEST_ERROR(r = pselect(nfds, prd, pwr, pex, timeout));
431
432     for (i = 0; i < nfds; i++) {
433         fd = TO_SOCKET(i);
434         bit = 1L<<(i % (sizeof(int)*8));
435         offset = i / (sizeof(int)*8);
436         if (rd[offset] & bit) {
437             if (!pWSAFDIsSet(fd, &nrd))
438                 rd[offset] &= ~bit;
439         }
440         if (wr[offset] & bit) {
441             if (!pWSAFDIsSet(fd, &nwr))
442                 wr[offset] &= ~bit;
443         }
444         if (ex[offset] & bit) {
445             if (!pWSAFDIsSet(fd, &nex))
446                 ex[offset] &= ~bit;
447         }
448     }
449     return r;
450 }
451
452 int
453 win32_send(SOCKET s, const char *buf, int len, int flags)
454 {
455     int r;
456
457     SOCKET_TEST_ERROR(r = psend(TO_SOCKET(s), buf, len, flags));
458     return r;
459 }
460
461 int
462 win32_sendto(SOCKET s, const char *buf, int len, int flags,
463              const struct sockaddr *to, int tolen)
464 {
465     int r;
466
467     SOCKET_TEST_ERROR(r = psendto(TO_SOCKET(s), buf, len, flags, to, tolen));
468     return r;
469 }
470
471 int
472 win32_setsockopt(SOCKET s, int level, int optname, const char *optval, int optlen)
473 {
474     int r;
475
476     SOCKET_TEST_ERROR(r = psetsockopt(TO_SOCKET(s), level, optname, optval, optlen));
477     return r;
478 }
479     
480 int
481 win32_shutdown(SOCKET s, int how)
482 {
483     int r;
484
485     SOCKET_TEST_ERROR(r = pshutdown(TO_SOCKET(s), how));
486     return r;
487 }
488
489 SOCKET
490 win32_socket(int af, int type, int protocol)
491 {
492     SOCKET s;
493
494 #ifndef USE_SOCKETS_AS_HANDLES
495     SOCKET_TEST(s = psocket(af, type, protocol), INVALID_SOCKET);
496 #else
497     if(hWinSockDll == 0)
498         StartSockets();
499
500     if((s = psocket(af, type, protocol)) == INVALID_SOCKET)
501         errno = pWSAGetLastError();
502     else
503         s = OPEN_SOCKET(s);
504 #endif  /* USE_SOCKETS_AS_HANDLES */
505
506     return s;
507 }
508
509 struct hostent *
510 win32_gethostbyaddr(const char *addr, int len, int type)
511 {
512     struct hostent *r;
513
514     SOCKET_TEST(r = pgethostbyaddr(addr, len, type), NULL);
515     return r;
516 }
517
518 struct hostent *
519 win32_gethostbyname(const char *name)
520 {
521     struct hostent *r;
522
523     SOCKET_TEST(r = pgethostbyname(name), NULL);
524     return r;
525 }
526
527 int
528 win32_gethostname(char *name, int len)
529 {
530     int r;
531
532     SOCKET_TEST_ERROR(r = pgethostname(name, len));
533     return r;
534 }
535
536 struct protoent *
537 win32_getprotobyname(const char *name)
538 {
539     struct protoent *r;
540
541     SOCKET_TEST(r = pgetprotobyname(name), NULL);
542     return r;
543 }
544
545 struct protoent *
546 win32_getprotobynumber(int num)
547 {
548     struct protoent *r;
549
550     SOCKET_TEST(r = pgetprotobynumber(num), NULL);
551     return r;
552 }
553
554 struct servent *
555 win32_getservbyname(const char *name, const char *proto)
556 {
557     struct servent *r;
558    
559     SOCKET_TEST(r = pgetservbyname(name, proto), NULL);
560     if (r) {
561         r = win32_savecopyservent(&myservent, r, proto);
562     }
563     return r;
564 }
565
566 struct servent *
567 win32_getservbyport(int port, const char *proto)
568 {
569     struct servent *r;
570
571     SOCKET_TEST(r = pgetservbyport(port, proto), NULL);
572     if (r) {
573         r = win32_savecopyservent(&myservent, r, proto);
574     }
575     return r;
576 }
577
578 char FAR *
579 win32_inet_ntoa(struct in_addr in)
580 {
581     if(hWinSockDll == 0)
582         LoadWinSock();
583
584     return pinet_ntoa(in);
585 }
586
587 unsigned long
588 win32_inet_addr(const char FAR *cp)
589 {
590     if(hWinSockDll == 0)
591         LoadWinSock();
592
593     return pinet_addr(cp);
594
595 }
596
597 /*
598  * Networking stubs
599  */
600 #undef CROAK 
601 #define CROAK croak
602
603 void
604 win32_endhostent() 
605 {
606     CROAK("endhostent not implemented!\n");
607 }
608
609 void
610 win32_endnetent()
611 {
612     CROAK("endnetent not implemented!\n");
613 }
614
615 void
616 win32_endprotoent()
617 {
618     CROAK("endprotoent not implemented!\n");
619 }
620
621 void
622 win32_endservent()
623 {
624     CROAK("endservent not implemented!\n");
625 }
626
627
628 struct netent *
629 win32_getnetent(void) 
630 {
631     CROAK("getnetent not implemented!\n");
632     return (struct netent *) NULL;
633 }
634
635 struct netent *
636 win32_getnetbyname(char *name) 
637 {
638     CROAK("getnetbyname not implemented!\n");
639     return (struct netent *)NULL;
640 }
641
642 struct netent *
643 win32_getnetbyaddr(long net, int type) 
644 {
645     CROAK("getnetbyaddr not implemented!\n");
646     return (struct netent *)NULL;
647 }
648
649 struct protoent *
650 win32_getprotoent(void) 
651 {
652     CROAK("getprotoent not implemented!\n");
653     return (struct protoent *) NULL;
654 }
655
656 struct servent *
657 win32_getservent(void) 
658 {
659     CROAK("getservent not implemented!\n");
660     return (struct servent *) NULL;
661 }
662
663 void
664 win32_sethostent(int stayopen)
665 {
666     CROAK("sethostent not implemented!\n");
667 }
668
669
670 void
671 win32_setnetent(int stayopen)
672 {
673     CROAK("setnetent not implemented!\n");
674 }
675
676
677 void
678 win32_setprotoent(int stayopen)
679 {
680     CROAK("setprotoent not implemented!\n");
681 }
682
683
684 void
685 win32_setservent(int stayopen)
686 {
687     CROAK("setservent not implemented!\n");
688 }
689
690 #define WIN32IO_IS_STDIO
691 #include <io.h>
692 #include "win32iop.h"
693
694 static struct servent*
695 win32_savecopyservent(struct servent*d, struct servent*s, const char *proto)
696 {
697     d->s_name = s->s_name;
698     d->s_aliases = s->s_aliases;
699     d->s_port = s->s_port;
700     if (!IsWin95() && s->s_proto && strlen(s->s_proto))
701         d->s_proto = s->s_proto;
702     else if (proto && strlen(proto))
703         d->s_proto = (char *)proto;
704     else
705         d->s_proto = "tcp";
706    
707     return d;
708 }
709
710